LIVE / thrust /testing /equal.cu
Xu Ma
update
1c3c0d9
#include <unittest/unittest.h>
#include <thrust/equal.h>
#include <thrust/functional.h>
#include <thrust/iterator/retag.h>
#include <thrust/device_malloc.h>
#include <thrust/device_free.h>
template <class Vector>
void TestEqualSimple(void)
{
typedef typename Vector::value_type T;
Vector v1(5);
Vector v2(5);
v1[0] = 5; v1[1] = 2; v1[2] = 0; v1[3] = 0; v1[4] = 0;
v2[0] = 5; v2[1] = 2; v2[2] = 0; v2[3] = 6; v2[4] = 1;
ASSERT_EQUAL(thrust::equal(v1.begin(), v1.end(), v1.begin()), true);
ASSERT_EQUAL(thrust::equal(v1.begin(), v1.end(), v2.begin()), false);
ASSERT_EQUAL(thrust::equal(v2.begin(), v2.end(), v2.begin()), true);
ASSERT_EQUAL(thrust::equal(v1.begin(), v1.begin() + 0, v1.begin()), true);
ASSERT_EQUAL(thrust::equal(v1.begin(), v1.begin() + 1, v1.begin()), true);
ASSERT_EQUAL(thrust::equal(v1.begin(), v1.begin() + 3, v2.begin()), true);
ASSERT_EQUAL(thrust::equal(v1.begin(), v1.begin() + 4, v2.begin()), false);
ASSERT_EQUAL(thrust::equal(v1.begin(), v1.end(), v2.begin(), thrust::less_equal<T>()), true);
ASSERT_EQUAL(thrust::equal(v1.begin(), v1.end(), v2.begin(), thrust::greater<T>()), false);
}
DECLARE_VECTOR_UNITTEST(TestEqualSimple);
template <typename T>
void TestEqual(const size_t n)
{
thrust::host_vector<T> h_data1 = unittest::random_samples<T>(n);
thrust::host_vector<T> h_data2 = unittest::random_samples<T>(n);
thrust::device_vector<T> d_data1 = h_data1;
thrust::device_vector<T> d_data2 = h_data2;
//empty ranges
ASSERT_EQUAL(thrust::equal(h_data1.begin(), h_data1.begin(), h_data1.begin()), true);
ASSERT_EQUAL(thrust::equal(d_data1.begin(), d_data1.begin(), d_data1.begin()), true);
//symmetric cases
ASSERT_EQUAL(thrust::equal(h_data1.begin(), h_data1.end(), h_data1.begin()), true);
ASSERT_EQUAL(thrust::equal(d_data1.begin(), d_data1.end(), d_data1.begin()), true);
if (n > 0)
{
h_data1[0] = 0; h_data2[0] = 1;
d_data1[0] = 0; d_data2[0] = 1;
//different vectors
ASSERT_EQUAL(thrust::equal(h_data1.begin(), h_data1.end(), h_data2.begin()), false);
ASSERT_EQUAL(thrust::equal(d_data1.begin(), d_data1.end(), d_data2.begin()), false);
//different predicates
ASSERT_EQUAL(thrust::equal(h_data1.begin(), h_data1.begin() + 1, h_data2.begin(), thrust::less<T>()), true);
ASSERT_EQUAL(thrust::equal(d_data1.begin(), d_data1.begin() + 1, d_data2.begin(), thrust::less<T>()), true);
ASSERT_EQUAL(thrust::equal(h_data1.begin(), h_data1.begin() + 1, h_data2.begin(), thrust::greater<T>()), false);
ASSERT_EQUAL(thrust::equal(d_data1.begin(), d_data1.begin() + 1, d_data2.begin(), thrust::greater<T>()), false);
}
}
DECLARE_VARIABLE_UNITTEST(TestEqual);
template<typename InputIterator1, typename InputIterator2>
bool equal(my_system &system, InputIterator1 /*first*/, InputIterator1, InputIterator2)
{
system.validate_dispatch();
return false;
}
void TestEqualDispatchExplicit()
{
thrust::device_vector<int> vec(1);
my_system sys(0);
thrust::equal(sys,
vec.begin(),
vec.end(),
vec.begin());
ASSERT_EQUAL(true, sys.is_valid());
}
DECLARE_UNITTEST(TestEqualDispatchExplicit);
template<typename InputIterator1, typename InputIterator2>
bool equal(my_tag, InputIterator1 first, InputIterator1, InputIterator2)
{
*first = 13;
return false;
}
void TestEqualDispatchImplicit()
{
thrust::device_vector<int> vec(1);
thrust::equal(thrust::retag<my_tag>(vec.begin()),
thrust::retag<my_tag>(vec.end()),
thrust::retag<my_tag>(vec.begin()));
ASSERT_EQUAL(13, vec.front());
}
DECLARE_UNITTEST(TestEqualDispatchImplicit);
struct only_set_when_both_expected
{
long long expected;
bool * flag;
__device__
bool operator()(long long x, long long y)
{
if (x == expected && y == expected)
{
*flag = true;
}
return x == y;
}
};
void TestEqualWithBigIndexesHelper(int magnitude)
{
thrust::counting_iterator<long long> begin(1);
thrust::counting_iterator<long long> end = begin + (1ll << magnitude);
ASSERT_EQUAL(thrust::distance(begin, end), 1ll << magnitude);
thrust::device_ptr<bool> has_executed = thrust::device_malloc<bool>(1);
*has_executed = false;
only_set_when_both_expected fn = { (1ll << magnitude) - 1,
thrust::raw_pointer_cast(has_executed) };
ASSERT_EQUAL(thrust::equal(thrust::device, begin, end, begin, fn), true);
bool has_executed_h = *has_executed;
thrust::device_free(has_executed);
ASSERT_EQUAL(has_executed_h, true);
}
void TestEqualWithBigIndexes()
{
TestEqualWithBigIndexesHelper(30);
TestEqualWithBigIndexesHelper(31);
TestEqualWithBigIndexesHelper(32);
TestEqualWithBigIndexesHelper(33);
}
DECLARE_UNITTEST(TestEqualWithBigIndexes);