//---------------------------------------------------------------------------// // Copyright (c) 2013 Kyle Lutz // // Distributed under the Boost Software License, Version 1.0 // See accompanying file LICENSE_1_0.txt or copy at // http://www.boost.org/LICENSE_1_0.txt // // See http://boostorg.github.com/compute for more information. //---------------------------------------------------------------------------// // Undefining BOOST_COMPUTE_USE_OFFLINE_CACHE macro as we want to modify cached // parameters for copy algorithm without any undesirable consequences (like // saving modified values of those parameters). #ifdef BOOST_COMPUTE_USE_OFFLINE_CACHE #undef BOOST_COMPUTE_USE_OFFLINE_CACHE #endif #define BOOST_TEST_MODULE TestExtrema #include #include #include #include #include #include #include #include #include #include #include #include #include "quirks.hpp" #include "context_setup.hpp" namespace bc = boost::compute; namespace compute = boost::compute; BOOST_AUTO_TEST_CASE(empyt_min) { using boost::compute::int_; boost::compute::vector vector(size_t(16), int_(0), queue); boost::compute::vector::iterator min_iter = boost::compute::min_element(vector.begin(), vector.begin(), queue); BOOST_CHECK(min_iter == vector.begin()); min_iter = boost::compute::min_element(vector.begin(), vector.begin() + 1, queue); BOOST_CHECK(min_iter == vector.begin()); } BOOST_AUTO_TEST_CASE(int_min_max) { using boost::compute::int_; using boost::compute::uint_; boost::compute::vector vector(size_t(4096), int_(0), queue); boost::compute::iota(vector.begin(), (vector.begin() + 512), 1, queue); boost::compute::fill((vector.end() - 512), vector.end(), 513, queue); boost::compute::vector::iterator min_iter = boost::compute::min_element(vector.begin(), vector.end(), queue); BOOST_CHECK(min_iter == vector.begin() + 512); BOOST_CHECK_EQUAL((vector.begin() + 512).read(queue), 0); BOOST_CHECK_EQUAL(min_iter.read(queue), 0); boost::compute::vector::iterator max_iter = boost::compute::max_element(vector.begin(), vector.end(), queue); BOOST_CHECK(max_iter == vector.end() - 512); BOOST_CHECK_EQUAL((vector.end() - 512).read(queue), 513); BOOST_CHECK_EQUAL(max_iter.read(queue), 513); // compare function boost::compute::less lessint; // test minmax_element std::pair< boost::compute::vector::iterator, boost::compute::vector::iterator > minmax_iter = boost::compute::minmax_element(vector.begin(), vector.end(), queue); BOOST_CHECK_EQUAL((minmax_iter.first).read(queue), 0); BOOST_CHECK_EQUAL((minmax_iter.second).read(queue), 513); minmax_iter = boost::compute::minmax_element(vector.begin(), vector.end(), lessint, queue); BOOST_CHECK_EQUAL((minmax_iter.first).read(queue), 0); BOOST_CHECK_EQUAL((minmax_iter.second).read(queue), 513); // find_extrama_on_cpu // make sure find_extrama_on_cpu is used, no serial_find_extrema std::string cache_key = "__boost_find_extrema_cpu_4"; boost::shared_ptr parameters = bc::detail::parameter_cache::get_global_cache(device); // save uint_ map_copy_threshold = parameters->get(cache_key, "serial_find_extrema_threshold", 0); // force find_extrama_on_cpu parameters->set(cache_key, "serial_find_extrema_threshold", 16); min_iter = boost::compute::detail::find_extrema_on_cpu( vector.begin(), vector.end(), lessint, true /* find minimum */, queue ); BOOST_CHECK(min_iter == vector.begin() + 512); BOOST_CHECK_EQUAL((vector.begin() + 512).read(queue), 0); BOOST_CHECK_EQUAL(min_iter.read(queue), 0); max_iter = boost::compute::detail::find_extrema_on_cpu( vector.begin(), vector.end(), lessint, false /* find minimum */, queue ); BOOST_CHECK(max_iter == vector.end() - 512); BOOST_CHECK_EQUAL((vector.end() - 512).read(queue), 513); BOOST_CHECK_EQUAL(max_iter.read(queue), 513); // restore parameters->set(cache_key, "serial_find_extrema_threshold", map_copy_threshold); if(is_apple_cpu_device(device)) { std::cerr << "skipping all further tests due to Apple platform" << " behavior when local memory is used on a CPU device" << std::endl; return; } // find_extrama_with_reduce min_iter = boost::compute::detail::find_extrema_with_reduce( vector.begin(), vector.end(), lessint, true /* find minimum */, queue ); BOOST_CHECK(min_iter == vector.begin() + 512); BOOST_CHECK_EQUAL((vector.begin() + 512).read(queue), 0); BOOST_CHECK_EQUAL(min_iter.read(queue), 0); max_iter = boost::compute::detail::find_extrema_with_reduce( vector.begin(), vector.end(), lessint, false /* find minimum */, queue ); BOOST_CHECK(max_iter == vector.end() - 512); BOOST_CHECK_EQUAL((vector.end() - 512).read(queue), 513); BOOST_CHECK_EQUAL(max_iter.read(queue), 513); // find_extram_with_atomics min_iter = boost::compute::detail::find_extrema_with_atomics( vector.begin(), vector.end(), lessint, true /* find minimum */, queue ); BOOST_CHECK(min_iter == vector.begin() + 512); BOOST_CHECK_EQUAL((vector.begin() + 512).read(queue), 0); BOOST_CHECK_EQUAL(min_iter.read(queue), 0); max_iter = boost::compute::detail::find_extrema_with_atomics( vector.begin(), vector.end(), lessint, false /* find minimum */, queue ); BOOST_CHECK(max_iter == vector.end() - 512); BOOST_CHECK_EQUAL((vector.end() - 512).read(queue), 513); BOOST_CHECK_EQUAL(max_iter.read(queue), 513); } BOOST_AUTO_TEST_CASE(int2_min_max_custom_comparision_function) { using boost::compute::int2_; using boost::compute::uint_; boost::compute::vector vector(context); vector.push_back(int2_(1, 10), queue); vector.push_back(int2_(2, -100), queue); vector.push_back(int2_(3, 30), queue); vector.push_back(int2_(4, 20), queue); vector.push_back(int2_(5, 5), queue); vector.push_back(int2_(6, -80), queue); vector.push_back(int2_(7, 21), queue); vector.push_back(int2_(8, -5), queue); BOOST_COMPUTE_FUNCTION(bool, compare_second, (const int2_ a, const int2_ b), { return a.y < b.y; }); boost::compute::vector::iterator min_iter = boost::compute::min_element( vector.begin(), vector.end(), compare_second, queue ); BOOST_CHECK(min_iter == vector.begin() + 1); BOOST_CHECK_EQUAL(*min_iter, int2_(2, -100)); boost::compute::vector::iterator max_iter = boost::compute::max_element( vector.begin(), vector.end(), compare_second, queue ); BOOST_CHECK(max_iter == vector.begin() + 2); BOOST_CHECK_EQUAL(*max_iter, int2_(3, 30)); // find_extrama_on_cpu // make sure find_extrama_on_cpu is used, no serial_find_extrema std::string cache_key = "__boost_find_extrema_cpu_8"; boost::shared_ptr parameters = bc::detail::parameter_cache::get_global_cache(device); // save uint_ map_copy_threshold = parameters->get(cache_key, "serial_find_extrema_threshold", 0); // force find_extrama_on_cpu parameters->set(cache_key, "serial_find_extrema_threshold", 16); min_iter = boost::compute::detail::find_extrema_on_cpu( vector.begin(), vector.end(), compare_second, true /* find minimum */, queue ); BOOST_CHECK(min_iter == vector.begin() + 1); BOOST_CHECK_EQUAL(*min_iter, int2_(2, -100)); max_iter = boost::compute::detail::find_extrema_on_cpu( vector.begin(), vector.end(), compare_second, false /* find minimum */, queue ); BOOST_CHECK(max_iter == vector.begin() + 2); BOOST_CHECK_EQUAL(*max_iter, int2_(3, 30)); // restore parameters->set(cache_key, "serial_find_extrema_threshold", map_copy_threshold); if(is_apple_cpu_device(device)) { std::cerr << "skipping all further tests due to Apple platform" << " behavior when local memory is used on a CPU device" << std::endl; return; } // find_extrama_with_reduce min_iter = boost::compute::detail::find_extrema_with_reduce( vector.begin(), vector.end(), compare_second, true /* find minimum */, queue ); BOOST_CHECK(min_iter == vector.begin() + 1); BOOST_CHECK_EQUAL(*min_iter, int2_(2, -100)); max_iter = boost::compute::detail::find_extrema_with_reduce( vector.begin(), vector.end(), compare_second, false /* find minimum */, queue ); BOOST_CHECK(max_iter == vector.begin() + 2); BOOST_CHECK_EQUAL(*max_iter, int2_(3, 30)); // find_extram_with_atomics min_iter = boost::compute::detail::find_extrema_with_atomics( vector.begin(), vector.end(), compare_second, true /* find minimum */, queue ); BOOST_CHECK(min_iter == vector.begin() + 1); BOOST_CHECK_EQUAL(*min_iter, int2_(2, -100)); max_iter = boost::compute::detail::find_extrema_with_atomics( vector.begin(), vector.end(), compare_second, false /* find minimum */, queue ); BOOST_CHECK(max_iter == vector.begin() + 2); BOOST_CHECK_EQUAL(*max_iter, int2_(3, 30)); } BOOST_AUTO_TEST_CASE(iota_min_max) { boost::compute::vector vector(5000, context); // fill with 0 -> 4999 boost::compute::iota(vector.begin(), vector.end(), 0, queue); boost::compute::vector::iterator min_iter = boost::compute::min_element(vector.begin(), vector.end(), queue); BOOST_CHECK(min_iter == vector.begin()); BOOST_CHECK_EQUAL(*min_iter, 0); boost::compute::vector::iterator max_iter = boost::compute::max_element(vector.begin(), vector.end(), queue); BOOST_CHECK(max_iter == vector.end() - 1); BOOST_CHECK_EQUAL(*max_iter, 4999); min_iter = boost::compute::min_element( vector.begin() + 1000, vector.end() - 1000, queue ); BOOST_CHECK(min_iter == vector.begin() + 1000); BOOST_CHECK_EQUAL(*min_iter, 1000); max_iter = boost::compute::max_element( vector.begin() + 1000, vector.end() - 1000, queue ); BOOST_CHECK(max_iter == vector.begin() + 3999); BOOST_CHECK_EQUAL(*max_iter, 3999); // fill with -2500 -> 2499 boost::compute::iota(vector.begin(), vector.end(), -2500, queue); min_iter = boost::compute::min_element(vector.begin(), vector.end(), queue); BOOST_CHECK(min_iter == vector.begin()); BOOST_CHECK_EQUAL(*min_iter, -2500); max_iter = boost::compute::max_element(vector.begin(), vector.end(), queue); BOOST_CHECK(max_iter == vector.end() - 1); BOOST_CHECK_EQUAL(*max_iter, 2499); } // uses max_element() and length() to find the longest 2d vector BOOST_AUTO_TEST_CASE(max_vector_length) { float data[] = { -1.5f, 3.2f, 10.0f, 0.0f, -4.2f, 2.0f, 0.0f, 0.5f, 1.9f, 1.9f }; boost::compute::vector vector( reinterpret_cast(data), reinterpret_cast(data) + 5, queue ); // find length of the longest vector typedef boost::compute::transform_iterator< boost::compute::vector::iterator, boost::compute::length > length_transform_iter; length_transform_iter max_iter = boost::compute::max_element( boost::compute::make_transform_iterator( vector.begin(), boost::compute::length() ), boost::compute::make_transform_iterator( vector.end(), boost::compute::length() ), queue ); BOOST_CHECK( max_iter == boost::compute::make_transform_iterator( vector.begin() + 1, boost::compute::length() ) ); BOOST_CHECK(max_iter.base() == vector.begin() + 1); BOOST_CHECK_EQUAL(*max_iter, float(10.0)); // find length of the shortest vector length_transform_iter min_iter = boost::compute::min_element( boost::compute::make_transform_iterator( vector.begin(), boost::compute::length() ), boost::compute::make_transform_iterator( vector.end(), boost::compute::length() ), queue ); BOOST_CHECK( min_iter == boost::compute::make_transform_iterator( vector.begin() + 3, boost::compute::length() ) ); BOOST_CHECK(min_iter.base() == vector.begin() + 3); BOOST_CHECK_EQUAL(*min_iter, float(0.5)); } // uses max_element() and popcount() to find the value with the most 1 bits BOOST_AUTO_TEST_CASE(max_bits_set) { using boost::compute::uint_; uint_ data[] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 }; boost::compute::vector vector(data, data + 10, queue); boost::compute::vector::iterator iter = boost::compute::max_element( boost::compute::make_transform_iterator( vector.begin(), boost::compute::popcount() ), boost::compute::make_transform_iterator( vector.end(), boost::compute::popcount() ), queue ).base(); BOOST_CHECK(iter == vector.begin() + 7); BOOST_CHECK_EQUAL(uint_(*iter), uint_(7)); } BOOST_AUTO_TEST_SUITE_END()