2017-05-15 05:34:26 -05:00
|
|
|
/*
|
|
|
|
Copyright 2017 Dr. Blatt - HPC-Simulation-Software & Services
|
|
|
|
Copyright 2017 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 <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef OPM_WELLCONNECTIONAUXILIARYMODULE_HEADER_INCLUDED
|
|
|
|
#define OPM_WELLCONNECTIONAUXILIARYMODULE_HEADER_INCLUDED
|
|
|
|
|
2019-09-11 06:46:48 -05:00
|
|
|
#include <opm/models/discretization/common/baseauxiliarymodule.hh>
|
2017-05-15 05:34:26 -05:00
|
|
|
|
2018-02-14 15:22:08 -06:00
|
|
|
#include <opm/grid/CpGrid.hpp>
|
2017-05-16 05:28:54 -05:00
|
|
|
|
|
|
|
#include <opm/parser/eclipse/EclipseState/EclipseState.hpp>
|
|
|
|
|
2017-05-15 05:34:26 -05:00
|
|
|
namespace Opm
|
|
|
|
{
|
|
|
|
template<class TypeTag>
|
|
|
|
class WellConnectionAuxiliaryModule
|
2019-09-05 10:04:39 -05:00
|
|
|
: public Opm::BaseAuxiliaryModule<TypeTag>
|
2017-05-15 05:34:26 -05:00
|
|
|
{
|
2020-08-26 03:49:52 -05:00
|
|
|
using GlobalEqVector = GetPropType<TypeTag, Properties::GlobalEqVector>;
|
|
|
|
using SparseMatrixAdapter = GetPropType<TypeTag, Properties::SparseMatrixAdapter>;
|
2017-05-15 05:34:26 -05:00
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
using NeighborSet = typename
|
2019-09-05 10:04:39 -05:00
|
|
|
Opm::BaseAuxiliaryModule<TypeTag>::NeighborSet;
|
2017-05-15 05:34:26 -05:00
|
|
|
|
2017-05-16 05:28:54 -05:00
|
|
|
WellConnectionAuxiliaryModule(const Schedule& schedule,
|
|
|
|
const Dune::CpGrid& grid)
|
2017-05-15 05:34:26 -05:00
|
|
|
{
|
2017-05-16 05:28:54 -05:00
|
|
|
// Create cartesian to compressed mapping
|
|
|
|
const auto& globalCell = grid.globalCell();
|
|
|
|
const auto& cartesianSize = grid.logicalCartesianSize();
|
|
|
|
|
|
|
|
auto size = cartesianSize[0]*cartesianSize[1]*cartesianSize[2];
|
|
|
|
|
|
|
|
std::vector<int> cartesianToCompressed(size, -1);
|
|
|
|
auto begin = globalCell.begin();
|
|
|
|
|
2018-02-26 02:09:12 -06:00
|
|
|
for ( auto cell = begin, end= globalCell.end(); cell != end; ++cell )
|
2017-05-16 05:28:54 -05:00
|
|
|
{
|
2019-05-02 05:51:25 -05:00
|
|
|
cartesianToCompressed[ *cell ] = cell - begin;
|
2017-05-16 05:28:54 -05:00
|
|
|
}
|
|
|
|
|
2019-11-13 16:16:11 -06:00
|
|
|
const auto& schedule_wells = schedule.getWellsatEnd();
|
2017-05-16 05:28:54 -05:00
|
|
|
wells_.reserve(schedule_wells.size());
|
|
|
|
|
|
|
|
// initialize the additional cell connections introduced by wells.
|
2020-09-01 06:25:26 -05:00
|
|
|
for ( const auto& well : schedule_wells )
|
2018-02-26 02:09:12 -06:00
|
|
|
{
|
2017-05-16 05:28:54 -05:00
|
|
|
std::vector<int> compressed_well_perforations;
|
|
|
|
// All possible completions of the well
|
2019-05-02 05:51:25 -05:00
|
|
|
const auto& completionSet = well.getConnections();
|
2017-05-16 05:28:54 -05:00
|
|
|
compressed_well_perforations.reserve(completionSet.size());
|
|
|
|
|
2018-02-26 02:09:12 -06:00
|
|
|
for ( size_t c=0; c < completionSet.size(); c++ )
|
|
|
|
{
|
2017-05-16 05:28:54 -05:00
|
|
|
const auto& completion = completionSet.get(c);
|
|
|
|
int i = completion.getI();
|
|
|
|
int j = completion.getJ();
|
|
|
|
int k = completion.getK();
|
|
|
|
int cart_grid_idx = i + cartesianSize[0]*(j + cartesianSize[1]*k);
|
|
|
|
int compressed_idx = cartesianToCompressed[cart_grid_idx];
|
|
|
|
|
|
|
|
if ( compressed_idx >= 0 ) // Ignore completions in inactive/remote cells.
|
|
|
|
{
|
|
|
|
compressed_well_perforations.push_back(compressed_idx);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-26 02:09:12 -06:00
|
|
|
if ( ! compressed_well_perforations.empty() )
|
2017-05-16 05:28:54 -05:00
|
|
|
{
|
|
|
|
std::sort(compressed_well_perforations.begin(),
|
|
|
|
compressed_well_perforations.end());
|
|
|
|
|
|
|
|
wells_.push_back(compressed_well_perforations);
|
|
|
|
}
|
|
|
|
}
|
2017-05-15 05:34:26 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned numDofs() const
|
|
|
|
{
|
|
|
|
// No extra dofs are inserted for wells.
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void addNeighbors(std::vector<NeighborSet>& neighbors) const
|
|
|
|
{
|
2020-09-01 06:25:26 -05:00
|
|
|
for(const auto& well_perforations : wells_)
|
2017-05-15 05:34:26 -05:00
|
|
|
{
|
2017-05-16 05:28:54 -05:00
|
|
|
for(const auto& perforation : well_perforations)
|
|
|
|
neighbors[perforation].insert(well_perforations.begin(),
|
|
|
|
well_perforations.end());
|
2017-05-15 05:34:26 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void applyInitial()
|
|
|
|
{}
|
|
|
|
|
2018-11-12 05:40:11 -06:00
|
|
|
void linearize(SparseMatrixAdapter& , GlobalEqVector&)
|
2017-05-15 05:34:26 -05:00
|
|
|
{
|
|
|
|
// Linearization is done in StandardDenseWells
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
2017-05-16 05:28:54 -05:00
|
|
|
std::vector<std::vector<int> > wells_;
|
2017-05-15 05:34:26 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
} // end namespace OPM
|
|
|
|
#endif
|