2
0
mirror of https://github.com/boostorg/compute.git synced 2026-02-19 14:22:12 +00:00

Merge pull request #305 from kylelutz/stable-sort-radix-sort

Use radix_sort() for stable_sort() when possible
This commit is contained in:
Kyle Lutz
2014-11-10 21:24:14 -08:00
5 changed files with 170 additions and 31 deletions

View File

@@ -177,4 +177,13 @@ BOOST_AUTO_TEST_CASE(sort_double_vector)
);
}
BOOST_AUTO_TEST_CASE(sort_partial_vector)
{
int data[] = { 9, 8, 7, 6, 5, 4, 3, 2, 1, 0 };
boost::compute::vector<int> vec(data, data + 10, queue);
boost::compute::detail::radix_sort(vec.begin() + 2, vec.end() - 2, queue);
CHECK_RANGE_EQUAL(int, 10, vec, (9, 8, 2, 3, 4, 5, 6, 7, 1, 0));
}
BOOST_AUTO_TEST_SUITE_END()

View File

@@ -12,6 +12,7 @@
#include <boost/test/unit_test.hpp>
#include <boost/compute/system.hpp>
#include <boost/compute/function.hpp>
#include <boost/compute/algorithm/stable_sort.hpp>
#include <boost/compute/algorithm/is_sorted.hpp>
#include <boost/compute/container/vector.hpp>
@@ -19,16 +20,73 @@
#include "check_macros.hpp"
#include "context_setup.hpp"
namespace compute = boost::compute;
BOOST_AUTO_TEST_CASE(sort_int_vector)
{
int data[] = { -4, 152, -5000, 963, 75321, -456, 0, 1112 };
boost::compute::vector<int> vector(data, data + 8);
compute::vector<int> vector(data, data + 8, queue);
BOOST_CHECK_EQUAL(vector.size(), size_t(8));
BOOST_CHECK(boost::compute::is_sorted(vector.begin(), vector.end()) == false);
BOOST_CHECK(compute::is_sorted(vector.begin(), vector.end(), queue) == false);
boost::compute::stable_sort(vector.begin(), vector.end());
BOOST_CHECK(boost::compute::is_sorted(vector.begin(), vector.end()) == true);
compute::stable_sort(vector.begin(), vector.end(), queue);
BOOST_CHECK(compute::is_sorted(vector.begin(), vector.end(), queue) == true);
CHECK_RANGE_EQUAL(int, 8, vector, (-5000, -456, -4, 0, 152, 963, 1112, 75321));
// sort reversed
compute::stable_sort(vector.begin(), vector.end(), compute::greater<int>(), queue);
CHECK_RANGE_EQUAL(int, 8, vector, (75321, 1112, 963, 152, 0, -4, -456, -5000));
}
BOOST_AUTO_TEST_CASE(sort_int2)
{
using compute::int2_;
// device vector of int2's
compute::vector<int2_> vec(context);
vec.push_back(int2_(2, 1), queue);
vec.push_back(int2_(2, 2), queue);
vec.push_back(int2_(1, 2), queue);
vec.push_back(int2_(1, 1), queue);
// function comparing the first component of each int2
BOOST_COMPUTE_FUNCTION(bool, compare_first, (int2_ a, int2_ b),
{
return a.x < b.x;
});
// ensure vector is not sorted
BOOST_CHECK(compute::is_sorted(vec.begin(), vec.end(), compare_first, queue) == false);
// sort elements based on their first component
compute::stable_sort(vec.begin(), vec.end(), compare_first, queue);
// ensure vector is now sorted
BOOST_CHECK(compute::is_sorted(vec.begin(), vec.end(), compare_first, queue) == true);
// check sorted vector order
std::vector<int2_> result(vec.size());
compute::copy(vec.begin(), vec.end(), result.begin(), queue);
BOOST_CHECK_EQUAL(result[0], int2_(1, 2));
BOOST_CHECK_EQUAL(result[1], int2_(1, 1));
BOOST_CHECK_EQUAL(result[2], int2_(2, 1));
BOOST_CHECK_EQUAL(result[3], int2_(2, 2));
// function comparing the second component of each int2
BOOST_COMPUTE_FUNCTION(bool, compare_second, (int2_ a, int2_ b),
{
return a.y < b.y;
});
// sort elements based on their second component
compute::stable_sort(vec.begin(), vec.end(), compare_second, queue);
// check sorted vector order
compute::copy(vec.begin(), vec.end(), result.begin(), queue);
BOOST_CHECK_EQUAL(result[0], int2_(1, 1));
BOOST_CHECK_EQUAL(result[1], int2_(2, 1));
BOOST_CHECK_EQUAL(result[2], int2_(1, 2));
BOOST_CHECK_EQUAL(result[3], int2_(2, 2));
}
BOOST_AUTO_TEST_SUITE_END()