mirror of
https://github.com/OPM/opm-simulators.git
synced 2025-01-14 04:51:56 -06:00
Support for FLOWS and FLORES
This commit is contained in:
parent
55a1747595
commit
8ff952fd84
@ -203,6 +203,7 @@ public:
|
||||
* read from the extensive quantities of the element context.
|
||||
*/
|
||||
static void computeFlux(RateVector& flux,
|
||||
RateVector& darcy,
|
||||
const Problem& problem,
|
||||
const unsigned globalIndexIn,
|
||||
const unsigned globalIndexEx,
|
||||
@ -213,6 +214,7 @@ public:
|
||||
const FaceDir::DirEnum facedir)
|
||||
{
|
||||
flux = 0.0;
|
||||
darcy = 0.0;
|
||||
Scalar Vin = problem.model().dofTotalVolume(globalIndexIn);
|
||||
Scalar Vex = problem.model().dofTotalVolume(globalIndexEx);
|
||||
|
||||
@ -236,6 +238,7 @@ public:
|
||||
Scalar distZ = zIn - zEx; // NB could be precalculated
|
||||
|
||||
calculateFluxes_(flux,
|
||||
darcy,
|
||||
intQuantsIn,
|
||||
intQuantsEx,
|
||||
Vin,
|
||||
@ -260,6 +263,7 @@ public:
|
||||
assert(timeIdx == 0);
|
||||
|
||||
flux = 0.0;
|
||||
RateVector darcy = 0.0;
|
||||
// need for dary flux calculation
|
||||
const auto& problem = elemCtx.problem();
|
||||
const auto& stencil = elemCtx.stencil(timeIdx);
|
||||
@ -304,6 +308,7 @@ public:
|
||||
Scalar distZ = zIn - zEx;
|
||||
|
||||
calculateFluxes_(flux,
|
||||
darcy,
|
||||
intQuantsIn,
|
||||
intQuantsEx,
|
||||
Vin,
|
||||
@ -318,6 +323,7 @@ public:
|
||||
}
|
||||
|
||||
static void calculateFluxes_(RateVector& flux,
|
||||
RateVector& darcy,
|
||||
const IntensiveQuantities& intQuantsIn,
|
||||
const IntensiveQuantities& intQuantsEx,
|
||||
const Scalar& Vin,
|
||||
@ -371,6 +377,7 @@ public:
|
||||
darcyFlux = pressureDifference *
|
||||
(Toolbox::value(up.mobility(phaseIdx, facedir)) * Toolbox::value(transMult) * (-trans / faceArea));
|
||||
}
|
||||
darcy[phaseIdx] = darcyFlux.value() * faceArea; // For the FLORES fluxes
|
||||
|
||||
unsigned pvtRegionIdx = up.pvtRegionIndex();
|
||||
// if (upIdx == globalFocusDofIdx){
|
||||
|
@ -32,6 +32,7 @@
|
||||
#include "linearizationtype.hh"
|
||||
|
||||
#include <opm/common/Exceptions.hpp>
|
||||
#include <opm/grid/utility/SparseTable.hpp>
|
||||
|
||||
#include <opm/models/parallel/gridcommhandles.hh>
|
||||
#include <opm/models/parallel/threadmanager.hh>
|
||||
@ -295,6 +296,22 @@ public:
|
||||
const std::map<unsigned, Constraints>& constraintsMap() const
|
||||
{ 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:
|
||||
Simulator& simulator_()
|
||||
{ return *simulatorPtr_; }
|
||||
@ -588,6 +605,16 @@ private:
|
||||
// EnableConstraints property is true)
|
||||
std::map<unsigned, Constraints> constraintsMap_;
|
||||
|
||||
|
||||
struct FlowInfo
|
||||
{
|
||||
int faceId;
|
||||
VectorBlock flow;
|
||||
unsigned int nncId;
|
||||
};
|
||||
SparseTable<FlowInfo> flowsInfo_;
|
||||
SparseTable<FlowInfo> floresInfo_;
|
||||
|
||||
// the jacobian matrix
|
||||
std::unique_ptr<SparseMatrixAdapter> jacobian_;
|
||||
|
||||
|
@ -266,6 +266,26 @@ public:
|
||||
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()
|
||||
{
|
||||
updateStoredTransmissibilities();
|
||||
@ -306,6 +326,9 @@ private:
|
||||
// initialize the Jacobian matrix and the vector for the residual function
|
||||
residual_.resize(model_().numTotalDof());
|
||||
resetSystem_();
|
||||
|
||||
// initialize the sparse tables for Flows and Flores
|
||||
createFlows_();
|
||||
}
|
||||
|
||||
// Construct the BCRS matrix for the Jacobian of the residual function
|
||||
@ -405,6 +428,73 @@ private:
|
||||
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:
|
||||
void setResAndJacobi(VectorBlock& res, MatrixBlock& bMat, const ADVectorBlock& resid) const
|
||||
{
|
||||
@ -428,6 +518,8 @@ private:
|
||||
const bool well_local = true;
|
||||
resetSystem_();
|
||||
unsigned numCells = model_().numTotalDof();
|
||||
const bool& enableFlows = simulator_().problem().eclWriter()->eclOutputModule().hasFlows();
|
||||
const bool& enableFlores = simulator_().problem().eclWriter()->eclOutputModule().hasFlores();
|
||||
#ifdef _OPENMP
|
||||
#pragma omp parallel for
|
||||
#endif
|
||||
@ -436,6 +528,7 @@ private:
|
||||
VectorBlock res(0.0);
|
||||
MatrixBlock bMat(0.0);
|
||||
ADVectorBlock adres(0.0);
|
||||
ADVectorBlock darcyFlux(0.0);
|
||||
const IntensiveQuantities* intQuantsInP = model_().cachedIntensiveQuantities(globI, /*timeIdx*/ 0);
|
||||
if (intQuantsInP == nullptr) {
|
||||
throw std::logic_error("Missing updated intensive quantities for cell " + std::to_string(globI));
|
||||
@ -450,15 +543,26 @@ private:
|
||||
res = 0.0;
|
||||
bMat = 0.0;
|
||||
adres = 0.0;
|
||||
darcyFlux = 0.0;
|
||||
const IntensiveQuantities* intQuantsExP = model_().cachedIntensiveQuantities(globJ, /*timeIdx*/ 0);
|
||||
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));
|
||||
}
|
||||
const IntensiveQuantities& intQuantsEx = *intQuantsExP;
|
||||
LocalResidual::computeFlux(
|
||||
adres, problem_(), globI, globJ, intQuantsIn, intQuantsEx,
|
||||
adres, darcyFlux, problem_(), globI, globJ, intQuantsIn, intQuantsEx,
|
||||
nbInfo.trans, nbInfo.faceArea, nbInfo.faceDirection);
|
||||
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);
|
||||
residual_[globI] += res;
|
||||
//SparseAdapter syntax: jacobian_->addToBlock(globI, globI, bMat);
|
||||
@ -564,6 +668,15 @@ private:
|
||||
SparseTable<NeighborInfo> neighborInfo_;
|
||||
std::vector<MatrixBlock*> diagMatAddress_;
|
||||
|
||||
struct FlowInfo
|
||||
{
|
||||
int faceId;
|
||||
VectorBlock flow;
|
||||
unsigned int nncId;
|
||||
};
|
||||
SparseTable<FlowInfo> flowsInfo_;
|
||||
SparseTable<FlowInfo> floresInfo_;
|
||||
|
||||
using ScalarFluidState = typename IntensiveQuantities::ScalarFluidState;
|
||||
struct BoundaryConditionData
|
||||
{
|
||||
|
Loading…
Reference in New Issue
Block a user