///////////////////////////////////////////////////////////////////////////////// // // Copyright (C) Statoil ASA // Copyright (C) Ceetron Solutions AS // // 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 "RigGeoMechWellLogExtractor.h" #include "RiaDefines.h" #include "RiaWeightedMeanCalculator.h" #include "RigFemTypes.h" #include "RigGeoMechBoreHoleStressCalculator.h" #include "RigFemPart.h" #include "RigFemPartCollection.h" #include "RigGeoMechCaseData.h" #include "RigFemPartResultsCollection.h" #include "RigWellLogExtractionTools.h" #include "RigWellPath.h" #include "RigWellPathIntersectionTools.h" #include "cafTensor3.h" #include "cvfGeometryTools.h" #include "cvfMath.h" #include const double RigGeoMechWellLogExtractor::UNIT_WEIGHT_OF_WATER = 9.81 * 1000.0; // N / m^3 //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- RigGeoMechWellLogExtractor::RigGeoMechWellLogExtractor(RigGeoMechCaseData* aCase, const RigWellPath* wellpath, const std::string& wellCaseErrorMsgName) : RigWellLogExtractor(wellpath, wellCaseErrorMsgName) , m_caseData(aCase) { calculateIntersection(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RigGeoMechWellLogExtractor::curveData(const RigFemResultAddress& resAddr, int frameIndex, std::vector* values) { CVF_TIGHT_ASSERT(values); if (resAddr.resultPosType == RIG_WELLPATH_DERIVED) { if (resAddr.fieldName == RiaDefines::wellPathFGResultName().toStdString() || resAddr.fieldName == RiaDefines::wellPathSFGResultName().toStdString()) { wellBoreWallCurveData(resAddr, frameIndex, values); return; } else if (resAddr.fieldName == "PP" || resAddr.fieldName == "OBG" || resAddr.fieldName == "SH") { wellPathScaledCurveData(resAddr, frameIndex, values); return; } else if (resAddr.fieldName == "Azimuth" || resAddr.fieldName == "Inclination") { wellPathAngles(resAddr, values); return; } } if (!resAddr.isValid()) return; RigFemResultAddress convResAddr = resAddr; // When showing POR results, always use the element nodal result, // to get correct handling of elements without POR results if (convResAddr.fieldName == "POR-Bar") convResAddr.resultPosType = RIG_ELEMENT_NODAL; CVF_ASSERT(resAddr.resultPosType != RIG_WELLPATH_DERIVED); const std::vector& resultValues = m_caseData->femPartResults()->resultValues(convResAddr, 0, frameIndex); if (!resultValues.size()) return; values->resize(m_intersections.size()); for (size_t intersectionIdx = 0; intersectionIdx < m_intersections.size(); ++intersectionIdx) { (*values)[intersectionIdx] = static_cast(interpolateGridResultValue(convResAddr.resultPosType, resultValues, intersectionIdx, false)); } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- float RigGeoMechWellLogExtractor::calculatePorePressureInSegment(int64_t intersectionIdx, float averageSegmentPorePressureBar, double hydroStaticPorePressureBar, double effectiveDepthMeters, const std::vector& poreElementPressuresPascal) const { double porePressure = hydroStaticPorePressureBar; // 1: Try pore pressure from the grid if (porePressure == hydroStaticPorePressureBar && averageSegmentPorePressureBar != std::numeric_limits::infinity() && averageSegmentPorePressureBar > 0.0) { porePressure = averageSegmentPorePressureBar; } // 2: Try mud weight from LAS-file to generate pore pressure if (porePressure == hydroStaticPorePressureBar && !m_wellLogMdAndMudWeightKgPerM3.empty()) { double lasMudWeightKgPerM3 = getWellLogSegmentValue(intersectionIdx, m_wellLogMdAndMudWeightKgPerM3); if (lasMudWeightKgPerM3 != std::numeric_limits::infinity()) { double specificMudWeightNPerM3 = lasMudWeightKgPerM3 * 9.81; double porePressurePascal = specificMudWeightNPerM3 * effectiveDepthMeters; porePressure = pascalToBar(porePressurePascal); } } size_t elmIdx = m_intersectedCellsGlobIdx[intersectionIdx]; // 3: Try pore pressure from element property tables if (porePressure == hydroStaticPorePressureBar && elmIdx < poreElementPressuresPascal.size()) { // Pore pressure from element property tables are in pascal. porePressure = pascalToBar(poreElementPressuresPascal[elmIdx]); } // 4: If no pore-pressure was found, the default value of hydrostatic pore pressure is used. CVF_ASSERT(porePressure >= 0.0); return porePressure; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- float RigGeoMechWellLogExtractor::calculatePoissonRatio(int64_t intersectionIdx, const std::vector& poissonRatios) const { const double defaultPoissonRatio = 0.25; double poissonRatio = defaultPoissonRatio; if (!m_wellLogMdAndPoissonRatios.empty()) { double lasPoissionRatio = getWellLogSegmentValue(intersectionIdx, m_wellLogMdAndPoissonRatios); if (lasPoissionRatio != std::numeric_limits::infinity()) { poissonRatio = lasPoissionRatio; } } size_t elmIdx = m_intersectedCellsGlobIdx[intersectionIdx]; if (poissonRatio == defaultPoissonRatio && elmIdx < poissonRatios.size()) { poissonRatio = poissonRatios[elmIdx]; } return poissonRatio; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- float RigGeoMechWellLogExtractor::calculateUcs(int64_t intersectionIdx, const std::vector& ucsValuesPascal) const { // Typical UCS: http://ceae.colorado.edu/~amadei/CVEN5768/PDF/NOTES8.pdf // Typical UCS for Shale is 5 - 100 MPa -> 50 - 1000 bar. const double defaultUniaxialStrengthInBar = 100.0; double uniaxialStrengthInBar = defaultUniaxialStrengthInBar; if (!m_wellLogMdAndUcsBar.empty()) { double lasUniaxialStrengthInBar = getWellLogSegmentValue(intersectionIdx, m_wellLogMdAndUcsBar); if (lasUniaxialStrengthInBar != std::numeric_limits::infinity()) { uniaxialStrengthInBar = lasUniaxialStrengthInBar; } } size_t elmIdx = m_intersectedCellsGlobIdx[intersectionIdx]; if (uniaxialStrengthInBar == defaultUniaxialStrengthInBar && elmIdx < ucsValuesPascal.size()) { // Read UCS from element table in Pascal uniaxialStrengthInBar = pascalToBar(ucsValuesPascal[elmIdx]); } return uniaxialStrengthInBar; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RigGeoMechWellLogExtractor::wellPathAngles(const RigFemResultAddress& resAddr, std::vector* values) { CVF_ASSERT(values); CVF_ASSERT(resAddr.fieldName == "Azimuth" || resAddr.fieldName == "Inclination"); values->resize(m_intersections.size(), 0.0f); const double epsilon = 1.0e-6 * 360; const cvf::Vec3d trueNorth(0.0, 1.0, 0.0); const cvf::Vec3d up(0.0, 0.0, 1.0); for (int64_t intersectionIdx = 0; intersectionIdx < (int64_t)m_intersections.size(); ++intersectionIdx) { cvf::Vec3d wellPathTangent = calculateWellPathTangent(intersectionIdx, TangentFollowWellPathSegments); // Deviation from vertical. Since well path is tending downwards we compare with negative z. double inclination = cvf::Math::toDegrees(std::acos(cvf::Vec3d(0.0, 0.0, -1.0) * wellPathTangent.getNormalized())); if (resAddr.fieldName == "Azimuth") { double azimuth = HUGE_VAL; // Azimuth is not defined when well path is vertical. We define it as infinite to avoid it showing up in the plot. if (cvf::Math::valueInRange(inclination, epsilon, 180.0 - epsilon)) { cvf::Vec3d projectedTangentXY = wellPathTangent; projectedTangentXY.z() = 0.0; // Do tangentXY to true north for clockwise angles. double dotProduct = projectedTangentXY * trueNorth; double crossProduct = (projectedTangentXY ^ trueNorth) * up; // http://www.glossary.oilfield.slb.com/Terms/a/azimuth.aspx azimuth = cvf::Math::toDegrees(std::atan2(crossProduct, dotProduct)); if (azimuth < 0.0) { // Straight atan2 gives angle from -PI to PI yielding angles from -180 to 180 // where the negative angles are counter clockwise. // To get all positive clockwise angles, we add 360 degrees to negative angles. azimuth = azimuth + 360.0; } } (*values)[intersectionIdx] = azimuth; } else { (*values)[intersectionIdx] = inclination; } } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RigGeoMechWellLogExtractor::wellPathScaledCurveData(const RigFemResultAddress& resAddr, int frameIndex, std::vector* values) { CVF_ASSERT(values); const RigFemPart* femPart = m_caseData->femParts()->part(0); RigFemPartResultsCollection* resultCollection = m_caseData->femPartResults(); std::string nativeFieldName; std::string nativeCompName; if (resAddr.fieldName == "PP") { nativeFieldName = "POR-Bar"; // More likely to be in memory than POR } else if (resAddr.fieldName == "OBG") { nativeFieldName = "ST"; nativeCompName = "S33"; } else if (resAddr.fieldName == "SH") { nativeFieldName = "ST"; nativeCompName = "S3"; } RigFemResultAddress nativeAddr(RIG_ELEMENT_NODAL, nativeFieldName, nativeCompName); RigFemResultAddress porElementResAddr(RIG_ELEMENT, "POR", ""); std::vector unscaledResultValues = resultCollection->resultValues(nativeAddr, 0, frameIndex); std::vector poreElementPressuresPascal = resultCollection->resultValues(porElementResAddr, 0, frameIndex); std::vector interpolatedInterfaceValues; interpolatedInterfaceValues.resize(m_intersections.size(), std::numeric_limits::infinity()); #pragma omp parallel for for (int64_t intersectionIdx = 0; intersectionIdx < (int64_t)m_intersections.size(); ++intersectionIdx) { size_t elmIdx = m_intersectedCellsGlobIdx[intersectionIdx]; RigElementType elmType = femPart->elementType(elmIdx); if (!(elmType == HEX8 || elmType == HEX8P)) continue; interpolatedInterfaceValues[intersectionIdx] = interpolateGridResultValue(nativeAddr.resultPosType, unscaledResultValues, intersectionIdx, false); } values->resize(m_intersections.size(), 0.0f); #pragma omp parallel for for (int64_t intersectionIdx = 0; intersectionIdx < (int64_t)m_intersections.size(); ++intersectionIdx) { // Set the value to invalid by default (*values)[intersectionIdx] = std::numeric_limits::infinity(); size_t elmIdx = m_intersectedCellsGlobIdx[intersectionIdx]; RigElementType elmType = femPart->elementType(elmIdx); if (!(elmType == HEX8 || elmType == HEX8P)) continue; cvf::Vec3f centroid = cellCentroid(intersectionIdx); double trueVerticalDepth = -centroid.z(); double effectiveDepthMeters = trueVerticalDepth + m_rkbDiff; double hydroStaticPorePressureBar = pascalToBar(effectiveDepthMeters * UNIT_WEIGHT_OF_WATER); float averageUnscaledValue = std::numeric_limits::infinity(); bool validAverage = averageIntersectionValuesToSegmentValue(intersectionIdx, interpolatedInterfaceValues, std::numeric_limits::infinity(), &averageUnscaledValue); if (resAddr.fieldName == "PP" && validAverage) { double segmentPorePressureFromGrid = averageUnscaledValue; averageUnscaledValue = calculatePorePressureInSegment(intersectionIdx, segmentPorePressureFromGrid, hydroStaticPorePressureBar, effectiveDepthMeters, poreElementPressuresPascal); } (*values)[intersectionIdx] = static_cast(averageUnscaledValue) / hydroStaticPorePressureBar; } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RigGeoMechWellLogExtractor::wellBoreWallCurveData(const RigFemResultAddress& resAddr, int frameIndex, std::vector* values) { CVF_ASSERT(values); CVF_ASSERT(resAddr.fieldName == RiaDefines::wellPathFGResultName().toStdString() || resAddr.fieldName == RiaDefines::wellPathSFGResultName().toStdString()); // The result addresses needed RigFemResultAddress stressResAddr(RIG_ELEMENT_NODAL, "ST", ""); RigFemResultAddress porBarResAddr(RIG_ELEMENT_NODAL, "POR-Bar", ""); // Allow POR as an element property value RigFemResultAddress porElementResAddr(RIG_ELEMENT, "POR", ""); RigFemResultAddress poissonResAddr(RIG_ELEMENT, "RATIO", ""); RigFemResultAddress ucsResAddr(RIG_ELEMENT, "UCS", ""); const RigFemPart* femPart = m_caseData->femParts()->part(0); RigFemPartResultsCollection* resultCollection = m_caseData->femPartResults(); // Load results std::vector vertexStressesFloat = resultCollection->tensors(stressResAddr, 0, frameIndex); if (!vertexStressesFloat.size()) return; std::vector vertexStresses; vertexStresses.reserve(vertexStressesFloat.size()); for (const caf::Ten3f& floatTensor : vertexStressesFloat) { vertexStresses.push_back(caf::Ten3d(floatTensor)); } std::vector porePressures = resultCollection->resultValues(porBarResAddr, 0, frameIndex); std::vector poreElementPressuresPascal = resultCollection->resultValues(porElementResAddr, 0, frameIndex); std::vector poissonRatios = resultCollection->resultValues(poissonResAddr, 0, frameIndex); std::vector ucsValuesPascal = resultCollection->resultValues(ucsResAddr, 0, frameIndex); std::vector interpolatedInterfacePorePressureBar; interpolatedInterfacePorePressureBar.resize(m_intersections.size(), std::numeric_limits::infinity()); std::vector interpolatedInterfaceStressBar; interpolatedInterfaceStressBar.resize(m_intersections.size()); #pragma omp parallel for for (int64_t intersectionIdx = 0; intersectionIdx < (int64_t)m_intersections.size(); ++intersectionIdx) { size_t elmIdx = m_intersectedCellsGlobIdx[intersectionIdx]; RigElementType elmType = femPart->elementType(elmIdx); if (!(elmType == HEX8 || elmType == HEX8P)) continue; interpolatedInterfacePorePressureBar[intersectionIdx] = interpolateGridResultValue(porBarResAddr.resultPosType, porePressures, intersectionIdx, false); interpolatedInterfaceStressBar[intersectionIdx] = interpolateGridResultValue(stressResAddr.resultPosType, vertexStresses, intersectionIdx, false); } values->resize(m_intersections.size(), 0.0f); #pragma omp parallel for for (int64_t intersectionIdx = 0; intersectionIdx < (int64_t) m_intersections.size(); ++intersectionIdx) { size_t elmIdx = m_intersectedCellsGlobIdx[intersectionIdx]; RigElementType elmType = femPart->elementType(elmIdx); if (!(elmType == HEX8 || elmType == HEX8P)) continue; cvf::Vec3f centroid = cellCentroid(intersectionIdx); double trueVerticalDepth = -centroid.z(); double effectiveDepthMeters = trueVerticalDepth + m_rkbDiff; double hydroStaticPorePressureBar = pascalToBar(effectiveDepthMeters * UNIT_WEIGHT_OF_WATER); float averagePorePressureBar = std::numeric_limits::infinity(); bool validGridPorePressure = averageIntersectionValuesToSegmentValue(intersectionIdx, interpolatedInterfacePorePressureBar, std::numeric_limits::infinity(), &averagePorePressureBar); bool isFGregion = validGridPorePressure; // FG is for sands, SFG for shale. Sands has PP, shale does not. double porePressureBar = calculatePorePressureInSegment(intersectionIdx, averagePorePressureBar, hydroStaticPorePressureBar, effectiveDepthMeters, poreElementPressuresPascal); double poissonRatio = calculatePoissonRatio(intersectionIdx, poissonRatios); double ucsBar = calculateUcs(intersectionIdx, ucsValuesPascal); caf::Ten3d segmentStress; bool validSegmentStress = averageIntersectionValuesToSegmentValue(intersectionIdx, interpolatedInterfaceStressBar, caf::Ten3d::invalid(), &segmentStress); cvf::Vec3d wellPathTangent = calculateWellPathTangent(intersectionIdx, TangentConstantWithinCell); caf::Ten3d wellPathStressFloat = transformTensorToWellPathOrientation(wellPathTangent, segmentStress); caf::Ten3d wellPathStressDouble(wellPathStressFloat); RigGeoMechBoreHoleStressCalculator sigmaCalculator(wellPathStressDouble, porePressureBar, poissonRatio, ucsBar, 32); double resultValue = std::numeric_limits::infinity(); if (resAddr.fieldName == RiaDefines::wellPathFGResultName().toStdString()) { if (isFGregion && validSegmentStress) { resultValue = sigmaCalculator.solveFractureGradient(); } } else { CVF_ASSERT(resAddr.fieldName == RiaDefines::wellPathSFGResultName().toStdString()); if (!isFGregion && validSegmentStress) { resultValue = sigmaCalculator.solveStassiDalia(); } } if (resultValue != std::numeric_limits::infinity()) { if (hydroStaticPorePressureBar > 1.0e-8) { resultValue /= hydroStaticPorePressureBar; } } (*values)[intersectionIdx] = resultValue; } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- const RigGeoMechCaseData* RigGeoMechWellLogExtractor::caseData() { return m_caseData.p(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RigGeoMechWellLogExtractor::setRkbDiff(double rkbDiff) { m_rkbDiff = rkbDiff; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RigGeoMechWellLogExtractor::setWellLogMdAndMudWeightKgPerM3(const std::vector>& porePressures) { m_wellLogMdAndMudWeightKgPerM3 = porePressures; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RigGeoMechWellLogExtractor::setWellLogMdAndUcsBar(const std::vector>& ucsValues) { m_wellLogMdAndUcsBar = ucsValues; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RigGeoMechWellLogExtractor::setWellLogMdAndPoissonRatio(const std::vector>& poissonRatios) { m_wellLogMdAndPoissonRatios = poissonRatios; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- template T RigGeoMechWellLogExtractor::interpolateGridResultValue(RigFemResultPosEnum resultPosType, const std::vector& gridResultValues, int64_t intersectionIdx, bool averageNodeElementResults) const { const RigFemPart* femPart = m_caseData->femParts()->part(0); const std::vector& nodeCoords = femPart->nodes().coordinates; size_t elmIdx = m_intersectedCellsGlobIdx[intersectionIdx]; RigElementType elmType = femPart->elementType(elmIdx); if (!(elmType == HEX8 || elmType == HEX8P)) return T(); if (resultPosType == RIG_FORMATION_NAMES) { resultPosType = RIG_ELEMENT_NODAL; // formation indices are stored per element node result. } if (resultPosType == RIG_ELEMENT) { return gridResultValues[elmIdx]; } cvf::StructGridInterface::FaceType cellFace = m_intersectedCellFaces[intersectionIdx]; if (cellFace == cvf::StructGridInterface::NO_FACE) { if (resultPosType == RIG_ELEMENT_NODAL_FACE) { return std::numeric_limits::infinity(); // undefined value. ELEMENT_NODAL_FACE values are only defined on a face. } // TODO: Should interpolate within the whole hexahedron. This requires converting to locals coordinates. // For now just pick the average value for the cell. size_t gridResultValueIdx = femPart->resultValueIdxFromResultPosType(resultPosType, static_cast(elmIdx), 0); T sumOfVertexValues = gridResultValues[gridResultValueIdx]; for (int i = 1; i < 8; ++i) { gridResultValueIdx = femPart->resultValueIdxFromResultPosType(resultPosType, static_cast(elmIdx), i); sumOfVertexValues = sumOfVertexValues + gridResultValues[gridResultValueIdx]; } return sumOfVertexValues * (1.0 / 8.0); } int faceNodeCount = 0; const int* elementLocalIndicesForFace = RigFemTypes::localElmNodeIndicesForFace(elmType, cellFace, &faceNodeCount); const int* elmNodeIndices = femPart->connectivities(elmIdx); cvf::Vec3d v0(nodeCoords[elmNodeIndices[elementLocalIndicesForFace[0]]]); cvf::Vec3d v1(nodeCoords[elmNodeIndices[elementLocalIndicesForFace[1]]]); cvf::Vec3d v2(nodeCoords[elmNodeIndices[elementLocalIndicesForFace[2]]]); cvf::Vec3d v3(nodeCoords[elmNodeIndices[elementLocalIndicesForFace[3]]]); std::vector nodeResIdx(4, cvf::UNDEFINED_SIZE_T); for (size_t i = 0; i < nodeResIdx.size(); ++i) { if (resultPosType == RIG_ELEMENT_NODAL_FACE) { nodeResIdx[i] = gridResultIndexFace(elmIdx, cellFace, static_cast(i)); } else { nodeResIdx[i] = femPart->resultValueIdxFromResultPosType(resultPosType, static_cast(elmIdx), elementLocalIndicesForFace[i]); } } std::vector nodeResultValues; nodeResultValues.reserve(4); if (resultPosType == RIG_ELEMENT_NODAL && averageNodeElementResults) { // Estimate nodal values as the average of the node values from each connected element. for (size_t i = 0; i < nodeResIdx.size(); ++i) { int nodeIndex = femPart->nodeIdxFromElementNodeResultIdx(nodeResIdx[i]); const std::vector& elements = femPart->elementsUsingNode(nodeIndex); const std::vector& localIndices = femPart->elementLocalIndicesForNode(nodeIndex); size_t otherGridResultValueIdx = femPart->resultValueIdxFromResultPosType(resultPosType, elements[0], static_cast(localIndices[0])); T nodeResultValue = gridResultValues[otherGridResultValueIdx]; for (size_t j = 1; j < elements.size(); ++j) { otherGridResultValueIdx = femPart->resultValueIdxFromResultPosType(resultPosType, elements[j], static_cast(localIndices[j])); nodeResultValue = nodeResultValue + gridResultValues[otherGridResultValueIdx]; } nodeResultValue = nodeResultValue * (1.0 / elements.size()); nodeResultValues.push_back(nodeResultValue); } } else { for (size_t i = 0; i < nodeResIdx.size(); ++i) { nodeResultValues.push_back(gridResultValues[nodeResIdx[i]]); } } T interpolatedValue = cvf::GeometryTools::interpolateQuad( v0, nodeResultValues[0], v1, nodeResultValues[1], v2, nodeResultValues[2], v3, nodeResultValues[3], m_intersections[intersectionIdx] ); return interpolatedValue; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- size_t RigGeoMechWellLogExtractor::gridResultIndexFace(size_t elementIdx, cvf::StructGridInterface::FaceType cellFace, int faceLocalNodeIdx) const { CVF_ASSERT(cellFace != cvf::StructGridInterface::NO_FACE && faceLocalNodeIdx < 4); return elementIdx * 24 + static_cast(cellFace) * 4 + faceLocalNodeIdx; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RigGeoMechWellLogExtractor::calculateIntersection() { CVF_ASSERT(m_caseData->femParts()->partCount() == 1); std::map uniqueIntersections; const RigFemPart* femPart = m_caseData->femParts()->part(0); const std::vector& nodeCoords = femPart->nodes().coordinates; for (size_t wpp = 0; wpp < m_wellPath->m_wellPathPoints.size() - 1; ++wpp) { std::vector intersections; cvf::Vec3d p1 = m_wellPath->m_wellPathPoints[wpp]; cvf::Vec3d p2 = m_wellPath->m_wellPathPoints[wpp+1]; cvf::BoundingBox bb; bb.add(p1); bb.add(p2); std::vector closeCells = findCloseCells(bb); cvf::Vec3d hexCorners[8]; for (size_t ccIdx = 0; ccIdx < closeCells.size(); ++ccIdx) { RigElementType elmType = femPart->elementType(closeCells[ccIdx]); if (!(elmType == HEX8 || elmType == HEX8P)) continue; const int* cornerIndices = femPart->connectivities(closeCells[ccIdx]); hexCorners[0] = cvf::Vec3d(nodeCoords[cornerIndices[0]]); hexCorners[1] = cvf::Vec3d(nodeCoords[cornerIndices[1]]); hexCorners[2] = cvf::Vec3d(nodeCoords[cornerIndices[2]]); hexCorners[3] = cvf::Vec3d(nodeCoords[cornerIndices[3]]); hexCorners[4] = cvf::Vec3d(nodeCoords[cornerIndices[4]]); hexCorners[5] = cvf::Vec3d(nodeCoords[cornerIndices[5]]); hexCorners[6] = cvf::Vec3d(nodeCoords[cornerIndices[6]]); hexCorners[7] = cvf::Vec3d(nodeCoords[cornerIndices[7]]); //int intersectionCount = RigHexIntersector::lineHexCellIntersection(p1, p2, hexCorners, closeCells[ccIdx], &intersections); RigHexIntersectionTools::lineHexCellIntersection(p1, p2, hexCorners, closeCells[ccIdx], &intersections); } // Now, with all the intersections of this piece of line, we need to // sort them in order, and set the measured depth and corresponding cell index // Inserting the intersections in this map will remove identical intersections // and sort them according to MD, CellIdx, Leave/enter double md1 = m_wellPath->m_measuredDepths[wpp]; double md2 = m_wellPath->m_measuredDepths[wpp+1]; insertIntersectionsInMap(intersections, p1, md1, p2, md2, &uniqueIntersections); } this->populateReturnArrays(uniqueIntersections); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- std::vector RigGeoMechWellLogExtractor::findCloseCells(const cvf::BoundingBox& bb) { std::vector closeCells; if (m_caseData->femParts()->partCount()) { m_caseData->femParts()->part(0)->findIntersectingCells(bb, &closeCells); } return closeCells; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- cvf::Vec3d RigGeoMechWellLogExtractor::calculateLengthInCell(size_t cellIndex, const cvf::Vec3d& startPoint, const cvf::Vec3d& endPoint) const { std::array hexCorners; const RigFemPart* femPart = m_caseData->femParts()->part(0); const std::vector& nodeCoords = femPart->nodes().coordinates; const int* cornerIndices = femPart->connectivities(cellIndex); hexCorners[0] = cvf::Vec3d(nodeCoords[cornerIndices[0]]); hexCorners[1] = cvf::Vec3d(nodeCoords[cornerIndices[1]]); hexCorners[2] = cvf::Vec3d(nodeCoords[cornerIndices[2]]); hexCorners[3] = cvf::Vec3d(nodeCoords[cornerIndices[3]]); hexCorners[4] = cvf::Vec3d(nodeCoords[cornerIndices[4]]); hexCorners[5] = cvf::Vec3d(nodeCoords[cornerIndices[5]]); hexCorners[6] = cvf::Vec3d(nodeCoords[cornerIndices[6]]); hexCorners[7] = cvf::Vec3d(nodeCoords[cornerIndices[7]]); return RigWellPathIntersectionTools::calculateLengthInCell(hexCorners, startPoint, endPoint); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- cvf::Vec3d RigGeoMechWellLogExtractor::calculateWellPathTangent(int64_t intersectionIdx, WellPathTangentCalculation calculationType) const { if (calculationType == TangentFollowWellPathSegments) { cvf::Vec3d segmentStart, segmentEnd; m_wellPath->twoClosestPoints(m_intersections[intersectionIdx], &segmentStart, &segmentEnd); return (segmentEnd - segmentStart).getNormalized(); } else { cvf::Vec3d wellPathTangent; if (intersectionIdx % 2 == 0) { wellPathTangent = m_intersections[intersectionIdx + 1] - m_intersections[intersectionIdx]; } else { wellPathTangent = m_intersections[intersectionIdx] - m_intersections[intersectionIdx - 1]; } CVF_ASSERT(wellPathTangent.length() > 1.0e-7); return wellPathTangent.getNormalized(); } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- caf::Ten3d RigGeoMechWellLogExtractor::transformTensorToWellPathOrientation(const cvf::Vec3d& wellPathTangent, const caf::Ten3d& tensor) { // Create local coordinate system for well path segment cvf::Vec3d local_z = wellPathTangent; cvf::Vec3d local_x = local_z.perpendicularVector().getNormalized(); cvf::Vec3d local_y = (local_z ^ local_x).getNormalized(); // Calculate the rotation matrix from global i, j, k to local x, y, z. cvf::Mat4d rotationMatrix = cvf::Mat4d::fromCoordSystemAxes(&local_x, &local_y, &local_z); return tensor.rotated(rotationMatrix.toMatrix3()); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- cvf::Vec3f RigGeoMechWellLogExtractor::cellCentroid(size_t intersectionIdx) const { const RigFemPart* femPart = m_caseData->femParts()->part(0); const std::vector& nodeCoords = femPart->nodes().coordinates; size_t elmIdx = m_intersectedCellsGlobIdx[intersectionIdx]; RigElementType elmType = femPart->elementType(elmIdx); int elementNodeCount = RigFemTypes::elmentNodeCount(elmType); const int* elmNodeIndices = femPart->connectivities(elmIdx); cvf::Vec3f centroid(0.0, 0.0, 0.0); for (int i = 0; i < elementNodeCount; ++i) { centroid += nodeCoords[elmNodeIndices[i]]; } return centroid / elementNodeCount; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- double RigGeoMechWellLogExtractor::getWellLogSegmentValue(size_t intersectionIdx, const std::vector>& wellLogValues) const { double startMD, endMD; if (intersectionIdx % 2 == 0) { startMD = m_intersectionMeasuredDepths[intersectionIdx]; endMD = m_intersectionMeasuredDepths[intersectionIdx + 1]; } else { startMD = m_intersectionMeasuredDepths [intersectionIdx - 1]; endMD = m_intersectionMeasuredDepths[intersectionIdx]; } RiaWeightedMeanCalculator averageCalc; for (auto& depthAndValue : wellLogValues) { if (cvf::Math::valueInRange(depthAndValue.first, startMD, endMD)) { cvf::Vec3d position = m_wellPath->interpolatedPointAlongWellPath(depthAndValue.first); cvf::Vec3d centroid(cellCentroid(intersectionIdx)); double weight = 1.0; double dist = (position - centroid).length(); if (dist > 1.0) { weight = 1.0 / dist; } averageCalc.addValueAndWeight(depthAndValue.second, weight); } } if (averageCalc.validAggregatedWeight()) { return averageCalc.weightedMean(); } return std::numeric_limits::infinity(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- double RigGeoMechWellLogExtractor::pascalToBar(double pascalValue) { return pascalValue * 1.0e-5; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- template bool RigGeoMechWellLogExtractor::averageIntersectionValuesToSegmentValue(size_t intersectionIdx, const std::vector& values, const T& invalidValue, T* averagedCellValue) const { CVF_ASSERT(values.size() >= 2); *averagedCellValue = invalidValue; T value1, value2; cvf::Vec3d centroid(cellCentroid(intersectionIdx)); double dist1 = 0.0, dist2 = 0.0; if (intersectionIdx % 2 == 0) { value1 = values[intersectionIdx]; value2 = values[intersectionIdx + 1]; dist1 = (centroid - m_intersections[intersectionIdx]).length(); dist2 = (centroid - m_intersections[intersectionIdx + 1]).length(); } else { value1 = values[intersectionIdx - 1]; value2 = values[intersectionIdx]; dist1 = (centroid - m_intersections[intersectionIdx - 1]).length(); dist2 = (centroid - m_intersections[intersectionIdx]).length(); } if (invalidValue == value1 || invalidValue == value2) { return false; } RiaWeightedMeanCalculator averageCalc; averageCalc.addValueAndWeight(value1, dist2); averageCalc.addValueAndWeight(value2, dist1); if (averageCalc.validAggregatedWeight()) { *averagedCellValue = averageCalc.weightedMean(); } return true; }