|
| 1 | +#ifndef STAN_MATH_PRIM_PROB_BETA_NEG_BINOMIAL_LPMF_HPP |
| 2 | +#define STAN_MATH_PRIM_PROB_BETA_NEG_BINOMIAL_LPMF_HPP |
| 3 | + |
| 4 | +#include <stan/math/prim/meta.hpp> |
| 5 | +#include <stan/math/prim/err.hpp> |
| 6 | +#include <stan/math/prim/fun/constants.hpp> |
| 7 | +#include <stan/math/prim/fun/digamma.hpp> |
| 8 | +#include <stan/math/prim/fun/lbeta.hpp> |
| 9 | +#include <stan/math/prim/fun/lgamma.hpp> |
| 10 | +#include <stan/math/prim/fun/max_size.hpp> |
| 11 | +#include <stan/math/prim/fun/scalar_seq_view.hpp> |
| 12 | +#include <stan/math/prim/fun/size.hpp> |
| 13 | +#include <stan/math/prim/fun/size_zero.hpp> |
| 14 | +#include <stan/math/prim/fun/value_of.hpp> |
| 15 | +#include <stan/math/prim/functor/partials_propagator.hpp> |
| 16 | + |
| 17 | +namespace stan { |
| 18 | +namespace math { |
| 19 | + |
| 20 | +/** \ingroup prob_dists |
| 21 | + * Returns the log PMF of the Beta Negative Binomial distribution with given |
| 22 | + * number of successes, prior success, and prior failure parameters. |
| 23 | + * Given containers of matching sizes, returns the log sum of probabilities. |
| 24 | + * |
| 25 | + * @tparam T_n type of failure parameter |
| 26 | + * @tparam T_r type of number of successes parameter |
| 27 | + * @tparam T_alpha type of prior success parameter |
| 28 | + * @tparam T_beta type of prior failure parameter |
| 29 | + * |
| 30 | + * @param n failure parameter |
| 31 | + * @param r Number of successes parameter |
| 32 | + * @param alpha prior success parameter |
| 33 | + * @param beta prior failure parameter |
| 34 | + * @return log probability or log sum of probabilities |
| 35 | + * @throw std::domain_error if r, alpha, or beta fails to be positive |
| 36 | + * @throw std::invalid_argument if container sizes mismatch |
| 37 | + */ |
| 38 | +template <bool propto, typename T_n, typename T_r, typename T_alpha, |
| 39 | + typename T_beta, |
| 40 | + require_all_not_nonscalar_prim_or_rev_kernel_expression_t< |
| 41 | + T_n, T_r, T_alpha, T_beta>* = nullptr> |
| 42 | +inline return_type_t<T_r, T_alpha, T_beta> beta_neg_binomial_lpmf( |
| 43 | + const T_n& n, const T_r& r, const T_alpha& alpha, const T_beta& beta) { |
| 44 | + using T_partials_return = partials_return_t<T_n, T_r, T_alpha, T_beta>; |
| 45 | + using T_n_ref = ref_type_t<T_n>; |
| 46 | + using T_r_ref = ref_type_t<T_r>; |
| 47 | + using T_alpha_ref = ref_type_t<T_alpha>; |
| 48 | + using T_beta_ref = ref_type_t<T_beta>; |
| 49 | + static constexpr const char* function = "beta_neg_binomial_lpmf"; |
| 50 | + check_consistent_sizes( |
| 51 | + function, "Failures variable", n, "Number of successes parameter", r, |
| 52 | + "Prior success parameter", alpha, "Prior failure parameter", beta); |
| 53 | + if (size_zero(n, r, alpha, beta)) { |
| 54 | + return 0.0; |
| 55 | + } |
| 56 | + |
| 57 | + T_n_ref n_ref = n; |
| 58 | + T_r_ref r_ref = r; |
| 59 | + T_alpha_ref alpha_ref = alpha; |
| 60 | + T_beta_ref beta_ref = beta; |
| 61 | + check_nonnegative(function, "Failures variable", n_ref); |
| 62 | + check_positive_finite(function, "Number of successes parameter", r_ref); |
| 63 | + check_positive_finite(function, "Prior success parameter", alpha_ref); |
| 64 | + check_positive_finite(function, "Prior failure parameter", beta_ref); |
| 65 | + |
| 66 | + if constexpr (!include_summand<propto, T_r, T_alpha, T_beta>::value) { |
| 67 | + return 0.0; |
| 68 | + } |
| 69 | + |
| 70 | + auto ops_partials = make_partials_propagator(r_ref, alpha_ref, beta_ref); |
| 71 | + |
| 72 | + scalar_seq_view<T_n> n_vec(n); |
| 73 | + scalar_seq_view<T_r_ref> r_vec(r_ref); |
| 74 | + scalar_seq_view<T_alpha_ref> alpha_vec(alpha_ref); |
| 75 | + scalar_seq_view<T_beta_ref> beta_vec(beta_ref); |
| 76 | + const size_t max_size_seq_view = max_size(n, r, alpha, beta); |
| 77 | + T_partials_return logp(0.0); |
| 78 | + for (size_t i = 0; i < max_size_seq_view; i++) { |
| 79 | + if constexpr (include_summand<propto>::value) { |
| 80 | + logp -= lgamma(n_vec[i] + 1); |
| 81 | + } |
| 82 | + T_partials_return lbeta_denominator = lbeta(r_vec.val(i), alpha_vec.val(i)); |
| 83 | + T_partials_return lgamma_numerator = lgamma(n_vec[i] + beta_vec.val(i)); |
| 84 | + T_partials_return lgamma_denominator = lgamma(beta_vec.val(i)); |
| 85 | + T_partials_return lbeta_numerator |
| 86 | + = lbeta(n_vec[i] + r_vec.val(i), alpha_vec.val(i) + beta_vec.val(i)); |
| 87 | + logp += lbeta_numerator + lgamma_numerator - lbeta_denominator |
| 88 | + - lgamma_denominator; |
| 89 | + if (!is_constant_all<T_r, T_alpha, T_beta>::value) { |
| 90 | + T_partials_return digamma_n_r_alpha_beta = digamma( |
| 91 | + n_vec[i] + r_vec.val(i) + alpha_vec.val(i) + beta_vec.val(i)); |
| 92 | + |
| 93 | + if constexpr (!is_constant<T_r>::value || !is_constant<T_alpha>::value) { |
| 94 | + T_partials_return digamma_r_alpha |
| 95 | + = digamma(r_vec.val(i) + alpha_vec.val(i)); |
| 96 | + if constexpr (!is_constant_all<T_r>::value) { |
| 97 | + partials<0>(ops_partials)[i] |
| 98 | + += digamma(n_vec[i] + r_vec.val(i)) - digamma_n_r_alpha_beta |
| 99 | + - (digamma(r_vec.val(i)) - digamma_r_alpha); |
| 100 | + } |
| 101 | + if constexpr (!is_constant_all<T_alpha>::value) { |
| 102 | + partials<1>(ops_partials)[i] |
| 103 | + += -digamma_n_r_alpha_beta |
| 104 | + - (digamma(alpha_vec.val(i)) - digamma_r_alpha); |
| 105 | + } |
| 106 | + } |
| 107 | + if constexpr (!is_constant<T_beta>::value |
| 108 | + || !is_constant<T_alpha>::value) { |
| 109 | + T_partials_return digamma_alpha_beta |
| 110 | + = digamma(alpha_vec.val(i) + beta_vec.val(i)); |
| 111 | + if constexpr (!is_constant_all<T_beta>::value) { |
| 112 | + partials<2>(ops_partials)[i] += digamma_alpha_beta |
| 113 | + - digamma_n_r_alpha_beta |
| 114 | + + digamma(n_vec[i] + beta_vec.val(i)) |
| 115 | + - digamma(beta_vec.val(i)); |
| 116 | + } |
| 117 | + if constexpr (!is_constant_all<T_alpha>::value) { |
| 118 | + partials<1>(ops_partials)[i] += digamma_alpha_beta; |
| 119 | + } |
| 120 | + } |
| 121 | + } |
| 122 | + } |
| 123 | + return ops_partials.build(logp); |
| 124 | +} |
| 125 | + |
| 126 | +template <typename T_n, typename T_r, typename T_alpha, typename T_beta> |
| 127 | +inline return_type_t<T_r, T_alpha, T_beta> beta_neg_binomial_lpmf( |
| 128 | + const T_n& n, const T_r& r, const T_alpha& alpha, const T_beta& beta) { |
| 129 | + return beta_neg_binomial_lpmf<false>(n, r, alpha, beta); |
| 130 | +} |
| 131 | + |
| 132 | +} // namespace math |
| 133 | +} // namespace stan |
| 134 | +#endif |
0 commit comments