///////////////////////////////////////////////////////////////////////////////// // // 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 // for more details. // ///////////////////////////////////////////////////////////////////////////////// #include "RigFlowDiagResults.h" #include "RigActiveCellInfo.h" #include "RigEclipseCaseData.h" #include "RigFlowDiagSolverInterface.h" #include "RigFlowDiagStatCalc.h" #include "RimEclipseCase.h" #include "RimEclipseResultCase.h" #include "RimFlowDiagSolution.h" //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RigFlowDiagResults::RigFlowDiagResults(RimFlowDiagSolution* flowSolution, size_t timeStepCount) : m_flowDiagSolution(flowSolution) { m_timeStepCount = timeStepCount; m_hasAtemptedNativeResults.resize(timeStepCount, false); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RigFlowDiagResults::~RigFlowDiagResults() { } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- const std::vector* RigFlowDiagResults::resultValues(const RigFlowDiagResultAddress& resVarAddr, size_t frameIndex) { CVF_ASSERT(m_timeStepCount != cvf::UNDEFINED_SIZE_T); // Forgotten to call init return findOrCalculateResult(resVarAddr, frameIndex); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- const RigActiveCellInfo * RigFlowDiagResults::activeCellInfo(const RigFlowDiagResultAddress& resVarAddr) { RimEclipseResultCase* eclCase; m_flowDiagSolution->firstAncestorOrThisOfType(eclCase); return eclCase->reservoirData()->activeCellInfo(RifReaderInterface::MATRIX_RESULTS); // Todo: base on resVarAddr member } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- const std::vector* RigFlowDiagResults::findOrCalculateResult(const RigFlowDiagResultAddress& resVarAddr, size_t frameIndex) { std::vector* 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 if (!solverInterface()) return nullptr; if (!m_hasAtemptedNativeResults[frameIndex]) { RigFlowDiagTimeStepResult nativeTimestepResults = solverInterface()->calculate(frameIndex, m_flowDiagSolution->allInjectorTracerActiveCellIndices(frameIndex), m_flowDiagSolution->allProducerTracerActiveCellIndices(frameIndex)); std::map >& nativeResults = nativeTimestepResults.nativeResults(); for ( auto& resIt: nativeResults ) { RigFlowDiagResultFrames* nativeResFrames = findScalarResult(resIt.first); if ( !nativeResFrames ) nativeResFrames = createScalarResult(resIt.first); nativeResFrames->frameData(frameIndex).swap(resIt.second); } m_hasAtemptedNativeResults[frameIndex] = true; } return findScalarResultFrame(resVarAddr, frameIndex); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- std::vector* RigFlowDiagResults::findScalarResultFrame(const RigFlowDiagResultAddress& resVarAddr, size_t frameIndex) { RigFlowDiagResultFrames* resFrames = findScalarResult (resVarAddr); if ( resFrames ) { std::vector& frame = resFrames->frameData(frameIndex); if ( frame.size() ) return(&frame); } return nullptr; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RigFlowDiagSolverInterface* RigFlowDiagResults::solverInterface() { RimEclipseResultCase* eclCase; m_flowDiagSolution->firstAncestorOrThisOfType(eclCase); return eclCase->flowDiagSolverInterface(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RigFlowDiagResultFrames* RigFlowDiagResults::createScalarResult(const RigFlowDiagResultAddress& resVarAddr) { cvf::ref 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* RigFlowDiagResults::calculateDerivedResult(const RigFlowDiagResultAddress& resVarAddr, size_t frameIndex) { if (resVarAddr.isNativeResult()) return nullptr; if (resVarAddr.variableName == RIG_FLD_TOF_RESNAME) { 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); } return nullptr; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- std::vector* RigFlowDiagResults::calculateAverageTOFResult(const RigFlowDiagResultAddress& resVarAddr, size_t frameIndex) { std::vector* > injectorTOFs = findResultsForSelectedTracers(resVarAddr, frameIndex, RIG_FLD_TOF_RESNAME, RimFlowDiagSolution::INJECTOR); std::vector* > injectorFractions = findResultsForSelectedTracers(resVarAddr, frameIndex, RIG_FLD_CELL_FRACTION_RESNAME, RimFlowDiagSolution::INJECTOR); std::vector* > producerTOFs = findResultsForSelectedTracers(resVarAddr, frameIndex, RIG_FLD_TOF_RESNAME, RimFlowDiagSolution::PRODUCER); std::vector* > producerFractions = findResultsForSelectedTracers(resVarAddr, frameIndex, RIG_FLD_CELL_FRACTION_RESNAME, RimFlowDiagSolution::PRODUCER); size_t activeCellCount = this->activeCellInfo(resVarAddr)->reservoirActiveCellCount(); std::vector injectorTotalFractions; std::vector injectorFractMultTof; calculateSumOfFractionAndFractionMultTOF(activeCellCount, injectorFractions, injectorTOFs, &injectorTotalFractions, &injectorFractMultTof); std::vector producerTotalFractions; std::vector producerFractMultTof; calculateSumOfFractionAndFractionMultTOF(activeCellCount, producerFractions, producerTOFs, &producerTotalFractions, &producerFractMultTof); RigFlowDiagResultFrames* averageTofFrames = this->createScalarResult(resVarAddr); std::vector& averageTof = averageTofFrames->frameData(frameIndex); averageTof.resize(activeCellCount, HUGE_VAL); for ( size_t acIdx = 0 ; acIdx < activeCellCount; ++acIdx ) { if ( injectorTotalFractions[acIdx] == 0.0 && producerTotalFractions[acIdx] == 0.0 ) { 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; } } /// Test to remove all averaging // if (injectorTOFs.size()) averageTof = (*injectorTOFs[0]); return &averageTof; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- std::vector* RigFlowDiagResults::calculateSumOfFractionsResult(const RigFlowDiagResultAddress& resVarAddr, size_t frameIndex) { std::vector* > fractions = findResultsForSelectedTracers(resVarAddr, frameIndex, RIG_FLD_CELL_FRACTION_RESNAME, RimFlowDiagSolution::UNDEFINED); RigFlowDiagResultFrames* sumOfFractionsFrames = this->createScalarResult(resVarAddr); std::vector& sumOfFractions = sumOfFractionsFrames->frameData(frameIndex); size_t activeCellCount = this->activeCellInfo(resVarAddr)->reservoirActiveCellCount(); calculateSumOfFractions(fractions, activeCellCount, &sumOfFractions); return &sumOfFractions; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- std::vector* RigFlowDiagResults::calculateTracerWithMaxFractionResult(const RigFlowDiagResultAddress &resVarAddr, size_t frameIndex) { std::vector selectedTracerIdxToGlobalTracerIdx; { selectedTracerIdxToGlobalTracerIdx.resize(resVarAddr.selectedTracerNames.size(), -1); std::vector allTracerNames = m_flowDiagSolution->tracerNames(); int selTracerIdx = 0; for ( const std::string& tracerName: resVarAddr.selectedTracerNames ) { for ( size_t globIdx = 0; globIdx < allTracerNames.size(); ++globIdx ) { if ( allTracerNames[globIdx].toStdString() == tracerName ) { selectedTracerIdxToGlobalTracerIdx[selTracerIdx] = static_cast(globIdx); break; } } ++selTracerIdx; } } size_t activeCellCount = this->activeCellInfo(resVarAddr)->reservoirActiveCellCount(); RigFlowDiagResultFrames* maxFractionTracerIdxFrames = this->createScalarResult(resVarAddr); std::vector& maxFractionTracerIdx = maxFractionTracerIdxFrames->frameData(frameIndex); { std::vector* > fractions = findResultsForSelectedTracers(resVarAddr, frameIndex, RIG_FLD_CELL_FRACTION_RESNAME, RimFlowDiagSolution::UNDEFINED); maxFractionTracerIdx.resize(activeCellCount, HUGE_VAL); std::vector maxFraction; maxFraction.resize(activeCellCount, -HUGE_VAL); for ( size_t frIdx = 0; frIdx < fractions.size(); ++frIdx ) { const std::vector * fr = fractions[frIdx]; for ( size_t acIdx = 0 ; acIdx < activeCellCount; ++acIdx ) { if ( (*fr)[acIdx] == HUGE_VAL ) continue; if ( maxFraction[acIdx] < (*fr)[acIdx] ) { maxFraction[acIdx] = (*fr)[acIdx]; maxFractionTracerIdx[acIdx] = frIdx; } } } } for ( size_t acIdx = 0 ; acIdx < activeCellCount; ++acIdx ) { if ( maxFractionTracerIdx[acIdx] == HUGE_VAL ) continue; double selectedTracerIdx = static_cast(maxFractionTracerIdx[acIdx]); maxFractionTracerIdx[acIdx] = selectedTracerIdxToGlobalTracerIdx[selectedTracerIdx]; } return &maxFractionTracerIdx; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- std::vector* RigFlowDiagResults::calculateCommunicationResult(const RigFlowDiagResultAddress& resVarAddr, size_t frameIndex) { std::vector* > injectorFractions = findResultsForSelectedTracers(resVarAddr, frameIndex, RIG_FLD_CELL_FRACTION_RESNAME, RimFlowDiagSolution::INJECTOR); std::vector* > producerFractions = findResultsForSelectedTracers(resVarAddr, frameIndex, RIG_FLD_CELL_FRACTION_RESNAME, RimFlowDiagSolution::PRODUCER); size_t activeCellCount = this->activeCellInfo(resVarAddr)->reservoirActiveCellCount(); std::vector sumOfInjectorFractions; calculateSumOfFractions(injectorFractions, activeCellCount, &sumOfInjectorFractions); std::vector sumOfProducerFractions; calculateSumOfFractions(producerFractions, activeCellCount, &sumOfProducerFractions); RigFlowDiagResultFrames* commFrames = this->createScalarResult(resVarAddr); std::vector& 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]; } return &commPI; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- std::vector* > RigFlowDiagResults::findResultsForSelectedTracers(const RigFlowDiagResultAddress& resVarAddr, size_t frameIndex, const std::string& nativeResultName, RimFlowDiagSolution::TracerStatusType wantedTracerType) { std::vector* > 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) ) { selectedTracersResults.push_back(findOrCalculateResult(RigFlowDiagResultAddress(nativeResultName, tracerName), frameIndex)); } } return selectedTracersResults; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RigFlowDiagResults::calculateSumOfFractionAndFractionMultTOF(size_t activeCellCount, const std::vector *> & fractions, const std::vector *> & TOFs, std::vector *sumOfFractions, std::vector *fractionMultTOF) { sumOfFractions->resize(activeCellCount, 0.0); fractionMultTOF->resize(activeCellCount, 0.0); for ( size_t iIdx = 0; iIdx < fractions.size() ; ++iIdx ) { const std::vector * frInj = fractions[iIdx]; const std::vector * 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]; } } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- 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; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RigFlowDiagResults::calculateSumOfFractions(const std::vector *> &fractions, size_t activeCellCount, std::vector* sumOfFractions) { sumOfFractions->resize(activeCellCount, HUGE_VAL); for ( size_t iIdx = 0; iIdx < fractions.size() ; ++iIdx ) { const std::vector * 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]; } } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- 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& RigFlowDiagResults::scalarValuesHistogram(const RigFlowDiagResultAddress& resVarAddr) { return this->statistics(resVarAddr)->cellScalarValuesHistogram(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- const std::vector& RigFlowDiagResults::scalarValuesHistogram(const RigFlowDiagResultAddress& resVarAddr, int frameIndex) { return this->statistics(resVarAddr)->cellScalarValuesHistogram(frameIndex); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- const std::vector& RigFlowDiagResults::uniqueCellScalarValues(const RigFlowDiagResultAddress& resVarAddr) { return this->statistics(resVarAddr)->uniqueCellScalarValues(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- const std::vector& RigFlowDiagResults::uniqueCellScalarValues(const RigFlowDiagResultAddress& resVarAddr, int frameIndex) { return this->statistics(resVarAddr)->uniqueCellScalarValues(frameIndex); }