///////////////////////////////////////////////////////////////////////////////// // // 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 "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" //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- 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.fieldName == "FractureGradient" || resAddr.fieldName == "ShearFailureGradient") { wellPathDerivedCurveData(resAddr, frameIndex, 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 RigFemPart* femPart = m_caseData->femParts()->part(0); 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)); } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RigGeoMechWellLogExtractor::wellPathDerivedCurveData(const RigFemResultAddress& resAddr, int frameIndex, std::vector* values) { // TODO: Read in these values: const double poissonRatio = 0.25; // TODO: Read this in. // Typical UCS: http://ceae.colorado.edu/~amadei/CVEN5768/PDF/NOTES8.pdf // Typical UCS for Shale is 5 - 100 MPa -> 50 - 1000 bar. const double uniaxialStrengthInBars = 100.0; CVF_TIGHT_ASSERT(values); CVF_ASSERT(resAddr.fieldName == "FractureGradient" || resAddr.fieldName == "ShearFailureGradient"); const RigFemPart* femPart = m_caseData->femParts()->part(0); const std::vector& nodeCoords = femPart->nodes().coordinates; RigFemPartResultsCollection* resultCollection = m_caseData->femPartResults(); RigFemResultAddress stressResAddr(RIG_ELEMENT_NODAL, std::string("ST"), ""); stressResAddr.fieldName = std::string("ST"); RigFemResultAddress porBarResAddr(RIG_ELEMENT_NODAL, std::string("POR-Bar"), ""); 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)); } values->resize(m_intersections.size(), 0.0f); std::vector porePressures = resultCollection->resultValues(porBarResAddr, 0, frameIndex); #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; double trueVerticalDepth = -m_intersections[intersectionIdx].z(); double porePressure = trueVerticalDepth * 9.81 / 100.0; if (false && !porePressures.empty()) { float interpolatedPorePressure = interpolateGridResultValue(porBarResAddr.resultPosType, porePressures, intersectionIdx, false); if (interpolatedPorePressure != std::numeric_limits::infinity() && interpolatedPorePressure != -std::numeric_limits::infinity()) { porePressure = static_cast(interpolatedPorePressure); } } caf::Ten3d interpolatedStress = interpolateGridResultValue(stressResAddr.resultPosType, vertexStresses, intersectionIdx, false); cvf::Vec3d wellPathTangent = calculateWellPathTangent(intersectionIdx); caf::Ten3d wellPathStressFloat = transformTensorToWellPathOrientation(wellPathTangent, interpolatedStress); caf::Ten3d wellPathStressDouble(wellPathStressFloat); RigGeoMechBoreHoleStressCalculator sigmaCalculator(wellPathStressDouble, porePressure, poissonRatio, uniaxialStrengthInBars, 32); double resultValue = 0.0; if (resAddr.fieldName == "FractureGradient") { resultValue = sigmaCalculator.solveFractureGradient(); } else { CVF_ASSERT(resAddr.fieldName == "ShearFailureGradient"); resultValue = sigmaCalculator.solveStassiDalia(); } double effectiveDepth = -m_intersections[intersectionIdx].z() + m_rkbDiff; if (effectiveDepth > 1.0e-8) { resultValue *= 100.0 / (effectiveDepth * 9.81); } else { resultValue = std::numeric_limits::infinity(); } (*values)[intersectionIdx] = resultValue; } } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- const RigGeoMechCaseData* RigGeoMechWellLogExtractor::caseData() { return m_caseData.p(); } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- void RigGeoMechWellLogExtractor::setRkbDiff(double rkbDiff) { m_rkbDiff = rkbDiff; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- 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_ELEMENT) { return gridResultValues[elmIdx]; } cvf::StructGridInterface::FaceType cellFace = m_intersectedCellFaces[intersectionIdx]; if (cellFace == cvf::StructGridInterface::NO_FACE) { // TODO: Should interpolate within the whole hexahedron. This requires converting to locals coordinates. // For now just pick the average value for the cell. T sumOfVertexValues = gridResultValues[femPart->elementNodeResultIdx(static_cast(elmIdx), 0)]; for (int i = 1; i < 8; ++i) { sumOfVertexValues = sumOfVertexValues + gridResultValues[femPart->elementNodeResultIdx(static_cast(elmIdx), i)]; } return sumOfVertexValues * (1.0 / 8.0); } int faceNodeCount = 0; const int* faceLocalIndices = RigFemTypes::localElmNodeIndicesForFace(elmType, cellFace, &faceNodeCount); const int* elmNodeIndices = femPart->connectivities(elmIdx); cvf::Vec3d v0(nodeCoords[elmNodeIndices[faceLocalIndices[0]]]); cvf::Vec3d v1(nodeCoords[elmNodeIndices[faceLocalIndices[1]]]); cvf::Vec3d v2(nodeCoords[elmNodeIndices[faceLocalIndices[2]]]); cvf::Vec3d v3(nodeCoords[elmNodeIndices[faceLocalIndices[3]]]); std::vector nodeResIdx(4, cvf::UNDEFINED_SIZE_T); if (resultPosType == RIG_NODAL) { for (size_t i = 0; i < nodeResIdx.size(); ++i) { nodeResIdx[i] = elmNodeIndices[faceLocalIndices[i]]; } } else { for (size_t i = 0; i < nodeResIdx.size(); ++i) { nodeResIdx[i] = (size_t)femPart->elementNodeResultIdx((int)elmIdx, faceLocalIndices[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 otherNodeResIdx = femPart->elementNodeResultIdx(elements[0], static_cast(localIndices[0])); T nodeResultValue = gridResultValues[otherNodeResIdx]; for (size_t j = 1; j < elements.size(); ++j) { otherNodeResIdx = femPart->elementNodeResultIdx(elements[j], static_cast(localIndices[j])); nodeResultValue = nodeResultValue + gridResultValues[otherNodeResIdx]; } 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; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- 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) const { 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()); }