//---------------------------------------------------------------------------// // 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. //---------------------------------------------------------------------------// #define BOOST_TEST_MODULE TestReduce #include #include #include #include #include #include #include #include #include #include #include "check_macros.hpp" #include "context_setup.hpp" namespace compute = boost::compute; BOOST_AUTO_TEST_CASE(reduce_int) { int data[] = { 1, 5, 9, 13, 17 }; compute::vector vector(data, data + 5, queue); int sum; compute::reduce(vector.begin(), vector.end(), &sum, compute::plus(), queue); BOOST_CHECK_EQUAL(sum, 45); int product; compute::reduce(vector.begin(), vector.end(), &product, compute::multiplies(), queue); BOOST_CHECK_EQUAL(product, 9945); } BOOST_AUTO_TEST_CASE(reduce_empty_vector) { compute::vector vector(context); short sum = 0; compute::reduce(vector.begin(), vector.end(), &sum, queue); BOOST_CHECK_EQUAL(sum, short(0)); } BOOST_AUTO_TEST_CASE(reduce_doctest) { int data[] = { 1, 2, 3, 4 }; boost::compute::vector vec(data, data + 4, queue); //! [sum_int] int sum = 0; boost::compute::reduce(vec.begin(), vec.end(), &sum, queue); //! [sum_int] BOOST_CHECK_EQUAL(sum, 10); } BOOST_AUTO_TEST_CASE(reduce_twos) { using compute::uint_; compute::vector vector(8, context); compute::fill(vector.begin(), vector.end(), uint_(2), queue); uint_ sum; compute::reduce(vector.begin(), vector.end(), &sum, compute::plus(), queue); BOOST_CHECK_EQUAL(sum, uint_(16)); uint_ product; compute::reduce(vector.begin(), vector.end(), &product, compute::multiplies(), queue); BOOST_CHECK_EQUAL(product, uint_(256)); } BOOST_AUTO_TEST_CASE(reduce_on_device) { int data[] = { 1, 2, 3, 4, 5, 6, 7, 8 }; compute::vector input(data, data + 8, queue); compute::vector result(2, context); compute::reduce(input.begin(), input.begin() + 4, result.begin(), queue); compute::reduce(input.begin() + 4, input.end(), result.end() - 1, queue); CHECK_RANGE_EQUAL(int, 2, result, (10, 26)); } BOOST_AUTO_TEST_CASE(reduce_int_min_max) { int data[] = { 11, 5, 92, 13, 42 }; compute::vector vector(data, data + 5, queue); int min_value; compute::reduce( vector.begin(), vector.end(), &min_value, compute::min(), queue ); BOOST_CHECK_EQUAL(min_value, 5); int max_value; compute::reduce( vector.begin(), vector.end(), &max_value, compute::max(), queue ); BOOST_CHECK_EQUAL(max_value, 92); } BOOST_AUTO_TEST_CASE(reduce_int2) { std::vector data; for(int i = 0; i < 6; i++){ compute::int2_ value; value[0] = i + 1; value[1] = 2 * i + 1; data.push_back(value); } compute::vector vector(data.begin(), data.end(), queue); compute::int2_ sum; compute::reduce( vector.begin(), vector.end(), &sum, queue ); BOOST_CHECK_EQUAL(sum, compute::int2_(21, 36)); } BOOST_AUTO_TEST_CASE(reduce_pinned_vector) { int data[] = { 2, 5, 8, 11, 15 }; std::vector vector(data, data + 5); compute::buffer buffer(context, vector.size() * sizeof(int), compute::buffer::read_only | compute::buffer::use_host_ptr, &vector[0]); int sum; compute::reduce( compute::make_buffer_iterator(buffer, 0), compute::make_buffer_iterator(buffer, 5), &sum, compute::plus() ); BOOST_CHECK_EQUAL(sum, 41); } BOOST_AUTO_TEST_CASE(reduce_constant_iterator) { int result; compute::reduce( compute::make_constant_iterator(1, 0), compute::make_constant_iterator(1, 5), &result, queue ); BOOST_CHECK_EQUAL(result, 5); compute::reduce( compute::make_constant_iterator(3, 0), compute::make_constant_iterator(3, 5), &result, queue ); BOOST_CHECK_EQUAL(result, 15); compute::reduce( compute::make_constant_iterator(2, 0), compute::make_constant_iterator(2, 5), &result, compute::multiplies(), queue ); BOOST_CHECK_EQUAL(result, 32); } BOOST_AUTO_TEST_CASE(reduce_counting_iterator) { int result; compute::reduce( compute::make_counting_iterator(1), compute::make_counting_iterator(10), &result, queue ); BOOST_CHECK_EQUAL(result, 45); compute::reduce( compute::make_counting_iterator(1), compute::make_counting_iterator(5), &result, compute::multiplies(), queue ); BOOST_CHECK_EQUAL(result, 24); } BOOST_AUTO_TEST_CASE(reduce_transform_iterator) { using ::boost::compute::_1; int data[] = { 1, 3, 5, 7, 9 }; compute::vector vector(data, data + 5, queue); int sum; compute::reduce( compute::make_transform_iterator(vector.begin(), _1 + 1), compute::make_transform_iterator(vector.end(), _1 + 1), &sum, queue ); BOOST_CHECK_EQUAL(sum, 30); compute::reduce( compute::make_transform_iterator(vector.begin(), _1 > 4), compute::make_transform_iterator(vector.end(), _1 > 4), &sum, compute::plus(), queue ); BOOST_CHECK_EQUAL(sum, 3); compute::reduce( compute::make_transform_iterator(vector.begin(), _1 * _1), compute::make_transform_iterator(vector.end(), _1 * _1), &sum, queue ); BOOST_CHECK_EQUAL(sum, 165); } BOOST_AUTO_TEST_CASE(reduce_complex) { std::vector > data; data.push_back(std::complex(1, 2)); data.push_back(std::complex(2, 4)); data.push_back(std::complex(3, 6)); data.push_back(std::complex(4, 8)); compute::vector > vector(data.size(), context); compute::copy(data.begin(), data.end(), vector.begin(), queue); std::complex result; compute::reduce( vector.begin(), vector.end(), &result, queue ); BOOST_CHECK(result == std::complex(10, 20)); compute::reduce( vector.begin(), vector.end(), &result, compute::plus >(), queue ); BOOST_CHECK(result == std::complex(10, 20)); compute::reduce( vector.begin(), vector.end(), &result, compute::multiplies >(), queue ); BOOST_CHECK(result == std::complex(-168, -576)); } BOOST_AUTO_TEST_CASE(reduce_uchar_to_float) { compute::vector data(context); data.push_back(250, queue); data.push_back(250, queue); float sum = 0; compute::reduce(data.begin(), data.end(), &sum, compute::plus(), queue); BOOST_CHECK_EQUAL(sum, 500); } // Test case for https://github.com/boostorg/compute/issues/746 BOOST_AUTO_TEST_CASE(buffer_reference_count_test) { using compute::uint_; compute::vector vector(8, context); const compute::buffer& b = vector.get_buffer(); uint_ rc1 = b.get_info(); { compute::fill(vector.begin(), vector.end(), uint_(2), queue); uint_ product; compute::reduce(vector.begin(), vector.end(), &product, compute::multiplies(), queue); BOOST_CHECK_EQUAL(product, uint_(256)); } uint_ rc2 = b.get_info(); BOOST_CHECK_EQUAL(rc1, rc2); } BOOST_AUTO_TEST_SUITE_END()