mirror of
https://github.com/OPM/opm-simulators.git
synced 2025-01-16 00:41:56 -06:00
203 lines
8.8 KiB
C++
203 lines
8.8 KiB
C++
/*
|
|
Copyright 2012 SINTEF ICT, Applied Mathematics.
|
|
|
|
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/>.
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include <opm/polymer/PolymerInflow.hpp>
|
|
#include <opm/core/wells.h>
|
|
#include <opm/parser/eclipse/Deck/Deck.hpp>
|
|
#include <opm/parser/eclipse/EclipseState/Schedule/Well.hpp>
|
|
#include <opm/parser/eclipse/EclipseState/EclipseState.hpp>
|
|
#include <opm/parser/eclipse/EclipseState/Schedule/Schedule.hpp>
|
|
#include <opm/parser/eclipse/EclipseState/Schedule/ScheduleEnums.hpp>
|
|
#include <opm/parser/eclipse/EclipseState/Schedule/WellPolymerProperties.hpp>
|
|
#include <opm/parser/eclipse/EclipseState/Schedule/WellInjectionProperties.hpp>
|
|
#include <map>
|
|
#include <unordered_map>
|
|
#include <memory>
|
|
namespace Opm
|
|
{
|
|
|
|
// ---------- Methods of PolymerInflowBasic ----------
|
|
|
|
/// Constructor.
|
|
/// @param[in] starttime Start time of injection in seconds.
|
|
/// @param[in] endtime End time of injection in seconds.
|
|
/// @param[in] amount Amount to be injected per second.
|
|
PolymerInflowBasic::PolymerInflowBasic(const double starttime,
|
|
const double endtime,
|
|
const double amount)
|
|
: stime_(starttime), etime_(endtime), amount_(amount)
|
|
{
|
|
}
|
|
|
|
void PolymerInflowBasic::getInflowValues(const double step_start,
|
|
const double step_end,
|
|
std::vector<double>& poly_inflow_c) const
|
|
{
|
|
const double eps = 1e-5*(step_end - step_start);
|
|
if (step_start + eps >= stime_ && step_end - eps <= etime_) {
|
|
std::fill(poly_inflow_c.begin(), poly_inflow_c.end(), amount_);
|
|
} else if (step_start + eps <= etime_ && step_end - eps >= stime_) {
|
|
OPM_MESSAGE("Warning: polymer injection set to change inside timestep. Using value at start of step.");
|
|
std::fill(poly_inflow_c.begin(), poly_inflow_c.end(), amount_);
|
|
} else {
|
|
std::fill(poly_inflow_c.begin(), poly_inflow_c.end(), 0.0);
|
|
}
|
|
}
|
|
|
|
|
|
// ---------- Methods of PolymerInflowFromDeck ----------
|
|
|
|
|
|
|
|
/// Constructor.
|
|
/// @param[in] deck Input deck expected to contain WPOLYMER.
|
|
PolymerInflowFromDeck::PolymerInflowFromDeck(Opm::DeckConstPtr deck,
|
|
const Wells& wells,
|
|
const int num_cells)
|
|
: sparse_inflow_(num_cells)
|
|
{
|
|
if (!deck->hasKeyword("WPOLYMER")) {
|
|
OPM_MESSAGE("PolymerInflowFromDeck initialized without WPOLYMER in current epoch.");
|
|
return;
|
|
}
|
|
|
|
// Extract concentrations and put into cell->concentration map.
|
|
Opm::DeckKeywordConstPtr wpolymerKeyword = deck->getKeyword("WPOLYMER");
|
|
const int num_wpl = wpolymerKeyword->size();
|
|
std::map<int, double> perfcell_conc;
|
|
for (int i = 0; i < num_wpl; ++i) {
|
|
// Only use well name and polymer concentration.
|
|
// That is, we ignore salt concentration and group
|
|
// names.
|
|
int wix = 0;
|
|
for (; wix < wells.number_of_wells; ++wix) {
|
|
if (wpolymerKeyword->getRecord(i)->getItem("WELL")->getString(0) == wells.name[wix]) {
|
|
break;
|
|
}
|
|
}
|
|
if (wix == wells.number_of_wells) {
|
|
OPM_THROW(std::runtime_error, "Could not find a match for well "
|
|
<< wpolymerKeyword->getRecord(i)->getItem("WELL")->getString(0)
|
|
<< " from WPOLYMER.");
|
|
}
|
|
for (int j = wells.well_connpos[wix]; j < wells.well_connpos[wix+1]; ++j) {
|
|
const int perf_cell = wells.well_cells[j];
|
|
perfcell_conc[perf_cell] =
|
|
wpolymerKeyword->getRecord(i)->getItem("POLYMER_CONCENTRATION")->getSIDouble(0);
|
|
}
|
|
}
|
|
|
|
// Build sparse vector from map.
|
|
std::map<int, double>::const_iterator it = perfcell_conc.begin();
|
|
for (; it != perfcell_conc.end(); ++it) {
|
|
sparse_inflow_.addElement(it->second, it->first);
|
|
}
|
|
}
|
|
|
|
void
|
|
PolymerInflowFromDeck::setInflowValues(Opm::DeckConstPtr deck,
|
|
Opm::EclipseStateConstPtr eclipseState,
|
|
size_t currentStep)
|
|
{
|
|
Opm::DeckKeywordConstPtr keyword = deck->getKeyword("WPOLYMER");
|
|
|
|
// Schedule schedule(deck);
|
|
ScheduleConstPtr schedule = eclipseState->getSchedule();
|
|
for (size_t recordNr = 0; recordNr < keyword->size(); recordNr++) {
|
|
DeckRecordConstPtr record = keyword->getRecord(recordNr);
|
|
|
|
const std::string& wellNamesPattern = record->getItem("WELL")->getTrimmedString(0);
|
|
std::string wellName = record->getItem("WELL")->getTrimmedString(0);
|
|
std::vector<WellPtr> wells = schedule->getWells(wellNamesPattern);
|
|
for (auto wellIter = wells.begin(); wellIter != wells.end(); ++wellIter) {
|
|
WellPtr well = *wellIter;
|
|
WellInjectionProperties injection = well->getInjectionProperties(currentStep);
|
|
if (injection.injectorType == WellInjector::WATER) {
|
|
WellPolymerProperties polymer = well->getPolymerProperties(currentStep);
|
|
wellPolymerRate_.insert(std::make_pair(wellName, polymer.m_polymerConcentration));
|
|
} else {
|
|
OPM_THROW(std::logic_error, "For polymer injector you must have a water injector");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Constructor.
|
|
/// @param[in] deck Input deck expected to contain WPOLYMER.
|
|
PolymerInflowFromDeck::PolymerInflowFromDeck(Opm::DeckConstPtr deck,
|
|
Opm::EclipseStateConstPtr eclipseState,
|
|
const Wells& wells,
|
|
const int num_cells,
|
|
size_t currentStep)
|
|
: sparse_inflow_(num_cells)
|
|
{
|
|
if (!deck->hasKeyword("WPOLYMER")) {
|
|
OPM_MESSAGE("PolymerInflowFromDeck initialized without WPOLYMER in current epoch.");
|
|
return;
|
|
}
|
|
setInflowValues(deck, eclipseState, currentStep);
|
|
|
|
std::unordered_map<std::string, double>::const_iterator map_it;
|
|
// Extract concentrations and put into cell->concentration map.
|
|
std::map<int, double> perfcell_conc;
|
|
for (size_t i = 0; i < wellPolymerRate_.size(); ++i) {
|
|
// Only use well name and polymer concentration.
|
|
// That is, we ignore salt concentration and group
|
|
// names.
|
|
int wix = 0;
|
|
for (; wix < wells.number_of_wells; ++wix) {
|
|
map_it = wellPolymerRate_.find(wells.name[wix]);
|
|
if (map_it == wellPolymerRate_.end()) {
|
|
OPM_THROW(std::runtime_error, "Could not find a match for well from WPOLYMER.");
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
for (int j = wells.well_connpos[wix]; j < wells.well_connpos[wix+1]; ++j) {
|
|
const int perf_cell = wells.well_cells[j];
|
|
perfcell_conc[perf_cell] = map_it->second;
|
|
}
|
|
}
|
|
|
|
// Build sparse vector from map.
|
|
std::map<int, double>::const_iterator it = perfcell_conc.begin();
|
|
for (; it != perfcell_conc.end(); ++it) {
|
|
sparse_inflow_.addElement(it->second, it->first);
|
|
}
|
|
}
|
|
|
|
void PolymerInflowFromDeck::getInflowValues(const double /*step_start*/,
|
|
const double /*step_end*/,
|
|
std::vector<double>& poly_inflow_c) const
|
|
{
|
|
// This method does not depend on the given time,
|
|
// instead one would have a new epoch (and create a new
|
|
// instance) for each change in WPOLYMER.
|
|
std::fill(poly_inflow_c.begin(), poly_inflow_c.end(), 0.0);
|
|
const int nnz = sparse_inflow_.nonzeroSize();
|
|
for (int i = 0; i < nnz; ++i) {
|
|
poly_inflow_c[sparse_inflow_.nonzeroIndex(i)] = sparse_inflow_.nonzeroElement(i) ;
|
|
}
|
|
}
|
|
|
|
|
|
} // namespace Opm
|