/* Copyright 2017 SINTEF Digital, Mathematics and Cybernetics. Copyright 2017 Statoil ASA. Copyright 2016 - 2017 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 . */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace Opm { template void MultisegmentWellPrimaryVariables:: resize(const int numSegments) { value_.resize(numSegments); evaluation_.resize(numSegments); } template void MultisegmentWellPrimaryVariables:: init() { for (size_t seg = 0; seg < value_.size(); ++seg) { for (int eq_idx = 0; eq_idx < numWellEq; ++eq_idx) { evaluation_[seg][eq_idx] = 0.0; evaluation_[seg][eq_idx].setValue(value_[seg][eq_idx]); evaluation_[seg][eq_idx].setDerivative(eq_idx + Indices::numEq, 1.0); } } } template void MultisegmentWellPrimaryVariables:: update(const WellState& well_state, const bool stop_or_zero_rate_target) { static constexpr int Water = BlackoilPhases::Aqua; static constexpr int Gas = BlackoilPhases::Vapour; // TODO: to test using rate conversion coefficients to see if it will be better than // this default one if (!well_.isOperableAndSolvable() && !well_.wellIsStopped()) return; const Well& well = well_.wellEcl(); // the index of the top segment in the WellState const auto& ws = well_state.well(well_.indexOfWell()); const auto& segments = ws.segments; // maybe a throw for parallel running? assert(segments.size() == value_.size()); const auto& segment_rates = segments.rates; const auto& segment_pressure = segments.pressure; const PhaseUsage& pu = well_.phaseUsage(); for (size_t seg = 0; seg < value_.size(); ++seg) { // calculate the total rate for each segment double total_seg_rate = 0.0; // the segment pressure value_[seg][SPres] = segment_pressure[seg]; // TODO: under what kind of circustances, the following will be wrong? // the definition of g makes the gas phase is always the last phase for (int p = 0; p < well_.numPhases(); p++) { total_seg_rate += well_.scalingFactor(p) * segment_rates[well_.numPhases() * seg + p]; } if (seg == 0) { if (well_.isInjector()) { total_seg_rate = std::max(total_seg_rate, 0.); } else { total_seg_rate = std::min(total_seg_rate, 0.); } } value_[seg][WQTotal] = total_seg_rate; if (stop_or_zero_rate_target && seg == 0) { value_[seg][WQTotal] = 0; } if (std::abs(total_seg_rate) > 0.) { if (has_wfrac_variable) { const int water_pos = pu.phase_pos[Water]; value_[seg][WFrac] = well_.scalingFactor(water_pos) * segment_rates[well_.numPhases() * seg + water_pos] / total_seg_rate; } if (has_gfrac_variable) { const int gas_pos = pu.phase_pos[Gas]; value_[seg][GFrac] = well_.scalingFactor(gas_pos) * segment_rates[well_.numPhases() * seg + gas_pos] / total_seg_rate; } } else { // total_seg_rate == 0 if (well_.isInjector()) { // only single phase injection handled auto phase = well.getInjectionProperties().injectorType; if (has_wfrac_variable) { if (phase == InjectorType::WATER) { value_[seg][WFrac] = 1.0; } else { value_[seg][WFrac] = 0.0; } } if (has_gfrac_variable) { if (phase == InjectorType::GAS) { value_[seg][GFrac] = 1.0; } else { value_[seg][GFrac] = 0.0; } } } else if (well_.isProducer()) { // producers if (has_wfrac_variable) { value_[seg][WFrac] = 1.0 / well_.numPhases(); } if (has_gfrac_variable) { value_[seg][GFrac] = 1.0 / well_.numPhases(); } } } } } template void MultisegmentWellPrimaryVariables:: updateNewton(const BVectorWell& dwells, const double relaxation_factor, const double dFLimit, const bool stop_or_zero_rate_target, const double max_pressure_change) { const std::vector> old_primary_variables = value_; for (size_t seg = 0; seg < value_.size(); ++seg) { if (has_wfrac_variable) { const int sign = dwells[seg][WFrac] > 0. ? 1 : -1; const double dx_limited = sign * std::min(std::abs(dwells[seg][WFrac]) * relaxation_factor, dFLimit); value_[seg][WFrac] = old_primary_variables[seg][WFrac] - dx_limited; } if (has_gfrac_variable) { const int sign = dwells[seg][GFrac] > 0. ? 1 : -1; const double dx_limited = sign * std::min(std::abs(dwells[seg][GFrac]) * relaxation_factor, dFLimit); value_[seg][GFrac] = old_primary_variables[seg][GFrac] - dx_limited; } // handling the overshooting or undershooting of the fractions this->processFractions(seg); // update the segment pressure { const int sign = dwells[seg][SPres] > 0.? 1 : -1; const double dx_limited = sign * std::min(std::abs(dwells[seg][SPres]) * relaxation_factor, max_pressure_change); value_[seg][SPres] = std::max(old_primary_variables[seg][SPres] - dx_limited, 1e5); } // update the total rate // TODO: should we have a limitation of the total rate change? { value_[seg][WQTotal] = old_primary_variables[seg][WQTotal] - relaxation_factor * dwells[seg][WQTotal]; // make sure that no injector produce and no producer inject if (seg == 0) { if (well_.isInjector()) { value_[seg][WQTotal] = std::max(value_[seg][WQTotal], 0.0); } else { value_[seg][WQTotal] = std::min(value_[seg][WQTotal], 0.0); } } } } if (stop_or_zero_rate_target) { value_[0][WQTotal] = 0.; } } template void MultisegmentWellPrimaryVariables:: copyToWellState(const MultisegmentWellGeneric& mswell, const double rho, const bool stop_or_zero_rate_target, WellState& well_state, const SummaryState& summary_state, DeferredLogger& deferred_logger) const { static constexpr int Gas = BlackoilPhases::Vapour; static constexpr int Oil = BlackoilPhases::Liquid; static constexpr int Water = BlackoilPhases::Aqua; const auto pvtReg = std::max(well_.wellEcl().pvt_table_number() - 1, 0); const PhaseUsage& pu = well_.phaseUsage(); assert( FluidSystem::phaseIsActive(FluidSystem::oilPhaseIdx) ); const int oil_pos = pu.phase_pos[Oil]; auto& ws = well_state.well(well_.indexOfWell()); auto& segments = ws.segments; auto& segment_rates = segments.rates; auto& disgas = segments.dissolved_gas_rate; auto& vapoil = segments.vaporized_oil_rate; auto& segment_pressure = segments.pressure; for (size_t seg = 0; seg < value_.size(); ++seg) { std::vector fractions(well_.numPhases(), 0.0); fractions[oil_pos] = 1.0; if (FluidSystem::phaseIsActive(FluidSystem::waterPhaseIdx)) { const int water_pos = pu.phase_pos[Water]; fractions[water_pos] = value_[seg][WFrac]; fractions[oil_pos] -= fractions[water_pos]; } if (FluidSystem::phaseIsActive(FluidSystem::gasPhaseIdx)) { const int gas_pos = pu.phase_pos[Gas]; fractions[gas_pos] = value_[seg][GFrac]; fractions[oil_pos] -= fractions[gas_pos]; } // convert the fractions to be Q_p / G_total to calculate the phase rates for (int p = 0; p < well_.numPhases(); ++p) { const double scale = well_.scalingFactor(p); // for injection wells, there should only one non-zero scaling factor if (scale > 0.) { fractions[p] /= scale; } else { // this should only happens to injection wells fractions[p] = 0.; } } // calculate the phase rates based on the primary variables const double g_total = value_[seg][WQTotal]; for (int p = 0; p < well_.numPhases(); ++p) { const double phase_rate = g_total * fractions[p]; segment_rates[seg * well_.numPhases() + p] = phase_rate; if (seg == 0) { // top segment ws.surface_rates[p] = phase_rate; } } // update the segment pressure segment_pressure[seg] = value_[seg][SPres]; if (seg == 0) { // top segment ws.bhp = segment_pressure[seg]; } // Calculate other per-phase dynamic quantities. const auto temperature = 0.0; // Ignore thermal effects const auto saltConc = 0.0; // Ignore salt precipitation const auto Rvw = 0.0; // Ignore vaporised water. auto rsMax = 0.0; auto rvMax = 0.0; if (FluidSystem::phaseIsActive(FluidSystem::gasPhaseIdx)) { // Both oil and gas active. rsMax = FluidSystem::oilPvt() .saturatedGasDissolutionFactor(pvtReg, temperature, segment_pressure[seg]); rvMax = FluidSystem::gasPvt() .saturatedOilVaporizationFactor(pvtReg, temperature, segment_pressure[seg]); } // 1) Infer phase splitting for oil/gas. const auto& [Rs, Rv] = well_.rateConverter().inferDissolvedVaporisedRatio (rsMax, rvMax, segment_rates.begin() + seg * well_.numPhases()); if (! FluidSystem::phaseIsActive(FluidSystem::gasPhaseIdx)) { vapoil[seg] = disgas[seg] = 0.0; } else { const auto* qs = &segment_rates[seg * well_.numPhases()]; const auto denom = 1.0 - (Rs * Rv); const auto io = pu.phase_pos[Oil]; const auto ig = pu.phase_pos[Gas]; disgas[seg] = Rs * (qs[io] - Rv*qs[ig]) / denom; vapoil[seg] = Rv * (qs[ig] - Rs*qs[io]) / denom; } // 2) Local condition volume flow rates { // Use std::span<> in C++20 and beyond. const auto rate_start = seg * well_.numPhases(); const auto* surf_rates = segment_rates.data() + rate_start; auto* resv_rates = segments.phase_resv_rates.data() + rate_start; well_.rateConverter().calcReservoirVoidageRates (pvtReg, segment_pressure[seg], std::max(0.0, Rs), std::max(0.0, Rv), 0.0, // Rsw 0.0, // Rvw temperature, saltConc, surf_rates, resv_rates); } // 3) Local condition holdup fractions. const auto tot_resv = std::accumulate(segments.phase_resv_rates.begin() + (seg + 0) * well_.numPhases(), segments.phase_resv_rates.begin() + (seg + 1) * well_.numPhases(), 0.0); std::transform(segments.phase_resv_rates.begin() + (seg + 0) * well_.numPhases(), segments.phase_resv_rates.begin() + (seg + 1) * well_.numPhases(), segments.phase_holdup.begin() + (seg + 0) * well_.numPhases(), [tot_resv](const auto qr) { return std::clamp(qr / tot_resv, 0.0, 1.0); }); // 4) Local condition flow velocities for segments other than top segment. if (seg > 0) { // Possibly poor approximation // Velocity = Flow rate / cross-sectional area. // Additionally ignores drift flux. const auto area = well_.wellEcl().getSegments() .getFromSegmentNumber(segments.segment_number()[seg]).crossArea(); const auto velocity = (area > 0.0) ? tot_resv / area : 0.0; std::transform(segments.phase_holdup.begin() + (seg + 0) * well_.numPhases(), segments.phase_holdup.begin() + (seg + 1) * well_.numPhases(), segments.phase_velocity.begin() + (seg + 0) * well_.numPhases(), [velocity](const auto hf) { return (hf > 0.0) ? velocity : 0.0; }); } // 5) Local condition phase viscosities. segments.phase_viscosity[seg * well_.numPhases() + pu.phase_pos[Oil]] = FluidSystem::oilPvt().viscosity(pvtReg, temperature, segment_pressure[seg], Rs); if (FluidSystem::phaseIsActive(FluidSystem::waterPhaseIdx)) { segments.phase_viscosity[seg * well_.numPhases() + pu.phase_pos[Water]] = FluidSystem::waterPvt().viscosity(pvtReg, temperature, segment_pressure[seg], 0.0 /*Rsw*/, saltConc); } if (FluidSystem::phaseIsActive(FluidSystem::gasPhaseIdx)) { segments.phase_viscosity[seg * well_.numPhases() + pu.phase_pos[Gas]] = FluidSystem::gasPvt().viscosity(pvtReg, temperature, segment_pressure[seg], Rv, Rvw); } } // Segment flow velocity in top segment. { const auto np = well_.numPhases(); auto segVel = [&segments, np](const auto segmentNumber) { auto v = 0.0; const auto* vel = segments.phase_velocity.data() + segmentNumber*np; for (auto p = 0*np; p < np; ++p) { if (std::abs(vel[p]) > std::abs(v)) { v = vel[p]; } } return v; }; const auto seg = 0; auto maxVel = 0.0; for (const auto& inlet : mswell.segmentSet()[seg].inletSegments()) { const auto v = segVel(mswell.segmentNumberToIndex(inlet)); if (std::abs(v) > std::abs(maxVel)) { maxVel = v; } } std::transform(segments.phase_holdup.begin() + (seg + 0) * well_.numPhases(), segments.phase_holdup.begin() + (seg + 1) * well_.numPhases(), segments.phase_velocity.begin() + (seg + 0) * well_.numPhases(), [maxVel](const auto hf) { return (hf > 0.0) ? maxVel : 0.0; }); } // Note: for the ALQ value, in the StandardWell, WellInterfaceGeneric::getALQ(well_state) is used. // We might want to unify the way regarding AQL value. WellBhpThpCalculator(well_) .updateThp(rho, stop_or_zero_rate_target, [this]() { return well_.wellEcl().alq_value(); }, {FluidSystem::phaseIsActive(FluidSystem::waterPhaseIdx), FluidSystem::phaseIsActive(FluidSystem::oilPhaseIdx), FluidSystem::phaseIsActive(FluidSystem::gasPhaseIdx)}, well_state, summary_state, deferred_logger); } template void MultisegmentWellPrimaryVariables:: processFractions(const int seg) { static constexpr int Water = BlackoilPhases::Aqua; static constexpr int Oil = BlackoilPhases::Liquid; static constexpr int Gas = BlackoilPhases::Vapour; const PhaseUsage& pu = well_.phaseUsage(); std::vector fractions(well_.numPhases(), 0.0); assert(FluidSystem::phaseIsActive(FluidSystem::oilPhaseIdx)); const int oil_pos = pu.phase_pos[Oil]; fractions[oil_pos] = 1.0; if ( FluidSystem::phaseIsActive(FluidSystem::waterPhaseIdx) ) { const int water_pos = pu.phase_pos[Water]; fractions[water_pos] = value_[seg][WFrac]; fractions[oil_pos] -= fractions[water_pos]; } if ( FluidSystem::phaseIsActive(FluidSystem::gasPhaseIdx) ) { const int gas_pos = pu.phase_pos[Gas]; fractions[gas_pos] = value_[seg][GFrac]; fractions[oil_pos] -= fractions[gas_pos]; } if (FluidSystem::phaseIsActive(FluidSystem::waterPhaseIdx)) { const int water_pos = pu.phase_pos[Water]; if (fractions[water_pos] < 0.0) { if ( FluidSystem::phaseIsActive(FluidSystem::gasPhaseIdx) ) { fractions[pu.phase_pos[Gas]] /= (1.0 - fractions[water_pos]); } fractions[oil_pos] /= (1.0 - fractions[water_pos]); fractions[water_pos] = 0.0; } } if (FluidSystem::phaseIsActive(FluidSystem::gasPhaseIdx)) { const int gas_pos = pu.phase_pos[Gas]; if (fractions[gas_pos] < 0.0) { if ( FluidSystem::phaseIsActive(FluidSystem::waterPhaseIdx) ) { fractions[pu.phase_pos[Water]] /= (1.0 - fractions[gas_pos]); } fractions[oil_pos] /= (1.0 - fractions[gas_pos]); fractions[gas_pos] = 0.0; } } if (fractions[oil_pos] < 0.0) { if ( FluidSystem::phaseIsActive(FluidSystem::waterPhaseIdx) ) { fractions[pu.phase_pos[Water]] /= (1.0 - fractions[oil_pos]); } if ( FluidSystem::phaseIsActive(FluidSystem::gasPhaseIdx) ) { fractions[pu.phase_pos[Gas]] /= (1.0 - fractions[oil_pos]); } fractions[oil_pos] = 0.0; } if ( FluidSystem::phaseIsActive(FluidSystem::waterPhaseIdx) ) { value_[seg][WFrac] = fractions[pu.phase_pos[Water]]; } if ( FluidSystem::phaseIsActive(FluidSystem::gasPhaseIdx) ) { value_[seg][GFrac] = fractions[pu.phase_pos[Gas]]; } } template typename MultisegmentWellPrimaryVariables::EvalWell MultisegmentWellPrimaryVariables:: volumeFraction(const int seg, const unsigned compIdx) const { if (has_wfrac_variable && compIdx == Indices::canonicalToActiveComponentIndex(FluidSystem::waterCompIdx)) { return evaluation_[seg][WFrac]; } if (has_gfrac_variable && compIdx == Indices::canonicalToActiveComponentIndex(FluidSystem::gasCompIdx)) { return evaluation_[seg][GFrac]; } // Oil fraction EvalWell oil_fraction = 1.0; if (has_wfrac_variable) { oil_fraction -= evaluation_[seg][WFrac]; } if (has_gfrac_variable) { oil_fraction -= evaluation_[seg][GFrac]; } /* if (has_solvent) { oil_fraction -= evaluation_[seg][SFrac]; } */ return oil_fraction; } template typename MultisegmentWellPrimaryVariables::EvalWell MultisegmentWellPrimaryVariables:: volumeFractionScaled(const int seg, const int comp_idx) const { // For reservoir rate control, the distr in well control is used for the // rate conversion coefficients. For the injection well, only the distr of the injection // phase is not zero. const double scale = well_.scalingFactor(well_.ebosCompIdxToFlowCompIdx(comp_idx)); if (scale > 0.) { return this->volumeFraction(seg, comp_idx) / scale; } return this->volumeFraction(seg, comp_idx); } template typename MultisegmentWellPrimaryVariables::EvalWell MultisegmentWellPrimaryVariables:: surfaceVolumeFraction(const int seg, const int comp_idx) const { EvalWell sum_volume_fraction_scaled = 0.; for (int idx = 0; idx < well_.numComponents(); ++idx) { sum_volume_fraction_scaled += this->volumeFractionScaled(seg, idx); } assert(sum_volume_fraction_scaled.value() != 0.); return this->volumeFractionScaled(seg, comp_idx) / sum_volume_fraction_scaled; } template typename MultisegmentWellPrimaryVariables::EvalWell MultisegmentWellPrimaryVariables:: getSegmentRateUpwinding(const int seg, const int seg_upwind, const size_t comp_idx) const { // the result will contain the derivative with respect to WQTotal in segment seg, // and the derivatives with respect to WFrac GFrac in segment seg_upwind. // the derivative with respect to SPres should be zero. if (seg == 0 && well_.isInjector()) { const Well& well = well_.wellEcl(); auto phase = well.getInjectionProperties().injectorType; if (FluidSystem::phaseIsActive(FluidSystem::waterPhaseIdx) && Indices::canonicalToActiveComponentIndex(FluidSystem::waterCompIdx) == comp_idx && phase == InjectorType::WATER) return evaluation_[seg][WQTotal] / well_.scalingFactor(well_.ebosCompIdxToFlowCompIdx(comp_idx)); if (FluidSystem::phaseIsActive(FluidSystem::oilPhaseIdx) && Indices::canonicalToActiveComponentIndex(FluidSystem::oilCompIdx) == comp_idx && phase == InjectorType::OIL) return evaluation_[seg][WQTotal] / well_.scalingFactor(well_.ebosCompIdxToFlowCompIdx(comp_idx)); if (FluidSystem::phaseIsActive(FluidSystem::gasPhaseIdx) && Indices::canonicalToActiveComponentIndex(FluidSystem::gasCompIdx) == comp_idx && phase == InjectorType::GAS) return evaluation_[seg][WQTotal] / well_.scalingFactor(well_.ebosCompIdxToFlowCompIdx(comp_idx)); return 0.0; } const EvalWell segment_rate = evaluation_[seg][WQTotal] * this->volumeFractionScaled(seg_upwind, comp_idx); assert(segment_rate.derivative(SPres + Indices::numEq) == 0.); return segment_rate; } template typename MultisegmentWellPrimaryVariables::EvalWell MultisegmentWellPrimaryVariables:: getSegmentPressure(const int seg) const { return evaluation_[seg][SPres]; } template typename MultisegmentWellPrimaryVariables::EvalWell MultisegmentWellPrimaryVariables:: getBhp() const { return this->getSegmentPressure(0); } template typename MultisegmentWellPrimaryVariables::EvalWell MultisegmentWellPrimaryVariables:: getSegmentRate(const int seg, const int comp_idx) const { return evaluation_[seg][WQTotal] * this->volumeFractionScaled(seg, comp_idx); } template typename MultisegmentWellPrimaryVariables::EvalWell MultisegmentWellPrimaryVariables:: getQs(const int comp_idx) const { return this->getSegmentRate(0, comp_idx); } template typename MultisegmentWellPrimaryVariables::EvalWell MultisegmentWellPrimaryVariables:: getWQTotal() const { return evaluation_[0][WQTotal]; } #define INSTANCE(...) \ template class MultisegmentWellPrimaryVariables,__VA_ARGS__,double>; // One phase INSTANCE(BlackOilOnePhaseIndices<0u,0u,0u,0u,false,false,0u,1u,0u>) INSTANCE(BlackOilOnePhaseIndices<0u,0u,0u,1u,false,false,0u,1u,0u>) INSTANCE(BlackOilOnePhaseIndices<0u,0u,0u,0u,false,false,0u,1u,5u>) // Two phase INSTANCE(BlackOilTwoPhaseIndices<0u,0u,0u,0u,false,false,0u,0u,0u>) INSTANCE(BlackOilTwoPhaseIndices<0u,0u,0u,0u,false,false,0u,1u,0u>) INSTANCE(BlackOilTwoPhaseIndices<0u,0u,0u,0u,false,false,0u,2u,0u>) INSTANCE(BlackOilTwoPhaseIndices<0u,0u,1u,0u,false,false,0u,2u,0u>) INSTANCE(BlackOilTwoPhaseIndices<0u,0u,0u,1u,false,false,0u,1u,0u>) INSTANCE(BlackOilTwoPhaseIndices<0u,0u,0u,0u,false,true,0u,0u,0u>) INSTANCE(BlackOilTwoPhaseIndices<0u,0u,0u,0u,false,true,0u,2u,0u>) INSTANCE(BlackOilTwoPhaseIndices<0u,0u,2u,0u,false,false,0u,2u,0u>) INSTANCE(BlackOilTwoPhaseIndices<0u,0u,0u,1u,false,false,0u,0u,0u>) INSTANCE(BlackOilTwoPhaseIndices<0u,0u,0u,1u,false,true,0u,0u,0u>) INSTANCE(BlackOilTwoPhaseIndices<1u,0u,0u,0u,false,false,0u,0u,0u>) // Blackoil INSTANCE(BlackOilIndices<0u,0u,0u,0u,false,false,0u,0u>) INSTANCE(BlackOilIndices<1u,0u,0u,0u,false,false,0u,0u>) INSTANCE(BlackOilIndices<0u,1u,0u,0u,false,false,0u,0u>) INSTANCE(BlackOilIndices<0u,0u,1u,0u,false,false,0u,0u>) INSTANCE(BlackOilIndices<0u,0u,0u,1u,false,false,0u,0u>) INSTANCE(BlackOilIndices<0u,0u,0u,0u,true,false,0u,0u>) INSTANCE(BlackOilIndices<0u,0u,0u,0u,false,true,0u,0u>) INSTANCE(BlackOilIndices<0u,0u,0u,1u,false,true,0u,0u>) INSTANCE(BlackOilIndices<0u,0u,0u,0u,false,false,1u,0u>) INSTANCE(BlackOilIndices<0u,0u,0u,0u,false,true,2u,0u>) INSTANCE(BlackOilIndices<1u,0u,0u,0u,true,false,0u,0u>) }