| 
									
										
										
										
											2014-03-18 11:23:05 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |   Copyright 2014 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/>.
 | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef OPM_WELLSTATEFULLYIMPLICITBLACKOIL_HEADER_INCLUDED
 | 
					
						
							|  |  |  | #define OPM_WELLSTATEFULLYIMPLICITBLACKOIL_HEADER_INCLUDED
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <opm/core/wells.h>
 | 
					
						
							|  |  |  | #include <opm/core/well_controls.h>
 | 
					
						
							|  |  |  | #include <opm/core/simulator/WellState.hpp>
 | 
					
						
							| 
									
										
										
										
											2014-08-13 15:55:50 +02:00
										 |  |  | #include <opm/core/utility/ErrorMacros.hpp>
 | 
					
						
							| 
									
										
										
										
											2014-03-18 11:23:05 +01:00
										 |  |  | #include <vector>
 | 
					
						
							|  |  |  | #include <cassert>
 | 
					
						
							| 
									
										
										
										
											2014-11-10 08:46:07 +01:00
										 |  |  | #include <string>
 | 
					
						
							|  |  |  | #include <utility>
 | 
					
						
							|  |  |  | #include <map>
 | 
					
						
							|  |  |  | #include <algorithm>
 | 
					
						
							| 
									
										
										
										
											2014-11-11 07:33:15 +01:00
										 |  |  | #include <array>
 | 
					
						
							| 
									
										
										
										
											2014-03-18 11:23:05 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Opm | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// The state of a set of wells, tailored for use by the fully
 | 
					
						
							|  |  |  |     /// implicit blackoil simulator.
 | 
					
						
							|  |  |  |     class WellStateFullyImplicitBlackoil | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     public: | 
					
						
							| 
									
										
										
										
											2014-11-11 07:33:15 +01:00
										 |  |  |         typedef std::array< int, 2 >  mapentry_t; | 
					
						
							| 
									
										
										
										
											2014-11-10 08:46:07 +01:00
										 |  |  |         typedef std::map< std::string, mapentry_t > WellMapType; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-11 07:33:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-18 11:23:05 +01:00
										 |  |  |         /// Allocate and initialize if wells is non-null.  Also tries
 | 
					
						
							|  |  |  |         /// to give useful initial values to the bhp(), wellRates()
 | 
					
						
							|  |  |  |         /// and perfPhaseRates() fields, depending on controls
 | 
					
						
							| 
									
										
										
										
											2014-11-10 08:46:07 +01:00
										 |  |  |         template <class State, class PrevState> | 
					
						
							|  |  |  |         void init(const Wells* wells, const State& state, const PrevState& prevState) | 
					
						
							| 
									
										
										
										
											2014-03-18 11:23:05 +01:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2014-11-10 08:46:07 +01:00
										 |  |  |             // clear old name mapping
 | 
					
						
							|  |  |  |             wellMap_.clear(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-18 11:23:05 +01:00
										 |  |  |             if (wells == 0) { | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // We use the WellState::init() function to do bhp and well rates init.
 | 
					
						
							|  |  |  |             // The alternative would be to copy that function wholesale.
 | 
					
						
							|  |  |  |             basic_well_state_.init(wells, state); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Initialize perfphaserates_, which must be done here.
 | 
					
						
							|  |  |  |             const int nw = wells->number_of_wells; | 
					
						
							|  |  |  |             const int np = wells->number_of_phases; | 
					
						
							|  |  |  |             const int nperf = wells->well_connpos[nw]; | 
					
						
							|  |  |  |             perfphaserates_.resize(nperf * np, 0.0); | 
					
						
							|  |  |  |             for (int w = 0; w < nw; ++w) { | 
					
						
							|  |  |  |                 assert((wells->type[w] == INJECTOR) || (wells->type[w] == PRODUCER)); | 
					
						
							|  |  |  |                 const WellControls* ctrl = wells->ctrls[w]; | 
					
						
							| 
									
										
										
										
											2014-11-10 08:46:07 +01:00
										 |  |  |                 std::string name( wells->name[ w ] ); | 
					
						
							|  |  |  |                 assert( name.size() > 0 ); | 
					
						
							|  |  |  |                 mapentry_t& wellMapEntry = wellMap_[ name ]; | 
					
						
							| 
									
										
										
										
											2014-11-11 07:33:15 +01:00
										 |  |  |                 wellMapEntry[ 0 ] = w ; | 
					
						
							|  |  |  |                 wellMapEntry[ 1 ] = wells->well_connpos[w ] ; | 
					
						
							| 
									
										
										
										
											2014-03-18 11:23:05 +01:00
										 |  |  |                 if (well_controls_well_is_shut(ctrl)) { | 
					
						
							|  |  |  |                     // Shut well: perfphaserates_ are all zero.
 | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     // Open well: Initialize perfphaserates_ to well
 | 
					
						
							|  |  |  |                     // rates divided by the number of perforations.
 | 
					
						
							|  |  |  |                     const int num_perf_this_well = wells->well_connpos[w + 1] - wells->well_connpos[w]; | 
					
						
							|  |  |  |                     for (int perf = wells->well_connpos[w]; perf < wells->well_connpos[w + 1]; ++perf) { | 
					
						
							|  |  |  |                         for (int p = 0; p < np; ++p) { | 
					
						
							|  |  |  |                             perfphaserates_[np*perf + p] = wellRates()[np*w + p] / double(num_perf_this_well); | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2014-03-25 11:11:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             // Initialize current_controls_.
 | 
					
						
							|  |  |  |             // The controls set in the Wells object are treated as defaults,
 | 
					
						
							|  |  |  |             // and also used for initial values.
 | 
					
						
							|  |  |  |             current_controls_.resize(nw); | 
					
						
							|  |  |  |             for (int w = 0; w < nw; ++w) { | 
					
						
							|  |  |  |                 current_controls_[w] = well_controls_get_current(wells->ctrls[w]); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2014-11-10 08:46:07 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             // intialize wells that have been there before
 | 
					
						
							|  |  |  |             // order may change so the mapping is based on the well name
 | 
					
						
							| 
									
										
										
										
											2014-11-11 07:33:15 +01:00
										 |  |  |             if( ! prevState.wellMap().empty() ) | 
					
						
							| 
									
										
										
										
											2014-11-10 08:46:07 +01:00
										 |  |  |             { | 
					
						
							| 
									
										
										
										
											2014-11-11 07:33:15 +01:00
										 |  |  |                 typedef typename WellMapType :: const_iterator const_iterator; | 
					
						
							|  |  |  |                 const_iterator end = prevState.wellMap().end(); | 
					
						
							| 
									
										
										
										
											2014-11-10 08:46:07 +01:00
										 |  |  |                 for (int w = 0; w < nw; ++w) { | 
					
						
							|  |  |  |                     std::string name( wells->name[ w ] ); | 
					
						
							| 
									
										
										
										
											2014-11-11 07:33:15 +01:00
										 |  |  |                     const_iterator it = prevState.wellMap().find( name ); | 
					
						
							| 
									
										
										
										
											2014-11-10 08:46:07 +01:00
										 |  |  |                     if( it != end ) | 
					
						
							|  |  |  |                     { | 
					
						
							| 
									
										
										
										
											2014-11-11 07:33:15 +01:00
										 |  |  |                         const int oldIndex = (*it).second[ 0 ]; | 
					
						
							| 
									
										
										
										
											2014-11-10 08:46:07 +01:00
										 |  |  |                         const int newIndex = w; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         // bhp
 | 
					
						
							|  |  |  |                         bhp()[ newIndex ] = prevState.bhp()[ oldIndex ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         // wellrates
 | 
					
						
							|  |  |  |                         for( int i=0, idx=newIndex*np, oldidx=oldIndex*np; i<np; ++i, ++idx, ++oldidx ) | 
					
						
							|  |  |  |                         { | 
					
						
							|  |  |  |                             wellRates()[ idx ] = prevState.wellRates()[ oldidx ]; | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         // perfPhaseRates
 | 
					
						
							| 
									
										
										
										
											2014-11-11 07:33:15 +01:00
										 |  |  |                         int oldPerf = (*it).second[ 1 ] * np; | 
					
						
							| 
									
										
										
										
											2014-11-10 08:46:07 +01:00
										 |  |  |                         for (int perf = wells->well_connpos[ newIndex ]*np; | 
					
						
							|  |  |  |                              perf < wells->well_connpos[ newIndex + 1]*np; ++perf, ++oldPerf ) | 
					
						
							|  |  |  |                         { | 
					
						
							|  |  |  |                             perfPhaseRates()[ perf ] = prevState.perfPhaseRates()[ oldPerf ]; | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         // currentControls
 | 
					
						
							|  |  |  |                         currentControls()[ newIndex ] = prevState.currentControls()[ oldIndex ]; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2014-03-18 11:23:05 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /// One bhp pressure per well.
 | 
					
						
							|  |  |  |         std::vector<double>& bhp() { return basic_well_state_.bhp(); } | 
					
						
							|  |  |  |         const std::vector<double>& bhp() const { return basic_well_state_.bhp(); } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /// One rate per well and phase.
 | 
					
						
							|  |  |  |         std::vector<double>& wellRates() { return basic_well_state_.wellRates(); } | 
					
						
							|  |  |  |         const std::vector<double>& wellRates() const { return basic_well_state_.wellRates(); } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /// One rate per phase and well connection.
 | 
					
						
							|  |  |  |         std::vector<double>& perfPhaseRates() { return perfphaserates_; } | 
					
						
							|  |  |  |         const std::vector<double>& perfPhaseRates() const { return perfphaserates_; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-25 11:11:19 +01:00
										 |  |  |         /// One current control per well.
 | 
					
						
							|  |  |  |         std::vector<int>& currentControls() { return current_controls_; } | 
					
						
							|  |  |  |         const std::vector<int>& currentControls() const { return current_controls_; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-18 11:23:05 +01:00
										 |  |  |         /// For interfacing with functions that take a WellState.
 | 
					
						
							|  |  |  |         const WellState& basicWellState() const | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             return basic_well_state_; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-13 15:55:50 +02:00
										 |  |  |         /// The number of wells present.
 | 
					
						
							|  |  |  |         int numWells() const | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             return bhp().size(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /// The number of phases present.
 | 
					
						
							|  |  |  |         int numPhases() const | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             return wellRates().size() / numWells(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-11 07:33:15 +01:00
										 |  |  |         const WellMapType& wellMap() const { return wellMap_; } | 
					
						
							| 
									
										
										
										
											2014-08-13 15:55:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-18 11:23:05 +01:00
										 |  |  |     private: | 
					
						
							|  |  |  |         WellState basic_well_state_; | 
					
						
							|  |  |  |         std::vector<double> perfphaserates_; | 
					
						
							| 
									
										
										
										
											2014-03-25 11:11:19 +01:00
										 |  |  |         std::vector<int> current_controls_; | 
					
						
							| 
									
										
										
										
											2014-11-11 07:33:15 +01:00
										 |  |  |         WellMapType wellMap_; | 
					
						
							| 
									
										
										
										
											2014-03-18 11:23:05 +01:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Opm
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif // OPM_WELLSTATEFULLYIMPLICITBLACKOIL_HEADER_INCLUDED
 |