// -*- mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- // vi: set et ts=4 sw=4 sts=4: /* This file is part of the Open Porous Media project (OPM). OPM is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 2 of the License, or (at your option) any later version. OPM is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with OPM. If not, see . Consult the COPYING file in the top-level source directory of this module for the precise wording of the license and the list of copyright holders. */ /*! * \file * * \copydoc Opm::EcfvDiscretization */ #ifndef EWOMS_ECFV_DISCRETIZATION_HH #define EWOMS_ECFV_DISCRETIZATION_HH #include #include "ecfvproperties.hh" #include "ecfvstencil.hh" #include "ecfvgridcommhandlefactory.hh" #include "ecfvbaseoutputmodule.hh" #include #include #if HAVE_DUNE_FEM #include #include #include #endif namespace Opm { template class EcfvDiscretization; } namespace Opm::Properties { //! Set the stencil template struct Stencil { private: using Scalar = GetPropType; using GridView = GetPropType; public: using type = EcfvStencil; }; //! Mapper for the degrees of freedoms. template struct DofMapper { using type = GetPropType; }; //! The concrete class which manages the spatial discretization template struct Discretization { using type = EcfvDiscretization; }; //! The base class for the output modules (decides whether to write //! element or vertex based fields) template struct DiscBaseOutputModule { using type = EcfvBaseOutputModule; }; //! The class to create grid communication handles template struct GridCommHandleFactory { using type = EcfvGridCommHandleFactory; }; #if HAVE_DUNE_FEM //! Set the DiscreteFunctionSpace template struct DiscreteFunctionSpace { private: using Scalar = GetPropType ; using GridPart = GetPropType; enum { numEq = getPropValue() }; using FunctionSpace = Dune::Fem::FunctionSpace; public: using type = Dune::Fem::FiniteVolumeSpace< FunctionSpace, GridPart, 0 >; }; #else template struct DummySpaceEcfv { using DiscreteFunctionSpace = GetPropType; DummySpaceEcfv(const DiscreteFunctionSpace&) {}; DummySpaceEcfv(const int&) {}; }; template struct DiscreteFunctionSpace { using type = DummySpaceEcfv; }; #endif //! Set the border list creator for to the one of an element based //! method template struct BorderListCreator { private: using ElementMapper = GetPropType; using GridView = GetPropType; public: using type = Linear::ElementBorderListFromGrid; }; //! For the element centered finite volume method, ghost and overlap elements must be //! assembled to calculate the fluxes over the process boundary faces of the local //! process' grid partition template struct LinearizeNonLocalElements { static constexpr bool value = true; }; //! locking is not required for the element centered finite volume method because race //! conditions cannot occur since each matrix/vector entry is written exactly once template struct UseLinearizationLock { static constexpr bool value = false; }; } // namespace Opm::Properties namespace Opm { /*! * \ingroup EcfvDiscretization * * \brief The base class for the element-centered finite-volume discretization scheme. */ template class EcfvDiscretization : public GetPropType { using ParentType = GetPropType; using Implementation = GetPropType; using DofMapper = GetPropType; using PrimaryVariables = GetPropType; using SolutionVector = GetPropType; using GridView = GetPropType; using Simulator = GetPropType; public: EcfvDiscretization(Simulator& simulator) : ParentType(simulator) { } /*! * \brief Returns a string of discretization's human-readable name */ static std::string discretizationName() { return "ecfv"; } /*! * \brief Returns the number of global degrees of freedom (DOFs) due to the grid */ size_t numGridDof() const { return static_cast(this->gridView_.size(/*codim=*/0)); } /*! * \brief Mapper to convert the Dune entities of the * discretization's degrees of freedoms are to indices. */ const DofMapper& dofMapper() const { return this->elementMapper(); } /*! * \brief Syncronize the values of the primary variables on the * degrees of freedom that overlap with the neighboring * processes. * * For the Element Centered Finite Volume discretization, this * method retrieves the primary variables corresponding to * overlap/ghost elements from their respective master process. */ void syncOverlap() { // syncronize the solution on the ghost and overlap elements using GhostSyncHandle = GridCommHandleGhostSync; auto ghostSync = GhostSyncHandle(this->solution(/*timeIdx=*/0), asImp_().dofMapper()); this->gridView().communicate(ghostSync, Dune::InteriorBorder_All_Interface, Dune::ForwardCommunication); } /*! * \brief Serializes the current state of the model. * * \tparam Restarter The type of the serializer class * * \param res The serializer object */ template void serialize(Restarter& res) { res.template serializeEntities(asImp_(), this->gridView_); } /*! * \brief Deserializes the state of the model. * * \tparam Restarter The type of the serializer class * * \param res The serializer object */ template void deserialize(Restarter& res) { res.template deserializeEntities(asImp_(), this->gridView_); this->solution(/*timeIdx=*/1) = this->solution(/*timeIdx=*/0); } private: Implementation& asImp_() { return *static_cast(this); } const Implementation& asImp_() const { return *static_cast(this); } }; } // namespace Opm #endif