mirror of
https://gitlab.kitware.com/vtk/vtk-m
synced 2024-09-08 13:23:51 +00:00
Consolidate UnitTestBitField
Previously, each device adapter implementation had their own version of this test by including a common header. Simplify this by making a single test in UnitTests_vtkm_cont_testing, which can now be compiled for and tested on a device.
This commit is contained in:
parent
05bf2e4518
commit
891cfcfa86
@ -9,7 +9,6 @@
|
||||
##============================================================================
|
||||
|
||||
set(unit_tests
|
||||
UnitTestCudaBitField.cu
|
||||
UnitTestCudaCellLocatorRectilinearGrid.cu
|
||||
UnitTestCudaCellLocatorTwoLevel.cu
|
||||
UnitTestCudaCellLocatorUniformGrid.cu
|
||||
|
@ -1,18 +0,0 @@
|
||||
//============================================================================
|
||||
// Copyright (c) Kitware, Inc.
|
||||
// All rights reserved.
|
||||
// See LICENSE.txt for details.
|
||||
//
|
||||
// This software is distributed WITHOUT ANY WARRANTY; without even
|
||||
// the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
|
||||
// PURPOSE. See the above copyright notice for more information.
|
||||
//============================================================================
|
||||
#include <vtkm/cont/cuda/DeviceAdapterCuda.h>
|
||||
#include <vtkm/cont/testing/TestingBitField.h>
|
||||
|
||||
int UnitTestCudaBitField(int argc, char* argv[])
|
||||
{
|
||||
auto& tracker = vtkm::cont::GetRuntimeDeviceTracker();
|
||||
tracker.ForceDevice(vtkm::cont::DeviceAdapterTagCuda{});
|
||||
return vtkm::cont::testing::TestingBitField<vtkm::cont::DeviceAdapterTagCuda>::Run(argc, argv);
|
||||
}
|
@ -9,7 +9,6 @@
|
||||
##============================================================================
|
||||
|
||||
set(unit_tests
|
||||
UnitTestKokkosBitField.cxx
|
||||
UnitTestKokkosCellLocatorRectilinearGrid.cxx
|
||||
UnitTestKokkosCellLocatorTwoLevel.cxx
|
||||
UnitTestKokkosCellLocatorUniformGrid.cxx
|
||||
|
@ -1,18 +0,0 @@
|
||||
//============================================================================
|
||||
// Copyright (c) Kitware, Inc.
|
||||
// All rights reserved.
|
||||
// See LICENSE.txt for details.
|
||||
//
|
||||
// This software is distributed WITHOUT ANY WARRANTY; without even
|
||||
// the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
|
||||
// PURPOSE. See the above copyright notice for more information.
|
||||
//============================================================================
|
||||
#include <vtkm/cont/kokkos/DeviceAdapterKokkos.h>
|
||||
#include <vtkm/cont/testing/TestingBitField.h>
|
||||
|
||||
int UnitTestKokkosBitField(int argc, char* argv[])
|
||||
{
|
||||
auto& tracker = vtkm::cont::GetRuntimeDeviceTracker();
|
||||
tracker.ForceDevice(vtkm::cont::DeviceAdapterTagKokkos{});
|
||||
return vtkm::cont::testing::TestingBitField<vtkm::cont::DeviceAdapterTagKokkos>::Run(argc, argv);
|
||||
}
|
@ -9,7 +9,6 @@
|
||||
##============================================================================
|
||||
|
||||
set(unit_tests
|
||||
UnitTestOpenMPBitField.cxx
|
||||
UnitTestOpenMPCellLocatorRectilinearGrid.cxx
|
||||
UnitTestOpenMPCellLocatorTwoLevel.cxx
|
||||
UnitTestOpenMPCellLocatorUniformGrid.cxx
|
||||
|
@ -1,18 +0,0 @@
|
||||
//============================================================================
|
||||
// Copyright (c) Kitware, Inc.
|
||||
// All rights reserved.
|
||||
// See LICENSE.txt for details.
|
||||
//
|
||||
// This software is distributed WITHOUT ANY WARRANTY; without even
|
||||
// the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
|
||||
// PURPOSE. See the above copyright notice for more information.
|
||||
//============================================================================
|
||||
#include <vtkm/cont/openmp/DeviceAdapterOpenMP.h>
|
||||
#include <vtkm/cont/testing/TestingBitField.h>
|
||||
|
||||
int UnitTestOpenMPBitField(int argc, char* argv[])
|
||||
{
|
||||
auto& tracker = vtkm::cont::GetRuntimeDeviceTracker();
|
||||
tracker.ForceDevice(vtkm::cont::DeviceAdapterTagOpenMP{});
|
||||
return vtkm::cont::testing::TestingBitField<vtkm::cont::DeviceAdapterTagOpenMP>::Run(argc, argv);
|
||||
}
|
@ -9,7 +9,6 @@
|
||||
##============================================================================
|
||||
|
||||
set(unit_tests
|
||||
UnitTestSerialBitField.cxx
|
||||
UnitTestSerialCellLocatorRectilinearGrid.cxx
|
||||
UnitTestSerialCellLocatorTwoLevel.cxx
|
||||
UnitTestSerialCellLocatorUniformGrid.cxx
|
||||
|
@ -1,18 +0,0 @@
|
||||
//============================================================================
|
||||
// Copyright (c) Kitware, Inc.
|
||||
// All rights reserved.
|
||||
// See LICENSE.txt for details.
|
||||
//
|
||||
// This software is distributed WITHOUT ANY WARRANTY; without even
|
||||
// the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
|
||||
// PURPOSE. See the above copyright notice for more information.
|
||||
//============================================================================
|
||||
#include <vtkm/cont/serial/DeviceAdapterSerial.h>
|
||||
#include <vtkm/cont/testing/TestingBitField.h>
|
||||
|
||||
int UnitTestSerialBitField(int argc, char* argv[])
|
||||
{
|
||||
auto& tracker = vtkm::cont::GetRuntimeDeviceTracker();
|
||||
tracker.ForceDevice(vtkm::cont::DeviceAdapterTagSerial{});
|
||||
return vtkm::cont::testing::TestingBitField<vtkm::cont::DeviceAdapterTagSerial>::Run(argc, argv);
|
||||
}
|
@ -9,7 +9,6 @@
|
||||
##============================================================================
|
||||
|
||||
set(unit_tests
|
||||
UnitTestTBBBitField.cxx
|
||||
UnitTestTBBCellLocatorRectilinearGrid.cxx
|
||||
UnitTestTBBCellLocatorTwoLevel.cxx
|
||||
UnitTestTBBCellLocatorUniformGrid.cxx
|
||||
|
@ -1,18 +0,0 @@
|
||||
//============================================================================
|
||||
// Copyright (c) Kitware, Inc.
|
||||
// All rights reserved.
|
||||
// See LICENSE.txt for details.
|
||||
//
|
||||
// This software is distributed WITHOUT ANY WARRANTY; without even
|
||||
// the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
|
||||
// PURPOSE. See the above copyright notice for more information.
|
||||
//============================================================================
|
||||
#include <vtkm/cont/tbb/DeviceAdapterTBB.h>
|
||||
#include <vtkm/cont/testing/TestingBitField.h>
|
||||
|
||||
int UnitTestTBBBitField(int argc, char* argv[])
|
||||
{
|
||||
auto& tracker = vtkm::cont::GetRuntimeDeviceTracker();
|
||||
tracker.ForceDevice(vtkm::cont::DeviceAdapterTagTBB{});
|
||||
return vtkm::cont::testing::TestingBitField<vtkm::cont::DeviceAdapterTagTBB>::Run(argc, argv);
|
||||
}
|
@ -98,6 +98,7 @@ set(unit_tests_device
|
||||
UnitTestArrayHandleXGCCoordinates.cxx
|
||||
UnitTestArrayHandleZip.cxx
|
||||
UnitTestArrayRangeCompute.cxx
|
||||
UnitTestBitField.cxx
|
||||
UnitTestCellLocatorChooser.cxx
|
||||
UnitTestCellLocatorGeneral.cxx
|
||||
UnitTestCellSet.cxx
|
||||
|
@ -1,714 +0,0 @@
|
||||
//============================================================================
|
||||
// Copyright (c) Kitware, Inc.
|
||||
// All rights reserved.
|
||||
// See LICENSE.txt for details.
|
||||
//
|
||||
// This software is distributed WITHOUT ANY WARRANTY; without even
|
||||
// the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
|
||||
// PURPOSE. See the above copyright notice for more information.
|
||||
//============================================================================
|
||||
#ifndef vtk_m_cont_testing_TestingBitFields_h
|
||||
#define vtk_m_cont_testing_TestingBitFields_h
|
||||
|
||||
#include <vtkm/cont/ArrayHandleBitField.h>
|
||||
#include <vtkm/cont/ArrayHandleCounting.h>
|
||||
#include <vtkm/cont/BitField.h>
|
||||
#include <vtkm/cont/DeviceAdapterAlgorithm.h>
|
||||
#include <vtkm/cont/Invoker.h>
|
||||
#include <vtkm/cont/RuntimeDeviceTracker.h>
|
||||
|
||||
#include <vtkm/worklet/WorkletMapField.h>
|
||||
|
||||
#include <vtkm/cont/testing/Testing.h>
|
||||
|
||||
#include <vtkm/exec/FunctorBase.h>
|
||||
|
||||
#include <cstdio>
|
||||
|
||||
#define DEVICE_ASSERT_MSG(cond, message) \
|
||||
do \
|
||||
{ \
|
||||
if (!(cond)) \
|
||||
{ \
|
||||
printf("Testing assert failed at %s:%d\n\t- Condition: %s\n\t- Subtest: %s\n", \
|
||||
__FILE__, \
|
||||
__LINE__, \
|
||||
#cond, \
|
||||
message); \
|
||||
return false; \
|
||||
} \
|
||||
} while (false)
|
||||
|
||||
#define DEVICE_ASSERT(cond) \
|
||||
do \
|
||||
{ \
|
||||
if (!(cond)) \
|
||||
{ \
|
||||
printf("Testing assert failed at %s:%d\n\t- Condition: %s\n", __FILE__, __LINE__, #cond); \
|
||||
return false; \
|
||||
} \
|
||||
} while (false)
|
||||
|
||||
// Test with some trailing bits in partial last word:
|
||||
#define NUM_BITS \
|
||||
vtkm::Id { 7681 }
|
||||
|
||||
using vtkm::cont::BitField;
|
||||
|
||||
namespace vtkm
|
||||
{
|
||||
namespace cont
|
||||
{
|
||||
namespace testing
|
||||
{
|
||||
|
||||
// Takes an ArrayHandleBitField as the boolean condition field
|
||||
class ConditionalMergeWorklet : public vtkm::worklet::WorkletMapField
|
||||
{
|
||||
public:
|
||||
using ControlSignature = void(FieldIn cond, FieldIn trueVals, FieldIn falseVals, FieldOut result);
|
||||
using ExecutionSignature = _4(_1, _2, _3);
|
||||
|
||||
template <typename T>
|
||||
VTKM_EXEC T operator()(bool cond, const T& trueVal, const T& falseVal) const
|
||||
{
|
||||
return cond ? trueVal : falseVal;
|
||||
}
|
||||
};
|
||||
|
||||
// Takes a BitFieldInOut as the condition information, and reverses
|
||||
// the bits in place after performing the merge.
|
||||
class ConditionalMergeWorklet2 : public vtkm::worklet::WorkletMapField
|
||||
{
|
||||
public:
|
||||
using ControlSignature = void(BitFieldInOut bits,
|
||||
FieldIn trueVals,
|
||||
FieldIn falseVal,
|
||||
FieldOut result);
|
||||
using ExecutionSignature = _4(InputIndex, _1, _2, _3);
|
||||
using InputDomain = _2;
|
||||
|
||||
template <typename BitPortal, typename T>
|
||||
VTKM_EXEC T
|
||||
operator()(const vtkm::Id i, BitPortal& bits, const T& trueVal, const T& falseVal) const
|
||||
{
|
||||
return bits.XorBitAtomic(i, true) ? trueVal : falseVal;
|
||||
}
|
||||
};
|
||||
|
||||
/// This class has a single static member, Run, that runs all tests with the
|
||||
/// given DeviceAdapter.
|
||||
template <class DeviceAdapterTag>
|
||||
struct TestingBitField
|
||||
{
|
||||
using Algo = vtkm::cont::DeviceAdapterAlgorithm<DeviceAdapterTag>;
|
||||
using Traits = vtkm::cont::detail::BitFieldTraits;
|
||||
using WordTypes = vtkm::AtomicTypesSupported;
|
||||
|
||||
VTKM_EXEC_CONT
|
||||
static bool RandomBitFromIndex(vtkm::Id idx) noexcept
|
||||
{
|
||||
// Some random operations that will give a pseudorandom stream of bits:
|
||||
auto m = idx + (idx * 2) - (idx / 3) + (idx * 5 / 7) - (idx * 11 / 13);
|
||||
return (m % 2) == 1;
|
||||
}
|
||||
|
||||
template <typename WordType>
|
||||
VTKM_EXEC_CONT static WordType RandomWordFromIndex(vtkm::Id idx) noexcept
|
||||
{
|
||||
vtkm::UInt64 m = static_cast<vtkm::UInt64>(idx * (NUM_BITS - 1) + (idx + 1) * NUM_BITS);
|
||||
m ^= m << 3;
|
||||
m ^= m << 7;
|
||||
m ^= m << 15;
|
||||
m ^= m << 31;
|
||||
m = (m << 32) | (m >> 32);
|
||||
|
||||
const size_t mBits = 64;
|
||||
const size_t wordBits = sizeof(WordType) * CHAR_BIT;
|
||||
|
||||
const WordType highWord = static_cast<WordType>(m >> (mBits - wordBits));
|
||||
return highWord;
|
||||
}
|
||||
|
||||
VTKM_CONT
|
||||
static BitField RandomBitField(vtkm::Id numBits = NUM_BITS)
|
||||
{
|
||||
BitField field;
|
||||
field.Allocate(numBits);
|
||||
auto portal = field.WritePortal();
|
||||
for (vtkm::Id i = 0; i < numBits; ++i)
|
||||
{
|
||||
portal.SetBit(i, RandomBitFromIndex(i));
|
||||
}
|
||||
|
||||
return field;
|
||||
}
|
||||
|
||||
VTKM_CONT
|
||||
static void TestBlockAllocation()
|
||||
{
|
||||
BitField field;
|
||||
field.Allocate(NUM_BITS);
|
||||
|
||||
// NumBits should be rounded up to the nearest block of bytes, as defined in
|
||||
// the traits:
|
||||
const vtkm::BufferSizeType bytesInFieldData = field.GetBuffer().GetNumberOfBytes();
|
||||
|
||||
const vtkm::BufferSizeType blockSize = vtkm::cont::detail::BitFieldTraits::BlockSize;
|
||||
const vtkm::BufferSizeType numBytes = (NUM_BITS + CHAR_BIT - 1) / CHAR_BIT;
|
||||
const vtkm::BufferSizeType numBlocks = (numBytes + blockSize - 1) / blockSize;
|
||||
const vtkm::BufferSizeType expectedBytes = numBlocks * blockSize;
|
||||
|
||||
VTKM_TEST_ASSERT(bytesInFieldData == expectedBytes,
|
||||
"The BitField allocation does not round up to the nearest "
|
||||
"block. This can cause access-by-word to read/write invalid "
|
||||
"memory.");
|
||||
}
|
||||
|
||||
template <typename PortalType>
|
||||
VTKM_EXEC_CONT static bool TestBitValue(const char* operation,
|
||||
vtkm::Id i,
|
||||
PortalType portal,
|
||||
bool& bit,
|
||||
bool originalBit)
|
||||
{
|
||||
auto expected = bit;
|
||||
auto result = portal.GetBitAtomic(i);
|
||||
DEVICE_ASSERT_MSG(result == expected, operation);
|
||||
|
||||
// Reset
|
||||
bit = originalBit;
|
||||
portal.SetBitAtomic(i, bit);
|
||||
return true;
|
||||
}
|
||||
|
||||
template <typename PortalType>
|
||||
VTKM_EXEC_CONT static bool HelpTestBit(vtkm::Id i, PortalType portal)
|
||||
{
|
||||
const auto origBit = RandomBitFromIndex(i);
|
||||
auto bit = origBit;
|
||||
|
||||
const auto mod = RandomBitFromIndex(i + NUM_BITS);
|
||||
|
||||
bit = mod;
|
||||
portal.SetBitAtomic(i, mod);
|
||||
DEVICE_ASSERT(TestBitValue("SetBitAtomic", i, portal, bit, origBit));
|
||||
|
||||
bit = !bit;
|
||||
portal.NotBitAtomic(i);
|
||||
DEVICE_ASSERT(TestBitValue("NotBitAtomic", i, portal, bit, origBit));
|
||||
|
||||
bit = bit && mod;
|
||||
portal.AndBitAtomic(i, mod);
|
||||
DEVICE_ASSERT(TestBitValue("AndBitAtomic", i, portal, bit, origBit));
|
||||
|
||||
bit = bit || mod;
|
||||
portal.OrBitAtomic(i, mod);
|
||||
DEVICE_ASSERT(TestBitValue("OrBitAtomic", i, portal, bit, origBit));
|
||||
|
||||
bit = bit != mod;
|
||||
portal.XorBitAtomic(i, mod);
|
||||
DEVICE_ASSERT(TestBitValue("XorBitAtomic", i, portal, bit, origBit));
|
||||
|
||||
const auto notBit = !bit;
|
||||
// A compare-exchange that should fail
|
||||
auto expectedBit = notBit;
|
||||
bool cxResult = portal.CompareExchangeBitAtomic(i, &expectedBit, bit);
|
||||
DEVICE_ASSERT(!cxResult);
|
||||
DEVICE_ASSERT(expectedBit != notBit);
|
||||
DEVICE_ASSERT(portal.GetBit(i) == expectedBit);
|
||||
DEVICE_ASSERT(portal.GetBit(i) == bit);
|
||||
|
||||
// A compare-exchange that should succeed.
|
||||
expectedBit = bit;
|
||||
cxResult = portal.CompareExchangeBitAtomic(i, &expectedBit, notBit);
|
||||
DEVICE_ASSERT(cxResult);
|
||||
DEVICE_ASSERT(expectedBit == bit);
|
||||
DEVICE_ASSERT(portal.GetBit(i) == notBit);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
template <typename WordType, typename PortalType>
|
||||
VTKM_EXEC_CONT static bool TestWordValue(const char* operation,
|
||||
vtkm::Id i,
|
||||
const PortalType& portal,
|
||||
WordType& word,
|
||||
WordType originalWord)
|
||||
{
|
||||
auto expected = word;
|
||||
auto result = portal.template GetWordAtomic<WordType>(i);
|
||||
DEVICE_ASSERT_MSG(result == expected, operation);
|
||||
|
||||
// Reset
|
||||
word = originalWord;
|
||||
portal.SetWordAtomic(i, word);
|
||||
return true;
|
||||
}
|
||||
|
||||
template <typename WordType, typename PortalType>
|
||||
VTKM_EXEC_CONT static bool HelpTestWord(vtkm::Id i, PortalType portal)
|
||||
{
|
||||
const auto origWord = RandomWordFromIndex<WordType>(i);
|
||||
auto word = origWord;
|
||||
|
||||
const auto mod = RandomWordFromIndex<WordType>(i + NUM_BITS);
|
||||
|
||||
portal.SetWord(i, word);
|
||||
DEVICE_ASSERT(TestWordValue("SetWord", i, portal, word, origWord));
|
||||
|
||||
word = mod;
|
||||
portal.SetWordAtomic(i, mod);
|
||||
DEVICE_ASSERT(TestWordValue("SetWordAtomic", i, portal, word, origWord));
|
||||
|
||||
// C++ promotes e.g. uint8 to int32 when performing bitwise not. Silence
|
||||
// conversion warning and mask unimportant bits:
|
||||
word = static_cast<WordType>(~word);
|
||||
portal.template NotWordAtomic<WordType>(i);
|
||||
DEVICE_ASSERT(TestWordValue("NotWordAtomic", i, portal, word, origWord));
|
||||
|
||||
word = word & mod;
|
||||
portal.AndWordAtomic(i, mod);
|
||||
DEVICE_ASSERT(TestWordValue("AndWordAtomic", i, portal, word, origWord));
|
||||
|
||||
word = word | mod;
|
||||
portal.OrWordAtomic(i, mod);
|
||||
DEVICE_ASSERT(TestWordValue("OrWordAtomic", i, portal, word, origWord));
|
||||
|
||||
word = word ^ mod;
|
||||
portal.XorWordAtomic(i, mod);
|
||||
DEVICE_ASSERT(TestWordValue("XorWordAtomic", i, portal, word, origWord));
|
||||
|
||||
// Compare-exchange that should fail
|
||||
const WordType notWord = static_cast<WordType>(~word);
|
||||
WordType expectedWord = notWord;
|
||||
bool cxResult = portal.CompareExchangeWordAtomic(i, &expectedWord, word);
|
||||
DEVICE_ASSERT(!cxResult);
|
||||
DEVICE_ASSERT(expectedWord != notWord);
|
||||
DEVICE_ASSERT(portal.template GetWord<WordType>(i) == expectedWord);
|
||||
DEVICE_ASSERT(portal.template GetWord<WordType>(i) == word);
|
||||
|
||||
// Compare-exchange that should succeed
|
||||
expectedWord = word;
|
||||
cxResult = portal.CompareExchangeWordAtomic(i, &expectedWord, notWord);
|
||||
DEVICE_ASSERT(cxResult);
|
||||
DEVICE_ASSERT(expectedWord == word);
|
||||
DEVICE_ASSERT(portal.template GetWord<WordType>(i) == notWord);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
template <typename PortalType>
|
||||
struct HelpTestWordOpsControl
|
||||
{
|
||||
PortalType Portal;
|
||||
|
||||
VTKM_CONT
|
||||
HelpTestWordOpsControl(PortalType portal)
|
||||
: Portal(portal)
|
||||
{
|
||||
}
|
||||
|
||||
template <typename WordType>
|
||||
VTKM_CONT void operator()(WordType)
|
||||
{
|
||||
const auto numWords = this->Portal.template GetNumberOfWords<WordType>();
|
||||
for (vtkm::Id i = 0; i < numWords; ++i)
|
||||
{
|
||||
VTKM_TEST_ASSERT(HelpTestWord<WordType>(i, this->Portal));
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
template <typename Portal>
|
||||
VTKM_CONT static void HelpTestPortalsControl(Portal portal)
|
||||
{
|
||||
const auto numWords8 = (NUM_BITS + 7) / 8;
|
||||
const auto numWords16 = (NUM_BITS + 15) / 16;
|
||||
const auto numWords32 = (NUM_BITS + 31) / 32;
|
||||
const auto numWords64 = (NUM_BITS + 63) / 64;
|
||||
|
||||
VTKM_TEST_ASSERT(portal.GetNumberOfBits() == NUM_BITS);
|
||||
VTKM_TEST_ASSERT(portal.template GetNumberOfWords<vtkm::UInt8>() == numWords8);
|
||||
VTKM_TEST_ASSERT(portal.template GetNumberOfWords<vtkm::UInt16>() == numWords16);
|
||||
VTKM_TEST_ASSERT(portal.template GetNumberOfWords<vtkm::UInt32>() == numWords32);
|
||||
VTKM_TEST_ASSERT(portal.template GetNumberOfWords<vtkm::UInt64>() == numWords64);
|
||||
|
||||
for (vtkm::Id i = 0; i < NUM_BITS; ++i)
|
||||
{
|
||||
HelpTestBit(i, portal);
|
||||
}
|
||||
|
||||
HelpTestWordOpsControl<Portal> test(portal);
|
||||
vtkm::ListForEach(test, vtkm::AtomicTypesSupported{});
|
||||
}
|
||||
|
||||
VTKM_CONT
|
||||
static void TestControlPortals()
|
||||
{
|
||||
auto field = RandomBitField();
|
||||
|
||||
HelpTestPortalsControl(field.WritePortal());
|
||||
}
|
||||
|
||||
template <typename Portal>
|
||||
VTKM_EXEC_CONT static bool HelpTestPortalSanityExecution(Portal portal)
|
||||
{
|
||||
const auto numWords8 = (NUM_BITS + 7) / 8;
|
||||
const auto numWords16 = (NUM_BITS + 15) / 16;
|
||||
const auto numWords32 = (NUM_BITS + 31) / 32;
|
||||
const auto numWords64 = (NUM_BITS + 63) / 64;
|
||||
|
||||
DEVICE_ASSERT(portal.GetNumberOfBits() == NUM_BITS);
|
||||
DEVICE_ASSERT(portal.template GetNumberOfWords<vtkm::UInt8>() == numWords8);
|
||||
DEVICE_ASSERT(portal.template GetNumberOfWords<vtkm::UInt16>() == numWords16);
|
||||
DEVICE_ASSERT(portal.template GetNumberOfWords<vtkm::UInt32>() == numWords32);
|
||||
DEVICE_ASSERT(portal.template GetNumberOfWords<vtkm::UInt64>() == numWords64);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
template <typename WordType, typename PortalType>
|
||||
struct HelpTestPortalsExecutionWordsFunctor : vtkm::exec::FunctorBase
|
||||
{
|
||||
PortalType Portal;
|
||||
|
||||
HelpTestPortalsExecutionWordsFunctor(PortalType portal)
|
||||
: Portal(portal)
|
||||
{
|
||||
}
|
||||
|
||||
VTKM_EXEC_CONT
|
||||
void operator()(vtkm::Id i) const
|
||||
{
|
||||
if (i == 0)
|
||||
{
|
||||
if (!HelpTestPortalSanityExecution(this->Portal))
|
||||
{
|
||||
this->RaiseError("Testing Portal sanity failed.");
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
if (!HelpTestWord<WordType>(i, this->Portal))
|
||||
{
|
||||
this->RaiseError("Testing word operations failed.");
|
||||
return;
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
template <typename PortalType>
|
||||
struct HelpTestPortalsExecutionBitsFunctor : vtkm::exec::FunctorBase
|
||||
{
|
||||
PortalType Portal;
|
||||
|
||||
HelpTestPortalsExecutionBitsFunctor(PortalType portal)
|
||||
: Portal(portal)
|
||||
{
|
||||
}
|
||||
|
||||
VTKM_EXEC_CONT
|
||||
void operator()(vtkm::Id i) const
|
||||
{
|
||||
if (!HelpTestBit(i, this->Portal))
|
||||
{
|
||||
this->RaiseError("Testing bit operations failed.");
|
||||
return;
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
template <typename PortalType>
|
||||
struct HelpTestWordOpsExecution
|
||||
{
|
||||
PortalType Portal;
|
||||
|
||||
VTKM_CONT
|
||||
HelpTestWordOpsExecution(PortalType portal)
|
||||
: Portal(portal)
|
||||
{
|
||||
}
|
||||
|
||||
template <typename WordType>
|
||||
VTKM_CONT void operator()(WordType)
|
||||
{
|
||||
const auto numWords = this->Portal.template GetNumberOfWords<WordType>();
|
||||
|
||||
using WordFunctor = HelpTestPortalsExecutionWordsFunctor<WordType, PortalType>;
|
||||
WordFunctor test{ this->Portal };
|
||||
Algo::Schedule(test, numWords);
|
||||
}
|
||||
};
|
||||
|
||||
template <typename Portal>
|
||||
VTKM_CONT static void HelpTestPortalsExecution(Portal portal)
|
||||
{
|
||||
HelpTestPortalsExecutionBitsFunctor<Portal> bitTest{ portal };
|
||||
Algo::Schedule(bitTest, portal.GetNumberOfBits());
|
||||
|
||||
|
||||
HelpTestWordOpsExecution<Portal> test(portal);
|
||||
vtkm::ListForEach(test, vtkm::AtomicTypesSupported{});
|
||||
}
|
||||
|
||||
VTKM_CONT
|
||||
static void TestExecutionPortals()
|
||||
{
|
||||
vtkm::cont::Token token;
|
||||
auto field = RandomBitField();
|
||||
auto portal = field.PrepareForInPlace(DeviceAdapterTag{}, token);
|
||||
|
||||
HelpTestPortalsExecution(portal);
|
||||
}
|
||||
|
||||
VTKM_CONT
|
||||
static void TestFinalWordMask()
|
||||
{
|
||||
auto testMask32 = [](vtkm::Id numBits, vtkm::UInt32 expectedMask) {
|
||||
vtkm::cont::BitField field;
|
||||
field.Allocate(numBits);
|
||||
auto mask = field.ReadPortal().GetFinalWordMask<vtkm::UInt32>();
|
||||
|
||||
VTKM_TEST_ASSERT(expectedMask == mask,
|
||||
"Unexpected mask for BitField size ",
|
||||
numBits,
|
||||
": Expected 0x",
|
||||
std::hex,
|
||||
expectedMask,
|
||||
" got 0x",
|
||||
mask);
|
||||
};
|
||||
|
||||
auto testMask64 = [](vtkm::Id numBits, vtkm::UInt64 expectedMask) {
|
||||
vtkm::cont::BitField field;
|
||||
field.Allocate(numBits);
|
||||
auto mask = field.ReadPortal().GetFinalWordMask<vtkm::UInt64>();
|
||||
|
||||
VTKM_TEST_ASSERT(expectedMask == mask,
|
||||
"Unexpected mask for BitField size ",
|
||||
numBits,
|
||||
": Expected 0x",
|
||||
std::hex,
|
||||
expectedMask,
|
||||
" got 0x",
|
||||
mask);
|
||||
};
|
||||
|
||||
testMask32(0, 0x00000000);
|
||||
testMask32(1, 0x00000001);
|
||||
testMask32(2, 0x00000003);
|
||||
testMask32(3, 0x00000007);
|
||||
testMask32(4, 0x0000000f);
|
||||
testMask32(5, 0x0000001f);
|
||||
testMask32(8, 0x000000ff);
|
||||
testMask32(16, 0x0000ffff);
|
||||
testMask32(24, 0x00ffffff);
|
||||
testMask32(25, 0x01ffffff);
|
||||
testMask32(31, 0x7fffffff);
|
||||
testMask32(32, 0xffffffff);
|
||||
testMask32(64, 0xffffffff);
|
||||
testMask32(128, 0xffffffff);
|
||||
testMask32(129, 0x00000001);
|
||||
|
||||
testMask64(0, 0x0000000000000000);
|
||||
testMask64(1, 0x0000000000000001);
|
||||
testMask64(2, 0x0000000000000003);
|
||||
testMask64(3, 0x0000000000000007);
|
||||
testMask64(4, 0x000000000000000f);
|
||||
testMask64(5, 0x000000000000001f);
|
||||
testMask64(8, 0x00000000000000ff);
|
||||
testMask64(16, 0x000000000000ffff);
|
||||
testMask64(24, 0x0000000000ffffff);
|
||||
testMask64(25, 0x0000000001ffffff);
|
||||
testMask64(31, 0x000000007fffffff);
|
||||
testMask64(32, 0x00000000ffffffff);
|
||||
testMask64(40, 0x000000ffffffffff);
|
||||
testMask64(48, 0x0000ffffffffffff);
|
||||
testMask64(56, 0x00ffffffffffffff);
|
||||
testMask64(64, 0xffffffffffffffff);
|
||||
testMask64(128, 0xffffffffffffffff);
|
||||
testMask64(129, 0x0000000000000001);
|
||||
}
|
||||
|
||||
VTKM_CONT static void TestFill()
|
||||
{
|
||||
vtkm::cont::BitField bitField;
|
||||
bitField.Allocate(NUM_BITS);
|
||||
|
||||
bitField.Fill(true);
|
||||
{
|
||||
auto portal = bitField.ReadPortal();
|
||||
for (vtkm::Id index = 0; index < NUM_BITS; ++index)
|
||||
{
|
||||
VTKM_TEST_ASSERT(portal.GetBit(index));
|
||||
}
|
||||
}
|
||||
|
||||
constexpr vtkm::UInt8 word8 = 0xA6;
|
||||
bitField.Fill(word8);
|
||||
{
|
||||
auto portal = bitField.ReadPortal();
|
||||
for (vtkm::Id index = 0; index < NUM_BITS; ++index)
|
||||
{
|
||||
VTKM_TEST_ASSERT(portal.GetBit(index) == ((word8 >> (index % 8)) & 0x01));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
struct ArrayHandleBitFieldChecker : vtkm::exec::FunctorBase
|
||||
{
|
||||
using PortalType = vtkm::cont::ArrayHandleBitField::WritePortalType;
|
||||
|
||||
PortalType Portal;
|
||||
bool InvertReference;
|
||||
|
||||
VTKM_EXEC_CONT
|
||||
ArrayHandleBitFieldChecker(PortalType portal, bool invert)
|
||||
: Portal(portal)
|
||||
, InvertReference(invert)
|
||||
{
|
||||
}
|
||||
|
||||
VTKM_EXEC
|
||||
void operator()(vtkm::Id i) const
|
||||
{
|
||||
const bool ref = this->InvertReference ? !RandomBitFromIndex(i) : RandomBitFromIndex(i);
|
||||
if (this->Portal.Get(i) != ref)
|
||||
{
|
||||
this->RaiseError("Unexpected value from ArrayHandleBitField portal.");
|
||||
return;
|
||||
}
|
||||
|
||||
// Flip the bit for the next kernel launch, which tests that the bitfield
|
||||
// is inverted.
|
||||
this->Portal.Set(i, !ref);
|
||||
}
|
||||
};
|
||||
|
||||
VTKM_CONT
|
||||
static void TestArrayHandleBitField()
|
||||
{
|
||||
auto handle = vtkm::cont::make_ArrayHandleBitField(RandomBitField());
|
||||
const vtkm::Id numBits = handle.GetNumberOfValues();
|
||||
|
||||
VTKM_TEST_ASSERT(numBits == NUM_BITS,
|
||||
"ArrayHandleBitField returned the wrong number of values. "
|
||||
"Expected: ",
|
||||
NUM_BITS,
|
||||
" got: ",
|
||||
numBits);
|
||||
|
||||
{
|
||||
vtkm::cont::Token token;
|
||||
Algo::Schedule(
|
||||
ArrayHandleBitFieldChecker{ handle.PrepareForInPlace(DeviceAdapterTag{}, token), false },
|
||||
numBits);
|
||||
Algo::Schedule(
|
||||
ArrayHandleBitFieldChecker{ handle.PrepareForInPlace(DeviceAdapterTag{}, token), true },
|
||||
numBits);
|
||||
}
|
||||
|
||||
handle.Fill(true);
|
||||
{
|
||||
auto portal = handle.ReadPortal();
|
||||
for (vtkm::Id index = 0; index < NUM_BITS; ++index)
|
||||
{
|
||||
VTKM_TEST_ASSERT(portal.Get(index));
|
||||
}
|
||||
}
|
||||
|
||||
handle.Fill(false, 24);
|
||||
handle.Fill(true, 64);
|
||||
{
|
||||
auto portal = handle.ReadPortal();
|
||||
for (vtkm::Id index = 0; index < NUM_BITS; ++index)
|
||||
{
|
||||
VTKM_TEST_ASSERT(portal.Get(index) == ((index < 24) || (index >= 64)));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
VTKM_CONT
|
||||
static void TestArrayInvokeWorklet()
|
||||
{
|
||||
auto condArray = vtkm::cont::make_ArrayHandleBitField(RandomBitField());
|
||||
auto trueArray = vtkm::cont::make_ArrayHandleCounting<vtkm::Id>(20, 2, NUM_BITS);
|
||||
auto falseArray = vtkm::cont::make_ArrayHandleCounting<vtkm::Id>(13, 2, NUM_BITS);
|
||||
vtkm::cont::ArrayHandle<vtkm::Id> output;
|
||||
|
||||
vtkm::cont::Invoker invoke;
|
||||
invoke(ConditionalMergeWorklet{}, condArray, trueArray, falseArray, output);
|
||||
|
||||
auto condVals = condArray.ReadPortal();
|
||||
auto trueVals = trueArray.ReadPortal();
|
||||
auto falseVals = falseArray.ReadPortal();
|
||||
auto outVals = output.ReadPortal();
|
||||
|
||||
VTKM_TEST_ASSERT(condVals.GetNumberOfValues() == trueVals.GetNumberOfValues());
|
||||
VTKM_TEST_ASSERT(condVals.GetNumberOfValues() == falseVals.GetNumberOfValues());
|
||||
VTKM_TEST_ASSERT(condVals.GetNumberOfValues() == outVals.GetNumberOfValues());
|
||||
|
||||
for (vtkm::Id i = 0; i < condVals.GetNumberOfValues(); ++i)
|
||||
{
|
||||
VTKM_TEST_ASSERT(outVals.Get(i) == (condVals.Get(i) ? trueVals.Get(i) : falseVals.Get(i)));
|
||||
}
|
||||
}
|
||||
|
||||
VTKM_CONT
|
||||
static void TestArrayInvokeWorklet2()
|
||||
{
|
||||
auto condBits = RandomBitField();
|
||||
auto trueArray = vtkm::cont::make_ArrayHandleCounting<vtkm::Id>(20, 2, NUM_BITS);
|
||||
auto falseArray = vtkm::cont::make_ArrayHandleCounting<vtkm::Id>(13, 2, NUM_BITS);
|
||||
vtkm::cont::ArrayHandle<vtkm::Id> output;
|
||||
|
||||
vtkm::cont::Invoker invoke;
|
||||
invoke(ConditionalMergeWorklet2{}, condBits, trueArray, falseArray, output);
|
||||
|
||||
auto condVals = condBits.ReadPortal();
|
||||
auto trueVals = trueArray.ReadPortal();
|
||||
auto falseVals = falseArray.ReadPortal();
|
||||
auto outVals = output.ReadPortal();
|
||||
|
||||
VTKM_TEST_ASSERT(condVals.GetNumberOfBits() == trueVals.GetNumberOfValues());
|
||||
VTKM_TEST_ASSERT(condVals.GetNumberOfBits() == falseVals.GetNumberOfValues());
|
||||
VTKM_TEST_ASSERT(condVals.GetNumberOfBits() == outVals.GetNumberOfValues());
|
||||
|
||||
for (vtkm::Id i = 0; i < condVals.GetNumberOfBits(); ++i)
|
||||
{
|
||||
// The worklet flips the bitfield in place after choosing true/false paths
|
||||
VTKM_TEST_ASSERT(condVals.GetBit(i) == !RandomBitFromIndex(i));
|
||||
VTKM_TEST_ASSERT(outVals.Get(i) ==
|
||||
(!condVals.GetBit(i) ? trueVals.Get(i) : falseVals.Get(i)));
|
||||
}
|
||||
}
|
||||
|
||||
struct TestRunner
|
||||
{
|
||||
VTKM_CONT
|
||||
void operator()() const
|
||||
{
|
||||
TestingBitField::TestBlockAllocation();
|
||||
TestingBitField::TestControlPortals();
|
||||
TestingBitField::TestExecutionPortals();
|
||||
TestingBitField::TestFinalWordMask();
|
||||
TestingBitField::TestFill();
|
||||
TestingBitField::TestArrayHandleBitField();
|
||||
TestingBitField::TestArrayInvokeWorklet();
|
||||
TestingBitField::TestArrayInvokeWorklet2();
|
||||
}
|
||||
};
|
||||
|
||||
public:
|
||||
static VTKM_CONT int Run(int argc, char* argv[])
|
||||
{
|
||||
vtkm::cont::GetRuntimeDeviceTracker().ForceDevice(DeviceAdapterTag());
|
||||
return vtkm::cont::testing::Testing::Run(TestRunner{}, argc, argv);
|
||||
}
|
||||
};
|
||||
}
|
||||
}
|
||||
} // namespace vtkm::cont::testing
|
||||
|
||||
#endif // vtk_m_cont_testing_TestingBitFields_h
|
686
vtkm/cont/testing/UnitTestBitField.cxx
Normal file
686
vtkm/cont/testing/UnitTestBitField.cxx
Normal file
@ -0,0 +1,686 @@
|
||||
//============================================================================
|
||||
// Copyright (c) Kitware, Inc.
|
||||
// All rights reserved.
|
||||
// See LICENSE.txt for details.
|
||||
//
|
||||
// This software is distributed WITHOUT ANY WARRANTY; without even
|
||||
// the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
|
||||
// PURPOSE. See the above copyright notice for more information.
|
||||
//============================================================================
|
||||
|
||||
#include <vtkm/cont/Algorithm.h>
|
||||
#include <vtkm/cont/ArrayHandleBitField.h>
|
||||
#include <vtkm/cont/ArrayHandleCounting.h>
|
||||
#include <vtkm/cont/BitField.h>
|
||||
#include <vtkm/cont/DeviceAdapterAlgorithm.h>
|
||||
#include <vtkm/cont/Invoker.h>
|
||||
#include <vtkm/cont/TryExecute.h>
|
||||
|
||||
#include <vtkm/worklet/WorkletMapField.h>
|
||||
|
||||
#include <vtkm/exec/FunctorBase.h>
|
||||
|
||||
#include <vtkm/cont/testing/Testing.h>
|
||||
|
||||
#include <cstdio>
|
||||
|
||||
#define DEVICE_ASSERT_MSG(cond, message) \
|
||||
do \
|
||||
{ \
|
||||
if (!(cond)) \
|
||||
{ \
|
||||
printf("Testing assert failed at %s:%d\n\t- Condition: %s\n\t- Subtest: %s\n", \
|
||||
__FILE__, \
|
||||
__LINE__, \
|
||||
#cond, \
|
||||
message); \
|
||||
return false; \
|
||||
} \
|
||||
} while (false)
|
||||
|
||||
#define DEVICE_ASSERT(cond) \
|
||||
do \
|
||||
{ \
|
||||
if (!(cond)) \
|
||||
{ \
|
||||
printf("Testing assert failed at %s:%d\n\t- Condition: %s\n", __FILE__, __LINE__, #cond); \
|
||||
return false; \
|
||||
} \
|
||||
} while (false)
|
||||
|
||||
// Test with some trailing bits in partial last word:
|
||||
#define NUM_BITS \
|
||||
vtkm::Id { 7681 }
|
||||
|
||||
namespace
|
||||
{
|
||||
|
||||
// Takes an ArrayHandleBitField as the boolean condition field
|
||||
class ConditionalMergeWorklet : public vtkm::worklet::WorkletMapField
|
||||
{
|
||||
public:
|
||||
using ControlSignature = void(FieldIn cond, FieldIn trueVals, FieldIn falseVals, FieldOut result);
|
||||
using ExecutionSignature = _4(_1, _2, _3);
|
||||
|
||||
template <typename T>
|
||||
VTKM_EXEC T operator()(bool cond, const T& trueVal, const T& falseVal) const
|
||||
{
|
||||
return cond ? trueVal : falseVal;
|
||||
}
|
||||
};
|
||||
|
||||
// Takes a BitFieldInOut as the condition information, and reverses
|
||||
// the bits in place after performing the merge.
|
||||
class ConditionalMergeWorklet2 : public vtkm::worklet::WorkletMapField
|
||||
{
|
||||
public:
|
||||
using ControlSignature = void(BitFieldInOut bits,
|
||||
FieldIn trueVals,
|
||||
FieldIn falseVal,
|
||||
FieldOut result);
|
||||
using ExecutionSignature = _4(InputIndex, _1, _2, _3);
|
||||
using InputDomain = _2;
|
||||
|
||||
template <typename BitPortal, typename T>
|
||||
VTKM_EXEC T
|
||||
operator()(const vtkm::Id i, BitPortal& bits, const T& trueVal, const T& falseVal) const
|
||||
{
|
||||
return bits.XorBitAtomic(i, true) ? trueVal : falseVal;
|
||||
}
|
||||
};
|
||||
|
||||
using Traits = vtkm::cont::detail::BitFieldTraits;
|
||||
using WordTypes = vtkm::AtomicTypesSupported;
|
||||
|
||||
VTKM_EXEC_CONT
|
||||
bool RandomBitFromIndex(vtkm::Id idx) noexcept
|
||||
{
|
||||
// Some random operations that will give a pseudorandom stream of bits:
|
||||
auto m = idx + (idx * 2) - (idx / 3) + (idx * 5 / 7) - (idx * 11 / 13);
|
||||
return (m % 2) == 1;
|
||||
}
|
||||
|
||||
template <typename WordType>
|
||||
VTKM_EXEC_CONT WordType RandomWordFromIndex(vtkm::Id idx) noexcept
|
||||
{
|
||||
vtkm::UInt64 m = static_cast<vtkm::UInt64>(idx * (NUM_BITS - 1) + (idx + 1) * NUM_BITS);
|
||||
m ^= m << 3;
|
||||
m ^= m << 7;
|
||||
m ^= m << 15;
|
||||
m ^= m << 31;
|
||||
m = (m << 32) | (m >> 32);
|
||||
|
||||
const size_t mBits = 64;
|
||||
const size_t wordBits = sizeof(WordType) * CHAR_BIT;
|
||||
|
||||
const WordType highWord = static_cast<WordType>(m >> (mBits - wordBits));
|
||||
return highWord;
|
||||
}
|
||||
|
||||
VTKM_CONT
|
||||
static vtkm::cont::BitField RandomBitField(vtkm::Id numBits = NUM_BITS)
|
||||
{
|
||||
vtkm::cont::BitField field;
|
||||
field.Allocate(numBits);
|
||||
auto portal = field.WritePortal();
|
||||
for (vtkm::Id i = 0; i < numBits; ++i)
|
||||
{
|
||||
portal.SetBit(i, RandomBitFromIndex(i));
|
||||
}
|
||||
|
||||
return field;
|
||||
}
|
||||
|
||||
VTKM_CONT
|
||||
void TestBlockAllocation()
|
||||
{
|
||||
vtkm::cont::BitField field;
|
||||
field.Allocate(NUM_BITS);
|
||||
|
||||
// NumBits should be rounded up to the nearest block of bytes, as defined in
|
||||
// the traits:
|
||||
const vtkm::BufferSizeType bytesInFieldData = field.GetBuffer().GetNumberOfBytes();
|
||||
|
||||
const vtkm::BufferSizeType blockSize = vtkm::cont::detail::BitFieldTraits::BlockSize;
|
||||
const vtkm::BufferSizeType numBytes = (NUM_BITS + CHAR_BIT - 1) / CHAR_BIT;
|
||||
const vtkm::BufferSizeType numBlocks = (numBytes + blockSize - 1) / blockSize;
|
||||
const vtkm::BufferSizeType expectedBytes = numBlocks * blockSize;
|
||||
|
||||
VTKM_TEST_ASSERT(bytesInFieldData == expectedBytes,
|
||||
"The BitField allocation does not round up to the nearest "
|
||||
"block. This can cause access-by-word to read/write invalid "
|
||||
"memory.");
|
||||
}
|
||||
|
||||
template <typename PortalType>
|
||||
VTKM_EXEC_CONT bool TestBitValue(const char* operation,
|
||||
vtkm::Id i,
|
||||
PortalType portal,
|
||||
bool& bit,
|
||||
bool originalBit)
|
||||
{
|
||||
auto expected = bit;
|
||||
auto result = portal.GetBitAtomic(i);
|
||||
DEVICE_ASSERT_MSG(result == expected, operation);
|
||||
|
||||
// Reset
|
||||
bit = originalBit;
|
||||
portal.SetBitAtomic(i, bit);
|
||||
return true;
|
||||
}
|
||||
|
||||
template <typename PortalType>
|
||||
VTKM_EXEC_CONT bool HelpTestBit(vtkm::Id i, PortalType portal)
|
||||
{
|
||||
const auto origBit = RandomBitFromIndex(i);
|
||||
auto bit = origBit;
|
||||
|
||||
const auto mod = RandomBitFromIndex(i + NUM_BITS);
|
||||
|
||||
bit = mod;
|
||||
portal.SetBitAtomic(i, mod);
|
||||
DEVICE_ASSERT(TestBitValue("SetBitAtomic", i, portal, bit, origBit));
|
||||
|
||||
bit = !bit;
|
||||
portal.NotBitAtomic(i);
|
||||
DEVICE_ASSERT(TestBitValue("NotBitAtomic", i, portal, bit, origBit));
|
||||
|
||||
bit = bit && mod;
|
||||
portal.AndBitAtomic(i, mod);
|
||||
DEVICE_ASSERT(TestBitValue("AndBitAtomic", i, portal, bit, origBit));
|
||||
|
||||
bit = bit || mod;
|
||||
portal.OrBitAtomic(i, mod);
|
||||
DEVICE_ASSERT(TestBitValue("OrBitAtomic", i, portal, bit, origBit));
|
||||
|
||||
bit = bit != mod;
|
||||
portal.XorBitAtomic(i, mod);
|
||||
DEVICE_ASSERT(TestBitValue("XorBitAtomic", i, portal, bit, origBit));
|
||||
|
||||
const auto notBit = !bit;
|
||||
// A compare-exchange that should fail
|
||||
auto expectedBit = notBit;
|
||||
bool cxResult = portal.CompareExchangeBitAtomic(i, &expectedBit, bit);
|
||||
DEVICE_ASSERT(!cxResult);
|
||||
DEVICE_ASSERT(expectedBit != notBit);
|
||||
DEVICE_ASSERT(portal.GetBit(i) == expectedBit);
|
||||
DEVICE_ASSERT(portal.GetBit(i) == bit);
|
||||
|
||||
// A compare-exchange that should succeed.
|
||||
expectedBit = bit;
|
||||
cxResult = portal.CompareExchangeBitAtomic(i, &expectedBit, notBit);
|
||||
DEVICE_ASSERT(cxResult);
|
||||
DEVICE_ASSERT(expectedBit == bit);
|
||||
DEVICE_ASSERT(portal.GetBit(i) == notBit);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
template <typename WordType, typename PortalType>
|
||||
VTKM_EXEC_CONT bool TestWordValue(const char* operation,
|
||||
vtkm::Id i,
|
||||
const PortalType& portal,
|
||||
WordType& word,
|
||||
WordType originalWord)
|
||||
{
|
||||
auto expected = word;
|
||||
auto result = portal.template GetWordAtomic<WordType>(i);
|
||||
DEVICE_ASSERT_MSG(result == expected, operation);
|
||||
|
||||
// Reset
|
||||
word = originalWord;
|
||||
portal.SetWordAtomic(i, word);
|
||||
return true;
|
||||
}
|
||||
|
||||
template <typename WordType, typename PortalType>
|
||||
VTKM_EXEC_CONT bool HelpTestWord(vtkm::Id i, PortalType portal)
|
||||
{
|
||||
const auto origWord = RandomWordFromIndex<WordType>(i);
|
||||
auto word = origWord;
|
||||
|
||||
const auto mod = RandomWordFromIndex<WordType>(i + NUM_BITS);
|
||||
|
||||
portal.SetWord(i, word);
|
||||
DEVICE_ASSERT(TestWordValue("SetWord", i, portal, word, origWord));
|
||||
|
||||
word = mod;
|
||||
portal.SetWordAtomic(i, mod);
|
||||
DEVICE_ASSERT(TestWordValue("SetWordAtomic", i, portal, word, origWord));
|
||||
|
||||
// C++ promotes e.g. uint8 to int32 when performing bitwise not. Silence
|
||||
// conversion warning and mask unimportant bits:
|
||||
word = static_cast<WordType>(~word);
|
||||
portal.template NotWordAtomic<WordType>(i);
|
||||
DEVICE_ASSERT(TestWordValue("NotWordAtomic", i, portal, word, origWord));
|
||||
|
||||
word = word & mod;
|
||||
portal.AndWordAtomic(i, mod);
|
||||
DEVICE_ASSERT(TestWordValue("AndWordAtomic", i, portal, word, origWord));
|
||||
|
||||
word = word | mod;
|
||||
portal.OrWordAtomic(i, mod);
|
||||
DEVICE_ASSERT(TestWordValue("OrWordAtomic", i, portal, word, origWord));
|
||||
|
||||
word = word ^ mod;
|
||||
portal.XorWordAtomic(i, mod);
|
||||
DEVICE_ASSERT(TestWordValue("XorWordAtomic", i, portal, word, origWord));
|
||||
|
||||
// Compare-exchange that should fail
|
||||
const WordType notWord = static_cast<WordType>(~word);
|
||||
WordType expectedWord = notWord;
|
||||
bool cxResult = portal.CompareExchangeWordAtomic(i, &expectedWord, word);
|
||||
DEVICE_ASSERT(!cxResult);
|
||||
DEVICE_ASSERT(expectedWord != notWord);
|
||||
DEVICE_ASSERT(portal.template GetWord<WordType>(i) == expectedWord);
|
||||
DEVICE_ASSERT(portal.template GetWord<WordType>(i) == word);
|
||||
|
||||
// Compare-exchange that should succeed
|
||||
expectedWord = word;
|
||||
cxResult = portal.CompareExchangeWordAtomic(i, &expectedWord, notWord);
|
||||
DEVICE_ASSERT(cxResult);
|
||||
DEVICE_ASSERT(expectedWord == word);
|
||||
DEVICE_ASSERT(portal.template GetWord<WordType>(i) == notWord);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
template <typename PortalType>
|
||||
struct HelpTestWordOpsControl
|
||||
{
|
||||
PortalType Portal;
|
||||
|
||||
VTKM_CONT
|
||||
HelpTestWordOpsControl(PortalType portal)
|
||||
: Portal(portal)
|
||||
{
|
||||
}
|
||||
|
||||
template <typename WordType>
|
||||
VTKM_CONT void operator()(WordType)
|
||||
{
|
||||
const auto numWords = this->Portal.template GetNumberOfWords<WordType>();
|
||||
for (vtkm::Id i = 0; i < numWords; ++i)
|
||||
{
|
||||
VTKM_TEST_ASSERT(HelpTestWord<WordType>(i, this->Portal));
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
template <typename Portal>
|
||||
VTKM_CONT void HelpTestPortalsControl(Portal portal)
|
||||
{
|
||||
const auto numWords8 = (NUM_BITS + 7) / 8;
|
||||
const auto numWords16 = (NUM_BITS + 15) / 16;
|
||||
const auto numWords32 = (NUM_BITS + 31) / 32;
|
||||
const auto numWords64 = (NUM_BITS + 63) / 64;
|
||||
|
||||
VTKM_TEST_ASSERT(portal.GetNumberOfBits() == NUM_BITS);
|
||||
VTKM_TEST_ASSERT(portal.template GetNumberOfWords<vtkm::UInt8>() == numWords8);
|
||||
VTKM_TEST_ASSERT(portal.template GetNumberOfWords<vtkm::UInt16>() == numWords16);
|
||||
VTKM_TEST_ASSERT(portal.template GetNumberOfWords<vtkm::UInt32>() == numWords32);
|
||||
VTKM_TEST_ASSERT(portal.template GetNumberOfWords<vtkm::UInt64>() == numWords64);
|
||||
|
||||
for (vtkm::Id i = 0; i < NUM_BITS; ++i)
|
||||
{
|
||||
HelpTestBit(i, portal);
|
||||
}
|
||||
|
||||
HelpTestWordOpsControl<Portal> test(portal);
|
||||
vtkm::ListForEach(test, vtkm::AtomicTypesSupported{});
|
||||
}
|
||||
|
||||
VTKM_CONT
|
||||
void TestControlPortals()
|
||||
{
|
||||
auto field = RandomBitField();
|
||||
|
||||
HelpTestPortalsControl(field.WritePortal());
|
||||
}
|
||||
|
||||
template <typename Portal>
|
||||
VTKM_EXEC_CONT bool HelpTestPortalSanityExecution(Portal portal)
|
||||
{
|
||||
const auto numWords8 = (NUM_BITS + 7) / 8;
|
||||
const auto numWords16 = (NUM_BITS + 15) / 16;
|
||||
const auto numWords32 = (NUM_BITS + 31) / 32;
|
||||
const auto numWords64 = (NUM_BITS + 63) / 64;
|
||||
|
||||
DEVICE_ASSERT(portal.GetNumberOfBits() == NUM_BITS);
|
||||
DEVICE_ASSERT(portal.template GetNumberOfWords<vtkm::UInt8>() == numWords8);
|
||||
DEVICE_ASSERT(portal.template GetNumberOfWords<vtkm::UInt16>() == numWords16);
|
||||
DEVICE_ASSERT(portal.template GetNumberOfWords<vtkm::UInt32>() == numWords32);
|
||||
DEVICE_ASSERT(portal.template GetNumberOfWords<vtkm::UInt64>() == numWords64);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
template <typename WordType, typename PortalType>
|
||||
struct HelpTestPortalsExecutionWordsFunctor : vtkm::exec::FunctorBase
|
||||
{
|
||||
PortalType Portal;
|
||||
|
||||
HelpTestPortalsExecutionWordsFunctor(PortalType portal)
|
||||
: Portal(portal)
|
||||
{
|
||||
}
|
||||
|
||||
VTKM_EXEC_CONT
|
||||
void operator()(vtkm::Id i) const
|
||||
{
|
||||
if (i == 0)
|
||||
{
|
||||
if (!HelpTestPortalSanityExecution(this->Portal))
|
||||
{
|
||||
this->RaiseError("Testing Portal sanity failed.");
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
if (!HelpTestWord<WordType>(i, this->Portal))
|
||||
{
|
||||
this->RaiseError("Testing word operations failed.");
|
||||
return;
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
template <typename PortalType>
|
||||
struct HelpTestPortalsExecutionBitsFunctor : vtkm::exec::FunctorBase
|
||||
{
|
||||
PortalType Portal;
|
||||
|
||||
HelpTestPortalsExecutionBitsFunctor(PortalType portal)
|
||||
: Portal(portal)
|
||||
{
|
||||
}
|
||||
|
||||
VTKM_EXEC_CONT
|
||||
void operator()(vtkm::Id i) const
|
||||
{
|
||||
if (!HelpTestBit(i, this->Portal))
|
||||
{
|
||||
this->RaiseError("Testing bit operations failed.");
|
||||
return;
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
template <typename PortalType, typename Device>
|
||||
struct HelpTestWordOpsExecution
|
||||
{
|
||||
PortalType Portal;
|
||||
|
||||
VTKM_CONT
|
||||
HelpTestWordOpsExecution(PortalType portal)
|
||||
: Portal(portal)
|
||||
{
|
||||
}
|
||||
|
||||
template <typename WordType>
|
||||
VTKM_CONT void operator()(WordType)
|
||||
{
|
||||
const auto numWords = this->Portal.template GetNumberOfWords<WordType>();
|
||||
|
||||
using WordFunctor = HelpTestPortalsExecutionWordsFunctor<WordType, PortalType>;
|
||||
WordFunctor test{ this->Portal };
|
||||
vtkm::cont::DeviceAdapterAlgorithm<Device>::Schedule(test, numWords);
|
||||
}
|
||||
};
|
||||
|
||||
template <typename Portal, typename Device>
|
||||
VTKM_CONT void HelpTestPortalsExecution(Portal portal, Device)
|
||||
{
|
||||
HelpTestPortalsExecutionBitsFunctor<Portal> bitTest{ portal };
|
||||
vtkm::cont::DeviceAdapterAlgorithm<Device>::Schedule(bitTest, portal.GetNumberOfBits());
|
||||
|
||||
|
||||
HelpTestWordOpsExecution<Portal, Device> test(portal);
|
||||
vtkm::ListForEach(test, vtkm::AtomicTypesSupported{});
|
||||
}
|
||||
|
||||
VTKM_CONT
|
||||
void TestExecutionPortals()
|
||||
{
|
||||
vtkm::cont::BitField field = RandomBitField();
|
||||
|
||||
vtkm::cont::TryExecute([&](auto device) {
|
||||
vtkm::cont::Token token;
|
||||
HelpTestPortalsExecution(field.PrepareForInPlace(device, token), device);
|
||||
return true;
|
||||
});
|
||||
}
|
||||
|
||||
VTKM_CONT
|
||||
void TestFinalWordMask()
|
||||
{
|
||||
auto testMask32 = [](vtkm::Id numBits, vtkm::UInt32 expectedMask) {
|
||||
vtkm::cont::BitField field;
|
||||
field.Allocate(numBits);
|
||||
auto mask = field.ReadPortal().GetFinalWordMask<vtkm::UInt32>();
|
||||
|
||||
VTKM_TEST_ASSERT(expectedMask == mask,
|
||||
"Unexpected mask for BitField size ",
|
||||
numBits,
|
||||
": Expected 0x",
|
||||
std::hex,
|
||||
expectedMask,
|
||||
" got 0x",
|
||||
mask);
|
||||
};
|
||||
|
||||
auto testMask64 = [](vtkm::Id numBits, vtkm::UInt64 expectedMask) {
|
||||
vtkm::cont::BitField field;
|
||||
field.Allocate(numBits);
|
||||
auto mask = field.ReadPortal().GetFinalWordMask<vtkm::UInt64>();
|
||||
|
||||
VTKM_TEST_ASSERT(expectedMask == mask,
|
||||
"Unexpected mask for BitField size ",
|
||||
numBits,
|
||||
": Expected 0x",
|
||||
std::hex,
|
||||
expectedMask,
|
||||
" got 0x",
|
||||
mask);
|
||||
};
|
||||
|
||||
testMask32(0, 0x00000000);
|
||||
testMask32(1, 0x00000001);
|
||||
testMask32(2, 0x00000003);
|
||||
testMask32(3, 0x00000007);
|
||||
testMask32(4, 0x0000000f);
|
||||
testMask32(5, 0x0000001f);
|
||||
testMask32(8, 0x000000ff);
|
||||
testMask32(16, 0x0000ffff);
|
||||
testMask32(24, 0x00ffffff);
|
||||
testMask32(25, 0x01ffffff);
|
||||
testMask32(31, 0x7fffffff);
|
||||
testMask32(32, 0xffffffff);
|
||||
testMask32(64, 0xffffffff);
|
||||
testMask32(128, 0xffffffff);
|
||||
testMask32(129, 0x00000001);
|
||||
|
||||
testMask64(0, 0x0000000000000000);
|
||||
testMask64(1, 0x0000000000000001);
|
||||
testMask64(2, 0x0000000000000003);
|
||||
testMask64(3, 0x0000000000000007);
|
||||
testMask64(4, 0x000000000000000f);
|
||||
testMask64(5, 0x000000000000001f);
|
||||
testMask64(8, 0x00000000000000ff);
|
||||
testMask64(16, 0x000000000000ffff);
|
||||
testMask64(24, 0x0000000000ffffff);
|
||||
testMask64(25, 0x0000000001ffffff);
|
||||
testMask64(31, 0x000000007fffffff);
|
||||
testMask64(32, 0x00000000ffffffff);
|
||||
testMask64(40, 0x000000ffffffffff);
|
||||
testMask64(48, 0x0000ffffffffffff);
|
||||
testMask64(56, 0x00ffffffffffffff);
|
||||
testMask64(64, 0xffffffffffffffff);
|
||||
testMask64(128, 0xffffffffffffffff);
|
||||
testMask64(129, 0x0000000000000001);
|
||||
}
|
||||
|
||||
VTKM_CONT void TestFill()
|
||||
{
|
||||
vtkm::cont::BitField bitField;
|
||||
bitField.Allocate(NUM_BITS);
|
||||
|
||||
bitField.Fill(true);
|
||||
{
|
||||
auto portal = bitField.ReadPortal();
|
||||
for (vtkm::Id index = 0; index < NUM_BITS; ++index)
|
||||
{
|
||||
VTKM_TEST_ASSERT(portal.GetBit(index));
|
||||
}
|
||||
}
|
||||
|
||||
constexpr vtkm::UInt8 word8 = 0xA6;
|
||||
bitField.Fill(word8);
|
||||
{
|
||||
auto portal = bitField.ReadPortal();
|
||||
for (vtkm::Id index = 0; index < NUM_BITS; ++index)
|
||||
{
|
||||
VTKM_TEST_ASSERT(portal.GetBit(index) == ((word8 >> (index % 8)) & 0x01));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
struct ArrayHandleBitFieldChecker : vtkm::worklet::WorkletMapField
|
||||
{
|
||||
using ControlSignature = void(FieldInOut);
|
||||
using ExecutionSignature = void(_1, WorkIndex);
|
||||
|
||||
bool InvertReference;
|
||||
|
||||
VTKM_CONT
|
||||
ArrayHandleBitFieldChecker(bool invert)
|
||||
: InvertReference(invert)
|
||||
{
|
||||
}
|
||||
|
||||
VTKM_EXEC
|
||||
void operator()(bool& bit, vtkm::Id i) const
|
||||
{
|
||||
const bool ref = this->InvertReference ? !RandomBitFromIndex(i) : RandomBitFromIndex(i);
|
||||
if (bit != ref)
|
||||
{
|
||||
this->RaiseError("Unexpected value from ArrayHandleBitField portal.");
|
||||
return;
|
||||
}
|
||||
|
||||
// Flip the bit for the next kernel launch, which tests that the bitfield
|
||||
// is inverted.
|
||||
bit = !ref;
|
||||
}
|
||||
};
|
||||
|
||||
VTKM_CONT
|
||||
void TestArrayHandleBitField()
|
||||
{
|
||||
vtkm::cont::Invoker invoke;
|
||||
|
||||
auto handle = vtkm::cont::make_ArrayHandleBitField(RandomBitField());
|
||||
const vtkm::Id numBits = handle.GetNumberOfValues();
|
||||
|
||||
VTKM_TEST_ASSERT(numBits == NUM_BITS,
|
||||
"ArrayHandleBitField returned the wrong number of values. "
|
||||
"Expected: ",
|
||||
NUM_BITS,
|
||||
" got: ",
|
||||
numBits);
|
||||
|
||||
invoke(ArrayHandleBitFieldChecker{ false }, handle);
|
||||
invoke(ArrayHandleBitFieldChecker{ true }, handle);
|
||||
|
||||
handle.Fill(true);
|
||||
{
|
||||
auto portal = handle.ReadPortal();
|
||||
for (vtkm::Id index = 0; index < NUM_BITS; ++index)
|
||||
{
|
||||
VTKM_TEST_ASSERT(portal.Get(index));
|
||||
}
|
||||
}
|
||||
|
||||
handle.Fill(false, 24);
|
||||
handle.Fill(true, 64);
|
||||
{
|
||||
auto portal = handle.ReadPortal();
|
||||
for (vtkm::Id index = 0; index < NUM_BITS; ++index)
|
||||
{
|
||||
VTKM_TEST_ASSERT(portal.Get(index) == ((index < 24) || (index >= 64)));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
VTKM_CONT
|
||||
void TestArrayInvokeWorklet()
|
||||
{
|
||||
auto condArray = vtkm::cont::make_ArrayHandleBitField(RandomBitField());
|
||||
auto trueArray = vtkm::cont::make_ArrayHandleCounting<vtkm::Id>(20, 2, NUM_BITS);
|
||||
auto falseArray = vtkm::cont::make_ArrayHandleCounting<vtkm::Id>(13, 2, NUM_BITS);
|
||||
vtkm::cont::ArrayHandle<vtkm::Id> output;
|
||||
|
||||
vtkm::cont::Invoker invoke;
|
||||
invoke(ConditionalMergeWorklet{}, condArray, trueArray, falseArray, output);
|
||||
|
||||
auto condVals = condArray.ReadPortal();
|
||||
auto trueVals = trueArray.ReadPortal();
|
||||
auto falseVals = falseArray.ReadPortal();
|
||||
auto outVals = output.ReadPortal();
|
||||
|
||||
VTKM_TEST_ASSERT(condVals.GetNumberOfValues() == trueVals.GetNumberOfValues());
|
||||
VTKM_TEST_ASSERT(condVals.GetNumberOfValues() == falseVals.GetNumberOfValues());
|
||||
VTKM_TEST_ASSERT(condVals.GetNumberOfValues() == outVals.GetNumberOfValues());
|
||||
|
||||
for (vtkm::Id i = 0; i < condVals.GetNumberOfValues(); ++i)
|
||||
{
|
||||
VTKM_TEST_ASSERT(outVals.Get(i) == (condVals.Get(i) ? trueVals.Get(i) : falseVals.Get(i)));
|
||||
}
|
||||
}
|
||||
|
||||
VTKM_CONT
|
||||
static void TestArrayInvokeWorklet2()
|
||||
{
|
||||
auto condBits = RandomBitField();
|
||||
auto trueArray = vtkm::cont::make_ArrayHandleCounting<vtkm::Id>(20, 2, NUM_BITS);
|
||||
auto falseArray = vtkm::cont::make_ArrayHandleCounting<vtkm::Id>(13, 2, NUM_BITS);
|
||||
vtkm::cont::ArrayHandle<vtkm::Id> output;
|
||||
|
||||
vtkm::cont::Invoker invoke;
|
||||
invoke(ConditionalMergeWorklet2{}, condBits, trueArray, falseArray, output);
|
||||
|
||||
auto condVals = condBits.ReadPortal();
|
||||
auto trueVals = trueArray.ReadPortal();
|
||||
auto falseVals = falseArray.ReadPortal();
|
||||
auto outVals = output.ReadPortal();
|
||||
|
||||
VTKM_TEST_ASSERT(condVals.GetNumberOfBits() == trueVals.GetNumberOfValues());
|
||||
VTKM_TEST_ASSERT(condVals.GetNumberOfBits() == falseVals.GetNumberOfValues());
|
||||
VTKM_TEST_ASSERT(condVals.GetNumberOfBits() == outVals.GetNumberOfValues());
|
||||
|
||||
for (vtkm::Id i = 0; i < condVals.GetNumberOfBits(); ++i)
|
||||
{
|
||||
// The worklet flips the bitfield in place after choosing true/false paths
|
||||
VTKM_TEST_ASSERT(condVals.GetBit(i) == !RandomBitFromIndex(i));
|
||||
VTKM_TEST_ASSERT(outVals.Get(i) == (!condVals.GetBit(i) ? trueVals.Get(i) : falseVals.Get(i)));
|
||||
}
|
||||
}
|
||||
|
||||
VTKM_CONT void Run()
|
||||
{
|
||||
TestBlockAllocation();
|
||||
TestControlPortals();
|
||||
TestExecutionPortals();
|
||||
TestFinalWordMask();
|
||||
TestFill();
|
||||
TestArrayHandleBitField();
|
||||
TestArrayInvokeWorklet();
|
||||
TestArrayInvokeWorklet2();
|
||||
}
|
||||
|
||||
} // anonymous namespace
|
||||
|
||||
int UnitTestBitField(int argc, char* argv[])
|
||||
{
|
||||
return vtkm::cont::testing::Testing::Run(Run, argc, argv);
|
||||
}
|
Loading…
Reference in New Issue
Block a user