From ed887c6a9a4bdaf4928ace80264a5e9651472648 Mon Sep 17 00:00:00 2001 From: "Dennis.Glaeser" <dennis.glaeser@iws.uni-stuttgart.de> Date: Sat, 9 Feb 2019 14:48:19 +0100 Subject: [PATCH] [facet][box][darcyslaw] use tpfa flux also on interior Dirichlet BCS --- dumux/multidomain/facet/box/darcyslaw.hh | 91 ++++++------------- .../facet/box/fluxvariablescache.hh | 10 +- 2 files changed, 34 insertions(+), 67 deletions(-) diff --git a/dumux/multidomain/facet/box/darcyslaw.hh b/dumux/multidomain/facet/box/darcyslaw.hh index 29de16a088..c204533701 100644 --- a/dumux/multidomain/facet/box/darcyslaw.hh +++ b/dumux/multidomain/facet/box/darcyslaw.hh @@ -83,82 +83,49 @@ public: const auto& insideScv = fvGeometry.scv(scvf.insideScvIdx()); const auto& insideVolVars = elemVolVars[insideScv]; - // evaluate user-defined interior boundary types - const auto bcTypes = problem.interiorBoundaryTypes(element, scvf); - - static const bool enableGravity = getParamFromGroup<bool>(problem.paramGroup(), "Problem.EnableGravity"); + // evaluate the flux in a tpfa manner + const auto& supportPointShapeValues = fluxVarCache.shapeValuesAtTpfaSupportPoint(); - // on interior Neumann boundaries, evaluate the flux using the facet permeability - if (bcTypes.hasOnlyNeumann()) + Scalar rho = 0.0; + Scalar supportPressure = 0.0; + for (const auto& scv : scvs(fvGeometry)) { - const auto& supportPointShapeValues = fluxVarCache.shapeValuesAtTpfaSupportPoint(); + const auto& volVars = elemVolVars[scv]; + rho += volVars.density(phaseIdx)*shapeValues[scv.indexInElement()][0]; + supportPressure += volVars.pressure(phaseIdx)*supportPointShapeValues[scv.indexInElement()][0]; + } - Scalar rho = 0.0; - Scalar supportPressure = 0.0; - for (const auto& scv : scvs(fvGeometry)) - { - const auto& volVars = elemVolVars[scv]; - rho += volVars.density(phaseIdx)*shapeValues[scv.indexInElement()][0]; - supportPressure += volVars.pressure(phaseIdx)*supportPointShapeValues[scv.indexInElement()][0]; - } + // the transmissibility on the matrix side + const auto dm = (scvf.ipGlobal() - fluxVarCache.tpfaSupportPoint()).two_norm(); + const auto tm = 1.0/dm*vtmv(scvf.unitOuterNormal(), insideVolVars.permeability(), scvf.unitOuterNormal()); - // compute tpfa flux such that flux and pressure continuity holds - const auto& facetVolVars = problem.couplingManager().getLowDimVolVars(element, scvf); + // compute flux depending on the user's choice of boundary types + const auto bcTypes = problem.interiorBoundaryTypes(element, scvf); + const auto& facetVolVars = problem.couplingManager().getLowDimVolVars(element, scvf); + // compute tpfa flux such that flux continuity holds with flux in fracture + Scalar flux; + if (bcTypes.hasOnlyNeumann()) + { // On surface grids, use sqrt of aperture as distance measur using std::sqrt; - const auto df = dim == dimWorld ? 0.5*facetVolVars.extrusionFactor() - : 0.5*sqrt(facetVolVars.extrusionFactor()); - const auto dm = (scvf.ipGlobal() - fluxVarCache.tpfaSupportPoint()).two_norm(); - - const auto tm = 1.0/dm*vtmv(scvf.unitOuterNormal(), insideVolVars.permeability(), scvf.unitOuterNormal()); + const auto df = dim == dimWorld ? 0.5*facetVolVars.extrusionFactor() : 0.5*sqrt(facetVolVars.extrusionFactor()); const auto tf = 1.0/df*vtmv(scvf.unitOuterNormal(), facetVolVars.permeability(), scvf.unitOuterNormal()); - auto flux = tm*tf/(tm+tf)*(supportPressure - facetVolVars.pressure(phaseIdx)) - *scvf.area()*insideVolVars.extrusionFactor(); - - if (enableGravity) - flux -= rho*scvf.area()*insideVolVars.extrusionFactor() - *vtmv(scvf.unitOuterNormal(), insideVolVars.permeability(), problem.gravityAtPos(scvf.center())); - - return flux; + flux = tm*tf/(tm+tf)*(supportPressure - facetVolVars.pressure(phaseIdx))*scvf.area()*insideVolVars.extrusionFactor(); } - - // on interior Dirichlet boundaries use the facet pressure and evaluate flux else if (bcTypes.hasOnlyDirichlet()) - { - // create vector with nodal pressures - std::vector<Scalar> pressures(element.subEntities(dim)); - for (const auto& scv : scvs(fvGeometry)) - pressures[scv.localDofIndex()] = elemVolVars[scv].pressure(phaseIdx); - - // substitute with facet pressures for those scvs touching this facet - for (const auto& scvfJ : scvfs(fvGeometry)) - if (scvfJ.interiorBoundary() && scvfJ.facetIndexInElement() == scvf.facetIndexInElement()) - pressures[ fvGeometry.scv(scvfJ.insideScvIdx()).localDofIndex() ] - = problem.couplingManager().getLowDimVolVars(element, scvfJ).pressure(phaseIdx); - - // evaluate gradP - rho*g at integration point - Scalar rho(0.0); - Dune::FieldVector<Scalar, dimWorld> gradP(0.0); - for (const auto& scv : scvs(fvGeometry)) - { - rho += elemVolVars[scv].density(phaseIdx)*shapeValues[scv.indexInElement()][0]; - gradP.axpy(pressures[scv.localDofIndex()], fluxVarCache.gradN(scv.indexInElement())); - } - - if (enableGravity) - gradP.axpy(-rho, problem.gravityAtPos(scvf.center())); - - // apply matrix permeability and return the flux - return -1.0*scvf.area() - *insideVolVars.extrusionFactor() - *vtmv(scvf.unitOuterNormal(), insideVolVars.permeability(), gradP); - } - + flux = tm*(supportPressure - facetVolVars.pressure(phaseIdx))*scvf.area()*insideVolVars.extrusionFactor(); // mixed boundary types are not supported else DUNE_THROW(Dune::NotImplemented, "Mixed boundary types are not supported"); + + static const bool enableGravity = getParamFromGroup<bool>(problem.paramGroup(), "Problem.EnableGravity"); + if (enableGravity) + flux -= rho*scvf.area()*insideVolVars.extrusionFactor() + *vtmv(scvf.unitOuterNormal(), insideVolVars.permeability(), problem.gravityAtPos(scvf.center())); + + return flux; } // compute transmissibilities ti for analytical jacobians diff --git a/dumux/multidomain/facet/box/fluxvariablescache.hh b/dumux/multidomain/facet/box/fluxvariablescache.hh index 41593ec3ef..43cc613541 100644 --- a/dumux/multidomain/facet/box/fluxvariablescache.hh +++ b/dumux/multidomain/facet/box/fluxvariablescache.hh @@ -69,9 +69,9 @@ public: // on interior boundaries with Neumann BCs, prepare the shape values at a point // inside the element whose orthogonal projection is the integration point on scvf - if (scvf.interiorBoundary() && problem.interiorBoundaryTypes(element, scvf).hasOnlyNeumann()) + if (scvf.interiorBoundary()) { - isInteriorNeumannCache_ = true; + isInteriorBoundaryCache_ = true; const auto& geometry = element.geometry(); const auto& insideScv = fvGeometry.scv(scvf.insideScvIdx()); @@ -96,14 +96,14 @@ public: //! returns the integration point inside the element for interior boundaries const GlobalPosition& tpfaSupportPoint() const - { assert(isInteriorNeumannCache_); return ipGlobalInside_; } + { assert(isInteriorBoundaryCache_); return ipGlobalInside_; } //! returns the shape values at ip inside the element for interior boundaries const std::vector<ShapeValue>& shapeValuesAtTpfaSupportPoint() const - { assert(isInteriorNeumannCache_); return shapeValuesInside_; } + { assert(isInteriorBoundaryCache_); return shapeValuesInside_; } private: - bool isInteriorNeumannCache_{false}; + bool isInteriorBoundaryCache_{false}; GlobalPosition ipGlobalInside_; std::vector<ShapeValue> shapeValuesInside_; }; -- GitLab