2024-12-23 16:30:28 +05:00
# include "boundarylayer.hpp"
# include <core/array.hpp>
namespace netgen
{
struct Intersection_
{
bool is_intersecting = false ;
double lam0 = - 1 , lam1 = - 1 ;
Point < 3 > p ;
double bary [ 3 ] ;
operator bool ( ) const { return is_intersecting ; }
} ;
struct GrowthVectorLimiter
{
typedef std : : array < Point < 3 > , 2 > Seg ;
typedef std : : array < Point < 3 > , 3 > Trig ;
BoundaryLayerTool & tool ;
const BoundaryLayerParameters & params ;
Mesh & mesh ;
double height ;
Array < double , PointIndex > limits ;
FlatArray < Vec < 3 > , PointIndex > growthvectors ;
BitArray changed_domains ;
unique_ptr < BoxTree < 3 > > tree ;
Array < PointIndex , PointIndex > map_from ;
Table < SurfaceElementIndex , PointIndex > p2sel ;
GrowthVectorLimiter ( BoundaryLayerTool & tool_ )
: tool ( tool_ ) , params ( tool_ . params ) , mesh ( tool_ . mesh ) , height ( tool_ . total_height ) , growthvectors ( tool_ . growthvectors ) , map_from ( mesh . Points ( ) . Size ( ) )
{
changed_domains = tool . domains ;
if ( ! params . outside )
changed_domains . Invert ( ) ;
map_from = tool . mapfrom ;
p2sel = ngcore : : CreateSortedTable < SurfaceElementIndex , PointIndex > (
tool . new_sels . Range ( ) ,
[ & ] ( auto & table , SurfaceElementIndex ei ) {
for ( PointIndex pi : tool . new_sels [ ei ] . PNums ( ) )
table . Add ( pi , ei ) ;
} ,
mesh . GetNP ( ) ) ;
}
auto SurfaceElementsRange ( ) { return Range ( tool . nse + tool . new_sels . Size ( ) ) ; }
void WriteErrorMesh ( string name )
{
if ( ! debugparam . write_mesh_on_error )
return ;
Mesh out_mesh ;
out_mesh = mesh ;
for ( auto [ pi , data ] : tool . growth_vector_map )
{
auto [ gw , height ] = data ;
out_mesh [ pi ] + = limits [ pi ] * height * ( * gw ) ;
}
out_mesh . Save ( name ) ;
}
const auto & Get ( SurfaceElementIndex sei )
{
if ( sei < tool . nse )
return mesh [ sei ] ;
return tool . new_sels [ sei - tool . nse ] ;
}
std : : pair < double , double > GetMinMaxLimit ( SurfaceElementIndex sei )
{
const auto & sel = Get ( sei ) ;
double min_limit = GetLimit ( sel [ 0 ] ) ;
double max_limit = min_limit ;
for ( auto i : IntRange ( 1 , sel . GetNP ( ) ) )
{
auto limit = GetLimit ( sel [ i ] ) ;
min_limit = min ( min_limit , limit ) ;
max_limit = max ( max_limit , limit ) ;
}
return { min_limit , max_limit } ;
}
double GetLimit ( PointIndex pi )
{
2024-12-29 02:58:44 +05:00
// if (pi <= tool.np)
if ( pi < tool . npi )
2024-12-23 16:30:28 +05:00
return limits [ pi ] ;
return limits [ map_from [ pi ] ] ;
}
bool SetLimit ( PointIndex pi , double new_limit )
{
2024-12-29 02:58:44 +05:00
// double& limit = (pi <= tool.np) ? limits[pi] : limits[map_from[pi]];
double & limit = ( pi < tool . npi ) ? limits [ pi ] : limits [ map_from [ pi ] ] ;
2024-12-23 16:30:28 +05:00
if ( limit < = new_limit )
return false ;
limit = new_limit ;
return true ;
}
bool ScaleLimit ( PointIndex pi , double factor )
{
2024-12-29 02:58:44 +05:00
// double& limit = (pi <= tool.np) ? limits[pi] : limits[map_from[pi]];
double & limit = ( pi < tool . npi ) ? limits [ pi ] : limits [ map_from [ pi ] ] ;
2024-12-23 16:30:28 +05:00
return SetLimit ( pi , limit * factor ) ;
}
Vec < 3 > GetVector ( PointIndex pi_to , double shift = 1. , bool apply_limit = false )
{
auto [ gw , height ] = tool . growth_vector_map [ pi_to ] ;
if ( apply_limit )
shift * = GetLimit ( pi_to ) ;
return shift * height * ( * gw ) ;
}
Point < 3 > GetPoint ( PointIndex pi_to , double shift = 1. , bool apply_limit = false )
{
2024-12-29 02:58:44 +05:00
// if (pi_to <= tool.np || tool.growth_vector_map.count(pi_to) == 0)
if ( pi_to < tool . npi | | tool . growth_vector_map . count ( pi_to ) = = 0 )
2024-12-23 16:30:28 +05:00
return mesh [ pi_to ] ;
return mesh [ pi_to ] + GetVector ( pi_to , shift , apply_limit ) ;
}
Point < 3 > GetMappedPoint ( PointIndex pi_from , double shift = 1. , bool apply_limit = false )
{
auto pi_to = tool . mapto [ pi_from ] . Last ( ) ;
return GetPoint ( pi_to , shift , apply_limit ) ;
}
Seg GetMappedSeg ( PointIndex pi_from , double shift = 1. )
{
return { mesh [ pi_from ] , GetMappedPoint ( pi_from , shift ) } ;
}
Seg GetSeg ( PointIndex pi_to , double shift = 1. , bool apply_limit = false )
{
return { GetPoint ( pi_to , 0 ) , GetPoint ( pi_to , shift , apply_limit ) } ;
}
Trig GetTrig ( SurfaceElementIndex sei , double shift = 0.0 , bool apply_limit = false )
{
auto sel = Get ( sei ) ;
Trig trig ;
for ( auto i : Range ( 3 ) )
trig [ i ] = GetPoint ( sel [ i ] , shift , apply_limit ) ;
return trig ;
}
Trig GetMappedTrig ( SurfaceElementIndex sei , double shift = 0.0 )
{
auto sel = Get ( sei ) ;
Trig trig ;
for ( auto i : Range ( 3 ) )
trig [ i ] = GetMappedPoint ( sel [ i ] , shift ) ;
return trig ;
}
Trig GetSideTrig ( SurfaceElementIndex sei , int index , double shift = 0.0 , bool grow_first_vertex = true )
{
auto trig = GetMappedTrig ( sei , 0.0 ) ;
auto sel = Get ( sei ) ;
auto index1 = ( index + 1 ) % 3 ;
if ( ! grow_first_vertex )
index1 = ( index + 2 ) % 3 ;
trig [ index ] = GetMappedPoint ( sel [ index1 ] , shift , true ) ;
return trig ;
}
array < Trig , 4 > GetSideTrigs ( SurfaceElementIndex sei , int i0 , double shift = 0.0 )
{
auto trig = GetMappedTrig ( sei , 0.0 ) ;
array < Trig , 4 > trigs { trig , trig , trig , trig } ;
auto sel = Get ( sei ) ;
auto i1 = ( i0 + 1 ) % 3 ;
auto i2 = ( i0 + 2 ) % 3 ;
auto p1 = GetMappedPoint ( sel [ i1 ] , shift , true ) ;
auto p2 = GetMappedPoint ( sel [ i2 ] , shift , true ) ;
// create four trigs to span the quad from i1,i2 and their shifted points
// i1, i2, shifted i1
trigs [ 0 ] [ i0 ] = p1 ;
// i1, i2, shifted i2
trigs [ 1 ] [ i0 ] = p2 ;
// i1, shifted i1, shifted i2
trigs [ 2 ] [ i0 ] = p1 ;
trigs [ 2 ] [ i2 ] = p2 ;
// i2, shifted i1, shifted i2
trigs [ 2 ] [ i0 ] = p2 ;
trigs [ 2 ] [ i1 ] = p1 ;
return trigs ;
}
static constexpr double INTERSECTION_SAFETY = .9 ;
bool LimitGrowthVector ( PointIndex pi_to , SurfaceElementIndex sei , double trig_shift , double seg_shift , bool check_prism_sides = false )
{
auto pi_from = map_from [ pi_to ] ;
if ( ! pi_from . IsValid ( ) )
return false ;
for ( auto pi : Get ( sei ) . PNums ( ) )
{
if ( pi = = pi_from )
return false ;
if ( map_from [ pi ] = = pi_from )
return false ;
}
if ( check_prism_sides | | trig_shift > .0 )
{
auto [ trig_min_limit , trig_max_limit ] = GetMinMaxLimit ( sei ) ;
if ( GetLimit ( pi_to ) < trig_min_limit )
return false ;
auto getTrigs = [ & ] ( double scaling = 1.0 ) - > ArrayMem < Trig , 3 > {
ArrayMem < Trig , 12 > trigs ;
if ( check_prism_sides )
for ( auto i : Range ( 3 ) )
for ( auto trig : GetSideTrigs ( sei , i , scaling * trig_shift ) )
trigs . Append ( trig ) ;
else
trigs . Append ( GetTrig ( sei , scaling * trig_shift , true ) ) ;
return trigs ;
} ;
if ( ! check_prism_sides )
{
// If the growth vectors of all points are pointing in the same direction,
// an intersection means, we also have an intersection with a prism side face
// this is an extra check and handled later
auto seg = GetSeg ( pi_to , 1.0 , false ) ;
auto gw = seg [ 1 ] - seg [ 0 ] ;
bool have_same_growth_direction = true ;
for ( auto pi : Get ( sei ) . PNums ( ) )
{
auto p_seg = GetSeg ( pi , 1.0 , false ) ;
auto p_gw = p_seg [ 1 ] - p_seg [ 0 ] ;
have_same_growth_direction & = ( gw * p_gw ) > 0 ;
}
if ( have_same_growth_direction )
return false ;
}
double scaling = 1.0 ;
while ( true )
{
bool have_intersection = false ;
auto seg = GetSeg ( pi_to , scaling * seg_shift , true ) ;
for ( auto trig : getTrigs ( scaling ) )
have_intersection | = isIntersectingTrig ( seg , trig ) ;
if ( ! have_intersection )
break ;
scaling * = 0.9 ;
}
if ( scaling = = 1.0 )
return false ;
double new_limit = scaling * max ( GetLimit ( pi_to ) , trig_max_limit ) ;
SetLimit ( pi_to , new_limit ) ;
for ( auto pi : Get ( sei ) . PNums ( ) )
SetLimit ( pi , new_limit ) ;
return true ;
}
else
{
auto seg = GetSeg ( pi_to , seg_shift , false ) ;
auto trig = GetTrig ( sei , 0.0 ) ;
auto intersection = isIntersectingTrig ( seg , trig ) ;
// checking with original surface elements -> allow only half the distance
auto new_seg_limit = 0.40 * intersection . lam0 * seg_shift ;
if ( intersection & & new_seg_limit < GetLimit ( pi_from ) )
return SetLimit ( pi_from , new_seg_limit ) ;
return false ;
}
}
void EqualizeLimits ( double factor = .5 )
{
static Timer t ( " GrowthVectorLimiter::EqualizeLimits " ) ;
PrintMessage ( 5 , " GrowthVectorLimiter - equalize limits " ) ;
RegionTimer reg ( t ) ;
if ( factor = = 0.0 )
return ;
2024-12-29 02:58:44 +05:00
// for (PointIndex pi : IntRange(tool.np, mesh.GetNP()))
for ( PointIndex pi : mesh . Points ( ) . Range ( ) . Modify ( tool . np , 0 ) )
2024-12-23 16:30:28 +05:00
{
2024-12-29 01:26:05 +05:00
// auto pi_from = map_from[pi];
2024-12-23 16:30:28 +05:00
std : : set < PointIndex > pis ;
for ( auto sei : p2sel [ pi ] )
for ( auto pi_ : tool . new_sels [ sei ] . PNums ( ) )
pis . insert ( pi_ ) ;
ArrayMem < double , 20 > limits ;
for ( auto pi1 : pis )
{
auto limit = GetLimit ( pi1 ) ;
if ( limit > 0.0 )
limits . Append ( GetLimit ( pi1 ) ) ;
}
if ( limits . Size ( ) = = 0 )
continue ;
double average = 0.0 ;
for ( auto l : limits )
average + = l ;
average / = limits . Size ( ) ;
SetLimit ( pi , factor * average + ( 1.0 - factor ) * GetLimit ( pi ) ) ;
}
}
void LimitSelfIntersection ( double safety = 1.4 )
{
static Timer t ( " GrowthVectorLimiter::LimitSelfIntersection " ) ;
PrintMessage ( 5 , " GrowthVectorLimiter - self intersection " ) ;
RegionTimer reg ( t ) ;
// check for self-intersection within new elements (prisms/hexes)
auto isIntersecting = [ & ] ( SurfaceElementIndex sei , double shift ) {
// checks if surface element is self intersecting when growing with factor
// shift
// ignore new surface elements, side trigs are only built
// from original surface elements
if ( sei > = tool . nse )
return false ;
const auto sel = Get ( sei ) ;
auto np = sel . GetNP ( ) ;
for ( auto i : Range ( np ) )
{
2024-12-29 02:58:44 +05:00
// if (sel[i] > tool.np)
if ( sel [ i ] > = tool . npi )
2024-12-23 16:30:28 +05:00
return false ;
if ( tool . mapto [ sel [ i ] ] . Size ( ) = = 0 )
return false ;
}
for ( auto i : Range ( np ) )
{
auto seg = GetMappedSeg ( sel [ i ] , shift * limits [ sel [ i ] ] ) ;
for ( auto fi : Range ( np - 2 ) )
{
for ( auto side : { true , false } )
{
auto trig = GetSideTrig ( sei , i + fi , 1.0 , side ) ;
if ( isIntersectingPlane ( seg , trig ) )
return true ;
}
}
}
return false ;
} ;
for ( SurfaceElementIndex sei : mesh . SurfaceElements ( ) . Range ( ) )
{
auto sel = mesh [ sei ] ;
if ( ! tool . moved_surfaces [ sel . GetIndex ( ) ] )
continue ;
if ( sel . GetNP ( ) = = 4 )
continue ;
2024-12-29 01:26:05 +05:00
// const auto& fd = mesh.GetFaceDescriptor(sel.GetIndex());
2024-12-23 16:30:28 +05:00
auto np = sel . GetNP ( ) ;
double shift = 1.0 ;
const double step_factor = 0.9 ;
while ( isIntersecting ( sei , shift * safety ) )
{
shift * = step_factor ;
double max_limit = 0 ;
for ( auto i : Range ( np ) )
max_limit = max ( max_limit , limits [ sel [ i ] ] ) ;
for ( auto i : Range ( np ) )
if ( max_limit = = limits [ sel [ i ] ] )
ScaleLimit ( sel [ i ] , step_factor ) ;
// if (max_limit < 0.01) break;
}
}
}
// checks if a segment is intersecting a plane, spanned by three points, lam
// will be set s.t. p_intersect = seg[0] + lam * (seg[1]-seg[0])
Intersection_ isIntersectingPlane ( const Seg & seg ,
const Trig & trig )
{
auto t1 = trig [ 1 ] - trig [ 0 ] ;
auto t2 = trig [ 2 ] - trig [ 0 ] ;
auto n = Cross ( t1 , t2 ) ;
auto v0n = ( seg [ 0 ] - trig [ 0 ] ) * n ;
auto v1n = ( seg [ 1 ] - trig [ 0 ] ) * n ;
Intersection_ intersection ;
intersection . lam0 = - v0n / ( v1n - v0n ) ;
intersection . p = seg [ 0 ] + intersection . lam0 * ( seg [ 1 ] - seg [ 0 ] ) ;
intersection . is_intersecting = ( v0n * v1n < 0 ) & & ( intersection . lam0 > - 1e-8 ) & & ( intersection . lam0 < 1 + 1e-8 ) ;
return intersection ;
}
Intersection_ isIntersectingTrig ( const Seg & seg , const Trig & trig )
{
auto intersection = isIntersectingPlane ( seg , trig ) ;
if ( ! intersection )
return intersection ;
auto p = seg [ 0 ] + intersection . lam0 * ( seg [ 1 ] - seg [ 0 ] ) - trig [ 0 ] ;
Vec3d col1 = trig [ 1 ] - trig [ 0 ] ;
Vec3d col2 = trig [ 2 ] - trig [ 0 ] ;
Vec3d col3 = Cross ( col1 , col2 ) ;
Vec3d rhs = p ;
Vec3d bary ;
SolveLinearSystem ( col1 , col2 , col3 , rhs , bary ) ;
intersection . lam1 = 0 ;
double eps = 1e-4 ;
if ( bary . X ( ) > = - eps & & bary . Y ( ) > = - eps & & bary . X ( ) + bary . Y ( ) < = 1 + eps )
{
intersection . bary [ 0 ] = bary . X ( ) ;
intersection . bary [ 1 ] = bary . Y ( ) ;
intersection . bary [ 2 ] = 1.0 - bary . X ( ) - bary . Y ( ) ;
}
else
intersection . is_intersecting = false ;
return intersection ;
}
Intersection_ isIntersectingTrig ( PointIndex pi_from , PointIndex pi_to , SurfaceElementIndex sei , double shift = 0.0 )
{
2024-12-29 02:58:44 +05:00
// JS: where is that GetSeg function ?
2024-12-23 16:30:28 +05:00
return isIntersectingTrig ( GetSeg ( pi_from , pi_to ) , GetTrig ( sei , shift ) ) ;
}
void BuildSearchTree ( double trig_shift )
{
static Timer t ( " BuildSearchTree " ) ;
RegionTimer rt ( t ) ;
Box < 3 > bbox ( Box < 3 > : : EMPTY_BOX ) ;
for ( PointIndex pi : mesh . Points ( ) . Range ( ) )
{
bbox . Add ( mesh [ pi ] ) ;
bbox . Add ( GetPoint ( pi , 1.1 ) ) ;
}
tree = make_unique < BoxTree < 3 > > ( bbox ) ;
for ( auto sei : SurfaceElementsRange ( ) )
{
const auto & sel = Get ( sei ) ;
2024-12-29 01:26:05 +05:00
// auto sel_index = sel.GetIndex();
2024-12-23 16:30:28 +05:00
Box < 3 > box ( Box < 3 > : : EMPTY_BOX ) ;
for ( auto pi : sel . PNums ( ) )
{
box . Add ( GetPoint ( pi , 0. ) ) ;
box . Add ( GetPoint ( pi , trig_shift * GetLimit ( pi ) ) ) ;
}
tree - > Insert ( box , sei ) ;
}
}
template < typename TFunc >
2024-12-29 02:58:44 +05:00
void FindTreeIntersections ( double trig_shift , double seg_shift , TFunc f ,
TBitArray < PointIndex > * relevant_points = nullptr )
2024-12-23 16:30:28 +05:00
{
static Timer t ( " GrowthVectorLimiter::FindTreeIntersections " ) ;
RegionTimer rt ( t ) ;
BuildSearchTree ( trig_shift ) ;
auto np_new = mesh . Points ( ) . Size ( ) ;
2024-12-29 01:26:05 +05:00
// int counter = 0;
2024-12-23 16:30:28 +05:00
for ( auto i : IntRange ( tool . np , np_new ) )
{
2024-12-29 02:58:44 +05:00
PointIndex pi_to = i + IndexBASE < PointIndex > ( ) ;
2024-12-23 16:30:28 +05:00
PointIndex pi_from = map_from [ pi_to ] ;
if ( ! pi_from . IsValid ( ) )
throw Exception ( " Point not mapped " ) ;
if ( relevant_points & & ! relevant_points - > Test ( pi_to ) & & ! relevant_points - > Test ( pi_from ) )
continue ;
Box < 3 > box ( Box < 3 > : : EMPTY_BOX ) ;
2024-12-29 01:26:05 +05:00
// auto seg = GetSeg(pi_to, seg_shift);
2024-12-23 16:30:28 +05:00
box . Add ( GetPoint ( pi_to , 0 ) ) ;
box . Add ( GetPoint ( pi_to , GetLimit ( pi_from ) ) ) ;
tree - > GetFirstIntersecting ( box . PMin ( ) , box . PMax ( ) , [ & ] ( SurfaceElementIndex sei ) {
const auto & sel = Get ( sei ) ;
if ( sel . PNums ( ) . Contains ( pi_from ) )
return false ;
if ( sel . PNums ( ) . Contains ( pi_to ) )
return false ;
2024-12-29 01:26:05 +05:00
// counter++;
2024-12-23 16:30:28 +05:00
f ( pi_to , sei ) ;
return false ;
} ) ;
}
}
void FixIntersectingSurfaceTrigs ( )
{
static Timer t ( " GrowthVectorLimiter::FixIntersectingSurfaceTrigs " ) ;
RegionTimer reg ( t ) ;
// check if surface trigs are intersecting each other
bool changed = true ;
while ( changed )
{
changed = false ;
Point3d pmin , pmax ;
mesh . GetBox ( pmin , pmax ) ;
BoxTree < 3 , SurfaceElementIndex > setree ( pmin , pmax ) ;
for ( auto sei : SurfaceElementsRange ( ) )
{
const Element2d & tri = Get ( sei ) ;
Box < 3 > box ( Box < 3 > : : EMPTY_BOX ) ;
for ( PointIndex pi : tri . PNums ( ) )
box . Add ( GetPoint ( pi , 1.0 , true ) ) ;
box . Increase ( 1e-3 * box . Diam ( ) ) ;
setree . Insert ( box , sei ) ;
}
for ( auto sei : SurfaceElementsRange ( ) )
{
const Element2d & tri = Get ( sei ) ;
Box < 3 > box ( Box < 3 > : : EMPTY_BOX ) ;
for ( PointIndex pi : tri . PNums ( ) )
box . Add ( GetPoint ( pi , 1.0 , true ) ) ;
setree . GetFirstIntersecting ( box . PMin ( ) , box . PMax ( ) , [ & ] ( size_t sej ) {
const Element2d & tri2 = Get ( sej ) ;
if ( mesh [ tri [ 0 ] ] . GetLayer ( ) ! = mesh [ tri2 [ 0 ] ] . GetLayer ( ) )
return false ;
netgen : : Point < 3 > tri1_points [ 3 ] , tri2_points [ 3 ] ;
const netgen : : Point < 3 > * trip1 [ 3 ] , * trip2 [ 3 ] ;
for ( int k = 0 ; k < 3 ; k + + )
{
trip1 [ k ] = & tri1_points [ k ] ;
trip2 [ k ] = & tri2_points [ k ] ;
}
auto set_points = [ & ] ( ) {
for ( int k = 0 ; k < 3 ; k + + )
{
tri1_points [ k ] = GetPoint ( tri [ k ] , 1.0 , true ) ;
tri2_points [ k ] = GetPoint ( tri2 [ k ] , 1.0 , true ) ;
}
} ;
set_points ( ) ;
int counter = 0 ;
while ( IntersectTriangleTriangle ( & trip1 [ 0 ] , & trip2 [ 0 ] ) )
{
changed = true ;
PointIndex pi_max_limit = PointIndex : : INVALID ;
for ( PointIndex pi :
{ tri [ 0 ] , tri [ 1 ] , tri [ 2 ] , tri2 [ 0 ] , tri2 [ 1 ] , tri2 [ 2 ] } )
2024-12-29 02:58:44 +05:00
// if (pi > tool.np && (!pi_max_limit.IsValid() || GetLimit(pi) > GetLimit(pi_max_limit)))
if ( pi > = tool . npi & & ( ! pi_max_limit . IsValid ( ) | | GetLimit ( pi ) > GetLimit ( pi_max_limit ) ) )
2024-12-23 16:30:28 +05:00
pi_max_limit = map_from [ pi ] ;
if ( ! pi_max_limit . IsValid ( ) )
break ;
ScaleLimit ( pi_max_limit , 0.9 ) ;
set_points ( ) ;
counter + + ;
if ( GetLimit ( pi_max_limit ) < 1e-10 )
{
WriteErrorMesh ( " error_blayer_self_intersection_pi " + ToString ( pi_max_limit ) + " .vol.gz " ) ;
throw NgException ( " Stop meshing in boundary layer thickness limitation: overlapping regions detected at elements " + ToString ( tri ) + " and " + ToString ( tri2 ) ) ;
}
if ( debugparam . debugoutput & & counter > 20 )
{
cerr < < " Limit intersecting surface elements: too many "
" limitation steps, sels: "
< < Get ( sei ) < < ' \t ' < < Get ( sej ) < < endl ;
for ( auto si : { sei , sej } )
{
auto sel = Get ( si ) ;
cerr < < " Limits: " ;
for ( auto pi : sel . PNums ( ) )
cerr < < GetLimit ( pi ) < < " , \t " ;
cerr < < endl ;
for ( auto pi : sel . PNums ( ) )
cerr < < GetPoint ( pi , 1.0 , true ) < < " \t " ;
cerr < < endl ;
}
cerr < < " pi_max_limit " < < pi_max_limit < < endl ;
break ;
}
}
return false ;
} ) ;
}
}
}
void LimitOriginalSurface ( double safety )
{
static Timer t ( " GrowthVectorLimiter::LimitOriginalSurface " ) ;
RegionTimer reg ( t ) ;
PrintMessage ( 5 , " GrowthVectorLimiter - original surface " ) ;
// limit to not intersect with other (original) surface elements
double trig_shift = 0 ;
double seg_shift = safety ;
FindTreeIntersections (
trig_shift , seg_shift , [ & ] ( PointIndex pi_to , SurfaceElementIndex sei ) {
if ( sei > = tool . nse )
return ; // ignore new surface elements in first pass
LimitGrowthVector ( pi_to , sei , trig_shift , seg_shift ) ;
} ) ;
}
void LimitBoundaryLayer ( double safety = 1.1 )
{
static Timer t ( " GrowthVectorLimiter::LimitBoundaryLayer " ) ;
PrintMessage ( 5 , " GrowthVectorLimiter - boundary layer " ) ;
// now limit again with shifted surface elements
double trig_shift = safety ;
double seg_shift = safety ;
size_t limit_counter = 1 ;
2024-12-29 02:58:44 +05:00
TBitArray < PointIndex > relevant_points , relevant_points_next ;
relevant_points . SetSize ( mesh . Points ( ) . Size ( ) + 1 ) ;
2024-12-23 16:30:28 +05:00
relevant_points_next . SetSize ( mesh . Points ( ) . Size ( ) + 1 ) ;
relevant_points . Set ( ) ;
while ( limit_counter )
{
RegionTimer reg ( t ) ;
size_t find_counter = 0 ;
limit_counter = 0 ;
relevant_points_next . Clear ( ) ;
FindTreeIntersections (
trig_shift , seg_shift , [ & ] ( PointIndex pi_to , SurfaceElementIndex sei ) {
find_counter + + ;
auto sel = Get ( sei ) ;
if ( LimitGrowthVector ( pi_to , sei , trig_shift , seg_shift ) )
{
limit_counter + + ;
relevant_points_next . SetBit ( pi_to ) ;
relevant_points_next . SetBit ( map_from [ pi_to ] ) ;
for ( auto pi : sel . PNums ( ) )
{
relevant_points_next . SetBit ( pi ) ;
2024-12-29 02:58:44 +05:00
// if (pi >= tool.np) // was this correct (JS) ?
if ( pi > = tool . npi )
2024-12-23 16:30:28 +05:00
relevant_points_next . SetBit ( map_from [ pi ] ) ;
else
relevant_points_next . SetBit ( map_from [ pi ] ) ;
}
}
for ( auto pi : sel . PNums ( ) )
{
2024-12-29 02:58:44 +05:00
//if (pi >= tool.np) // was this correct (JS) ?
if ( pi > = tool . npi )
2024-12-23 16:30:28 +05:00
return ;
if ( tool . mapto [ pi ] . Size ( ) = = 0 )
return ;
}
if ( LimitGrowthVector ( pi_to , sei , trig_shift , seg_shift , true ) )
limit_counter + + ;
} ,
& relevant_points ) ;
relevant_points = relevant_points_next ;
}
}
void CheckLimits ( int line )
{
auto check_point = [ & ] ( PointIndex pi ) {
if ( limits [ pi ] < 1e-8 )
{
WriteErrorMesh ( " error_blayer_intersection_pi " + ToString ( pi ) + " .vol.gz " ) ;
throw NgException ( __FILE__ + ToString ( line ) + " : Stop meshing in boundary layer thickness limitation: overlapping regions detected at point " + ToString ( pi ) ) ;
}
} ;
for ( auto pi : Range ( growthvectors ) )
check_point ( pi ) ;
for ( auto & [ special_pi , special_point ] : tool . special_boundary_points )
check_point ( special_pi ) ;
}
void Perform ( )
{
limits . SetSize ( mesh . Points ( ) . Size ( ) ) ;
limits = 1.0 ;
if ( tool . special_boundary_points . size ( ) )
{
auto point_to_sel = tool . mesh . CreatePoint2SurfaceElementTable ( ) ;
for ( auto & [ pi , special_point ] : tool . special_boundary_points )
{
auto maxh = mesh . GetH ( mesh [ pi ] ) ;
auto new_limit = min ( 0.3 * maxh / tool . total_height , 1.0 ) ;
if ( new_limit < 1.0 )
{
limits [ pi ] = new_limit ;
for ( auto sei : point_to_sel [ pi ] )
for ( auto pi_ : Get ( sei ) . PNums ( ) )
limits [ pi_ ] = new_limit ;
}
}
}
std : : array safeties = { 0.5 , 1.1 , 1.5 , 1.5 } ;
// No smoothing in the last pass, to avoid generating new intersections
std : : array smoothing_factors = { 0.8 , 0.7 , 0.5 , 0.0 } ;
for ( auto i_pass : Range ( safeties . size ( ) ) )
{
PrintMessage ( 4 , " GrowthVectorLimiter pass " , i_pass ) ;
double safety = safeties [ i_pass ] ;
CheckLimits ( __LINE__ ) ;
// intersect segment with original surface elements
LimitOriginalSurface ( 2.1 ) ;
CheckLimits ( __LINE__ ) ;
// intersect prisms with themself
LimitSelfIntersection ( 1.3 * safety ) ;
CheckLimits ( __LINE__ ) ;
// intesect segment with prism
LimitBoundaryLayer ( safety ) ;
CheckLimits ( __LINE__ ) ;
2024-12-30 01:36:37 +05:00
for ( [[maybe_unused]] auto i : Range ( 10 ) )
2024-12-23 16:30:28 +05:00
EqualizeLimits ( smoothing_factors [ i_pass ] ) ;
CheckLimits ( __LINE__ ) ;
if ( i_pass = = safeties . size ( ) - 1 )
FixIntersectingSurfaceTrigs ( ) ;
CheckLimits ( __LINE__ ) ;
}
for ( auto i : Range ( growthvectors ) )
growthvectors [ i ] * = limits [ i ] ;
for ( auto & [ special_pi , special_point ] : tool . special_boundary_points )
{
for ( auto & group : special_point . growth_groups )
{
group . growth_vector * = limits [ special_pi ] ;
}
}
}
} ;
} // namespace netgen