vtk-m/vtkm/filter/LagrangianStructures.hxx
Abhishek Yenpure 4c781374c2 Removing virtuals v1
-- Remove virtuals from Integrators, Fields, Particles
-- Remaining virtuals are in the CellInterpolationHelpers
2021-03-22 23:00:36 -07:00

197 lines
7.3 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.
//============================================================================
#ifndef vtk_m_filter_LagrangianStructures_hxx
#define vtk_m_filter_LagrangianStructures_hxx
#include <vtkm/cont/ArrayCopy.h>
#include <vtkm/cont/ArrayHandleIndex.h>
#include <vtkm/cont/ErrorFilterExecution.h>
#include <vtkm/cont/Invoker.h>
#include <vtkm/worklet/particleadvection/Field.h>
#include <vtkm/worklet/particleadvection/GridEvaluators.h>
#include <vtkm/worklet/particleadvection/Particles.h>
#include <vtkm/worklet/particleadvection/RK4Integrator.h>
#include <vtkm/worklet/particleadvection/Stepper.h>
#include <vtkm/worklet/LagrangianStructures.h>
namespace vtkm
{
namespace filter
{
namespace detail
{
class ExtractParticlePosition : public vtkm::worklet::WorkletMapField
{
public:
using ControlSignature = void(FieldIn particle, FieldOut position);
using ExecutionSignature = void(_1, _2);
using InputDomain = _1;
VTKM_EXEC void operator()(const vtkm::Particle& particle, vtkm::Vec3f& pt) const
{
pt = particle.Pos;
}
};
class MakeParticles : public vtkm::worklet::WorkletMapField
{
public:
using ControlSignature = void(FieldIn seed, FieldOut particle);
using ExecutionSignature = void(WorkIndex, _1, _2);
using InputDomain = _1;
VTKM_EXEC void operator()(const vtkm::Id index,
const vtkm::Vec3f& seed,
vtkm::Particle& particle) const
{
particle.ID = index;
particle.Pos = seed;
}
};
} //detail
//-----------------------------------------------------------------------------
inline VTKM_CONT LagrangianStructures::LagrangianStructures()
: vtkm::filter::FilterDataSetWithField<LagrangianStructures>()
{
OutputFieldName = std::string("FTLE");
}
//-----------------------------------------------------------------------------
template <typename T, typename StorageType, typename DerivedPolicy>
inline VTKM_CONT vtkm::cont::DataSet LagrangianStructures::DoExecute(
const vtkm::cont::DataSet& input,
const vtkm::cont::ArrayHandle<vtkm::Vec<T, 3>, StorageType>& field,
const vtkm::filter::FieldMetadata& fieldMeta,
const vtkm::filter::PolicyBase<DerivedPolicy>&)
{
if (!fieldMeta.IsPointField())
{
throw vtkm::cont::ErrorFilterExecution("Point field expected.");
}
using Structured2DType = vtkm::cont::CellSetStructured<2>;
using Structured3DType = vtkm::cont::CellSetStructured<3>;
using FieldHandle = vtkm::cont::ArrayHandle<vtkm::Vec<T, 3>, StorageType>;
using FieldType = vtkm::worklet::particleadvection::VelocityField<FieldHandle>;
using GridEvaluator = vtkm::worklet::particleadvection::GridEvaluator<FieldType>;
using IntegratorType = vtkm::worklet::particleadvection::RK4Integrator<GridEvaluator>;
using Stepper = vtkm::worklet::particleadvection::Stepper<IntegratorType, GridEvaluator>;
vtkm::FloatDefault stepSize = this->GetStepSize();
vtkm::Id numberOfSteps = this->GetNumberOfSteps();
vtkm::cont::CoordinateSystem coordinates = input.GetCoordinateSystem();
vtkm::cont::DynamicCellSet cellset = input.GetCellSet();
vtkm::cont::DataSet lcsInput;
if (this->GetUseAuxiliaryGrid())
{
vtkm::Id3 lcsGridDims = this->GetAuxiliaryGridDimensions();
vtkm::Bounds inputBounds = coordinates.GetBounds();
vtkm::Vec3f origin(static_cast<vtkm::FloatDefault>(inputBounds.X.Min),
static_cast<vtkm::FloatDefault>(inputBounds.Y.Min),
static_cast<vtkm::FloatDefault>(inputBounds.Z.Min));
vtkm::Vec3f spacing;
spacing[0] = static_cast<vtkm::FloatDefault>(inputBounds.X.Length()) /
static_cast<vtkm::FloatDefault>(lcsGridDims[0] - 1);
spacing[1] = static_cast<vtkm::FloatDefault>(inputBounds.Y.Length()) /
static_cast<vtkm::FloatDefault>(lcsGridDims[1] - 1);
spacing[2] = static_cast<vtkm::FloatDefault>(inputBounds.Z.Length()) /
static_cast<vtkm::FloatDefault>(lcsGridDims[2] - 1);
vtkm::cont::DataSetBuilderUniform uniformDatasetBuilder;
lcsInput = uniformDatasetBuilder.Create(lcsGridDims, origin, spacing);
}
else
{
// Check if input dataset is structured.
// If not, we cannot proceed.
if (!(cellset.IsType<Structured2DType>() || cellset.IsType<Structured3DType>()))
throw vtkm::cont::ErrorFilterExecution(
"Provided data is not structured, provide parameters for an auxiliary grid.");
lcsInput = input;
}
vtkm::cont::ArrayHandle<vtkm::Vec3f> lcsInputPoints, lcsOutputPoints;
vtkm::cont::ArrayCopy(lcsInput.GetCoordinateSystem().GetData(), lcsInputPoints);
if (this->GetUseFlowMapOutput())
{
// Check if there is a 1:1 correspondense between the flow map
// and the input points
lcsOutputPoints = this->GetFlowMapOutput();
if (lcsInputPoints.GetNumberOfValues() != lcsOutputPoints.GetNumberOfValues())
throw vtkm::cont::ErrorFilterExecution(
"Provided flow map does not correspond to the input points for LCS filter.");
}
else
{
vtkm::cont::Invoker invoke;
FieldType velocities(field);
GridEvaluator evaluator(input.GetCoordinateSystem(), input.GetCellSet(), velocities);
Stepper integrator(evaluator, stepSize);
vtkm::worklet::ParticleAdvection particles;
vtkm::worklet::ParticleAdvectionResult<vtkm::Particle> advectionResult;
vtkm::cont::ArrayHandle<vtkm::Particle> advectionPoints;
invoke(detail::MakeParticles{}, lcsInputPoints, advectionPoints);
advectionResult = particles.Run(integrator, advectionPoints, numberOfSteps);
invoke(detail::ExtractParticlePosition{}, advectionResult.Particles, lcsOutputPoints);
}
// FTLE output field
vtkm::cont::ArrayHandle<vtkm::FloatDefault> outputField;
vtkm::FloatDefault advectionTime = this->GetAdvectionTime();
vtkm::cont::DynamicCellSet lcsCellSet = lcsInput.GetCellSet();
if (lcsCellSet.IsType<Structured2DType>())
{
using AnalysisType = vtkm::worklet::LagrangianStructures<2>;
AnalysisType ftleCalculator(advectionTime, lcsCellSet);
vtkm::worklet::DispatcherMapField<AnalysisType> dispatcher(ftleCalculator);
dispatcher.Invoke(lcsInputPoints, lcsOutputPoints, outputField);
}
else if (lcsCellSet.IsType<Structured3DType>())
{
using AnalysisType = vtkm::worklet::LagrangianStructures<3>;
AnalysisType ftleCalculator(advectionTime, lcsCellSet);
vtkm::worklet::DispatcherMapField<AnalysisType> dispatcher(ftleCalculator);
dispatcher.Invoke(lcsInputPoints, lcsOutputPoints, outputField);
}
vtkm::cont::DataSet output;
output.AddCoordinateSystem(lcsInput.GetCoordinateSystem());
output.SetCellSet(lcsInput.GetCellSet());
output.AddPointField(this->GetOutputFieldName(), outputField);
return output;
}
//-----------------------------------------------------------------------------
template <typename DerivedPolicy>
inline VTKM_CONT bool LagrangianStructures::MapFieldOntoOutput(
vtkm::cont::DataSet& result,
const vtkm::cont::Field& field,
vtkm::filter::PolicyBase<DerivedPolicy>)
{
if (field.IsFieldGlobal())
{
result.AddField(field);
return true;
}
else
{
return false;
}
}
}
} // namespace vtkm::filter
#endif