vtk-m/vtkm/filter/LagrangianStructures.hxx

192 lines
7.2 KiB
C++
Raw Normal View History

//============================================================================
// 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>
2019-12-11 20:32:40 +00:00
#include <vtkm/cont/Invoker.h>
#include <vtkm/worklet/particleadvection/Field.h>
#include <vtkm/worklet/particleadvection/GridEvaluators.h>
#include <vtkm/worklet/particleadvection/Particles.h>
2020-08-15 14:16:55 +00:00
#include <vtkm/worklet/particleadvection/RK4Integrator.h>
#include <vtkm/worklet/particleadvection/Stepper.h>
2019-07-29 16:50:54 +00:00
#include <vtkm/worklet/LagrangianStructures.h>
namespace vtkm
{
namespace filter
{
2019-12-11 20:32:40 +00:00
namespace detail
{
class ExtractParticlePosition : public vtkm::worklet::WorkletMapField
{
public:
using ControlSignature = void(FieldIn particle, FieldOut position);
using ExecutionSignature = void(_1, _2);
using InputDomain = _1;
2020-08-26 12:59:51 +00:00
VTKM_EXEC void operator()(const vtkm::Particle& particle, vtkm::Vec3f& pt) const
2019-12-11 20:32:40 +00:00
{
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,
2020-08-26 12:59:51 +00:00
vtkm::Particle& particle) const
{
particle.ID = index;
particle.Pos = seed;
}
};
2019-12-11 20:32:40 +00:00
} //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>&)
{
using Structured2DType = vtkm::cont::CellSetStructured<2>;
using Structured3DType = vtkm::cont::CellSetStructured<3>;
2019-07-29 20:05:29 +00:00
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();
2019-07-29 20:05:29 +00:00
vtkm::Id numberOfSteps = this->GetNumberOfSteps();
vtkm::cont::CoordinateSystem coordinates = input.GetCoordinateSystem();
vtkm::cont::DynamicCellSet cellset = input.GetCellSet();
2019-07-29 20:05:29 +00:00
vtkm::cont::DataSet lcsInput;
2019-07-29 16:50:54 +00:00
if (this->GetUseAuxiliaryGrid())
{
vtkm::Id3 lcsGridDims = this->GetAuxiliaryGridDimensions();
2019-07-29 20:05:29 +00:00
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);
2019-07-29 16:50:54 +00:00
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>()))
2019-07-29 20:05:29 +00:00
throw vtkm::cont::ErrorFilterExecution(
"Provided data is not structured, provide parameters for an auxiliary grid.");
2019-07-29 16:50:54 +00:00
lcsInput = input;
}
vtkm::cont::ArrayHandle<vtkm::Vec3f> lcsInputPoints, lcsOutputPoints;
2019-07-29 16:50:54 +00:00
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())
2019-07-29 20:05:29 +00:00
throw vtkm::cont::ErrorFilterExecution(
"Provided flow map does not correspond to the input points for LCS filter.");
2019-07-29 16:50:54 +00:00
}
else
{
vtkm::cont::Invoker invoke;
FieldType velocities(field, fieldMeta.GetAssociation());
GridEvaluator evaluator(input.GetCoordinateSystem(), input.GetCellSet(), velocities);
Stepper integrator(evaluator, stepSize);
2019-07-29 16:50:54 +00:00
vtkm::worklet::ParticleAdvection particles;
2020-08-26 12:59:51 +00:00
vtkm::worklet::ParticleAdvectionResult<vtkm::Particle> advectionResult;
vtkm::cont::ArrayHandle<vtkm::Particle> advectionPoints;
invoke(detail::MakeParticles{}, lcsInputPoints, advectionPoints);
2019-07-29 16:50:54 +00:00
advectionResult = particles.Run(integrator, advectionPoints, numberOfSteps);
2019-12-11 20:32:40 +00:00
invoke(detail::ExtractParticlePosition{}, advectionResult.Particles, lcsOutputPoints);
2019-07-29 16:50:54 +00:00
}
// FTLE output field
vtkm::cont::ArrayHandle<vtkm::FloatDefault> outputField;
2019-07-29 16:50:54 +00:00
vtkm::FloatDefault advectionTime = this->GetAdvectionTime();
2019-07-29 16:50:54 +00:00
vtkm::cont::DynamicCellSet lcsCellSet = lcsInput.GetCellSet();
if (lcsCellSet.IsType<Structured2DType>())
{
2019-07-29 16:50:54 +00:00
using AnalysisType = vtkm::worklet::LagrangianStructures<2>;
AnalysisType ftleCalculator(advectionTime, lcsCellSet);
vtkm::worklet::DispatcherMapField<AnalysisType> dispatcher(ftleCalculator);
2019-07-29 16:50:54 +00:00
dispatcher.Invoke(lcsInputPoints, lcsOutputPoints, outputField);
}
2019-07-29 16:50:54 +00:00
else if (lcsCellSet.IsType<Structured3DType>())
{
2019-07-29 16:50:54 +00:00
using AnalysisType = vtkm::worklet::LagrangianStructures<3>;
AnalysisType ftleCalculator(advectionTime, lcsCellSet);
vtkm::worklet::DispatcherMapField<AnalysisType> dispatcher(ftleCalculator);
2019-07-29 16:50:54 +00:00
dispatcher.Invoke(lcsInputPoints, lcsOutputPoints, outputField);
}
vtkm::cont::DataSet output;
2019-07-29 16:50:54 +00:00
output.AddCoordinateSystem(lcsInput.GetCoordinateSystem());
output.SetCellSet(lcsInput.GetCellSet());
2020-05-27 19:27:47 +00:00
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