/* Copyright 2017 TNO - Heat Transfer & Fluid Dynamics, Modelling & Optimization of the Subsurface Copyright 2017 Statoil ASA. 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 3 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 . */ namespace Opm { template BlackoilAquiferModel::BlackoilAquiferModel(Simulator& simulator) : simulator_(simulator) { // Grid needs to support Facetag using Grid = std::remove_const_t>; static_assert(SupportsFaceTag::value, "Grid has to support assumptions about face tag."); init(); } template void BlackoilAquiferModel::initialSolutionApplied() { for (auto& aquifer : aquifers) aquifer->initialSolutionApplied(); } template void BlackoilAquiferModel::initFromRestart(const data::Aquifers& aquiferSoln) { for (auto& aquifer : this->aquifers) aquifer->initFromRestart(aquiferSoln); } template void BlackoilAquiferModel::beginEpisode() {} template void BlackoilAquiferModel::beginIteration() {} template void BlackoilAquiferModel::beginTimeStep() { for (auto& aquifer : aquifers) aquifer->beginTimeStep(); } template template void BlackoilAquiferModel::addToSource(RateVector& rates, const Context& context, unsigned spaceIdx, unsigned timeIdx) const { for (auto& aquifer : aquifers) aquifer->addToSource(rates, context, spaceIdx, timeIdx); } template void BlackoilAquiferModel::addToSource(RateVector& rates, unsigned globalSpaceIdx, unsigned timeIdx) const { for (auto& aquifer : aquifers) aquifer->addToSource(rates, globalSpaceIdx, timeIdx); } template void BlackoilAquiferModel::endIteration() {} template void BlackoilAquiferModel::endTimeStep() { for (auto& aquifer : aquifers) { aquifer->endTimeStep(); using NumAq = AquiferNumerical; NumAq* num = dynamic_cast(aquifer.get()); if (num) this->simulator_.vanguard().grid().comm().barrier(); } } template void BlackoilAquiferModel::endEpisode() {} template template void BlackoilAquiferModel::serialize(Restarter& /* res */) { // TODO (?) throw std::logic_error("BlackoilAquiferModel::serialize() is not yet implemented"); } template template void BlackoilAquiferModel::deserialize(Restarter& /* res */) { // TODO (?) throw std::logic_error("BlackoilAquiferModel::deserialize() is not yet implemented"); } // Initialize the aquifers in the deck template void BlackoilAquiferModel::init() { const auto& aquifer = this->simulator_.vanguard().eclState().aquifer(); if (!aquifer.active()) { return; } const auto& connections = aquifer.connections(); for (const auto& aq : aquifer.ct()) { if (!connections.hasAquiferConnections(aq.aquiferID)) { auto msg = fmt::format("No valid connections for Carter-Tracy aquifer {}, aquifer {} will be ignored.", aq.aquiferID, aq.aquiferID); OpmLog::warning(msg); continue; } auto aqf = std::make_unique>(connections.getConnections(aq.aquiferID), this->simulator_, aq); aquifers.push_back(std::move(aqf)); } for (const auto& aq : aquifer.fetp()) { if (!connections.hasAquiferConnections(aq.aquiferID)) { auto msg = fmt::format("No valid connections for Fetkovich aquifer {}, aquifer {} will be ignored.", aq.aquiferID, aq.aquiferID); OpmLog::warning(msg); continue; } auto aqf = std::make_unique>(connections.getConnections(aq.aquiferID), this->simulator_, aq); aquifers.push_back(std::move(aqf)); } if (aquifer.hasNumericalAquifer()) { const auto& num_aquifers = aquifer.numericalAquifers().aquifers(); for ([[maybe_unused]]const auto& [id, aqu] : num_aquifers) { auto aqf = std::make_unique>(aqu, this->simulator_); aquifers.push_back(std::move(aqf)); } } } template data::Aquifers BlackoilAquiferModel::aquiferData() const { data::Aquifers data; for (const auto& aqu : this->aquifers) data.insert_or_assign(aqu->aquiferID(), aqu->aquiferData()); return data; } } // namespace Opm