test_functional_bind.cpp 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237
  1. //---------------------------------------------------------------------------//
  2. // Copyright (c) 2013-2014 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. #define BOOST_TEST_MODULE TestFunctionalBind
  11. #include <boost/test/unit_test.hpp>
  12. #include <iostream>
  13. #include <boost/compute/function.hpp>
  14. #include <boost/compute/algorithm/copy_n.hpp>
  15. #include <boost/compute/algorithm/count_if.hpp>
  16. #include <boost/compute/algorithm/find_if.hpp>
  17. #include <boost/compute/algorithm/transform.hpp>
  18. #include <boost/compute/container/vector.hpp>
  19. #include <boost/compute/functional/bind.hpp>
  20. #include <boost/compute/functional/common.hpp>
  21. #include <boost/compute/functional/operator.hpp>
  22. #include <boost/compute/types/struct.hpp>
  23. // simple test struct
  24. struct data_struct
  25. {
  26. int int_value;
  27. float float_value;
  28. };
  29. BOOST_COMPUTE_ADAPT_STRUCT(data_struct, data_struct, (int_value, float_value))
  30. #include "quirks.hpp"
  31. #include "check_macros.hpp"
  32. #include "context_setup.hpp"
  33. namespace compute = boost::compute;
  34. using compute::placeholders::_1;
  35. using compute::placeholders::_2;
  36. BOOST_AUTO_TEST_CASE(transform_plus_two)
  37. {
  38. int data[] = { 1, 2, 3, 4 };
  39. compute::vector<int> vector(4, context);
  40. compute::copy_n(data, 4, vector.begin(), queue);
  41. compute::transform(
  42. vector.begin(), vector.end(), vector.begin(),
  43. compute::bind(compute::plus<int>(), _1, 2),
  44. queue
  45. );
  46. CHECK_RANGE_EQUAL(int, 4, vector, (3, 4, 5, 6));
  47. }
  48. BOOST_AUTO_TEST_CASE(transform_pow_two)
  49. {
  50. float data[] = { 2, 3, 4, 5 };
  51. compute::vector<float> vector(4, context);
  52. compute::copy_n(data, 4, vector.begin(), queue);
  53. compute::transform(
  54. vector.begin(), vector.end(), vector.begin(),
  55. compute::bind(compute::pow<float>(), 2.0f, _1),
  56. queue
  57. );
  58. compute::copy(vector.begin(), vector.end(), data, queue);
  59. BOOST_CHECK_CLOSE(data[0], 4.0f, 1e-4);
  60. BOOST_CHECK_CLOSE(data[1], 8.0f, 1e-4);
  61. BOOST_CHECK_CLOSE(data[2], 16.0f, 1e-4);
  62. BOOST_CHECK_CLOSE(data[3], 32.0f, 1e-4);
  63. }
  64. BOOST_AUTO_TEST_CASE(find_if_equal)
  65. {
  66. int data[] = { 1, 2, 3, 4 };
  67. compute::vector<int> vector(4, context);
  68. compute::copy_n(data, 4, vector.begin(), queue);
  69. BOOST_CHECK(
  70. compute::find_if(
  71. vector.begin(), vector.end(),
  72. compute::bind(compute::equal_to<int>(), _1, 3),
  73. queue
  74. ) == vector.begin() + 2
  75. );
  76. }
  77. BOOST_AUTO_TEST_CASE(compare_less_than)
  78. {
  79. int data[] = { 1, 2, 3, 4 };
  80. compute::vector<int> vector(data, data + 4, queue);
  81. int count = boost::compute::count_if(
  82. vector.begin(), vector.end(),
  83. compute::bind(compute::less<int>(), _1, int(3)),
  84. queue
  85. );
  86. BOOST_CHECK_EQUAL(count, 2);
  87. count = boost::compute::count_if(
  88. vector.begin(), vector.end(),
  89. compute::bind(compute::less<int>(), int(3), _1),
  90. queue
  91. );
  92. BOOST_CHECK_EQUAL(count, 1);
  93. }
  94. BOOST_AUTO_TEST_CASE(subtract_ranges)
  95. {
  96. int data1[] = { 1, 2, 3, 4 };
  97. int data2[] = { 4, 3, 2, 1 };
  98. compute::vector<int> vector1(data1, data1 + 4, queue);
  99. compute::vector<int> vector2(data2, data2 + 4, queue);
  100. compute::vector<int> result(4, context);
  101. compute::transform(
  102. vector1.begin(),
  103. vector1.end(),
  104. vector2.begin(),
  105. result.begin(),
  106. compute::bind(compute::minus<int>(), _1, _2),
  107. queue
  108. );
  109. CHECK_RANGE_EQUAL(int, 4, result, (-3, -1, 1, 3));
  110. compute::transform(
  111. vector1.begin(),
  112. vector1.end(),
  113. vector2.begin(),
  114. result.begin(),
  115. compute::bind(compute::minus<int>(), _2, _1),
  116. queue
  117. );
  118. CHECK_RANGE_EQUAL(int, 4, result, (3, 1, -1, -3));
  119. compute::transform(
  120. vector1.begin(),
  121. vector1.end(),
  122. vector2.begin(),
  123. result.begin(),
  124. compute::bind(compute::minus<int>(), 5, _1),
  125. queue
  126. );
  127. CHECK_RANGE_EQUAL(int, 4, result, (4, 3, 2, 1));
  128. compute::transform(
  129. vector1.begin(),
  130. vector1.end(),
  131. vector2.begin(),
  132. result.begin(),
  133. compute::bind(compute::minus<int>(), 5, _2),
  134. queue
  135. );
  136. CHECK_RANGE_EQUAL(int, 4, result, (1, 2, 3, 4));
  137. }
  138. BOOST_AUTO_TEST_CASE(clamp_values)
  139. {
  140. int data[] = { 1, 2, 3, 4 };
  141. compute::vector<int> vector(data, data + 4, queue);
  142. compute::transform(
  143. vector.begin(), vector.end(), vector.begin(),
  144. compute::bind(compute::clamp<int>(), _1, 2, 3),
  145. queue
  146. );
  147. CHECK_RANGE_EQUAL(int, 4, vector, (2, 2, 3, 3));
  148. }
  149. BOOST_AUTO_TEST_CASE(bind_custom_function)
  150. {
  151. int data[] = { 1, 2, 3, 4 };
  152. compute::vector<int> vector(data, data + 4, queue);
  153. BOOST_COMPUTE_FUNCTION(int, x_if_odd_else_y, (int x, int y),
  154. {
  155. if(x & 1)
  156. return x;
  157. else
  158. return y;
  159. });
  160. compute::transform(
  161. vector.begin(), vector.end(), vector.begin(),
  162. compute::bind(x_if_odd_else_y, _1, 9),
  163. queue
  164. );
  165. CHECK_RANGE_EQUAL(int, 4, vector, (1, 9, 3, 9));
  166. compute::copy(
  167. data, data + 4, vector.begin(), queue
  168. );
  169. compute::transform(
  170. vector.begin(), vector.end(), vector.begin(),
  171. compute::bind(x_if_odd_else_y, 2, _1),
  172. queue
  173. );
  174. CHECK_RANGE_EQUAL(int, 4, vector, (1, 2, 3, 4));
  175. }
  176. BOOST_AUTO_TEST_CASE(bind_struct)
  177. {
  178. if(bug_in_struct_assignment(device)){
  179. std::cerr << "skipping bind_struct test" << std::endl;
  180. return;
  181. }
  182. BOOST_COMPUTE_FUNCTION(int, add_struct_value, (int x, data_struct s),
  183. {
  184. return s.int_value + x;
  185. });
  186. data_struct data;
  187. data.int_value = 3;
  188. data.float_value = 4.56f;
  189. int input[] = { 1, 2, 3, 4 };
  190. compute::vector<int> vec(input, input + 4, queue);
  191. compute::transform(
  192. vec.begin(), vec.end(), vec.begin(),
  193. compute::bind(add_struct_value, _1, data),
  194. queue
  195. );
  196. CHECK_RANGE_EQUAL(int, 4, vec, (4, 5, 6, 7));
  197. }
  198. BOOST_AUTO_TEST_SUITE_END()