2015-07-31 06:36:15 -05:00
|
|
|
/*
|
|
|
|
Copyright 2015 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_WELLMULTISEGMENT_HEADER_INCLUDED
|
|
|
|
#define OPM_WELLMULTISEGMENT_HEADER_INCLUDED
|
|
|
|
|
|
|
|
|
2015-10-07 04:19:53 -05:00
|
|
|
#include <opm/common/utility/platform_dependent/disable_warnings.h>
|
2015-08-28 08:57:27 -05:00
|
|
|
#include <Eigen/Eigen>
|
|
|
|
#include <Eigen/Sparse>
|
2015-10-07 04:19:53 -05:00
|
|
|
#include <opm/common/utility/platform_dependent/reenable_warnings.h>
|
2015-08-28 08:57:27 -05:00
|
|
|
|
2015-07-31 06:36:15 -05:00
|
|
|
#include <opm/core/wells.h>
|
|
|
|
#include <opm/core/well_controls.h>
|
|
|
|
#include <opm/core/simulator/WellState.hpp>
|
2015-10-12 07:06:50 -05:00
|
|
|
#include <opm/common/ErrorMacros.hpp>
|
2015-07-31 06:36:15 -05:00
|
|
|
#include <opm/parser/eclipse/EclipseState/Schedule/ScheduleEnums.hpp>
|
2015-11-12 06:15:27 -06:00
|
|
|
#include <opm/parser/eclipse/EclipseState/Schedule/MSW/SegmentSet.hpp>
|
2015-07-31 06:36:15 -05:00
|
|
|
#include <opm/parser/eclipse/EclipseState/Schedule/Well.hpp>
|
|
|
|
#include <vector>
|
|
|
|
#include <cassert>
|
|
|
|
#include <string>
|
|
|
|
#include <utility>
|
|
|
|
#include <map>
|
|
|
|
#include <algorithm>
|
|
|
|
#include <array>
|
|
|
|
|
|
|
|
namespace Opm
|
|
|
|
{
|
|
|
|
|
|
|
|
class WellMultiSegment
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
2015-12-01 03:44:05 -06:00
|
|
|
typedef Eigen::SparseMatrix<double> Matrix;
|
2015-08-28 08:57:27 -05:00
|
|
|
|
2015-11-23 07:17:39 -06:00
|
|
|
/// Constructor of WellMultiSegment
|
|
|
|
/// \param[in] well information from EclipseState
|
|
|
|
/// \param[in] current time step
|
|
|
|
/// \param[in[ pointer to Wells structure, to be removed eventually
|
2015-08-26 06:42:01 -05:00
|
|
|
WellMultiSegment(WellConstPtr well, size_t time_step, const Wells* wells);
|
2015-07-31 06:36:15 -05:00
|
|
|
|
2015-11-23 07:17:39 -06:00
|
|
|
/// Well name.
|
2015-08-28 08:57:27 -05:00
|
|
|
const std::string& name() const;
|
2015-11-23 07:17:39 -06:00
|
|
|
|
|
|
|
/// Flag indicating if the well is a multi-segment well.
|
2015-10-07 04:19:53 -05:00
|
|
|
bool isMultiSegmented() const;
|
2015-11-23 07:17:39 -06:00
|
|
|
|
|
|
|
/// Number of the perforations.
|
2015-10-07 04:19:53 -05:00
|
|
|
int numberOfPerforations() const;
|
2015-11-23 07:17:39 -06:00
|
|
|
|
|
|
|
/// Number of the segments.
|
2015-10-07 04:19:53 -05:00
|
|
|
int numberOfSegments() const;
|
2015-11-23 07:17:39 -06:00
|
|
|
|
|
|
|
/// Components of the pressure drop invloved.
|
|
|
|
/// HFA Hydrostatic + friction + acceleration
|
|
|
|
/// HF- Hydrostatic + friction
|
|
|
|
/// H-- Hydrostatic only.
|
2015-10-16 07:38:10 -05:00
|
|
|
std::string compPressureDrop() const;
|
2015-07-31 06:36:15 -05:00
|
|
|
|
2015-11-23 07:17:39 -06:00
|
|
|
/// Well control.
|
2015-10-07 04:19:53 -05:00
|
|
|
const WellControls* wellControls() const;
|
2015-11-23 07:17:39 -06:00
|
|
|
|
|
|
|
/// Component fractions for each well.
|
2015-08-26 06:42:01 -05:00
|
|
|
const std::vector<double>& compFrac() const;
|
|
|
|
|
2015-11-23 07:17:39 -06:00
|
|
|
/// Number of phases.
|
2015-10-07 04:19:53 -05:00
|
|
|
int numberOfPhases() const;
|
2015-08-26 06:42:01 -05:00
|
|
|
|
2015-11-23 07:17:39 -06:00
|
|
|
/// Well type.
|
2015-10-07 04:19:53 -05:00
|
|
|
WellType wellType() const;
|
2015-11-23 07:17:39 -06:00
|
|
|
|
|
|
|
/// Well productivity index.
|
2015-07-31 06:36:15 -05:00
|
|
|
const std::vector<double>& wellIndex() const;
|
2015-11-23 07:17:39 -06:00
|
|
|
|
|
|
|
/// Depth of the perforations.
|
2015-07-31 06:36:15 -05:00
|
|
|
const std::vector<double>& perfDepth() const;
|
2015-11-23 07:17:39 -06:00
|
|
|
|
|
|
|
/// Indices of the grid blocks that perforations are completed in.
|
2015-08-28 08:57:27 -05:00
|
|
|
const std::vector<int>& wellCells() const;
|
2015-11-23 07:17:39 -06:00
|
|
|
|
|
|
|
/// Indices of the gird blocks that segments locate at.
|
2015-10-12 04:33:30 -05:00
|
|
|
const std::vector<int>& segmentCells() const;
|
2015-11-23 07:17:39 -06:00
|
|
|
|
|
|
|
/// Outlet segments, a segment (except top segment) can only have one outlet segment.
|
|
|
|
/// For top segment, its outlet segments is -1 always, which means no outlet segment for top segment.
|
2015-07-31 06:36:15 -05:00
|
|
|
const std::vector<int>& outletSegment() const;
|
2015-11-23 07:17:39 -06:00
|
|
|
|
|
|
|
/// Inlet segments. a segment can have more than one inlet segments.
|
2015-08-28 08:57:27 -05:00
|
|
|
const std::vector<std::vector<int>>& inletSegments() const;
|
2015-11-23 07:17:39 -06:00
|
|
|
|
|
|
|
/// The length of the segment nodes down the wellbore from the reference point.
|
2015-07-31 06:36:15 -05:00
|
|
|
const std::vector<double>& segmentLength() const;
|
2015-11-23 07:17:39 -06:00
|
|
|
|
|
|
|
/// The depth of the segment nodes.
|
2015-07-31 06:36:15 -05:00
|
|
|
const std::vector<double>& segmentDepth() const;
|
2015-11-23 07:17:39 -06:00
|
|
|
|
|
|
|
/// Tubing internal diameter.
|
2015-10-16 10:07:43 -05:00
|
|
|
const std::vector<double>& segmentDiameter() const;
|
2015-11-23 07:17:39 -06:00
|
|
|
|
|
|
|
/// Cross-sectional area.
|
2015-07-31 06:36:15 -05:00
|
|
|
const std::vector<double>& segmentCrossArea() const;
|
2015-11-23 07:17:39 -06:00
|
|
|
|
|
|
|
/// Effective absolute roughness of the tube.
|
2015-07-31 06:36:15 -05:00
|
|
|
const std::vector<double>& segmentRoughness() const;
|
2015-11-23 07:17:39 -06:00
|
|
|
|
|
|
|
/// Volume of segment.
|
2015-07-31 06:36:15 -05:00
|
|
|
const std::vector<double>& segmentVolume() const;
|
2015-11-23 07:17:39 -06:00
|
|
|
|
|
|
|
/// Perforations related to each segment.
|
2015-08-28 08:57:27 -05:00
|
|
|
const std::vector<std::vector<int>>& segmentPerforations() const;
|
|
|
|
|
2015-11-23 07:17:39 -06:00
|
|
|
/// Struct for the well operator matrices.
|
|
|
|
/// All the operator matrics only apply to the one specifi well.
|
2015-08-28 08:57:27 -05:00
|
|
|
struct WellOps {
|
2015-12-01 03:44:05 -06:00
|
|
|
Matrix s2p; // segment -> perf (scatter)
|
|
|
|
Matrix p2s; // perf -> segment (gather)
|
|
|
|
Matrix p2s_average; // perf -> segment (avarage)
|
|
|
|
Matrix s2s_gather; // segment -> segment (in an accumlative way)
|
2015-08-28 08:57:27 -05:00
|
|
|
// means the outlet segments will gather all the contribution
|
|
|
|
// from all the inlet segments in a recurisive way
|
2015-12-01 03:44:05 -06:00
|
|
|
Matrix p2s_gather; // perforation -> segment (in an accumative way)
|
|
|
|
Matrix s2s_inlets; // segment -> its inlet segments
|
|
|
|
Matrix s2s_outlet; // segment -> its outlet segment
|
2015-08-28 08:57:27 -05:00
|
|
|
};
|
|
|
|
|
2015-11-23 07:17:39 -06:00
|
|
|
/// Well operator matrics
|
2015-08-28 08:57:27 -05:00
|
|
|
const WellOps& wellOps() const;
|
2015-07-31 06:36:15 -05:00
|
|
|
|
|
|
|
private:
|
|
|
|
// for the moment, we use the information from wells.
|
|
|
|
// TODO: remove the dependency on wells from opm-core.
|
2015-11-23 03:30:48 -06:00
|
|
|
void initMultiSegmentWell(WellConstPtr well, size_t time_step, const Wells* wells);
|
|
|
|
void initNonMultiSegmentWell(WellConstPtr well, size_t time_step, const Wells* wells);
|
|
|
|
void updateWellOps();
|
2015-07-31 06:36:15 -05:00
|
|
|
|
|
|
|
private:
|
|
|
|
// name of the well
|
2015-08-26 06:42:01 -05:00
|
|
|
std::string m_well_name_;
|
2015-07-31 06:36:15 -05:00
|
|
|
// flag to indicate if this well is a
|
|
|
|
// multi-segmented well
|
|
|
|
bool m_is_multi_segment_;
|
2015-08-26 06:42:01 -05:00
|
|
|
// well type
|
|
|
|
// INJECTOR or PRODUCER
|
|
|
|
enum WellType m_well_type_;
|
|
|
|
// number of phases
|
2015-10-06 09:53:22 -05:00
|
|
|
int m_number_of_phases_;
|
2015-08-26 06:42:01 -05:00
|
|
|
// component fractions for each well
|
|
|
|
std::vector<double> m_comp_frac_;
|
|
|
|
// controls for this well
|
|
|
|
// using pointer for temporary
|
|
|
|
// changing it when figuring out how to using it
|
|
|
|
struct WellControls *m_well_controls_;
|
2015-07-31 06:36:15 -05:00
|
|
|
// components of the pressure drop to be included
|
2015-11-12 06:15:27 -06:00
|
|
|
WellSegment::CompPressureDropEnum m_comp_pressure_drop_;
|
2015-07-31 06:36:15 -05:00
|
|
|
// multi-phase flow model
|
|
|
|
WellSegment::MultiPhaseModelEnum m_multiphase_model_;
|
|
|
|
// number of perforation for this well
|
2015-10-06 09:53:22 -05:00
|
|
|
int m_number_of_perforations_;
|
2015-07-31 06:36:15 -05:00
|
|
|
// number of segments for this well
|
2015-10-06 09:53:22 -05:00
|
|
|
int m_number_of_segments_;
|
2015-07-31 06:36:15 -05:00
|
|
|
// well index for each completion
|
|
|
|
std::vector<double> m_well_index_;
|
2015-08-26 06:42:01 -05:00
|
|
|
// depth for each completion // form the keyword COMPSEGS?
|
2015-07-31 06:36:15 -05:00
|
|
|
std::vector<double> m_perf_depth_;
|
|
|
|
// well cell for each completion
|
|
|
|
std::vector<int> m_well_cell_;
|
2015-10-12 04:33:30 -05:00
|
|
|
// cell for each segment
|
|
|
|
std::vector<int> m_segment_cell_;
|
2015-07-31 06:36:15 -05:00
|
|
|
// how to organize the segment structure here?
|
|
|
|
// indicate the outlet segment for each segment
|
|
|
|
// maybe here we can use the location in the vector
|
|
|
|
// at the moment, we still use the ID number
|
|
|
|
// then a mapping from the ID number to the actual location will be required
|
2015-08-28 08:57:27 -05:00
|
|
|
// The ID is already changed to the location now.
|
2015-07-31 06:36:15 -05:00
|
|
|
std::vector<int> m_outlet_segment_;
|
2015-08-28 08:57:27 -05:00
|
|
|
// for convinience, we store the inlet segments for each segment
|
|
|
|
std::vector<std::vector<int>> m_inlet_segments_;
|
|
|
|
// this one is not necessary any more, since the segment number is its location.
|
|
|
|
// std::map<int, int> m_number_to_location_;
|
2015-07-31 06:36:15 -05:00
|
|
|
// has not decided to use the absolute length from the well head
|
|
|
|
// or the length of this single segment
|
|
|
|
// using the absolute length first
|
|
|
|
std::vector<double> m_segment_length_;
|
|
|
|
// the depth of the segmnet node
|
|
|
|
std::vector<double> m_segment_depth_;
|
|
|
|
// the internal diameter of the segment
|
|
|
|
std::vector<double> m_segment_internal_diameter_;
|
|
|
|
// the roughness of the segment
|
|
|
|
std::vector<double> m_segment_roughness_;
|
|
|
|
// the cross area of the segment
|
|
|
|
std::vector<double> m_segment_cross_area_;
|
|
|
|
// the volume of the segment
|
|
|
|
std::vector<double> m_segment_volume_;
|
|
|
|
// the completions that is related to each segment
|
|
|
|
// the completions's ids are their location in the vector m_well_index_
|
|
|
|
// m_well_cell_
|
2015-08-26 06:42:01 -05:00
|
|
|
// This is also assuming the order of the completions in Well is the same with
|
|
|
|
// the order of the completions in wells.
|
2015-07-31 06:36:15 -05:00
|
|
|
std::vector<std::vector<int>> m_segment_perforations_;
|
2015-08-28 08:57:27 -05:00
|
|
|
|
|
|
|
WellOps m_wops_;
|
2015-07-31 06:36:15 -05:00
|
|
|
};
|
|
|
|
|
2015-09-22 08:54:21 -05:00
|
|
|
typedef std::shared_ptr<WellMultiSegment> WellMultiSegmentPtr;
|
|
|
|
typedef std::shared_ptr<const WellMultiSegment> WellMultiSegmentConstPtr;
|
2015-07-31 06:36:15 -05:00
|
|
|
|
|
|
|
} // namespace Opm
|
|
|
|
|
|
|
|
|
|
|
|
#endif // OPM_WELLMULTISEGMENT_HEADER_INCLUDE
|