Merge branch 'boundarylayer_cleanup' into 'master'

restructure BoundaryLayer code

See merge request jschoeberl/netgen!489
This commit is contained in:
Matthias Hochsteger 2022-03-09 12:43:47 +00:00
commit 0d475f4c43
2 changed files with 181 additions and 91 deletions

View File

@ -9,25 +9,6 @@
namespace netgen namespace netgen
{ {
Vec<3> getSurfaceNormal(const Mesh & mesh, const Element2d & el)
{
auto v0 = mesh[el[0]];
return Cross(mesh[el[1]]-v0, mesh[el[2]]-v0).Normalize();
};
Vec<3> getEdgeTangent(const Mesh & mesh, PointIndex pi)
{
Vec<3> tangent = 0.0;
for(auto segi : mesh.GetTopology().GetVertexSegments(pi))
{
auto & seg = mesh[segi];
tangent += (mesh[seg[1]] - mesh[seg[0]]);
}
return tangent.Normalize();
}
void InsertVirtualBoundaryLayer (Mesh & mesh) void InsertVirtualBoundaryLayer (Mesh & mesh)
{ {
cout << "Insert virt. b.l." << endl; cout << "Insert virt. b.l." << endl;
@ -265,59 +246,56 @@ namespace netgen
growthvectors[pi] += normals[pi]; growthvectors[pi] += normals[pi];
} }
void GenerateBoundaryLayer(Mesh& mesh, const BoundaryLayerParameters& blp)
BoundaryLayerTool::BoundaryLayerTool(Mesh & mesh_, const BoundaryLayerParameters & params_)
: mesh(mesh_), topo(mesh_.GetTopology()), params(params_)
{ {
static Timer timer("Create Boundarylayers"); static Timer timer("BoundaryLayerTool::ctor");
RegionTimer regt(timer); RegionTimer regt(timer);
int max_edge_nr = -1; max_edge_nr = -1;
for(const auto& seg : mesh.LineSegments()) for(const auto& seg : mesh.LineSegments())
if(seg.edgenr > max_edge_nr) if(seg.edgenr > max_edge_nr)
max_edge_nr = seg.edgenr; max_edge_nr = seg.edgenr;
int new_mat_nr = mesh.GetNDomains() +1; new_mat_nr = mesh.GetNDomains() +1;
mesh.SetMaterial(new_mat_nr, blp.new_mat); mesh.SetMaterial(new_mat_nr, params.new_mat);
auto domains = blp.domains; domains = params.domains;
if(!blp.outside) if(!params.outside)
domains.Invert(); domains.Invert();
auto& meshtopo = mesh.GetTopology(); topo.SetBuildVertex2Element(true);
meshtopo.SetBuildVertex2Element(true);
mesh.UpdateTopology(); mesh.UpdateTopology();
bool have_single_segments = HaveSingleSegments(mesh); have_single_segments = HaveSingleSegments(mesh);
Array<Segment> segments;
if(have_single_segments) if(have_single_segments)
segments = BuildSegments(mesh); segments = BuildSegments(mesh);
else else
segments = mesh.LineSegments(); segments = mesh.LineSegments();
int np = mesh.GetNP(); np = mesh.GetNP();
int ne = mesh.GetNE(); ne = mesh.GetNE();
int nse = mesh.GetNSE(); nse = mesh.GetNSE();
int nseg = segments.Size(); nseg = segments.Size();
Array<Array<PointIndex>, PointIndex> mapto(np); p2sel = mesh.CreatePoint2SurfaceElementTable();
Array<Vec<3>, PointIndex> growthvectors(np); nfd_old = mesh.GetNFD();
growthvectors = 0.; si_map.SetSize(nfd_old+1);
Array<double> surfacefacs(mesh.GetNFD()+1);
surfacefacs = 0.;
// surface index map
Array<int> si_map(mesh.GetNFD()+1);
si_map = -1; si_map = -1;
}
int fd_old = mesh.GetNFD(); void BoundaryLayerTool :: CreateNewFaceDescriptors()
{
surfacefacs.SetSize(nfd_old+1);
surfacefacs = 0.0;
// create new FaceDescriptors // create new FaceDescriptors
for(auto i : Range(1, fd_old+1)) for(auto i : Range(1, nfd_old+1))
{ {
const auto& fd = mesh.GetFaceDescriptor(i); const auto& fd = mesh.GetFaceDescriptor(i);
string name = fd.GetBCName(); string name = fd.GetBCName();
if(blp.surfid.Contains(i)) if(params.surfid.Contains(i))
{ {
if(auto isIn = domains.Test(fd.DomainIn()); isIn != domains.Test(fd.DomainOut())) if(auto isIn = domains.Test(fd.DomainIn()); isIn != domains.Test(fd.DomainOut()))
{ {
@ -333,8 +311,12 @@ namespace netgen
} }
} }
} }
}
const auto & p2sel = mesh.CreatePoint2SurfaceElementTable(); void BoundaryLayerTool :: CalculateGrowthVectors()
{
growthvectors.SetSize(np);
growthvectors = 0.;
for(auto pi : mesh.Points().Range()) for(auto pi : mesh.Points().Range())
{ {
@ -349,10 +331,10 @@ namespace netgen
{ {
const auto & sel = mesh[sei]; const auto & sel = mesh[sei];
auto facei = sel.GetIndex(); auto facei = sel.GetIndex();
if(!blp.surfid.Contains(facei)) if(!params.surfid.Contains(facei))
continue; continue;
auto n = surfacefacs[sel.GetIndex()] * getSurfaceNormal(mesh, sel); auto n = surfacefacs[sel.GetIndex()] * getNormal(sel);
int itrig = sel.PNums().Pos(pi); int itrig = sel.PNums().Pos(pi);
itrig += sel.GetNP(); itrig += sel.GetNP();
@ -378,7 +360,10 @@ namespace netgen
np += (nn - npn)/(nn - npn*npn/npnp) * (n - npn/npnp * np); np += (nn - npn)/(nn - npn*npn/npnp) * (n - npn/npnp * np);
} }
} }
}
Array<Array<pair<SegmentIndex, int>>, SegmentIndex> BoundaryLayerTool :: BuildSegMap()
{
// Bit array to keep track of segments already processed // Bit array to keep track of segments already processed
BitArray segs_done(nseg+1); BitArray segs_done(nseg+1);
segs_done.Clear(); segs_done.Clear();
@ -393,16 +378,14 @@ namespace netgen
Array<Array<pair<SegmentIndex, int>>, SegmentIndex> segmap(segments.Size()); Array<Array<pair<SegmentIndex, int>>, SegmentIndex> segmap(segments.Size());
// moved segments // moved segments
Array<SegmentIndex> moved_segs; is_edge_moved.SetSize(max_edge_nr+1);
BitArray moved_edges(max_edge_nr+1); is_edge_moved = false;
moved_edges = false;
Array<Segment> new_segments;
// boundaries to project endings to // boundaries to project endings to
BitArray project_boundaries(fd_old+1); is_boundary_projected.SetSize(nfd_old+1);
BitArray move_boundaries(fd_old+1); is_boundary_projected.Clear();
project_boundaries.Clear(); is_boundary_moved.SetSize(nfd_old+1);
move_boundaries.Clear(); is_boundary_moved.Clear();
for(auto si : Range(segments)) for(auto si : Range(segments))
{ {
@ -412,7 +395,7 @@ namespace netgen
segs_done.SetBit(si); segs_done.SetBit(si);
segmap[si].Append(make_pair(si, 0)); segmap[si].Append(make_pair(si, 0));
moved_segs.Append(si); moved_segs.Append(si);
moved_edges.SetBit(segi.edgenr); is_edge_moved.SetBit(segi.edgenr);
for(auto sj : Range(segments)) for(auto sj : Range(segments))
{ {
if(segs_done.Test(sj)) continue; if(segs_done.Test(sj)) continue;
@ -428,36 +411,40 @@ namespace netgen
{ {
type = 2; type = 2;
if(fd.DomainIn() == 0 || fd.DomainOut() == 0) if(fd.DomainIn() == 0 || fd.DomainOut() == 0)
project_boundaries.SetBit(segj.si); is_boundary_projected.SetBit(segj.si);
} }
else if(const auto& fd = mesh.GetFaceDescriptor(segj.si); !domains.Test(fd.DomainIn()) && !domains.Test(fd.DomainOut())) else if(const auto& fd = mesh.GetFaceDescriptor(segj.si); !domains.Test(fd.DomainIn()) && !domains.Test(fd.DomainOut()))
{ {
type = 3; type = 3;
if(fd.DomainIn() == 0 || fd.DomainOut() == 0) if(fd.DomainIn() == 0 || fd.DomainOut() == 0)
project_boundaries.SetBit(segj.si); is_boundary_projected.SetBit(segj.si);
move_boundaries.SetBit(segj.si); is_boundary_moved.SetBit(segj.si);
} }
else else
{ {
type = 1; type = 1;
// in case 1 we project the growthvector onto the surface // in case 1 we project the growthvector onto the surface
project_boundaries.SetBit(segj.si); is_boundary_projected.SetBit(segj.si);
} }
segmap[si].Append(make_pair(sj, type)); segmap[si].Append(make_pair(sj, type));
} }
} }
} }
BitArray in_surface_direction(fd_old+1); return segmap;
in_surface_direction.Clear(); }
BitArray BoundaryLayerTool :: ProjectGrowthVectorsOnSurface()
{
BitArray in_surface_direction(nfd_old+1);
in_surface_direction.Clear();
// project growthvector on surface for inner angles // project growthvector on surface for inner angles
if(blp.grow_edges) if(params.grow_edges)
{ {
for(const auto& sel : mesh.SurfaceElements()) for(const auto& sel : mesh.SurfaceElements())
if(project_boundaries.Test(sel.GetIndex())) if(is_boundary_projected.Test(sel.GetIndex()))
{ {
auto n = getSurfaceNormal(mesh, sel); auto n = getNormal(sel);
for(auto i : Range(sel.PNums())) for(auto i : Range(sel.PNums()))
{ {
auto pi = sel.PNums()[i]; auto pi = sel.PNums()[i];
@ -492,7 +479,7 @@ namespace netgen
{ {
int count = 0; int count = 0;
for(const auto& seg2 : segments) for(const auto& seg2 : segments)
if(((seg[0] == seg2[0] && seg[1] == seg2[1]) || (seg[0] == seg2[1] && seg[1] == seg2[0])) && blp.surfid.Contains(seg2.si)) if(((seg[0] == seg2[0] && seg[1] == seg2[1]) || (seg[0] == seg2[1] && seg[1] == seg2[0])) && params.surfid.Contains(seg2.si))
count++; count++;
if(count == 1) if(count == 1)
{ {
@ -502,10 +489,15 @@ namespace netgen
} }
} }
return in_surface_direction;
}
void BoundaryLayerTool :: InterpolateGrowthVectors()
{
// interpolate tangential component of growth vector along edge // interpolate tangential component of growth vector along edge
for(auto edgenr : Range(max_edge_nr)) for(auto edgenr : Range(max_edge_nr))
{ {
if(!moved_edges[edgenr+1]) continue; if(!is_edge_moved[edgenr+1]) continue;
const auto& geo = *mesh.GetGeometry(); const auto& geo = *mesh.GetGeometry();
if(edgenr >= geo.GetNEdges()) if(edgenr >= geo.GetNEdges())
continue; continue;
@ -527,7 +519,7 @@ namespace netgen
while(true) while(true)
{ {
bool point_found = false; bool point_found = false;
for(auto si : meshtopo.GetVertexSegments(points.Last())) for(auto si : topo.GetVertexSegments(points.Last()))
{ {
const auto& seg = mesh[si]; const auto& seg = mesh[si];
if(seg.edgenr-1 != edgenr) if(seg.edgenr-1 != edgenr)
@ -555,30 +547,34 @@ namespace netgen
{ {
auto pi = points[i]; auto pi = points[i];
len += (mesh[pi] - mesh[points[i-1]]).Length(); len += (mesh[pi] - mesh[points[i-1]]).Length();
auto t = getEdgeTangent(mesh, pi); auto t = getEdgeTangent(pi);
auto lam = len/edge_len; auto lam = len/edge_len;
auto interpol = (1-lam) * (gt1 * t) * t + lam * (gt2 * t) * t; auto interpol = (1-lam) * (gt1 * t) * t + lam * (gt2 * t) * t;
growthvectors[pi] += interpol; growthvectors[pi] += interpol;
} }
} }
InterpolateSurfaceGrowthVectors(mesh, blp, fd_old, growthvectors, p2sel); InterpolateSurfaceGrowthVectors(mesh, params, nfd_old, growthvectors, p2sel);
}
void BoundaryLayerTool :: InsertNewElements( FlatArray<Array<pair<SegmentIndex, int>>, SegmentIndex> segmap, const BitArray & in_surface_direction )
{
Array<Array<PointIndex>, PointIndex> mapto(np);
// insert new points // insert new points
for (PointIndex pi = 1; pi <= np; pi++) for (PointIndex pi = 1; pi <= np; pi++)
if (growthvectors[pi].Length2() != 0) if (growthvectors[pi].Length2() != 0)
{ {
Point<3> p = mesh[pi]; Point<3> p = mesh[pi];
for(auto i : Range(blp.heights)) for(auto i : Range(params.heights))
{ {
p += blp.heights[i] * growthvectors[pi]; p += params.heights[i] * growthvectors[pi];
mapto[pi].Append(mesh.AddPoint(p)); mapto[pi].Append(mesh.AddPoint(p));
} }
} }
// add 2d quads on required surfaces // add 2d quads on required surfaces
map<pair<PointIndex, PointIndex>, int> seg2edge; map<pair<PointIndex, PointIndex>, int> seg2edge;
if(blp.grow_edges) if(params.grow_edges)
{ {
for(auto sei : moved_segs) for(auto sei : moved_segs)
{ {
@ -609,7 +605,7 @@ namespace netgen
if(in_surface_direction.Test(segj.si)) if(in_surface_direction.Test(segj.si))
{ {
Swap(pp1, pp2); Swap(pp1, pp2);
move_boundaries.SetBit(segj.si); is_boundary_moved.SetBit(segj.si);
} }
PointIndex p1 = pp1; PointIndex p1 = pp1;
PointIndex p2 = pp2; PointIndex p2 = pp2;
@ -622,7 +618,7 @@ namespace netgen
s0.si = segj.si; s0.si = segj.si;
new_segments.Append(s0); new_segments.Append(s0);
for(auto i : Range(blp.heights)) for(auto i : Range(params.heights))
{ {
Element2d sel(QUAD); Element2d sel(QUAD);
p3 = mapto[pp2][i]; p3 = mapto[pp2][i];
@ -690,7 +686,7 @@ namespace netgen
{ {
Array<PointIndex> points(sel.PNums()); Array<PointIndex> points(sel.PNums());
if(surfacefacs[sel.GetIndex()] > 0) Swap(points[0], points[2]); if(surfacefacs[sel.GetIndex()] > 0) Swap(points[0], points[2]);
for(auto j : Range(blp.heights)) for(auto j : Range(params.heights))
{ {
auto eltype = points.Size() == 3 ? PRISM : HEX; auto eltype = points.Size() == 3 ? PRISM : HEX;
Element el(eltype); Element el(eltype);
@ -722,12 +718,12 @@ namespace netgen
if(!mapto[p].Size()) if(!mapto[p].Size())
{ {
fixed_points.SetBit(p); fixed_points.SetBit(p);
if(move_boundaries.Test(sel.GetIndex())) if(is_boundary_moved.Test(sel.GetIndex()))
moveboundarypoint.SetBit(p); moveboundarypoint.SetBit(p);
} }
} }
} }
if(move_boundaries.Test(sel.GetIndex())) if(is_boundary_moved.Test(sel.GetIndex()))
{ {
for(auto& p : mesh[si].PNums()) for(auto& p : mesh[si].PNums())
if(mapto[p].Size()) if(mapto[p].Size())
@ -738,7 +734,7 @@ namespace netgen
for(SegmentIndex sei = 0; sei < nseg; sei++) for(SegmentIndex sei = 0; sei < nseg; sei++)
{ {
auto& seg = segments[sei]; auto& seg = segments[sei];
if(move_boundaries.Test(seg.si)) if(is_boundary_moved.Test(seg.si))
for(auto& p : seg.PNums()) for(auto& p : seg.PNums())
if(mapto[p].Size()) if(mapto[p].Size())
p = mapto[p].Last(); p = mapto[p].Last();
@ -795,7 +791,7 @@ namespace netgen
PointIndex p4 = p1; PointIndex p4 = p1;
PointIndex p5 = p2; PointIndex p5 = p2;
PointIndex p6 = p3; PointIndex p6 = p3;
for(auto i : Range(blp.heights)) for(auto i : Range(params.heights))
{ {
Element nel(PRISM); Element nel(PRISM);
nel[0] = p4; nel[1] = p5; nel[2] = p6; nel[0] = p4; nel[1] = p5; nel[2] = p6;
@ -811,7 +807,7 @@ namespace netgen
{ {
PointIndex p1 = moved[0]; PointIndex p1 = moved[0];
PointIndex p2 = moved[1]; PointIndex p2 = moved[1];
for(auto i : Range(blp.heights)) for(auto i : Range(params.heights))
{ {
PointIndex p3 = mapto[moved[1]][i]; PointIndex p3 = mapto[moved[1]][i];
PointIndex p4 = mapto[moved[0]][i]; PointIndex p4 = mapto[moved[0]][i];
@ -833,7 +829,7 @@ namespace netgen
if(moved.Size() == 1 && fixed.Size() == 1) if(moved.Size() == 1 && fixed.Size() == 1)
{ {
PointIndex p1 = moved[0]; PointIndex p1 = moved[0];
for(auto i : Range(blp.heights)) for(auto i : Range(params.heights))
{ {
Element nel = el; Element nel = el;
PointIndex p2 = mapto[moved[0]][i]; PointIndex p2 = mapto[moved[0]][i];
@ -857,7 +853,7 @@ namespace netgen
throw Exception("This case is not implemented yet! Fixed size = " + ToString(fixed.Size())); throw Exception("This case is not implemented yet! Fixed size = " + ToString(fixed.Size()));
PointIndex p1 = moved[0]; PointIndex p1 = moved[0];
PointIndex p2 = moved[1]; PointIndex p2 = moved[1];
for(auto i : Range(blp.heights)) for(auto i : Range(params.heights))
{ {
PointIndex p3 = mapto[moved[1]][i]; PointIndex p3 = mapto[moved[1]][i];
PointIndex p4 = mapto[moved[0]][i]; PointIndex p4 = mapto[moved[0]][i];
@ -882,8 +878,11 @@ namespace netgen
throw Exception("Boundarylayer only implemented for tets and pyramids outside yet!"); throw Exception("Boundarylayer only implemented for tets and pyramids outside yet!");
} }
} }
}
for(auto i : Range(1, fd_old+1)) void BoundaryLayerTool :: SetDomInOut()
{
for(auto i : Range(1, nfd_old+1))
if(si_map[i] != -1) if(si_map[i] != -1)
{ {
if(mesh.GetFaceDescriptor(mesh.GetNFD()).DomainIn() == new_mat_nr) if(mesh.GetFaceDescriptor(mesh.GetNFD()).DomainIn() == new_mat_nr)
@ -891,6 +890,10 @@ namespace netgen
else else
mesh.GetFaceDescriptor(i).SetDomainIn(new_mat_nr); mesh.GetFaceDescriptor(i).SetDomainIn(new_mat_nr);
} }
}
void BoundaryLayerTool :: AddSegments()
{
if(have_single_segments) if(have_single_segments)
MergeAndAddSegments(mesh, new_segments); MergeAndAddSegments(mesh, new_segments);
else else
@ -898,12 +901,35 @@ namespace netgen
for(auto & seg : new_segments) for(auto & seg : new_segments)
mesh.AddSegment(seg); mesh.AddSegment(seg);
} }
}
void BoundaryLayerTool :: Perform()
{
CreateNewFaceDescriptors();
CalculateGrowthVectors();
auto segmap = BuildSegMap();
auto in_surface_direction = ProjectGrowthVectorsOnSurface();
InterpolateGrowthVectors();
InsertNewElements(segmap, in_surface_direction);
SetDomInOut();
AddSegments();
mesh.GetTopology().ClearEdges(); mesh.GetTopology().ClearEdges();
mesh.SetNextMajorTimeStamp(); mesh.SetNextMajorTimeStamp();
mesh.UpdateTopology(); mesh.UpdateTopology();
} }
void GenerateBoundaryLayer(Mesh& mesh, const BoundaryLayerParameters& blp)
{
static Timer timer("Create Boundarylayers");
RegionTimer regt(timer);
BoundaryLayerTool tool(mesh, blp);
tool.Perform();
}
void AddDirection( Vec<3> & a, Vec<3> b ) void AddDirection( Vec<3> & a, Vec<3> b )
{ {
if(a.Length2()==0.) if(a.Length2()==0.)

View File

@ -26,4 +26,68 @@ DLL_HEADER void GenerateBoundaryLayer (Mesh & mesh,
DLL_HEADER int /* new_domain_number */ GenerateBoundaryLayer2 (Mesh & mesh, int domain, const Array<double> & thicknesses, bool should_make_new_domain=true, const Array<int> & boundaries=Array<int>{}); DLL_HEADER int /* new_domain_number */ GenerateBoundaryLayer2 (Mesh & mesh, int domain, const Array<double> & thicknesses, bool should_make_new_domain=true, const Array<int> & boundaries=Array<int>{});
class BoundaryLayerTool
{
public:
BoundaryLayerTool(Mesh & mesh_, const BoundaryLayerParameters & params_);
void Perform();
protected:
Mesh & mesh;
MeshTopology & topo;
BoundaryLayerParameters params;
Array<Vec<3>, PointIndex> growthvectors;
Table<SurfaceElementIndex, PointIndex> p2sel;
BitArray domains, is_edge_moved, is_boundary_projected, is_boundary_moved;
Array<SegmentIndex> moved_segs;
int max_edge_nr, new_mat_nr, nfd_old;
int np, nseg, nse, ne;
bool have_single_segments;
Array<Segment> segments, new_segments;
Array<double> surfacefacs;
Array<int> si_map;
// major steps called in Perform()
void CreateNewFaceDescriptors();
void CalculateGrowthVectors();
Array<Array<pair<SegmentIndex, int>>, SegmentIndex> BuildSegMap();
BitArray ProjectGrowthVectorsOnSurface();
void InterpolateGrowthVectors();
void InsertNewElements(FlatArray<Array<pair<SegmentIndex, int>>, SegmentIndex> segmap, const BitArray & in_surface_direction);
void SetDomInOut();
void AddSegments();
// utility functions
array<Point<3>, 2> GetGrowSeg( PointIndex pi0 );
ArrayMem<Point<3>, 4> GetGrowFace( SurfaceElementIndex sei );
ArrayMem<Point<3>, 4> GetGrowFace( SurfaceElementIndex sei, int face );
bool IsSegIntersectingPlane ( array<Point<3>, 2> seg, array<Point<3>, 3> trig, double & lam);
bool IsIntersectingTrig ( array<Point<3>, 2> seg, array<Point<3>, 3> trig, double & lam);
Vec<3> getNormal(const Element2d & el)
{
auto v0 = mesh[el[0]];
return Cross(mesh[el[1]]-v0, mesh[el[2]]-v0).Normalize();
}
Vec<3> getEdgeTangent(PointIndex pi)
{
Vec<3> tangent = 0.0;
for(auto segi : topo.GetVertexSegments(pi))
{
auto & seg = mesh[segi];
tangent += (mesh[seg[1]] - mesh[seg[0]]);
}
return tangent.Normalize();
}
};
#endif #endif