/* Copyright 2016 SINTEF ICT, Applied Mathematics. Copyright 2016 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 . */ #ifndef OPM_STANDARDWELLS_HEADER_INCLUDED #define OPM_STANDARDWELLS_HEADER_INCLUDED #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace Opm { /// Class for handling the standard well model. class StandardWells { public: struct WellOps { explicit WellOps(const Wells* wells); Eigen::SparseMatrix w2p; // well -> perf (scatter) Eigen::SparseMatrix p2w; // perf -> well (gather) std::vector well_cells; // the set of perforated cells }; // --------- Types --------- using ADB = AutoDiffBlock; using Vector = ADB::V; using Communication = Dune::CollectiveCommunication; // copied from BlackoilModelBase // should put to somewhere better using DataBlock = Eigen::Array; // --------- Public methods --------- StandardWells(const Wells* wells_arg, WellCollection* well_collection); void init(const BlackoilPropsAdFromDeck* fluid_arg, const std::vector* active_arg, const std::vector* pc_arg, const VFPProperties* vfp_properties_arg, const double gravity_arg, const Vector& depth_arg); const WellOps& wellOps() const; int numPhases() const { return wells().number_of_phases; }; const Wells& wells() const; const Wells* wellsPointer() const; /// return true if wells are available in the reservoir bool wellsActive() const; void setWellsActive(const bool wells_active); /// return true if wells are available on this process bool localWellsActive() const; int numWellVars() const; /// Density of each well perforation Vector& wellPerforationDensities(); // mutable version kept for BlackoilMultisegmentModel const Vector& wellPerforationDensities() const; /// Diff to bhp for each well perforation. Vector& wellPerforationPressureDiffs(); // mutable version kept for BlackoilMultisegmentModel const Vector& wellPerforationPressureDiffs() const; template void extractWellPerfProperties(const SolutionState& state, const std::vector& rq, std::vector& mob_perfcells, std::vector& b_perfcells) const; template void computeWellFlux(const SolutionState& state, const std::vector& mob_perfcells, const std::vector& b_perfcells, Vector& aliveWells, std::vector& cq_s) const; template void updatePerfPhaseRatesAndPressures(const std::vector& cq_s, const SolutionState& state, WellState& xw) const; template void updateWellState(const Vector& dwells, const double dpmaxrel, WellState& well_state); template void updateWellControls(WellState& xw) const; // TODO: should LinearisedBlackoilResidual also be a template class? template void addWellFluxEq(const std::vector& cq_s, const SolutionState& state, LinearisedBlackoilResidual& residual); // TODO: some parameters, like gravity, maybe it is better to put in the member list template void addWellControlEq(const SolutionState& state, const WellState& xw, const Vector& aliveWells, LinearisedBlackoilResidual& residual); template void computeWellConnectionPressures(const SolutionState& state, const WellState& xw); // state0 is non-constant, while it will not be used outside of the function template void computeWellPotentials(const std::vector& mob_perfcells, const std::vector& b_perfcells, const WellState& well_state, SolutionState& state0, std::vector& well_potentials) const; template void variableStateExtractWellsVars(const std::vector& indices, std::vector& vars, SolutionState& state) const; void variableStateWellIndices(std::vector& indices, int& next) const; std::vector variableWellStateIndices() const; template void variableWellStateInitials(const WellState& xw, std::vector& vars0) const; /// If set, computeWellFlux() will additionally store the /// total reservoir volume perforation fluxes. void setStoreWellPerforationFluxesFlag(const bool store_fluxes); /// Retrieves the stored fluxes. It is an error to call this /// unless setStoreWellPerforationFluxesFlag(true) has been /// called. const Vector& getStoredWellPerforationFluxes() const; /// upate the dynamic lists related to economic limits template void updateListEconLimited(const Schedule& schedule, const int current_step, const Wells* wells, const WellState& well_state, DynamicListEconLimited& list_econ_limited) const; WellCollection* wellCollection() const; void calculateEfficiencyFactors(); const Vector& wellPerfEfficiencyFactors() const; protected: bool wells_active_; const Wells* wells_; const WellOps wops_; // It will probably need to be updated during running time. WellCollection* well_collection_; // The efficiency factor for each connection // It is specified based on wells and groups // By default, they should all be one. Vector well_perforation_efficiency_factors_; const BlackoilPropsAdFromDeck* fluid_; const std::vector* active_; const std::vector* phase_condition_; const VFPProperties* vfp_properties_; double gravity_; // the depth of the all the cell centers // for standard Wells, it the same with the perforation depth Vector perf_cell_depth_; Vector well_perforation_densities_; Vector well_perforation_pressure_diffs_; bool store_well_perforation_fluxes_; Vector well_perforation_fluxes_; // protected methods template void computePropertiesForWellConnectionPressures(const SolutionState& state, const WellState& xw, std::vector& b_perf, std::vector& rsmax_perf, std::vector& rvmax_perf, std::vector& surf_dens_perf); template void computeWellConnectionDensitesPressures(const WellState& xw, const std::vector& b_perf, const std::vector& rsmax_perf, const std::vector& rvmax_perf, const std::vector& surf_dens_perf, const std::vector& depth_perf, const double grav); template bool checkRateEconLimits(const WellEconProductionLimits& econ_production_limits, const WellState& well_state, const int well_number) const; using WellMapType = typename WellState::WellMapType; using WellMapEntryType = typename WellState::mapentry_t; // a tuple type for ratio limit check. // first value indicates whether ratio limit is violated, when the ratio limit is not violated, the following three // values should not be used. // second value indicates whehter there is only one connection left. // third value indicates the indx of the worst-offending connection. // the last value indicates the extent of the violation for the worst-offending connection, which is defined by // the ratio of the actual value to the value of the violated limit. using RatioCheckTuple = std::tuple; enum ConnectionIndex { INVALIDCONNECTION = -10000 }; template RatioCheckTuple checkRatioEconLimits(const WellEconProductionLimits& econ_production_limits, const WellState& well_state, const WellMapEntryType& map_entry) const; template RatioCheckTuple checkMaxWaterCutLimit(const WellEconProductionLimits& econ_production_limits, const WellState& well_state, const WellMapEntryType& map_entry) const; template void updateWellStateWithTarget(const WellControls* wc, const int current, const int well_index, WellState& xw) const; }; } // namespace Opm #include "StandardWells_impl.hpp" #endif