vtk-m/vtkm/cont/testing/UnitTestArrayHandleExtractComponent.cxx
Kenneth Moreland c55d15f397 Deprecate ArrayHandle::ExecutionTypes
The newer version of `ArrayHandle` no longer supports different types of
portals for different devices. Thus, the `ReadPortalType` and
`WritePortalType` are sufficient for all types of portals across all
devices.

This significantly simplifies supporting execution objects on devices,
and thus this change also includes many changes to various execution
objects to remove their dependence on the device adapter tag.
2021-02-08 12:17:37 -07:00

236 lines
7.8 KiB
C++

//============================================================================
// 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/ArrayHandle.h>
#include <vtkm/cont/ArrayHandleCompositeVector.h>
#include <vtkm/cont/ArrayHandleCounting.h>
#include <vtkm/cont/ArrayHandleExtractComponent.h>
#include <vtkm/cont/DeviceAdapter.h>
#include <vtkm/cont/DeviceAdapterAlgorithm.h>
#include <vtkm/cont/testing/Testing.h>
namespace
{
template <typename ValueType>
struct ExtractComponentTests
{
using InputArray = vtkm::cont::ArrayHandle<vtkm::Vec<ValueType, 4>>;
using ExtractArray = vtkm::cont::ArrayHandleExtractComponent<InputArray>;
using ReferenceComponentArray = vtkm::cont::ArrayHandleCounting<ValueType>;
using ReferenceCompositeArray =
typename vtkm::cont::ArrayHandleCompositeVector<ReferenceComponentArray,
ReferenceComponentArray,
ReferenceComponentArray,
ReferenceComponentArray>;
using DeviceTag = vtkm::cont::DeviceAdapterTagSerial;
using Algo = vtkm::cont::DeviceAdapterAlgorithm<DeviceTag>;
// This is used to build a ArrayHandleExtractComponent's internal array.
ReferenceCompositeArray RefComposite;
void ConstructReferenceArray()
{
// Build the Ref array
const vtkm::Id numValues = 32;
ReferenceComponentArray c1 = vtkm::cont::make_ArrayHandleCounting<ValueType>(3, 2, numValues);
ReferenceComponentArray c2 = vtkm::cont::make_ArrayHandleCounting<ValueType>(2, 3, numValues);
ReferenceComponentArray c3 = vtkm::cont::make_ArrayHandleCounting<ValueType>(4, 4, numValues);
ReferenceComponentArray c4 = vtkm::cont::make_ArrayHandleCounting<ValueType>(1, 3, numValues);
this->RefComposite = vtkm::cont::make_ArrayHandleCompositeVector(c1, c2, c3, c4);
}
InputArray BuildInputArray() const
{
InputArray result;
Algo::Copy(this->RefComposite, result);
return result;
}
void SanityCheck(vtkm::IdComponent component) const
{
InputArray composite = this->BuildInputArray();
ExtractArray extract(composite, component);
VTKM_TEST_ASSERT(composite.GetNumberOfValues() == extract.GetNumberOfValues(),
"Number of values in copied ExtractComponent array does not match input.");
}
void ReadTestComponentExtraction(vtkm::IdComponent component) const
{
// Test that the expected values are read from an ExtractComponent array.
InputArray composite = this->BuildInputArray();
ExtractArray extract(composite, component);
// Test reading the data back in the control env:
this->ValidateReadTestArray(extract, component);
// Copy the extract array in the execution environment to test reading:
vtkm::cont::ArrayHandle<ValueType> execCopy;
Algo::Copy(extract, execCopy);
this->ValidateReadTestArray(execCopy, component);
}
template <typename ArrayHandleType>
void ValidateReadTestArray(ArrayHandleType testArray, vtkm::IdComponent component) const
{
using RefVectorType = typename ReferenceCompositeArray::ValueType;
using Traits = vtkm::VecTraits<RefVectorType>;
auto testPortal = testArray.ReadPortal();
auto refPortal = this->RefComposite.ReadPortal();
VTKM_TEST_ASSERT(testPortal.GetNumberOfValues() == refPortal.GetNumberOfValues(),
"Number of values in read test output do not match input.");
for (vtkm::Id i = 0; i < testPortal.GetNumberOfValues(); ++i)
{
VTKM_TEST_ASSERT(
test_equal(testPortal.Get(i), Traits::GetComponent(refPortal.Get(i), component), 0.),
"Value mismatch in read test.");
}
}
// Doubles the specified component (reading from RefVectorType).
template <typename PortalType, typename RefPortalType>
struct WriteTestFunctor : vtkm::exec::FunctorBase
{
using RefVectorType = typename RefPortalType::ValueType;
using Traits = vtkm::VecTraits<RefVectorType>;
PortalType Portal;
RefPortalType RefPortal;
vtkm::IdComponent Component;
VTKM_CONT
WriteTestFunctor(const PortalType& portal,
const RefPortalType& ref,
vtkm::IdComponent component)
: Portal(portal)
, RefPortal(ref)
, Component(component)
{
}
VTKM_EXEC_CONT
void operator()(vtkm::Id index) const
{
this->Portal.Set(index,
Traits::GetComponent(this->RefPortal.Get(index), this->Component) * 2);
}
};
void WriteTestComponentExtraction(vtkm::IdComponent component) const
{
// Control test:
{
InputArray composite = this->BuildInputArray();
ExtractArray extract(composite, component);
{
WriteTestFunctor<typename ExtractArray::WritePortalType,
typename ReferenceCompositeArray::ReadPortalType>
functor(extract.WritePortal(), this->RefComposite.ReadPortal(), component);
for (vtkm::Id i = 0; i < extract.GetNumberOfValues(); ++i)
{
functor(i);
}
}
this->ValidateWriteTestArray(composite, component);
}
// Exec test:
{
InputArray composite = this->BuildInputArray();
ExtractArray extract(composite, component);
using Portal = typename ExtractArray::WritePortalType;
using RefPortal = typename ReferenceCompositeArray::ReadPortalType;
{
vtkm::cont::Token token;
WriteTestFunctor<Portal, RefPortal> functor(
extract.PrepareForInPlace(DeviceTag(), token),
this->RefComposite.PrepareForInput(DeviceTag(), token),
component);
Algo::Schedule(functor, extract.GetNumberOfValues());
}
this->ValidateWriteTestArray(composite, component);
}
}
void ValidateWriteTestArray(InputArray testArray, vtkm::IdComponent component) const
{
using VectorType = typename ReferenceCompositeArray::ValueType;
using Traits = vtkm::VecTraits<VectorType>;
// Check that the indicated component is twice the reference value.
auto refPortal = this->RefComposite.ReadPortal();
auto portal = testArray.ReadPortal();
VTKM_TEST_ASSERT(portal.GetNumberOfValues() == refPortal.GetNumberOfValues(),
"Number of values in write test output do not match input.");
for (vtkm::Id i = 0; i < portal.GetNumberOfValues(); ++i)
{
auto value = portal.Get(i);
auto refValue = refPortal.Get(i);
Traits::SetComponent(refValue, component, Traits::GetComponent(refValue, component) * 2);
VTKM_TEST_ASSERT(test_equal(refValue, value, 0.), "Value mismatch in write test.");
}
}
void TestComponent(vtkm::IdComponent component) const
{
this->SanityCheck(component);
this->ReadTestComponentExtraction(component);
this->WriteTestComponentExtraction(component);
}
void operator()()
{
this->ConstructReferenceArray();
this->TestComponent(0);
this->TestComponent(1);
this->TestComponent(2);
this->TestComponent(3);
}
};
struct ArgToTemplateType
{
template <typename ValueType>
void operator()(ValueType) const
{
ExtractComponentTests<ValueType>()();
}
};
void TestArrayHandleExtractComponent()
{
using TestTypes = vtkm::List<vtkm::Int32, vtkm::Int64, vtkm::Float32, vtkm::Float64>;
vtkm::testing::Testing::TryTypes(ArgToTemplateType(), TestTypes());
}
} // end anon namespace
int UnitTestArrayHandleExtractComponent(int argc, char* argv[])
{
return vtkm::cont::testing::Testing::Run(TestArrayHandleExtractComponent, argc, argv);
}