2019-09-16 04:15:31 -05:00
|
|
|
// -*- 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 <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
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
|
|
|
|
*
|
|
|
|
* \brief Contains the classes required to extend the black-oil model by energy.
|
|
|
|
*/
|
|
|
|
#ifndef EWOMS_BLACK_OIL_ENERGY_MODULE_HH
|
|
|
|
#define EWOMS_BLACK_OIL_ENERGY_MODULE_HH
|
|
|
|
|
|
|
|
#include "blackoilproperties.hh"
|
2019-09-16 03:09:33 -05:00
|
|
|
#include <opm/models/io/vtkblackoilenergymodule.hh>
|
2019-09-16 04:15:31 -05:00
|
|
|
#include <opm/models/common/quantitycallbacks.hh>
|
2023-08-23 04:46:09 -05:00
|
|
|
#include <opm/models/discretization/common/linearizationtype.hh>
|
2019-09-16 04:15:31 -05:00
|
|
|
|
|
|
|
#include <opm/material/common/Tabulated1DFunction.hpp>
|
|
|
|
|
|
|
|
#include <opm/material/common/Valgrind.hpp>
|
|
|
|
|
|
|
|
#include <dune/common/fvector.hh>
|
|
|
|
|
|
|
|
#include <string>
|
|
|
|
|
|
|
|
namespace Opm {
|
|
|
|
/*!
|
|
|
|
* \ingroup BlackOil
|
|
|
|
* \brief Contains the high level supplements required to extend the black oil
|
|
|
|
* model by energy.
|
|
|
|
*/
|
2020-06-08 09:41:02 -05:00
|
|
|
template <class TypeTag, bool enableEnergyV = getPropValue<TypeTag, Properties::EnableEnergy>()>
|
2019-09-16 04:15:31 -05:00
|
|
|
class BlackOilEnergyModule
|
|
|
|
{
|
2020-06-10 06:49:42 -05:00
|
|
|
using Scalar = GetPropType<TypeTag, Properties::Scalar>;
|
|
|
|
using Evaluation = GetPropType<TypeTag, Properties::Evaluation>;
|
|
|
|
using PrimaryVariables = GetPropType<TypeTag, Properties::PrimaryVariables>;
|
|
|
|
using IntensiveQuantities = GetPropType<TypeTag, Properties::IntensiveQuantities>;
|
|
|
|
using Model = GetPropType<TypeTag, Properties::Model>;
|
|
|
|
using Simulator = GetPropType<TypeTag, Properties::Simulator>;
|
|
|
|
using ElementContext = GetPropType<TypeTag, Properties::ElementContext>;
|
|
|
|
using FluidSystem = GetPropType<TypeTag, Properties::FluidSystem>;
|
|
|
|
using EqVector = GetPropType<TypeTag, Properties::EqVector>;
|
|
|
|
using RateVector = GetPropType<TypeTag, Properties::RateVector>;
|
|
|
|
using Indices = GetPropType<TypeTag, Properties::Indices>;
|
2019-09-16 04:15:31 -05:00
|
|
|
|
|
|
|
static constexpr unsigned temperatureIdx = Indices::temperatureIdx;
|
|
|
|
static constexpr unsigned contiEnergyEqIdx = Indices::contiEnergyEqIdx;
|
|
|
|
|
|
|
|
static constexpr unsigned enableEnergy = enableEnergyV;
|
2020-06-08 09:41:02 -05:00
|
|
|
static constexpr unsigned numEq = getPropValue<TypeTag, Properties::NumEq>();
|
2019-09-16 04:15:31 -05:00
|
|
|
static constexpr unsigned numPhases = FluidSystem::numPhases;
|
|
|
|
|
|
|
|
public:
|
2023-08-23 04:46:09 -05:00
|
|
|
using ExtensiveQuantities = GetPropType<TypeTag, Properties::ExtensiveQuantities>;
|
2019-09-16 04:15:31 -05:00
|
|
|
/*!
|
|
|
|
* \brief Register all run-time parameters for the black-oil energy module.
|
|
|
|
*/
|
|
|
|
static void registerParameters()
|
|
|
|
{
|
2022-08-09 01:55:04 -05:00
|
|
|
if constexpr (enableEnergy)
|
|
|
|
VtkBlackOilEnergyModule<TypeTag>::registerParameters();
|
2019-09-16 04:15:31 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/*!
|
|
|
|
* \brief Register all energy specific VTK and ECL output modules.
|
|
|
|
*/
|
|
|
|
static void registerOutputModules(Model& model,
|
|
|
|
Simulator& simulator)
|
|
|
|
{
|
2022-08-09 01:55:04 -05:00
|
|
|
if constexpr (enableEnergy)
|
|
|
|
model.addOutputModule(new VtkBlackOilEnergyModule<TypeTag>(simulator));
|
2019-09-16 04:15:31 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool primaryVarApplies(unsigned pvIdx)
|
|
|
|
{
|
2022-08-09 01:55:04 -05:00
|
|
|
if constexpr (enableEnergy)
|
|
|
|
return pvIdx == temperatureIdx;
|
|
|
|
else
|
2019-09-16 04:15:31 -05:00
|
|
|
return false;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2022-08-02 03:40:48 -05:00
|
|
|
static std::string primaryVarName([[maybe_unused]] unsigned pvIdx)
|
2019-09-16 04:15:31 -05:00
|
|
|
{
|
|
|
|
assert(primaryVarApplies(pvIdx));
|
|
|
|
|
|
|
|
return "temperature";
|
|
|
|
}
|
|
|
|
|
2022-08-02 03:40:48 -05:00
|
|
|
static Scalar primaryVarWeight([[maybe_unused]] unsigned pvIdx)
|
2019-09-16 04:15:31 -05:00
|
|
|
{
|
|
|
|
assert(primaryVarApplies(pvIdx));
|
|
|
|
|
|
|
|
// TODO: it may be beneficial to chose this differently.
|
|
|
|
return static_cast<Scalar>(1.0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool eqApplies(unsigned eqIdx)
|
|
|
|
{
|
2022-08-09 01:55:04 -05:00
|
|
|
if constexpr (enableEnergy)
|
|
|
|
return eqIdx == contiEnergyEqIdx;
|
|
|
|
else
|
2019-09-16 04:15:31 -05:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-08-02 03:40:48 -05:00
|
|
|
static std::string eqName([[maybe_unused]] unsigned eqIdx)
|
2019-09-16 04:15:31 -05:00
|
|
|
{
|
|
|
|
assert(eqApplies(eqIdx));
|
|
|
|
|
|
|
|
return "conti^energy";
|
|
|
|
}
|
|
|
|
|
2022-08-02 03:40:48 -05:00
|
|
|
static Scalar eqWeight([[maybe_unused]] unsigned eqIdx)
|
2019-09-16 04:15:31 -05:00
|
|
|
{
|
|
|
|
assert(eqApplies(eqIdx));
|
|
|
|
|
|
|
|
return 1.0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// must be called after water storage is computed
|
|
|
|
template <class LhsEval>
|
|
|
|
static void addStorage(Dune::FieldVector<LhsEval, numEq>& storage,
|
|
|
|
const IntensiveQuantities& intQuants)
|
|
|
|
{
|
2022-08-09 01:55:04 -05:00
|
|
|
if constexpr (enableEnergy) {
|
|
|
|
const auto& poro = decay<LhsEval>(intQuants.porosity());
|
2019-09-16 04:15:31 -05:00
|
|
|
|
2022-08-09 01:55:04 -05:00
|
|
|
// accumulate the internal energy of the fluids
|
|
|
|
const auto& fs = intQuants.fluidState();
|
|
|
|
for (unsigned phaseIdx = 0; phaseIdx < numPhases; ++ phaseIdx) {
|
|
|
|
if (!FluidSystem::phaseIsActive(phaseIdx))
|
|
|
|
continue;
|
2019-09-16 04:15:31 -05:00
|
|
|
|
2022-08-09 01:55:04 -05:00
|
|
|
const auto& u = decay<LhsEval>(fs.internalEnergy(phaseIdx));
|
|
|
|
const auto& S = decay<LhsEval>(fs.saturation(phaseIdx));
|
|
|
|
const auto& rho = decay<LhsEval>(fs.density(phaseIdx));
|
2019-09-16 04:15:31 -05:00
|
|
|
|
2022-08-09 01:55:04 -05:00
|
|
|
storage[contiEnergyEqIdx] += poro*S*u*rho;
|
|
|
|
}
|
2019-09-16 04:15:31 -05:00
|
|
|
|
2022-08-09 01:55:04 -05:00
|
|
|
// add the internal energy of the rock
|
2023-03-27 08:51:04 -05:00
|
|
|
Scalar rockFraction = intQuants.rockFraction();
|
2022-08-09 01:55:04 -05:00
|
|
|
const auto& uRock = decay<LhsEval>(intQuants.rockInternalEnergy());
|
2023-03-27 08:51:04 -05:00
|
|
|
storage[contiEnergyEqIdx] += rockFraction*uRock;
|
2022-08-09 01:55:04 -05:00
|
|
|
storage[contiEnergyEqIdx] *= getPropValue<TypeTag, Properties::BlackOilEnergyScalingFactor>();
|
2019-09-16 04:15:31 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-09 01:55:04 -05:00
|
|
|
static void computeFlux([[maybe_unused]] RateVector& flux,
|
|
|
|
[[maybe_unused]] const ElementContext& elemCtx,
|
|
|
|
[[maybe_unused]] unsigned scvfIdx,
|
|
|
|
[[maybe_unused]] unsigned timeIdx)
|
2019-09-16 04:15:31 -05:00
|
|
|
{
|
2022-08-09 01:55:04 -05:00
|
|
|
if constexpr (enableEnergy) {
|
|
|
|
flux[contiEnergyEqIdx] = 0.0;
|
|
|
|
|
|
|
|
const auto& extQuants = elemCtx.extensiveQuantities(scvfIdx, timeIdx);
|
|
|
|
unsigned focusIdx = elemCtx.focusDofIndex();
|
|
|
|
for (unsigned phaseIdx = 0; phaseIdx < numPhases; ++phaseIdx) {
|
|
|
|
if (!FluidSystem::phaseIsActive(phaseIdx))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
unsigned upIdx = extQuants.upstreamIndex(phaseIdx);
|
|
|
|
if (upIdx == focusIdx)
|
|
|
|
addPhaseEnthalpyFlux_<Evaluation>(flux, phaseIdx, elemCtx, scvfIdx, timeIdx);
|
|
|
|
else
|
|
|
|
addPhaseEnthalpyFlux_<Scalar>(flux, phaseIdx, elemCtx, scvfIdx, timeIdx);
|
|
|
|
}
|
2019-09-16 04:15:31 -05:00
|
|
|
|
2022-08-09 01:55:04 -05:00
|
|
|
// diffusive energy flux
|
|
|
|
flux[contiEnergyEqIdx] += extQuants.energyFlux();
|
|
|
|
flux[contiEnergyEqIdx] *= getPropValue<TypeTag, Properties::BlackOilEnergyScalingFactor>();
|
2019-09-16 04:15:31 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-23 04:46:09 -05:00
|
|
|
static void addHeatFlux(RateVector& flux,
|
|
|
|
const Evaluation& heatFlux)
|
|
|
|
{
|
|
|
|
if constexpr (enableEnergy) {
|
|
|
|
// diffusive energy flux
|
|
|
|
flux[contiEnergyEqIdx] += heatFlux;
|
|
|
|
flux[contiEnergyEqIdx] *= getPropValue<TypeTag, Properties::BlackOilEnergyScalingFactor>();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
template <class UpEval, class Eval, class FluidState>
|
|
|
|
static void addPhaseEnthalpyFluxes_(RateVector& flux,
|
|
|
|
unsigned phaseIdx,
|
|
|
|
const Eval& volumeFlux,
|
|
|
|
const FluidState& upFs)
|
|
|
|
{
|
|
|
|
flux[contiEnergyEqIdx] +=
|
|
|
|
decay<UpEval>(upFs.enthalpy(phaseIdx))
|
|
|
|
* decay<UpEval>(upFs.density(phaseIdx))
|
|
|
|
* volumeFlux;
|
|
|
|
}
|
|
|
|
|
2019-09-16 04:15:31 -05:00
|
|
|
template <class UpstreamEval>
|
|
|
|
static void addPhaseEnthalpyFlux_(RateVector& flux,
|
|
|
|
unsigned phaseIdx,
|
|
|
|
const ElementContext& elemCtx,
|
|
|
|
unsigned scvfIdx,
|
|
|
|
unsigned timeIdx)
|
|
|
|
{
|
|
|
|
const auto& extQuants = elemCtx.extensiveQuantities(scvfIdx, timeIdx);
|
|
|
|
unsigned upIdx = extQuants.upstreamIndex(phaseIdx);
|
|
|
|
const auto& up = elemCtx.intensiveQuantities(upIdx, timeIdx);
|
|
|
|
const auto& fs = up.fluidState();
|
|
|
|
const auto& volFlux = extQuants.volumeFlux(phaseIdx);
|
2023-08-23 04:46:09 -05:00
|
|
|
addPhaseEnthalpyFluxes_<UpstreamEval>(flux,
|
|
|
|
phaseIdx,
|
|
|
|
volFlux,
|
|
|
|
fs);
|
2019-09-16 04:15:31 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static void addToEnthalpyRate(RateVector& flux,
|
|
|
|
const Evaluation& hRate)
|
|
|
|
{
|
2022-08-09 01:55:04 -05:00
|
|
|
if constexpr (enableEnergy)
|
|
|
|
flux[contiEnergyEqIdx] += hRate;
|
2019-09-16 04:15:31 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/*!
|
|
|
|
* \brief Assign the energy specific primary variables to a PrimaryVariables object
|
|
|
|
*/
|
|
|
|
static void assignPrimaryVars(PrimaryVariables& priVars,
|
2021-08-03 03:10:01 -05:00
|
|
|
Scalar)
|
2019-09-16 04:15:31 -05:00
|
|
|
{
|
2022-08-09 01:55:04 -05:00
|
|
|
if constexpr (enableEnergy)
|
|
|
|
priVars[temperatureIdx] = temperatureIdx;
|
2019-09-16 04:15:31 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/*!
|
|
|
|
* \brief Assign the energy specific primary variables to a PrimaryVariables object
|
|
|
|
*/
|
|
|
|
template <class FluidState>
|
|
|
|
static void assignPrimaryVars(PrimaryVariables& priVars,
|
|
|
|
const FluidState& fluidState)
|
|
|
|
{
|
2022-08-09 01:55:04 -05:00
|
|
|
if constexpr (enableEnergy)
|
|
|
|
priVars[temperatureIdx] = fluidState.temperature(/*phaseIdx=*/0);
|
2019-09-16 04:15:31 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/*!
|
|
|
|
* \brief Do a Newton-Raphson update the primary variables of the energys.
|
|
|
|
*/
|
|
|
|
static void updatePrimaryVars(PrimaryVariables& newPv,
|
|
|
|
const PrimaryVariables& oldPv,
|
|
|
|
const EqVector& delta)
|
|
|
|
{
|
2022-08-09 01:55:04 -05:00
|
|
|
if constexpr (enableEnergy)
|
|
|
|
// do a plain unchopped Newton update
|
|
|
|
newPv[temperatureIdx] = oldPv[temperatureIdx] - delta[temperatureIdx];
|
2019-09-16 04:15:31 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/*!
|
|
|
|
* \brief Return how much a Newton-Raphson update is considered an error
|
|
|
|
*/
|
2021-08-03 03:10:01 -05:00
|
|
|
static Scalar computeUpdateError(const PrimaryVariables&,
|
|
|
|
const EqVector&)
|
2019-09-16 04:15:31 -05:00
|
|
|
{
|
|
|
|
// do not consider consider the cange of energy primary variables for
|
|
|
|
// convergence
|
|
|
|
// TODO: maybe this should be changed
|
|
|
|
return static_cast<Scalar>(0.0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*!
|
|
|
|
* \brief Return how much a residual is considered an error
|
|
|
|
*/
|
|
|
|
static Scalar computeResidualError(const EqVector& resid)
|
|
|
|
{
|
|
|
|
// do not weight the residual of energy when it comes to convergence
|
2021-05-05 02:50:05 -05:00
|
|
|
return std::abs(scalarValue(resid[contiEnergyEqIdx]));
|
2019-09-16 04:15:31 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
template <class DofEntity>
|
|
|
|
static void serializeEntity(const Model& model, std::ostream& outstream, const DofEntity& dof)
|
|
|
|
{
|
2022-08-09 01:55:04 -05:00
|
|
|
if constexpr (enableEnergy) {
|
|
|
|
unsigned dofIdx = model.dofMapper().index(dof);
|
|
|
|
const PrimaryVariables& priVars = model.solution(/*timeIdx=*/0)[dofIdx];
|
|
|
|
outstream << priVars[temperatureIdx];
|
|
|
|
}
|
2019-09-16 04:15:31 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
template <class DofEntity>
|
|
|
|
static void deserializeEntity(Model& model, std::istream& instream, const DofEntity& dof)
|
|
|
|
{
|
2022-08-09 01:55:04 -05:00
|
|
|
if constexpr (enableEnergy) {
|
|
|
|
unsigned dofIdx = model.dofMapper().index(dof);
|
|
|
|
PrimaryVariables& priVars0 = model.solution(/*timeIdx=*/0)[dofIdx];
|
|
|
|
PrimaryVariables& priVars1 = model.solution(/*timeIdx=*/1)[dofIdx];
|
2019-09-16 04:15:31 -05:00
|
|
|
|
2022-08-09 01:55:04 -05:00
|
|
|
instream >> priVars0[temperatureIdx];
|
2019-09-16 04:15:31 -05:00
|
|
|
|
2022-08-09 01:55:04 -05:00
|
|
|
// set the primary variables for the beginning of the current time step.
|
|
|
|
priVars1 = priVars0[temperatureIdx];
|
|
|
|
}
|
2019-09-16 04:15:31 -05:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/*!
|
|
|
|
* \ingroup BlackOil
|
|
|
|
* \class Opm::BlackOilEnergyIntensiveQuantities
|
|
|
|
*
|
|
|
|
* \brief Provides the volumetric quantities required for the equations needed by the
|
|
|
|
* energys extension of the black-oil model.
|
|
|
|
*/
|
2020-06-08 09:41:02 -05:00
|
|
|
template <class TypeTag, bool enableEnergyV = getPropValue<TypeTag, Properties::EnableEnergy>()>
|
2019-09-16 04:15:31 -05:00
|
|
|
class BlackOilEnergyIntensiveQuantities
|
|
|
|
{
|
2020-06-10 06:49:42 -05:00
|
|
|
using Implementation = GetPropType<TypeTag, Properties::IntensiveQuantities>;
|
2019-09-16 04:15:31 -05:00
|
|
|
|
2020-06-10 06:49:42 -05:00
|
|
|
using Scalar = GetPropType<TypeTag, Properties::Scalar>;
|
|
|
|
using Evaluation = GetPropType<TypeTag, Properties::Evaluation>;
|
|
|
|
using PrimaryVariables = GetPropType<TypeTag, Properties::PrimaryVariables>;
|
|
|
|
using FluidSystem = GetPropType<TypeTag, Properties::FluidSystem>;
|
|
|
|
using SolidEnergyLaw = GetPropType<TypeTag, Properties::SolidEnergyLaw>;
|
|
|
|
using ThermalConductionLaw = GetPropType<TypeTag, Properties::ThermalConductionLaw>;
|
|
|
|
using Indices = GetPropType<TypeTag, Properties::Indices>;
|
|
|
|
using ElementContext = GetPropType<TypeTag, Properties::ElementContext>;
|
2023-08-23 04:46:09 -05:00
|
|
|
using Problem = GetPropType<TypeTag, Properties::Problem>;
|
2019-09-16 04:15:31 -05:00
|
|
|
|
2020-06-10 06:49:42 -05:00
|
|
|
using EnergyModule = BlackOilEnergyModule<TypeTag>;
|
2019-09-16 04:15:31 -05:00
|
|
|
|
2020-06-08 09:41:02 -05:00
|
|
|
enum { numPhases = getPropValue<TypeTag, Properties::NumPhases>() };
|
2019-09-16 04:15:31 -05:00
|
|
|
static constexpr int temperatureIdx = Indices::temperatureIdx;
|
|
|
|
static constexpr int waterPhaseIdx = FluidSystem::waterPhaseIdx;
|
|
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
/*!
|
|
|
|
* \brief Update the temperature of the intensive quantity's fluid state
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
void updateTemperature_(const ElementContext& elemCtx,
|
|
|
|
unsigned dofIdx,
|
|
|
|
unsigned timeIdx)
|
|
|
|
{
|
|
|
|
auto& fs = asImp_().fluidState_;
|
|
|
|
const auto& priVars = elemCtx.primaryVars(dofIdx, timeIdx);
|
|
|
|
|
|
|
|
// set temperature
|
2020-06-04 08:37:44 -05:00
|
|
|
fs.setTemperature(priVars.makeEvaluation(temperatureIdx, timeIdx, elemCtx.linearizationType()));
|
2019-09-16 04:15:31 -05:00
|
|
|
}
|
|
|
|
|
2023-08-23 04:46:09 -05:00
|
|
|
/*!
|
|
|
|
* \brief Update the temperature of the intensive quantity's fluid state
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
void updateTemperature_([[maybe_unused]] const Problem& problem,
|
|
|
|
const PrimaryVariables& priVars,
|
|
|
|
[[maybe_unused]] unsigned globalDofIdx,
|
|
|
|
const unsigned timeIdx,
|
|
|
|
const LinearizationType& lintype)
|
|
|
|
{
|
|
|
|
auto& fs = asImp_().fluidState_;
|
|
|
|
fs.setTemperature(priVars.makeEvaluation(temperatureIdx, timeIdx, lintype));
|
|
|
|
}
|
|
|
|
|
2019-09-16 04:15:31 -05:00
|
|
|
/*!
|
|
|
|
* \brief Compute the intensive quantities needed to handle energy conservation
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
void updateEnergyQuantities_(const ElementContext& elemCtx,
|
|
|
|
unsigned dofIdx,
|
|
|
|
unsigned timeIdx,
|
|
|
|
const typename FluidSystem::template ParameterCache<Evaluation>& paramCache)
|
|
|
|
{
|
|
|
|
auto& fs = asImp_().fluidState_;
|
|
|
|
|
|
|
|
// compute the specific enthalpy of the fluids, the specific enthalpy of the rock
|
|
|
|
// and the thermal condictivity coefficients
|
|
|
|
for (int phaseIdx = 0; phaseIdx < numPhases; ++ phaseIdx) {
|
|
|
|
if (!FluidSystem::phaseIsActive(phaseIdx)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto& h = FluidSystem::enthalpy(fs, paramCache, phaseIdx);
|
|
|
|
fs.setEnthalpy(phaseIdx, h);
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto& solidEnergyLawParams = elemCtx.problem().solidEnergyLawParams(elemCtx, dofIdx, timeIdx);
|
|
|
|
rockInternalEnergy_ = SolidEnergyLaw::solidInternalEnergy(solidEnergyLawParams, fs);
|
|
|
|
|
|
|
|
const auto& thermalConductionLawParams = elemCtx.problem().thermalConductionLawParams(elemCtx, dofIdx, timeIdx);
|
|
|
|
totalThermalConductivity_ = ThermalConductionLaw::thermalConductivity(thermalConductionLawParams, fs);
|
2023-03-27 08:51:04 -05:00
|
|
|
|
|
|
|
// Retrieve the rock fraction from the problem
|
|
|
|
// Usually 1 - porosity, but if pvmult is used to modify porosity
|
|
|
|
// we will apply the same multiplier to the rock fraction
|
|
|
|
// i.e. pvmult*(1 - porosity) and thus interpret multpv as a volume
|
|
|
|
// multiplier. This is to avoid negative rock volume for pvmult*porosity > 1
|
|
|
|
const unsigned cell_idx = elemCtx.globalSpaceIndex(dofIdx, timeIdx);
|
|
|
|
rockFraction_ = elemCtx.problem().rockFraction(cell_idx, timeIdx);
|
2019-09-16 04:15:31 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
const Evaluation& rockInternalEnergy() const
|
|
|
|
{ return rockInternalEnergy_; }
|
|
|
|
|
|
|
|
const Evaluation& totalThermalConductivity() const
|
|
|
|
{ return totalThermalConductivity_; }
|
|
|
|
|
2023-03-27 08:51:04 -05:00
|
|
|
const Scalar& rockFraction() const
|
|
|
|
{ return rockFraction_; }
|
|
|
|
|
2019-09-16 04:15:31 -05:00
|
|
|
protected:
|
|
|
|
Implementation& asImp_()
|
|
|
|
{ return *static_cast<Implementation*>(this); }
|
|
|
|
|
|
|
|
Evaluation rockInternalEnergy_;
|
|
|
|
Evaluation totalThermalConductivity_;
|
2023-03-27 08:51:04 -05:00
|
|
|
Scalar rockFraction_;
|
2019-09-16 04:15:31 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
template <class TypeTag>
|
|
|
|
class BlackOilEnergyIntensiveQuantities<TypeTag, false>
|
|
|
|
{
|
2020-06-10 06:49:42 -05:00
|
|
|
using Implementation = GetPropType<TypeTag, Properties::IntensiveQuantities>;
|
2023-08-23 04:46:09 -05:00
|
|
|
using PrimaryVariables = GetPropType<TypeTag, Properties::PrimaryVariables>;
|
2020-06-10 06:49:42 -05:00
|
|
|
using Evaluation = GetPropType<TypeTag, Properties::Evaluation>;
|
|
|
|
using ElementContext = GetPropType<TypeTag, Properties::ElementContext>;
|
|
|
|
using FluidSystem = GetPropType<TypeTag, Properties::FluidSystem>;
|
|
|
|
using Scalar = GetPropType<TypeTag, Properties::Scalar>;
|
2019-09-16 04:15:31 -05:00
|
|
|
|
2023-08-23 04:46:09 -05:00
|
|
|
using Problem = GetPropType<TypeTag, Properties::Problem>;
|
2020-06-08 09:41:02 -05:00
|
|
|
static constexpr bool enableTemperature = getPropValue<TypeTag, Properties::EnableTemperature>();
|
2019-09-16 04:15:31 -05:00
|
|
|
|
|
|
|
public:
|
2022-08-09 01:55:04 -05:00
|
|
|
void updateTemperature_([[maybe_unused]] const ElementContext& elemCtx,
|
|
|
|
[[maybe_unused]] unsigned dofIdx,
|
|
|
|
[[maybe_unused]] unsigned timeIdx)
|
2019-09-16 04:15:31 -05:00
|
|
|
{
|
2022-08-09 01:55:04 -05:00
|
|
|
if constexpr (enableTemperature) {
|
2019-09-16 04:15:31 -05:00
|
|
|
// even if energy is conserved, the temperature can vary over the spatial
|
|
|
|
// domain if the EnableTemperature property is set to true
|
|
|
|
auto& fs = asImp_().fluidState_;
|
2023-08-23 04:46:09 -05:00
|
|
|
const Scalar T = elemCtx.problem().temperature(elemCtx, dofIdx, timeIdx);
|
|
|
|
fs.setTemperature(T);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class Problem>
|
|
|
|
void updateTemperature_([[maybe_unused]] const Problem& problem,
|
|
|
|
[[maybe_unused]] const PrimaryVariables& priVars,
|
|
|
|
[[maybe_unused]] unsigned globalDofIdx,
|
|
|
|
[[maybe_unused]] unsigned timeIdx,
|
|
|
|
[[maybe_unused]] const LinearizationType& lintype
|
|
|
|
)
|
|
|
|
{
|
|
|
|
if constexpr (enableTemperature) {
|
|
|
|
auto& fs = asImp_().fluidState_;
|
|
|
|
// even if energy is conserved, the temperature can vary over the spatial
|
|
|
|
// domain if the EnableTemperature property is set to true
|
|
|
|
const Scalar T = problem.temperature(globalDofIdx, timeIdx);
|
2019-09-16 04:15:31 -05:00
|
|
|
fs.setTemperature(T);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-08-03 03:10:01 -05:00
|
|
|
void updateEnergyQuantities_(const ElementContext&,
|
|
|
|
unsigned,
|
|
|
|
unsigned,
|
|
|
|
const typename FluidSystem::template ParameterCache<Evaluation>&)
|
2019-09-16 04:15:31 -05:00
|
|
|
{ }
|
|
|
|
|
|
|
|
const Evaluation& rockInternalEnergy() const
|
|
|
|
{ throw std::logic_error("Requested the rock internal energy, which is "
|
|
|
|
"unavailable because energy is not conserved"); }
|
|
|
|
|
|
|
|
const Evaluation& totalThermalConductivity() const
|
|
|
|
{ throw std::logic_error("Requested the total thermal conductivity, which is "
|
|
|
|
"unavailable because energy is not conserved"); }
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Implementation& asImp_()
|
|
|
|
{ return *static_cast<Implementation*>(this); }
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/*!
|
|
|
|
* \ingroup BlackOil
|
|
|
|
* \class Opm::BlackOilEnergyExtensiveQuantities
|
|
|
|
*
|
|
|
|
* \brief Provides the energy specific extensive quantities to the generic black-oil
|
|
|
|
* module's extensive quantities.
|
|
|
|
*/
|
2020-06-08 09:41:02 -05:00
|
|
|
template <class TypeTag, bool enableEnergyV = getPropValue<TypeTag, Properties::EnableEnergy>()>
|
2019-09-16 04:15:31 -05:00
|
|
|
class BlackOilEnergyExtensiveQuantities
|
|
|
|
{
|
2020-06-10 06:49:42 -05:00
|
|
|
using Implementation = GetPropType<TypeTag, Properties::ExtensiveQuantities>;
|
2019-09-16 04:15:31 -05:00
|
|
|
|
2020-06-10 06:49:42 -05:00
|
|
|
using Scalar = GetPropType<TypeTag, Properties::Scalar>;
|
|
|
|
using Evaluation = GetPropType<TypeTag, Properties::Evaluation>;
|
|
|
|
using ElementContext = GetPropType<TypeTag, Properties::ElementContext>;
|
|
|
|
using IntensiveQuantities = GetPropType<TypeTag, Properties::IntensiveQuantities>;
|
|
|
|
using ExtensiveQuantities = GetPropType<TypeTag, Properties::ExtensiveQuantities>;
|
|
|
|
using FluidSystem = GetPropType<TypeTag, Properties::FluidSystem>;
|
|
|
|
using GridView = GetPropType<TypeTag, Properties::GridView>;
|
2019-09-16 04:15:31 -05:00
|
|
|
|
2021-05-05 02:50:05 -05:00
|
|
|
using Toolbox = MathToolbox<Evaluation>;
|
2019-09-16 04:15:31 -05:00
|
|
|
|
2020-06-10 06:49:42 -05:00
|
|
|
using EnergyModule = BlackOilEnergyModule<TypeTag>;
|
2019-09-16 04:15:31 -05:00
|
|
|
|
|
|
|
static const int dimWorld = GridView::dimensionworld;
|
2020-06-10 06:49:42 -05:00
|
|
|
using DimVector = Dune::FieldVector<Scalar, dimWorld>;
|
|
|
|
using DimEvalVector = Dune::FieldVector<Evaluation, dimWorld>;
|
2019-09-16 04:15:31 -05:00
|
|
|
public:
|
2023-08-23 04:46:09 -05:00
|
|
|
template<class FluidState>
|
|
|
|
static void updateEnergy(Evaluation& energyFlux,
|
|
|
|
const unsigned& focusDofIndex,
|
|
|
|
const unsigned& inIdx,
|
|
|
|
const unsigned& exIdx,
|
|
|
|
const IntensiveQuantities& inIq,
|
|
|
|
const IntensiveQuantities& exIq,
|
|
|
|
const FluidState& inFs,
|
|
|
|
const FluidState& exFs,
|
|
|
|
const Scalar& inAlpha,
|
|
|
|
const Scalar& outAlpha,
|
|
|
|
const Scalar& faceArea)
|
2019-09-16 04:15:31 -05:00
|
|
|
{
|
|
|
|
Evaluation deltaT;
|
2023-08-23 04:46:09 -05:00
|
|
|
if (focusDofIndex == inIdx)
|
2019-09-16 04:15:31 -05:00
|
|
|
deltaT =
|
2021-05-05 02:50:05 -05:00
|
|
|
decay<Scalar>(exFs.temperature(/*phaseIdx=*/0))
|
2019-09-16 04:15:31 -05:00
|
|
|
- inFs.temperature(/*phaseIdx=*/0);
|
2023-08-23 04:46:09 -05:00
|
|
|
else if (focusDofIndex == exIdx)
|
2019-09-16 04:15:31 -05:00
|
|
|
deltaT =
|
|
|
|
exFs.temperature(/*phaseIdx=*/0)
|
2021-05-05 02:50:05 -05:00
|
|
|
- decay<Scalar>(inFs.temperature(/*phaseIdx=*/0));
|
2019-09-16 04:15:31 -05:00
|
|
|
else
|
|
|
|
deltaT =
|
2021-05-05 02:50:05 -05:00
|
|
|
decay<Scalar>(exFs.temperature(/*phaseIdx=*/0))
|
|
|
|
- decay<Scalar>(inFs.temperature(/*phaseIdx=*/0));
|
2019-09-16 04:15:31 -05:00
|
|
|
|
|
|
|
Evaluation inLambda;
|
2023-08-23 04:46:09 -05:00
|
|
|
if (focusDofIndex == inIdx)
|
2019-09-16 04:15:31 -05:00
|
|
|
inLambda = inIq.totalThermalConductivity();
|
|
|
|
else
|
2021-05-05 02:50:05 -05:00
|
|
|
inLambda = decay<Scalar>(inIq.totalThermalConductivity());
|
2019-09-16 04:15:31 -05:00
|
|
|
|
|
|
|
Evaluation exLambda;
|
2023-08-23 04:46:09 -05:00
|
|
|
if (focusDofIndex == exIdx)
|
2019-09-16 04:15:31 -05:00
|
|
|
exLambda = exIq.totalThermalConductivity();
|
|
|
|
else
|
2021-05-05 02:50:05 -05:00
|
|
|
exLambda = decay<Scalar>(exIq.totalThermalConductivity());
|
2019-09-16 04:15:31 -05:00
|
|
|
|
|
|
|
Evaluation H;
|
2024-01-16 05:20:20 -06:00
|
|
|
const Evaluation& inH = inLambda*inAlpha;
|
|
|
|
const Evaluation& exH = exLambda*outAlpha;
|
|
|
|
if (inH > 0 && exH > 0) {
|
2019-09-16 04:15:31 -05:00
|
|
|
// compute the "thermal transmissibility". In contrast to the normal
|
|
|
|
// transmissibility this cannot be done as a preprocessing step because the
|
2023-08-23 04:46:09 -05:00
|
|
|
// average thermal conductivity is analogous to the permeability but
|
2019-09-16 04:15:31 -05:00
|
|
|
// depends on the solution.
|
|
|
|
H = 1.0/(1.0/inH + 1.0/exH);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
H = 0.0;
|
|
|
|
|
2023-08-23 04:46:09 -05:00
|
|
|
energyFlux = deltaT * (-H/faceArea);
|
|
|
|
}
|
|
|
|
|
|
|
|
void updateEnergy(const ElementContext& elemCtx,
|
|
|
|
unsigned scvfIdx,
|
|
|
|
unsigned timeIdx)
|
|
|
|
{
|
|
|
|
const auto& stencil = elemCtx.stencil(timeIdx);
|
|
|
|
const auto& scvf = stencil.interiorFace(scvfIdx);
|
|
|
|
|
|
|
|
const Scalar faceArea = scvf.area();
|
|
|
|
const unsigned inIdx = scvf.interiorIndex();
|
|
|
|
const unsigned exIdx = scvf.exteriorIndex();
|
|
|
|
const auto& inIq = elemCtx.intensiveQuantities(inIdx, timeIdx);
|
|
|
|
const auto& exIq = elemCtx.intensiveQuantities(exIdx, timeIdx);
|
|
|
|
const auto& inFs = inIq.fluidState();
|
|
|
|
const auto& exFs = exIq.fluidState();
|
|
|
|
const Scalar inAlpha = elemCtx.problem().thermalHalfTransmissibilityIn(elemCtx, scvfIdx, timeIdx);
|
|
|
|
const Scalar outAlpha = elemCtx.problem().thermalHalfTransmissibilityOut(elemCtx, scvfIdx, timeIdx);
|
|
|
|
updateEnergy(energyFlux_,
|
|
|
|
elemCtx.focusDofIndex(),
|
|
|
|
inIdx,
|
|
|
|
exIdx,
|
|
|
|
inIq,
|
|
|
|
exIq,
|
|
|
|
inFs,
|
|
|
|
exFs,
|
|
|
|
inAlpha,
|
|
|
|
outAlpha,
|
|
|
|
faceArea);
|
2019-09-16 04:15:31 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
template <class Context, class BoundaryFluidState>
|
|
|
|
void updateEnergyBoundary(const Context& ctx,
|
|
|
|
unsigned scvfIdx,
|
|
|
|
unsigned timeIdx,
|
|
|
|
const BoundaryFluidState& boundaryFs)
|
|
|
|
{
|
|
|
|
const auto& stencil = ctx.stencil(timeIdx);
|
|
|
|
const auto& scvf = stencil.boundaryFace(scvfIdx);
|
|
|
|
|
|
|
|
unsigned inIdx = scvf.interiorIndex();
|
|
|
|
const auto& inIq = ctx.intensiveQuantities(inIdx, timeIdx);
|
2023-08-23 04:46:09 -05:00
|
|
|
const auto& focusDofIdx = ctx.focusDofIndex();
|
|
|
|
const Scalar alpha = ctx.problem().thermalHalfTransmissibilityBoundary(ctx, scvfIdx);
|
|
|
|
updateEnergyBoundary(energyFlux_, inIq, focusDofIdx, inIdx, alpha, boundaryFs);
|
|
|
|
}
|
2019-09-16 04:15:31 -05:00
|
|
|
|
2023-08-23 04:46:09 -05:00
|
|
|
template <class BoundaryFluidState>
|
|
|
|
static void updateEnergyBoundary(Evaluation& energyFlux,
|
|
|
|
const IntensiveQuantities& inIq,
|
|
|
|
unsigned focusDofIndex,
|
|
|
|
unsigned inIdx,
|
|
|
|
Scalar alpha,
|
|
|
|
const BoundaryFluidState& boundaryFs)
|
|
|
|
{
|
|
|
|
const auto& inFs = inIq.fluidState();
|
2019-09-16 04:15:31 -05:00
|
|
|
Evaluation deltaT;
|
2023-08-23 04:46:09 -05:00
|
|
|
if (focusDofIndex == inIdx)
|
2019-09-16 04:15:31 -05:00
|
|
|
deltaT =
|
|
|
|
boundaryFs.temperature(/*phaseIdx=*/0)
|
|
|
|
- inFs.temperature(/*phaseIdx=*/0);
|
|
|
|
else
|
|
|
|
deltaT =
|
2021-05-05 02:50:05 -05:00
|
|
|
decay<Scalar>(boundaryFs.temperature(/*phaseIdx=*/0))
|
|
|
|
- decay<Scalar>(inFs.temperature(/*phaseIdx=*/0));
|
2019-09-16 04:15:31 -05:00
|
|
|
|
|
|
|
Evaluation lambda;
|
2023-08-23 04:46:09 -05:00
|
|
|
if (focusDofIndex == inIdx)
|
2019-09-16 04:15:31 -05:00
|
|
|
lambda = inIq.totalThermalConductivity();
|
|
|
|
else
|
2021-05-05 02:50:05 -05:00
|
|
|
lambda = decay<Scalar>(inIq.totalThermalConductivity());
|
2019-09-16 04:15:31 -05:00
|
|
|
|
|
|
|
|
|
|
|
if (lambda > 0.0) {
|
|
|
|
// compute the "thermal transmissibility". In contrast to the normal
|
|
|
|
// transmissibility this cannot be done as a preprocessing step because the
|
|
|
|
// average thermal conductivity is analogous to the permeability but depends
|
|
|
|
// on the solution.
|
2023-08-23 04:46:09 -05:00
|
|
|
energyFlux = deltaT*lambda*(-alpha);
|
2019-09-16 04:15:31 -05:00
|
|
|
}
|
|
|
|
else
|
2023-08-23 04:46:09 -05:00
|
|
|
energyFlux = 0.0;
|
2019-09-16 04:15:31 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
const Evaluation& energyFlux() const
|
|
|
|
{ return energyFlux_; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
Implementation& asImp_()
|
|
|
|
{ return *static_cast<Implementation*>(this); }
|
|
|
|
|
|
|
|
Evaluation energyFlux_;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <class TypeTag>
|
|
|
|
class BlackOilEnergyExtensiveQuantities<TypeTag, false>
|
|
|
|
{
|
2020-06-10 06:49:42 -05:00
|
|
|
using ElementContext = GetPropType<TypeTag, Properties::ElementContext>;
|
|
|
|
using Evaluation = GetPropType<TypeTag, Properties::Evaluation>;
|
2023-08-23 04:46:09 -05:00
|
|
|
using IntensiveQuantities = GetPropType<TypeTag, Properties::IntensiveQuantities>;
|
|
|
|
using Scalar = GetPropType<TypeTag, Properties::Scalar>;
|
2019-09-16 04:15:31 -05:00
|
|
|
public:
|
2023-08-23 04:46:09 -05:00
|
|
|
template<class FluidState>
|
2023-09-04 07:34:30 -05:00
|
|
|
static void updateEnergy(Evaluation& /*energyFlux*/,
|
|
|
|
const unsigned& /*focusDofIndex*/,
|
|
|
|
const unsigned& /*inIdx*/,
|
|
|
|
const unsigned& /*exIdx*/,
|
|
|
|
const IntensiveQuantities& /*inIq*/,
|
|
|
|
const IntensiveQuantities& /*exIq*/,
|
|
|
|
const FluidState& /*inFs*/,
|
|
|
|
const FluidState& /*exFs*/,
|
|
|
|
const Scalar& /*inAlpha*/,
|
|
|
|
const Scalar& /*outAlpha*/,
|
|
|
|
const Scalar& /*faceArea*/)
|
2023-11-20 09:56:56 -06:00
|
|
|
{}
|
2023-08-23 04:46:09 -05:00
|
|
|
|
2021-08-03 03:10:01 -05:00
|
|
|
void updateEnergy(const ElementContext&,
|
|
|
|
unsigned,
|
|
|
|
unsigned)
|
2019-09-16 04:15:31 -05:00
|
|
|
{}
|
|
|
|
|
|
|
|
template <class Context, class BoundaryFluidState>
|
2021-08-03 03:10:01 -05:00
|
|
|
void updateEnergyBoundary(const Context&,
|
|
|
|
unsigned,
|
|
|
|
unsigned,
|
|
|
|
const BoundaryFluidState&)
|
2019-09-16 04:15:31 -05:00
|
|
|
{}
|
|
|
|
|
2023-08-23 04:46:09 -05:00
|
|
|
template <class BoundaryFluidState>
|
2023-09-04 07:34:30 -05:00
|
|
|
static void updateEnergyBoundary(Evaluation& /*heatFlux*/,
|
|
|
|
const IntensiveQuantities& /*inIq*/,
|
|
|
|
unsigned /*focusDofIndex*/,
|
|
|
|
unsigned /*inIdx*/,
|
|
|
|
unsigned /*timeIdx*/,
|
|
|
|
Scalar /*alpha*/,
|
|
|
|
const BoundaryFluidState& /*boundaryFs*/)
|
|
|
|
{}
|
2019-09-16 04:15:31 -05:00
|
|
|
const Evaluation& energyFlux() const
|
|
|
|
{ throw std::logic_error("Requested the energy flux, but energy is not conserved"); }
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
} // namespace Opm
|
|
|
|
|
|
|
|
#endif
|