2018-03-02 20:49:07 +00:00
|
|
|
//============================================================================
|
|
|
|
// Copyright (c) Kitware, Inc.
|
|
|
|
// All rights reserved.
|
|
|
|
// See LICENSE.txt for details.
|
2019-04-15 23:24:21 +00:00
|
|
|
//
|
2018-03-02 20:49:07 +00:00
|
|
|
// 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/filter/FieldMetadata.h>
|
|
|
|
#include <vtkm/filter/PolicyDefault.h>
|
|
|
|
|
|
|
|
#include <vtkm/filter/internal/ResolveFieldTypeAndExecute.h>
|
|
|
|
#include <vtkm/filter/internal/ResolveFieldTypeAndMap.h>
|
|
|
|
|
2018-04-09 19:07:32 +00:00
|
|
|
#include <vtkm/cont/ErrorFilterExecution.h>
|
2018-03-02 20:49:07 +00:00
|
|
|
#include <vtkm/cont/Field.h>
|
2018-08-30 17:29:36 +00:00
|
|
|
#include <vtkm/cont/Logging.h>
|
2018-03-02 20:49:07 +00:00
|
|
|
|
|
|
|
namespace vtkm
|
|
|
|
{
|
|
|
|
namespace filter
|
|
|
|
{
|
|
|
|
|
2018-04-09 19:07:32 +00:00
|
|
|
namespace internal
|
|
|
|
{
|
|
|
|
|
|
|
|
template <typename T, typename InputType, typename DerivedPolicy>
|
|
|
|
struct SupportsPreExecute
|
|
|
|
{
|
|
|
|
template <typename U,
|
|
|
|
typename S = decltype(std::declval<U>().PreExecute(
|
|
|
|
std::declval<InputType>(),
|
|
|
|
std::declval<vtkm::filter::PolicyBase<DerivedPolicy>>()))>
|
|
|
|
static std::true_type has(int);
|
|
|
|
template <typename U>
|
|
|
|
static std::false_type has(...);
|
|
|
|
using type = decltype(has<T>(0));
|
|
|
|
};
|
|
|
|
|
|
|
|
template <typename T, typename InputType, typename DerivedPolicy>
|
|
|
|
struct SupportsPostExecute
|
|
|
|
{
|
|
|
|
template <typename U,
|
|
|
|
typename S = decltype(std::declval<U>().PostExecute(
|
|
|
|
std::declval<InputType>(),
|
|
|
|
std::declval<InputType&>(),
|
|
|
|
std::declval<vtkm::filter::PolicyBase<DerivedPolicy>>()))>
|
|
|
|
static std::true_type has(int);
|
|
|
|
template <typename U>
|
|
|
|
static std::false_type has(...);
|
|
|
|
using type = decltype(has<T>(0));
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
template <typename T, typename InputType, typename DerivedPolicy>
|
|
|
|
struct SupportsPrepareForExecution
|
|
|
|
{
|
|
|
|
template <typename U,
|
|
|
|
typename S = decltype(std::declval<U>().PrepareForExecution(
|
|
|
|
std::declval<InputType>(),
|
|
|
|
std::declval<vtkm::filter::PolicyBase<DerivedPolicy>>()))>
|
|
|
|
static std::true_type has(int);
|
|
|
|
template <typename U>
|
|
|
|
static std::false_type has(...);
|
|
|
|
using type = decltype(has<T>(0));
|
|
|
|
};
|
|
|
|
|
|
|
|
template <typename T, typename DerivedPolicy>
|
|
|
|
struct SupportsMapFieldOntoOutput
|
|
|
|
{
|
|
|
|
template <typename U,
|
|
|
|
typename S = decltype(std::declval<U>().MapFieldOntoOutput(
|
|
|
|
std::declval<vtkm::cont::DataSet&>(),
|
|
|
|
std::declval<vtkm::cont::Field>(),
|
|
|
|
std::declval<vtkm::filter::PolicyBase<DerivedPolicy>>()))>
|
|
|
|
static std::true_type has(int);
|
|
|
|
template <typename U>
|
|
|
|
static std::false_type has(...);
|
|
|
|
using type = decltype(has<T>(0));
|
|
|
|
};
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------
|
|
|
|
template <typename Derived, typename... Args>
|
|
|
|
void CallPreExecuteInternal(std::true_type, Derived* self, Args&&... args)
|
|
|
|
{
|
|
|
|
return self->PreExecute(std::forward<Args>(args)...);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------
|
|
|
|
template <typename Derived, typename... Args>
|
|
|
|
void CallPreExecuteInternal(std::false_type, Derived*, Args&&...)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------
|
|
|
|
template <typename Derived, typename InputType, typename DerivedPolicy>
|
|
|
|
void CallPreExecute(Derived* self,
|
|
|
|
const InputType& input,
|
|
|
|
const vtkm::filter::PolicyBase<DerivedPolicy>& policy)
|
|
|
|
{
|
|
|
|
using call_supported_t = typename SupportsPreExecute<Derived, InputType, DerivedPolicy>::type;
|
|
|
|
CallPreExecuteInternal(call_supported_t(), self, input, policy);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------
|
|
|
|
template <typename Derived, typename DerivedPolicy>
|
|
|
|
void CallMapFieldOntoOutputInternal(std::true_type,
|
|
|
|
Derived* self,
|
|
|
|
const vtkm::cont::DataSet& input,
|
|
|
|
vtkm::cont::DataSet& output,
|
|
|
|
const vtkm::filter::PolicyBase<DerivedPolicy>& policy)
|
|
|
|
{
|
|
|
|
for (vtkm::IdComponent cc = 0; cc < input.GetNumberOfFields(); ++cc)
|
|
|
|
{
|
|
|
|
auto field = input.GetField(cc);
|
|
|
|
if (self->GetFieldsToPass().IsFieldSelected(field))
|
|
|
|
{
|
|
|
|
self->MapFieldOntoOutput(output, field, policy);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------
|
|
|
|
template <typename Derived, typename DerivedPolicy>
|
|
|
|
void CallMapFieldOntoOutputInternal(std::false_type,
|
2018-04-20 16:26:18 +00:00
|
|
|
Derived* self,
|
|
|
|
const vtkm::cont::DataSet& input,
|
|
|
|
vtkm::cont::DataSet& output,
|
2018-04-09 19:07:32 +00:00
|
|
|
const vtkm::filter::PolicyBase<DerivedPolicy>&)
|
|
|
|
{
|
2018-04-20 16:26:18 +00:00
|
|
|
// no MapFieldOntoOutput method is present. In that case, we simply copy the
|
|
|
|
// requested input fields to the output.
|
|
|
|
for (vtkm::IdComponent cc = 0; cc < input.GetNumberOfFields(); ++cc)
|
|
|
|
{
|
|
|
|
auto field = input.GetField(cc);
|
|
|
|
if (self->GetFieldsToPass().IsFieldSelected(field))
|
|
|
|
{
|
|
|
|
output.AddField(field);
|
|
|
|
}
|
|
|
|
}
|
2018-04-09 19:07:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------
|
|
|
|
template <typename Derived, typename DerivedPolicy>
|
|
|
|
void CallMapFieldOntoOutput(Derived* self,
|
|
|
|
const vtkm::cont::DataSet& input,
|
|
|
|
vtkm::cont::DataSet& output,
|
2019-09-17 12:04:59 +00:00
|
|
|
vtkm::filter::PolicyBase<DerivedPolicy> policy)
|
2018-04-09 19:07:32 +00:00
|
|
|
{
|
|
|
|
using call_supported_t = typename SupportsMapFieldOntoOutput<Derived, DerivedPolicy>::type;
|
|
|
|
CallMapFieldOntoOutputInternal(call_supported_t(), self, input, output, policy);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------
|
|
|
|
// forward declare.
|
|
|
|
template <typename Derived, typename InputType, typename DerivedPolicy>
|
|
|
|
InputType CallPrepareForExecution(Derived* self,
|
|
|
|
const InputType& input,
|
|
|
|
const vtkm::filter::PolicyBase<DerivedPolicy>& policy);
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------
|
|
|
|
template <typename Derived, typename InputType, typename DerivedPolicy>
|
|
|
|
InputType CallPrepareForExecutionInternal(std::true_type,
|
|
|
|
Derived* self,
|
|
|
|
const InputType& input,
|
|
|
|
const vtkm::filter::PolicyBase<DerivedPolicy>& policy)
|
|
|
|
{
|
|
|
|
return self->PrepareForExecution(input, policy);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------
|
2019-09-02 14:38:47 +00:00
|
|
|
// specialization for PartitionedDataSet input when `PrepareForExecution` is not provided
|
2018-04-09 19:07:32 +00:00
|
|
|
// by the subclass. we iterate over blocks and execute for each block
|
|
|
|
// individually.
|
|
|
|
template <typename Derived, typename DerivedPolicy>
|
2019-09-02 14:38:47 +00:00
|
|
|
vtkm::cont::PartitionedDataSet CallPrepareForExecutionInternal(
|
2018-04-09 19:07:32 +00:00
|
|
|
std::false_type,
|
|
|
|
Derived* self,
|
2019-09-02 14:38:47 +00:00
|
|
|
const vtkm::cont::PartitionedDataSet& input,
|
2018-04-09 19:07:32 +00:00
|
|
|
const vtkm::filter::PolicyBase<DerivedPolicy>& policy)
|
|
|
|
{
|
2019-09-02 14:38:47 +00:00
|
|
|
vtkm::cont::PartitionedDataSet output;
|
2018-04-09 19:07:32 +00:00
|
|
|
for (const auto& inBlock : input)
|
|
|
|
{
|
|
|
|
vtkm::cont::DataSet outBlock = CallPrepareForExecution(self, inBlock, policy);
|
|
|
|
CallMapFieldOntoOutput(self, inBlock, outBlock, policy);
|
2019-09-02 14:38:47 +00:00
|
|
|
output.AppendPartition(outBlock);
|
2018-04-09 19:07:32 +00:00
|
|
|
}
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------
|
|
|
|
template <typename Derived, typename InputType, typename DerivedPolicy>
|
|
|
|
InputType CallPrepareForExecution(Derived* self,
|
|
|
|
const InputType& input,
|
|
|
|
const vtkm::filter::PolicyBase<DerivedPolicy>& policy)
|
|
|
|
{
|
|
|
|
using call_supported_t =
|
|
|
|
typename SupportsPrepareForExecution<Derived, InputType, DerivedPolicy>::type;
|
|
|
|
return CallPrepareForExecutionInternal(call_supported_t(), self, input, policy);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------
|
|
|
|
template <typename Derived, typename InputType, typename DerivedPolicy>
|
|
|
|
void CallPostExecuteInternal(std::true_type,
|
|
|
|
Derived* self,
|
|
|
|
const InputType& input,
|
|
|
|
InputType& output,
|
|
|
|
const vtkm::filter::PolicyBase<DerivedPolicy>& policy)
|
|
|
|
{
|
|
|
|
self->PostExecute(input, output, policy);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------
|
|
|
|
template <typename Derived, typename... Args>
|
|
|
|
void CallPostExecuteInternal(std::false_type, Derived*, Args&&...)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------
|
|
|
|
template <typename Derived, typename InputType, typename DerivedPolicy>
|
|
|
|
void CallPostExecute(Derived* self,
|
|
|
|
const InputType& input,
|
|
|
|
InputType& output,
|
|
|
|
const vtkm::filter::PolicyBase<DerivedPolicy>& policy)
|
|
|
|
{
|
|
|
|
using call_supported_t = typename SupportsPostExecute<Derived, InputType, DerivedPolicy>::type;
|
|
|
|
CallPostExecuteInternal(call_supported_t(), self, input, output, policy);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-02 20:49:07 +00:00
|
|
|
//----------------------------------------------------------------------------
|
|
|
|
template <typename Derived>
|
|
|
|
inline VTKM_CONT Filter<Derived>::Filter()
|
2019-08-07 16:54:47 +00:00
|
|
|
: Invoke()
|
|
|
|
, FieldsToPass(vtkm::filter::FieldSelection::MODE_ALL)
|
2018-03-02 20:49:07 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------------
|
|
|
|
template <typename Derived>
|
|
|
|
inline VTKM_CONT Filter<Derived>::~Filter()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------------
|
|
|
|
template <typename Derived>
|
2018-03-22 16:25:09 +00:00
|
|
|
inline VTKM_CONT vtkm::cont::DataSet Filter<Derived>::Execute(const vtkm::cont::DataSet& input)
|
2018-03-02 20:49:07 +00:00
|
|
|
{
|
2020-05-15 23:47:30 +00:00
|
|
|
Derived* self = static_cast<Derived*>(this);
|
|
|
|
vtkm::cont::PartitionedDataSet output = self->Execute(vtkm::cont::PartitionedDataSet(input));
|
|
|
|
if (output.GetNumberOfPartitions() > 1)
|
|
|
|
{
|
|
|
|
throw vtkm::cont::ErrorFilterExecution("Expecting at most 1 block.");
|
|
|
|
}
|
|
|
|
return output.GetNumberOfPartitions() == 1 ? output.GetPartition(0) : vtkm::cont::DataSet();
|
2018-03-02 20:49:07 +00:00
|
|
|
}
|
|
|
|
|
2018-04-09 19:07:32 +00:00
|
|
|
//----------------------------------------------------------------------------
|
|
|
|
template <typename Derived>
|
2019-09-02 14:38:47 +00:00
|
|
|
inline VTKM_CONT vtkm::cont::PartitionedDataSet Filter<Derived>::Execute(
|
|
|
|
const vtkm::cont::PartitionedDataSet& input)
|
2018-04-09 19:07:32 +00:00
|
|
|
{
|
2020-05-15 23:47:30 +00:00
|
|
|
VTKM_LOG_SCOPE(vtkm::cont::LogLevel::Perf,
|
2020-10-26 15:00:09 +00:00
|
|
|
"Filter (%d partitions): '%s'",
|
|
|
|
(int)input.GetNumberOfPartitions(),
|
2020-05-15 23:47:30 +00:00
|
|
|
vtkm::cont::TypeToString<Derived>().c_str());
|
|
|
|
|
|
|
|
Derived* self = static_cast<Derived*>(this);
|
|
|
|
|
|
|
|
vtkm::filter::PolicyDefault policy;
|
|
|
|
|
|
|
|
// Call `void Derived::PreExecute<DerivedPolicy>(input, policy)`, if defined.
|
|
|
|
internal::CallPreExecute(self, input, policy);
|
|
|
|
|
|
|
|
// Call `PrepareForExecution` (which should probably be renamed at some point)
|
|
|
|
vtkm::cont::PartitionedDataSet output = internal::CallPrepareForExecution(self, input, policy);
|
|
|
|
|
|
|
|
// Call `Derived::PostExecute<DerivedPolicy>(input, output, policy)` if defined.
|
|
|
|
internal::CallPostExecute(self, input, output, policy);
|
|
|
|
return output;
|
2018-04-09 19:07:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-03-02 20:49:07 +00:00
|
|
|
//----------------------------------------------------------------------------
|
|
|
|
template <typename Derived>
|
|
|
|
template <typename DerivedPolicy>
|
2019-09-09 13:39:23 +00:00
|
|
|
VTKM_CONT vtkm::cont::DataSet Filter<Derived>::Execute(
|
2018-03-02 20:49:07 +00:00
|
|
|
const vtkm::cont::DataSet& input,
|
2019-09-17 12:04:59 +00:00
|
|
|
vtkm::filter::PolicyBase<DerivedPolicy> policy)
|
2018-03-02 20:49:07 +00:00
|
|
|
{
|
|
|
|
Derived* self = static_cast<Derived*>(this);
|
2020-05-15 23:47:30 +00:00
|
|
|
VTKM_DEPRECATED_SUPPRESS_BEGIN
|
2019-09-02 14:38:47 +00:00
|
|
|
vtkm::cont::PartitionedDataSet output =
|
|
|
|
self->Execute(vtkm::cont::PartitionedDataSet(input), policy);
|
2020-05-15 23:47:30 +00:00
|
|
|
VTKM_DEPRECATED_SUPPRESS_END
|
2019-09-02 14:38:47 +00:00
|
|
|
if (output.GetNumberOfPartitions() > 1)
|
2018-03-02 20:49:07 +00:00
|
|
|
{
|
2018-04-09 19:07:32 +00:00
|
|
|
throw vtkm::cont::ErrorFilterExecution("Expecting at most 1 block.");
|
2018-03-02 20:49:07 +00:00
|
|
|
}
|
2019-09-02 14:38:47 +00:00
|
|
|
return output.GetNumberOfPartitions() == 1 ? output.GetPartition(0) : vtkm::cont::DataSet();
|
2018-03-02 20:49:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------------
|
|
|
|
template <typename Derived>
|
2018-04-09 19:07:32 +00:00
|
|
|
template <typename DerivedPolicy>
|
2019-09-09 13:39:23 +00:00
|
|
|
VTKM_CONT vtkm::cont::PartitionedDataSet Filter<Derived>::Execute(
|
2019-09-02 14:38:47 +00:00
|
|
|
const vtkm::cont::PartitionedDataSet& input,
|
2019-09-17 12:04:59 +00:00
|
|
|
vtkm::filter::PolicyBase<DerivedPolicy> policy)
|
2018-03-02 20:49:07 +00:00
|
|
|
{
|
2018-08-30 17:29:36 +00:00
|
|
|
VTKM_LOG_SCOPE(vtkm::cont::LogLevel::Perf,
|
2020-10-26 15:00:09 +00:00
|
|
|
"Filter (%d partitions): '%s'",
|
|
|
|
(int)input.GetNumberOfPartitions(),
|
2019-02-28 18:15:27 +00:00
|
|
|
vtkm::cont::TypeToString<Derived>().c_str());
|
2018-08-30 17:29:36 +00:00
|
|
|
|
2018-04-09 19:07:32 +00:00
|
|
|
Derived* self = static_cast<Derived*>(this);
|
|
|
|
|
|
|
|
// Call `void Derived::PreExecute<DerivedPolicy>(input, policy)`, if defined.
|
|
|
|
internal::CallPreExecute(self, input, policy);
|
|
|
|
|
|
|
|
// Call `PrepareForExecution` (which should probably be renamed at some point)
|
2019-09-02 14:38:47 +00:00
|
|
|
vtkm::cont::PartitionedDataSet output = internal::CallPrepareForExecution(self, input, policy);
|
2018-04-09 19:07:32 +00:00
|
|
|
|
|
|
|
// Call `Derived::PostExecute<DerivedPolicy>(input, output, policy)` if defined.
|
|
|
|
internal::CallPostExecute(self, input, output, policy);
|
|
|
|
return output;
|
2018-03-02 20:49:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------------
|
|
|
|
template <typename Derived>
|
|
|
|
template <typename DerivedPolicy>
|
2018-04-09 19:07:32 +00:00
|
|
|
inline VTKM_CONT void Filter<Derived>::MapFieldsToPass(
|
|
|
|
const vtkm::cont::DataSet& input,
|
|
|
|
vtkm::cont::DataSet& output,
|
2019-09-17 12:04:59 +00:00
|
|
|
vtkm::filter::PolicyBase<DerivedPolicy> policy)
|
2018-03-02 20:49:07 +00:00
|
|
|
{
|
2018-04-09 19:07:32 +00:00
|
|
|
Derived* self = static_cast<Derived*>(this);
|
|
|
|
for (vtkm::IdComponent cc = 0; cc < input.GetNumberOfFields(); ++cc)
|
2018-03-02 20:49:07 +00:00
|
|
|
{
|
2018-04-09 19:07:32 +00:00
|
|
|
auto field = input.GetField(cc);
|
|
|
|
if (this->GetFieldsToPass().IsFieldSelected(field))
|
|
|
|
{
|
|
|
|
internal::CallMapFieldOntoOutput(self, output, field, policy);
|
|
|
|
}
|
2018-03-02 20:49:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|