//---------------------------------------------------------------------------// // 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 TestFunction #include #include #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(add_three) { BOOST_COMPUTE_FUNCTION(int, add_three, (int x), { return x + 3; }); int data[] = { 1, 2, 3, 4 }; compute::vector vector(data, data + 4, queue); compute::transform( vector.begin(), vector.end(), vector.begin(), add_three, queue ); CHECK_RANGE_EQUAL(int, 4, vector, (4, 5, 6, 7)); } BOOST_AUTO_TEST_CASE(sum_odd_values) { BOOST_COMPUTE_FUNCTION(int, add_odd_value, (int sum, int value), { if(value & 1){ return sum + value; } else { return sum + 0; } }); int data[] = { 1, 2, 3, 4, 5, 6, 7, 8 }; compute::vector vector(data, data + 8, queue); int result = compute::accumulate( vector.begin(), vector.end(), 0, add_odd_value, queue ); BOOST_CHECK_EQUAL(result, 16); } BOOST_AUTO_TEST_CASE(sort_pairs) { if(device.vendor() == "NVIDIA" && device.platform().name() == "Apple"){ // FIXME: this test currently segfaults on NVIDIA GPUs on Apple std::cerr << "skipping sort_pairs test on NVIDIA GPU on Apple platform" << std::endl; return; } std::vector > data; data.push_back(std::make_pair(1, 2.3f)); data.push_back(std::make_pair(0, 4.2f)); data.push_back(std::make_pair(2, 1.0f)); compute::vector > vector(data.begin(), data.end(), queue); // sort by first component BOOST_COMPUTE_FUNCTION(bool, compare_first, (std::pair a, std::pair b), { return a.first < b.first; }); compute::sort(vector.begin(), vector.end(), compare_first, queue); compute::copy(vector.begin(), vector.end(), data.begin(), queue); BOOST_CHECK(data[0] == std::make_pair(0, 4.2f)); BOOST_CHECK(data[1] == std::make_pair(1, 2.3f)); BOOST_CHECK(data[2] == std::make_pair(2, 1.0f)); // sort by second component BOOST_COMPUTE_FUNCTION(bool, compare_second, (std::pair a, std::pair b), { return a.second < b.second; }); compute::sort(vector.begin(), vector.end(), compare_second, queue); compute::copy(vector.begin(), vector.end(), data.begin(), queue); BOOST_CHECK(data[0] == std::make_pair(2, 1.0f)); BOOST_CHECK(data[1] == std::make_pair(1, 2.3f)); BOOST_CHECK(data[2] == std::make_pair(0, 4.2f)); } BOOST_AUTO_TEST_CASE(transform_zip_iterator) { float float_data[] = { 1.f, 2.f, 3.f, 4.f }; compute::vector input_floats(float_data, float_data + 4, queue); int int_data[] = { 2, 4, 6, 8 }; compute::vector input_ints(int_data, int_data + 4, queue); compute::vector results(4, context); BOOST_COMPUTE_FUNCTION(float, tuple_pown, (boost::tuple x), { return pown(boost_tuple_get(x, 0), boost_tuple_get(x, 1)); }); compute::transform( compute::make_zip_iterator( boost::make_tuple(input_floats.begin(), input_ints.begin()) ), compute::make_zip_iterator( boost::make_tuple(input_floats.end(), input_ints.end()) ), results.begin(), tuple_pown, queue ); float results_data[4]; compute::copy(results.begin(), results.end(), results_data, queue); BOOST_CHECK_CLOSE(results_data[0], 1.f, 1e-4); BOOST_CHECK_CLOSE(results_data[1], 16.f, 1e-4); BOOST_CHECK_CLOSE(results_data[2], 729.f, 1e-4); BOOST_CHECK_CLOSE(results_data[3], 65536.f, 1e-4); } static BOOST_COMPUTE_FUNCTION(int, static_function, (int x), { return x + 5; }); BOOST_AUTO_TEST_CASE(test_static_function) { int data[] = { 1, 2, 3, 4}; compute::vector vec(data, data + 4, queue); compute::transform( vec.begin(), vec.end(), vec.begin(), static_function, queue ); CHECK_RANGE_EQUAL(int, 4, vec, (6, 7, 8, 9)); } template inline compute::function make_negate_function() { BOOST_COMPUTE_FUNCTION(T, negate, (const T x), { return -x; }); return negate; } BOOST_AUTO_TEST_CASE(test_templated_function) { int int_data[] = { 1, 2, 3, 4 }; compute::vector int_vec(int_data, int_data + 4, queue); compute::function negate_int = make_negate_function(); compute::transform( int_vec.begin(), int_vec.end(), int_vec.begin(), negate_int, queue ); CHECK_RANGE_EQUAL(int, 4, int_vec, (-1, -2, -3, -4)); float float_data[] = { 1.1f, 2.2f, 3.3f, 4.4f }; compute::vector float_vec(float_data, float_data + 4, queue); compute::function negate_float = make_negate_function(); compute::transform( float_vec.begin(), float_vec.end(), float_vec.begin(), negate_float, queue ); CHECK_RANGE_EQUAL(float, 4, float_vec, (-1.1f, -2.2f, -3.3f, -4.4f)); } BOOST_AUTO_TEST_CASE(define) { BOOST_COMPUTE_FUNCTION(int, return_number, (), { return NUMBER; }); return_number.define("NUMBER", "4"); compute::vector vec(1, context); compute::generate(vec.begin(), vec.end(), return_number, queue); CHECK_RANGE_EQUAL(int, 1, vec, (4)); return_number.define("NUMBER", "2"); compute::generate(vec.begin(), vec.end(), return_number, queue); CHECK_RANGE_EQUAL(int, 1, vec, (2)); } BOOST_AUTO_TEST_SUITE_END()