diff --git a/ProcessLib/LIE/HydroMechanics/LocalAssembler/CreateLocalAssemblers.h b/ProcessLib/LIE/HydroMechanics/LocalAssembler/CreateLocalAssemblers.h new file mode 100644 index 0000000000000000000000000000000000000000..f005ccc05be7755131303d000593308b65a55811 --- /dev/null +++ b/ProcessLib/LIE/HydroMechanics/LocalAssembler/CreateLocalAssemblers.h @@ -0,0 +1,105 @@ +/** + * \copyright + * Copyright (c) 2012-2016, OpenGeoSys Community (http://www.opengeosys.org) + * Distributed under a Modified BSD License. + * See accompanying file LICENSE.txt or + * http://www.opengeosys.org/project/license + */ + +#ifndef PROCESSLIB_LIE_HYDROMECHANICS_CREATE_LOCAL_ASSEMBLERS_H_ +#define PROCESSLIB_LIE_HYDROMECHANICS_CREATE_LOCAL_ASSEMBLERS_H_ + +#include <vector> + +#include <logog/include/logog.hpp> + +#include "NumLib/DOF/LocalToGlobalIndexMap.h" + +#include "LocalDataInitializer.h" + +namespace ProcessLib +{ +namespace LIE +{ +namespace HydroMechanics +{ +namespace detail +{ +template <unsigned GlobalDim, + template <typename, typename, typename, unsigned> + class LocalAssemblerMatrixImplementation, + template <typename, typename, typename, unsigned> + class LocalAssemblerMatrixNearFractureImplementation, + template <typename, typename, typename, unsigned> + class LocalAssemblerFractureImplementation, + typename LocalAssemblerInterface, typename... ExtraCtorArgs> +void createLocalAssemblers( + NumLib::LocalToGlobalIndexMap const& dof_table, + const unsigned shapefunction_order, + std::vector<MeshLib::Element*> const& mesh_elements, + std::vector<std::unique_ptr<LocalAssemblerInterface>>& local_assemblers, + ExtraCtorArgs&&... extra_ctor_args) +{ + // Shape matrices initializer + using LocalDataInitializer = + LocalDataInitializer<LocalAssemblerInterface, + LocalAssemblerMatrixImplementation, + LocalAssemblerMatrixNearFractureImplementation, + LocalAssemblerFractureImplementation, + GlobalDim, ExtraCtorArgs...>; + + DBUG("Create local assemblers."); + // Populate the vector of local assemblers. + local_assemblers.resize(mesh_elements.size()); + + LocalDataInitializer initializer(dof_table, shapefunction_order); + + DBUG("Calling local assembler builder for all mesh elements."); + GlobalExecutor::transformDereferenced( + initializer, mesh_elements, local_assemblers, + std::forward<ExtraCtorArgs>(extra_ctor_args)...); +} + +} // namespace detail + +/*! Creates local assemblers for each element of the given \c mesh. + * + * \tparam LocalAssemblerImplementation the individual local assembler type + * \tparam LocalAssemblerInterface the general local assembler interface + * \tparam ExtraCtorArgs types of additional constructor arguments. + * Those arguments will be passed to the constructor of + * \c LocalAssemblerImplementation. + * + * The first two template parameters cannot be deduced from the arguments. + * Therefore they always have to be provided manually. + */ +template <unsigned GlobalDim, + template <typename, typename, typename, unsigned> + class LocalAssemblerMatrixImplementation, + template <typename, typename, typename, unsigned> + class LocalAssemblerMatrixNearFractureImplementation, + template <typename, typename, typename, unsigned> + class LocalAssemblerFractureImplementation, + typename LocalAssemblerInterface, typename... ExtraCtorArgs> +void createLocalAssemblers( + std::vector<MeshLib::Element*> const& mesh_elements, + NumLib::LocalToGlobalIndexMap const& dof_table, + const unsigned shapefunction_order, + std::vector<std::unique_ptr<LocalAssemblerInterface>>& local_assemblers, + ExtraCtorArgs&&... extra_ctor_args) +{ + DBUG("Create local assemblers."); + + detail::createLocalAssemblers<GlobalDim, + LocalAssemblerMatrixImplementation, + LocalAssemblerMatrixNearFractureImplementation, + LocalAssemblerFractureImplementation>( + dof_table, shapefunction_order, mesh_elements, local_assemblers, + std::forward<ExtraCtorArgs>(extra_ctor_args)...); +} + +} // namespace HydroMechanics +} // namespace LIE +} // namespace ProcessLib + +#endif // PROCESSLIB_LIE_HYDROMECHANICS_CREATE_LOCAL_ASSEMBLERS_H_ diff --git a/ProcessLib/LIE/HydroMechanics/LocalAssembler/LocalDataInitializer.h b/ProcessLib/LIE/HydroMechanics/LocalAssembler/LocalDataInitializer.h new file mode 100644 index 0000000000000000000000000000000000000000..e6f8ebe2433bda00777bbc7adfabe2513cf8f9b5 --- /dev/null +++ b/ProcessLib/LIE/HydroMechanics/LocalAssembler/LocalDataInitializer.h @@ -0,0 +1,369 @@ +/** + * \copyright + * Copyright (c) 2012-2016, OpenGeoSys Community (http://www.opengeosys.org) + * Distributed under a Modified BSD License. + * See accompanying file LICENSE.txt or + * http://www.opengeosys.org/project/license + * + */ + +#ifndef PROCESSLIB_LIE_HYDROMECHANICS_LOCALDATAINITIALIZER_H_ +#define PROCESSLIB_LIE_HYDROMECHANICS_LOCALDATAINITIALIZER_H_ + +#include <functional> +#include <memory> +#include <typeindex> +#include <typeinfo> +#include <type_traits> +#include <unordered_map> + +#include "MeshLib/Elements/Elements.h" +#include "NumLib/DOF/LocalToGlobalIndexMap.h" +#include "NumLib/Fem/FiniteElement/LowerDimShapeTable.h" +#include "NumLib/Fem/Integration/GaussIntegrationPolicy.h" + +#ifndef OGS_MAX_ELEMENT_DIM +static_assert(false, "The macro OGS_MAX_ELEMENT_DIM is undefined."); +#endif + +#ifndef OGS_MAX_ELEMENT_ORDER +static_assert(false, "The macro OGS_MAX_ELEMENT_ORDER is undefined."); +#endif + +// The following macros decide which element types will be compiled, i.e. +// which element types will be available for use in simulations. + +#ifdef OGS_ENABLE_ELEMENT_SIMPLEX +#define ENABLED_ELEMENT_TYPE_SIMPLEX 1u +#else +#define ENABLED_ELEMENT_TYPE_SIMPLEX 0u +#endif + +#ifdef OGS_ENABLE_ELEMENT_CUBOID +#define ENABLED_ELEMENT_TYPE_CUBOID 1u << 1 +#else +#define ENABLED_ELEMENT_TYPE_CUBOID 0u +#endif + +#ifdef OGS_ENABLE_ELEMENT_PRISM +#define ENABLED_ELEMENT_TYPE_PRISM 1u << 2 +#else +#define ENABLED_ELEMENT_TYPE_PRISM 0u +#endif + +#ifdef OGS_ENABLE_ELEMENT_PYRAMID +#define ENABLED_ELEMENT_TYPE_PYRAMID 1u << 3 +#else +#define ENABLED_ELEMENT_TYPE_PYRAMID 0u +#endif + +// Dependent element types. +// Faces of tets, pyramids and prisms are triangles +#define ENABLED_ELEMENT_TYPE_TRI \ + ((ENABLED_ELEMENT_TYPE_SIMPLEX) | (ENABLED_ELEMENT_TYPE_PYRAMID) | \ + (ENABLED_ELEMENT_TYPE_PRISM)) +// Faces of hexes, pyramids and prisms are quads +#define ENABLED_ELEMENT_TYPE_QUAD \ + ((ENABLED_ELEMENT_TYPE_CUBOID) | (ENABLED_ELEMENT_TYPE_PYRAMID) | \ + (ENABLED_ELEMENT_TYPE_PRISM)) + +// All enabled element types +#define OGS_ENABLED_ELEMENTS \ + ((ENABLED_ELEMENT_TYPE_SIMPLEX) | (ENABLED_ELEMENT_TYPE_CUBOID) | \ + (ENABLED_ELEMENT_TYPE_PYRAMID) | (ENABLED_ELEMENT_TYPE_PRISM)) + + +// Include only what is needed (Well, the conditions are not sharp). +#if OGS_ENABLED_ELEMENTS != 0 +#include "NumLib/Fem/ShapeFunction/ShapeLine2.h" +#include "NumLib/Fem/ShapeFunction/ShapeLine3.h" +#endif + +#if (OGS_ENABLED_ELEMENTS & ENABLED_ELEMENT_TYPE_SIMPLEX) != 0 +#include "NumLib/Fem/ShapeFunction/ShapeTet10.h" +#include "NumLib/Fem/ShapeFunction/ShapeTet4.h" +#endif + +#if (OGS_ENABLED_ELEMENTS & ENABLED_ELEMENT_TYPE_TRI) != 0 +#include "NumLib/Fem/ShapeFunction/ShapeTri3.h" +#include "NumLib/Fem/ShapeFunction/ShapeTri6.h" +#endif + +#if (OGS_ENABLED_ELEMENTS & ENABLED_ELEMENT_TYPE_CUBOID) != 0 +#include "NumLib/Fem/ShapeFunction/ShapeHex20.h" +#include "NumLib/Fem/ShapeFunction/ShapeHex8.h" +#endif + +#if (OGS_ENABLED_ELEMENTS & ENABLED_ELEMENT_TYPE_QUAD) != 0 +#include "NumLib/Fem/ShapeFunction/ShapeQuad4.h" +#include "NumLib/Fem/ShapeFunction/ShapeQuad8.h" +#include "NumLib/Fem/ShapeFunction/ShapeQuad9.h" +#endif + +#if (OGS_ENABLED_ELEMENTS & ENABLED_ELEMENT_TYPE_PRISM) != 0 +#include "NumLib/Fem/ShapeFunction/ShapePrism15.h" +#include "NumLib/Fem/ShapeFunction/ShapePrism6.h" +#endif + +#if (OGS_ENABLED_ELEMENTS & ENABLED_ELEMENT_TYPE_PYRAMID) != 0 +#include "NumLib/Fem/ShapeFunction/ShapePyra13.h" +#include "NumLib/Fem/ShapeFunction/ShapePyra5.h" +#endif + +namespace ProcessLib +{ +namespace LIE +{ +namespace HydroMechanics +{ + +/// The LocalDataInitializer is a functor creating a local assembler data with +/// corresponding to the mesh element type shape functions and calling +/// initialization of the new local assembler data. +/// For example for MeshLib::Quad a local assembler data with template argument +/// NumLib::ShapeQuad4 is created. +/// +/// \attention This is modified version of the ProcessLib::LocalDataInitializer +/// class which does not include line elements, allows only shapefunction of +/// order 2, and provides additional template argument GlobalDim. +template < + typename LocalAssemblerInterface, + template <typename, typename, typename, unsigned> class LocalAssemblerDataMatrix, + template <typename, typename, typename, unsigned> class LocalAssemblerDataMatrixNearFracture, + template <typename, typename, typename, unsigned> class LocalAssemblerDataFracture, + unsigned GlobalDim, typename... ConstructorArgs> +class LocalDataInitializer final +{ +public: + using LADataIntfPtr = std::unique_ptr<LocalAssemblerInterface>; + + LocalDataInitializer(NumLib::LocalToGlobalIndexMap const& dof_table, + const unsigned shapefunction_order) + : _dof_table(dof_table) + { + if (shapefunction_order != 2) + OGS_FATAL( + "The given shape function order %d is not supported.\nOnly " + "shape functions of order 2 are supported.", + shapefunction_order); + // /// Quads and Hexahedra /////////////////////////////////// + +#if (OGS_ENABLED_ELEMENTS & ENABLED_ELEMENT_TYPE_QUAD) != 0 && \ + OGS_MAX_ELEMENT_DIM >= 2 && OGS_MAX_ELEMENT_ORDER >= 2 + _builder[std::type_index(typeid(MeshLib::Quad8))] = + makeLocalAssemblerBuilder<NumLib::ShapeQuad8>(); + _builder[std::type_index(typeid(MeshLib::Quad9))] = + makeLocalAssemblerBuilder<NumLib::ShapeQuad9>(); +#endif + + // /// Simplices //////////////////////////////////////////////// + +#if (OGS_ENABLED_ELEMENTS & ENABLED_ELEMENT_TYPE_TRI) != 0 && \ + OGS_MAX_ELEMENT_DIM >= 2 && OGS_MAX_ELEMENT_ORDER >= 2 + _builder[std::type_index(typeid(MeshLib::Tri6))] = + makeLocalAssemblerBuilder<NumLib::ShapeTri6>(); +#endif + + // /// Lines /////////////////////////////////// +#if OGS_MAX_ELEMENT_DIM >= 3 && OGS_MAX_ELEMENT_ORDER >= 2 + _builder[std::type_index(typeid(MeshLib::Line3))] = + makeLocalAssemblerBuilder<NumLib::ShapeLine3>(); +#endif + } + + /// Sets the provided \c data_ptr to the newly created local assembler data. + /// + /// \attention + /// The index \c id is not necessarily the mesh item's id. Especially when + /// having multiple meshes it will differ from the latter. + void operator()( + std::size_t const id, + MeshLib::Element const& mesh_item, + LADataIntfPtr& data_ptr, + ConstructorArgs&&... args) const + { + auto const type_idx = std::type_index(typeid(mesh_item)); + auto const it = _builder.find(type_idx); + + if (it != _builder.end()) + { + auto const n_local_dof = _dof_table.getNumberOfElementDOF(id); + auto const n_global_components = _dof_table.getNumberOfElementComponents(id); + const std::vector<std::size_t> varIDs(_dof_table.getElementVariableIDs(id)); + + std::vector<unsigned> dofIndex_to_localIndex; + if (mesh_item.getDimension() == GlobalDim + && n_global_components == GlobalDim + 1) // +1 for pressure + { + // normal matrix assembler + dofIndex_to_localIndex.resize(n_local_dof); + int count = 0; + std::generate(dofIndex_to_localIndex.begin(), dofIndex_to_localIndex.end(), [&](){return count++;}); + } + else + { + // matrix and fracture assemblers with enrichments + dofIndex_to_localIndex.resize(n_local_dof); + unsigned dof_id = 0; + unsigned local_id = 0; + std::vector<unsigned> vec_n_element_nodes; + //TODO how to get the shape function order for each variable? + vec_n_element_nodes.push_back(mesh_item.getNumberOfBaseNodes()); // pressure + for (unsigned i=1; i<varIDs.size(); i++) + vec_n_element_nodes.push_back(mesh_item.getNumberOfNodes()); // displacements + + for (unsigned i=0; i<varIDs.size(); i++) + { + auto const var_id = varIDs[i]; + auto const n_var_comp = _dof_table.getNumberOfVariableComponents(var_id); + auto const n_var_element_nodes = vec_n_element_nodes[i]; + for (unsigned var_comp_id = 0; var_comp_id < n_var_comp; var_comp_id++) + { + auto& mss = _dof_table.getMeshSubsets(var_id, var_comp_id); + assert(mss.size() == 1); + auto mesh_id = mss.getMeshSubset(0).getMeshID(); + for (unsigned k=0; k<n_var_element_nodes; k++) + { + MeshLib::Location l(mesh_id, + MeshLib::MeshItemType::Node, + mesh_item.getNodeIndex(k)); + auto global_index = _dof_table.getGlobalIndex(l, var_id, var_comp_id); + if (global_index != NumLib::MeshComponentMap::nop) + dofIndex_to_localIndex[dof_id++] = local_id; + local_id++; + } + } + } + + } + + + data_ptr = it->second( + mesh_item, varIDs.size(), n_local_dof, dofIndex_to_localIndex, + std::forward<ConstructorArgs>(args)...); + } else { + OGS_FATAL("You are trying to build a local assembler for an unknown mesh element type (%s)." + " Maybe you have disabled this mesh element type in your build configuration.", + type_idx.name()); + } + } + +private: + using LADataBuilder = std::function<LADataIntfPtr( + MeshLib::Element const& e, + std::size_t const n_variables, + std::size_t const local_matrix_size, + std::vector<unsigned> const& dofIndex_to_localIndex, + ConstructorArgs&&... + )>; + + template <typename ShapeFunctionDisplacement> + using IntegrationMethod = typename NumLib::GaussIntegrationPolicy< + typename ShapeFunctionDisplacement::MeshElement>::IntegrationMethod; + + template <typename ShapeFunctionDisplacement, + typename ShapeFunctionPressure> + using LADataMatrix = + LocalAssemblerDataMatrix<ShapeFunctionDisplacement, + ShapeFunctionPressure, + IntegrationMethod<ShapeFunctionDisplacement>, + GlobalDim>; + + template <typename ShapeFunctionDisplacement, + typename ShapeFunctionPressure> + using LADataMatrixNearFracture = LocalAssemblerDataMatrixNearFracture< + ShapeFunctionDisplacement, ShapeFunctionPressure, + IntegrationMethod<ShapeFunctionDisplacement>, GlobalDim>; + + template <typename ShapeFunctionDisplacement, + typename ShapeFunctionPressure> + using LAFractureData = + LocalAssemblerDataFracture<ShapeFunctionDisplacement, + ShapeFunctionPressure, + IntegrationMethod<ShapeFunctionDisplacement>, + GlobalDim>; + + /// A helper forwarding to the correct version of makeLocalAssemblerBuilder + /// depending whether the global dimension is less than the shape function's + /// dimension or not. + template <typename ShapeFunctionDisplacement> + static LADataBuilder makeLocalAssemblerBuilder() + { + return makeLocalAssemblerBuilder<ShapeFunctionDisplacement>( + static_cast<std::integral_constant< + bool, (GlobalDim >= ShapeFunctionDisplacement::DIM)>*>( + nullptr)); + } + + /// Mapping of element types to local assembler constructors. + std::unordered_map<std::type_index, LADataBuilder> _builder; + + NumLib::LocalToGlobalIndexMap const& _dof_table; + + // local assembler builder implementations. +private: + /// Generates a function that creates a new LocalAssembler of type + /// LAData<ShapeFunctionDisplacement>. Only functions with shape function's + /// dimension less or equal to the global dimension are instantiated, e.g. + /// following combinations of shape functions and global dimensions: (Line2, + /// 1), + /// (Line2, 2), (Line2, 3), (Hex20, 3) but not (Hex20, 2) or (Hex20, 1). + template <typename ShapeFunctionDisplacement> + static LADataBuilder makeLocalAssemblerBuilder(std::true_type*) + { + // (Lower order elements = Order(ShapeFunctionDisplacement) - 1). + using ShapeFunctionPressure = + typename NumLib::LowerDim<ShapeFunctionDisplacement>::type; + return [](MeshLib::Element const& e, + std::size_t const n_variables, + std::size_t const local_matrix_size, + std::vector<unsigned> const& dofIndex_to_localIndex, + ConstructorArgs&&... args) + { + if (e.getDimension() == GlobalDim) + { + if (n_variables == 2) + { + return LADataIntfPtr{ + new LADataMatrix<ShapeFunctionDisplacement, ShapeFunctionPressure>{ + e, n_variables, local_matrix_size, dofIndex_to_localIndex, + std::forward<ConstructorArgs>(args)... + }}; + } else { + return LADataIntfPtr{ + new LADataMatrixNearFracture<ShapeFunctionDisplacement, ShapeFunctionPressure>{ + e, n_variables, local_matrix_size, dofIndex_to_localIndex, + std::forward<ConstructorArgs>(args)... + }}; + } + } + return LADataIntfPtr{ + new LAFractureData<ShapeFunctionDisplacement, ShapeFunctionPressure>{ + e, local_matrix_size, dofIndex_to_localIndex, + std::forward<ConstructorArgs>(args)... + }}; + }; + } + + /// Returns nullptr for shape functions whose dimensions are less than the + /// global dimension. + template <typename ShapeFunctionDisplacement> + static LADataBuilder makeLocalAssemblerBuilder(std::false_type*) + { + return nullptr; + } +}; + +} // namespace HydroMechanics +} // namespace LIE +} // namespace ProcessLib + +#undef ENABLED_ELEMENT_TYPE_SIMPLEX +#undef ENABLED_ELEMENT_TYPE_CUBOID +#undef ENABLED_ELEMENT_TYPE_PYRAMID +#undef ENABLED_ELEMENT_TYPE_PRISM +#undef ENABLED_ELEMENT_TYPE_TRI +#undef ENABLED_ELEMENT_TYPE_QUAD +#undef OGS_ENABLED_ELEMENTS + +#endif // PROCESSLIB_LIE_HYDROMECHANICS_LOCALDATAINITIALIZER_H_