//============================================================================ // 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_cont_Field_h #define vtk_m_cont_Field_h #include #include #include #include #include #include namespace vtkm { namespace cont { /// A \c Field encapsulates an array on some piece of the mesh, such as /// the points, a cell set, a point logical dimension, or the whole mesh. /// class VTKM_CONT_EXPORT Field { public: enum struct Association { ANY, WHOLE_MESH, POINTS, CELL_SET }; VTKM_CONT Field() = default; VTKM_CONT Field(std::string name, Association association, const vtkm::cont::UnknownArrayHandle& data); template VTKM_CONT Field(std::string name, Association association, const vtkm::cont::ArrayHandle& data) : Field(name, association, vtkm::cont::UnknownArrayHandle{ data }) { } Field(const vtkm::cont::Field& src); Field(vtkm::cont::Field&& src) noexcept; VTKM_CONT virtual ~Field(); VTKM_CONT Field& operator=(const vtkm::cont::Field& src); VTKM_CONT Field& operator=(vtkm::cont::Field&& src) noexcept; VTKM_CONT bool IsFieldCell() const { return this->FieldAssociation == Association::CELL_SET; } VTKM_CONT bool IsFieldPoint() const { return this->FieldAssociation == Association::POINTS; } VTKM_CONT bool IsFieldGlobal() const { return this->FieldAssociation == Association::WHOLE_MESH; } /// Returns true if the array of the field has a value type that matches something in /// `VTKM_FIELD_TYPE_LIST` and a storage that matches something in `VTKM_FIELD_STORAGE_LIST`. VTKM_CONT bool IsSupportedType() const; VTKM_CONT vtkm::Id GetNumberOfValues() const { return this->Data.GetNumberOfValues(); } VTKM_CONT const std::string& GetName() const { return this->Name; } VTKM_CONT Association GetAssociation() const { return this->FieldAssociation; } const vtkm::cont::UnknownArrayHandle& GetData() const; vtkm::cont::UnknownArrayHandle& GetData(); VTKM_CONT const vtkm::cont::ArrayHandle& GetRange() const; VTKM_CONT void GetRange(vtkm::Range* range) const; /// \brief Get the data as an array with `vtkm::FloatDefault` components. /// /// Returns a `vtkm::cont::UnknownArrayHandle` that contains an array that either contains /// values of type `vtkm::FloatDefault` or contains `Vec`s with components of type /// `vtkm::FloatDefault`. If the array has value types that do not match this type, then /// it will be copied into an array that does. /// /// Additionally, the returned array will have a storage that is compatible with /// something in `VTKM_FIELD_STORAGE_LIST`. If this condition is not met, then the /// array will be copied. /// /// If the array contained in the field already matches the required criteria, the array /// will be returned without copying. /// VTKM_CONT vtkm::cont::UnknownArrayHandle GetDataAsDefaultFloat() const; /// \brief Get the data as an array of an expected type. /// /// Returns a `vtkm::cont::UnknownArrayHandle` that contains an array that (probably) has /// a value type that matches something in `VTKM_FIELD_TYPE_LIST` and a storage that matches /// something in `VTKM_FIELD_STORAGE_LIST`. If the array has a value type and storage that /// match `VTKM_FIELD_TYPE_LIST` and `VTKM_FIELD_STORAGE_LIST` respectively, then the same /// array is returned. If something does not match, then the data are copied to a /// `vtkm::cont::ArrayHandleBasic` with a value type component of `vtkm::FloatDefault`. /// /// Note that the returned array is likely to be compatible with `VTKM_FIELD_TYPE_LIST`, but /// not guaranteed. In particular, if this field contains `Vec`s, the returned array will also /// contain `Vec`s of the same size. For example, if the field contains `vtkm::Vec2i_16` values, /// they will (likely) be converted to `vtkm::Vec2f`. Howver, `vtkm::Vec2f` may still not be /// in `VTKM_FIELD_TYPE_LIST`. /// VTKM_CONT vtkm::cont::UnknownArrayHandle GetDataWithExpectedTypes() const; /// \brief Convert this field to use an array of an expected type. /// /// Copies the internal data, as necessary, to an array that (probably) has a value type /// that matches something in `VTKM_FIELD_TYPE_LIST` and a storage that matches something /// in `VTKM_FIELD_STORAGE_LIST`. If the field already has a value type and storage that /// match `VTKM_FIELD_TYPE_LIST` and `VTKM_FIELD_STORAGE_LIST` respectively, then nothing /// in the field is changed. If something does not match, then the data are copied to a /// `vtkm::cont::ArrayHandleBasic` with a value type component of `vtkm::FloatDefault`. /// /// Note that the returned array is likely to be compatible with `VTKM_FIELD_TYPE_LIST`, but /// not guaranteed. In particular, if this field contains `Vec`s, the returned array will also /// contain `Vec`s of the same size. For example, if the field contains `vtkm::Vec2i_16` values, /// they will (likely) be converted to `vtkm::Vec2f`. Howver, `vtkm::Vec2f` may still not be /// in `VTKM_FIELD_TYPE_LIST`. /// VTKM_CONT void ConvertToExpected(); template VTKM_DEPRECATED(1.6, "TypeList no longer supported in Field::GetRange.") VTKM_CONT void GetRange(vtkm::Range* range, TypeList) const { this->GetRange(range); } template VTKM_DEPRECATED(1.6, "TypeList no longer supported in Field::GetRange.") VTKM_CONT const vtkm::cont::ArrayHandle& GetRange(TypeList) const { return this->GetRange(); } VTKM_CONT void SetData(const vtkm::cont::UnknownArrayHandle& newdata); template VTKM_CONT void SetData(const vtkm::cont::ArrayHandle& newdata) { this->SetData(vtkm::cont::UnknownArrayHandle(newdata)); } VTKM_CONT virtual void PrintSummary(std::ostream& out) const; VTKM_CONT virtual void ReleaseResourcesExecution() { this->Data.ReleaseResourcesExecution(); this->Range.ReleaseResourcesExecution(); } private: std::string Name; ///< name of field Association FieldAssociation = Association::ANY; vtkm::cont::UnknownArrayHandle Data; mutable vtkm::cont::ArrayHandle Range; mutable bool ModifiedFlag = true; }; template void CastAndCall(const vtkm::cont::Field& field, Functor&& f, Args&&... args) { vtkm::cont::CastAndCall(field.GetData(), std::forward(f), std::forward(args)...); } //@{ /// Convenience functions to build fields from C style arrays and std::vector template vtkm::cont::Field make_Field(std::string name, Field::Association association, const T* data, vtkm::Id size, vtkm::CopyFlag copy) { return vtkm::cont::Field(name, association, vtkm::cont::make_ArrayHandle(data, size, copy)); } template VTKM_DEPRECATED(1.6, "Specify a vtkm::CopyFlag or use a move version of make_Field.") vtkm::cont::Field make_Field(std::string name, Field::Association association, const T* data, vtkm::Id size) { return make_Field(name, association, data, size, vtkm::CopyFlag::Off); } template vtkm::cont::Field make_Field(std::string name, Field::Association association, const std::vector& data, vtkm::CopyFlag copy) { return vtkm::cont::Field(name, association, vtkm::cont::make_ArrayHandle(data, copy)); } template VTKM_DEPRECATED(1.6, "Specify a vtkm::CopyFlag or use a move version of make_Field.") vtkm::cont::Field make_Field(std::string name, Field::Association association, const std::vector& data) { return make_Field(name, association, data, vtkm::CopyFlag::Off); } template vtkm::cont::Field make_FieldMove(std::string name, Field::Association association, std::vector&& data) { return vtkm::cont::Field(name, association, vtkm::cont::make_ArrayHandleMove(data)); } template vtkm::cont::Field make_Field(std::string name, Field::Association association, std::vector&& data, vtkm::CopyFlag vtkmNotUsed(copy)) { return make_FieldMove(name, association, std::move(data)); } template vtkm::cont::Field make_Field(std::string name, Field::Association association, std::initializer_list&& data) { return make_FieldMove(name, association, vtkm::cont::make_ArrayHandle(std::move(data))); } //@} /// Convenience function to build point fields from vtkm::cont::ArrayHandle template vtkm::cont::Field make_FieldPoint(std::string name, const vtkm::cont::ArrayHandle& data) { return vtkm::cont::Field(name, vtkm::cont::Field::Association::POINTS, data); } /// Convenience function to build point fields from vtkm::cont::UnknownArrayHandle inline vtkm::cont::Field make_FieldPoint(std::string name, const vtkm::cont::UnknownArrayHandle& data) { return vtkm::cont::Field(name, vtkm::cont::Field::Association::POINTS, data); } /// Convenience function to build cell fields from vtkm::cont::ArrayHandle template vtkm::cont::Field make_FieldCell(std::string name, const vtkm::cont::ArrayHandle& data) { return vtkm::cont::Field(name, vtkm::cont::Field::Association::CELL_SET, data); } /// Convenience function to build cell fields from vtkm::cont::UnknownArrayHandle inline vtkm::cont::Field make_FieldCell(std::string name, const vtkm::cont::UnknownArrayHandle& data) { return vtkm::cont::Field(name, vtkm::cont::Field::Association::CELL_SET, data); } } // namespace cont } // namespace vtkm namespace vtkm { namespace cont { namespace internal { template <> struct DynamicTransformTraits { using DynamicTag = vtkm::cont::internal::DynamicTransformTagCastAndCall; }; } // namespace internal } // namespace cont } // namespace vtkm //============================================================================= // Specializations of serialization related classes /// @cond SERIALIZATION namespace vtkm { namespace cont { template struct VTKM_DEPRECATED(1.6, "You can now directly serialize Field.") SerializableField { SerializableField() = default; explicit SerializableField(const vtkm::cont::Field& field) : Field(field) { } vtkm::cont::Field Field; }; } // namespace cont } // namespace vtkm namespace mangled_diy_namespace { template <> struct VTKM_CONT_EXPORT Serialization { static VTKM_CONT void save(BinaryBuffer& bb, const vtkm::cont::Field& field); static VTKM_CONT void load(BinaryBuffer& bb, vtkm::cont::Field& field); }; // Implement deprecated code VTKM_DEPRECATED_SUPPRESS_BEGIN template struct Serialization> { private: using Type = vtkm::cont::SerializableField; public: static VTKM_CONT void save(BinaryBuffer& bb, const Type& serializable) { Serialization::save(bb, serializable.Field); } static VTKM_CONT void load(BinaryBuffer& bb, Type& serializable) { Serialization::load(bb, serializable.Field); } }; VTKM_DEPRECATED_SUPPRESS_END } // diy /// @endcond SERIALIZATION #endif //vtk_m_cont_Field_h