function.hpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454
  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_FUNCTION_HPP
  11. #define BOOST_COMPUTE_FUNCTION_HPP
  12. #include <map>
  13. #include <string>
  14. #include <sstream>
  15. #include <vector>
  16. #include <boost/assert.hpp>
  17. #include <boost/config.hpp>
  18. #include <boost/function_types/parameter_types.hpp>
  19. #include <boost/preprocessor/repetition.hpp>
  20. #include <boost/mpl/for_each.hpp>
  21. #include <boost/mpl/size.hpp>
  22. #include <boost/mpl/transform.hpp>
  23. #include <boost/static_assert.hpp>
  24. #include <boost/tuple/tuple.hpp>
  25. #include <boost/type_traits/add_pointer.hpp>
  26. #include <boost/type_traits/function_traits.hpp>
  27. #include <boost/compute/cl.hpp>
  28. #include <boost/compute/config.hpp>
  29. #include <boost/compute/type_traits/type_name.hpp>
  30. namespace boost {
  31. namespace compute {
  32. namespace detail {
  33. template<class ResultType, class ArgTuple>
  34. class invoked_function
  35. {
  36. public:
  37. typedef ResultType result_type;
  38. BOOST_STATIC_CONSTANT(
  39. size_t, arity = boost::tuples::length<ArgTuple>::value
  40. );
  41. invoked_function(const std::string &name,
  42. const std::string &source)
  43. : m_name(name),
  44. m_source(source)
  45. {
  46. }
  47. invoked_function(const std::string &name,
  48. const std::string &source,
  49. const std::map<std::string, std::string> &definitions)
  50. : m_name(name),
  51. m_source(source),
  52. m_definitions(definitions)
  53. {
  54. }
  55. invoked_function(const std::string &name,
  56. const std::string &source,
  57. const ArgTuple &args)
  58. : m_name(name),
  59. m_source(source),
  60. m_args(args)
  61. {
  62. }
  63. invoked_function(const std::string &name,
  64. const std::string &source,
  65. const std::map<std::string, std::string> &definitions,
  66. const ArgTuple &args)
  67. : m_name(name),
  68. m_source(source),
  69. m_definitions(definitions),
  70. m_args(args)
  71. {
  72. }
  73. std::string name() const
  74. {
  75. return m_name;
  76. }
  77. std::string source() const
  78. {
  79. return m_source;
  80. }
  81. const std::map<std::string, std::string>& definitions() const
  82. {
  83. return m_definitions;
  84. }
  85. const ArgTuple& args() const
  86. {
  87. return m_args;
  88. }
  89. private:
  90. std::string m_name;
  91. std::string m_source;
  92. std::map<std::string, std::string> m_definitions;
  93. ArgTuple m_args;
  94. };
  95. } // end detail namespace
  96. /// \class function
  97. /// \brief A function object.
  98. template<class Signature>
  99. class function
  100. {
  101. public:
  102. /// \internal_
  103. typedef typename
  104. boost::function_traits<Signature>::result_type result_type;
  105. /// \internal_
  106. BOOST_STATIC_CONSTANT(
  107. size_t, arity = boost::function_traits<Signature>::arity
  108. );
  109. /// \internal_
  110. typedef Signature signature;
  111. /// Creates a new function object with \p name.
  112. function(const std::string &name)
  113. : m_name(name)
  114. {
  115. }
  116. /// Destroys the function object.
  117. ~function()
  118. {
  119. }
  120. /// \internal_
  121. std::string name() const
  122. {
  123. return m_name;
  124. }
  125. /// \internal_
  126. void set_source(const std::string &source)
  127. {
  128. m_source = source;
  129. }
  130. /// \internal_
  131. std::string source() const
  132. {
  133. return m_source;
  134. }
  135. /// \internal_
  136. void define(std::string name, std::string value = std::string())
  137. {
  138. m_definitions[name] = value;
  139. }
  140. /// \internal_
  141. detail::invoked_function<result_type, boost::tuple<> >
  142. operator()() const
  143. {
  144. BOOST_STATIC_ASSERT_MSG(
  145. arity == 0,
  146. "Non-nullary function invoked with zero arguments"
  147. );
  148. return detail::invoked_function<result_type, boost::tuple<> >(
  149. m_name, m_source, m_definitions
  150. );
  151. }
  152. /// \internal_
  153. template<class Arg1>
  154. detail::invoked_function<result_type, boost::tuple<Arg1> >
  155. operator()(const Arg1 &arg1) const
  156. {
  157. BOOST_STATIC_ASSERT_MSG(
  158. arity == 1,
  159. "Non-unary function invoked one argument"
  160. );
  161. return detail::invoked_function<result_type, boost::tuple<Arg1> >(
  162. m_name, m_source, m_definitions, boost::make_tuple(arg1)
  163. );
  164. }
  165. /// \internal_
  166. template<class Arg1, class Arg2>
  167. detail::invoked_function<result_type, boost::tuple<Arg1, Arg2> >
  168. operator()(const Arg1 &arg1, const Arg2 &arg2) const
  169. {
  170. BOOST_STATIC_ASSERT_MSG(
  171. arity == 2,
  172. "Non-binary function invoked with two arguments"
  173. );
  174. return detail::invoked_function<result_type, boost::tuple<Arg1, Arg2> >(
  175. m_name, m_source, m_definitions, boost::make_tuple(arg1, arg2)
  176. );
  177. }
  178. /// \internal_
  179. template<class Arg1, class Arg2, class Arg3>
  180. detail::invoked_function<result_type, boost::tuple<Arg1, Arg2, Arg3> >
  181. operator()(const Arg1 &arg1, const Arg2 &arg2, const Arg3 &arg3) const
  182. {
  183. BOOST_STATIC_ASSERT_MSG(
  184. arity == 3,
  185. "Non-ternary function invoked with two arguments"
  186. );
  187. return detail::invoked_function<result_type, boost::tuple<Arg1, Arg2, Arg3> >(
  188. m_name, m_source, m_definitions, boost::make_tuple(arg1, arg2, arg3)
  189. );
  190. }
  191. private:
  192. std::string m_name;
  193. std::string m_source;
  194. std::map<std::string, std::string> m_definitions;
  195. };
  196. /// Creates a function object given its \p name and \p source.
  197. ///
  198. /// \param name The function name.
  199. /// \param source The function source code.
  200. ///
  201. /// \see BOOST_COMPUTE_FUNCTION()
  202. template<class Signature>
  203. inline function<Signature>
  204. make_function_from_source(const std::string &name, const std::string &source)
  205. {
  206. function<Signature> f(name);
  207. f.set_source(source);
  208. return f;
  209. }
  210. namespace detail {
  211. // given a string containing the arguments declaration for a function
  212. // like: "(int a, const float b)", returns a vector containing the name
  213. // of each argument (e.g. ["a", "b"]).
  214. inline std::vector<std::string> parse_argument_names(const char *arguments)
  215. {
  216. BOOST_ASSERT_MSG(
  217. arguments[0] == '(' && arguments[std::strlen(arguments)-1] == ')',
  218. "Arguments should start and end with parentheses"
  219. );
  220. std::vector<std::string> args;
  221. size_t last_space = 0;
  222. size_t skip_comma = 0;
  223. for(size_t i = 1; i < std::strlen(arguments) - 2; i++){
  224. const char c = arguments[i];
  225. if(c == ' '){
  226. last_space = i;
  227. }
  228. else if(c == ',' && !skip_comma){
  229. std::string name(
  230. arguments + last_space + 1, i - last_space - 1
  231. );
  232. args.push_back(name);
  233. }
  234. else if(c == '<'){
  235. skip_comma++;
  236. }
  237. else if(c == '>'){
  238. skip_comma--;
  239. }
  240. }
  241. std::string last_argument(
  242. arguments + last_space + 1, std::strlen(arguments) - last_space - 2
  243. );
  244. args.push_back(last_argument);
  245. return args;
  246. }
  247. struct signature_argument_inserter
  248. {
  249. signature_argument_inserter(std::stringstream &s_, const char *arguments, size_t last)
  250. : s(s_)
  251. {
  252. n = 0;
  253. m_last = last;
  254. m_argument_names = parse_argument_names(arguments);
  255. BOOST_ASSERT_MSG(
  256. m_argument_names.size() == last,
  257. "Wrong number of arguments"
  258. );
  259. }
  260. template<class T>
  261. void operator()(const T*)
  262. {
  263. s << type_name<T>() << " " << m_argument_names[n];
  264. if(n+1 < m_last){
  265. s << ", ";
  266. }
  267. n++;
  268. }
  269. size_t n;
  270. size_t m_last;
  271. std::stringstream &s;
  272. std::vector<std::string> m_argument_names;
  273. };
  274. template<class Signature>
  275. inline std::string make_function_declaration(const char *name, const char *arguments)
  276. {
  277. typedef typename
  278. boost::function_traits<Signature>::result_type result_type;
  279. typedef typename
  280. boost::function_types::parameter_types<Signature>::type parameter_types;
  281. typedef typename
  282. mpl::size<parameter_types>::type arity_type;
  283. std::stringstream s;
  284. s << "inline " << type_name<result_type>() << " " << name;
  285. s << "(";
  286. if(arity_type::value > 0){
  287. signature_argument_inserter i(s, arguments, arity_type::value);
  288. mpl::for_each<
  289. typename mpl::transform<parameter_types, boost::add_pointer<mpl::_1>
  290. >::type>(i);
  291. }
  292. s << ")";
  293. return s.str();
  294. }
  295. struct argument_list_inserter
  296. {
  297. argument_list_inserter(std::stringstream &s_, const char first, size_t last)
  298. : s(s_)
  299. {
  300. n = 0;
  301. m_last = last;
  302. m_name = first;
  303. }
  304. template<class T>
  305. void operator()(const T*)
  306. {
  307. s << type_name<T>() << " " << m_name++;
  308. if(n+1 < m_last){
  309. s << ", ";
  310. }
  311. n++;
  312. }
  313. size_t n;
  314. size_t m_last;
  315. char m_name;
  316. std::stringstream &s;
  317. };
  318. template<class Signature>
  319. inline std::string generate_argument_list(const char first = 'a')
  320. {
  321. typedef typename
  322. boost::function_types::parameter_types<Signature>::type parameter_types;
  323. typedef typename
  324. mpl::size<parameter_types>::type arity_type;
  325. std::stringstream s;
  326. s << '(';
  327. if(arity_type::value > 0){
  328. argument_list_inserter i(s, first, arity_type::value);
  329. mpl::for_each<
  330. typename mpl::transform<parameter_types, boost::add_pointer<mpl::_1>
  331. >::type>(i);
  332. }
  333. s << ')';
  334. return s.str();
  335. }
  336. // used by the BOOST_COMPUTE_FUNCTION() macro to create a function
  337. // with the given signature, name, arguments, and source.
  338. template<class Signature>
  339. inline function<Signature>
  340. make_function_impl(const char *name, const char *arguments, const char *source)
  341. {
  342. std::stringstream s;
  343. s << make_function_declaration<Signature>(name, arguments);
  344. s << source;
  345. return make_function_from_source<Signature>(name, s.str());
  346. }
  347. } // end detail namespace
  348. } // end compute namespace
  349. } // end boost namespace
  350. /// Creates a function object with \p name and \p source.
  351. ///
  352. /// \param return_type The return type for the function.
  353. /// \param name The name of the function.
  354. /// \param arguments A list of arguments for the function.
  355. /// \param source The OpenCL C source code for the function.
  356. ///
  357. /// The function declaration and signature are automatically created using
  358. /// the \p return_type, \p name, and \p arguments macro parameters.
  359. ///
  360. /// The source code for the function is interpreted as OpenCL C99 source code
  361. /// which is stringified and passed to the OpenCL compiler when the function
  362. /// is invoked.
  363. ///
  364. /// For example, to create a function which squares a number:
  365. /// \code
  366. /// BOOST_COMPUTE_FUNCTION(float, square, (float x),
  367. /// {
  368. /// return x * x;
  369. /// });
  370. /// \endcode
  371. ///
  372. /// And to create a function which sums two numbers:
  373. /// \code
  374. /// BOOST_COMPUTE_FUNCTION(int, sum_two, (int x, int y),
  375. /// {
  376. /// return x + y;
  377. /// });
  378. /// \endcode
  379. ///
  380. /// \see BOOST_COMPUTE_CLOSURE()
  381. #ifdef BOOST_COMPUTE_DOXYGEN_INVOKED
  382. #define BOOST_COMPUTE_FUNCTION(return_type, name, arguments, source)
  383. #else
  384. #define BOOST_COMPUTE_FUNCTION(return_type, name, arguments, ...) \
  385. ::boost::compute::function<return_type arguments> name = \
  386. ::boost::compute::detail::make_function_impl<return_type arguments>( \
  387. #name, #arguments, #__VA_ARGS__ \
  388. )
  389. #endif
  390. #endif // BOOST_COMPUTE_FUNCTION_HPP