test_extrema.cpp 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396
  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. // Undefining BOOST_COMPUTE_USE_OFFLINE_CACHE macro as we want to modify cached
  11. // parameters for copy algorithm without any undesirable consequences (like
  12. // saving modified values of those parameters).
  13. #ifdef BOOST_COMPUTE_USE_OFFLINE_CACHE
  14. #undef BOOST_COMPUTE_USE_OFFLINE_CACHE
  15. #endif
  16. #define BOOST_TEST_MODULE TestExtrema
  17. #include <boost/test/unit_test.hpp>
  18. #include <boost/compute/system.hpp>
  19. #include <boost/compute/command_queue.hpp>
  20. #include <boost/compute/algorithm/copy.hpp>
  21. #include <boost/compute/algorithm/iota.hpp>
  22. #include <boost/compute/algorithm/fill.hpp>
  23. #include <boost/compute/algorithm/max_element.hpp>
  24. #include <boost/compute/algorithm/min_element.hpp>
  25. #include <boost/compute/algorithm/minmax_element.hpp>
  26. #include <boost/compute/container/vector.hpp>
  27. #include <boost/compute/iterator/transform_iterator.hpp>
  28. #include <boost/compute/detail/parameter_cache.hpp>
  29. #include "quirks.hpp"
  30. #include "context_setup.hpp"
  31. namespace bc = boost::compute;
  32. namespace compute = boost::compute;
  33. BOOST_AUTO_TEST_CASE(empyt_min)
  34. {
  35. using boost::compute::int_;
  36. boost::compute::vector<int_> vector(size_t(16), int_(0), queue);
  37. boost::compute::vector<int_>::iterator min_iter =
  38. boost::compute::min_element(vector.begin(), vector.begin(), queue);
  39. BOOST_CHECK(min_iter == vector.begin());
  40. min_iter =
  41. boost::compute::min_element(vector.begin(), vector.begin() + 1, queue);
  42. BOOST_CHECK(min_iter == vector.begin());
  43. }
  44. BOOST_AUTO_TEST_CASE(int_min_max)
  45. {
  46. using boost::compute::int_;
  47. using boost::compute::uint_;
  48. boost::compute::vector<int_> vector(size_t(4096), int_(0), queue);
  49. boost::compute::iota(vector.begin(), (vector.begin() + 512), 1, queue);
  50. boost::compute::fill((vector.end() - 512), vector.end(), 513, queue);
  51. boost::compute::vector<int_>::iterator min_iter =
  52. boost::compute::min_element(vector.begin(), vector.end(), queue);
  53. BOOST_CHECK(min_iter == vector.begin() + 512);
  54. BOOST_CHECK_EQUAL((vector.begin() + 512).read(queue), 0);
  55. BOOST_CHECK_EQUAL(min_iter.read(queue), 0);
  56. boost::compute::vector<int_>::iterator max_iter =
  57. boost::compute::max_element(vector.begin(), vector.end(), queue);
  58. BOOST_CHECK(max_iter == vector.end() - 512);
  59. BOOST_CHECK_EQUAL((vector.end() - 512).read(queue), 513);
  60. BOOST_CHECK_EQUAL(max_iter.read(queue), 513);
  61. // compare function
  62. boost::compute::less<int_> lessint;
  63. // test minmax_element
  64. std::pair<
  65. boost::compute::vector<int_>::iterator,
  66. boost::compute::vector<int_>::iterator
  67. > minmax_iter =
  68. boost::compute::minmax_element(vector.begin(), vector.end(), queue);
  69. BOOST_CHECK_EQUAL((minmax_iter.first).read(queue), 0);
  70. BOOST_CHECK_EQUAL((minmax_iter.second).read(queue), 513);
  71. minmax_iter =
  72. boost::compute::minmax_element(vector.begin(), vector.end(), lessint, queue);
  73. BOOST_CHECK_EQUAL((minmax_iter.first).read(queue), 0);
  74. BOOST_CHECK_EQUAL((minmax_iter.second).read(queue), 513);
  75. // find_extrama_on_cpu
  76. // make sure find_extrama_on_cpu is used, no serial_find_extrema
  77. std::string cache_key =
  78. "__boost_find_extrema_cpu_4";
  79. boost::shared_ptr<bc::detail::parameter_cache> parameters =
  80. bc::detail::parameter_cache::get_global_cache(device);
  81. // save
  82. uint_ map_copy_threshold =
  83. parameters->get(cache_key, "serial_find_extrema_threshold", 0);
  84. // force find_extrama_on_cpu
  85. parameters->set(cache_key, "serial_find_extrema_threshold", 16);
  86. min_iter = boost::compute::detail::find_extrema_on_cpu(
  87. vector.begin(), vector.end(), lessint, true /* find minimum */, queue
  88. );
  89. BOOST_CHECK(min_iter == vector.begin() + 512);
  90. BOOST_CHECK_EQUAL((vector.begin() + 512).read(queue), 0);
  91. BOOST_CHECK_EQUAL(min_iter.read(queue), 0);
  92. max_iter = boost::compute::detail::find_extrema_on_cpu(
  93. vector.begin(), vector.end(), lessint, false /* find minimum */, queue
  94. );
  95. BOOST_CHECK(max_iter == vector.end() - 512);
  96. BOOST_CHECK_EQUAL((vector.end() - 512).read(queue), 513);
  97. BOOST_CHECK_EQUAL(max_iter.read(queue), 513);
  98. // restore
  99. parameters->set(cache_key, "serial_find_extrema_threshold", map_copy_threshold);
  100. if(is_apple_cpu_device(device)) {
  101. std::cerr
  102. << "skipping all further tests due to Apple platform"
  103. << " behavior when local memory is used on a CPU device"
  104. << std::endl;
  105. return;
  106. }
  107. // find_extrama_with_reduce
  108. min_iter = boost::compute::detail::find_extrema_with_reduce(
  109. vector.begin(), vector.end(), lessint, true /* find minimum */, queue
  110. );
  111. BOOST_CHECK(min_iter == vector.begin() + 512);
  112. BOOST_CHECK_EQUAL((vector.begin() + 512).read(queue), 0);
  113. BOOST_CHECK_EQUAL(min_iter.read(queue), 0);
  114. max_iter = boost::compute::detail::find_extrema_with_reduce(
  115. vector.begin(), vector.end(), lessint, false /* find minimum */, queue
  116. );
  117. BOOST_CHECK(max_iter == vector.end() - 512);
  118. BOOST_CHECK_EQUAL((vector.end() - 512).read(queue), 513);
  119. BOOST_CHECK_EQUAL(max_iter.read(queue), 513);
  120. // find_extram_with_atomics
  121. min_iter = boost::compute::detail::find_extrema_with_atomics(
  122. vector.begin(), vector.end(), lessint, true /* find minimum */, queue
  123. );
  124. BOOST_CHECK(min_iter == vector.begin() + 512);
  125. BOOST_CHECK_EQUAL((vector.begin() + 512).read(queue), 0);
  126. BOOST_CHECK_EQUAL(min_iter.read(queue), 0);
  127. max_iter = boost::compute::detail::find_extrema_with_atomics(
  128. vector.begin(), vector.end(), lessint, false /* find minimum */, queue
  129. );
  130. BOOST_CHECK(max_iter == vector.end() - 512);
  131. BOOST_CHECK_EQUAL((vector.end() - 512).read(queue), 513);
  132. BOOST_CHECK_EQUAL(max_iter.read(queue), 513);
  133. }
  134. BOOST_AUTO_TEST_CASE(int2_min_max_custom_comparision_function)
  135. {
  136. using boost::compute::int2_;
  137. using boost::compute::uint_;
  138. boost::compute::vector<int2_> vector(context);
  139. vector.push_back(int2_(1, 10), queue);
  140. vector.push_back(int2_(2, -100), queue);
  141. vector.push_back(int2_(3, 30), queue);
  142. vector.push_back(int2_(4, 20), queue);
  143. vector.push_back(int2_(5, 5), queue);
  144. vector.push_back(int2_(6, -80), queue);
  145. vector.push_back(int2_(7, 21), queue);
  146. vector.push_back(int2_(8, -5), queue);
  147. BOOST_COMPUTE_FUNCTION(bool, compare_second, (const int2_ a, const int2_ b),
  148. {
  149. return a.y < b.y;
  150. });
  151. boost::compute::vector<int2_>::iterator min_iter =
  152. boost::compute::min_element(
  153. vector.begin(), vector.end(), compare_second, queue
  154. );
  155. BOOST_CHECK(min_iter == vector.begin() + 1);
  156. BOOST_CHECK_EQUAL(*min_iter, int2_(2, -100));
  157. boost::compute::vector<int2_>::iterator max_iter =
  158. boost::compute::max_element(
  159. vector.begin(), vector.end(), compare_second, queue
  160. );
  161. BOOST_CHECK(max_iter == vector.begin() + 2);
  162. BOOST_CHECK_EQUAL(*max_iter, int2_(3, 30));
  163. // find_extrama_on_cpu
  164. // make sure find_extrama_on_cpu is used, no serial_find_extrema
  165. std::string cache_key =
  166. "__boost_find_extrema_cpu_8";
  167. boost::shared_ptr<bc::detail::parameter_cache> parameters =
  168. bc::detail::parameter_cache::get_global_cache(device);
  169. // save
  170. uint_ map_copy_threshold =
  171. parameters->get(cache_key, "serial_find_extrema_threshold", 0);
  172. // force find_extrama_on_cpu
  173. parameters->set(cache_key, "serial_find_extrema_threshold", 16);
  174. min_iter = boost::compute::detail::find_extrema_on_cpu(
  175. vector.begin(), vector.end(), compare_second, true /* find minimum */, queue
  176. );
  177. BOOST_CHECK(min_iter == vector.begin() + 1);
  178. BOOST_CHECK_EQUAL(*min_iter, int2_(2, -100));
  179. max_iter = boost::compute::detail::find_extrema_on_cpu(
  180. vector.begin(), vector.end(), compare_second, false /* find minimum */, queue
  181. );
  182. BOOST_CHECK(max_iter == vector.begin() + 2);
  183. BOOST_CHECK_EQUAL(*max_iter, int2_(3, 30));
  184. // restore
  185. parameters->set(cache_key, "serial_find_extrema_threshold", map_copy_threshold);
  186. if(is_apple_cpu_device(device)) {
  187. std::cerr
  188. << "skipping all further tests due to Apple platform"
  189. << " behavior when local memory is used on a CPU device"
  190. << std::endl;
  191. return;
  192. }
  193. // find_extrama_with_reduce
  194. min_iter = boost::compute::detail::find_extrema_with_reduce(
  195. vector.begin(), vector.end(), compare_second, true /* find minimum */, queue
  196. );
  197. BOOST_CHECK(min_iter == vector.begin() + 1);
  198. BOOST_CHECK_EQUAL(*min_iter, int2_(2, -100));
  199. max_iter = boost::compute::detail::find_extrema_with_reduce(
  200. vector.begin(), vector.end(), compare_second, false /* find minimum */, queue
  201. );
  202. BOOST_CHECK(max_iter == vector.begin() + 2);
  203. BOOST_CHECK_EQUAL(*max_iter, int2_(3, 30));
  204. // find_extram_with_atomics
  205. min_iter = boost::compute::detail::find_extrema_with_atomics(
  206. vector.begin(), vector.end(), compare_second, true /* find minimum */, queue
  207. );
  208. BOOST_CHECK(min_iter == vector.begin() + 1);
  209. BOOST_CHECK_EQUAL(*min_iter, int2_(2, -100));
  210. max_iter = boost::compute::detail::find_extrema_with_atomics(
  211. vector.begin(), vector.end(), compare_second, false /* find minimum */, queue
  212. );
  213. BOOST_CHECK(max_iter == vector.begin() + 2);
  214. BOOST_CHECK_EQUAL(*max_iter, int2_(3, 30));
  215. }
  216. BOOST_AUTO_TEST_CASE(iota_min_max)
  217. {
  218. boost::compute::vector<int> vector(5000, context);
  219. // fill with 0 -> 4999
  220. boost::compute::iota(vector.begin(), vector.end(), 0, queue);
  221. boost::compute::vector<int>::iterator min_iter =
  222. boost::compute::min_element(vector.begin(), vector.end(), queue);
  223. BOOST_CHECK(min_iter == vector.begin());
  224. BOOST_CHECK_EQUAL(*min_iter, 0);
  225. boost::compute::vector<int>::iterator max_iter =
  226. boost::compute::max_element(vector.begin(), vector.end(), queue);
  227. BOOST_CHECK(max_iter == vector.end() - 1);
  228. BOOST_CHECK_EQUAL(*max_iter, 4999);
  229. min_iter =
  230. boost::compute::min_element(
  231. vector.begin() + 1000,
  232. vector.end() - 1000,
  233. queue
  234. );
  235. BOOST_CHECK(min_iter == vector.begin() + 1000);
  236. BOOST_CHECK_EQUAL(*min_iter, 1000);
  237. max_iter =
  238. boost::compute::max_element(
  239. vector.begin() + 1000,
  240. vector.end() - 1000,
  241. queue
  242. );
  243. BOOST_CHECK(max_iter == vector.begin() + 3999);
  244. BOOST_CHECK_EQUAL(*max_iter, 3999);
  245. // fill with -2500 -> 2499
  246. boost::compute::iota(vector.begin(), vector.end(), -2500, queue);
  247. min_iter =
  248. boost::compute::min_element(vector.begin(), vector.end(), queue);
  249. BOOST_CHECK(min_iter == vector.begin());
  250. BOOST_CHECK_EQUAL(*min_iter, -2500);
  251. max_iter =
  252. boost::compute::max_element(vector.begin(), vector.end(), queue);
  253. BOOST_CHECK(max_iter == vector.end() - 1);
  254. BOOST_CHECK_EQUAL(*max_iter, 2499);
  255. }
  256. // uses max_element() and length() to find the longest 2d vector
  257. BOOST_AUTO_TEST_CASE(max_vector_length)
  258. {
  259. float data[] = { -1.5f, 3.2f,
  260. 10.0f, 0.0f,
  261. -4.2f, 2.0f,
  262. 0.0f, 0.5f,
  263. 1.9f, 1.9f };
  264. boost::compute::vector<boost::compute::float2_> vector(
  265. reinterpret_cast<boost::compute::float2_ *>(data),
  266. reinterpret_cast<boost::compute::float2_ *>(data) + 5,
  267. queue
  268. );
  269. // find length of the longest vector
  270. typedef boost::compute::transform_iterator<
  271. boost::compute::vector<boost::compute::float2_>::iterator,
  272. boost::compute::length<boost::compute::float2_>
  273. > length_transform_iter;
  274. length_transform_iter max_iter =
  275. boost::compute::max_element(
  276. boost::compute::make_transform_iterator(
  277. vector.begin(),
  278. boost::compute::length<boost::compute::float2_>()
  279. ),
  280. boost::compute::make_transform_iterator(
  281. vector.end(),
  282. boost::compute::length<boost::compute::float2_>()
  283. ),
  284. queue
  285. );
  286. BOOST_CHECK(
  287. max_iter == boost::compute::make_transform_iterator(
  288. vector.begin() + 1,
  289. boost::compute::length<boost::compute::float2_>()
  290. )
  291. );
  292. BOOST_CHECK(max_iter.base() == vector.begin() + 1);
  293. BOOST_CHECK_EQUAL(*max_iter, float(10.0));
  294. // find length of the shortest vector
  295. length_transform_iter min_iter =
  296. boost::compute::min_element(
  297. boost::compute::make_transform_iterator(
  298. vector.begin(),
  299. boost::compute::length<boost::compute::float2_>()
  300. ),
  301. boost::compute::make_transform_iterator(
  302. vector.end(),
  303. boost::compute::length<boost::compute::float2_>()
  304. ),
  305. queue
  306. );
  307. BOOST_CHECK(
  308. min_iter == boost::compute::make_transform_iterator(
  309. vector.begin() + 3,
  310. boost::compute::length<boost::compute::float2_>()
  311. )
  312. );
  313. BOOST_CHECK(min_iter.base() == vector.begin() + 3);
  314. BOOST_CHECK_EQUAL(*min_iter, float(0.5));
  315. }
  316. // uses max_element() and popcount() to find the value with the most 1 bits
  317. BOOST_AUTO_TEST_CASE(max_bits_set)
  318. {
  319. using boost::compute::uint_;
  320. uint_ data[] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 };
  321. boost::compute::vector<uint_> vector(data, data + 10, queue);
  322. boost::compute::vector<uint_>::iterator iter =
  323. boost::compute::max_element(
  324. boost::compute::make_transform_iterator(
  325. vector.begin(),
  326. boost::compute::popcount<uint_>()
  327. ),
  328. boost::compute::make_transform_iterator(
  329. vector.end(),
  330. boost::compute::popcount<uint_>()
  331. ),
  332. queue
  333. ).base();
  334. BOOST_CHECK(iter == vector.begin() + 7);
  335. BOOST_CHECK_EQUAL(uint_(*iter), uint_(7));
  336. }
  337. BOOST_AUTO_TEST_SUITE_END()