mirror of
https://github.com/OPM/opm-simulators.git
synced 2025-02-25 18:55:30 -06:00
i.e. it now supports stuff like MULTFLT in the schedule section. Possibly, the MPI-parallel code paths need some fixes. (but if the geology is not changed during the simulation, the parallel code will do the same as before.) the most fundamental change of this patch is that the reference/pointer to the DerivedGeology object is made non-constant. IMO that's okay, though, becase the geology can no longer assumed to be constant over the whole simulation run.
146 lines
6.6 KiB
C++
146 lines
6.6 KiB
C++
/*
|
|
Copyright 2015 IRIS AS
|
|
|
|
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 <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#ifndef OPM_SIMULATORFULLYIMPLICITBLACKOILSOLVENT_IMPL_HEADER_INCLUDED
|
|
#define OPM_SIMULATORFULLYIMPLICITBLACKOILSOLVENT_IMPL_HEADER_INCLUDED
|
|
|
|
namespace Opm
|
|
{
|
|
template <class GridT>
|
|
SimulatorFullyImplicitBlackoilSolvent<GridT>::
|
|
SimulatorFullyImplicitBlackoilSolvent(const parameter::ParameterGroup& param,
|
|
const GridT& grid,
|
|
DerivedGeology& geo,
|
|
BlackoilPropsAdInterface& props,
|
|
const SolventPropsAdFromDeck& solvent_props,
|
|
const RockCompressibility* rock_comp_props,
|
|
NewtonIterationBlackoilInterface& linsolver,
|
|
const double* gravity,
|
|
const bool has_disgas,
|
|
const bool has_vapoil,
|
|
std::shared_ptr<EclipseState> eclipse_state,
|
|
BlackoilOutputWriter& output_writer,
|
|
Opm::DeckConstPtr& deck,
|
|
const std::vector<double>& threshold_pressures_by_face,
|
|
const bool has_solvent)
|
|
: BaseType(param,
|
|
grid,
|
|
geo,
|
|
props,
|
|
rock_comp_props,
|
|
linsolver,
|
|
gravity,
|
|
has_disgas,
|
|
has_vapoil,
|
|
eclipse_state,
|
|
output_writer,
|
|
threshold_pressures_by_face)
|
|
, has_solvent_(has_solvent)
|
|
, deck_(deck)
|
|
, solvent_props_(solvent_props)
|
|
{
|
|
if(deck->hasKeyword("MISCIBLE")) {
|
|
std::cerr << "MISICIBLE keyword is present. Mixing is not currently supported" << std::endl;
|
|
}
|
|
}
|
|
|
|
template <class GridT>
|
|
auto SimulatorFullyImplicitBlackoilSolvent<GridT>::
|
|
createSolver(const Wells* wells)
|
|
-> std::unique_ptr<Solver>
|
|
{
|
|
typedef typename Traits::Model Model;
|
|
|
|
|
|
auto model = std::unique_ptr<Model>(new Model(BaseType::model_param_,
|
|
BaseType::grid_,
|
|
BaseType::props_,
|
|
BaseType::geo_,
|
|
BaseType::rock_comp_props_,
|
|
solvent_props_,
|
|
wells,
|
|
BaseType::solver_,
|
|
BaseType::eclipse_state_,
|
|
BaseType::has_disgas_,
|
|
BaseType::has_vapoil_,
|
|
BaseType::terminal_output_,
|
|
has_solvent_));
|
|
|
|
if (!BaseType::threshold_pressures_by_face_.empty()) {
|
|
model->setThresholdPressures(BaseType::threshold_pressures_by_face_);
|
|
}
|
|
|
|
return std::unique_ptr<Solver>(new Solver(BaseType::solver_param_, std::move(model)));
|
|
}
|
|
|
|
template <class GridT>
|
|
void SimulatorFullyImplicitBlackoilSolvent<GridT>::
|
|
handleAdditionalWellInflow(SimulatorTimer& timer,
|
|
WellsManager& /*wells_manager*/,
|
|
typename BaseType::WellState& well_state,
|
|
const Wells* wells)
|
|
{
|
|
// compute solvent inflow
|
|
const int nw = wells->number_of_wells;
|
|
std::vector<double> perfcells_fraction(wells->well_connpos[nw], 0.0);
|
|
|
|
if (deck_->hasKeyword("WSOLVENT")) {
|
|
|
|
size_t currentStep = timer.currentStepNum();
|
|
ScheduleConstPtr schedule = BaseType::eclipse_state_->getSchedule();
|
|
Opm::DeckKeywordConstPtr keyword = deck_->getKeyword("WSOLVENT");
|
|
const int num_keywords = keyword->size();
|
|
|
|
for (int recordNr = 0; recordNr < num_keywords; ++recordNr) {
|
|
DeckRecordConstPtr record = keyword->getRecord(recordNr);
|
|
const std::string& wellNamesPattern = record->getItem("WELL")->getTrimmedString(0);
|
|
std::vector<WellPtr> wells_solvent = schedule->getWells(wellNamesPattern);
|
|
for (auto wellIter = wells_solvent.begin(); wellIter != wells_solvent.end(); ++wellIter) {
|
|
WellPtr well_solvent = *wellIter;
|
|
WellInjectionProperties injection = well_solvent->getInjectionProperties(currentStep);
|
|
if (injection.injectorType == WellInjector::GAS) {
|
|
double solventFraction = well_solvent->getSolventFraction(currentStep);
|
|
// Find the solvent well in the well list and add properties to it
|
|
int wix = 0;
|
|
for (; wix < nw; ++wix) {
|
|
if (wellNamesPattern == wells->name[wix]) {
|
|
break;
|
|
}
|
|
}
|
|
if (wix == wells->number_of_wells) {
|
|
OPM_THROW(std::runtime_error, "Could not find a match for well "
|
|
<< wellNamesPattern
|
|
<< " from WSOLVENT.");
|
|
}
|
|
for (int j = wells->well_connpos[wix]; j < wells->well_connpos[wix+1]; ++j) {
|
|
perfcells_fraction[j] = solventFraction;
|
|
}
|
|
} else {
|
|
OPM_THROW(std::logic_error, "For solvent injector you must have a gas injector");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
well_state.solventFraction() = perfcells_fraction;
|
|
}
|
|
|
|
} // namespace Opm
|
|
|
|
#endif // OPM_SIMULATORFULLYIMPLICITBLACKOILSOLVENT_IMPL_HEADER_INCLUDED
|