Support for FLOWS and FLORES

This commit is contained in:
David Landa Marban 2023-02-02 11:37:47 +01:00
parent 55a1747595
commit 8ff952fd84
3 changed files with 148 additions and 1 deletions

View File

@ -203,6 +203,7 @@ public:
* read from the extensive quantities of the element context. * read from the extensive quantities of the element context.
*/ */
static void computeFlux(RateVector& flux, static void computeFlux(RateVector& flux,
RateVector& darcy,
const Problem& problem, const Problem& problem,
const unsigned globalIndexIn, const unsigned globalIndexIn,
const unsigned globalIndexEx, const unsigned globalIndexEx,
@ -213,6 +214,7 @@ public:
const FaceDir::DirEnum facedir) const FaceDir::DirEnum facedir)
{ {
flux = 0.0; flux = 0.0;
darcy = 0.0;
Scalar Vin = problem.model().dofTotalVolume(globalIndexIn); Scalar Vin = problem.model().dofTotalVolume(globalIndexIn);
Scalar Vex = problem.model().dofTotalVolume(globalIndexEx); Scalar Vex = problem.model().dofTotalVolume(globalIndexEx);
@ -236,6 +238,7 @@ public:
Scalar distZ = zIn - zEx; // NB could be precalculated Scalar distZ = zIn - zEx; // NB could be precalculated
calculateFluxes_(flux, calculateFluxes_(flux,
darcy,
intQuantsIn, intQuantsIn,
intQuantsEx, intQuantsEx,
Vin, Vin,
@ -260,6 +263,7 @@ public:
assert(timeIdx == 0); assert(timeIdx == 0);
flux = 0.0; flux = 0.0;
RateVector darcy = 0.0;
// need for dary flux calculation // need for dary flux calculation
const auto& problem = elemCtx.problem(); const auto& problem = elemCtx.problem();
const auto& stencil = elemCtx.stencil(timeIdx); const auto& stencil = elemCtx.stencil(timeIdx);
@ -304,6 +308,7 @@ public:
Scalar distZ = zIn - zEx; Scalar distZ = zIn - zEx;
calculateFluxes_(flux, calculateFluxes_(flux,
darcy,
intQuantsIn, intQuantsIn,
intQuantsEx, intQuantsEx,
Vin, Vin,
@ -318,6 +323,7 @@ public:
} }
static void calculateFluxes_(RateVector& flux, static void calculateFluxes_(RateVector& flux,
RateVector& darcy,
const IntensiveQuantities& intQuantsIn, const IntensiveQuantities& intQuantsIn,
const IntensiveQuantities& intQuantsEx, const IntensiveQuantities& intQuantsEx,
const Scalar& Vin, const Scalar& Vin,
@ -371,6 +377,7 @@ public:
darcyFlux = pressureDifference * darcyFlux = pressureDifference *
(Toolbox::value(up.mobility(phaseIdx, facedir)) * Toolbox::value(transMult) * (-trans / faceArea)); (Toolbox::value(up.mobility(phaseIdx, facedir)) * Toolbox::value(transMult) * (-trans / faceArea));
} }
darcy[phaseIdx] = darcyFlux.value() * faceArea; // For the FLORES fluxes
unsigned pvtRegionIdx = up.pvtRegionIndex(); unsigned pvtRegionIdx = up.pvtRegionIndex();
// if (upIdx == globalFocusDofIdx){ // if (upIdx == globalFocusDofIdx){

View File

@ -32,6 +32,7 @@
#include "linearizationtype.hh" #include "linearizationtype.hh"
#include <opm/common/Exceptions.hpp> #include <opm/common/Exceptions.hpp>
#include <opm/grid/utility/SparseTable.hpp>
#include <opm/models/parallel/gridcommhandles.hh> #include <opm/models/parallel/gridcommhandles.hh>
#include <opm/models/parallel/threadmanager.hh> #include <opm/models/parallel/threadmanager.hh>
@ -295,6 +296,22 @@ public:
const std::map<unsigned, Constraints>& constraintsMap() const const std::map<unsigned, Constraints>& constraintsMap() const
{ return constraintsMap_; } { return constraintsMap_; }
/*!
* \brief Return constant reference to the flowsInfo.
*
* (This object has been only implemented for the tpfalinearizer.)
*/
const auto& getFlowsInfo() const
{return flowsInfo_;}
/*!
* \brief Return constant reference to the floresInfo.
*
* (This object has been only implemented for the tpfalinearizer.)
*/
const auto& getFloresInfo() const
{return floresInfo_;}
private: private:
Simulator& simulator_() Simulator& simulator_()
{ return *simulatorPtr_; } { return *simulatorPtr_; }
@ -588,6 +605,16 @@ private:
// EnableConstraints property is true) // EnableConstraints property is true)
std::map<unsigned, Constraints> constraintsMap_; std::map<unsigned, Constraints> constraintsMap_;
struct FlowInfo
{
int faceId;
VectorBlock flow;
unsigned int nncId;
};
SparseTable<FlowInfo> flowsInfo_;
SparseTable<FlowInfo> floresInfo_;
// the jacobian matrix // the jacobian matrix
std::unique_ptr<SparseMatrixAdapter> jacobian_; std::unique_ptr<SparseMatrixAdapter> jacobian_;

View File

@ -266,6 +266,26 @@ public:
return linearizationType_; return linearizationType_;
}; };
/*!
* \brief Return constant reference to the flowsInfo.
*
* (This object is only non-empty if the FLOWS keyword is true.)
*/
const auto& getFlowsInfo() const{
return flowsInfo_;
}
/*!
* \brief Return constant reference to the floresInfo.
*
* (This object is only non-empty if the FLORES keyword is true.)
*/
const auto& getFloresInfo() const{
return floresInfo_;
}
void updateDiscretizationParameters() void updateDiscretizationParameters()
{ {
updateStoredTransmissibilities(); updateStoredTransmissibilities();
@ -306,6 +326,9 @@ private:
// initialize the Jacobian matrix and the vector for the residual function // initialize the Jacobian matrix and the vector for the residual function
residual_.resize(model_().numTotalDof()); residual_.resize(model_().numTotalDof());
resetSystem_(); resetSystem_();
// initialize the sparse tables for Flows and Flores
createFlows_();
} }
// Construct the BCRS matrix for the Jacobian of the residual function // Construct the BCRS matrix for the Jacobian of the residual function
@ -405,6 +428,73 @@ private:
jacobian_->clear(); jacobian_->clear();
} }
// Initialize the flows and flores sparse tables
void createFlows_()
{
// If FLOWS/FLORES is set in any RPTRST in the schedule, then we initializate the sparse tables
const bool anyFlows = simulator_().problem().eclWriter()->eclOutputModule().anyFlows();
const bool anyFlores = simulator_().problem().eclWriter()->eclOutputModule().anyFlores();
if ((!anyFlows || !flowsInfo_.empty()) && (!anyFlores || !floresInfo_.empty())) {
return;
}
const auto& model = model_();
const auto& nncOutput = simulator_().problem().eclWriter()->getOutputNnc();
Stencil stencil(gridView_(), model_().dofMapper());
unsigned numCells = model.numTotalDof();
std::unordered_multimap<int, std::pair<int, int>> nncIndices;
std::vector<FlowInfo> loc_flinfo;
unsigned int nncId = 0;
VectorBlock flow(0.0);
// Create a nnc structure to use fast lookup
for (unsigned int nncIdx = 0; nncIdx < nncOutput.size(); ++nncIdx) {
const int ci1 = nncOutput[nncIdx].cell1;
const int ci2 = nncOutput[nncIdx].cell2;
nncIndices.emplace(ci1, std::make_pair(ci2, nncIdx));
}
if (anyFlows) {
flowsInfo_.reserve(numCells, 6 * numCells);
}
if (anyFlores) {
floresInfo_.reserve(numCells, 6 * numCells);
}
for (const auto& elem : elements(gridView_())) {
stencil.update(elem);
for (unsigned primaryDofIdx = 0; primaryDofIdx < stencil.numPrimaryDof(); ++primaryDofIdx) {
unsigned myIdx = stencil.globalSpaceIndex(primaryDofIdx);
loc_flinfo.resize(stencil.numDof() - 1);
for (unsigned dofIdx = 0; dofIdx < stencil.numDof(); ++dofIdx) {
unsigned neighborIdx = stencil.globalSpaceIndex(dofIdx);
if (dofIdx > 0) {
const auto scvfIdx = dofIdx - 1;
const auto& scvf = stencil.interiorFace(scvfIdx);
int faceId = scvf.dirId();
const int cartMyIdx = simulator_().vanguard().cartesianIndex(myIdx);
const int cartNeighborIdx = simulator_().vanguard().cartesianIndex(neighborIdx);
const auto& range = nncIndices.equal_range(cartMyIdx);
for (auto it = range.first; it != range.second; ++it) {
if (it->second.first == cartNeighborIdx){
// -1 gives problem since is used for the nncInput from the deck
faceId = -2;
// the index is stored to be used for writting the outputs
nncId = it->second.second;
}
}
loc_flinfo[dofIdx - 1] = FlowInfo{faceId, flow, nncId};
}
}
if (anyFlows) {
flowsInfo_.appendRow(loc_flinfo.begin(), loc_flinfo.end());
}
if (anyFlores) {
floresInfo_.appendRow(loc_flinfo.begin(), loc_flinfo.end());
}
}
}
}
public: public:
void setResAndJacobi(VectorBlock& res, MatrixBlock& bMat, const ADVectorBlock& resid) const void setResAndJacobi(VectorBlock& res, MatrixBlock& bMat, const ADVectorBlock& resid) const
{ {
@ -428,6 +518,8 @@ private:
const bool well_local = true; const bool well_local = true;
resetSystem_(); resetSystem_();
unsigned numCells = model_().numTotalDof(); unsigned numCells = model_().numTotalDof();
const bool& enableFlows = simulator_().problem().eclWriter()->eclOutputModule().hasFlows();
const bool& enableFlores = simulator_().problem().eclWriter()->eclOutputModule().hasFlores();
#ifdef _OPENMP #ifdef _OPENMP
#pragma omp parallel for #pragma omp parallel for
#endif #endif
@ -436,6 +528,7 @@ private:
VectorBlock res(0.0); VectorBlock res(0.0);
MatrixBlock bMat(0.0); MatrixBlock bMat(0.0);
ADVectorBlock adres(0.0); ADVectorBlock adres(0.0);
ADVectorBlock darcyFlux(0.0);
const IntensiveQuantities* intQuantsInP = model_().cachedIntensiveQuantities(globI, /*timeIdx*/ 0); const IntensiveQuantities* intQuantsInP = model_().cachedIntensiveQuantities(globI, /*timeIdx*/ 0);
if (intQuantsInP == nullptr) { if (intQuantsInP == nullptr) {
throw std::logic_error("Missing updated intensive quantities for cell " + std::to_string(globI)); throw std::logic_error("Missing updated intensive quantities for cell " + std::to_string(globI));
@ -450,15 +543,26 @@ private:
res = 0.0; res = 0.0;
bMat = 0.0; bMat = 0.0;
adres = 0.0; adres = 0.0;
darcyFlux = 0.0;
const IntensiveQuantities* intQuantsExP = model_().cachedIntensiveQuantities(globJ, /*timeIdx*/ 0); const IntensiveQuantities* intQuantsExP = model_().cachedIntensiveQuantities(globJ, /*timeIdx*/ 0);
if (intQuantsExP == nullptr) { if (intQuantsExP == nullptr) {
throw std::logic_error("Missing updated intensive quantities for cell " + std::to_string(globJ) + " when assembling fluxes for cell " + std::to_string(globI)); throw std::logic_error("Missing updated intensive quantities for cell " + std::to_string(globJ) + " when assembling fluxes for cell " + std::to_string(globI));
} }
const IntensiveQuantities& intQuantsEx = *intQuantsExP; const IntensiveQuantities& intQuantsEx = *intQuantsExP;
LocalResidual::computeFlux( LocalResidual::computeFlux(
adres, problem_(), globI, globJ, intQuantsIn, intQuantsEx, adres, darcyFlux, problem_(), globI, globJ, intQuantsIn, intQuantsEx,
nbInfo.trans, nbInfo.faceArea, nbInfo.faceDirection); nbInfo.trans, nbInfo.faceArea, nbInfo.faceDirection);
adres *= nbInfo.faceArea; adres *= nbInfo.faceArea;
if (enableFlows) {
for (unsigned phaseIdx = 0; phaseIdx < numEq; ++ phaseIdx) {
flowsInfo_[globI][loc].flow[phaseIdx] = adres[phaseIdx].value();
}
}
if (enableFlores) {
for (unsigned phaseIdx = 0; phaseIdx < numEq; ++ phaseIdx) {
floresInfo_[globI][loc].flow[phaseIdx] = darcyFlux[phaseIdx].value();
}
}
setResAndJacobi(res, bMat, adres); setResAndJacobi(res, bMat, adres);
residual_[globI] += res; residual_[globI] += res;
//SparseAdapter syntax: jacobian_->addToBlock(globI, globI, bMat); //SparseAdapter syntax: jacobian_->addToBlock(globI, globI, bMat);
@ -564,6 +668,15 @@ private:
SparseTable<NeighborInfo> neighborInfo_; SparseTable<NeighborInfo> neighborInfo_;
std::vector<MatrixBlock*> diagMatAddress_; std::vector<MatrixBlock*> diagMatAddress_;
struct FlowInfo
{
int faceId;
VectorBlock flow;
unsigned int nncId;
};
SparseTable<FlowInfo> flowsInfo_;
SparseTable<FlowInfo> floresInfo_;
using ScalarFluidState = typename IntensiveQuantities::ScalarFluidState; using ScalarFluidState = typename IntensiveQuantities::ScalarFluidState;
struct BoundaryConditionData struct BoundaryConditionData
{ {