generic implementation of InterpolateSurfaceGrowthVectors

This commit is contained in:
mhochsteger@cerbsim.com 2022-03-04 15:42:16 +01:00
parent f0b10d696e
commit 8e861d1773

View File

@ -5,9 +5,18 @@
#include "global.hpp" #include "global.hpp"
#include "../geom2d/csg2d.hpp" #include "../geom2d/csg2d.hpp"
#include <set>
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();
};
void InsertVirtualBoundaryLayer (Mesh & mesh) void InsertVirtualBoundaryLayer (Mesh & mesh)
{ {
cout << "Insert virt. b.l." << endl; cout << "Insert virt. b.l." << endl;
@ -182,115 +191,67 @@ namespace netgen
void InterpolateSurfaceGrowthVectors(const Mesh & mesh, const BoundaryLayerParameters& blp, int fd_old, FlatArray<Vec<3>, PointIndex> growthvectors, const Table<SurfaceElementIndex, PointIndex> & p2sel) void InterpolateSurfaceGrowthVectors(const Mesh & mesh, const BoundaryLayerParameters& blp, int fd_old, FlatArray<Vec<3>, PointIndex> growthvectors, const Table<SurfaceElementIndex, PointIndex> & p2sel)
{ {
static Timer tall("InterpolateSurfaceGrowthVectors"); RegionTimer rtall(tall);
static Timer tsmooth("InterpolateSurfaceGrowthVectors-Smoothing");
auto np = mesh.GetNP(); auto np = mesh.GetNP();
// interpolate growth vectors at inner surface points from surrounding edge points BitArray is_point_on_bl_surface(np+1);
Array<Point<2>, PointIndex> delaunay_points(np); is_point_on_bl_surface.Clear();
Array<PointIndex, PointIndex> pmap(np); // maps duplicated points Array<Vec<3>, PointIndex> normals(np);
for(auto pi : Range(growthvectors))
normals[pi] = growthvectors[pi];
Array<SurfaceElementIndex> surface_els; ParallelForRange( mesh.SurfaceElements().Range(), [&] ( auto myrange )
Array<PointIndex> edge_points;
Array<PointIndex> surface_points;
for(auto facei : Range(1, fd_old+1))
{ {
if(!blp.surfid.Contains(facei)) for(SurfaceElementIndex sei : myrange)
continue; {
auto facei = mesh[sei].GetIndex();
edge_points.SetSize(0); if(facei < fd_old && !blp.surfid.Contains(facei))
surface_points.SetSize(0);
surface_els.SetSize(0);
delaunay_points.SetSize(np);
pmap.SetSize(np);
mesh.GetSurfaceElementsOfFace (facei, surface_els);
Box<2> bbox ( Box<2>::EMPTY_BOX );
for(auto sei : surface_els)
{
auto sel = mesh[sei];
for (auto i : Range(sel.GetNP()))
{
auto & gi = sel.GeomInfo()[i];
Point<2> p = {gi.u, gi.v};
bbox.Add(p);
}
}
BoxTree<2> tree(bbox);
for(auto pi : mesh.Points().Range())
{
auto n_surf_els = p2sel[pi].Size();
bool has_relevant_sel = false;
for(auto sei : p2sel[pi])
if(mesh[sei].GetIndex()==facei)
{
has_relevant_sel = true;
break;
}
if(!has_relevant_sel)
continue; continue;
for(auto pi : mesh[sei].PNums())
if(mesh[pi].Type() == SURFACEPOINT)
is_point_on_bl_surface.SetBitAtomic(pi);
}
});
if(mesh[pi].Type() <= EDGEPOINT) Array<PointIndex> points;
edge_points.Append(pi); for(PointIndex pi : mesh.Points().Range())
else if(is_point_on_bl_surface[pi])
surface_points.Append(pi);
// the same point might have different uv coordinates (closed edges for instance)
// duplicate these points for the delaunay tree
bool inserted = false;
for(auto sei : p2sel[pi])
{
auto sel = mesh[sei];
if(sel.GetIndex()!=facei)
continue;
PointGeomInfo gi = sel.GeomInfo()[sel.PNums().Pos(pi)];
Point<2> p = {gi.u, gi.v};
bool found = false;
tree.GetFirstIntersecting( p, p, [&] (const auto pi_found) { return found = true; });
if(found)
continue;
auto pi_new = pi;
if(inserted)
{
pi_new = delaunay_points.Append(p);
pmap.Append(pi);
edge_points.Append(pi_new);
}
else
{
delaunay_points[pi] = p;
pmap[pi] = pi;
}
tree.Insert(p, pi_new);
inserted = true;
}
}
if(surface_points.Size()==0)
continue;
DelaunayMesh dmesh( delaunay_points, bbox );
for(auto pi : edge_points)
dmesh.AddPoint(pi);
std::map<PointIndex, double> weights;
for(auto pi : surface_points)
{ {
dmesh.CalcIntersecting(pi); points.Append(pi);
dmesh.CalcWeights(pi, weights); growthvectors[pi] = 0.0;
auto & v = growthvectors[pi];
auto n = 1./v.Length() * v;
for(auto & [pi_other, weight] : weights)
{
// interpolate only the tangential part of the growth vector
auto t = weight * growthvectors[pmap[pi_other]];
t -= (t * n) * n;
v += t;
}
} }
}
// smooth tangential part of growth vectors from edges to surface elements
RegionTimer rtsmooth(tsmooth);
for(auto i : Range(10))
{
for(auto pi : points)
{
auto sels = p2sel[pi];
Vec<3> new_gw = growthvectors[pi];
int cnt = 1;
std::set<PointIndex> suround;
suround.insert(pi);
auto normal = normals[pi];
for(auto sei: sels)
{
const auto & sel = mesh[sei];
for(auto pi1 : sel.PNums())
if(suround.count(pi1)==0)
{
suround.insert(pi1);
auto gw_other = growthvectors[pi1];
auto tangent_part = gw_other - (gw_other*normal)*normal;
new_gw += tangent_part;
}
}
growthvectors[pi] = 1.0/suround.size() * new_gw;
}
}
for(auto pi : points)
growthvectors[pi] += normals[pi];
} }
void GenerateBoundaryLayer(Mesh& mesh, const BoundaryLayerParameters& blp) void GenerateBoundaryLayer(Mesh& mesh, const BoundaryLayerParameters& blp)
@ -298,10 +259,6 @@ namespace netgen
static Timer timer("Create Boundarylayers"); static Timer timer("Create Boundarylayers");
RegionTimer regt(timer); RegionTimer regt(timer);
bool interpolate_growth_vectors = false;
if(mesh.GetGeometry())
interpolate_growth_vectors = mesh.GetGeometry()->GetGeomType() == Mesh::GEOM_OCC;
int max_edge_nr = -1; int 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)
@ -338,12 +295,6 @@ namespace netgen
Array<double> surfacefacs(mesh.GetNFD()+1); Array<double> surfacefacs(mesh.GetNFD()+1);
surfacefacs = 0.; surfacefacs = 0.;
auto getSurfaceNormal = [&mesh] (const Element2d& el)
{
auto v0 = mesh[el[0]];
return Cross(mesh[el[1]]-v0, mesh[el[2]]-v0).Normalize();
};
// surface index map // surface index map
Array<int> si_map(mesh.GetNFD()+1); Array<int> si_map(mesh.GetNFD()+1);
si_map = -1; si_map = -1;
@ -390,7 +341,7 @@ namespace netgen
if(!blp.surfid.Contains(facei)) if(!blp.surfid.Contains(facei))
continue; continue;
auto n = surfacefacs[sel.GetIndex()] * getSurfaceNormal(sel); auto n = surfacefacs[sel.GetIndex()] * getSurfaceNormal(mesh, sel);
int itrig = sel.PNums().Pos(pi); int itrig = sel.PNums().Pos(pi);
itrig += sel.GetNP(); itrig += sel.GetNP();
@ -495,7 +446,7 @@ namespace netgen
for(const auto& sel : mesh.SurfaceElements()) for(const auto& sel : mesh.SurfaceElements())
if(project_boundaries.Test(sel.GetIndex())) if(project_boundaries.Test(sel.GetIndex()))
{ {
auto n = getSurfaceNormal(sel); auto n = getSurfaceNormal(mesh, sel);
for(auto i : Range(sel.PNums())) for(auto i : Range(sel.PNums()))
{ {
auto pi = sel.PNums()[i]; auto pi = sel.PNums()[i];
@ -541,7 +492,6 @@ namespace netgen
} }
// interpolate tangential component of growth vector along edge // interpolate tangential component of growth vector along edge
if(interpolate_growth_vectors)
for(auto edgenr : Range(max_edge_nr)) for(auto edgenr : Range(max_edge_nr))
{ {
if(!moved_edges[edgenr+1]) continue; if(!moved_edges[edgenr+1]) continue;
@ -609,8 +559,7 @@ namespace netgen
} }
} }
if(interpolate_growth_vectors) InterpolateSurfaceGrowthVectors(mesh, blp, fd_old, growthvectors, p2sel);
InterpolateSurfaceGrowthVectors(mesh, blp, fd_old, growthvectors, p2sel);
// insert new points // insert new points
for (PointIndex pi = 1; pi <= np; pi++) for (PointIndex pi = 1; pi <= np; pi++)