// -*- mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- // vi: set et ts=4 sw=4 sts=4: /* 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 2 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 . Consult the COPYING file in the top-level source directory of this module for the precise wording of the license and the list of copyright holders. */ #include #include #include #include #include #include #include #include #include #include #include #include #if HAVE_DUNE_FEM #include #include #include #endif #include #include #include namespace Opm { template EclGenericTracerModel:: EclGenericTracerModel(const GridView& gridView, const EclipseState& eclState, const CartesianIndexMapper& cartMapper, const DofMapper& dofMapper) : gridView_(gridView) , eclState_(eclState) , cartMapper_(cartMapper) , dofMapper_(dofMapper) { } template const std::string& EclGenericTracerModel:: tracerName(int tracerIdx) const { if (tracerNames_.empty()) throw std::logic_error("This method should never be called when there are no tracers in the model"); return tracerNames_[tracerIdx]; } template Scalar EclGenericTracerModel:: tracerConcentration(int tracerIdx, int globalDofIdx) const { if (tracerConcentration_.empty()) return 0.0; return tracerConcentration_[tracerIdx][globalDofIdx]; } template void EclGenericTracerModel:: doInit(bool enabled, size_t numGridDof, size_t gasPhaseIdx, size_t oilPhaseIdx, size_t waterPhaseIdx) { const auto& tracers = eclState_.tracer(); const auto& comm = gridView_.comm(); if (tracers.size() == 0) return; // tracer treatment is supposed to be disabled if (!enabled) { if (gridView_.comm().rank() == 0) { OpmLog::warning("Keyword TRACERS has only experimental support, and is hence ignored.\n" "The experimental tracer model can still be used, but must be set explicitely.\n" "To use tracers, set the command line option: --enable-tracer-model=true" "\n"); } return; // Tracer transport must be enabled by the user } if (comm.size() > 1) { tracerNames_.resize(0); if (comm.rank() == 0) std::cout << "Warning: The tracer model currently does not work for parallel runs\n" << std::flush; return; } // retrieve the number of tracers from the deck const size_t numTracers = tracers.size(); tracerNames_.resize(numTracers); tracerConcentration_.resize(numTracers); storageOfTimeIndex1_.resize(numTracers); // the phase where the tracer is tracerPhaseIdx_.resize(numTracers); size_t tracerIdx = 0; for (const auto& tracer : tracers) { tracerNames_[tracerIdx] = tracer.name; if (tracer.phase == Phase::WATER) tracerPhaseIdx_[tracerIdx] = waterPhaseIdx; else if (tracer.phase == Phase::OIL) tracerPhaseIdx_[tracerIdx] = oilPhaseIdx; else if (tracer.phase == Phase::GAS) tracerPhaseIdx_[tracerIdx] = gasPhaseIdx; tracerConcentration_[tracerIdx].resize(numGridDof); storageOfTimeIndex1_[tracerIdx].resize(numGridDof); //TBLK keyword if (!tracer.concentration.empty()){ int tblkDatasize = tracer.concentration.size(); if (tblkDatasize < cartMapper_.cartesianSize()){ throw std::runtime_error("Wrong size of TBLK for" + tracer.name); } for (size_t globalDofIdx = 0; globalDofIdx < numGridDof; ++globalDofIdx){ int cartDofIdx = cartMapper_.cartesianIndex(globalDofIdx); tracerConcentration_[tracerIdx][globalDofIdx] = tracer.concentration[cartDofIdx]; } } //TVDPF keyword else { const auto& eclGrid = eclState_.getInputGrid(); for (size_t globalDofIdx = 0; globalDofIdx < numGridDof; ++globalDofIdx){ int cartDofIdx = cartMapper_.cartesianIndex(globalDofIdx); const auto& center = eclGrid.getCellCenter(cartDofIdx); tracerConcentration_[tracerIdx][globalDofIdx] = tracer.tvdpf.evaluate("TRACER_CONCENTRATION", center[2]); } } ++tracerIdx; } // initial tracer concentration tracerConcentrationInitial_ = tracerConcentration_; // residual of tracers tracerResidual_.resize(numGridDof); // allocate matrix for storing the Jacobian of the tracer residual tracerMatrix_ = new TracerMatrix(numGridDof, numGridDof, TracerMatrix::random); // find the sparsity pattern of the tracer matrix using NeighborSet = std::set; std::vector neighbors(numGridDof); Stencil stencil(gridView_, dofMapper_); auto elemIt = gridView_.template begin<0>(); const auto elemEndIt = gridView_.template end<0>(); for (; elemIt != elemEndIt; ++elemIt) { const auto& elem = *elemIt; stencil.update(elem); for (unsigned primaryDofIdx = 0; primaryDofIdx < stencil.numPrimaryDof(); ++primaryDofIdx) { unsigned myIdx = stencil.globalSpaceIndex(primaryDofIdx); for (unsigned dofIdx = 0; dofIdx < stencil.numDof(); ++dofIdx) { unsigned neighborIdx = stencil.globalSpaceIndex(dofIdx); neighbors[myIdx].insert(neighborIdx); } } } // allocate space for the rows of the matrix for (unsigned dofIdx = 0; dofIdx < numGridDof; ++ dofIdx) tracerMatrix_->setrowsize(dofIdx, neighbors[dofIdx].size()); tracerMatrix_->endrowsizes(); // fill the rows with indices. each degree of freedom talks to // all of its neighbors. (it also talks to itself since // degrees of freedom are sometimes quite egocentric.) for (unsigned dofIdx = 0; dofIdx < numGridDof; ++ dofIdx) { typename NeighborSet::iterator nIt = neighbors[dofIdx].begin(); typename NeighborSet::iterator nEndIt = neighbors[dofIdx].end(); for (; nIt != nEndIt; ++nIt) tracerMatrix_->addindex(dofIdx, *nIt); } tracerMatrix_->endindices(); const int sizeCartGrid = cartMapper_.cartesianSize(); cartToGlobal_.resize(sizeCartGrid); for (unsigned i = 0; i < numGridDof; ++i) { int cartIdx = cartMapper_.cartesianIndex(i); cartToGlobal_[cartIdx] = i; } } template bool EclGenericTracerModel:: linearSolve_(const TracerMatrix& M, TracerVector& x, TracerVector& b) { #if ! DUNE_VERSION_NEWER(DUNE_COMMON, 2,7) Dune::FMatrixPrecision::set_singular_limit(1.e-30); Dune::FMatrixPrecision::set_absolute_limit(1.e-30); #endif x = 0.0; Scalar tolerance = 1e-2; int maxIter = 100; int verbosity = 0; using TracerSolver = Dune::BiCGSTABSolver; using TracerOperator = Dune::MatrixAdapter; using TracerScalarProduct = Dune::SeqScalarProduct; using TracerPreconditioner = Dune::SeqILU< TracerMatrix,TracerVector,TracerVector>; TracerOperator tracerOperator(M); TracerScalarProduct tracerScalarProduct; TracerPreconditioner tracerPreconditioner(M, 0, 1); // results in ILU0 TracerSolver solver (tracerOperator, tracerScalarProduct, tracerPreconditioner, tolerance, maxIter, verbosity); Dune::InverseOperatorResult result; solver.apply(x, b, result); // return the result of the solver return result.converged; } #if HAVE_DUNE_FEM template class EclGenericTracerModel>>, Dune::MultipleCodimMultipleGeomTypeMapper>>>, Opm::EcfvStencil>>,false,false>, double>; #else template class EclGenericTracerModel>, Dune::MultipleCodimMultipleGeomTypeMapper>,Dune::Impl::MCMGFailLayout>, Opm::EcfvStencil>,false,false>, double>; #endif template class EclGenericTracerModel, Dune::GridView>,Dune::MultipleCodimMultipleGeomTypeMapper>,Dune::Impl::MCMGFailLayout>, Opm::EcfvStencil>,false,false>, double>; } // namespace Opm