20 #ifndef GEOS_MESH_UTILITIES_COMPUTATIONALGEOMETRY_HPP_
21 #define GEOS_MESH_UTILITIES_COMPUTATIONALGEOMETRY_HPP_
29 #include "LvArray/src/output.hpp"
30 #include "LvArray/src/tensorOps.hpp"
34 namespace computationalGeometry
53 template<
typename LINEDIR_TYPE,
57 typename INTPOINT_TYPE >
59 POINT_TYPE
const & linePoint,
60 NORMAL_TYPE
const & planeNormal,
61 ORIGIN_TYPE
const & planeOrigin,
62 INTPOINT_TYPE & intersectionPoint )
70 real64 dummy[ 3 ] = LVARRAY_TENSOROPS_INIT_LOCAL_3( planeOrigin );
71 LvArray::tensorOps::subtract< 3 >( dummy, linePoint );
72 real64 const d = LvArray::tensorOps::AiBi< 3 >( dummy, planeNormal ) /
73 LvArray::tensorOps::AiBi< 3 >( lineDir, planeNormal );
75 LvArray::tensorOps::copy< 3 >( intersectionPoint, linePoint );
76 LvArray::tensorOps::scaledAdd< 3 >( intersectionPoint, lineDir, d );
86 template<
typename NORMAL_TYPE >
88 NORMAL_TYPE
const & normal )
99 LvArray::tensorOps::fill< 3 >( centroid, 0 );
102 LvArray::tensorOps::add< 3 >( centroid, points[ a ] );
106 LvArray::tensorOps::scale< 3 >( centroid, 1.0 / numPoints );
108 real64 v0[3] = LVARRAY_TENSOROPS_INIT_LOCAL_3( centroid );
109 LvArray::tensorOps::subtract< 3 >( v0, points[ 0 ] );
110 LvArray::tensorOps::normalize< 3 >( v0 );
116 real64 v[3] = LVARRAY_TENSOROPS_INIT_LOCAL_3( centroid );
117 LvArray::tensorOps::subtract< 3 >( v, points[ a ] );
118 real64 const dot = LvArray::tensorOps::AiBi< 3 >( v, v0 );
121 LvArray::tensorOps::crossProduct( crossProduct, v, v0 );
122 real64 const det = LvArray::tensorOps::AiBi< 3 >( normal, crossProduct );
124 angle[ a ] = std::atan2( det, dot );
128 std::sort( indices.begin(), indices.end(), [&](
int i,
int j ) { return angle[ i ] < angle[ j ]; } );
134 LvArray::tensorOps::copy< 3 >( orderedPoints[ a ], points[ indices[ a ] ] );
139 LvArray::tensorOps::copy< 3 >( points[a], orderedPoints[a] );
152 template<
typename NORMAL_TYPE >
154 NORMAL_TYPE
const && normal )
160 for(
localIndex a = 0; a < points.size( 0 ); a++ )
162 LvArray::tensorOps::copy< 3 >( orderedPoints[a], points[a] );
167 for(
localIndex a = 0; a < points.size( 0 ) - 2; ++a )
169 real64 v1[ 3 ] = LVARRAY_TENSOROPS_INIT_LOCAL_3( orderedPoints[ a + 1 ] );
170 real64 v2[ 3 ] = LVARRAY_TENSOROPS_INIT_LOCAL_3( orderedPoints[ a + 2 ] );
172 LvArray::tensorOps::subtract< 3 >( v1, orderedPoints[ 0 ] );
173 LvArray::tensorOps::subtract< 3 >( v2, orderedPoints[ 0 ] );
175 real64 triangleNormal[ 3 ];
176 LvArray::tensorOps::crossProduct( triangleNormal, v1, v2 );
177 surfaceArea += LvArray::tensorOps::l2Norm< 3 >( triangleNormal );
180 return surfaceArea * 0.5;
191 template< localIndex DIMENSION,
typename POINT_COORDS_TYPE >
198 for(
localIndex numPoint = 0; numPoint < numPoints; ++numPoint )
200 for(
localIndex numOthPoint = 0; numOthPoint < numPoint; ++numOthPoint )
202 real64 candidateDiameter = 0.0;
205 real64 coordDiff = points[numPoint][i] - points[numOthPoint][i];
206 candidateDiameter += coordDiff * coordDiff;
208 if( diameter < candidateDiameter )
210 diameter = candidateDiameter;
214 return LvArray::math::sqrt< real64 >( diameter );
231 template<
typename CENTER_TYPE,
typename NORMAL_TYPE >
236 CENTER_TYPE && center,
237 NORMAL_TYPE && normal,
238 real64 const areaTolerance = 0.0 )
241 LvArray::tensorOps::fill< 3 >( center, 0 );
242 LvArray::tensorOps::fill< 3 >( normal, 0 );
244 localIndex const numberOfPoints = pointsIndices.size();
248 real64 current[ 3 ], next[ 3 ], origin[ 3 ], crossProduct[ 3 ];
250 LvArray::tensorOps::copy< 3 >( next, points[ pointsIndices[ numberOfPoints - 1 ] ] );
251 LvArray::tensorOps::copy< 3 >( origin, points[ pointsIndices[ 0 ]] );
255 LvArray::tensorOps::copy< 3 >( current, points[ pointsIndices[ a++ ]] );
256 LvArray::tensorOps::scaledAdd< 3 >( current, origin, -1. );
257 LvArray::tensorOps::copy< 3 >( next, points[ pointsIndices[ a % numberOfPoints ] ] );
258 LvArray::tensorOps::scaledAdd< 3 >( next, origin, -1. );
260 LvArray::tensorOps::crossProduct( crossProduct, current, next );
262 LvArray::tensorOps::add< 3 >( normal, crossProduct );
263 LvArray::tensorOps::add< 3 >( center, next );
266 area = LvArray::tensorOps::l2Norm< 3 >( normal );
267 LvArray::tensorOps::scale< 3 >( center, 1.0 / numberOfPoints );
268 LvArray::tensorOps::scaledAdd< 3 >( center, origin, 1. );
270 if( area > areaTolerance )
272 LvArray::tensorOps::normalize< 3 >( normal );
275 else if( area < -areaTolerance )
279 GEOS_LOG_RANK(
"Points: " << points[ pointsIndices[ a ] ] <<
" " << pointsIndices[ a ] );
281 #if defined(GEOS_DEVICE_COMPILE)
284 GEOS_ERROR( GEOS_FMT(
"Negative area found : {}", area ) );
291 GEOS_LOG_RANK(
"Points: " << points[ pointsIndices[ a ] ] <<
" " << pointsIndices[ a ] );
293 #if defined(GEOS_DEVICE_COMPILE)
296 GEOS_ERROR( GEOS_FMT(
"Null area found : {}", area ) );
310 template<
typename NORMAL_TYPE >
318 if( normal[ 2 ] <= -orientationTolerance )
320 LvArray::tensorOps::scale< 3 >( normal, -1.0 );
322 else if( std::fabs( normal[ 2 ] ) < orientationTolerance )
325 if( normal[ 1 ] <= -orientationTolerance )
327 LvArray::tensorOps::scale< 3 >( normal, -1.0 );
329 else if( fabs( normal[ 1 ] ) < orientationTolerance )
332 if( normal[ 0 ] <= -orientationTolerance )
334 LvArray::tensorOps::scale< 3 >( normal, -1.0 );
347 template<
typename NORMAL_TYPE,
typename MATRIX_TYPE >
350 MATRIX_TYPE && rotationMatrix )
352 real64 m1[ 3 ] = { normal[ 2 ], 0.0, -normal[ 0 ] };
353 real64 m2[ 3 ] = { 0.0, normal[ 2 ], -normal[ 1 ] };
354 real64 const norm_m1 = LvArray::tensorOps::l2Norm< 3 >( m1 );
355 real64 const norm_m2 = LvArray::tensorOps::l2Norm< 3 >( m2 );
361 LvArray::tensorOps::crossProduct( m2, normal, m1 );
362 LvArray::tensorOps::normalize< 3 >( m2 );
363 LvArray::tensorOps::normalize< 3 >( m1 );
367 LvArray::tensorOps::crossProduct( m1, normal, m2 );
368 LvArray::tensorOps::scale< 3 >( m1, -1 );
369 LvArray::tensorOps::normalize< 3 >( m1 );
370 LvArray::tensorOps::normalize< 3 >( m2 );
374 rotationMatrix[ 0 ][ 0 ] = normal[ 0 ];
375 rotationMatrix[ 1 ][ 0 ] = normal[ 1 ];
376 rotationMatrix[ 2 ][ 0 ] = normal[ 2 ];
377 rotationMatrix[ 0 ][ 1 ] = m1[ 0 ];
378 rotationMatrix[ 1 ][ 1 ] = m1[ 1 ];
379 rotationMatrix[ 2 ][ 1 ] = m1[ 2 ];
380 rotationMatrix[ 0 ][ 2 ] = m2[ 0 ];
381 rotationMatrix[ 1 ][ 2 ] = m2[ 1 ];
382 rotationMatrix[ 2 ][ 2 ] = m2[ 2 ];
385 "Rotation matrix with determinant different from +1.0" );
394 template<
typename T >
399 return (T( 0 ) < val) - (val < T( 0 ));
415 template<
typename POINT_TYPE >
420 POINT_TYPE
const & elemCenter,
421 POINT_TYPE
const & point,
422 real64 const areaTolerance = 0.0 )
424 localIndex const numFaces = faceIndices.size();
425 R1Tensor faceCenter, faceNormal, cellToFaceVec;
431 centroid_3DPolygon( facesToNodes[faceIndex], nodeCoordinates, faceCenter, faceNormal, areaTolerance );
434 LvArray::tensorOps::copy< 3 >( cellToFaceVec, faceCenter );
435 LvArray::tensorOps::subtract< 3 >( cellToFaceVec, elemCenter );
436 if( LvArray::tensorOps::AiBi< 3 >( cellToFaceVec, faceNormal ) < 0.0 )
438 LvArray::tensorOps::scale< 3 >( faceNormal, -1 );
442 LvArray::tensorOps::subtract< 3 >( faceCenter, point );
443 int const s =
sign( LvArray::tensorOps::AiBi< 3 >( faceNormal, faceCenter ) );
464 template<
typename POLYGON_TYPE,
typename POINT_TYPE >
467 POINT_TYPE
const & point,
468 real64 const tol = 1e-10 )
472 for(
integer i = 0; i < n; ++i )
474 auto const & p1 = polygon[i];
475 auto const & p2 = polygon[(i + 1) % n];
477 real64 y1 = p1[1], y2 = p2[1];
478 real64 x1 = p1[0], x2 = p2[0];
479 real64 py = point[1], px = point[0];
482 if( py + tol < std::min( y1, y2 ) || py - tol > std::max( y1, y2 ) )
487 if( std::abs( (x2 - x1) * (py - y1) - (px - x1) * (y2 - y1) ) < tol *
488 ( std::hypot( x2 - x1, y2 - y1 ) + 1.0 ) )
491 if( px + tol >= std::min( x1, x2 ) && px - tol <= std::max( x1, x2 ) &&
492 py + tol >= std::min( y1, y2 ) && py - tol <= std::max( y1, y2 ) )
497 if( std::abs( y2 - y1 ) < tol )
501 real64 xIntersect = x1 + (py - y1) * (x2 - x1) / (y2 - y1);
504 if( px < xIntersect - tol )
508 return (count % 2) == 1;
521 template<
typename POLYGON_TYPE,
typename POINT_TYPE >
524 POINT_TYPE
const & point,
525 real64 const tol = 1e-10 )
528 auto const & p0 = polygon[0];
529 POINT_TYPE normal = {0, 0, 0};
530 for(
integer i = 1; i < n - 1; i++ )
532 auto const & p1 = polygon[i];
533 auto const & p2 = polygon[i + 1];
534 normal[0] += (p1[1] - p0[1]) * (p2[2] - p0[2]) - (p1[2] - p0[2]) * (p2[1] - p0[1]);
535 normal[1] += (p1[2] - p0[2]) * (p2[0] - p0[0]) - (p1[0] - p0[0]) * (p2[2] - p0[2]);
536 normal[2] += (p1[0] - p0[0]) * (p2[1] - p0[1]) - (p1[1] - p0[1]) * (p2[0] - p0[0]);
539 real64 const dist = normal[0] * point[0] + normal[1] * point[1] + normal[2] * point[2] -(normal[0] * p0[0] + normal[1] * p0[1] + normal[2] * p0[2]);
541 if( std::abs( dist ) > tol )
547 int dominantIndex = 0;
548 if( std::abs( normal[1] ) > std::abs( normal[0] ))
552 if( std::abs( normal[2] ) > std::abs( normal[dominantIndex] ))
558 POLYGON_TYPE projectedPolygon( n );
559 POINT_TYPE projectedPoint;
560 if( dominantIndex == 0 )
562 for(
int i = 0; i < n; i++ )
564 projectedPolygon[i][0] = polygon[i][1];
565 projectedPolygon[i][1] = polygon[i][2];
567 projectedPoint[0] = point[1];
568 projectedPoint[1] = point[2];
570 else if( dominantIndex == 1 )
572 for(
int i = 0; i < n; i++ )
574 projectedPolygon[i][0] = polygon[i][0];
575 projectedPolygon[i][1] = polygon[i][2];
577 projectedPoint[0] = point[0];
578 projectedPoint[1] = point[2];
582 for(
int i = 0; i < n; i++ )
584 projectedPolygon[i][0] = polygon[i][0];
585 projectedPolygon[i][1] = polygon[i][1];
587 projectedPoint[0] = point[0];
588 projectedPoint[1] = point[1];
606 template<
typename COORD_TYPE,
typename POINT_TYPE >
609 COORD_TYPE
const bx, COORD_TYPE
const by, COORD_TYPE
const bz )
641 template<
typename COORD_TYPE,
typename POINT_TYPE >
644 COORD_TYPE
const e1x, COORD_TYPE
const e1y, COORD_TYPE
const e1z,
645 COORD_TYPE
const e2x, COORD_TYPE
const e2y, COORD_TYPE
const e2z )
648 ( e1z - az ) * ( e2x - ax ),
649 ( e1z - az ) * ( e2y - ay ),
650 ( e1x - ax ) * ( e2y - ay ),
651 ( e1x - ax ) * ( e2z - az ),
652 ( e1y - ay ) * ( e2z - az ) );
673 template<
typename COORD_TYPE,
typename POINT_TYPE >
676 COORD_TYPE
const t1x, COORD_TYPE
const t1y, COORD_TYPE
const t1z,
677 COORD_TYPE
const t2x, COORD_TYPE
const t2y, COORD_TYPE
const t2z,
678 COORD_TYPE
const t3x, COORD_TYPE
const t3y, COORD_TYPE
const t3z )
680 COORD_TYPE v1x = t1x - ax;
681 COORD_TYPE v1y = t1y - ay;
682 COORD_TYPE v1z = t1z - az;
683 COORD_TYPE v2x = t2x - ax;
684 COORD_TYPE v2y = t2y - ay;
685 COORD_TYPE v2z = t2z - az;
686 COORD_TYPE v3x = t3x - ax;
687 COORD_TYPE v3y = t3y - ay;
688 COORD_TYPE v3z = t3z - az;
689 COORD_TYPE
sign = ( v1x * v2y - v1y * v2x ) * v3z +
690 ( v2x * v3y - v2y * v3x ) * v1z +
691 ( v3x * v1y - v3y * v1x ) * v2z;
705 template<
typename ... LIST_TYPE >
711 globalIndex minElementGID = LvArray::NumericLimits< globalIndex >::max;
712 for(
int i = 0; i < nodeElements.size(); i++ )
715 if( elementGlobalIndex[ e ] < minElementGID )
717 minElementGID = elementGlobalIndex[ e ];
732 template<
typename ... LIST_TYPE >
739 globalIndex minElementGID = LvArray::NumericLimits< globalIndex >::max;
740 for(
int i = 0; i < nodeElements1.size(); i++ )
743 for(
int j = 0; j < nodeElements2.size(); j++ )
748 if( elementGlobalIndex[ e1 ] < minElementGID )
750 minElementGID = elementGlobalIndex[ e1 ];
768 template<
typename ... LIST_TYPE >
776 globalIndex minElementGID = LvArray::NumericLimits< globalIndex >::max;
777 for(
int i = 0; i < nodeElements1.size(); i++ )
780 for(
int j = 0; j < nodeElements2.size(); j++ )
783 for(
int k = 0; k < nodeElements3.size(); k++ )
786 if( e1 == e2 && e2 == e3 )
788 if( elementGlobalIndex[ e1 ] < minElementGID )
790 minElementGID = elementGlobalIndex[ e1 ];
814 template<
typename COORD_TYPE,
typename POINT_TYPE >
823 POINT_TYPE
const & elemCenter,
824 POINT_TYPE
const & point )
827 localIndex const numFaces = faceIndices.size();
834 globalIndex minGlobalId = LvArray::NumericLimits< globalIndex >::max;
836 localIndex numFaceVertices = facesToNodes[faceIndex].size();
837 for(
localIndex v = 0; v < numFaceVertices; v++ )
839 localIndex vIndex = facesToNodes( faceIndex, v );
840 globalIndex globalId = nodeLocalToGlobal[ vIndex ];
841 if( globalId < minGlobalId )
843 minGlobalId = globalId;
849 for(
localIndex v = 0; v < numFaceVertices; v++ )
851 vi[ 1 ] = facesToNodes( faceIndex, v );
852 vi[ 2 ] = facesToNodes( faceIndex, (v + 1) % numFaceVertices );
853 if( vi[ 1 ] != minVertex && vi[ 2 ] != minVertex )
856 if( nodeLocalToGlobal[ vi[ 1 ] ] > nodeLocalToGlobal[ vi[ 2 ] ] )
862 COORD_TYPE v1x = nodeCoordinates( vi[ 0 ], 0 );
863 COORD_TYPE v1y = nodeCoordinates( vi[ 0 ], 1 );
864 COORD_TYPE v1z = nodeCoordinates( vi[ 0 ], 2 );
865 COORD_TYPE v2x = nodeCoordinates( vi[ 1 ], 0 );
866 COORD_TYPE v2y = nodeCoordinates( vi[ 1 ], 1 );
867 COORD_TYPE v2z = nodeCoordinates( vi[ 1 ], 2 );
868 COORD_TYPE v3x = nodeCoordinates( vi[ 2 ], 0 );
869 COORD_TYPE v3y = nodeCoordinates( vi[ 2 ], 1 );
870 COORD_TYPE v3z = nodeCoordinates( vi[ 2 ], 2 );
872 R1Tensor vv1 = { v2x - v1x, v2y - v1y, v2z - v1z };
873 R1Tensor vv2 = { v3x - v1x, v3y - v1y, v3z - v1z };
874 R1Tensor dist = { elemCenter[ 0 ] - ( v1x + v2x + v3x )/3.0,
875 elemCenter[ 1 ] - ( v1y + v2y + v3y )/3.0,
876 elemCenter[ 2 ] - ( v1z + v2z + v3z )/3.0 };
878 LvArray::tensorOps::crossProduct( norm, vv1, vv2 );
880 int sign = LvArray::tensorOps::AiBi< 3 >( norm, dist ) > 0 ? -1 : +1;
906 return findEdgeRefElement( nodesToElements[ vi[ 0 ] ], nodesToElements[ vi[ 1 ] ], elementLocalToGlobal ) == element;
908 facecmp +=
sign * edgecmp;
917 return findEdgeRefElement( nodesToElements[ vi[ 1 ] ], nodesToElements[ vi[ 2 ] ], elementLocalToGlobal ) == element;
919 facecmp +=
sign * edgecmp;
928 return findEdgeRefElement( nodesToElements[ vi[ 0 ] ], nodesToElements[ vi[ 2 ] ], elementLocalToGlobal ) == element;
930 facecmp +=
sign * edgecmp;
942 return findTriangleRefElement( nodesToElements[ vi[ 0 ] ], nodesToElements[ vi[ 1 ] ], nodesToElements[ vi[ 2 ] ], elementLocalToGlobal ) == element;
944 omega +=
sign * facecmp;
975 template<
typename COORD_TYPE,
typename POINT_TYPE >
984 POINT_TYPE
const & elemCenter,
985 POINT_TYPE
const & point )
987 return computeWindingNumber( element, nodeCoordinates, elementsToFaces, facesToNodes, nodesToElements, nodeLocalToGlobal, elementLocalToGlobal, elemCenter, point ) > 0;
999 template<
typename NODE_MAP_TYPE,
typename VEC_TYPE >
1002 NODE_MAP_TYPE
const & pointIndices,
1004 VEC_TYPE && boxDims )
1007 R1Tensor minCoords = { LvArray::NumericLimits< real64 >::max,
1008 LvArray::NumericLimits< real64 >::max,
1009 LvArray::NumericLimits< real64 >::max };
1012 LvArray::tensorOps::fill< 3 >( boxDims, LvArray::NumericLimits< real64 >::lowest );
1015 for(
localIndex a = 0; a < pointIndices[elemIndex].size(); ++a )
1017 localIndex const id = pointIndices( elemIndex, a );
1020 minCoords[ d ] = fmin( minCoords[ d ], pointCoordinates(
id, d ) );
1021 boxDims[ d ] = fmax( boxDims[ d ], pointCoordinates(
id, d ) );
1025 LvArray::tensorOps::subtract< 3 >( boxDims, minCoords );
1034 template<
typename FE_TYPE >
1039 for(
localIndex q=0; q<FE_TYPE::numQuadraturePoints; ++q )
1041 result = result + FE_TYPE::transformedQuadratureWeight( q, X );
1055 return elementVolume< finiteElement::H1_Hexahedron_Lagrange1_GaussLegendre2 >( X );
1067 return elementVolume< finiteElement::H1_Tetrahedron_Lagrange1_Gauss1 >( X );
1079 return elementVolume< finiteElement::H1_Wedge_Lagrange1_Gauss6 >( X );
1091 return elementVolume< finiteElement::H1_Pyramid_Lagrange1_Gauss5 >( X );
1104 template<
integer N >
1109 static_assert( N > 4,
1110 "Function prismVolume can be called for a prism with N-sided polygon base where N > 5." );
1117 for(
integer a = 0; a < N; ++a )
1119 LvArray::tensorOps::add< 3 >( XGBot, X[a] );
1121 for(
integer a = N; a < 2 * N; ++a )
1123 LvArray::tensorOps::add< 3 >( XGTop, X[a] );
1125 LvArray::tensorOps::scale< 3 >( XGBot, 1.0 / N );
1126 LvArray::tensorOps::scale< 3 >( XGTop, 1.0 / N );
1129 for(
int a = 0; a < N - 1; ++a )
1132 LvArray::tensorOps::copy< 3 >( XWedge[0], X[a] );
1133 LvArray::tensorOps::copy< 3 >( XWedge[1], X[a+N] );
1134 LvArray::tensorOps::copy< 3 >( XWedge[2], X[a+1] );
1135 LvArray::tensorOps::copy< 3 >( XWedge[3], X[a+1+N] );
1136 LvArray::tensorOps::copy< 3 >( XWedge[4], XGBot );
1137 LvArray::tensorOps::copy< 3 >( XWedge[5], XGTop );
1138 result = result + computationalGeometry::elementVolume< finiteElement::H1_Wedge_Lagrange1_Gauss6 >( XWedge );
1140 LvArray::tensorOps::copy< 3 >( XWedge[0], X[N-1] );
1141 LvArray::tensorOps::copy< 3 >( XWedge[1], X[2*N-1] );
1142 LvArray::tensorOps::copy< 3 >( XWedge[2], X[0] );
1143 LvArray::tensorOps::copy< 3 >( XWedge[3], X[N] );
1144 LvArray::tensorOps::copy< 3 >( XWedge[4], XGBot );
1145 LvArray::tensorOps::copy< 3 >( XWedge[5], XGTop );
1146 result = result + computationalGeometry::elementVolume< finiteElement::H1_Wedge_Lagrange1_Gauss6 >( XWedge );
GEOS_HOST_DEVICE int lexicographicalCompareTriangle(POINT_TYPE const ax, POINT_TYPE const ay, POINT_TYPE const az, COORD_TYPE const t1x, COORD_TYPE const t1y, COORD_TYPE const t1z, COORD_TYPE const t2x, COORD_TYPE const t2y, COORD_TYPE const t2z, COORD_TYPE const t3x, COORD_TYPE const t3y, COORD_TYPE const t3z)
Method to perform lexicographic comparison of a node and a triangle based on coordinates.
GEOS_HOST_DEVICE void getBoundingBox(localIndex const elemIndex, NODE_MAP_TYPE const &pointIndices, arrayView2d< real64 const, nodes::REFERENCE_POSITION_USD > const &pointCoordinates, VEC_TYPE &&boxDims)
Compute the dimensions of the bounding box containing the element defined here by the coordinates of ...
GEOS_HOST_DEVICE bool isPointInsideConvexPolyhedronRobust(localIndex element, arrayView2d< COORD_TYPE const, nodes::REFERENCE_POSITION_USD > const &nodeCoordinates, arrayView2d< localIndex const > const &elementsToFaces, ArrayOfArraysView< localIndex const > const &facesToNodes, ArrayOfArraysView< localIndex const > const &nodesToElements, arrayView1d< globalIndex const > const &nodeLocalToGlobal, arrayView1d< globalIndex const > const &elementLocalToGlobal, POINT_TYPE const &elemCenter, POINT_TYPE const &point)
Check if a point is inside a convex polyhedron (3D polygon), using a robust method to avoid ambiguity...
GEOS_HOST_DEVICE void FixNormalOrientation_3D(NORMAL_TYPE &&normal)
Change the orientation of the input vector to be consistent in a global sense.
bool isPointInPolygon3d(POLYGON_TYPE const &polygon, integer const n, POINT_TYPE const &point, real64 const tol=1e-10)
Check if a point is inside a polygon (3D version)
GEOS_HOST_DEVICE real64 prismVolume(real64 const (&X)[2 *N][3])
Compute the volume of a prism with N-sided polygon base.
GEOS_HOST_DEVICE bool isPointInsidePolyhedron(arrayView2d< real64 const, nodes::REFERENCE_POSITION_USD > const &nodeCoordinates, arraySlice1d< localIndex const > const &faceIndices, ArrayOfArraysView< localIndex const > const &facesToNodes, POINT_TYPE const &elemCenter, POINT_TYPE const &point, real64 const areaTolerance=0.0)
Check if a point is inside a convex polyhedron (3D polygon)
GEOS_HOST_DEVICE GEOS_FORCE_INLINE real64 centroid_3DPolygon(arraySlice1d< localIndex const > const pointsIndices, arrayView2d< real64 const, nodes::REFERENCE_POSITION_USD > const &points, CENTER_TYPE &¢er, NORMAL_TYPE &&normal, real64 const areaTolerance=0.0)
Calculate the centroid of a convex 3D polygon as well as the normal.
GEOS_HOST_DEVICE void RotationMatrix_3D(NORMAL_TYPE const &normal, MATRIX_TYPE &&rotationMatrix)
Calculate the rotation matrix for a face in the 3D space.
GEOS_HOST_DEVICE bool computeWindingNumber(localIndex element, arrayView2d< COORD_TYPE const, nodes::REFERENCE_POSITION_USD > const &nodeCoordinates, arrayView2d< localIndex const > const &elementsToFaces, ArrayOfArraysView< localIndex const > const &facesToNodes, ArrayOfArraysView< localIndex const > const &nodesToElements, arrayView1d< globalIndex const > const &nodeLocalToGlobal, arrayView1d< globalIndex const > const &elementLocalToGlobal, POINT_TYPE const &elemCenter, POINT_TYPE const &point)
Computes the winding number of a point with respect to a mesh element.
GEOS_HOST_DEVICE int findVertexRefElement(arraySlice1d< localIndex const > const &nodeElements, arrayView1d< globalIndex const > const &elementGlobalIndex)
Method to find the reference element touching a vertex. The element with the lowest global ID is chos...
bool isPointInPolygon2d(POLYGON_TYPE const &polygon, integer n, POINT_TYPE const &point, real64 const tol=1e-10)
Check if a point is inside a polygon (2D version)
constexpr real64 machinePrecision
Machine epsilon for double-precision calculations.
GEOS_HOST_DEVICE real64 elementVolume(real64 const (&X)[FE_TYPE::numNodes][3])
Compute the volume of an element (tetrahedron, pyramid, wedge, hexahedron)
GEOS_HOST_DEVICE int findTriangleRefElement(arraySlice1d< localIndex const > const &nodeElements1, arraySlice1d< localIndex const > const &nodeElements2, arraySlice1d< localIndex const > const &nodeElements3, arrayView1d< globalIndex const > const &elementGlobalIndex)
Method to find the reference element for a triangle. The element with the lowest global ID is chosen ...
GEOS_HOST_DEVICE real64 hexahedronVolume(real64 const (&X)[8][3])
Compute the volume of an hexahedron.
array1d< int > orderPointsCCW(arrayView2d< real64 > const &points, NORMAL_TYPE const &normal)
Reorder a set of points counter-clockwise.
GEOS_HOST_DEVICE int lexicographicalCompareVertex(POINT_TYPE const ax, POINT_TYPE const ay, POINT_TYPE const az, COORD_TYPE const bx, COORD_TYPE const by, COORD_TYPE const bz)
Method to perform lexicographic comparison of two nodes based on coordinates.
GEOS_HOST_DEVICE real64 tetrahedronVolume(real64 const (&X)[4][3])
Compute the volume of an tetrahedron.
void LinePlaneIntersection(LINEDIR_TYPE const &lineDir, POINT_TYPE const &linePoint, NORMAL_TYPE const &planeNormal, ORIGIN_TYPE const &planeOrigin, INTPOINT_TYPE &intersectionPoint)
Calculate the intersection between a line and a plane.
GEOS_HOST_DEVICE int findEdgeRefElement(arraySlice1d< localIndex const > const &nodeElements1, arraySlice1d< localIndex const > const &nodeElements2, arrayView1d< globalIndex const > const &elementGlobalIndex)
Method to find the reference element for an edge. The element with the lowest global ID is chosen fro...
real64 ComputeSurfaceArea(arrayView2d< real64 const > const &points, NORMAL_TYPE const &&normal)
Calculate the area of a polygon given the set of points in ccw order defining it.
GEOS_HOST_DEVICE real64 wedgeVolume(real64 const (&X)[6][3])
Compute the volume of a wedge.
GEOS_HOST_DEVICE real64 pyramidVolume(real64 const (&X)[5][3])
Compute the volume of a pyramid.
GEOS_HOST_DEVICE GEOS_FORCE_INLINE real64 computeDiameter(POINT_COORDS_TYPE points, localIndex const &numPoints)
Calculate the diameter of a set of points in a given dimension.
GEOS_HOST_DEVICE GEOS_FORCE_INLINE int sign(T const val)
Return the sign of a given value as an integer.
GEOS_HOST_DEVICE int lexicographicalCompareEdge(POINT_TYPE const ax, POINT_TYPE const ay, POINT_TYPE const az, COORD_TYPE const e1x, COORD_TYPE const e1y, COORD_TYPE const e1z, COORD_TYPE const e2x, COORD_TYPE const e2y, COORD_TYPE const e2z)
Method to perform lexicographic comparison of a node and an edge based on coordinates.
#define GEOS_HOST_DEVICE
Marks a host-device function.
#define GEOS_FORCE_INLINE
Marks a function or lambda for inlining.
#define GEOS_ERROR(...)
Raise a hard error and terminate the program.
#define GEOS_LOG_RANK(msg)
Log a message to the rank output stream.
#define GEOS_ERROR_IF_LT(lhs, rhs)
Raise a hard error if one value compares less than the other.
#define GEOS_ERROR_IF(COND,...)
Conditionally raise a hard error and terminate the program.
ArrayView< T, 1 > arrayView1d
Alias for 1D array view.
Array< T, 2, PERMUTATION > array2d
Alias for 2D array.
GEOS_GLOBALINDEX_TYPE globalIndex
Global index type (for indexing objects across MPI partitions).
LvArray::ArrayOfArraysView< T, INDEX_TYPE const, CONST_SIZES, LvArray::ChaiBuffer > ArrayOfArraysView
View of array of variable-sized arrays. See LvArray::ArrayOfArraysView for details.
double real64
64-bit floating point type.
GEOS_LOCALINDEX_TYPE localIndex
Local index type (for indexing objects within an MPI partition).
ArraySlice< T, 1, USD > arraySlice1d
Alias for 1D array slice.
ArrayView< T, 2, USD > arrayView2d
Alias for 2D array view.
int integer
Signed integer type.
Array< T, 1 > array1d
Alias for 1D array.