Skip to content
Snippets Groups Projects
ProcessVariable.cpp 13.8 KiB
Newer Older
  • Learn to ignore specific revisions
  •  * Copyright (c) 2012-2023, OpenGeoSys Community (http://www.opengeosys.org)
    
     *            Distributed under a Modified BSD License.
     *              See accompanying file LICENSE.txt or
     *              http://www.opengeosys.org/project/license
     *
     */
    
    
    #include "ProcessVariable.h"
    
    
    #include <utility>
    
    #include "BaseLib/Algorithm.h"
    
    #include "BaseLib/Logging.h"
    
    #include "MeshGeoToolsLib/ConstructMeshesFromGeometries.h"
    
    #include "MeshLib/Mesh.h"
    
    #include "MeshLib/Utils/getOrCreateMeshProperty.h"
    
    #include "ParameterLib/Utils.h"
    
    #include "ProcessLib/BoundaryConditionAndSourceTerm/BoundaryCondition.h"
    
    #include "ProcessLib/BoundaryConditionAndSourceTerm/BoundaryConditionConfig.h"
    
    #include "ProcessLib/BoundaryConditionAndSourceTerm/CreateBoundaryCondition.h"
    #include "ProcessLib/BoundaryConditionAndSourceTerm/CreateSourceTerm.h"
    #include "ProcessLib/BoundaryConditionAndSourceTerm/DeactivatedSubdomainDirichlet.h"
    #include "ProcessLib/BoundaryConditionAndSourceTerm/SourceTerm.h"
    
    #include "ProcessLib/BoundaryConditionAndSourceTerm/SourceTermConfig.h"
    
    #include "ProcessLib/CreateDeactivatedSubdomain.h"
    
    #include "ProcessLib/DeactivatedSubdomain.h"
    
    namespace
    {
    MeshLib::Mesh const& findMeshInConfig(
        BaseLib::ConfigTree const& config,
        std::vector<std::unique_ptr<MeshLib::Mesh>> const& meshes)
    {
        //
        // Get the mesh name from the config.
        //
        std::string mesh_name;  // Either given directly in <mesh> or constructed
                                // from <geometrical_set>_<geometry>.
    
    
    #ifdef DOXYGEN_DOCU_ONLY
        //! \ogs_file_param{prj__process_variables__process_variable__source_terms__source_term__mesh}
        config.getConfigParameterOptional<std::string>("mesh");
    #endif  // DOXYGEN_DOCU_ONLY
    
    
        auto optional_mesh_name =
            //! \ogs_file_param{prj__process_variables__process_variable__boundary_conditions__boundary_condition__mesh}
            config.getConfigParameterOptional<std::string>("mesh");
        if (optional_mesh_name)
        {
            mesh_name = *optional_mesh_name;
        }
        else
        {
    
    #ifdef DOXYGEN_DOCU_ONLY
            //! \ogs_file_param{prj__process_variables__process_variable__source_terms__source_term__geometrical_set}
            config.getConfigParameterOptional<std::string>("geometrical_set");
            //! \ogs_file_param{prj__process_variables__process_variable__source_terms__source_term__geometry}
            config.getConfigParameter<std::string>("geometry");
    #endif  // DOXYGEN_DOCU_ONLY
    
    
            // Looking for the mesh created before for the given geometry.
    
            auto const geometrical_set_name =
                //! \ogs_file_param{prj__process_variables__process_variable__boundary_conditions__boundary_condition__geometrical_set}
                config.getConfigParameter<std::string>("geometrical_set");
            auto const geometry_name =
                //! \ogs_file_param{prj__process_variables__process_variable__boundary_conditions__boundary_condition__geometry}
                config.getConfigParameter<std::string>("geometry");
    
    
            mesh_name = MeshGeoToolsLib::meshNameFromGeometry(geometrical_set_name,
                                                              geometry_name);
    
        }
    
        //
        // Find and extract mesh from the list of meshes.
        //
    
        auto const& mesh = *BaseLib::findElementOrError(
            begin(meshes), end(meshes),
    
    Lars Bilke's avatar
    Lars Bilke committed
            [&mesh_name](auto const& mesh)
            {
    
                assert(mesh != nullptr);
                return mesh->getName() == mesh_name;
    
            },
            "Required mesh with name '" + mesh_name + "' not found.");
    
        DBUG("Found mesh '{:s}' with id {:d}.", mesh.getName(), mesh.getID());
    
    
        return mesh;
    }
    }  // namespace
    
    
    namespace ProcessLib
    {
    
    bool parseCompensateNonEquilibriumInitialResiduum(
        BaseLib::ConfigTree const& config)
    {
        auto const compensate_non_equilibrium_initial_residuum_ptr =
            //! \ogs_file_param{prj__process_variables__process_variable__compensate_non_equilibrium_initial_residuum}
            config.getConfigParameterOptional<bool>(
                "compensate_non_equilibrium_initial_residuum");
    
        return (compensate_non_equilibrium_initial_residuum_ptr &&
                *compensate_non_equilibrium_initial_residuum_ptr);
    }
    
    
    ProcessVariable::ProcessVariable(
    
        BaseLib::ConfigTree const& config, MeshLib::Mesh& mesh,
    
        std::vector<std::unique_ptr<MeshLib::Mesh>> const& meshes,
    
        std::vector<std::unique_ptr<ParameterLib::ParameterBase>> const& parameters,
        std::map<std::string,
                 std::unique_ptr<MathLib::PiecewiseLinearInterpolation>> const&
            curves)
    
        :  //! \ogs_file_param{prj__process_variables__process_variable__name}
          _name(config.getConfigParameter<std::string>("name")),
    
          //! \ogs_file_param{prj__process_variables__process_variable__components}
    
          _n_components(config.getConfigParameter<int>("components")),
    
          //! \ogs_file_param{prj__process_variables__process_variable__order}
          _shapefunction_order(config.getConfigParameter<unsigned>("order")),
    
          _deactivated_subdomains(
    
              createDeactivatedSubdomains(config, mesh, parameters, curves)),
    
          _initial_condition(ParameterLib::findParameter<double>(
    
              //! \ogs_file_param{prj__process_variables__process_variable__initial_condition}
              config.getConfigParameter<std::string>("initial_condition"),
    
              parameters, _n_components, &mesh)),
          _compensate_non_equilibrium_initial_residuum(
              parseCompensateNonEquilibriumInitialResiduum(config))
    
        DBUG("Constructing process variable {:s}", _name);
    
        if (_shapefunction_order < 1 || 2 < _shapefunction_order)
    
    Dmitri Naumov's avatar
    Dmitri Naumov committed
            OGS_FATAL("The given shape function order {:d} is not supported",
    
        // Boundary conditions
    
                //! \ogs_file_param{prj__process_variables__process_variable__boundary_conditions}
    
            config.getConfigSubtreeOptional("boundary_conditions"))
    
            for (
                auto bc_config :
                //! \ogs_file_param{prj__process_variables__process_variable__boundary_conditions__boundary_condition}
                bcs_config->getConfigSubtreeList("boundary_condition"))
    
                auto const& bc_mesh = findMeshInConfig(bc_config, meshes);
    
                auto component_id =
    
                    //! \ogs_file_param{prj__process_variables__process_variable__boundary_conditions__boundary_condition__component}
    
                    bc_config.getConfigParameterOptional<int>("component");
    
    
                if (!component_id && _n_components == 1)
    
                    // default value for single component vars.
                    component_id = 0;
    
                _bc_configs.emplace_back(std::move(bc_config), bc_mesh,
                                         component_id);
    
    Dmitri Naumov's avatar
    Dmitri Naumov committed
            INFO("No boundary conditions for process variable '{:s}' found.",
    
    
        // Source terms
        //! \ogs_file_param{prj__process_variables__process_variable__source_terms}
        if (auto sts_config = config.getConfigSubtreeOptional("source_terms"))
        {
    
            for (
                auto st_config :
                //! \ogs_file_param{prj__process_variables__process_variable__source_terms__source_term}
                sts_config->getConfigSubtreeList("source_term"))
    
                MeshLib::Mesh const& st_mesh = findMeshInConfig(st_config, meshes);
    
                auto component_id =
                    //! \ogs_file_param{prj__process_variables__process_variable__source_terms__source_term__component}
                    st_config.getConfigParameterOptional<int>("component");
    
                if (!component_id && _n_components == 1)
    
                    // default value for single component vars.
                    component_id = 0;
    
                _source_term_configs.emplace_back(std::move(st_config), st_mesh,
    
            INFO("No source terms for process variable '{:s}' found.", _name);
    
        if (!_deactivated_subdomains.empty())
        {
            _is_active = getOrCreateMeshProperty<unsigned char>(
                _mesh, _name + "_active", MeshLib::MeshItemType::Cell, 1);
            std::fill(std::begin(*_is_active), std::end(*_is_active), 1u);
        }
    
    ProcessVariable::ProcessVariable(ProcessVariable&&) = default;
    
    
    std::string const& ProcessVariable::getName() const
    {
    
        return _name;
    
    MeshLib::Mesh const& ProcessVariable::getMesh() const
    {
    
        return _mesh;
    
    std::vector<std::unique_ptr<BoundaryCondition>>
    
    ProcessVariable::createBoundaryConditions(
    
        const NumLib::LocalToGlobalIndexMap& dof_table,
        const int variable_id,
    
        unsigned const integration_order,
    
        std::vector<std::unique_ptr<ParameterLib::ParameterBase>> const& parameters,
    
        Process const& process,
        std::vector<std::reference_wrapper<ProcessVariable>> const&
    
            all_process_variables_for_this_process,
        std::map<int, std::shared_ptr<MaterialPropertyLib::Medium>> const& media)
    
    {
        std::vector<std::unique_ptr<BoundaryCondition>> bcs;
    
        bcs.reserve(_bc_configs.size());
    
        for (auto const& config : _bc_configs)
    
            auto bc = createBoundaryCondition(
                config, dof_table, _mesh, variable_id, integration_order,
    
                _shapefunction_order, parameters, process,
    
                all_process_variables_for_this_process, media);
    
    #ifdef USE_PETSC
            if (bc == nullptr)
            {
                continue;
            }
    #endif  // USE_PETSC
    
            bcs.push_back(std::move(bc));
        }
    
        createBoundaryConditionsForDeactivatedSubDomains(dof_table, variable_id,
                                                         parameters, bcs);
    
    void ProcessVariable::createBoundaryConditionsForDeactivatedSubDomains(
        const NumLib::LocalToGlobalIndexMap& dof_table, const int variable_id,
    
        std::vector<std::unique_ptr<ParameterLib::ParameterBase>> const& parameters,
    
        std::vector<std::unique_ptr<BoundaryCondition>>& bcs)
    {
        for (auto const& deactivated_subdomain : _deactivated_subdomains)
        {
    
            auto const& deactivated_subdomain_mesh =
    
                deactivated_subdomain.deactivated_subdomain_mesh;
    
            auto const* parameter = &ParameterLib::findParameter<double>(
                DeactivatedSubdomain::zero_parameter_name, parameters, 1);
            bool const set_outer_nodes_dirichlet_values =
    
                deactivated_subdomain.boundary_value_parameter != nullptr;
    
            if (set_outer_nodes_dirichlet_values)
    
                parameter = deactivated_subdomain.boundary_value_parameter;
    
            for (int component_id = 0;
                 component_id <
                 dof_table.getNumberOfVariableComponents(variable_id);
                 component_id++)
            {
                auto bc = std::make_unique<DeactivatedSubdomainDirichlet>(
    
                    *_is_active, deactivated_subdomain.time_interval, *parameter,
    
                    set_outer_nodes_dirichlet_values, deactivated_subdomain_mesh,
                    dof_table, variable_id, component_id);
    
                // TODO: make it work under PETSc too.
                if (bc == nullptr)
                {
                    continue;
    
    #endif  // USE_PETSC
                bcs.push_back(std::move(bc));
    
    void ProcessVariable::updateDeactivatedSubdomains(double const time)
    
        if (_deactivated_subdomains.empty())
        {
            return;
        }
    
    
        _ids_of_active_elements.clear();
    
    
        // If none of the deactivated subdomains is active at current time, then the
        // _ids_of_active_elements remain empty.
        if (std::none_of(
                begin(_deactivated_subdomains), end(_deactivated_subdomains),
    
                [&](auto const& ds) { return ds.isInTimeSupportInterval(time); }))
    
            // Also mark all of the elements as active.
            assert(_is_active != nullptr);  // guaranteed by constructor
            std::fill(std::begin(*_is_active), std::end(*_is_active), 1u);
    
    
        auto is_active_in_subdomain = [&](std::size_t const element_id,
    
    Lars Bilke's avatar
    Lars Bilke committed
                                          DeactivatedSubdomain const& ds) -> bool
        {
    
            return (!ds.isInTimeSupportInterval(time)) ||
                   !ds.isDeactivated(*_mesh.getElement(element_id), time);
    
        auto is_active_in_all_subdomains = [&](std::size_t const element_id) -> bool
        {
            return std::all_of(begin(_deactivated_subdomains),
                               end(_deactivated_subdomains),
                               [&](auto const& ds)
                               { return is_active_in_subdomain(element_id, ds); });
        };
    
    
        auto const number_of_elements = _mesh.getNumberOfElements();
    
        for (std::size_t element_id = 0; element_id < number_of_elements;
             element_id++)
    
            if (is_active_in_all_subdomains(element_id))
    
                _ids_of_active_elements.push_back(element_id);
    
    
        // all elements are deactivated
        std::fill(std::begin(*_is_active), std::end(*_is_active), 0u);
    
        for (auto const id : _ids_of_active_elements)
        {
            (*_is_active)[id] = 1u;
        }
    
    std::vector<std::unique_ptr<SourceTerm>> ProcessVariable::createSourceTerms(
    
        const NumLib::LocalToGlobalIndexMap& dof_table,
        const int variable_id,
    
        unsigned const integration_order,
    
        std::vector<std::unique_ptr<ParameterLib::ParameterBase>> const& parameters,
        std::vector<std::reference_wrapper<ProcessVariable>> const&
            all_process_variables_for_this_process)
    
        std::vector<std::unique_ptr<SourceTerm>> source_terms;
    
        transform(cbegin(_source_term_configs), cend(_source_term_configs),
    
    Lars Bilke's avatar
    Lars Bilke committed
                  back_inserter(source_terms),
                  [&](auto const& config)
                  {
    
                      return createSourceTerm(
                          config, dof_table, config.mesh, variable_id,
                          integration_order, _shapefunction_order, parameters,
                          all_process_variables_for_this_process);
    
    ProcessVariable::~ProcessVariable() = default;
    
    
    }  // namespace ProcessLib