mirror of
https://gitlab.kitware.com/vtk/vtk-m
synced 2024-10-08 11:29:02 +00:00
e0e24090a4
The condition was completely backwards so no file could ever be loaded.
1054 lines
32 KiB
C++
1054 lines
32 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 "Benchmarker.h"
|
|
|
|
#include <vtkm/Math.h>
|
|
#include <vtkm/Range.h>
|
|
#include <vtkm/VecTraits.h>
|
|
|
|
#include <vtkm/cont/ArrayGetValues.h>
|
|
#include <vtkm/cont/ArrayHandle.h>
|
|
#include <vtkm/cont/ArrayHandleUniformPointCoordinates.h>
|
|
#include <vtkm/cont/CellSetExplicit.h>
|
|
#include <vtkm/cont/CellSetSingleType.h>
|
|
#include <vtkm/cont/CellSetStructured.h>
|
|
#include <vtkm/cont/DataSet.h>
|
|
#include <vtkm/cont/ErrorInternal.h>
|
|
#include <vtkm/cont/Logging.h>
|
|
#include <vtkm/cont/RuntimeDeviceTracker.h>
|
|
#include <vtkm/cont/Timer.h>
|
|
#include <vtkm/cont/testing/MakeTestDataSet.h>
|
|
|
|
#include <vtkm/cont/internal/OptionParser.h>
|
|
|
|
#include <vtkm/filter/CellAverage.h>
|
|
#include <vtkm/filter/Contour.h>
|
|
#include <vtkm/filter/ExternalFaces.h>
|
|
#include <vtkm/filter/FieldSelection.h>
|
|
#include <vtkm/filter/Gradient.h>
|
|
#include <vtkm/filter/PointAverage.h>
|
|
#include <vtkm/filter/PolicyBase.h>
|
|
#include <vtkm/filter/Tetrahedralize.h>
|
|
#include <vtkm/filter/Threshold.h>
|
|
#include <vtkm/filter/ThresholdPoints.h>
|
|
#include <vtkm/filter/Triangulate.h>
|
|
#include <vtkm/filter/VectorMagnitude.h>
|
|
#include <vtkm/filter/VertexClustering.h>
|
|
#include <vtkm/filter/WarpScalar.h>
|
|
#include <vtkm/filter/WarpVector.h>
|
|
|
|
#include <vtkm/io/VTKDataSetReader.h>
|
|
|
|
#include <vtkm/source/Wavelet.h>
|
|
#include <vtkm/worklet/DispatcherMapField.h>
|
|
#include <vtkm/worklet/WorkletMapField.h>
|
|
|
|
#include <cctype> // for std::tolower
|
|
#include <sstream>
|
|
#include <type_traits>
|
|
|
|
#ifdef VTKM_ENABLE_TBB
|
|
#include <tbb/task_scheduler_init.h>
|
|
#endif
|
|
#ifdef VTKM_ENABLE_OPENMP
|
|
#include <omp.h>
|
|
#endif
|
|
|
|
// A specific vtk dataset can be used during benchmarking using the
|
|
// "Filename [filename]" argument.
|
|
|
|
// Otherwise a wavelet dataset will be used. The size of the wavelet can be
|
|
// specified via "WaveletDim [integer]" argument. The default is 256, resulting
|
|
// in a 256x256x256 (cell extent) dataset.
|
|
|
|
// Passing in the "Tetra" option will pass the input dataset through the
|
|
// Tetrahedralize filter to generate an unstructured, single cell type dataset.
|
|
|
|
// For the filters that require fields, the desired fields may be specified
|
|
// using these arguments:
|
|
//
|
|
// PointScalars [fieldname]
|
|
// CellScalars [fieldname]
|
|
// PointVectors [fieldname]
|
|
//
|
|
// If the fields are not specified, the first field with the correct association
|
|
// is used. If no such field exists, one will be generated from the data.
|
|
|
|
// For the TBB/OpenMP implementations, the number of threads can be customized
|
|
// using a "NumThreads [numThreads]" argument.
|
|
|
|
namespace
|
|
{
|
|
|
|
// Hold configuration state (e.g. active device):
|
|
vtkm::cont::InitializeResult Config;
|
|
|
|
// The input dataset we'll use on the filters:
|
|
static vtkm::cont::DataSet InputDataSet;
|
|
static vtkm::cont::DataSet UnstructuredInputDataSet;
|
|
// The point scalars to use:
|
|
static std::string PointScalarsName;
|
|
// The cell scalars to use:
|
|
static std::string CellScalarsName;
|
|
// The point vectors to use:
|
|
static std::string PointVectorsName;
|
|
// Whether the input is a file or is generated
|
|
bool FileAsInput = false;
|
|
|
|
bool InputIsStructured()
|
|
{
|
|
return InputDataSet.GetCellSet().IsType<vtkm::cont::CellSetStructured<3>>() ||
|
|
InputDataSet.GetCellSet().IsType<vtkm::cont::CellSetStructured<2>>() ||
|
|
InputDataSet.GetCellSet().IsType<vtkm::cont::CellSetStructured<1>>();
|
|
}
|
|
|
|
enum GradOpts : int
|
|
{
|
|
Gradient = 1,
|
|
PointGradient = 1 << 1,
|
|
Divergence = 1 << 2,
|
|
Vorticity = 1 << 3,
|
|
QCriterion = 1 << 4,
|
|
RowOrdering = 1 << 5,
|
|
ScalarInput = 1 << 6
|
|
};
|
|
|
|
void BenchGradient(::benchmark::State& state, int options)
|
|
{
|
|
const vtkm::cont::DeviceAdapterId device = Config.Device;
|
|
|
|
vtkm::filter::Gradient filter;
|
|
|
|
if (options & ScalarInput)
|
|
{
|
|
// Some outputs require vectors:
|
|
if (options & Divergence || options & Vorticity || options & QCriterion)
|
|
{
|
|
throw vtkm::cont::ErrorInternal("A requested gradient output is "
|
|
"incompatible with scalar input.");
|
|
}
|
|
filter.SetActiveField(PointScalarsName, vtkm::cont::Field::Association::POINTS);
|
|
}
|
|
else
|
|
{
|
|
filter.SetActiveField(PointVectorsName, vtkm::cont::Field::Association::POINTS);
|
|
}
|
|
|
|
filter.SetComputeGradient(static_cast<bool>(options & Gradient));
|
|
filter.SetComputePointGradient(static_cast<bool>(options & PointGradient));
|
|
filter.SetComputeDivergence(static_cast<bool>(options & Divergence));
|
|
filter.SetComputeVorticity(static_cast<bool>(options & Vorticity));
|
|
filter.SetComputeQCriterion(static_cast<bool>(options & QCriterion));
|
|
|
|
if (options & RowOrdering)
|
|
{
|
|
filter.SetRowMajorOrdering();
|
|
}
|
|
else
|
|
{
|
|
filter.SetColumnMajorOrdering();
|
|
}
|
|
|
|
vtkm::cont::Timer timer{ device };
|
|
for (auto _ : state)
|
|
{
|
|
(void)_;
|
|
timer.Start();
|
|
auto result = filter.Execute(InputDataSet);
|
|
::benchmark::DoNotOptimize(result);
|
|
timer.Stop();
|
|
|
|
state.SetIterationTime(timer.GetElapsedTime());
|
|
}
|
|
}
|
|
|
|
#define VTKM_PRIVATE_GRADIENT_BENCHMARK(Name, Opts) \
|
|
void BenchGradient##Name(::benchmark::State& state) { BenchGradient(state, Opts); } \
|
|
VTKM_BENCHMARK(BenchGradient##Name)
|
|
|
|
VTKM_PRIVATE_GRADIENT_BENCHMARK(Scalar, Gradient | ScalarInput);
|
|
VTKM_PRIVATE_GRADIENT_BENCHMARK(Vector, Gradient);
|
|
VTKM_PRIVATE_GRADIENT_BENCHMARK(VectorRow, Gradient | RowOrdering);
|
|
VTKM_PRIVATE_GRADIENT_BENCHMARK(Point, PointGradient);
|
|
VTKM_PRIVATE_GRADIENT_BENCHMARK(Divergence, Divergence);
|
|
VTKM_PRIVATE_GRADIENT_BENCHMARK(Vorticity, Vorticity);
|
|
VTKM_PRIVATE_GRADIENT_BENCHMARK(QCriterion, QCriterion);
|
|
VTKM_PRIVATE_GRADIENT_BENCHMARK(All,
|
|
Gradient | PointGradient | Divergence | Vorticity | QCriterion);
|
|
|
|
#undef VTKM_PRIVATE_GRADIENT_BENCHMARK
|
|
|
|
void BenchThreshold(::benchmark::State& state)
|
|
{
|
|
const vtkm::cont::DeviceAdapterId device = Config.Device;
|
|
|
|
// Lookup the point scalar range
|
|
const auto range = []() -> vtkm::Range {
|
|
auto ptScalarField =
|
|
InputDataSet.GetField(PointScalarsName, vtkm::cont::Field::Association::POINTS);
|
|
return vtkm::cont::ArrayGetValue(0, ptScalarField.GetRange());
|
|
}();
|
|
|
|
// Extract points with values between 25-75% of the range
|
|
vtkm::Float64 quarter = range.Length() / 4.;
|
|
vtkm::Float64 mid = range.Center();
|
|
|
|
vtkm::filter::Threshold filter;
|
|
filter.SetActiveField(PointScalarsName, vtkm::cont::Field::Association::POINTS);
|
|
filter.SetLowerThreshold(mid - quarter);
|
|
filter.SetUpperThreshold(mid + quarter);
|
|
|
|
vtkm::cont::Timer timer{ device };
|
|
for (auto _ : state)
|
|
{
|
|
(void)_;
|
|
timer.Start();
|
|
auto result = filter.Execute(InputDataSet);
|
|
::benchmark::DoNotOptimize(result);
|
|
timer.Stop();
|
|
|
|
state.SetIterationTime(timer.GetElapsedTime());
|
|
}
|
|
}
|
|
VTKM_BENCHMARK(BenchThreshold);
|
|
|
|
void BenchThresholdPoints(::benchmark::State& state)
|
|
{
|
|
const vtkm::cont::DeviceAdapterId device = Config.Device;
|
|
const bool compactPoints = static_cast<bool>(state.range(0));
|
|
|
|
// Lookup the point scalar range
|
|
const auto range = []() -> vtkm::Range {
|
|
auto ptScalarField =
|
|
InputDataSet.GetField(PointScalarsName, vtkm::cont::Field::Association::POINTS);
|
|
return vtkm::cont::ArrayGetValue(0, ptScalarField.GetRange());
|
|
}();
|
|
|
|
// Extract points with values between 25-75% of the range
|
|
vtkm::Float64 quarter = range.Length() / 4.;
|
|
vtkm::Float64 mid = range.Center();
|
|
|
|
vtkm::filter::ThresholdPoints filter;
|
|
filter.SetActiveField(PointScalarsName, vtkm::cont::Field::Association::POINTS);
|
|
filter.SetLowerThreshold(mid - quarter);
|
|
filter.SetUpperThreshold(mid + quarter);
|
|
filter.SetCompactPoints(compactPoints);
|
|
|
|
vtkm::cont::Timer timer{ device };
|
|
for (auto _ : state)
|
|
{
|
|
(void)_;
|
|
timer.Start();
|
|
auto result = filter.Execute(InputDataSet);
|
|
::benchmark::DoNotOptimize(result);
|
|
timer.Stop();
|
|
|
|
state.SetIterationTime(timer.GetElapsedTime());
|
|
}
|
|
}
|
|
VTKM_BENCHMARK_OPTS(BenchThresholdPoints, ->ArgName("CompactPts")->DenseRange(0, 1));
|
|
|
|
void BenchCellAverage(::benchmark::State& state)
|
|
{
|
|
const vtkm::cont::DeviceAdapterId device = Config.Device;
|
|
|
|
vtkm::filter::CellAverage filter;
|
|
filter.SetActiveField(PointScalarsName, vtkm::cont::Field::Association::POINTS);
|
|
|
|
vtkm::cont::Timer timer{ device };
|
|
for (auto _ : state)
|
|
{
|
|
(void)_;
|
|
timer.Start();
|
|
auto result = filter.Execute(InputDataSet);
|
|
::benchmark::DoNotOptimize(result);
|
|
timer.Stop();
|
|
|
|
state.SetIterationTime(timer.GetElapsedTime());
|
|
}
|
|
}
|
|
VTKM_BENCHMARK(BenchCellAverage);
|
|
|
|
void BenchPointAverage(::benchmark::State& state)
|
|
{
|
|
const vtkm::cont::DeviceAdapterId device = Config.Device;
|
|
|
|
vtkm::filter::PointAverage filter;
|
|
filter.SetActiveField(CellScalarsName, vtkm::cont::Field::Association::CELL_SET);
|
|
|
|
vtkm::cont::Timer timer{ device };
|
|
for (auto _ : state)
|
|
{
|
|
(void)_;
|
|
timer.Start();
|
|
auto result = filter.Execute(InputDataSet);
|
|
::benchmark::DoNotOptimize(result);
|
|
timer.Stop();
|
|
|
|
state.SetIterationTime(timer.GetElapsedTime());
|
|
}
|
|
}
|
|
VTKM_BENCHMARK(BenchPointAverage);
|
|
|
|
void BenchWarpScalar(::benchmark::State& state)
|
|
{
|
|
const vtkm::cont::DeviceAdapterId device = Config.Device;
|
|
|
|
vtkm::filter::WarpScalar filter{ 2. };
|
|
filter.SetUseCoordinateSystemAsField(true);
|
|
filter.SetNormalField(PointVectorsName, vtkm::cont::Field::Association::POINTS);
|
|
filter.SetScalarFactorField(PointScalarsName, vtkm::cont::Field::Association::POINTS);
|
|
|
|
vtkm::cont::Timer timer{ device };
|
|
for (auto _ : state)
|
|
{
|
|
(void)_;
|
|
timer.Start();
|
|
auto result = filter.Execute(InputDataSet);
|
|
::benchmark::DoNotOptimize(result);
|
|
timer.Stop();
|
|
|
|
state.SetIterationTime(timer.GetElapsedTime());
|
|
}
|
|
}
|
|
VTKM_BENCHMARK(BenchWarpScalar);
|
|
|
|
void BenchWarpVector(::benchmark::State& state)
|
|
{
|
|
const vtkm::cont::DeviceAdapterId device = Config.Device;
|
|
|
|
vtkm::filter::WarpVector filter{ 2. };
|
|
filter.SetUseCoordinateSystemAsField(true);
|
|
filter.SetVectorField(PointVectorsName, vtkm::cont::Field::Association::POINTS);
|
|
|
|
vtkm::cont::Timer timer{ device };
|
|
for (auto _ : state)
|
|
{
|
|
(void)_;
|
|
timer.Start();
|
|
auto result = filter.Execute(InputDataSet);
|
|
::benchmark::DoNotOptimize(result);
|
|
timer.Stop();
|
|
|
|
state.SetIterationTime(timer.GetElapsedTime());
|
|
}
|
|
}
|
|
VTKM_BENCHMARK(BenchWarpVector);
|
|
|
|
void BenchContour(::benchmark::State& state)
|
|
{
|
|
const vtkm::cont::DeviceAdapterId device = Config.Device;
|
|
|
|
const bool isStructured = static_cast<vtkm::Id>(state.range(0));
|
|
const vtkm::Id numIsoVals = static_cast<vtkm::Id>(state.range(1));
|
|
const bool mergePoints = static_cast<bool>(state.range(2));
|
|
const bool normals = static_cast<bool>(state.range(3));
|
|
const bool fastNormals = static_cast<bool>(state.range(4));
|
|
|
|
vtkm::filter::Contour filter;
|
|
filter.SetActiveField(PointScalarsName, vtkm::cont::Field::Association::POINTS);
|
|
|
|
// Set up some equally spaced contours, with the min/max slightly inside the
|
|
// scalar range:
|
|
const vtkm::Range scalarRange = []() -> vtkm::Range {
|
|
auto field = InputDataSet.GetField(PointScalarsName, vtkm::cont::Field::Association::POINTS);
|
|
return vtkm::cont::ArrayGetValue(0, field.GetRange());
|
|
}();
|
|
const auto step = scalarRange.Length() / static_cast<vtkm::Float64>(numIsoVals + 1);
|
|
const auto minIsoVal = scalarRange.Min + (step / 2.);
|
|
|
|
filter.SetNumberOfIsoValues(numIsoVals);
|
|
for (vtkm::Id i = 0; i < numIsoVals; ++i)
|
|
{
|
|
filter.SetIsoValue(i, minIsoVal + (step * static_cast<vtkm::Float64>(i)));
|
|
}
|
|
|
|
filter.SetMergeDuplicatePoints(mergePoints);
|
|
filter.SetGenerateNormals(normals);
|
|
filter.SetComputeFastNormalsForStructured(fastNormals);
|
|
filter.SetComputeFastNormalsForUnstructured(fastNormals);
|
|
|
|
vtkm::cont::Timer timer{ device };
|
|
|
|
vtkm::cont::DataSet input = isStructured ? InputDataSet : UnstructuredInputDataSet;
|
|
|
|
for (auto _ : state)
|
|
{
|
|
(void)_;
|
|
timer.Start();
|
|
auto result = filter.Execute(input);
|
|
::benchmark::DoNotOptimize(result);
|
|
timer.Stop();
|
|
|
|
state.SetIterationTime(timer.GetElapsedTime());
|
|
}
|
|
}
|
|
|
|
void BenchContourGenerator(::benchmark::internal::Benchmark* bm)
|
|
{
|
|
bm->ArgNames({ "IsStructuredDataSet", "NIsoVals", "MergePts", "GenNormals", "FastNormals" });
|
|
|
|
auto helper = [&](const vtkm::Id numIsoVals) {
|
|
bm->Args({ 0, numIsoVals, 0, 0, 0 });
|
|
bm->Args({ 0, numIsoVals, 1, 0, 0 });
|
|
bm->Args({ 0, numIsoVals, 0, 1, 0 });
|
|
bm->Args({ 0, numIsoVals, 0, 1, 1 });
|
|
bm->Args({ 1, numIsoVals, 0, 0, 0 });
|
|
bm->Args({ 1, numIsoVals, 1, 0, 0 });
|
|
bm->Args({ 1, numIsoVals, 0, 1, 0 });
|
|
bm->Args({ 1, numIsoVals, 0, 1, 1 });
|
|
};
|
|
|
|
helper(1);
|
|
helper(3);
|
|
helper(12);
|
|
}
|
|
|
|
// :TODO: Disabled until SIGSEGV in Countour when passings field is resolved
|
|
VTKM_BENCHMARK_APPLY(BenchContour, BenchContourGenerator);
|
|
|
|
void BenchExternalFaces(::benchmark::State& state)
|
|
{
|
|
const vtkm::cont::DeviceAdapterId device = Config.Device;
|
|
const bool compactPoints = static_cast<bool>(state.range(0));
|
|
|
|
vtkm::filter::ExternalFaces filter;
|
|
filter.SetCompactPoints(compactPoints);
|
|
|
|
vtkm::cont::Timer timer{ device };
|
|
for (auto _ : state)
|
|
{
|
|
(void)_;
|
|
timer.Start();
|
|
auto result = filter.Execute(InputDataSet);
|
|
::benchmark::DoNotOptimize(result);
|
|
timer.Stop();
|
|
|
|
state.SetIterationTime(timer.GetElapsedTime());
|
|
}
|
|
}
|
|
VTKM_BENCHMARK_OPTS(BenchExternalFaces, ->ArgName("Compact")->DenseRange(0, 1));
|
|
|
|
void BenchTetrahedralize(::benchmark::State& state)
|
|
{
|
|
const vtkm::cont::DeviceAdapterId device = Config.Device;
|
|
|
|
// This filter only supports structured datasets:
|
|
if (FileAsInput && !InputIsStructured())
|
|
{
|
|
state.SkipWithError("Tetrahedralize Filter requires structured data.");
|
|
}
|
|
|
|
vtkm::filter::Tetrahedralize filter;
|
|
|
|
vtkm::cont::Timer timer{ device };
|
|
for (auto _ : state)
|
|
{
|
|
(void)_;
|
|
timer.Start();
|
|
auto result = filter.Execute(InputDataSet);
|
|
::benchmark::DoNotOptimize(result);
|
|
timer.Stop();
|
|
|
|
state.SetIterationTime(timer.GetElapsedTime());
|
|
}
|
|
}
|
|
VTKM_BENCHMARK(BenchTetrahedralize);
|
|
|
|
void BenchVertexClustering(::benchmark::State& state)
|
|
{
|
|
const vtkm::cont::DeviceAdapterId device = Config.Device;
|
|
const vtkm::Id numDivs = static_cast<vtkm::Id>(state.range(0));
|
|
|
|
// This filter only supports unstructured datasets:
|
|
if (FileAsInput && InputIsStructured())
|
|
{
|
|
state.SkipWithError("VertexClustering Filter requires unstructured data (use --tetra).");
|
|
}
|
|
|
|
vtkm::filter::VertexClustering filter;
|
|
filter.SetNumberOfDivisions({ numDivs });
|
|
|
|
vtkm::cont::Timer timer{ device };
|
|
for (auto _ : state)
|
|
{
|
|
(void)_;
|
|
|
|
timer.Start();
|
|
auto result = filter.Execute(UnstructuredInputDataSet);
|
|
::benchmark::DoNotOptimize(result);
|
|
timer.Stop();
|
|
|
|
state.SetIterationTime(timer.GetElapsedTime());
|
|
}
|
|
}
|
|
VTKM_BENCHMARK_OPTS(BenchVertexClustering,
|
|
->RangeMultiplier(2)
|
|
->Range(32, 1024)
|
|
->ArgName("NumDivs"));
|
|
|
|
// Helper for resetting the reverse connectivity table:
|
|
struct PrepareForInput
|
|
{
|
|
mutable vtkm::cont::Timer Timer;
|
|
|
|
PrepareForInput()
|
|
: Timer{ Config.Device }
|
|
{
|
|
}
|
|
|
|
void operator()(const vtkm::cont::CellSet& cellSet) const
|
|
{
|
|
static bool warned{ false };
|
|
if (!warned)
|
|
{
|
|
std::cerr << "Invalid cellset type for benchmark.\n";
|
|
cellSet.PrintSummary(std::cerr);
|
|
warned = true;
|
|
}
|
|
}
|
|
|
|
template <typename T1, typename T2, typename T3>
|
|
VTKM_CONT void operator()(const vtkm::cont::CellSetExplicit<T1, T2, T3>& cellSet) const
|
|
{
|
|
vtkm::cont::TryExecuteOnDevice(Config.Device, *this, cellSet);
|
|
}
|
|
|
|
template <typename T1, typename T2, typename T3, typename DeviceTag>
|
|
VTKM_CONT bool operator()(DeviceTag, const vtkm::cont::CellSetExplicit<T1, T2, T3>& cellSet) const
|
|
{
|
|
// Why does CastAndCall insist on making the cellset const?
|
|
using CellSetT = vtkm::cont::CellSetExplicit<T1, T2, T3>;
|
|
CellSetT& mcellSet = const_cast<CellSetT&>(cellSet);
|
|
mcellSet.ResetConnectivity(vtkm::TopologyElementTagPoint{}, vtkm::TopologyElementTagCell{});
|
|
|
|
vtkm::cont::Token token;
|
|
this->Timer.Start();
|
|
auto result = cellSet.PrepareForInput(
|
|
DeviceTag{}, vtkm::TopologyElementTagPoint{}, vtkm::TopologyElementTagCell{}, token);
|
|
::benchmark::DoNotOptimize(result);
|
|
this->Timer.Stop();
|
|
|
|
return true;
|
|
}
|
|
};
|
|
|
|
void BenchReverseConnectivityGen(::benchmark::State& state)
|
|
{
|
|
if (FileAsInput && InputIsStructured())
|
|
{
|
|
state.SkipWithError("ReverseConnectivityGen requires unstructured data (--use tetra).");
|
|
}
|
|
|
|
auto cellset = UnstructuredInputDataSet.GetCellSet();
|
|
PrepareForInput functor;
|
|
for (auto _ : state)
|
|
{
|
|
(void)_;
|
|
cellset.CastAndCall(functor);
|
|
state.SetIterationTime(functor.Timer.GetElapsedTime());
|
|
}
|
|
}
|
|
VTKM_BENCHMARK(BenchReverseConnectivityGen);
|
|
|
|
// Generates a Vec3 field from point coordinates.
|
|
struct PointVectorGenerator : public vtkm::worklet::WorkletMapField
|
|
{
|
|
public:
|
|
using ControlSignature = void(FieldIn, FieldOut);
|
|
using ExecutionSignature = _2(_1);
|
|
|
|
vtkm::Bounds Bounds;
|
|
vtkm::Vec3f_64 Center;
|
|
vtkm::Vec3f_64 Scale;
|
|
|
|
VTKM_CONT
|
|
PointVectorGenerator(const vtkm::Bounds& bounds)
|
|
: Bounds(bounds)
|
|
, Center(bounds.Center())
|
|
, Scale((6. * vtkm::Pi()) / bounds.X.Length(),
|
|
(2. * vtkm::Pi()) / bounds.Y.Length(),
|
|
(7. * vtkm::Pi()) / bounds.Z.Length())
|
|
{
|
|
}
|
|
|
|
template <typename T>
|
|
VTKM_EXEC vtkm::Vec<T, 3> operator()(vtkm::Vec<T, 3> val) const
|
|
{
|
|
using Vec3T = vtkm::Vec<T, 3>;
|
|
using Vec3F64 = vtkm::Vec3f_64;
|
|
|
|
Vec3F64 valF64{ val };
|
|
Vec3F64 periodic = (valF64 - this->Center) * this->Scale;
|
|
periodic[0] = vtkm::Sin(periodic[0]);
|
|
periodic[1] = vtkm::Sin(periodic[1]);
|
|
periodic[2] = vtkm::Cos(periodic[2]);
|
|
|
|
if (vtkm::MagnitudeSquared(periodic) > 0.)
|
|
{
|
|
vtkm::Normalize(periodic);
|
|
}
|
|
if (vtkm::MagnitudeSquared(valF64) > 0.)
|
|
{
|
|
vtkm::Normalize(valF64);
|
|
}
|
|
return Vec3T{ vtkm::Normal(periodic + valF64) };
|
|
}
|
|
};
|
|
|
|
// Get the number of components in a VariantArrayHandle, ArrayHandle, or Field's
|
|
// ValueType.
|
|
struct NumberOfComponents
|
|
{
|
|
vtkm::IdComponent NumComponents;
|
|
|
|
template <typename ArrayHandleT>
|
|
VTKM_CONT void operator()(const ArrayHandleT&)
|
|
{
|
|
using ValueType = typename ArrayHandleT::ValueType;
|
|
using Traits = vtkm::VecTraits<ValueType>;
|
|
this->NumComponents = Traits::NUM_COMPONENTS;
|
|
}
|
|
|
|
template <typename DynamicType>
|
|
VTKM_CONT static vtkm::IdComponent Check(const DynamicType& obj)
|
|
{
|
|
NumberOfComponents functor;
|
|
vtkm::cont::CastAndCall(obj, functor);
|
|
return functor.NumComponents;
|
|
}
|
|
};
|
|
|
|
void FindFields()
|
|
{
|
|
if (PointScalarsName.empty())
|
|
{
|
|
for (vtkm::Id i = 0; i < InputDataSet.GetNumberOfFields(); ++i)
|
|
{
|
|
auto field = InputDataSet.GetField(i);
|
|
if (field.GetAssociation() == vtkm::cont::Field::Association::POINTS &&
|
|
NumberOfComponents::Check(field) == 1)
|
|
{
|
|
PointScalarsName = field.GetName();
|
|
std::cerr << "[FindFields] Found PointScalars: " << PointScalarsName << "\n";
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (CellScalarsName.empty())
|
|
{
|
|
for (vtkm::Id i = 0; i < InputDataSet.GetNumberOfFields(); ++i)
|
|
{
|
|
auto field = InputDataSet.GetField(i);
|
|
if (field.GetAssociation() == vtkm::cont::Field::Association::CELL_SET &&
|
|
NumberOfComponents::Check(field) == 1)
|
|
{
|
|
CellScalarsName = field.GetName();
|
|
std::cerr << "[FindFields] CellScalars: " << CellScalarsName << "\n";
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (PointVectorsName.empty())
|
|
{
|
|
for (vtkm::Id i = 0; i < InputDataSet.GetNumberOfFields(); ++i)
|
|
{
|
|
auto field = InputDataSet.GetField(i);
|
|
if (field.GetAssociation() == vtkm::cont::Field::Association::POINTS &&
|
|
NumberOfComponents::Check(field) == 3)
|
|
{
|
|
PointVectorsName = field.GetName();
|
|
std::cerr << "[FindFields] Found PointVectors: " << PointVectorsName << "\n";
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void CreateMissingFields()
|
|
{
|
|
// Do point vectors first, so we can generate the scalars from them if needed
|
|
if (PointVectorsName.empty())
|
|
{
|
|
// Construct them from the coordinates:
|
|
auto coords = InputDataSet.GetCoordinateSystem();
|
|
auto bounds = coords.GetBounds();
|
|
auto points = coords.GetData();
|
|
vtkm::cont::ArrayHandle<vtkm::Vec3f> pvecs;
|
|
|
|
PointVectorGenerator worklet(bounds);
|
|
vtkm::worklet::DispatcherMapField<PointVectorGenerator> dispatch(worklet);
|
|
dispatch.Invoke(points, pvecs);
|
|
InputDataSet.AddField(
|
|
vtkm::cont::Field("GeneratedPointVectors", vtkm::cont::Field::Association::POINTS, pvecs));
|
|
PointVectorsName = "GeneratedPointVectors";
|
|
std::cerr << "[CreateFields] Generated point vectors '" << PointVectorsName
|
|
<< "' from coordinate data.\n";
|
|
}
|
|
|
|
if (PointScalarsName.empty())
|
|
{
|
|
if (!CellScalarsName.empty())
|
|
{ // Generate from found cell field:
|
|
vtkm::filter::PointAverage avg;
|
|
avg.SetActiveField(CellScalarsName, vtkm::cont::Field::Association::CELL_SET);
|
|
avg.SetOutputFieldName("GeneratedPointScalars");
|
|
auto outds = avg.Execute(InputDataSet);
|
|
InputDataSet.AddField(
|
|
outds.GetField("GeneratedPointScalars", vtkm::cont::Field::Association::POINTS));
|
|
PointScalarsName = "GeneratedPointScalars";
|
|
std::cerr << "[CreateFields] Generated point scalars '" << PointScalarsName
|
|
<< "' from cell scalars, '" << CellScalarsName << "'.\n";
|
|
}
|
|
else
|
|
{
|
|
// Compute the magnitude of the vectors:
|
|
VTKM_ASSERT(!PointVectorsName.empty());
|
|
vtkm::filter::VectorMagnitude mag;
|
|
mag.SetActiveField(PointVectorsName, vtkm::cont::Field::Association::POINTS);
|
|
mag.SetOutputFieldName("GeneratedPointScalars");
|
|
auto outds = mag.Execute(InputDataSet);
|
|
InputDataSet.AddField(
|
|
outds.GetField("GeneratedPointScalars", vtkm::cont::Field::Association::POINTS));
|
|
PointScalarsName = "GeneratedPointScalars";
|
|
std::cerr << "[CreateFields] Generated point scalars '" << PointScalarsName
|
|
<< "' from point vectors, '" << PointVectorsName << "'.\n";
|
|
}
|
|
}
|
|
|
|
if (CellScalarsName.empty())
|
|
{ // Attempt to construct them from a point field:
|
|
VTKM_ASSERT(!PointScalarsName.empty());
|
|
vtkm::filter::CellAverage avg;
|
|
avg.SetActiveField(PointScalarsName, vtkm::cont::Field::Association::POINTS);
|
|
avg.SetOutputFieldName("GeneratedCellScalars");
|
|
auto outds = avg.Execute(InputDataSet);
|
|
InputDataSet.AddField(
|
|
outds.GetField("GeneratedCellScalars", vtkm::cont::Field::Association::CELL_SET));
|
|
CellScalarsName = "GeneratedCellScalars";
|
|
std::cerr << "[CreateFields] Generated cell scalars '" << CellScalarsName
|
|
<< "' from point scalars, '" << PointScalarsName << "'.\n";
|
|
}
|
|
}
|
|
|
|
struct Arg : vtkm::cont::internal::option::Arg
|
|
{
|
|
static vtkm::cont::internal::option::ArgStatus Number(
|
|
const vtkm::cont::internal::option::Option& option,
|
|
bool msg)
|
|
{
|
|
bool argIsNum = ((option.arg != nullptr) && (option.arg[0] != '\0'));
|
|
const char* c = option.arg;
|
|
while (argIsNum && (*c != '\0'))
|
|
{
|
|
argIsNum &= static_cast<bool>(std::isdigit(*c));
|
|
++c;
|
|
}
|
|
|
|
if (argIsNum)
|
|
{
|
|
return vtkm::cont::internal::option::ARG_OK;
|
|
}
|
|
else
|
|
{
|
|
if (msg)
|
|
{
|
|
std::cerr << "Option " << option.name << " requires a numeric argument." << std::endl;
|
|
}
|
|
|
|
return vtkm::cont::internal::option::ARG_ILLEGAL;
|
|
}
|
|
}
|
|
|
|
static vtkm::cont::internal::option::ArgStatus Required(
|
|
const vtkm::cont::internal::option::Option& option,
|
|
bool msg)
|
|
{
|
|
if ((option.arg != nullptr) && (option.arg[0] != '\0'))
|
|
{
|
|
return vtkm::cont::internal::option::ARG_OK;
|
|
}
|
|
else
|
|
{
|
|
if (msg)
|
|
{
|
|
std::cerr << "Option " << option.name << " requires an argument." << std::endl;
|
|
}
|
|
return vtkm::cont::internal::option::ARG_ILLEGAL;
|
|
}
|
|
}
|
|
};
|
|
|
|
enum optionIndex
|
|
{
|
|
UNKNOWN,
|
|
HELP,
|
|
NUM_THREADS,
|
|
FILENAME,
|
|
POINT_SCALARS,
|
|
CELL_SCALARS,
|
|
POINT_VECTORS,
|
|
WAVELET_DIM,
|
|
TETRA
|
|
};
|
|
|
|
void InitDataSet(int& argc, char** argv)
|
|
{
|
|
int numThreads = 0;
|
|
std::string filename;
|
|
vtkm::Id waveletDim = 256;
|
|
bool tetra = false;
|
|
|
|
namespace option = vtkm::cont::internal::option;
|
|
|
|
std::vector<option::Descriptor> usage;
|
|
std::string usageHeader{ "Usage: " };
|
|
usageHeader.append(argv[0]);
|
|
usageHeader.append(" [input data options] [benchmark options]");
|
|
usage.push_back({ UNKNOWN, 0, "", "", Arg::None, usageHeader.c_str() });
|
|
usage.push_back({ UNKNOWN, 0, "", "", Arg::None, "Input data options are:" });
|
|
usage.push_back({ HELP, 0, "h", "help", Arg::None, " -h, --help\tDisplay this help." });
|
|
usage.push_back({ UNKNOWN, 0, "", "", Arg::None, Config.Usage.c_str() });
|
|
usage.push_back({ NUM_THREADS,
|
|
0,
|
|
"",
|
|
"num-threads",
|
|
Arg::Number,
|
|
" --num-threads <N> \tSpecify the number of threads to use." });
|
|
usage.push_back({ FILENAME,
|
|
0,
|
|
"",
|
|
"file",
|
|
Arg::Required,
|
|
" --file <filename> \tFile (in legacy vtk format) to read as input. "
|
|
"If not specified, a wavelet source is generated." });
|
|
usage.push_back({ POINT_SCALARS,
|
|
0,
|
|
"",
|
|
"point-scalars",
|
|
Arg::Required,
|
|
" --point-scalars <name> \tName of the point scalar field to operate on." });
|
|
usage.push_back({ CELL_SCALARS,
|
|
0,
|
|
"",
|
|
"cell-scalars",
|
|
Arg::Required,
|
|
" --cell-scalars <name> \tName of the cell scalar field to operate on." });
|
|
usage.push_back({ POINT_VECTORS,
|
|
0,
|
|
"",
|
|
"point-vectors",
|
|
Arg::Required,
|
|
" --point-vectors <name> \tName of the point vector field to operate on." });
|
|
usage.push_back({ WAVELET_DIM,
|
|
0,
|
|
"",
|
|
"wavelet-dim",
|
|
Arg::Number,
|
|
" --wavelet-dim <N> \tThe size in each dimension of the wavelet grid "
|
|
"(if generated)." });
|
|
usage.push_back({ TETRA,
|
|
0,
|
|
"",
|
|
"tetra",
|
|
Arg::None,
|
|
" --tetra \tTetrahedralize data set before running benchmark." });
|
|
usage.push_back({ 0, 0, nullptr, nullptr, nullptr, nullptr });
|
|
|
|
|
|
vtkm::cont::internal::option::Stats stats(usage.data(), argc - 1, argv + 1);
|
|
std::unique_ptr<option::Option[]> options{ new option::Option[stats.options_max] };
|
|
std::unique_ptr<option::Option[]> buffer{ new option::Option[stats.buffer_max] };
|
|
option::Parser commandLineParse(usage.data(), argc - 1, argv + 1, options.get(), buffer.get());
|
|
|
|
if (options[HELP])
|
|
{
|
|
option::printUsage(std::cout, usage.data());
|
|
// Print google benchmark usage too
|
|
const char* helpstr = "--help";
|
|
char* tmpargv[] = { argv[0], const_cast<char*>(helpstr), nullptr };
|
|
int tmpargc = 2;
|
|
VTKM_EXECUTE_BENCHMARKS(tmpargc, tmpargv);
|
|
exit(0);
|
|
}
|
|
|
|
if (options[NUM_THREADS])
|
|
{
|
|
std::istringstream parse(options[NUM_THREADS].arg);
|
|
parse >> numThreads;
|
|
if (Config.Device == vtkm::cont::DeviceAdapterTagTBB() ||
|
|
Config.Device == vtkm::cont::DeviceAdapterTagOpenMP())
|
|
{
|
|
std::cout << "Selected " << numThreads << " " << Config.Device.GetName() << " threads."
|
|
<< std::endl;
|
|
}
|
|
else
|
|
{
|
|
std::cerr << options[NUM_THREADS].name << " not valid on this device. Ignoring." << std::endl;
|
|
}
|
|
}
|
|
|
|
if (options[FILENAME])
|
|
{
|
|
filename = options[FILENAME].arg;
|
|
}
|
|
|
|
if (options[POINT_SCALARS])
|
|
{
|
|
PointScalarsName = options[POINT_SCALARS].arg;
|
|
}
|
|
if (options[CELL_SCALARS])
|
|
{
|
|
CellScalarsName = options[CELL_SCALARS].arg;
|
|
}
|
|
if (options[POINT_VECTORS])
|
|
{
|
|
PointVectorsName = options[POINT_VECTORS].arg;
|
|
}
|
|
|
|
if (options[WAVELET_DIM])
|
|
{
|
|
std::istringstream parse(options[WAVELET_DIM].arg);
|
|
parse >> waveletDim;
|
|
}
|
|
|
|
tetra = (options[TETRA] != nullptr);
|
|
|
|
#ifdef VTKM_ENABLE_TBB
|
|
// Must not be destroyed as long as benchmarks are running:
|
|
tbb::task_scheduler_init init((numThreads > 0) ? numThreads
|
|
: tbb::task_scheduler_init::automatic);
|
|
#endif
|
|
#ifdef VTKM_ENABLE_OPENMP
|
|
omp_set_num_threads((numThreads > 0) ? numThreads : omp_get_max_threads());
|
|
#endif
|
|
|
|
// Now go back through the arg list and remove anything that is not in the list of
|
|
// unknown options or non-option arguments.
|
|
int destArg = 1;
|
|
// This is copy/pasted from vtkm::cont::Initialize(), should probably be abstracted eventually:
|
|
for (int srcArg = 1; srcArg < argc; ++srcArg)
|
|
{
|
|
std::string thisArg{ argv[srcArg] };
|
|
bool copyArg = false;
|
|
|
|
// Special case: "--" gets removed by optionparser but should be passed.
|
|
if (thisArg == "--")
|
|
{
|
|
copyArg = true;
|
|
}
|
|
for (const option::Option* opt = options[UNKNOWN]; !copyArg && opt != nullptr;
|
|
opt = opt->next())
|
|
{
|
|
if (thisArg == opt->name)
|
|
{
|
|
copyArg = true;
|
|
}
|
|
if ((opt->arg != nullptr) && (thisArg == opt->arg))
|
|
{
|
|
copyArg = true;
|
|
}
|
|
// Special case: optionparser sometimes removes a single "-" from an option
|
|
if (thisArg.substr(1) == opt->name)
|
|
{
|
|
copyArg = true;
|
|
}
|
|
}
|
|
for (int nonOpt = 0; !copyArg && nonOpt < commandLineParse.nonOptionsCount(); ++nonOpt)
|
|
{
|
|
if (thisArg == commandLineParse.nonOption(nonOpt))
|
|
{
|
|
copyArg = true;
|
|
}
|
|
}
|
|
if (copyArg)
|
|
{
|
|
if (destArg != srcArg)
|
|
{
|
|
argv[destArg] = argv[srcArg];
|
|
}
|
|
++destArg;
|
|
}
|
|
}
|
|
argc = destArg;
|
|
|
|
// Load / generate the dataset
|
|
vtkm::cont::Timer inputGenTimer{ Config.Device };
|
|
inputGenTimer.Start();
|
|
|
|
if (!filename.empty())
|
|
{
|
|
std::cerr << "[InitDataSet] Loading file: " << filename << "\n";
|
|
vtkm::io::VTKDataSetReader reader(filename);
|
|
InputDataSet = reader.ReadDataSet();
|
|
FileAsInput = true;
|
|
}
|
|
else
|
|
{
|
|
std::cerr << "[InitDataSet] Generating " << waveletDim << "x" << waveletDim << "x" << waveletDim
|
|
<< " wavelet...\n";
|
|
vtkm::source::Wavelet source;
|
|
source.SetExtent({ 0 }, { waveletDim - 1 });
|
|
|
|
InputDataSet = source.Execute();
|
|
}
|
|
|
|
FindFields();
|
|
CreateMissingFields();
|
|
|
|
std::cerr
|
|
<< "[InitDataSet] Create UnstructuredInputDataSet from Tetrahedralized InputDataSet...\n";
|
|
vtkm::filter::Tetrahedralize tet;
|
|
tet.SetFieldsToPass(vtkm::filter::FieldSelection(vtkm::filter::FieldSelection::MODE_ALL));
|
|
UnstructuredInputDataSet = tet.Execute(InputDataSet);
|
|
|
|
if (tetra)
|
|
{
|
|
InputDataSet = UnstructuredInputDataSet;
|
|
}
|
|
|
|
inputGenTimer.Stop();
|
|
|
|
std::cerr << "[InitDataSet] DataSet initialization took " << inputGenTimer.GetElapsedTime()
|
|
<< " seconds.\n\n-----------------";
|
|
}
|
|
|
|
} // end anon namespace
|
|
|
|
int main(int argc, char* argv[])
|
|
{
|
|
auto opts = vtkm::cont::InitializeOptions::RequireDevice;
|
|
|
|
std::vector<char*> args(argv, argv + argc);
|
|
vtkm::bench::detail::InitializeArgs(&argc, args, opts);
|
|
|
|
// Parse VTK-m options:
|
|
Config = vtkm::cont::Initialize(argc, args.data(), opts);
|
|
|
|
// This opts changes when it is help
|
|
if (opts != vtkm::cont::InitializeOptions::None)
|
|
{
|
|
vtkm::cont::GetRuntimeDeviceTracker().ForceDevice(Config.Device);
|
|
}
|
|
InitDataSet(argc, args.data());
|
|
|
|
const std::string dataSetSummary = []() -> std::string {
|
|
std::ostringstream out;
|
|
InputDataSet.PrintSummary(out);
|
|
return out.str();
|
|
}();
|
|
|
|
// handle benchmarking related args and run benchmarks:
|
|
VTKM_EXECUTE_BENCHMARKS_PREAMBLE(argc, args.data(), dataSetSummary);
|
|
}
|