56class AquiferAnalytical :
public AquiferInterface<TypeTag>
78 static constexpr int numEq = BlackoilIndices::numEq;
80 using FluidState = BlackOilFluidState<Eval,
82 energyModuleType != EnergyModules::NoTemperature,
83 energyModuleType == EnergyModules::FullyImplicitThermal,
84 BlackoilIndices::gasEnabled,
87 enableSaltPrecipitation,
90 BlackoilIndices::numPhases>;
93 AquiferAnalytical(
const int aqID,
94 const std::vector<Aquancon::AquancCell>& connections,
95 const Simulator& simulator)
96 : AquiferInterface<TypeTag>(aqID, simulator)
97 , connections_(connections)
99 this->initializeConnectionMappings();
102 void computeFaceAreaFraction(
const std::vector<Scalar>& total_face_area)
override
104 assert (total_face_area.size() >=
static_cast<typename std::vector<Scalar>::size_type
>(this->aquiferID()));
106 const auto tfa = total_face_area[this->aquiferID() - 1];
107 const auto eps_sqrt = std::sqrt(std::numeric_limits<Scalar>::epsilon());
109 if (tfa < eps_sqrt) {
110 this->alphai_.assign(this->size(), Scalar{0});
113 std::ranges::transform(this->faceArea_connected_, this->alphai_.begin(),
114 [tfa](
const Scalar area)
115 { return area / tfa; });
118 this->area_fraction_ = this->totalFaceArea() / tfa;
121 Scalar totalFaceArea()
const override
123 return this->total_face_area_;
126 void initFromRestart(
const data::Aquifers& aquiferSoln)
override
128 auto xaqPos = aquiferSoln.find(this->aquiferID());
129 if (xaqPos == aquiferSoln.end())
132 this->assignRestartData(xaqPos->second);
134 this->W_flux_ = xaqPos->second.volume * this->area_fraction_;
135 this->pa0_ = xaqPos->second.initPressure;
137 this->solution_set_from_restart_ =
true;
140 void initialSolutionApplied()
override
145 void beginTimeStep()
override
147 ElementContext elemCtx(this->simulator_);
148 OPM_BEGIN_PARALLEL_TRY_CATCH();
150 for (
const auto& elem : elements(this->simulator_.gridView())) {
151 elemCtx.updatePrimaryStencil(elem);
153 const int cellIdx = elemCtx.globalSpaceIndex(0, 0);
154 const int idx = cellToConnectionIdx_[cellIdx];
158 elemCtx.updateIntensiveQuantities(0);
159 const auto& iq = elemCtx.intensiveQuantities(0, 0);
160 pressure_previous_[idx] = getValue(iq.fluidState().pressure(this->phaseIdx_()));
163 OPM_END_PARALLEL_TRY_CATCH(
"AquiferAnalytical::beginTimeStep() failed: ",
164 this->simulator_.vanguard().grid().comm());
167 void addToSource(RateVector& rates,
168 const unsigned cellIdx,
169 const unsigned timeIdx)
override
171 const auto& model = this->simulator_.model();
173 const int idx = this->cellToConnectionIdx_[cellIdx];
177 const auto& intQuants = model.intensiveQuantities(cellIdx, timeIdx);
180 this->updateCellPressure(this->pressure_current_, idx, intQuants);
181 this->calculateInflowRate(idx, this->simulator_);
183 rates[BlackoilIndices::conti0EqIdx + compIdx_()]
184 += this->Qai_[idx] / model.dofTotalVolume(cellIdx);
186 if constexpr (energyModuleType == EnergyModules::FullyImplicitThermal) {
187 auto fs = intQuants.fluidState();
188 if (this->Ta0_.has_value() && this->Qai_[idx] > 0)
190 fs.setTemperature(this->Ta0_.value());
191 typedef typename std::decay<
decltype(fs)>::type::ValueType FsValueType;
192 typename FluidSystem::template ParameterCache<FsValueType> paramCache;
193 const unsigned pvtRegionIdx = intQuants.pvtRegionIndex();
194 paramCache.setRegionIndex(pvtRegionIdx);
195 paramCache.updatePhase(fs, this->phaseIdx_());
196 const auto& h = FluidSystem::enthalpy(fs, paramCache, this->phaseIdx_());
197 fs.setEnthalpy(this->phaseIdx_(), h);
199 rates[BlackoilIndices::contiEnergyEqIdx]
200 += this->Qai_[idx] *fs.enthalpy(this->phaseIdx_()) * FluidSystem::referenceDensity( this->phaseIdx_(), intQuants.pvtRegionIndex()) / model.dofTotalVolume(cellIdx);
205 std::size_t size()
const
207 return this->connections_.size();
210 template<
class Serializer>
211 void serializeOp(Serializer& serializer)
213 serializer(pressure_previous_);
214 serializer(pressure_current_);
220 bool operator==(
const AquiferAnalytical& rhs)
const
222 return this->pressure_previous_ == rhs.pressure_previous_ &&
223 this->pressure_current_ == rhs.pressure_current_ &&
224 this->Qai_ == rhs.Qai_ &&
225 this->rhow_ == rhs.rhow_ &&
226 this->W_flux_ == rhs.W_flux_;
230 virtual void assignRestartData(
const data::AquiferData& xaq) = 0;
231 virtual void calculateInflowRate(
int idx,
const Simulator& simulator) = 0;
232 virtual void calculateAquiferCondition() = 0;
233 virtual void calculateAquiferConstants() = 0;
234 virtual Scalar aquiferDepth()
const = 0;
236 Scalar gravity_()
const
238 return this->simulator_.problem().gravity()[2];
243 if (this->co2store_or_h2store_())
244 return FluidSystem::oilCompIdx;
246 return FluidSystem::waterCompIdx;
249 void initQuantities()
252 if (! this->solution_set_from_restart_) {
256 this->initializeConnectionDepths();
257 this->calculateAquiferCondition();
258 this->calculateAquiferConstants();
260 this->pressure_previous_.resize(this->size(), Scalar{0});
261 this->pressure_current_.resize(this->size(), Scalar{0});
262 this->Qai_.resize(this->size(), Scalar{0});
265 void updateCellPressure(std::vector<Eval>& pressure_water,
267 const IntensiveQuantities& intQuants)
269 const auto& fs = intQuants.fluidState();
270 pressure_water.at(idx) = fs.pressure(this->phaseIdx_());
273 void updateCellPressure(std::vector<Scalar>& pressure_water,
275 const IntensiveQuantities& intQuants)
277 const auto& fs = intQuants.fluidState();
278 pressure_water.at(idx) = fs.pressure(this->phaseIdx_()).value();
281 void initializeConnectionMappings()
283 this->alphai_.resize(this->size(), 1.0);
284 this->faceArea_connected_.resize(this->size(), Scalar{0});
287 this->total_face_area_ = Scalar{0};
288 this->cellToConnectionIdx_.resize(this->simulator_.gridView().size(0), -1);
289 const auto& gridView = this->simulator_.vanguard().gridView();
290 for (std::size_t idx = 0; idx < this->size(); ++idx) {
291 const auto global_index = this->connections_[idx].global_index;
292 const int cell_index = this->simulator_.vanguard().compressedIndex(global_index);
293 if (cell_index < 0) {
297 auto elemIt = gridView.template begin< 0>();
298 std::advance(elemIt, cell_index);
301 if (elemIt->partitionType() != Dune::InteriorEntity) {
305 this->cellToConnectionIdx_[cell_index] = idx;
309 FaceDir::DirEnum faceDirection;
312 const auto& elemMapper = this->simulator_.model().dofMapper();
313 for (
const auto& elem : elements(gridView)) {
314 const unsigned cell_index = elemMapper.index(elem);
315 const int idx = this->cellToConnectionIdx_[cell_index];
322 for (
const auto& intersection : intersections(gridView, elem)) {
324 if (! intersection.boundary()) {
328 switch (intersection.indexInInside()) {
330 faceDirection = FaceDir::XMinus;
333 faceDirection = FaceDir::XPlus;
336 faceDirection = FaceDir::YMinus;
339 faceDirection = FaceDir::YPlus;
342 faceDirection = FaceDir::ZMinus;
345 faceDirection = FaceDir::ZPlus;
348 OPM_THROW(std::logic_error,
349 "Internal error in initialization of aquifer.");
352 if (faceDirection == this->connections_[idx].face_dir) {
353 this->faceArea_connected_[idx] = this->connections_[idx].influx_coeff;
358 this->total_face_area_ += this->faceArea_connected_.at(idx);
362 void initializeConnectionDepths()
364 this->cell_depth_.resize(this->size(), this->aquiferDepth());
366 const auto& gridView = this->simulator_.vanguard().gridView();
367 for (std::size_t idx = 0; idx < this->size(); ++idx) {
368 const int cell_index = this->simulator_.vanguard()
369 .compressedIndex(this->connections_[idx].global_index);
370 if (cell_index < 0) {
374 auto elemIt = gridView.template begin< 0>();
375 std::advance(elemIt, cell_index);
378 if (elemIt->partitionType() != Dune::InteriorEntity) {
382 this->cell_depth_.at(idx) = this->simulator_.vanguard().cellCenterDepth(cell_index);
387 Scalar calculateReservoirEquilibrium()
390 std::vector<Scalar> pw_aquifer;
391 Scalar water_pressure_reservoir;
393 ElementContext elemCtx(this->simulator_);
394 const auto& gridView = this->simulator_.gridView();
395 for (
const auto& elem : elements(gridView)) {
396 elemCtx.updatePrimaryStencil(elem);
398 const auto cellIdx = elemCtx.globalSpaceIndex(0, 0);
399 const auto idx = this->cellToConnectionIdx_[cellIdx];
403 elemCtx.updatePrimaryIntensiveQuantities(0);
404 const auto& iq0 = elemCtx.intensiveQuantities(0, 0);
405 const auto& fs = iq0.fluidState();
407 water_pressure_reservoir = fs.pressure(this->phaseIdx_()).value();
408 const auto water_density = fs.density(this->phaseIdx_());
411 this->gravity_() * (this->cell_depth_[idx] - this->aquiferDepth());
413 pw_aquifer.push_back(this->alphai_[idx] *
414 (water_pressure_reservoir - water_density.value()*gdz));
418 const auto& comm = this->simulator_.vanguard().grid().comm();
421 vals[0] = std::accumulate(this->alphai_.begin(), this->alphai_.end(), Scalar{0});
422 vals[1] = std::accumulate(pw_aquifer.begin(), pw_aquifer.end(), Scalar{0});
426 return vals[1] / vals[0];
429 const std::vector<Aquancon::AquancCell> connections_;
432 std::vector<Scalar> faceArea_connected_;
433 std::vector<int> cellToConnectionIdx_;
436 std::vector<Scalar> cell_depth_;
437 std::vector<Scalar> pressure_previous_;
438 std::vector<Eval> pressure_current_;
439 std::vector<Eval> Qai_;
440 std::vector<Scalar> alphai_;
444 std::optional<Scalar> Ta0_{};
447 Scalar total_face_area_{};
448 Scalar area_fraction_{Scalar{1}};
452 bool solution_set_from_restart_ {
false};
453 bool has_active_connection_on_proc_{
false};