LIVE / thrust /testing /reduce.cu
Xu Ma
update
1c3c0d9
raw
history blame
5.99 kB
#include <unittest/unittest.h>
#include <thrust/reduce.h>
#include <thrust/iterator/counting_iterator.h>
#include <thrust/iterator/constant_iterator.h>
#include <thrust/iterator/retag.h>
#include <limits>
template<typename T>
struct plus_mod_10
{
__host__ __device__
T operator()(T lhs, T rhs) const
{
return ((lhs % 10) + (rhs % 10)) % 10;
}
};
template<typename T>
struct is_equal_div_10_reduce
{
__host__ __device__
bool operator()(const T x, const T& y) const { return ((int) x / 10) == ((int) y / 10); }
};
template <class Vector>
void TestReduceSimple(void)
{
typedef typename Vector::value_type T;
Vector v(3);
v[0] = 1; v[1] = -2; v[2] = 3;
// no initializer
ASSERT_EQUAL(thrust::reduce(v.begin(), v.end()), 2);
// with initializer
ASSERT_EQUAL(thrust::reduce(v.begin(), v.end(), (T) 10), 12);
}
DECLARE_VECTOR_UNITTEST(TestReduceSimple);
template<typename InputIterator>
int reduce(my_system &system, InputIterator, InputIterator)
{
system.validate_dispatch();
return 13;
}
void TestReduceDispatchExplicit()
{
thrust::device_vector<int> vec;
my_system sys(0);
thrust::reduce(sys, vec.begin(), vec.end());
ASSERT_EQUAL(true, sys.is_valid());
}
DECLARE_UNITTEST(TestReduceDispatchExplicit);
template<typename InputIterator>
int reduce(my_tag, InputIterator, InputIterator)
{
return 13;
}
void TestReduceDispatchImplicit()
{
thrust::device_vector<int> vec;
int result = thrust::reduce(thrust::retag<my_tag>(vec.begin()),
thrust::retag<my_tag>(vec.end()));
ASSERT_EQUAL(13, result);
}
DECLARE_UNITTEST(TestReduceDispatchImplicit);
template <typename T>
struct TestReduce
{
void operator()(const size_t n)
{
thrust::host_vector<T> h_data = unittest::random_integers<T>(n);
thrust::device_vector<T> d_data = h_data;
T init = 13;
T h_result = thrust::reduce(h_data.begin(), h_data.end(), init);
T d_result = thrust::reduce(d_data.begin(), d_data.end(), init);
ASSERT_EQUAL(h_result, d_result);
}
};
VariableUnitTest<TestReduce, IntegralTypes> TestReduceInstance;
template <class IntVector, class FloatVector>
void TestReduceMixedTypes(void)
{
// make sure we get types for default args and operators correct
IntVector int_input(4);
int_input[0] = 1;
int_input[1] = 2;
int_input[2] = 3;
int_input[3] = 4;
FloatVector float_input(4);
float_input[0] = 1.5;
float_input[1] = 2.5;
float_input[2] = 3.5;
float_input[3] = 4.5;
// float -> int should use using plus<int> operator by default
ASSERT_EQUAL(thrust::reduce(float_input.begin(), float_input.end(), (int) 0), 10);
// int -> float should use using plus<float> operator by default
ASSERT_EQUAL(thrust::reduce(int_input.begin(), int_input.end(), (float) 0.5), 10.5);
}
void TestReduceMixedTypesHost(void)
{
TestReduceMixedTypes< thrust::host_vector<int>, thrust::host_vector<float> >();
}
DECLARE_UNITTEST(TestReduceMixedTypesHost);
void TestReduceMixedTypesDevice(void)
{
TestReduceMixedTypes< thrust::device_vector<int>, thrust::device_vector<float> >();
}
DECLARE_UNITTEST(TestReduceMixedTypesDevice);
template <typename T>
struct TestReduceWithOperator
{
void operator()(const size_t n)
{
thrust::host_vector<T> h_data = unittest::random_integers<T>(n);
thrust::device_vector<T> d_data = h_data;
T init = 3;
T cpu_result = thrust::reduce(h_data.begin(), h_data.end(), init, plus_mod_10<T>());
T gpu_result = thrust::reduce(d_data.begin(), d_data.end(), init, plus_mod_10<T>());
ASSERT_EQUAL(cpu_result, gpu_result);
}
};
VariableUnitTest<TestReduceWithOperator, UnsignedIntegralTypes> TestReduceWithOperatorInstance;
template <typename T>
struct plus_mod3
{
T * table;
plus_mod3(T * table) : table(table) {}
__host__ __device__
T operator()(T a, T b)
{
return table[(int) (a + b)];
}
};
template <typename Vector>
void TestReduceWithIndirection(void)
{
// add numbers modulo 3 with external lookup table
typedef typename Vector::value_type T;
Vector data(7);
data[0] = 0;
data[1] = 1;
data[2] = 2;
data[3] = 1;
data[4] = 2;
data[5] = 0;
data[6] = 1;
Vector table(6);
table[0] = 0;
table[1] = 1;
table[2] = 2;
table[3] = 0;
table[4] = 1;
table[5] = 2;
T result = thrust::reduce(data.begin(), data.end(), T(0), plus_mod3<T>(thrust::raw_pointer_cast(&table[0])));
ASSERT_EQUAL(result, T(1));
}
DECLARE_INTEGRAL_VECTOR_UNITTEST(TestReduceWithIndirection);
template<typename T>
void TestReduceCountingIterator()
{
size_t const n = 15 * sizeof(T);
ASSERT_LEQUAL(T(n), unittest::truncate_to_max_representable<T>(n));
thrust::counting_iterator<T, thrust::host_system_tag> h_first = thrust::make_counting_iterator<T>(0);
thrust::counting_iterator<T, thrust::device_system_tag> d_first = thrust::make_counting_iterator<T>(0);
T init = unittest::random_integer<T>();
T h_result = thrust::reduce(h_first, h_first + n, init);
T d_result = thrust::reduce(d_first, d_first + n, init);
// we use ASSERT_ALMOST_EQUAL because we're testing floating point types
ASSERT_ALMOST_EQUAL(h_result, d_result);
}
DECLARE_GENERIC_UNITTEST(TestReduceCountingIterator);
void TestReduceWithBigIndexesHelper(int magnitude)
{
thrust::constant_iterator<long long> begin(1);
thrust::constant_iterator<long long> end = begin + (1ll << magnitude);
ASSERT_EQUAL(thrust::distance(begin, end), 1ll << magnitude);
long long result = thrust::reduce(thrust::device, begin, end);
ASSERT_EQUAL(result, 1ll << magnitude);
}
void TestReduceWithBigIndexes()
{
TestReduceWithBigIndexesHelper(30);
TestReduceWithBigIndexesHelper(31);
TestReduceWithBigIndexesHelper(32);
TestReduceWithBigIndexesHelper(33);
}
DECLARE_UNITTEST(TestReduceWithBigIndexes);