/* Copyright 2020 Equinor 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 . */ #include #include #include #include #include #include #include #include #include #include #include namespace Opm { namespace WellGroupHelpers { TargetCalculator::TargetCalculator(const Group::ProductionCMode cmode, const PhaseUsage& pu, const std::vector& resv_coeff, const double group_grat_target_from_sales, const std::string& group_name, const GroupState& group_state, const bool use_gpmaint) : cmode_(cmode) , pu_(pu) , resv_coeff_(resv_coeff) , group_grat_target_from_sales_(group_grat_target_from_sales) , group_name_(group_name) , group_state_(group_state) , use_gpmaint_(use_gpmaint) { } template RateType TargetCalculator::calcModeRateFromRates(const RateType* rates) const { switch (cmode_) { case Group::ProductionCMode::ORAT: { assert(pu_.phase_used[BlackoilPhases::Liquid]); const int pos = pu_.phase_pos[BlackoilPhases::Liquid]; return rates[pos]; } case Group::ProductionCMode::WRAT: { assert(pu_.phase_used[BlackoilPhases::Aqua]); const int pos = pu_.phase_pos[BlackoilPhases::Aqua]; return rates[pos]; } case Group::ProductionCMode::GRAT: { assert(pu_.phase_used[BlackoilPhases::Vapour]); const int pos = pu_.phase_pos[BlackoilPhases::Vapour]; return rates[pos]; } case Group::ProductionCMode::LRAT: { assert(pu_.phase_used[BlackoilPhases::Liquid]); assert(pu_.phase_used[BlackoilPhases::Aqua]); const int opos = pu_.phase_pos[BlackoilPhases::Liquid]; const int wpos = pu_.phase_pos[BlackoilPhases::Aqua]; return rates[opos] + rates[wpos]; } case Group::ProductionCMode::RESV: { auto mode_rate = rates[0] * resv_coeff_[0]; for (int phase = 1; phase < pu_.num_phases; ++phase) { mode_rate += rates[phase] * resv_coeff_[phase]; } return mode_rate; } default: // Should never be here. assert(false); return rates[0]; } } double TargetCalculator::groupTarget(const Group::ProductionControls ctrl) const { switch (cmode_) { case Group::ProductionCMode::ORAT: return ctrl.oil_target; case Group::ProductionCMode::WRAT: return ctrl.water_target; case Group::ProductionCMode::GRAT: { // gas target may have been adjusted by GCONSALE if ( group_grat_target_from_sales_ > 0) return group_grat_target_from_sales_; return ctrl.gas_target; } case Group::ProductionCMode::LRAT: return ctrl.liquid_target; case Group::ProductionCMode::RESV: { if(use_gpmaint_ && this->group_state_.has_gpmaint_target(this->group_name_)) return this->group_state_.gpmaint_target(this->group_name_); return ctrl.resv_target; } default: // Should never be here. assert(false); return 0.0; } } GuideRateModel::Target TargetCalculator::guideTargetMode() const { switch (cmode_) { case Group::ProductionCMode::ORAT: return GuideRateModel::Target::OIL; case Group::ProductionCMode::WRAT: return GuideRateModel::Target::WAT; case Group::ProductionCMode::GRAT: return GuideRateModel::Target::GAS; case Group::ProductionCMode::LRAT: return GuideRateModel::Target::LIQ; case Group::ProductionCMode::RESV: return GuideRateModel::Target::RES; default: // Should never be here. assert(false); return GuideRateModel::Target::NONE; } } InjectionTargetCalculator::InjectionTargetCalculator(const Group::InjectionCMode& cmode, const PhaseUsage& pu, const std::vector& resv_coeff, const std::string& group_name, const double sales_target, const GroupState& group_state, const Phase& injection_phase, const bool use_gpmaint, DeferredLogger& deferred_logger) : cmode_(cmode) , pu_(pu) , resv_coeff_(resv_coeff) , group_name_(group_name) , sales_target_(sales_target) , group_state_(group_state) , use_gpmaint_(use_gpmaint) { // initialize to avoid warning pos_ = pu.phase_pos[BlackoilPhases::Aqua]; target_ = GuideRateModel::Target::WAT; switch (injection_phase) { case Phase::WATER: { pos_ = pu.phase_pos[BlackoilPhases::Aqua]; target_ = GuideRateModel::Target::WAT; break; } case Phase::OIL: { pos_ = pu.phase_pos[BlackoilPhases::Liquid]; target_ = GuideRateModel::Target::OIL; break; } case Phase::GAS: { pos_ = pu.phase_pos[BlackoilPhases::Vapour]; target_ = GuideRateModel::Target::GAS; break; } default: OPM_DEFLOG_THROW(std::logic_error, "Invalid injection phase in InjectionTargetCalculator", deferred_logger); } } double InjectionTargetCalculator::groupTarget(const Group::InjectionControls& ctrl, Opm::DeferredLogger& deferred_logger) const { switch (cmode_) { case Group::InjectionCMode::RATE: if(use_gpmaint_ && this->group_state_.has_gpmaint_target(this->group_name_)) return this->group_state_.gpmaint_target(this->group_name_); return ctrl.surface_max_rate; case Group::InjectionCMode::RESV: if(use_gpmaint_ && this->group_state_.has_gpmaint_target(this->group_name_)) return this->group_state_.gpmaint_target(this->group_name_) / resv_coeff_[pos_]; return ctrl.resv_max_rate / resv_coeff_[pos_]; case Group::InjectionCMode::REIN: { double production_rate = this->group_state_.injection_rein_rates(ctrl.reinj_group)[pos_]; return ctrl.target_reinj_fraction * production_rate; } case Group::InjectionCMode::VREP: { const std::vector& group_injection_reductions = this->group_state_.injection_reduction_rates(this->group_name_); double voidage_rate = group_state_.injection_vrep_rate(ctrl.voidage_group) * ctrl.target_void_fraction; double inj_reduction = 0.0; if (ctrl.phase != Phase::WATER) inj_reduction += group_injection_reductions[pu_.phase_pos[BlackoilPhases::Aqua]] * resv_coeff_[pu_.phase_pos[BlackoilPhases::Aqua]]; if (ctrl.phase != Phase::OIL) inj_reduction += group_injection_reductions[pu_.phase_pos[BlackoilPhases::Liquid]] * resv_coeff_[pu_.phase_pos[BlackoilPhases::Liquid]]; if (ctrl.phase != Phase::GAS) inj_reduction += group_injection_reductions[pu_.phase_pos[BlackoilPhases::Vapour]] * resv_coeff_[pu_.phase_pos[BlackoilPhases::Vapour]]; voidage_rate -= inj_reduction; return voidage_rate / resv_coeff_[pos_]; } case Group::InjectionCMode::SALE: { assert(pos_ == pu_.phase_pos[BlackoilPhases::Vapour]); // Gas injection rate = Total gas production rate + gas import rate - gas consumption rate - sales rate; // Gas import and consumption is already included in the REIN rates double inj_rate = group_state_.injection_rein_rates(this->group_name_)[pos_]; inj_rate -= sales_target_; return inj_rate; } default: OPM_DEFLOG_THROW(std::logic_error, "Invalid Group::InjectionCMode in InjectionTargetCalculator", deferred_logger); return 0.0; } } GuideRateModel::Target InjectionTargetCalculator::guideTargetMode() const { return target_; } #define INSTANCE_TARGET_CALCULATOR(...) \ template __VA_ARGS__ TargetCalculator::calcModeRateFromRates<__VA_ARGS__>(const __VA_ARGS__* rates) const; INSTANCE_TARGET_CALCULATOR(double) INSTANCE_TARGET_CALCULATOR(DenseAd::Evaluation) INSTANCE_TARGET_CALCULATOR(DenseAd::Evaluation) INSTANCE_TARGET_CALCULATOR(DenseAd::Evaluation) INSTANCE_TARGET_CALCULATOR(DenseAd::Evaluation) INSTANCE_TARGET_CALCULATOR(DenseAd::Evaluation) INSTANCE_TARGET_CALCULATOR(DenseAd::Evaluation) INSTANCE_TARGET_CALCULATOR(DenseAd::Evaluation) INSTANCE_TARGET_CALCULATOR(DenseAd::Evaluation) INSTANCE_TARGET_CALCULATOR(DenseAd::Evaluation) INSTANCE_TARGET_CALCULATOR(DenseAd::Evaluation) INSTANCE_TARGET_CALCULATOR(DenseAd::Evaluation) INSTANCE_TARGET_CALCULATOR(DenseAd::Evaluation) INSTANCE_TARGET_CALCULATOR(DenseAd::Evaluation) } // namespace WellGroupHelpers } // namespace Opm