-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdbeta.c
70 lines (65 loc) · 2.08 KB
/
dbeta.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
/*
* AUTHOR
* Catherine Loader, [email protected].
* October 23, 2000.
*
* Merge in to R:
* Copyright (C) 2000, The R Core Team
* Changes to case a, b < 2, use logs to avoid underflow
* Copyright (C) 2006, The R Core Team
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, a copy is available at
* http://www.r-project.org/Licenses/
*
*
* DESCRIPTION
* Beta density,
* (a+b-1)! a-1 b-1
* p(x;a,b) = ------------ x (1-x)
* (a-1)!(b-1)!
*
* = (a+b-1) dbinom(a-1; a+b-2,x)
*
* The basic formula for the log density is thus
* (a-1) log x + (b-1) log (1-x) - lbeta(a, b)
* If either a or b <= 2 then 0 < lbeta(a, b) < 710 and so no
* term is large. We use Loader's code only if both a and b > 2.
*/
#include "nmath.h"
#include "dpq.h"
double dbeta(double x, double a, double b, int give_log)
{
double lval;
#ifdef IEEE_754
/* NaNs propagated correctly */
if (ISNAN(x) || ISNAN(a) || ISNAN(b)) return x + a + b;
#endif
if (a <= 0 || b <= 0) ML_ERR_return_NAN;
if (x < 0 || x > 1) return(R_D__0);
if (x == 0) {
if(a > 1) return(R_D__0);
if(a < 1) return(ML_POSINF);
/* a == 1 : */ return(R_D_val(b));
}
if (x == 1) {
if(b > 1) return(R_D__0);
if(b < 1) return(ML_POSINF);
/* b == 1 : */ return(R_D_val(a));
}
if (a <= 2 || b <= 2)
lval = (a-1)*log(x) + (b-1)*log1p(-x) - lbeta(a, b);
else
lval = log(a+b-1) + dbinom_raw(a-1, a+b-2, x, 1-x, TRUE);
return R_D_exp(lval);
}