functional.hpp 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. //---------------------------------------------------------------------------//
  2. // Copyright (c) 2013 Kyle Lutz <kyle.r.lutz@gmail.com>
  3. //
  4. // Distributed under the Boost Software License, Version 1.0
  5. // See accompanying file LICENSE_1_0.txt or copy at
  6. // http://www.boost.org/LICENSE_1_0.txt
  7. //
  8. // See http://boostorg.github.com/compute for more information.
  9. //---------------------------------------------------------------------------//
  10. #ifndef BOOST_COMPUTE_LAMBDA_FUNCTIONAL_HPP
  11. #define BOOST_COMPUTE_LAMBDA_FUNCTIONAL_HPP
  12. #include <boost/tuple/tuple.hpp>
  13. #include <boost/lexical_cast.hpp>
  14. #include <boost/proto/core.hpp>
  15. #include <boost/preprocessor/cat.hpp>
  16. #include <boost/preprocessor/stringize.hpp>
  17. #include <boost/compute/functional/get.hpp>
  18. #include <boost/compute/lambda/result_of.hpp>
  19. #include <boost/compute/lambda/placeholder.hpp>
  20. namespace boost {
  21. namespace compute {
  22. namespace lambda {
  23. namespace mpl = boost::mpl;
  24. namespace proto = boost::proto;
  25. // wraps a unary boolean function
  26. #define BOOST_COMPUTE_LAMBDA_WRAP_BOOLEAN_UNARY_FUNCTION(name) \
  27. namespace detail { \
  28. struct BOOST_PP_CAT(name, _func) \
  29. { \
  30. template<class Expr, class Args> \
  31. struct lambda_result \
  32. { \
  33. typedef int type; \
  34. }; \
  35. \
  36. template<class Context, class Arg> \
  37. static void apply(Context &ctx, const Arg &arg) \
  38. { \
  39. ctx.stream << #name << "("; \
  40. proto::eval(arg, ctx); \
  41. ctx.stream << ")"; \
  42. } \
  43. }; \
  44. } \
  45. template<class Arg> \
  46. inline typename proto::result_of::make_expr< \
  47. proto::tag::function, BOOST_PP_CAT(detail::name, _func), const Arg& \
  48. >::type const \
  49. name(const Arg &arg) \
  50. { \
  51. return proto::make_expr<proto::tag::function>( \
  52. BOOST_PP_CAT(detail::name, _func)(), ::boost::ref(arg) \
  53. ); \
  54. }
  55. // wraps a unary function who's return type is the same as the argument type
  56. #define BOOST_COMPUTE_LAMBDA_WRAP_UNARY_FUNCTION_T(name) \
  57. namespace detail { \
  58. struct BOOST_PP_CAT(name, _func) \
  59. { \
  60. template<class Expr, class Args> \
  61. struct lambda_result \
  62. { \
  63. typedef typename proto::result_of::child_c<Expr, 1>::type Arg1; \
  64. typedef typename ::boost::compute::lambda::result_of<Arg1, Args>::type type; \
  65. }; \
  66. \
  67. template<class Context, class Arg> \
  68. static void apply(Context &ctx, const Arg &arg) \
  69. { \
  70. ctx.stream << #name << "("; \
  71. proto::eval(arg, ctx); \
  72. ctx.stream << ")"; \
  73. } \
  74. }; \
  75. } \
  76. template<class Arg> \
  77. inline typename proto::result_of::make_expr< \
  78. proto::tag::function, BOOST_PP_CAT(detail::name, _func), const Arg& \
  79. >::type const \
  80. name(const Arg &arg) \
  81. { \
  82. return proto::make_expr<proto::tag::function>( \
  83. BOOST_PP_CAT(detail::name, _func)(), ::boost::ref(arg) \
  84. ); \
  85. }
  86. // wraps a binary function
  87. #define BOOST_COMPUTE_LAMBDA_WRAP_BINARY_FUNCTION(name) \
  88. namespace detail { \
  89. struct BOOST_PP_CAT(name, _func) \
  90. { \
  91. template<class Expr, class Args> \
  92. struct lambda_result \
  93. { \
  94. typedef typename proto::result_of::child_c<Expr, 1>::type Arg1; \
  95. typedef typename ::boost::compute::lambda::result_of<Arg1, Args>::type type; \
  96. }; \
  97. \
  98. template<class Context, class Arg1, class Arg2> \
  99. static void apply(Context &ctx, const Arg1 &arg1, const Arg2 &arg2) \
  100. { \
  101. ctx.stream << #name << "("; \
  102. proto::eval(arg1, ctx); \
  103. ctx.stream << ", "; \
  104. proto::eval(arg2, ctx); \
  105. ctx.stream << ")"; \
  106. } \
  107. }; \
  108. } \
  109. template<class Arg1, class Arg2> \
  110. inline typename proto::result_of::make_expr< \
  111. proto::tag::function, BOOST_PP_CAT(detail::name, _func), const Arg1&, const Arg2& \
  112. >::type const \
  113. name(const Arg1 &arg1, const Arg2 &arg2) \
  114. { \
  115. return proto::make_expr<proto::tag::function>( \
  116. BOOST_PP_CAT(detail::name, _func)(), ::boost::ref(arg1), ::boost::ref(arg2) \
  117. ); \
  118. }
  119. // wraps a binary function who's result type is the scalar type of the first argument
  120. #define BOOST_COMPUTE_LAMBDA_WRAP_BINARY_FUNCTION_ST(name) \
  121. namespace detail { \
  122. struct BOOST_PP_CAT(name, _func) \
  123. { \
  124. template<class Expr, class Args> \
  125. struct lambda_result \
  126. { \
  127. typedef typename proto::result_of::child_c<Expr, 1>::type Arg1; \
  128. typedef typename ::boost::compute::lambda::result_of<Arg1, Args>::type result_type; \
  129. typedef typename ::boost::compute::scalar_type<result_type>::type type; \
  130. }; \
  131. \
  132. template<class Context, class Arg1, class Arg2> \
  133. static void apply(Context &ctx, const Arg1 &arg1, const Arg2 &arg2) \
  134. { \
  135. ctx.stream << #name << "("; \
  136. proto::eval(arg1, ctx); \
  137. ctx.stream << ", "; \
  138. proto::eval(arg2, ctx); \
  139. ctx.stream << ")"; \
  140. } \
  141. }; \
  142. } \
  143. template<class Arg1, class Arg2> \
  144. inline typename proto::result_of::make_expr< \
  145. proto::tag::function, BOOST_PP_CAT(detail::name, _func), const Arg1&, const Arg2& \
  146. >::type const \
  147. name(const Arg1 &arg1, const Arg2 &arg2) \
  148. { \
  149. return proto::make_expr<proto::tag::function>( \
  150. BOOST_PP_CAT(detail::name, _func)(), ::boost::ref(arg1), ::boost::ref(arg2) \
  151. ); \
  152. }
  153. // wraps a ternary function
  154. #define BOOST_COMPUTE_LAMBDA_WRAP_TERNARY_FUNCTION(name) \
  155. namespace detail { \
  156. struct BOOST_PP_CAT(name, _func) \
  157. { \
  158. template<class Expr, class Args> \
  159. struct lambda_result \
  160. { \
  161. typedef typename proto::result_of::child_c<Expr, 1>::type Arg1; \
  162. typedef typename ::boost::compute::lambda::result_of<Arg1, Args>::type type; \
  163. }; \
  164. \
  165. template<class Context, class Arg1, class Arg2, class Arg3> \
  166. static void apply(Context &ctx, const Arg1 &arg1, const Arg2 &arg2, const Arg3 &arg3) \
  167. { \
  168. ctx.stream << #name << "("; \
  169. proto::eval(arg1, ctx); \
  170. ctx.stream << ", "; \
  171. proto::eval(arg2, ctx); \
  172. ctx.stream << ", "; \
  173. proto::eval(arg3, ctx); \
  174. ctx.stream << ")"; \
  175. } \
  176. }; \
  177. } \
  178. template<class Arg1, class Arg2, class Arg3> \
  179. inline typename proto::result_of::make_expr< \
  180. proto::tag::function, BOOST_PP_CAT(detail::name, _func), const Arg1&, const Arg2&, const Arg3& \
  181. >::type const \
  182. name(const Arg1 &arg1, const Arg2 &arg2, const Arg3 &arg3) \
  183. { \
  184. return proto::make_expr<proto::tag::function>( \
  185. BOOST_PP_CAT(detail::name, _func)(), ::boost::ref(arg1), ::boost::ref(arg2), ::boost::ref(arg3) \
  186. ); \
  187. }
  188. BOOST_COMPUTE_LAMBDA_WRAP_BOOLEAN_UNARY_FUNCTION(all)
  189. BOOST_COMPUTE_LAMBDA_WRAP_BOOLEAN_UNARY_FUNCTION(any)
  190. BOOST_COMPUTE_LAMBDA_WRAP_BOOLEAN_UNARY_FUNCTION(isinf)
  191. BOOST_COMPUTE_LAMBDA_WRAP_BOOLEAN_UNARY_FUNCTION(isnan)
  192. BOOST_COMPUTE_LAMBDA_WRAP_BOOLEAN_UNARY_FUNCTION(isfinite)
  193. BOOST_COMPUTE_LAMBDA_WRAP_UNARY_FUNCTION_T(abs)
  194. BOOST_COMPUTE_LAMBDA_WRAP_UNARY_FUNCTION_T(cos)
  195. BOOST_COMPUTE_LAMBDA_WRAP_UNARY_FUNCTION_T(acos)
  196. BOOST_COMPUTE_LAMBDA_WRAP_UNARY_FUNCTION_T(sin)
  197. BOOST_COMPUTE_LAMBDA_WRAP_UNARY_FUNCTION_T(asin)
  198. BOOST_COMPUTE_LAMBDA_WRAP_UNARY_FUNCTION_T(tan)
  199. BOOST_COMPUTE_LAMBDA_WRAP_UNARY_FUNCTION_T(atan)
  200. BOOST_COMPUTE_LAMBDA_WRAP_UNARY_FUNCTION_T(sqrt)
  201. BOOST_COMPUTE_LAMBDA_WRAP_UNARY_FUNCTION_T(rsqrt)
  202. BOOST_COMPUTE_LAMBDA_WRAP_UNARY_FUNCTION_T(exp)
  203. BOOST_COMPUTE_LAMBDA_WRAP_UNARY_FUNCTION_T(exp2)
  204. BOOST_COMPUTE_LAMBDA_WRAP_UNARY_FUNCTION_T(exp10)
  205. BOOST_COMPUTE_LAMBDA_WRAP_UNARY_FUNCTION_T(log)
  206. BOOST_COMPUTE_LAMBDA_WRAP_UNARY_FUNCTION_T(log2)
  207. BOOST_COMPUTE_LAMBDA_WRAP_UNARY_FUNCTION_T(log10)
  208. BOOST_COMPUTE_LAMBDA_WRAP_UNARY_FUNCTION_T(round)
  209. BOOST_COMPUTE_LAMBDA_WRAP_UNARY_FUNCTION_T(length)
  210. BOOST_COMPUTE_LAMBDA_WRAP_BINARY_FUNCTION(cross)
  211. BOOST_COMPUTE_LAMBDA_WRAP_BINARY_FUNCTION(pow)
  212. BOOST_COMPUTE_LAMBDA_WRAP_BINARY_FUNCTION(pown)
  213. BOOST_COMPUTE_LAMBDA_WRAP_BINARY_FUNCTION(powr)
  214. BOOST_COMPUTE_LAMBDA_WRAP_BINARY_FUNCTION_ST(dot)
  215. BOOST_COMPUTE_LAMBDA_WRAP_BINARY_FUNCTION_ST(distance)
  216. BOOST_COMPUTE_LAMBDA_WRAP_TERNARY_FUNCTION(clamp)
  217. BOOST_COMPUTE_LAMBDA_WRAP_TERNARY_FUNCTION(fma)
  218. BOOST_COMPUTE_LAMBDA_WRAP_TERNARY_FUNCTION(mad)
  219. BOOST_COMPUTE_LAMBDA_WRAP_TERNARY_FUNCTION(smoothstep)
  220. } // end lambda namespace
  221. } // end compute namespace
  222. } // end boost namespace
  223. #endif // BOOST_COMPUTE_LAMBDA_FUNCTIONAL_HPP