2016-12-16 07:17:56 -06:00
|
|
|
/////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// Copyright (C) 2016- Statoil ASA
|
|
|
|
//
|
|
|
|
// ResInsight 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.
|
|
|
|
//
|
|
|
|
// ResInsight 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 at <http://www.gnu.org/licenses/gpl.html>
|
|
|
|
// for more details.
|
|
|
|
//
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
#include "RigFlowDiagResults.h"
|
2017-01-09 12:14:07 -06:00
|
|
|
|
|
|
|
#include "RigActiveCellInfo.h"
|
2017-01-10 02:51:39 -06:00
|
|
|
#include "RigEclipseCaseData.h"
|
2017-01-09 12:14:07 -06:00
|
|
|
#include "RigFlowDiagSolverInterface.h"
|
2016-12-20 04:41:05 -06:00
|
|
|
#include "RigFlowDiagStatCalc.h"
|
2016-12-16 07:17:56 -06:00
|
|
|
|
2017-01-09 12:14:07 -06:00
|
|
|
#include "RimEclipseCase.h"
|
|
|
|
#include "RimEclipseResultCase.h"
|
|
|
|
#include "RimFlowDiagSolution.h"
|
2017-02-17 04:13:27 -06:00
|
|
|
#include "RigFlowDiagResultFrames.h"
|
|
|
|
#include "RigStatisticsDataCache.h"
|
2017-01-09 12:14:07 -06:00
|
|
|
|
2017-03-28 02:47:06 -05:00
|
|
|
#include <cmath> // Needed for HUGE_VAL on Linux
|
|
|
|
|
2016-12-16 07:17:56 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2016-12-17 03:46:57 -06:00
|
|
|
RigFlowDiagResults::RigFlowDiagResults(RimFlowDiagSolution* flowSolution, size_t timeStepCount)
|
2016-12-20 04:41:05 -06:00
|
|
|
: m_flowDiagSolution(flowSolution)
|
2016-12-16 07:17:56 -06:00
|
|
|
{
|
|
|
|
|
2016-12-17 03:46:57 -06:00
|
|
|
m_timeStepCount = timeStepCount;
|
2017-08-10 08:08:30 -05:00
|
|
|
m_hasAtemptedNativeResults.resize(timeStepCount);
|
2017-03-14 13:21:55 -05:00
|
|
|
m_injProdPairFluxCommunicationTimesteps.resize(timeStepCount);
|
2016-12-16 07:17:56 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
RigFlowDiagResults::~RigFlowDiagResults()
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2016-12-17 03:46:57 -06:00
|
|
|
|
2016-12-16 07:17:56 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
const std::vector<double>* RigFlowDiagResults::resultValues(const RigFlowDiagResultAddress& resVarAddr, size_t frameIndex)
|
|
|
|
{
|
2016-12-17 03:46:57 -06:00
|
|
|
CVF_ASSERT(m_timeStepCount != cvf::UNDEFINED_SIZE_T); // Forgotten to call init
|
|
|
|
|
|
|
|
return findOrCalculateResult(resVarAddr, frameIndex);
|
|
|
|
}
|
|
|
|
|
2016-12-20 04:41:05 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
const RigActiveCellInfo * RigFlowDiagResults::activeCellInfo(const RigFlowDiagResultAddress& resVarAddr)
|
|
|
|
{
|
|
|
|
RimEclipseResultCase* eclCase;
|
|
|
|
m_flowDiagSolution->firstAncestorOrThisOfType(eclCase);
|
|
|
|
|
2017-08-11 07:05:59 -05:00
|
|
|
return eclCase->eclipseCaseData()->activeCellInfo(RiaDefines::MATRIX_MODEL); // Todo: base on resVarAddr member
|
2016-12-20 04:41:05 -06:00
|
|
|
}
|
|
|
|
|
2016-12-17 03:46:57 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
const std::vector<double>* RigFlowDiagResults::findOrCalculateResult(const RigFlowDiagResultAddress& resVarAddr, size_t frameIndex)
|
|
|
|
{
|
|
|
|
|
|
|
|
std::vector<double>* frameData = findScalarResultFrame(resVarAddr, frameIndex);
|
|
|
|
|
|
|
|
if ( frameData ) return frameData;
|
|
|
|
|
|
|
|
frameData = calculateDerivedResult(resVarAddr, frameIndex);
|
|
|
|
|
|
|
|
if ( frameData ) return frameData;
|
|
|
|
|
|
|
|
// We need to access the native data from the opm solver
|
|
|
|
|
2017-01-12 07:16:39 -06:00
|
|
|
if (!solverInterface()) return nullptr;
|
|
|
|
|
2017-08-10 08:08:30 -05:00
|
|
|
calculateNativeResultsIfNotPreviouslyAttempted(frameIndex, resVarAddr.phaseSelection);
|
2017-03-14 13:21:55 -05:00
|
|
|
|
|
|
|
return findScalarResultFrame(resVarAddr, frameIndex);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2017-08-10 08:08:30 -05:00
|
|
|
void RigFlowDiagResults::calculateNativeResultsIfNotPreviouslyAttempted(size_t frameIndex, RigFlowDiagResultAddress::PhaseSelection phaseSelection)
|
2017-03-14 13:21:55 -05:00
|
|
|
{
|
2017-08-10 08:08:30 -05:00
|
|
|
auto it = m_hasAtemptedNativeResults[frameIndex].find(phaseSelection);
|
|
|
|
if ( it == m_hasAtemptedNativeResults[frameIndex].end() || !it->second )
|
2016-12-17 03:46:57 -06:00
|
|
|
{
|
2017-03-14 13:21:55 -05:00
|
|
|
|
|
|
|
RigFlowDiagTimeStepResult nativeTimestepResults = solverInterface()->calculate(frameIndex,
|
2017-08-10 08:08:30 -05:00
|
|
|
phaseSelection,
|
2016-12-20 04:41:05 -06:00
|
|
|
m_flowDiagSolution->allInjectorTracerActiveCellIndices(frameIndex),
|
|
|
|
m_flowDiagSolution->allProducerTracerActiveCellIndices(frameIndex));
|
2016-12-17 03:46:57 -06:00
|
|
|
|
|
|
|
std::map<RigFlowDiagResultAddress, std::vector<double> >& nativeResults = nativeTimestepResults.nativeResults();
|
|
|
|
|
|
|
|
for ( auto& resIt: nativeResults )
|
|
|
|
{
|
|
|
|
RigFlowDiagResultFrames* nativeResFrames = findScalarResult(resIt.first);
|
|
|
|
if ( !nativeResFrames ) nativeResFrames = createScalarResult(resIt.first);
|
|
|
|
|
|
|
|
nativeResFrames->frameData(frameIndex).swap(resIt.second);
|
|
|
|
}
|
|
|
|
|
2017-08-10 08:08:30 -05:00
|
|
|
m_injProdPairFluxCommunicationTimesteps[frameIndex][phaseSelection].swap(nativeTimestepResults.injProdWellPairFluxes());
|
2017-03-14 13:21:55 -05:00
|
|
|
|
2017-08-10 08:08:30 -05:00
|
|
|
m_hasAtemptedNativeResults[frameIndex][phaseSelection] = true;
|
2016-12-17 03:46:57 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
std::vector<double>* RigFlowDiagResults::findScalarResultFrame(const RigFlowDiagResultAddress& resVarAddr, size_t frameIndex)
|
|
|
|
{
|
|
|
|
RigFlowDiagResultFrames* resFrames = findScalarResult (resVarAddr);
|
|
|
|
|
|
|
|
if ( resFrames )
|
|
|
|
{
|
|
|
|
std::vector<double>& frame = resFrames->frameData(frameIndex);
|
|
|
|
if ( frame.size() ) return(&frame);
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2016-12-20 04:41:05 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
RigFlowDiagSolverInterface* RigFlowDiagResults::solverInterface()
|
|
|
|
{
|
|
|
|
RimEclipseResultCase* eclCase;
|
|
|
|
m_flowDiagSolution->firstAncestorOrThisOfType(eclCase);
|
|
|
|
|
|
|
|
return eclCase->flowDiagSolverInterface();
|
|
|
|
}
|
|
|
|
|
2016-12-17 03:46:57 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
RigFlowDiagResultFrames* RigFlowDiagResults::createScalarResult(const RigFlowDiagResultAddress& resVarAddr)
|
|
|
|
{
|
|
|
|
cvf::ref<RigFlowDiagResultFrames> newFrameSet = new RigFlowDiagResultFrames(m_timeStepCount);
|
|
|
|
m_resultSets[resVarAddr] = newFrameSet;
|
|
|
|
|
|
|
|
return newFrameSet.p();
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
RigFlowDiagResultFrames* RigFlowDiagResults::findScalarResult(const RigFlowDiagResultAddress& resVarAddr)
|
|
|
|
{
|
|
|
|
decltype(m_resultSets)::iterator it = m_resultSets.find(resVarAddr);
|
|
|
|
|
|
|
|
if ( it == m_resultSets.end() ) return nullptr;
|
|
|
|
|
|
|
|
return it->second.p();
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
std::vector<double>* RigFlowDiagResults::calculateDerivedResult(const RigFlowDiagResultAddress& resVarAddr, size_t frameIndex)
|
|
|
|
{
|
2016-12-21 10:14:13 -06:00
|
|
|
if (resVarAddr.isNativeResult()) return nullptr;
|
|
|
|
|
|
|
|
if (resVarAddr.variableName == RIG_FLD_TOF_RESNAME)
|
|
|
|
{
|
2017-01-06 07:41:00 -06:00
|
|
|
return calculateAverageTOFResult(resVarAddr, frameIndex);
|
|
|
|
}
|
|
|
|
else if (resVarAddr.variableName == RIG_FLD_CELL_FRACTION_RESNAME)
|
|
|
|
{
|
|
|
|
return calculateSumOfFractionsResult(resVarAddr, frameIndex);
|
|
|
|
}
|
|
|
|
else if ( resVarAddr.variableName == RIG_FLD_COMMUNICATION_RESNAME )
|
|
|
|
{
|
|
|
|
return calculateCommunicationResult(resVarAddr, frameIndex);
|
|
|
|
}
|
|
|
|
else if ( resVarAddr.variableName == RIG_FLD_MAX_FRACTION_TRACER_RESNAME )
|
|
|
|
{
|
|
|
|
return calculateTracerWithMaxFractionResult(resVarAddr, frameIndex);
|
|
|
|
}
|
2016-12-21 10:14:13 -06:00
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
std::vector<double>* RigFlowDiagResults::calculateAverageTOFResult(const RigFlowDiagResultAddress& resVarAddr, size_t frameIndex)
|
|
|
|
{
|
|
|
|
std::vector<const std::vector<double>* > injectorTOFs = findResultsForSelectedTracers(resVarAddr, frameIndex,
|
|
|
|
RIG_FLD_TOF_RESNAME, RimFlowDiagSolution::INJECTOR);
|
|
|
|
std::vector<const std::vector<double>* > injectorFractions = findResultsForSelectedTracers(resVarAddr, frameIndex,
|
|
|
|
RIG_FLD_CELL_FRACTION_RESNAME, RimFlowDiagSolution::INJECTOR);
|
|
|
|
|
|
|
|
std::vector<const std::vector<double>* > producerTOFs = findResultsForSelectedTracers(resVarAddr, frameIndex,
|
|
|
|
RIG_FLD_TOF_RESNAME, RimFlowDiagSolution::PRODUCER);
|
|
|
|
std::vector<const std::vector<double>* > producerFractions = findResultsForSelectedTracers(resVarAddr, frameIndex,
|
|
|
|
RIG_FLD_CELL_FRACTION_RESNAME, RimFlowDiagSolution::PRODUCER);
|
|
|
|
size_t activeCellCount = this->activeCellInfo(resVarAddr)->reservoirActiveCellCount();
|
2016-12-21 10:14:13 -06:00
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
std::vector<double> injectorTotalFractions;
|
|
|
|
std::vector<double> injectorFractMultTof;
|
|
|
|
calculateSumOfFractionAndFractionMultTOF(activeCellCount, injectorFractions, injectorTOFs, &injectorTotalFractions, &injectorFractMultTof);
|
2016-12-21 10:14:13 -06:00
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
std::vector<double> producerTotalFractions;
|
|
|
|
std::vector<double> producerFractMultTof;
|
|
|
|
calculateSumOfFractionAndFractionMultTOF(activeCellCount, producerFractions, producerTOFs, &producerTotalFractions, &producerFractMultTof);
|
2016-12-21 10:14:13 -06:00
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
RigFlowDiagResultFrames* averageTofFrames = this->createScalarResult(resVarAddr);
|
|
|
|
std::vector<double>& averageTof = averageTofFrames->frameData(frameIndex);
|
|
|
|
averageTof.resize(activeCellCount, HUGE_VAL);
|
2016-12-21 10:14:13 -06:00
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
for ( size_t acIdx = 0 ; acIdx < activeCellCount; ++acIdx )
|
|
|
|
{
|
|
|
|
if ( injectorTotalFractions[acIdx] == 0.0 && producerTotalFractions[acIdx] == 0.0 )
|
2016-12-21 10:14:13 -06:00
|
|
|
{
|
2017-01-06 07:41:00 -06:00
|
|
|
averageTof[acIdx] = HUGE_VAL;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
double retVal = 0.0;
|
|
|
|
if ( injectorTotalFractions[acIdx] != 0.0 ) retVal += (1.0/injectorTotalFractions[acIdx]) * injectorFractMultTof[acIdx];
|
|
|
|
if ( producerTotalFractions[acIdx] != 0.0 ) retVal += (1.0/producerTotalFractions[acIdx]) * producerFractMultTof[acIdx];
|
|
|
|
averageTof[acIdx] = retVal;
|
2016-12-21 10:14:13 -06:00
|
|
|
}
|
2017-01-06 07:41:00 -06:00
|
|
|
}
|
2016-12-21 10:14:13 -06:00
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
/// Test to remove all averaging
|
|
|
|
// if (injectorTOFs.size()) averageTof = (*injectorTOFs[0]);
|
|
|
|
|
|
|
|
return &averageTof;
|
|
|
|
}
|
2017-01-03 06:14:07 -06:00
|
|
|
|
2017-01-02 06:34:51 -06:00
|
|
|
|
2017-05-05 04:50:08 -05:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void RigFlowDiagResults::calculateSumOfFractionAndFractionMultTOF(size_t activeCellCount,
|
|
|
|
const std::vector<const std::vector<double> *> & fractions,
|
|
|
|
const std::vector<const std::vector<double> *> & TOFs,
|
|
|
|
std::vector<double> *sumOfFractions,
|
|
|
|
std::vector<double> *fractionMultTOF)
|
|
|
|
{
|
|
|
|
sumOfFractions->resize(activeCellCount, 0.0);
|
|
|
|
fractionMultTOF->resize(activeCellCount, 0.0);
|
|
|
|
|
|
|
|
for ( size_t iIdx = 0; iIdx < fractions.size() ; ++iIdx )
|
|
|
|
{
|
|
|
|
const std::vector<double> * frInj = fractions[iIdx];
|
|
|
|
const std::vector<double> * tofInj = TOFs[iIdx];
|
|
|
|
|
|
|
|
if ( ! (frInj && tofInj) ) continue;
|
|
|
|
|
|
|
|
for ( size_t acIdx = 0 ; acIdx < activeCellCount; ++acIdx )
|
|
|
|
{
|
|
|
|
if ( (*frInj)[acIdx] == HUGE_VAL ) continue;
|
|
|
|
|
|
|
|
(*sumOfFractions)[acIdx] += (*frInj)[acIdx];
|
|
|
|
(*fractionMultTOF)[acIdx] += (*frInj)[acIdx] * (*tofInj)[acIdx];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
std::vector<double>* RigFlowDiagResults::calculateSumOfFractionsResult(const RigFlowDiagResultAddress& resVarAddr, size_t frameIndex)
|
|
|
|
{
|
|
|
|
std::vector<const std::vector<double>* > fractions = findResultsForSelectedTracers(resVarAddr,
|
|
|
|
frameIndex,
|
|
|
|
RIG_FLD_CELL_FRACTION_RESNAME,
|
|
|
|
RimFlowDiagSolution::UNDEFINED);
|
2017-01-02 06:34:51 -06:00
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
RigFlowDiagResultFrames* sumOfFractionsFrames = this->createScalarResult(resVarAddr);
|
|
|
|
std::vector<double>& sumOfFractions = sumOfFractionsFrames->frameData(frameIndex);
|
2017-01-02 06:34:51 -06:00
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
size_t activeCellCount = this->activeCellInfo(resVarAddr)->reservoirActiveCellCount();
|
2017-01-02 07:46:12 -06:00
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
calculateSumOfFractions(fractions, activeCellCount, &sumOfFractions);
|
2017-01-02 06:34:51 -06:00
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
return &sumOfFractions;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
std::vector<double>* RigFlowDiagResults::calculateTracerWithMaxFractionResult(const RigFlowDiagResultAddress &resVarAddr, size_t frameIndex)
|
|
|
|
{
|
2017-03-21 09:56:36 -05:00
|
|
|
std::vector< std::pair<std::string, const std::vector<double>* > > fractions = findNamedResultsForSelectedTracers(resVarAddr,
|
|
|
|
frameIndex,
|
|
|
|
RIG_FLD_CELL_FRACTION_RESNAME,
|
|
|
|
RimFlowDiagSolution::UNDEFINED);
|
|
|
|
|
|
|
|
std::vector<int> resultTracerIdxToGlobalTracerIdx;
|
2017-01-03 02:27:25 -06:00
|
|
|
{
|
2017-03-21 09:56:36 -05:00
|
|
|
resultTracerIdxToGlobalTracerIdx.resize(fractions.size(), -1);
|
2017-01-03 02:27:25 -06:00
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
std::vector<QString> allTracerNames = m_flowDiagSolution->tracerNames();
|
|
|
|
int selTracerIdx = 0;
|
2017-03-21 09:56:36 -05:00
|
|
|
for ( const auto& trNameFractionPair: fractions )
|
2017-01-06 07:41:00 -06:00
|
|
|
{
|
2017-01-10 01:48:58 -06:00
|
|
|
for ( size_t globIdx = 0; globIdx < allTracerNames.size(); ++globIdx )
|
2017-01-03 02:27:25 -06:00
|
|
|
{
|
2017-03-21 09:56:36 -05:00
|
|
|
if ( allTracerNames[globIdx].toStdString() == trNameFractionPair.first )
|
2017-01-03 02:27:25 -06:00
|
|
|
{
|
2017-03-21 09:56:36 -05:00
|
|
|
resultTracerIdxToGlobalTracerIdx[selTracerIdx] = static_cast<int>(globIdx);
|
2017-01-06 07:41:00 -06:00
|
|
|
break;
|
2017-01-03 02:27:25 -06:00
|
|
|
}
|
|
|
|
}
|
2017-01-06 07:41:00 -06:00
|
|
|
|
|
|
|
++selTracerIdx;
|
2017-01-03 02:27:25 -06:00
|
|
|
}
|
2017-01-06 07:41:00 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
size_t activeCellCount = this->activeCellInfo(resVarAddr)->reservoirActiveCellCount();
|
2017-01-03 02:27:25 -06:00
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
RigFlowDiagResultFrames* maxFractionTracerIdxFrames = this->createScalarResult(resVarAddr);
|
|
|
|
std::vector<double>& maxFractionTracerIdx = maxFractionTracerIdxFrames->frameData(frameIndex);
|
|
|
|
{
|
2017-01-03 02:27:25 -06:00
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
maxFractionTracerIdx.resize(activeCellCount, HUGE_VAL);
|
2017-01-03 02:27:25 -06:00
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
std::vector<double> maxFraction;
|
|
|
|
maxFraction.resize(activeCellCount, -HUGE_VAL);
|
2017-01-03 02:27:25 -06:00
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
for ( size_t frIdx = 0; frIdx < fractions.size(); ++frIdx )
|
|
|
|
{
|
2017-03-21 09:56:36 -05:00
|
|
|
const std::vector<double> * fr = fractions[frIdx].second;
|
2017-01-03 02:27:25 -06:00
|
|
|
|
2017-03-22 08:04:13 -05:00
|
|
|
if (!fr) continue;
|
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
for ( size_t acIdx = 0 ; acIdx < activeCellCount; ++acIdx )
|
2017-01-03 02:27:25 -06:00
|
|
|
{
|
2017-01-06 07:41:00 -06:00
|
|
|
if ( (*fr)[acIdx] == HUGE_VAL ) continue;
|
2017-01-03 02:27:25 -06:00
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
if ( maxFraction[acIdx] < (*fr)[acIdx] )
|
2017-01-03 02:27:25 -06:00
|
|
|
{
|
2017-01-06 07:41:00 -06:00
|
|
|
maxFraction[acIdx] = (*fr)[acIdx];
|
2017-03-21 09:56:36 -05:00
|
|
|
maxFractionTracerIdx[acIdx] = resultTracerIdxToGlobalTracerIdx[frIdx];
|
2017-01-03 02:27:25 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-01-06 07:41:00 -06:00
|
|
|
}
|
2017-01-03 02:27:25 -06:00
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
return &maxFractionTracerIdx;
|
|
|
|
}
|
2017-01-03 02:27:25 -06:00
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
std::vector<double>* RigFlowDiagResults::calculateCommunicationResult(const RigFlowDiagResultAddress& resVarAddr, size_t frameIndex)
|
|
|
|
{
|
|
|
|
std::vector<const std::vector<double>* > injectorFractions = findResultsForSelectedTracers(resVarAddr,
|
|
|
|
frameIndex,
|
|
|
|
RIG_FLD_CELL_FRACTION_RESNAME,
|
|
|
|
RimFlowDiagSolution::INJECTOR);
|
|
|
|
std::vector<const std::vector<double>* > producerFractions = findResultsForSelectedTracers(resVarAddr,
|
|
|
|
frameIndex,
|
|
|
|
RIG_FLD_CELL_FRACTION_RESNAME,
|
|
|
|
RimFlowDiagSolution::PRODUCER);
|
|
|
|
size_t activeCellCount = this->activeCellInfo(resVarAddr)->reservoirActiveCellCount();
|
|
|
|
|
|
|
|
std::vector<double> sumOfInjectorFractions;
|
|
|
|
calculateSumOfFractions(injectorFractions, activeCellCount, &sumOfInjectorFractions);
|
|
|
|
|
|
|
|
std::vector<double> sumOfProducerFractions;
|
|
|
|
calculateSumOfFractions(producerFractions, activeCellCount, &sumOfProducerFractions);
|
|
|
|
|
|
|
|
RigFlowDiagResultFrames* commFrames = this->createScalarResult(resVarAddr);
|
|
|
|
std::vector<double>& commPI = commFrames->frameData(frameIndex);
|
|
|
|
commPI.resize(activeCellCount, HUGE_VAL);
|
|
|
|
|
|
|
|
for ( size_t acIdx = 0 ; acIdx < activeCellCount; ++acIdx )
|
|
|
|
{
|
|
|
|
if ( (sumOfInjectorFractions)[acIdx] == HUGE_VAL ) continue;
|
|
|
|
if ( (sumOfProducerFractions)[acIdx] == HUGE_VAL ) continue;
|
|
|
|
|
|
|
|
(commPI)[acIdx] = (sumOfInjectorFractions)[acIdx] * (sumOfProducerFractions)[acIdx];
|
2017-01-03 02:27:25 -06:00
|
|
|
}
|
2016-12-21 10:14:13 -06:00
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
return &commPI;
|
2016-12-16 07:17:56 -06:00
|
|
|
}
|
|
|
|
|
2017-01-02 07:46:12 -06:00
|
|
|
|
2017-01-06 07:41:00 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
std::vector<const std::vector<double>* > RigFlowDiagResults::findResultsForSelectedTracers(const RigFlowDiagResultAddress& resVarAddr,
|
2017-01-02 07:46:12 -06:00
|
|
|
size_t frameIndex,
|
|
|
|
const std::string& nativeResultName,
|
|
|
|
RimFlowDiagSolution::TracerStatusType wantedTracerType)
|
|
|
|
{
|
|
|
|
|
|
|
|
std::vector<const std::vector<double>* > selectedTracersResults;
|
|
|
|
|
|
|
|
for ( const std::string& tracerName: resVarAddr.selectedTracerNames )
|
|
|
|
{
|
|
|
|
RimFlowDiagSolution::TracerStatusType tracerType = m_flowDiagSolution->tracerStatusInTimeStep(QString::fromStdString(tracerName), frameIndex);
|
|
|
|
|
2017-03-07 05:19:19 -06:00
|
|
|
if (tracerType != RimFlowDiagSolution::CLOSED
|
|
|
|
&& ( tracerType == wantedTracerType || wantedTracerType == RimFlowDiagSolution::UNDEFINED) )
|
2017-01-02 07:46:12 -06:00
|
|
|
{
|
2017-08-10 08:08:30 -05:00
|
|
|
selectedTracersResults.push_back(findOrCalculateResult(RigFlowDiagResultAddress(nativeResultName, resVarAddr.phaseSelection, tracerName), frameIndex));
|
2017-01-02 07:46:12 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return selectedTracersResults;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-21 09:56:36 -05:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
std::vector< std::pair<std::string, const std::vector<double>*> >
|
|
|
|
RigFlowDiagResults::findNamedResultsForSelectedTracers(const RigFlowDiagResultAddress& resVarAddr,
|
|
|
|
size_t frameIndex,
|
|
|
|
const std::string& nativeResultName,
|
|
|
|
RimFlowDiagSolution::TracerStatusType wantedTracerType)
|
|
|
|
{
|
|
|
|
|
|
|
|
std::vector<std::pair<std::string, const std::vector<double>* > > selectedTracersResults;
|
|
|
|
|
|
|
|
for ( const std::string& tracerName: resVarAddr.selectedTracerNames )
|
|
|
|
{
|
|
|
|
RimFlowDiagSolution::TracerStatusType tracerType = m_flowDiagSolution->tracerStatusInTimeStep(QString::fromStdString(tracerName), frameIndex);
|
|
|
|
|
|
|
|
if (tracerType != RimFlowDiagSolution::CLOSED
|
|
|
|
&& ( tracerType == wantedTracerType || wantedTracerType == RimFlowDiagSolution::UNDEFINED) )
|
|
|
|
{
|
2017-08-10 08:08:30 -05:00
|
|
|
selectedTracersResults.push_back(std::make_pair(tracerName, findOrCalculateResult(RigFlowDiagResultAddress(nativeResultName, resVarAddr.phaseSelection, tracerName), frameIndex)));
|
2017-03-21 09:56:36 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return selectedTracersResults;
|
|
|
|
}
|
|
|
|
|
2017-01-02 07:46:12 -06:00
|
|
|
|
2016-12-20 04:41:05 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
RigStatisticsDataCache* RigFlowDiagResults::statistics(const RigFlowDiagResultAddress& resVarAddr)
|
|
|
|
{
|
|
|
|
RigStatisticsDataCache* statCache = m_resultStatistics[resVarAddr].p();
|
|
|
|
if ( !statCache )
|
|
|
|
{
|
|
|
|
RigFlowDiagStatCalc* calculator = new RigFlowDiagStatCalc(this, resVarAddr);
|
|
|
|
statCache = new RigStatisticsDataCache(calculator);
|
|
|
|
m_resultStatistics[resVarAddr] = statCache;
|
|
|
|
}
|
|
|
|
|
|
|
|
return statCache;
|
|
|
|
}
|
|
|
|
|
2017-01-02 07:46:12 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void RigFlowDiagResults::calculateSumOfFractions(const std::vector<const std::vector<double> *> &fractions,
|
|
|
|
size_t activeCellCount,
|
|
|
|
std::vector<double>* sumOfFractions)
|
|
|
|
{
|
|
|
|
sumOfFractions->resize(activeCellCount, HUGE_VAL);
|
|
|
|
|
|
|
|
for ( size_t iIdx = 0; iIdx < fractions.size() ; ++iIdx )
|
|
|
|
{
|
|
|
|
const std::vector<double> * fraction = fractions[iIdx];
|
|
|
|
|
|
|
|
if ( ! (fraction) ) continue;
|
|
|
|
|
|
|
|
for ( size_t acIdx = 0 ; acIdx < activeCellCount; ++acIdx )
|
|
|
|
{
|
|
|
|
if ( (*fraction)[acIdx] == HUGE_VAL ) continue;
|
|
|
|
|
|
|
|
if ( (*sumOfFractions)[acIdx] == HUGE_VAL ) (*sumOfFractions)[acIdx] = 0.0;
|
|
|
|
|
|
|
|
(*sumOfFractions)[acIdx] += (*fraction)[acIdx];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-20 04:41:05 -06:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void RigFlowDiagResults::minMaxScalarValues(const RigFlowDiagResultAddress& resVarAddr, int frameIndex,
|
|
|
|
double* localMin, double* localMax)
|
|
|
|
{
|
|
|
|
this->statistics(resVarAddr)->minMaxCellScalarValues(frameIndex, *localMin, *localMax);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void RigFlowDiagResults::minMaxScalarValues(const RigFlowDiagResultAddress& resVarAddr,
|
|
|
|
double* globalMin, double* globalMax)
|
|
|
|
{
|
|
|
|
this->statistics(resVarAddr)->minMaxCellScalarValues(*globalMin, *globalMax);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void RigFlowDiagResults::posNegClosestToZero(const RigFlowDiagResultAddress& resVarAddr, int frameIndex, double* localPosClosestToZero, double* localNegClosestToZero)
|
|
|
|
{
|
|
|
|
this->statistics(resVarAddr)->posNegClosestToZero(frameIndex, *localPosClosestToZero, *localNegClosestToZero);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void RigFlowDiagResults::posNegClosestToZero(const RigFlowDiagResultAddress& resVarAddr, double* globalPosClosestToZero, double* globalNegClosestToZero)
|
|
|
|
{
|
|
|
|
this->statistics(resVarAddr)->posNegClosestToZero(*globalPosClosestToZero, *globalNegClosestToZero);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void RigFlowDiagResults::meanScalarValue(const RigFlowDiagResultAddress& resVarAddr, double* meanValue)
|
|
|
|
{
|
|
|
|
CVF_ASSERT(meanValue);
|
|
|
|
|
|
|
|
this->statistics(resVarAddr)->meanCellScalarValues(*meanValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void RigFlowDiagResults::meanScalarValue(const RigFlowDiagResultAddress& resVarAddr, int frameIndex, double* meanValue)
|
|
|
|
{
|
|
|
|
this->statistics(resVarAddr)->meanCellScalarValues(frameIndex, *meanValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void RigFlowDiagResults::p10p90ScalarValues(const RigFlowDiagResultAddress& resVarAddr, double* p10, double* p90)
|
|
|
|
{
|
|
|
|
this->statistics(resVarAddr)->p10p90CellScalarValues(*p10, *p90);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void RigFlowDiagResults::p10p90ScalarValues(const RigFlowDiagResultAddress& resVarAddr, int frameIndex, double* p10, double* p90)
|
|
|
|
{
|
|
|
|
this->statistics(resVarAddr)->p10p90CellScalarValues(frameIndex, *p10, *p90);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void RigFlowDiagResults::sumScalarValue(const RigFlowDiagResultAddress& resVarAddr, double* sum)
|
|
|
|
{
|
|
|
|
CVF_ASSERT(sum);
|
|
|
|
|
|
|
|
this->statistics(resVarAddr)->sumCellScalarValues(*sum);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void RigFlowDiagResults::sumScalarValue(const RigFlowDiagResultAddress& resVarAddr, int frameIndex, double* sum)
|
|
|
|
{
|
|
|
|
CVF_ASSERT(sum);
|
|
|
|
|
|
|
|
this->statistics(resVarAddr)->sumCellScalarValues(frameIndex, *sum);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
const std::vector<size_t>& RigFlowDiagResults::scalarValuesHistogram(const RigFlowDiagResultAddress& resVarAddr)
|
|
|
|
{
|
|
|
|
return this->statistics(resVarAddr)->cellScalarValuesHistogram();
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
const std::vector<size_t>& RigFlowDiagResults::scalarValuesHistogram(const RigFlowDiagResultAddress& resVarAddr, int frameIndex)
|
|
|
|
{
|
|
|
|
return this->statistics(resVarAddr)->cellScalarValuesHistogram(frameIndex);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
const std::vector<int>& RigFlowDiagResults::uniqueCellScalarValues(const RigFlowDiagResultAddress& resVarAddr)
|
|
|
|
{
|
|
|
|
return this->statistics(resVarAddr)->uniqueCellScalarValues();
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
const std::vector<int>& RigFlowDiagResults::uniqueCellScalarValues(const RigFlowDiagResultAddress& resVarAddr, int frameIndex)
|
|
|
|
{
|
|
|
|
return this->statistics(resVarAddr)->uniqueCellScalarValues(frameIndex);
|
|
|
|
}
|
2017-03-14 13:21:55 -05:00
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
std::pair<double, double> RigFlowDiagResults::injectorProducerPairFluxes(const std::string& injTracername,
|
|
|
|
const std::string& prodTracerName,
|
|
|
|
int frameIndex)
|
|
|
|
{
|
2017-08-10 08:08:30 -05:00
|
|
|
calculateNativeResultsIfNotPreviouslyAttempted(frameIndex, RigFlowDiagResultAddress::PHASE_ALL);
|
2017-03-14 13:21:55 -05:00
|
|
|
|
2017-08-10 08:08:30 -05:00
|
|
|
auto commPair = m_injProdPairFluxCommunicationTimesteps[frameIndex][RigFlowDiagResultAddress::PHASE_ALL].find(std::make_pair(injTracername, prodTracerName));
|
|
|
|
if (commPair != m_injProdPairFluxCommunicationTimesteps[frameIndex][RigFlowDiagResultAddress::PHASE_ALL].end())
|
2017-03-14 13:21:55 -05:00
|
|
|
{
|
|
|
|
return commPair->second;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return std::make_pair(0.0, 0.0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
double RigFlowDiagResults::maxAbsPairFlux(int frameIndex)
|
|
|
|
{
|
2017-08-10 08:08:30 -05:00
|
|
|
calculateNativeResultsIfNotPreviouslyAttempted(frameIndex, RigFlowDiagResultAddress::PHASE_ALL);
|
2017-03-14 13:21:55 -05:00
|
|
|
double maxFlux = 0.0;
|
|
|
|
|
2017-08-10 08:08:30 -05:00
|
|
|
for (const auto& commPair : m_injProdPairFluxCommunicationTimesteps[frameIndex][RigFlowDiagResultAddress::PHASE_ALL])
|
2017-03-14 13:21:55 -05:00
|
|
|
{
|
2017-03-16 10:51:28 -05:00
|
|
|
if (fabs(commPair.second.first) > maxFlux ) maxFlux = fabs(commPair.second.first);
|
|
|
|
if (fabs(commPair.second.second) > maxFlux ) maxFlux = fabs(commPair.second.second);
|
2017-03-14 13:21:55 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return maxFlux;
|
|
|
|
}
|
2017-03-24 09:10:39 -05:00
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2017-08-10 08:08:30 -05:00
|
|
|
std::vector<int> RigFlowDiagResults::calculatedTimeSteps(RigFlowDiagResultAddress::PhaseSelection phaseSelection)
|
2017-03-24 09:10:39 -05:00
|
|
|
{
|
|
|
|
std::vector<int> timestepIndices;
|
|
|
|
for (size_t tsIdx = 0; tsIdx < m_timeStepCount; ++tsIdx)
|
|
|
|
{
|
2017-08-10 08:08:30 -05:00
|
|
|
auto it = m_hasAtemptedNativeResults[tsIdx].find(phaseSelection);
|
|
|
|
if (it != m_hasAtemptedNativeResults[tsIdx].end() && it->second)
|
|
|
|
{
|
|
|
|
timestepIndices.push_back(static_cast<int>(tsIdx));
|
|
|
|
}
|
2017-03-24 09:10:39 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return timestepIndices;
|
|
|
|
}
|
2017-03-28 02:47:06 -05:00
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2017-08-09 07:10:39 -05:00
|
|
|
RigFlowDiagSolverInterface::FlowCharacteristicsResultFrame RigFlowDiagResults::flowCharacteristicsResults(int frameIndex, double max_pv_fraction)
|
2017-03-28 02:47:06 -05:00
|
|
|
{
|
2017-08-09 07:10:39 -05:00
|
|
|
std::vector<QString> tracerNames = m_flowDiagSolution->tracerNames();
|
|
|
|
|
|
|
|
std::set<std::string> injectorNames;
|
|
|
|
std::set<std::string> producerNames;
|
|
|
|
|
|
|
|
for (const QString& tracerName : tracerNames)
|
|
|
|
{
|
|
|
|
RimFlowDiagSolution::TracerStatusType status = m_flowDiagSolution->tracerStatusInTimeStep(tracerName, frameIndex);
|
|
|
|
if (status == RimFlowDiagSolution::INJECTOR)
|
|
|
|
{
|
|
|
|
injectorNames.insert(tracerName.toStdString());
|
|
|
|
}
|
|
|
|
else if (status == RimFlowDiagSolution::PRODUCER)
|
|
|
|
{
|
|
|
|
producerNames.insert(tracerName.toStdString());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-10 08:08:30 -05:00
|
|
|
RigFlowDiagResultAddress injectorAddress(RIG_FLD_TOF_RESNAME, RigFlowDiagResultAddress::PHASE_ALL, injectorNames);
|
|
|
|
RigFlowDiagResultAddress producerAddress(RIG_FLD_TOF_RESNAME, RigFlowDiagResultAddress::PHASE_ALL, producerNames);
|
2017-08-09 07:10:39 -05:00
|
|
|
|
|
|
|
const std::vector<double>* injectorResults = resultValues(injectorAddress, frameIndex);
|
|
|
|
const std::vector<double>* producerResults = resultValues(producerAddress, frameIndex);
|
2017-03-28 02:47:06 -05:00
|
|
|
|
2017-08-10 03:09:01 -05:00
|
|
|
return solverInterface()->calculateFlowCharacteristics(injectorResults, producerResults, max_pv_fraction);
|
2017-03-28 02:47:06 -05:00
|
|
|
}
|