annotate DEPENDENCIES/generic/include/boost/math/special_functions/trigamma.hpp @ 133:4acb5d8d80b6 tip

Don't fail environmental check if README.md exists (but .txt and no-suffix don't)
author Chris Cannam
date Tue, 30 Jul 2019 12:25:44 +0100
parents f46d142149f5
children
rev   line source
Chris@102 1 // (C) Copyright John Maddock 2006.
Chris@102 2 // Use, modification and distribution are subject to the
Chris@102 3 // Boost Software License, Version 1.0. (See accompanying file
Chris@102 4 // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
Chris@102 5
Chris@102 6 #ifndef BOOST_MATH_SF_TRIGAMMA_HPP
Chris@102 7 #define BOOST_MATH_SF_TRIGAMMA_HPP
Chris@102 8
Chris@102 9 #ifdef _MSC_VER
Chris@102 10 #pragma once
Chris@102 11 #endif
Chris@102 12
Chris@102 13 #include <boost/math/special_functions/math_fwd.hpp>
Chris@102 14 #include <boost/math/tools/rational.hpp>
Chris@102 15 #include <boost/math/tools/series.hpp>
Chris@102 16 #include <boost/math/tools/promotion.hpp>
Chris@102 17 #include <boost/math/policies/error_handling.hpp>
Chris@102 18 #include <boost/math/constants/constants.hpp>
Chris@102 19 #include <boost/mpl/comparison.hpp>
Chris@102 20 #include <boost/math/tools/big_constant.hpp>
Chris@102 21 #include <boost/math/special_functions/polygamma.hpp>
Chris@102 22
Chris@102 23 namespace boost{
Chris@102 24 namespace math{
Chris@102 25 namespace detail{
Chris@102 26
Chris@102 27 template<class T, class Policy>
Chris@102 28 T polygamma_imp(const int n, T x, const Policy &pol);
Chris@102 29
Chris@102 30 template <class T, class Policy>
Chris@102 31 T trigamma_prec(T x, const mpl::int_<53>*, const Policy&)
Chris@102 32 {
Chris@102 33 // Max error in interpolated form: 3.736e-017
Chris@102 34 static const T offset = BOOST_MATH_BIG_CONSTANT(T, 53, 2.1093254089355469);
Chris@102 35 static const T P_1_2[] = {
Chris@102 36 BOOST_MATH_BIG_CONSTANT(T, 53, -1.1093280605946045),
Chris@102 37 BOOST_MATH_BIG_CONSTANT(T, 53, -3.8310674472619321),
Chris@102 38 BOOST_MATH_BIG_CONSTANT(T, 53, -3.3703848401898283),
Chris@102 39 BOOST_MATH_BIG_CONSTANT(T, 53, 0.28080574467981213),
Chris@102 40 BOOST_MATH_BIG_CONSTANT(T, 53, 1.6638069578676164),
Chris@102 41 BOOST_MATH_BIG_CONSTANT(T, 53, 0.64468386819102836),
Chris@102 42 };
Chris@102 43 static const T Q_1_2[] = {
Chris@102 44 BOOST_MATH_BIG_CONSTANT(T, 53, 1.0),
Chris@102 45 BOOST_MATH_BIG_CONSTANT(T, 53, 3.4535389668541151),
Chris@102 46 BOOST_MATH_BIG_CONSTANT(T, 53, 4.5208926987851437),
Chris@102 47 BOOST_MATH_BIG_CONSTANT(T, 53, 2.7012734178351534),
Chris@102 48 BOOST_MATH_BIG_CONSTANT(T, 53, 0.64468798399785611),
Chris@102 49 BOOST_MATH_BIG_CONSTANT(T, 53, -0.20314516859987728e-6),
Chris@102 50 };
Chris@102 51 // Max error in interpolated form: 1.159e-017
Chris@102 52 static const T P_2_4[] = {
Chris@102 53 BOOST_MATH_BIG_CONSTANT(T, 53, -0.13803835004508849e-7),
Chris@102 54 BOOST_MATH_BIG_CONSTANT(T, 53, 0.50000049158540261),
Chris@102 55 BOOST_MATH_BIG_CONSTANT(T, 53, 1.6077979838469348),
Chris@102 56 BOOST_MATH_BIG_CONSTANT(T, 53, 2.5645435828098254),
Chris@102 57 BOOST_MATH_BIG_CONSTANT(T, 53, 2.0534873203680393),
Chris@102 58 BOOST_MATH_BIG_CONSTANT(T, 53, 0.74566981111565923),
Chris@102 59 };
Chris@102 60 static const T Q_2_4[] = {
Chris@102 61 BOOST_MATH_BIG_CONSTANT(T, 53, 1.0),
Chris@102 62 BOOST_MATH_BIG_CONSTANT(T, 53, 2.8822787662376169),
Chris@102 63 BOOST_MATH_BIG_CONSTANT(T, 53, 4.1681660554090917),
Chris@102 64 BOOST_MATH_BIG_CONSTANT(T, 53, 2.7853527819234466),
Chris@102 65 BOOST_MATH_BIG_CONSTANT(T, 53, 0.74967671848044792),
Chris@102 66 BOOST_MATH_BIG_CONSTANT(T, 53, -0.00057069112416246805),
Chris@102 67 };
Chris@102 68 // Maximum Deviation Found: 6.896e-018
Chris@102 69 // Expected Error Term : -6.895e-018
Chris@102 70 // Maximum Relative Change in Control Points : 8.497e-004
Chris@102 71 static const T P_4_inf[] = {
Chris@102 72 0.68947581948701249e-17L,
Chris@102 73 0.49999999999998975L,
Chris@102 74 1.0177274392923795L,
Chris@102 75 2.498208511343429L,
Chris@102 76 2.1921221359427595L,
Chris@102 77 1.5897035272532764L,
Chris@102 78 0.40154388356961734L,
Chris@102 79 };
Chris@102 80 static const T Q_4_inf[] = {
Chris@102 81 1.0L,
Chris@102 82 1.7021215452463932L,
Chris@102 83 4.4290431747556469L,
Chris@102 84 2.9745631894384922L,
Chris@102 85 2.3013614809773616L,
Chris@102 86 0.28360399799075752L,
Chris@102 87 0.022892987908906897L,
Chris@102 88 };
Chris@102 89
Chris@102 90 if(x <= 2)
Chris@102 91 {
Chris@102 92 return (offset + boost::math::tools::evaluate_polynomial(P_1_2, x) / tools::evaluate_polynomial(Q_1_2, x)) / (x * x);
Chris@102 93 }
Chris@102 94 else if(x <= 4)
Chris@102 95 {
Chris@102 96 T y = 1 / x;
Chris@102 97 return (1 + tools::evaluate_polynomial(P_2_4, y) / tools::evaluate_polynomial(Q_2_4, y)) / x;
Chris@102 98 }
Chris@102 99 T y = 1 / x;
Chris@102 100 return (1 + tools::evaluate_polynomial(P_4_inf, y) / tools::evaluate_polynomial(Q_4_inf, y)) / x;
Chris@102 101 }
Chris@102 102
Chris@102 103 template <class T, class Policy>
Chris@102 104 T trigamma_prec(T x, const mpl::int_<64>*, const Policy&)
Chris@102 105 {
Chris@102 106 // Max error in interpolated form: 1.178e-020
Chris@102 107 static const T offset_1_2 = BOOST_MATH_BIG_CONSTANT(T, 64, 2.109325408935546875);
Chris@102 108 static const T P_1_2[] = {
Chris@102 109 BOOST_MATH_BIG_CONSTANT(T, 64, -1.10932535608960258341),
Chris@102 110 BOOST_MATH_BIG_CONSTANT(T, 64, -4.18793841543017129052),
Chris@102 111 BOOST_MATH_BIG_CONSTANT(T, 64, -4.63865531898487734531),
Chris@102 112 BOOST_MATH_BIG_CONSTANT(T, 64, -0.919832884430500908047),
Chris@102 113 BOOST_MATH_BIG_CONSTANT(T, 64, 1.68074038333180423012),
Chris@102 114 BOOST_MATH_BIG_CONSTANT(T, 64, 1.21172611429185622377),
Chris@102 115 BOOST_MATH_BIG_CONSTANT(T, 64, 0.259635673503366427284),
Chris@102 116 };
Chris@102 117 static const T Q_1_2[] = {
Chris@102 118 BOOST_MATH_BIG_CONSTANT(T, 64, 1.0),
Chris@102 119 BOOST_MATH_BIG_CONSTANT(T, 64, 3.77521119359546982995),
Chris@102 120 BOOST_MATH_BIG_CONSTANT(T, 64, 5.664338024578956321),
Chris@102 121 BOOST_MATH_BIG_CONSTANT(T, 64, 4.25995134879278028361),
Chris@102 122 BOOST_MATH_BIG_CONSTANT(T, 64, 1.62956638448940402182),
Chris@102 123 BOOST_MATH_BIG_CONSTANT(T, 64, 0.259635512844691089868),
Chris@102 124 BOOST_MATH_BIG_CONSTANT(T, 64, 0.629642219810618032207e-8),
Chris@102 125 };
Chris@102 126 // Max error in interpolated form: 3.912e-020
Chris@102 127 static const T P_2_8[] = {
Chris@102 128 BOOST_MATH_BIG_CONSTANT(T, 64, -0.387540035162952880976e-11),
Chris@102 129 BOOST_MATH_BIG_CONSTANT(T, 64, 0.500000000276430504),
Chris@102 130 BOOST_MATH_BIG_CONSTANT(T, 64, 3.21926880986360957306),
Chris@102 131 BOOST_MATH_BIG_CONSTANT(T, 64, 10.2550347708483445775),
Chris@102 132 BOOST_MATH_BIG_CONSTANT(T, 64, 18.9002075150709144043),
Chris@102 133 BOOST_MATH_BIG_CONSTANT(T, 64, 21.0357215832399705625),
Chris@102 134 BOOST_MATH_BIG_CONSTANT(T, 64, 13.4346512182925923978),
Chris@102 135 BOOST_MATH_BIG_CONSTANT(T, 64, 3.98656291026448279118),
Chris@102 136 };
Chris@102 137 static const T Q_2_8[] = {
Chris@102 138 BOOST_MATH_BIG_CONSTANT(T, 64, 1.0),
Chris@102 139 BOOST_MATH_BIG_CONSTANT(T, 64, 6.10520430478613667724),
Chris@102 140 BOOST_MATH_BIG_CONSTANT(T, 64, 18.475001060603645512),
Chris@102 141 BOOST_MATH_BIG_CONSTANT(T, 64, 31.7087534567758405638),
Chris@102 142 BOOST_MATH_BIG_CONSTANT(T, 64, 31.908814523890465398),
Chris@102 143 BOOST_MATH_BIG_CONSTANT(T, 64, 17.4175479039227084798),
Chris@102 144 BOOST_MATH_BIG_CONSTANT(T, 64, 3.98749106958394941276),
Chris@102 145 BOOST_MATH_BIG_CONSTANT(T, 64, -0.000115917322224411128566),
Chris@102 146 };
Chris@102 147 // Maximum Deviation Found: 2.635e-020
Chris@102 148 // Expected Error Term : 2.635e-020
Chris@102 149 // Maximum Relative Change in Control Points : 1.791e-003
Chris@102 150 static const T P_8_inf[] = {
Chris@102 151 BOOST_MATH_BIG_CONSTANT(T, 64, -0.263527875092466899848e-19),
Chris@102 152 BOOST_MATH_BIG_CONSTANT(T, 64, 0.500000000000000058145),
Chris@102 153 BOOST_MATH_BIG_CONSTANT(T, 64, 0.0730121433777364138677),
Chris@102 154 BOOST_MATH_BIG_CONSTANT(T, 64, 1.94505878379957149534),
Chris@102 155 BOOST_MATH_BIG_CONSTANT(T, 64, 0.0517092358874932620529),
Chris@102 156 BOOST_MATH_BIG_CONSTANT(T, 64, 1.07995383547483921121),
Chris@102 157 };
Chris@102 158 static const T Q_8_inf[] = {
Chris@102 159 BOOST_MATH_BIG_CONSTANT(T, 64, 1.0),
Chris@102 160 BOOST_MATH_BIG_CONSTANT(T, 64, -0.187309046577818095504),
Chris@102 161 BOOST_MATH_BIG_CONSTANT(T, 64, 3.95255391645238842975),
Chris@102 162 BOOST_MATH_BIG_CONSTANT(T, 64, -1.14743283327078949087),
Chris@102 163 BOOST_MATH_BIG_CONSTANT(T, 64, 2.52989799376344914499),
Chris@102 164 BOOST_MATH_BIG_CONSTANT(T, 64, -0.627414303172402506396),
Chris@102 165 BOOST_MATH_BIG_CONSTANT(T, 64, 0.141554248216425512536),
Chris@102 166 };
Chris@102 167
Chris@102 168 if(x <= 2)
Chris@102 169 {
Chris@102 170 return (offset_1_2 + boost::math::tools::evaluate_polynomial(P_1_2, x) / tools::evaluate_polynomial(Q_1_2, x)) / (x * x);
Chris@102 171 }
Chris@102 172 else if(x <= 8)
Chris@102 173 {
Chris@102 174 T y = 1 / x;
Chris@102 175 return (1 + tools::evaluate_polynomial(P_2_8, y) / tools::evaluate_polynomial(Q_2_8, y)) / x;
Chris@102 176 }
Chris@102 177 T y = 1 / x;
Chris@102 178 return (1 + tools::evaluate_polynomial(P_8_inf, y) / tools::evaluate_polynomial(Q_8_inf, y)) / x;
Chris@102 179 }
Chris@102 180
Chris@102 181 template <class T, class Policy>
Chris@102 182 T trigamma_prec(T x, const mpl::int_<113>*, const Policy&)
Chris@102 183 {
Chris@102 184 // Max error in interpolated form: 1.916e-035
Chris@102 185
Chris@102 186 static const T P_1_2[] = {
Chris@102 187 BOOST_MATH_BIG_CONSTANT(T, 113, -0.999999999999999082554457936871832533),
Chris@102 188 BOOST_MATH_BIG_CONSTANT(T, 113, -4.71237311120865266379041700054847734),
Chris@102 189 BOOST_MATH_BIG_CONSTANT(T, 113, -7.94125711970499027763789342500817316),
Chris@102 190 BOOST_MATH_BIG_CONSTANT(T, 113, -5.74657746697664735258222071695644535),
Chris@102 191 BOOST_MATH_BIG_CONSTANT(T, 113, -0.404213349456398905981223965160595687),
Chris@102 192 BOOST_MATH_BIG_CONSTANT(T, 113, 2.47877781178642876561595890095758896),
Chris@102 193 BOOST_MATH_BIG_CONSTANT(T, 113, 2.07714151702455125992166949812126433),
Chris@102 194 BOOST_MATH_BIG_CONSTANT(T, 113, 0.858877899162360138844032265418028567),
Chris@102 195 BOOST_MATH_BIG_CONSTANT(T, 113, 0.20499222604410032375789018837922397),
Chris@102 196 BOOST_MATH_BIG_CONSTANT(T, 113, 0.0272103140348194747360175268778415049),
Chris@102 197 BOOST_MATH_BIG_CONSTANT(T, 113, 0.0015764849020876949848954081173520686),
Chris@102 198 };
Chris@102 199 static const T Q_1_2[] = {
Chris@102 200 BOOST_MATH_BIG_CONSTANT(T, 113, 1.0),
Chris@102 201 BOOST_MATH_BIG_CONSTANT(T, 113, 4.71237311120863419878375031457715223),
Chris@102 202 BOOST_MATH_BIG_CONSTANT(T, 113, 9.58619118655339853449127952145877467),
Chris@102 203 BOOST_MATH_BIG_CONSTANT(T, 113, 11.0940067269829372437561421279054968),
Chris@102 204 BOOST_MATH_BIG_CONSTANT(T, 113, 8.09075424749327792073276309969037885),
Chris@102 205 BOOST_MATH_BIG_CONSTANT(T, 113, 3.87705890159891405185343806884451286),
Chris@102 206 BOOST_MATH_BIG_CONSTANT(T, 113, 1.22758678701914477836330837816976782),
Chris@102 207 BOOST_MATH_BIG_CONSTANT(T, 113, 0.249092040606385004109672077814668716),
Chris@102 208 BOOST_MATH_BIG_CONSTANT(T, 113, 0.0295750413900655597027079600025569048),
Chris@102 209 BOOST_MATH_BIG_CONSTANT(T, 113, 0.00157648490200498142247694709728858139),
Chris@102 210 BOOST_MATH_BIG_CONSTANT(T, 113, 0.161264050344059471721062360645432809e-14),
Chris@102 211 };
Chris@102 212
Chris@102 213 // Max error in interpolated form: 8.958e-035
Chris@102 214 static const T P_2_4[] = {
Chris@102 215 BOOST_MATH_BIG_CONSTANT(T, 113, -2.55843734739907925764326773972215085),
Chris@102 216 BOOST_MATH_BIG_CONSTANT(T, 113, -12.2830208240542011967952466273455887),
Chris@102 217 BOOST_MATH_BIG_CONSTANT(T, 113, -23.9195022162767993526575786066414403),
Chris@102 218 BOOST_MATH_BIG_CONSTANT(T, 113, -24.9256431504823483094158828285470862),
Chris@102 219 BOOST_MATH_BIG_CONSTANT(T, 113, -14.7979122765478779075108064826412285),
Chris@102 220 BOOST_MATH_BIG_CONSTANT(T, 113, -4.46654453928610666393276765059122272),
Chris@102 221 BOOST_MATH_BIG_CONSTANT(T, 113, -0.0191439033405649675717082465687845002),
Chris@102 222 BOOST_MATH_BIG_CONSTANT(T, 113, 0.515412052554351265708917209749037352),
Chris@102 223 BOOST_MATH_BIG_CONSTANT(T, 113, 0.195378348786064304378247325360320038),
Chris@102 224 BOOST_MATH_BIG_CONSTANT(T, 113, 0.0334761282624174313035014426794245393),
Chris@102 225 BOOST_MATH_BIG_CONSTANT(T, 113, 0.002373665205942206348500250056602687),
Chris@102 226 };
Chris@102 227 static const T Q_2_4[] = {
Chris@102 228 BOOST_MATH_BIG_CONSTANT(T, 113, 1.0),
Chris@102 229 BOOST_MATH_BIG_CONSTANT(T, 113, 4.80098558454419907830670928248659245),
Chris@102 230 BOOST_MATH_BIG_CONSTANT(T, 113, 9.99220727843170133895059300223445265),
Chris@102 231 BOOST_MATH_BIG_CONSTANT(T, 113, 11.8896146167631330735386697123464976),
Chris@102 232 BOOST_MATH_BIG_CONSTANT(T, 113, 8.96613256683809091593793565879092581),
Chris@102 233 BOOST_MATH_BIG_CONSTANT(T, 113, 4.47254136149624110878909334574485751),
Chris@102 234 BOOST_MATH_BIG_CONSTANT(T, 113, 1.48600982028196527372434773913633152),
Chris@102 235 BOOST_MATH_BIG_CONSTANT(T, 113, 0.319570735766764237068541501137990078),
Chris@102 236 BOOST_MATH_BIG_CONSTANT(T, 113, 0.0407358345787680953107374215319322066),
Chris@102 237 BOOST_MATH_BIG_CONSTANT(T, 113, 0.00237366520593271641375755486420859837),
Chris@102 238 BOOST_MATH_BIG_CONSTANT(T, 113, 0.239554887903526152679337256236302116e-15),
Chris@102 239 BOOST_MATH_BIG_CONSTANT(T, 113, -0.294749244740618656265237072002026314e-17),
Chris@102 240 };
Chris@102 241
Chris@102 242 static const T y_offset_2_4 = BOOST_MATH_BIG_CONSTANT(T, 113, 3.558437347412109375);
Chris@102 243
Chris@102 244 // Max error in interpolated form: 4.319e-035
Chris@102 245 static const T P_4_8[] = {
Chris@102 246 BOOST_MATH_BIG_CONSTANT(T, 113, 0.166626112697021464248967707021688845e-16),
Chris@102 247 BOOST_MATH_BIG_CONSTANT(T, 113, 0.499999999999997739552090249208808197),
Chris@102 248 BOOST_MATH_BIG_CONSTANT(T, 113, 6.40270945019053817915772473771553187),
Chris@102 249 BOOST_MATH_BIG_CONSTANT(T, 113, 41.3833374155000608013677627389343329),
Chris@102 250 BOOST_MATH_BIG_CONSTANT(T, 113, 166.803341854562809335667241074035245),
Chris@102 251 BOOST_MATH_BIG_CONSTANT(T, 113, 453.39964786925369319960722793414521),
Chris@102 252 BOOST_MATH_BIG_CONSTANT(T, 113, 851.153712317697055375935433362983944),
Chris@102 253 BOOST_MATH_BIG_CONSTANT(T, 113, 1097.70657567285059133109286478004458),
Chris@102 254 BOOST_MATH_BIG_CONSTANT(T, 113, 938.431232478455316020076349367632922),
Chris@102 255 BOOST_MATH_BIG_CONSTANT(T, 113, 487.268001604651932322080970189930074),
Chris@102 256 BOOST_MATH_BIG_CONSTANT(T, 113, 119.953445242335730062471193124820659),
Chris@102 257 };
Chris@102 258 static const T Q_4_8[] = {
Chris@102 259 BOOST_MATH_BIG_CONSTANT(T, 113, 1.0),
Chris@102 260 BOOST_MATH_BIG_CONSTANT(T, 113, 12.4720855670474488978638945855932398),
Chris@102 261 BOOST_MATH_BIG_CONSTANT(T, 113, 78.6093129753298570701376952709727391),
Chris@102 262 BOOST_MATH_BIG_CONSTANT(T, 113, 307.470246050318322489781182863190127),
Chris@102 263 BOOST_MATH_BIG_CONSTANT(T, 113, 805.140686101151538537565264188630079),
Chris@102 264 BOOST_MATH_BIG_CONSTANT(T, 113, 1439.12019760292146454787601409644413),
Chris@102 265 BOOST_MATH_BIG_CONSTANT(T, 113, 1735.6105285756048831268586001383127),
Chris@102 266 BOOST_MATH_BIG_CONSTANT(T, 113, 1348.32500712856328019355198611280536),
Chris@102 267 BOOST_MATH_BIG_CONSTANT(T, 113, 607.225985860570846699704222144650563),
Chris@102 268 BOOST_MATH_BIG_CONSTANT(T, 113, 119.952317857277045332558673164517227),
Chris@102 269 BOOST_MATH_BIG_CONSTANT(T, 113, 0.000140165918355036060868680809129436084),
Chris@102 270 };
Chris@102 271
Chris@102 272 // Maximum Deviation Found: 2.867e-035
Chris@102 273 // Expected Error Term : 2.866e-035
Chris@102 274 // Maximum Relative Change in Control Points : 2.662e-004
Chris@102 275 static const T P_8_16[] = {
Chris@102 276 BOOST_MATH_BIG_CONSTANT(T, 113, -0.184828315274146610610872315609837439e-19),
Chris@102 277 BOOST_MATH_BIG_CONSTANT(T, 113, 0.500000000000000004122475157735807738),
Chris@102 278 BOOST_MATH_BIG_CONSTANT(T, 113, 3.02533865247313349284875558880415875),
Chris@102 279 BOOST_MATH_BIG_CONSTANT(T, 113, 13.5995927517457371243039532492642734),
Chris@102 280 BOOST_MATH_BIG_CONSTANT(T, 113, 35.3132224283087906757037999452941588),
Chris@102 281 BOOST_MATH_BIG_CONSTANT(T, 113, 67.1639424550714159157603179911505619),
Chris@102 282 BOOST_MATH_BIG_CONSTANT(T, 113, 83.5767733658513967581959839367419891),
Chris@102 283 BOOST_MATH_BIG_CONSTANT(T, 113, 71.073491212235705900866411319363501),
Chris@102 284 BOOST_MATH_BIG_CONSTANT(T, 113, 35.8621515614725564575893663483998663),
Chris@102 285 BOOST_MATH_BIG_CONSTANT(T, 113, 8.72152231639983491987779743154333318),
Chris@102 286 };
Chris@102 287 static const T Q_8_16[] = {
Chris@102 288 BOOST_MATH_BIG_CONSTANT(T, 113, 1.0),
Chris@102 289 BOOST_MATH_BIG_CONSTANT(T, 113, 5.71734397161293452310624822415866372),
Chris@102 290 BOOST_MATH_BIG_CONSTANT(T, 113, 25.293404179620438179337103263274815),
Chris@102 291 BOOST_MATH_BIG_CONSTANT(T, 113, 62.2619767967468199111077640625328469),
Chris@102 292 BOOST_MATH_BIG_CONSTANT(T, 113, 113.955048909238993473389714972250235),
Chris@102 293 BOOST_MATH_BIG_CONSTANT(T, 113, 130.807138328938966981862203944329408),
Chris@102 294 BOOST_MATH_BIG_CONSTANT(T, 113, 102.423146902337654110717764213057753),
Chris@102 295 BOOST_MATH_BIG_CONSTANT(T, 113, 44.0424772805245202514468199602123565),
Chris@102 296 BOOST_MATH_BIG_CONSTANT(T, 113, 8.89898032477904072082994913461386099),
Chris@102 297 BOOST_MATH_BIG_CONSTANT(T, 113, -0.0296627336872039988632793863671456398),
Chris@102 298 };
Chris@102 299 // Maximum Deviation Found: 1.079e-035
Chris@102 300 // Expected Error Term : -1.079e-035
Chris@102 301 // Maximum Relative Change in Control Points : 7.884e-003
Chris@102 302 static const T P_16_inf[] = {
Chris@102 303 BOOST_MATH_BIG_CONSTANT(T, 113, 0.0),
Chris@102 304 BOOST_MATH_BIG_CONSTANT(T, 113, 0.500000000000000000000000000000087317),
Chris@102 305 BOOST_MATH_BIG_CONSTANT(T, 113, 0.345625669885456215194494735902663968),
Chris@102 306 BOOST_MATH_BIG_CONSTANT(T, 113, 9.62895499360842232127552650044647769),
Chris@102 307 BOOST_MATH_BIG_CONSTANT(T, 113, 3.5936085382439026269301003761320812),
Chris@102 308 BOOST_MATH_BIG_CONSTANT(T, 113, 49.459599118438883265036646019410669),
Chris@102 309 BOOST_MATH_BIG_CONSTANT(T, 113, 7.77519237321893917784735690560496607),
Chris@102 310 BOOST_MATH_BIG_CONSTANT(T, 113, 74.4536074488178075948642351179304121),
Chris@102 311 BOOST_MATH_BIG_CONSTANT(T, 113, 2.75209340397069050436806159297952699),
Chris@102 312 BOOST_MATH_BIG_CONSTANT(T, 113, 23.9292359711471667884504840186561598),
Chris@102 313 };
Chris@102 314 static const T Q_16_inf[] = {
Chris@102 315 BOOST_MATH_BIG_CONSTANT(T, 113, 1.0),
Chris@102 316 BOOST_MATH_BIG_CONSTANT(T, 113, 0.357918006437579097055656138920742037),
Chris@102 317 BOOST_MATH_BIG_CONSTANT(T, 113, 19.1386039850709849435325005484512944),
Chris@102 318 BOOST_MATH_BIG_CONSTANT(T, 113, 0.874349081464143606016221431763364517),
Chris@102 319 BOOST_MATH_BIG_CONSTANT(T, 113, 98.6516097434855572678195488061432509),
Chris@102 320 BOOST_MATH_BIG_CONSTANT(T, 113, -16.1051972833382893468655223662534306),
Chris@102 321 BOOST_MATH_BIG_CONSTANT(T, 113, 154.316860216253720989145047141653727),
Chris@102 322 BOOST_MATH_BIG_CONSTANT(T, 113, -40.2026880424378986053105969312264534),
Chris@102 323 BOOST_MATH_BIG_CONSTANT(T, 113, 60.1679136674264778074736441126810223),
Chris@102 324 BOOST_MATH_BIG_CONSTANT(T, 113, -13.3414844622256422644504472438320114),
Chris@102 325 BOOST_MATH_BIG_CONSTANT(T, 113, 2.53795636200649908779512969030363442),
Chris@102 326 };
Chris@102 327
Chris@102 328 if(x <= 2)
Chris@102 329 {
Chris@102 330 return (2 + boost::math::tools::evaluate_polynomial(P_1_2, x) / tools::evaluate_polynomial(Q_1_2, x)) / (x * x);
Chris@102 331 }
Chris@102 332 else if(x <= 4)
Chris@102 333 {
Chris@102 334 return (y_offset_2_4 + boost::math::tools::evaluate_polynomial(P_2_4, x) / tools::evaluate_polynomial(Q_2_4, x)) / (x * x);
Chris@102 335 }
Chris@102 336 else if(x <= 8)
Chris@102 337 {
Chris@102 338 T y = 1 / x;
Chris@102 339 return (1 + tools::evaluate_polynomial(P_4_8, y) / tools::evaluate_polynomial(Q_4_8, y)) / x;
Chris@102 340 }
Chris@102 341 else if(x <= 16)
Chris@102 342 {
Chris@102 343 T y = 1 / x;
Chris@102 344 return (1 + tools::evaluate_polynomial(P_8_16, y) / tools::evaluate_polynomial(Q_8_16, y)) / x;
Chris@102 345 }
Chris@102 346 T y = 1 / x;
Chris@102 347 return (1 + tools::evaluate_polynomial(P_16_inf, y) / tools::evaluate_polynomial(Q_16_inf, y)) / x;
Chris@102 348 }
Chris@102 349
Chris@102 350 template <class T, class Tag, class Policy>
Chris@102 351 T trigamma_imp(T x, const Tag* t, const Policy& pol)
Chris@102 352 {
Chris@102 353 //
Chris@102 354 // This handles reflection of negative arguments, and all our
Chris@102 355 // error handling, then forwards to the T-specific approximation.
Chris@102 356 //
Chris@102 357 BOOST_MATH_STD_USING // ADL of std functions.
Chris@102 358
Chris@102 359 T result = 0;
Chris@102 360 //
Chris@102 361 // Check for negative arguments and use reflection:
Chris@102 362 //
Chris@102 363 if(x <= 0)
Chris@102 364 {
Chris@102 365 // Reflect:
Chris@102 366 T z = 1 - x;
Chris@102 367 // Argument reduction for tan:
Chris@102 368 if(floor(x) == x)
Chris@102 369 {
Chris@102 370 return policies::raise_pole_error<T>("boost::math::trigamma<%1%>(%1%)", 0, (1-x), pol);
Chris@102 371 }
Chris@102 372 T s = fabs(x) < fabs(z) ? boost::math::sin_pi(x, pol) : boost::math::sin_pi(z, pol);
Chris@102 373 return -trigamma_imp(z, t, pol) + boost::math::pow<2>(constants::pi<T>()) / (s * s);
Chris@102 374 }
Chris@102 375 if(x < 1)
Chris@102 376 {
Chris@102 377 result = 1 / (x * x);
Chris@102 378 x += 1;
Chris@102 379 }
Chris@102 380 return result + trigamma_prec(x, t, pol);
Chris@102 381 }
Chris@102 382
Chris@102 383 template <class T, class Policy>
Chris@102 384 T trigamma_imp(T x, const mpl::int_<0>*, const Policy& pol)
Chris@102 385 {
Chris@102 386 return polygamma_imp(1, x, pol);
Chris@102 387 }
Chris@102 388 //
Chris@102 389 // Initializer: ensure all our constants are initialized prior to the first call of main:
Chris@102 390 //
Chris@102 391 template <class T, class Policy>
Chris@102 392 struct trigamma_initializer
Chris@102 393 {
Chris@102 394 struct init
Chris@102 395 {
Chris@102 396 init()
Chris@102 397 {
Chris@102 398 typedef typename policies::precision<T, Policy>::type precision_type;
Chris@102 399 do_init(mpl::bool_<precision_type::value && (precision_type::value <= 113)>());
Chris@102 400 }
Chris@102 401 void do_init(const mpl::true_&)
Chris@102 402 {
Chris@102 403 boost::math::trigamma(T(2.5), Policy());
Chris@102 404 }
Chris@102 405 void do_init(const mpl::false_&){}
Chris@102 406 void force_instantiate()const{}
Chris@102 407 };
Chris@102 408 static const init initializer;
Chris@102 409 static void force_instantiate()
Chris@102 410 {
Chris@102 411 initializer.force_instantiate();
Chris@102 412 }
Chris@102 413 };
Chris@102 414
Chris@102 415 template <class T, class Policy>
Chris@102 416 const typename trigamma_initializer<T, Policy>::init trigamma_initializer<T, Policy>::initializer;
Chris@102 417
Chris@102 418 } // namespace detail
Chris@102 419
Chris@102 420 template <class T, class Policy>
Chris@102 421 inline typename tools::promote_args<T>::type
Chris@102 422 trigamma(T x, const Policy&)
Chris@102 423 {
Chris@102 424 typedef typename tools::promote_args<T>::type result_type;
Chris@102 425 typedef typename policies::evaluation<result_type, Policy>::type value_type;
Chris@102 426 typedef typename policies::precision<T, Policy>::type precision_type;
Chris@102 427 typedef typename mpl::if_<
Chris@102 428 mpl::or_<
Chris@102 429 mpl::less_equal<precision_type, mpl::int_<0> >,
Chris@102 430 mpl::greater<precision_type, mpl::int_<114> >
Chris@102 431 >,
Chris@102 432 mpl::int_<0>,
Chris@102 433 typename mpl::if_<
Chris@102 434 mpl::less<precision_type, mpl::int_<54> >,
Chris@102 435 mpl::int_<53>,
Chris@102 436 typename mpl::if_<
Chris@102 437 mpl::less<precision_type, mpl::int_<65> >,
Chris@102 438 mpl::int_<64>,
Chris@102 439 mpl::int_<113>
Chris@102 440 >::type
Chris@102 441 >::type
Chris@102 442 >::type tag_type;
Chris@102 443
Chris@102 444 typedef typename policies::normalise<
Chris@102 445 Policy,
Chris@102 446 policies::promote_float<false>,
Chris@102 447 policies::promote_double<false>,
Chris@102 448 policies::discrete_quantile<>,
Chris@102 449 policies::assert_undefined<> >::type forwarding_policy;
Chris@102 450
Chris@102 451 // Force initialization of constants:
Chris@102 452 detail::trigamma_initializer<value_type, forwarding_policy>::force_instantiate();
Chris@102 453
Chris@102 454 return policies::checked_narrowing_cast<result_type, Policy>(detail::trigamma_imp(
Chris@102 455 static_cast<value_type>(x),
Chris@102 456 static_cast<const tag_type*>(0), forwarding_policy()), "boost::math::trigamma<%1%>(%1%)");
Chris@102 457 }
Chris@102 458
Chris@102 459 template <class T>
Chris@102 460 inline typename tools::promote_args<T>::type
Chris@102 461 trigamma(T x)
Chris@102 462 {
Chris@102 463 return trigamma(x, policies::policy<>());
Chris@102 464 }
Chris@102 465
Chris@102 466 } // namespace math
Chris@102 467 } // namespace boost
Chris@102 468 #endif
Chris@102 469