cmath.hpp 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676
  1. // Boost.Units - A C++ library for zero-overhead dimensional analysis and
  2. // unit/quantity manipulation and conversion
  3. //
  4. // Copyright (C) 2003-2008 Matthias Christian Schabel
  5. // Copyright (C) 2007-2008 Steven Watanabe
  6. //
  7. // Distributed under the Boost Software License, Version 1.0. (See
  8. // accompanying file LICENSE_1_0.txt or copy at
  9. // http://www.boost.org/LICENSE_1_0.txt)
  10. #ifndef BOOST_UNITS_CMATH_HPP
  11. #define BOOST_UNITS_CMATH_HPP
  12. #include <boost/config/no_tr1/cmath.hpp>
  13. #include <cstdlib>
  14. #include <boost/math/special_functions/fpclassify.hpp>
  15. #include <boost/math/special_functions/hypot.hpp>
  16. #include <boost/math/special_functions/next.hpp>
  17. #include <boost/math/special_functions/round.hpp>
  18. #include <boost/math/special_functions/sign.hpp>
  19. #include <boost/units/dimensionless_quantity.hpp>
  20. #include <boost/units/pow.hpp>
  21. #include <boost/units/quantity.hpp>
  22. #include <boost/units/detail/cmath_impl.hpp>
  23. #include <boost/units/detail/dimensionless_unit.hpp>
  24. #include <boost/units/systems/si/plane_angle.hpp>
  25. /// \file
  26. /// \brief Overloads of functions in \<cmath\> for quantities.
  27. /// \details Only functions for which a dimensionally-correct result type
  28. /// can be determined are overloaded.
  29. /// All functions work with dimensionless quantities.
  30. // BOOST_PREVENT_MACRO_SUBSTITUTION is needed on certain compilers that define
  31. // some <cmath> functions as macros; it is used for all functions even though it
  32. // isn't necessary -- I didn't want to think :)
  33. //
  34. // the form using namespace detail; return(f(x)); is used
  35. // to enable ADL for UDTs.
  36. namespace boost {
  37. namespace units {
  38. template<class Unit,class Y>
  39. inline
  40. bool
  41. isfinite BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q)
  42. {
  43. using boost::math::isfinite;
  44. return isfinite BOOST_PREVENT_MACRO_SUBSTITUTION (q.value());
  45. }
  46. template<class Unit,class Y>
  47. inline
  48. bool
  49. isinf BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q)
  50. {
  51. using boost::math::isinf;
  52. return isinf BOOST_PREVENT_MACRO_SUBSTITUTION (q.value());
  53. }
  54. template<class Unit,class Y>
  55. inline
  56. bool
  57. isnan BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q)
  58. {
  59. using boost::math::isnan;
  60. return isnan BOOST_PREVENT_MACRO_SUBSTITUTION (q.value());
  61. }
  62. template<class Unit,class Y>
  63. inline
  64. bool
  65. isnormal BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q)
  66. {
  67. using boost::math::isnormal;
  68. return isnormal BOOST_PREVENT_MACRO_SUBSTITUTION (q.value());
  69. }
  70. template<class Unit,class Y>
  71. inline
  72. bool
  73. isgreater BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q1,
  74. const quantity<Unit,Y>& q2)
  75. {
  76. using namespace detail;
  77. return isgreater BOOST_PREVENT_MACRO_SUBSTITUTION (q1.value(),q2.value());
  78. }
  79. template<class Unit,class Y>
  80. inline
  81. bool
  82. isgreaterequal BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q1,
  83. const quantity<Unit,Y>& q2)
  84. {
  85. using namespace detail;
  86. return isgreaterequal BOOST_PREVENT_MACRO_SUBSTITUTION (q1.value(),q2.value());
  87. }
  88. template<class Unit,class Y>
  89. inline
  90. bool
  91. isless BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q1,
  92. const quantity<Unit,Y>& q2)
  93. {
  94. using namespace detail;
  95. return isless BOOST_PREVENT_MACRO_SUBSTITUTION (q1.value(),q2.value());
  96. }
  97. template<class Unit,class Y>
  98. inline
  99. bool
  100. islessequal BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q1,
  101. const quantity<Unit,Y>& q2)
  102. {
  103. using namespace detail;
  104. return islessequal BOOST_PREVENT_MACRO_SUBSTITUTION (q1.value(),q2.value());
  105. }
  106. template<class Unit,class Y>
  107. inline
  108. bool
  109. islessgreater BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q1,
  110. const quantity<Unit,Y>& q2)
  111. {
  112. using namespace detail;
  113. return islessgreater BOOST_PREVENT_MACRO_SUBSTITUTION (q1.value(),q2.value());
  114. }
  115. template<class Unit,class Y>
  116. inline
  117. bool
  118. isunordered BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q1,
  119. const quantity<Unit,Y>& q2)
  120. {
  121. using namespace detail;
  122. return isunordered BOOST_PREVENT_MACRO_SUBSTITUTION (q1.value(),q2.value());
  123. }
  124. template<class Unit,class Y>
  125. inline
  126. quantity<Unit,Y>
  127. abs BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q)
  128. {
  129. using std::abs;
  130. typedef quantity<Unit,Y> quantity_type;
  131. return quantity_type::from_value(abs BOOST_PREVENT_MACRO_SUBSTITUTION (q.value()));
  132. }
  133. template<class Unit,class Y>
  134. inline
  135. quantity<Unit,Y>
  136. ceil BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q)
  137. {
  138. using std::ceil;
  139. typedef quantity<Unit,Y> quantity_type;
  140. return quantity_type::from_value(ceil BOOST_PREVENT_MACRO_SUBSTITUTION (q.value()));
  141. }
  142. template<class Unit,class Y>
  143. inline
  144. quantity<Unit,Y>
  145. copysign BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q1,
  146. const quantity<Unit,Y>& q2)
  147. {
  148. using boost::math::copysign;
  149. typedef quantity<Unit,Y> quantity_type;
  150. return quantity_type::from_value(copysign BOOST_PREVENT_MACRO_SUBSTITUTION (q1.value(),q2.value()));
  151. }
  152. template<class Unit,class Y>
  153. inline
  154. quantity<Unit,Y>
  155. fabs BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q)
  156. {
  157. using std::fabs;
  158. typedef quantity<Unit,Y> quantity_type;
  159. return quantity_type::from_value(fabs BOOST_PREVENT_MACRO_SUBSTITUTION (q.value()));
  160. }
  161. template<class Unit,class Y>
  162. inline
  163. quantity<Unit,Y>
  164. floor BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q)
  165. {
  166. using std::floor;
  167. typedef quantity<Unit,Y> quantity_type;
  168. return quantity_type::from_value(floor BOOST_PREVENT_MACRO_SUBSTITUTION (q.value()));
  169. }
  170. template<class Unit,class Y>
  171. inline
  172. quantity<Unit,Y>
  173. fdim BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q1,
  174. const quantity<Unit,Y>& q2)
  175. {
  176. using namespace detail;
  177. typedef quantity<Unit,Y> quantity_type;
  178. return quantity_type::from_value(fdim BOOST_PREVENT_MACRO_SUBSTITUTION (q1.value(),q2.value()));
  179. }
  180. #if 0
  181. template<class Unit1,class Unit2,class Unit3,class Y>
  182. inline
  183. typename add_typeof_helper<
  184. typename multiply_typeof_helper<quantity<Unit1,Y>,
  185. quantity<Unit2,Y> >::type,
  186. quantity<Unit3,Y> >::type
  187. fma BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit1,Y>& q1,
  188. const quantity<Unit2,Y>& q2,
  189. const quantity<Unit3,Y>& q3)
  190. {
  191. using namespace detail;
  192. typedef quantity<Unit1,Y> type1;
  193. typedef quantity<Unit2,Y> type2;
  194. typedef quantity<Unit3,Y> type3;
  195. typedef typename multiply_typeof_helper<type1,type2>::type prod_type;
  196. typedef typename add_typeof_helper<prod_type,type3>::type quantity_type;
  197. return quantity_type::from_value(fma BOOST_PREVENT_MACRO_SUBSTITUTION (q1.value(),q2.value(),q3.value()));
  198. }
  199. #endif
  200. template<class Unit,class Y>
  201. inline
  202. quantity<Unit,Y>
  203. fmax BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q1,
  204. const quantity<Unit,Y>& q2)
  205. {
  206. using namespace detail;
  207. typedef quantity<Unit,Y> quantity_type;
  208. return quantity_type::from_value(fmax BOOST_PREVENT_MACRO_SUBSTITUTION (q1.value(),q2.value()));
  209. }
  210. template<class Unit,class Y>
  211. inline
  212. quantity<Unit,Y>
  213. fmin BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q1,
  214. const quantity<Unit,Y>& q2)
  215. {
  216. using namespace detail;
  217. typedef quantity<Unit,Y> quantity_type;
  218. return quantity_type::from_value(fmin BOOST_PREVENT_MACRO_SUBSTITUTION (q1.value(),q2.value()));
  219. }
  220. template<class Unit,class Y>
  221. inline
  222. int
  223. fpclassify BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q)
  224. {
  225. using boost::math::fpclassify;
  226. return fpclassify BOOST_PREVENT_MACRO_SUBSTITUTION (q.value());
  227. }
  228. template<class Unit,class Y>
  229. inline
  230. typename root_typeof_helper<
  231. typename add_typeof_helper<
  232. typename power_typeof_helper<quantity<Unit,Y>,
  233. static_rational<2> >::type,
  234. typename power_typeof_helper<quantity<Unit,Y>,
  235. static_rational<2> >::type>::type,
  236. static_rational<2> >::type
  237. hypot BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q1,const quantity<Unit,Y>& q2)
  238. {
  239. using boost::math::hypot;
  240. typedef quantity<Unit,Y> type1;
  241. typedef typename power_typeof_helper<type1,static_rational<2> >::type pow_type;
  242. typedef typename add_typeof_helper<pow_type,pow_type>::type add_type;
  243. typedef typename root_typeof_helper<add_type,static_rational<2> >::type quantity_type;
  244. return quantity_type::from_value(hypot BOOST_PREVENT_MACRO_SUBSTITUTION (q1.value(),q2.value()));
  245. }
  246. // does ISO C++ support long long? g++ claims not
  247. //template<class Unit,class Y>
  248. //inline
  249. //quantity<Unit,long long>
  250. //llrint BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q)
  251. //{
  252. // using namespace detail;
  253. //
  254. // typedef quantity<Unit,long long> quantity_type;
  255. //
  256. // return quantity_type::from_value(llrint BOOST_PREVENT_MACRO_SUBSTITUTION (q.value()));
  257. //}
  258. // does ISO C++ support long long? g++ claims not
  259. //template<class Unit,class Y>
  260. //inline
  261. //quantity<Unit,long long>
  262. //llround BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q)
  263. //{
  264. // using namespace detail;
  265. //
  266. // typedef quantity<Unit,long long> quantity_type;
  267. //
  268. // return quantity_type::from_value(llround BOOST_PREVENT_MACRO_SUBSTITUTION (q.value()));
  269. //}
  270. #if 0
  271. template<class Unit,class Y>
  272. inline
  273. quantity<Unit,Y>
  274. nearbyint BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q)
  275. {
  276. using namespace detail;
  277. typedef quantity<Unit,Y> quantity_type;
  278. return quantity_type::from_value(nearbyint BOOST_PREVENT_MACRO_SUBSTITUTION (q.value()));
  279. }
  280. #endif
  281. template<class Unit,class Y>
  282. inline
  283. quantity<Unit,Y> nextafter BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q1,
  284. const quantity<Unit,Y>& q2)
  285. {
  286. using boost::math::nextafter;
  287. typedef quantity<Unit,Y> quantity_type;
  288. return quantity_type::from_value(nextafter BOOST_PREVENT_MACRO_SUBSTITUTION (q1.value(),q2.value()));
  289. }
  290. template<class Unit,class Y>
  291. inline
  292. quantity<Unit,Y> nexttoward BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q1,
  293. const quantity<Unit,Y>& q2)
  294. {
  295. // the only difference between nextafter and nexttowards is
  296. // in the argument types. Since we are requiring identical
  297. // argument types, there is no difference.
  298. using boost::math::nextafter;
  299. typedef quantity<Unit,Y> quantity_type;
  300. return quantity_type::from_value(nextafter BOOST_PREVENT_MACRO_SUBSTITUTION (q1.value(),q2.value()));
  301. }
  302. #if 0
  303. template<class Unit,class Y>
  304. inline
  305. quantity<Unit,Y>
  306. rint BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q)
  307. {
  308. using namespace detail;
  309. typedef quantity<Unit,Y> quantity_type;
  310. return quantity_type::from_value(rint BOOST_PREVENT_MACRO_SUBSTITUTION (q.value()));
  311. }
  312. #endif
  313. template<class Unit,class Y>
  314. inline
  315. quantity<Unit,Y>
  316. round BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q)
  317. {
  318. using boost::math::round;
  319. typedef quantity<Unit,Y> quantity_type;
  320. return quantity_type::from_value(round BOOST_PREVENT_MACRO_SUBSTITUTION (q.value()));
  321. }
  322. template<class Unit,class Y>
  323. inline
  324. int
  325. signbit BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q)
  326. {
  327. using boost::math::signbit;
  328. return signbit BOOST_PREVENT_MACRO_SUBSTITUTION (q.value());
  329. }
  330. template<class Unit,class Y>
  331. inline
  332. quantity<Unit,Y>
  333. trunc BOOST_PREVENT_MACRO_SUBSTITUTION (const quantity<Unit,Y>& q)
  334. {
  335. using namespace detail;
  336. typedef quantity<Unit,Y> quantity_type;
  337. return quantity_type::from_value(trunc BOOST_PREVENT_MACRO_SUBSTITUTION (q.value()));
  338. }
  339. template<class Unit,class Y>
  340. inline
  341. quantity<Unit, Y>
  342. fmod(const quantity<Unit,Y>& q1, const quantity<Unit,Y>& q2)
  343. {
  344. using std::fmod;
  345. typedef quantity<Unit,Y> quantity_type;
  346. return quantity_type::from_value(fmod(q1.value(), q2.value()));
  347. }
  348. template<class Unit, class Y>
  349. inline
  350. quantity<Unit, Y>
  351. modf(const quantity<Unit, Y>& q1, quantity<Unit, Y>* q2)
  352. {
  353. using std::modf;
  354. typedef quantity<Unit,Y> quantity_type;
  355. return quantity_type::from_value(modf(q1.value(), &quantity_cast<Y&>(*q2)));
  356. }
  357. template<class Unit, class Y, class Int>
  358. inline
  359. quantity<Unit, Y>
  360. frexp(const quantity<Unit, Y>& q,Int* ex)
  361. {
  362. using std::frexp;
  363. typedef quantity<Unit,Y> quantity_type;
  364. return quantity_type::from_value(frexp(q.value(),ex));
  365. }
  366. /// For non-dimensionless quantities, integral and rational powers
  367. /// and roots can be computed by @c pow<Ex> and @c root<Rt> respectively.
  368. template<class S, class Y>
  369. inline
  370. quantity<BOOST_UNITS_DIMENSIONLESS_UNIT(S), Y>
  371. pow(const quantity<BOOST_UNITS_DIMENSIONLESS_UNIT(S), Y>& q1,
  372. const quantity<BOOST_UNITS_DIMENSIONLESS_UNIT(S), Y>& q2)
  373. {
  374. using std::pow;
  375. typedef quantity<BOOST_UNITS_DIMENSIONLESS_UNIT(S),Y> quantity_type;
  376. return quantity_type::from_value(pow(q1.value(), q2.value()));
  377. }
  378. template<class S, class Y>
  379. inline
  380. quantity<BOOST_UNITS_DIMENSIONLESS_UNIT(S), Y>
  381. exp(const quantity<BOOST_UNITS_DIMENSIONLESS_UNIT(S), Y>& q)
  382. {
  383. using std::exp;
  384. typedef quantity<BOOST_UNITS_DIMENSIONLESS_UNIT(S), Y> quantity_type;
  385. return quantity_type::from_value(exp(q.value()));
  386. }
  387. template<class Unit, class Y, class Int>
  388. inline
  389. quantity<Unit, Y>
  390. ldexp(const quantity<Unit, Y>& q,const Int& ex)
  391. {
  392. using std::ldexp;
  393. typedef quantity<Unit,Y> quantity_type;
  394. return quantity_type::from_value(ldexp(q.value(), ex));
  395. }
  396. template<class S, class Y>
  397. inline
  398. quantity<BOOST_UNITS_DIMENSIONLESS_UNIT(S), Y>
  399. log(const quantity<BOOST_UNITS_DIMENSIONLESS_UNIT(S), Y>& q)
  400. {
  401. using std::log;
  402. typedef quantity<BOOST_UNITS_DIMENSIONLESS_UNIT(S), Y> quantity_type;
  403. return quantity_type::from_value(log(q.value()));
  404. }
  405. template<class S, class Y>
  406. inline
  407. quantity<BOOST_UNITS_DIMENSIONLESS_UNIT(S), Y>
  408. log10(const quantity<BOOST_UNITS_DIMENSIONLESS_UNIT(S), Y>& q)
  409. {
  410. using std::log10;
  411. typedef quantity<BOOST_UNITS_DIMENSIONLESS_UNIT(S), Y> quantity_type;
  412. return quantity_type::from_value(log10(q.value()));
  413. }
  414. template<class Unit,class Y>
  415. inline
  416. typename root_typeof_helper<
  417. quantity<Unit,Y>,
  418. static_rational<2>
  419. >::type
  420. sqrt(const quantity<Unit,Y>& q)
  421. {
  422. using std::sqrt;
  423. typedef typename root_typeof_helper<
  424. quantity<Unit,Y>,
  425. static_rational<2>
  426. >::type quantity_type;
  427. return quantity_type::from_value(sqrt(q.value()));
  428. }
  429. } // namespace units
  430. } // namespace boost
  431. namespace boost {
  432. namespace units {
  433. // trig functions with si argument/return types
  434. /// cos of theta in radians
  435. template<class Y>
  436. typename dimensionless_quantity<si::system,Y>::type
  437. cos(const quantity<si::plane_angle,Y>& theta)
  438. {
  439. using std::cos;
  440. return cos(theta.value());
  441. }
  442. /// sin of theta in radians
  443. template<class Y>
  444. typename dimensionless_quantity<si::system,Y>::type
  445. sin(const quantity<si::plane_angle,Y>& theta)
  446. {
  447. using std::sin;
  448. return sin(theta.value());
  449. }
  450. /// tan of theta in radians
  451. template<class Y>
  452. typename dimensionless_quantity<si::system,Y>::type
  453. tan(const quantity<si::plane_angle,Y>& theta)
  454. {
  455. using std::tan;
  456. return tan(theta.value());
  457. }
  458. /// cos of theta in other angular units
  459. template<class System,class Y>
  460. typename dimensionless_quantity<System,Y>::type
  461. cos(const quantity<unit<plane_angle_dimension,System>,Y>& theta)
  462. {
  463. return cos(quantity<si::plane_angle,Y>(theta));
  464. }
  465. /// sin of theta in other angular units
  466. template<class System,class Y>
  467. typename dimensionless_quantity<System,Y>::type
  468. sin(const quantity<unit<plane_angle_dimension,System>,Y>& theta)
  469. {
  470. return sin(quantity<si::plane_angle,Y>(theta));
  471. }
  472. /// tan of theta in other angular units
  473. template<class System,class Y>
  474. typename dimensionless_quantity<System,Y>::type
  475. tan(const quantity<unit<plane_angle_dimension,System>,Y>& theta)
  476. {
  477. return tan(quantity<si::plane_angle,Y>(theta));
  478. }
  479. /// acos of dimensionless quantity returning angle in same system
  480. template<class Y,class System>
  481. quantity<unit<plane_angle_dimension, homogeneous_system<System> >,Y>
  482. acos(const quantity<unit<dimensionless_type, homogeneous_system<System> >,Y>& val)
  483. {
  484. using std::acos;
  485. return quantity<unit<plane_angle_dimension, homogeneous_system<System> >,Y>(acos(val.value())*si::radians);
  486. }
  487. /// acos of dimensionless quantity returning angle in radians
  488. template<class Y>
  489. quantity<angle::radian_base_unit::unit_type,Y>
  490. acos(const quantity<unit<dimensionless_type, heterogeneous_dimensionless_system>,Y>& val)
  491. {
  492. using std::acos;
  493. return quantity<angle::radian_base_unit::unit_type,Y>::from_value(acos(val.value()));
  494. }
  495. /// asin of dimensionless quantity returning angle in same system
  496. template<class Y,class System>
  497. quantity<unit<plane_angle_dimension, homogeneous_system<System> >,Y>
  498. asin(const quantity<unit<dimensionless_type, homogeneous_system<System> >,Y>& val)
  499. {
  500. using std::asin;
  501. return quantity<unit<plane_angle_dimension, homogeneous_system<System> >,Y>(asin(val.value())*si::radians);
  502. }
  503. /// asin of dimensionless quantity returning angle in radians
  504. template<class Y>
  505. quantity<angle::radian_base_unit::unit_type,Y>
  506. asin(const quantity<unit<dimensionless_type, heterogeneous_dimensionless_system>,Y>& val)
  507. {
  508. using std::asin;
  509. return quantity<angle::radian_base_unit::unit_type,Y>::from_value(asin(val.value()));
  510. }
  511. /// atan of dimensionless quantity returning angle in same system
  512. template<class Y,class System>
  513. quantity<unit<plane_angle_dimension, homogeneous_system<System> >,Y>
  514. atan(const quantity<unit<dimensionless_type, homogeneous_system<System> >, Y>& val)
  515. {
  516. using std::atan;
  517. return quantity<unit<plane_angle_dimension, homogeneous_system<System> >,Y>(atan(val.value())*si::radians);
  518. }
  519. /// atan of dimensionless quantity returning angle in radians
  520. template<class Y>
  521. quantity<angle::radian_base_unit::unit_type,Y>
  522. atan(const quantity<unit<dimensionless_type, heterogeneous_dimensionless_system>, Y>& val)
  523. {
  524. using std::atan;
  525. return quantity<angle::radian_base_unit::unit_type,Y>::from_value(atan(val.value()));
  526. }
  527. /// atan2 of @c value_type returning angle in radians
  528. template<class Y, class Dimension, class System>
  529. quantity<unit<plane_angle_dimension, homogeneous_system<System> >, Y>
  530. atan2(const quantity<unit<Dimension, homogeneous_system<System> >, Y>& y,
  531. const quantity<unit<Dimension, homogeneous_system<System> >, Y>& x)
  532. {
  533. using std::atan2;
  534. return quantity<unit<plane_angle_dimension, homogeneous_system<System> >, Y>(atan2(y.value(),x.value())*si::radians);
  535. }
  536. /// atan2 of @c value_type returning angle in radians
  537. template<class Y, class Dimension, class System>
  538. quantity<angle::radian_base_unit::unit_type,Y>
  539. atan2(const quantity<unit<Dimension, heterogeneous_system<System> >, Y>& y,
  540. const quantity<unit<Dimension, heterogeneous_system<System> >, Y>& x)
  541. {
  542. using std::atan2;
  543. return quantity<angle::radian_base_unit::unit_type,Y>::from_value(atan2(y.value(),x.value()));
  544. }
  545. } // namespace units
  546. } // namespace boost
  547. #endif // BOOST_UNITS_CMATH_HPP