// -*- 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::EclTransmissibility */ #ifndef EWOMS_ECL_TRANSMISSIBILITY_HH #define EWOMS_ECL_TRANSMISSIBILITY_HH #include #include #include #if HAVE_DUNE_ALUGRID #include #include #include "alucartesianindexmapper.hh" #endif // HAVE_DUNE_ALUGRID #include #include #include #include #include #include namespace Opm { class EclipseState; struct NNCdata; class TransMult; template class EclTransmissibility { // Grid and world dimension enum { dimWorld = GridView::dimensionworld }; public: using DimMatrix = Dune::FieldMatrix; using DimVector = Dune::FieldVector; EclTransmissibility(const EclipseState& eclState, const GridView& gridView, const CartesianIndexMapper& cartMapper, const Grid& grid, std::function(int)> centroids, bool enableEnergy, bool enableDiffusivity); /*! * \brief Return the permeability for an element. */ const DimMatrix& permeability(unsigned elemIdx) const { return permeability_[elemIdx]; } /*! * \brief Return the transmissibility for the intersection between two elements. */ Scalar transmissibility(unsigned elemIdx1, unsigned elemIdx2) const; /*! * \brief Return the transmissibility for a given boundary segment. */ Scalar transmissibilityBoundary(unsigned elemIdx, unsigned boundaryFaceIdx) const; /*! * \brief Return the thermal "half transmissibility" for the intersection between two * elements. * * The "half transmissibility" features all sub-expressions of the "thermal * transmissibility" which can be precomputed, i.e. they are not dependent on the * current solution: * * H_t = A * (n*d)/(d*d); * * where A is the area of the intersection between the inside and outside elements, n * is the outer unit normal, and d is the distance between the center of the inside * cell and the center of the intersection. */ Scalar thermalHalfTrans(unsigned insideElemIdx, unsigned outsideElemIdx) const; Scalar thermalHalfTransBoundary(unsigned insideElemIdx, unsigned boundaryFaceIdx) const; /*! * \brief Return the diffusivity for the intersection between two elements. */ Scalar diffusivity(unsigned elemIdx1, unsigned elemIdx2) const; /*! * \brief Actually compute the transmissibility over a face as a pre-compute step. * * This code actually uses the direction specific "centroids" of * each element. These "centroids" are _not_ the identical * barycenter of the element, but the middle of the centers of the * faces of the logical Cartesian cells, i.e., the centers of the * faces of the reference elements. We do things this way because * the barycenter of the element can be located outside of the * element for sufficiently "ugly" (i.e., thin and "non-flat") * elements which in turn leads to quite wrong * permeabilities. This approach is probably not always correct * either but at least it seems to be much better. */ void finishInit(const std::function& map = {}) { update(true,map); } /*! * \brief Compute all transmissibilities * * \param global If true, update is called on all processes * Also, this updates the "thermal half transmissibilities" if energy is enabled. */ void update(bool global, const std::function& map = {}); protected: void updateFromEclState_(bool global); void removeSmallNonCartesianTransmissibilities_(); /// \brief Apply the Multipliers for the case PINCH(4)==TOPBOT /// /// \param pinchTop Whether PINCH(5) is TOP, otherwise ALL is assumed. void applyAllZMultipliers_(Scalar& trans, unsigned insideFaceIdx, unsigned outsideFaceIdx, unsigned insideCartElemIdx, unsigned outsideCartElemIdx, const TransMult& transMult, const std::array& cartDims, bool pinchTop); /// \brief Creates TRANS{XYZ} arrays for modification by FieldProps data /// /// \param is_tran Whether TRAN{XYZ} will be modified. If entry is false the array will be empty /// \returns an array of vector (TRANX, TRANY, TRANZ} std::array,3> createTransmissibilityArrays_(const std::array& is_tran); /// \brief overwrites calculated transmissibilities /// /// \param is_tran Whether TRAN{XYZ} have been modified. /// \param trans Arrays with modified transmissibilities TRAN{XYZ} void resetTransmissibilityFromArrays_(const std::array& is_tran, const std::array,3>& trans); template void computeFaceProperties(const Intersection& intersection, const int, const int, const int, const int, DimVector& faceCenterInside, DimVector& faceCenterOutside, DimVector& faceAreaNormal, /*isCpGrid=*/std::false_type) const; template void computeFaceProperties(const Intersection& intersection, const int insideElemIdx, const int insideFaceIdx, const int outsideElemIdx, const int outsideFaceIdx, DimVector& faceCenterInside, DimVector& faceCenterOutside, DimVector& faceAreaNormal, /*isCpGrid=*/std::true_type) const; /* * \brief Applies additional transmissibilities specified via NNC keyword. * * Applies only those NNC that are actually represented by the grid. These may * NNCs due to faults or NNCs that are actually neighbours. In both cases that * specified transmissibilities (scaled by EDITNNC) will be added to the already * existing models. * * \param cartesianToCompressed Vector containing the compressed index (or -1 for inactive * cells) as the element at the cartesian index. * \return Two vector of NNCs (scaled by EDITNNC). The first one are the NNCs that have been applied * and the second the NNCs not resembled by faces of the grid. NNCs specified for * inactive cells are omitted in these vectors. */ std::tuple, std::vector> applyNncToGridTrans_(const std::unordered_map& cartesianToCompressed); /// \brief Multiplies the grid transmissibilities according to EDITNNC. void applyEditNncToGridTrans_(const std::unordered_map& globalToLocal); void extractPermeability_(); void extractPermeability_(const std::function& map); void extractPorosity_(); void computeHalfTrans_(Scalar& halfTrans, const DimVector& areaNormal, int faceIdx, // in the reference element that contains the intersection const DimVector& distance, const DimMatrix& perm) const; void computeHalfDiffusivity_(Scalar& halfDiff, const DimVector& areaNormal, const DimVector& distance, const Scalar& poro) const; DimVector distanceVector_(const DimVector& center, int faceIdx, // in the reference element that contains the intersection unsigned elemIdx, const std::array, dimWorld>& axisCentroids) const; void applyMultipliers_(Scalar& trans, unsigned faceIdx, unsigned cartElemIdx, const TransMult& transMult) const; void applyNtg_(Scalar& trans, unsigned faceIdx, unsigned elemIdx, const std::vector& ntg) const; std::vector permeability_; std::vector porosity_; std::unordered_map trans_; const EclipseState& eclState_; const GridView& gridView_; const CartesianIndexMapper& cartMapper_; const Grid& grid_; std::function(int)> centroids_; Scalar transmissibilityThreshold_; std::map, Scalar> transBoundary_; std::map, Scalar> thermalHalfTransBoundary_; bool enableEnergy_; bool enableDiffusivity_; std::unordered_map thermalHalfTrans_; std::unordered_map diffusivity_; }; } // namespace Opm #endif