netgen/libsrc/meshing/meshing3.cpp
Joachim Schoeberl 8f73a00d2d code cleanup
2024-12-15 18:00:50 +01:00

1290 lines
31 KiB
C++

#include <mystdlib.h>
#include "meshing.hpp"
namespace netgen
{
double minother;
double minwithoutother;
MeshingStat3d :: MeshingStat3d ()
{
cntsucc = cnttrials = cntelem = qualclass = 0;
vol0 = h = 1;
problemindex = 1;
}
Meshing3 :: Meshing3 (const string & rulefilename)
{
tolfak = 1;
LoadRules (rulefilename.c_str(), NULL);
adfront = new AdFront3;
problems.SetSize (rules.Size());
foundmap.SetSize (rules.Size());
canuse.SetSize (rules.Size());
ruleused.SetSize (rules.Size());
for (int i = 1; i <= rules.Size(); i++)
{
problems.Elem(i) = new char[255];
foundmap.Elem(i) = 0;
canuse.Elem(i) = 0;
ruleused.Elem(i) = 0;
}
}
Meshing3 :: Meshing3 (const char ** rulep)
{
tolfak = 1;
LoadRules (NULL, rulep);
adfront = new AdFront3;
problems.SetSize (rules.Size());
foundmap.SetSize (rules.Size());
canuse.SetSize (rules.Size());
ruleused.SetSize (rules.Size());
for (int i = 0; i < rules.Size(); i++)
{
problems[i] = new char[255];
foundmap[i] = 0;
canuse[i] = 0;
ruleused[i] = 0;
}
}
Meshing3 :: ~Meshing3 ()
{
delete adfront;
for (int i = 0; i < rules.Size(); i++)
{
delete [] problems[i];
delete rules[i];
}
}
/*
// was war das ????
static double CalcLocH (const NgArray<Point3d> & locpoints,
const NgArray<MiniElement2d> & locfaces,
double h)
{
return h;
int i, j;
double hi, h1, d, dn, sum, weight, wi;
Point3d p0, pc;
Vec3d n, v1, v2;
p0.X() = p0.Y() = p0.Z() = 0;
for (j = 1; j <= 3; j++)
{
p0.X() += locpoints.Get(locfaces.Get(1).PNum(j)).X();
p0.Y() += locpoints.Get(locfaces.Get(1).PNum(j)).Y();
p0.Z() += locpoints.Get(locfaces.Get(1).PNum(j)).Z();
}
p0.X() /= 3; p0.Y() /= 3; p0.Z() /= 3;
v1 = locpoints.Get(locfaces.Get(1).PNum(2)) -
locpoints.Get(locfaces.Get(1).PNum(1));
v2 = locpoints.Get(locfaces.Get(1).PNum(3)) -
locpoints.Get(locfaces.Get(1).PNum(1));
h1 = v1.Length();
n = Cross (v2, v1);
n /= n.Length();
sum = 0;
weight = 0;
for(int i = 1; i <= locfaces.Size(); i++)
{
pc.X() = pc.Y() = pc.Z() = 0;
for (j = 1; j <= 3; j++)
{
pc.X() += locpoints.Get(locfaces.Get(i).PNum(j)).X();
pc.Y() += locpoints.Get(locfaces.Get(i).PNum(j)).Y();
pc.Z() += locpoints.Get(locfaces.Get(i).PNum(j)).Z();
}
pc.X() /= 3; pc.Y() /= 3; pc.Z() /= 3;
d = Dist (p0, pc);
dn = n * (pc - p0);
hi = Dist (locpoints.Get(locfaces.Get(i).PNum(1)),
locpoints.Get(locfaces.Get(i).PNum(2)));
if (dn > -0.2 * h1)
{
wi = 1 / (h1 + d);
wi *= wi;
}
else
wi = 0;
sum += hi * wi;
weight += wi;
}
return sum/weight;
}
*/
PointIndex Meshing3 :: AddPoint (const Point3d & p, PointIndex globind)
{
return adfront -> AddPoint (p, globind);
}
void Meshing3 :: AddBoundaryElement (const Element2d & elem)
{
MiniElement2d mini(elem.GetNP());
for (int j = 0; j < elem.GetNP(); j++)
mini[j] = elem[j];
adfront -> AddFace(mini);
}
void Meshing3 :: AddBoundaryElement (const MiniElement2d & elem)
{
adfront -> AddFace(elem);
}
int Meshing3 :: AddConnectedPair (const INDEX_2 & apair)
{
return adfront -> AddConnectedPair (apair);
}
MESHING3_RESULT Meshing3 ::
GenerateMesh (Mesh & mesh, const MeshingParameters & mp)
{
static Timer t("Meshing3::GenerateMesh"); RegionTimer reg(t);
// static Timer meshing3_timer_a("Meshing3::GenerateMesh a", 2);
// static Timer meshing3_timer_b("Meshing3::GenerateMesh b", 2);
// static Timer meshing3_timer_c("Meshing3::GenerateMesh c", 1);
// static Timer meshing3_timer_d("Meshing3::GenerateMesh d", 2);
// static int meshing3_timer = NgProfiler::CreateTimer ("Meshing3::GenerateMesh");
// static int meshing3_timer_a = NgProfiler::CreateTimer ("Meshing3::GenerateMesh a");
// static int meshing3_timer_b = NgProfiler::CreateTimer ("Meshing3::GenerateMesh b");
// static int meshing3_timer_c = NgProfiler::CreateTimer ("Meshing3::GenerateMesh c");
// static int meshing3_timer_d = NgProfiler::CreateTimer ("Meshing3::GenerateMesh d");
// NgProfiler::RegionTimer reg (meshing3_timer);
Array<Point3d, PointIndex> locpoints; // local points
Array<MiniElement2d> locfaces; // local faces
Array<PointIndex, PointIndex> pindex; // mapping from local to front point numbering
Array<int, PointIndex> allowpoint; // point is allowed ?
Array<INDEX> findex; // mapping from local to front face numbering
//INDEX_2_HASHTABLE<int> connectedpairs(100); // connecgted pairs for prism meshing
Array<Point3d, PointIndex> plainpoints; // points in reference coordinates
NgArray<int> delpoints, delfaces; // points and lines to be deleted
NgArray<Element> locelements; // new generated elements
int j, oldnp, oldnf;
int found;
referencetransform trans;
Point3d inp;
float err;
INDEX locfacesplit; //index for faces in outer area
bool loktestmode = false;
int uselocalh = mp.uselocalh;
// int giveuptol = mp.giveuptol; //
MeshingStat3d stat; // statistics
int plotstat_oldne = -1;
// for star-shaped domain meshing
Array<MeshPoint, PointIndex> grouppoints;
Array<MiniElement2d> groupfaces;
Array<PointIndex, PointIndex> grouppindex;
Array<INDEX> groupfindex;
float minerr;
int hasfound;
[[maybe_unused]] double tetvol;
// int giveup = 0;
NgArray<Point3d> tempnewpoints;
NgArray<MiniElement2d> tempnewfaces;
NgArray<int> tempdelfaces;
NgArray<Element> templocelements;
stat.h = mp.maxh;
adfront->SetStartFront (mp.baseelnp);
found = 0;
stat.vol0 = adfront -> Volume();
tetvol = 0;
stat.qualclass = 1;
while (1)
{
if (multithread.terminate)
throw NgException ("Meshing stopped");
// break if advancing front is empty
if (!mp.baseelnp && adfront->Empty())
break;
// break, if advancing front has no elements with
// mp.baseelnp nodes
if (mp.baseelnp && adfront->Empty (mp.baseelnp))
break;
locpoints.SetSize(0);
locfaces.SetSize(0);
locelements.SetSize(0);
pindex.SetSize(0);
findex.SetSize(0);
INDEX_2_HASHTABLE<int> connectedpairs(100); // connected pairs for prism meshing
// select base-element (will be locface[1])
// and get local environment of radius (safety * h)
int baseelem = adfront -> SelectBaseElement ();
if (mp.baseelnp && adfront->GetFace (baseelem).GetNP() != mp.baseelnp)
{
adfront->IncrementClass (baseelem);
continue;
}
const MiniElement2d & bel = adfront->GetFace (baseelem);
const Point<3> p1 = adfront->GetPoint (bel[0]);
const Point<3> p2 = adfront->GetPoint (bel[1]);
const Point<3> p3 = adfront->GetPoint (bel[2]);
Point<3> pmid = Center (p1, p2, p3);
double his = (Dist (p1, p2) + Dist(p1, p3) + Dist(p2, p3)) / 3;
double hshould = mesh.GetH (pmid);
if (adfront->GetFace (baseelem).GetNP() == 4)
hshould = max2 (his, hshould);
double hmax = (his > hshould) ? his : hshould;
// qualclass should come from baseelem !!!!!
double hinner = hmax * (1 + stat.qualclass);
double houter = hmax * (1 + 2 * stat.qualclass);
// meshing3_timer_a.Start();
stat.qualclass =
adfront -> GetLocals (baseelem, locpoints, locfaces,
pindex, findex, connectedpairs,
houter, hinner,
locfacesplit);
// meshing3_timer_a.Stop();
// (*testout) << "locfaces = " << endl << locfaces << endl;
//loktestmode = 1;
testmode = loktestmode; //changed
// loktestmode = testmode = (adfront->GetFace (baseelem).GetNP() == 4) && (rules.Size() == 5);
loktestmode = stat.qualclass > 5;
if (loktestmode)
{
(*testout) << "baseel = " << baseelem << ", ind = " << findex[0] << endl;
int pi1 = pindex[locfaces[0].PNum(1)];
int pi2 = pindex[locfaces[0].PNum(2)];
int pi3 = pindex[locfaces[0].PNum(3)];
(*testout) << "pi = " << pi1 << ", " << pi2 << ", " << pi3 << endl;
}
if (testmode)
{
(*testout) << "baseelem = " << baseelem << " qualclass = " << stat.qualclass << endl;
(*testout) << "locpoints = " << endl << locpoints << endl;
(*testout) << "connected = " << endl << connectedpairs << endl;
}
// loch = CalcLocH (locpoints, locfaces, h);
stat.nff = adfront->GetNF();
stat.vol = adfront->Volume();
if (stat.vol < 0) break;
oldnp = locpoints.Size();
oldnf = locfaces.Size();
allowpoint.SetSize(locpoints.Size());
if (uselocalh && stat.qualclass <= 3)
//for(int i = 1; i <= allowpoint.Size(); i++)
for(auto i : allowpoint.Range())
{
allowpoint[i] =
(mesh.GetH (locpoints[i]) > 0.4 * hshould / mp.sloppy) ? 2 : 1;
}
else
allowpoint = 2;
if (stat.qualclass >= mp.starshapeclass &&
mp.baseelnp != 4)
{
// NgProfiler::RegionTimer reg1 (meshing3_timer_b);
// star-shaped domain removing
grouppoints.SetSize (0);
groupfaces.SetSize (0);
grouppindex.SetSize (0);
groupfindex.SetSize (0);
adfront -> GetGroup (findex[0], grouppoints, groupfaces,
grouppindex, groupfindex);
bool onlytri = 1;
for (auto & f : groupfaces)
if (f.GetNP() != 3)
onlytri = 0;
if (onlytri && groupfaces.Size() <= 20 + 2*stat.qualclass &&
FindInnerPoint (grouppoints, groupfaces, inp) &&
!adfront->PointInsideGroup(grouppindex, groupfaces))
{
(*testout) << "inner point found" << endl;
for(int i = 1; i <= groupfaces.Size(); i++)
adfront -> DeleteFace (groupfindex[i-1]);
for(int i = 1; i <= groupfaces.Size(); i++)
for (j = 1; j <= locfaces.Size(); j++)
if (findex[j-1] == groupfindex[i-1])
delfaces.Append (j);
delfaces.SetSize (0);
INDEX npi;
Element newel(TET);
npi = mesh.AddPoint (inp);
newel.SetNP(4);
newel.PNum(4) = npi;
for(int i = 0; i < groupfaces.Size(); i++)
{
for (j = 1; j <= 3; j++)
{
newel.PNum(j) =
adfront->GetGlobalIndex
(grouppindex[groupfaces[i].PNum(j)]);
}
mesh.AddVolumeElement (newel);
}
continue;
}
}
found = 0;
hasfound = 0;
minerr = 1e6;
// int optother = 0;
/*
for(int i = 1; i <= locfaces.Size(); i++)
{
(*testout) << "Face " << i << ": ";
for (j = 1; j <= locfaces.Get(i).GetNP(); j++)
(*testout) << pindex.Get(locfaces.Get(i).PNum(j)) << " ";
(*testout) << endl;
}
for(int i = 1; i <= locpoints.Size(); i++)
{
(*testout) << "p" << i
<< ", gi = " << pindex.Get(i)
<< " = " << locpoints.Get(i) << endl;
}
*/
minother = 1e10;
minwithoutother = 1e10;
bool impossible = 1;
for (int rotind = 1; rotind <= locfaces[0].GetNP(); rotind++)
{
// set transformatino to reference coordinates
if (locfaces[0].GetNP() == 3)
{
trans.Set (locpoints[locfaces[0].PNumMod(1+rotind)],
locpoints[locfaces[0].PNumMod(2+rotind)],
locpoints[locfaces[0].PNumMod(3+rotind)], hshould);
}
else
{
trans.Set (locpoints[locfaces[0].PNumMod(1+rotind)],
locpoints[locfaces[0].PNumMod(2+rotind)],
locpoints[locfaces[0].PNumMod(4+rotind)], hshould);
}
// trans.ToPlain (locpoints, plainpoints);
plainpoints.SetSize (locpoints.Size());
for (auto i : locpoints.Range())
trans.ToPlain (locpoints[i], plainpoints[i]);
for (auto i : allowpoint.Range())
if (plainpoints[i].Z() > 0)
allowpoint[i] = false;
stat.cnttrials++;
if (stat.cnttrials % 100 == 0)
{
(*testout) << "\n";
for(int i = 1; i <= canuse.Size(); i++)
{
(*testout) << foundmap.Get(i) << "/"
<< canuse.Get(i) << "/"
<< ruleused.Get(i) << " map/can/use rule " << rules.Get(i)->Name() << "\n";
}
(*testout) << endl;
}
// NgProfiler::StartTimer (meshing3_timer_c);
// meshing3_timer_c.Start();
found = ApplyRules (plainpoints, allowpoint,
locfaces, locfacesplit, connectedpairs,
locelements, delfaces,
stat.qualclass, mp.sloppy, rotind, err);
if (found >= 0) impossible = 0;
if (found < 0) found = 0;
// meshing3_timer_c.Stop();
// NgProfiler::StopTimer (meshing3_timer_c);
if (!found) loktestmode = 0;
// NgProfiler::RegionTimer reg2 (meshing3_timer_d);
if (loktestmode)
{
(*testout) << "plainpoints = " << endl << plainpoints << endl;
(*testout) << "Applyrules found " << found << endl;
}
if (found) stat.cntsucc++;
locpoints.SetSize (plainpoints.Size());
/*
for (int i = oldnp+1; i <= plainpoints.Size(); i++)
trans.FromPlain (plainpoints.Elem(i), locpoints.Elem(i));
*/
for (auto i : plainpoints.Range().Modify(oldnp,0))
trans.FromPlain (plainpoints[i], locpoints[i]);
// avoid meshing from large to small mesh-size
if (uselocalh && found && stat.qualclass <= 3)
{
for (int i = 1; i <= locelements.Size(); i++)
{
Point3d pmin = locpoints[locelements.Get(i).PNum(1)];
Point3d pmax = pmin;
for (j = 2; j <= 4; j++)
{
const Point3d & hp = locpoints[locelements.Get(i).PNum(j)];
pmin.SetToMin (hp);
pmax.SetToMax (hp);
}
if (mesh.GetMinH (pmin, pmax) < 0.4 * hshould / mp.sloppy)
found = 0;
}
}
if (found)
{
for (int i = 1; i <= locelements.Size(); i++)
for (int j = 1; j <= 4; j++)
{
const Point3d & hp = locpoints[locelements.Get(i).PNum(j)];
if (Dist (hp, pmid) > hinner)
found = 0;
}
}
if (found)
ruleused.Elem(found)++;
// plotstat->Plot(stat);
if (stat.cntelem != plotstat_oldne)
{
plotstat_oldne = stat.cntelem;
PrintMessageCR (5, "El: ", stat.cntelem,
// << " trials: " << stat.cnttrials
" faces: ", stat.nff,
" vol = ", float(100 * stat.vol / stat.vol0));
multithread.percent = 100 - 100.0 * stat.vol / stat.vol0;
}
if (found && (!hasfound || err < minerr) )
{
if (testmode)
{
(*testout) << "found is active, 3" << endl;
//for(int i = 1; i <= plainpoints.Size(); i++)
for(auto i : plainpoints.Range())
{
(*testout) << "p";
if (i < pindex.Range().Next())
(*testout) << pindex[i] << ": ";
else
(*testout) << "new: ";
(*testout) << plainpoints[i] << endl;
}
}
hasfound = found;
minerr = err;
tempnewpoints.SetSize (0);
// for(int i = oldnp+1; i <= locpoints.Size(); i++)
for (auto i : locpoints.Range().Modify(oldnp,0))
tempnewpoints.Append (locpoints[i]);
tempnewfaces.SetSize (0);
// for(int i = oldnf+1; i <= locfaces.Size(); i++)
for (auto i : locfaces.Range().Modify(oldnf,0))
tempnewfaces.Append (locfaces[i]);
tempdelfaces.SetSize (0);
// for(int i = 1; i <= delfaces.Size(); i++)
for (auto i : delfaces.Range())
tempdelfaces.Append (delfaces[i]);
templocelements.SetSize (0);
// for(int i = 1; i <= locelements.Size(); i++)
for (auto i : locelements.Range())
templocelements.Append (locelements[i]);
/*
optother =
strcmp (problems[found], "other") == 0;
*/
}
locpoints.SetSize (oldnp);
locfaces.SetSize (oldnf);
delfaces.SetSize (0);
locelements.SetSize (0);
}
if (hasfound)
{
/*
if (optother)
(*testout) << "Other is optimal" << endl;
if (minother < minwithoutother)
{
(*testout) << "Other is better, " << minother << " less " << minwithoutother << endl;
}
*/
for(int i = 1; i <= tempnewpoints.Size(); i++)
locpoints.Append (tempnewpoints.Get(i));
for(int i = 1; i <= tempnewfaces.Size(); i++)
locfaces.Append (tempnewfaces.Get(i));
for(int i = 1; i <= tempdelfaces.Size(); i++)
delfaces.Append (tempdelfaces.Get(i));
for(int i = 1; i <= templocelements.Size(); i++)
locelements.Append (templocelements.Get(i));
if (loktestmode)
{
(*testout) << "apply rule" << endl;
for (auto i : locpoints.Range())
{
(*testout) << "p";
if (pindex.Range().Contains(i))
(*testout) << pindex[i] << ": ";
else
(*testout) << "new: ";
(*testout) << locpoints[i] << endl;
}
}
pindex.SetSize(locpoints.Size());
// for (int i = oldnp+1; i <= locpoints.Size(); i++)
for (auto i : locpoints.Range().Modify(oldnp,0))
{
PointIndex globind = mesh.AddPoint (locpoints[i]);
pindex[i] = adfront -> AddPoint (locpoints[i], globind);
}
for (int i = 1; i <= locelements.Size(); i++)
{
Point3d * hp1, * hp2, * hp3, * hp4;
hp1 = &locpoints[locelements.Get(i).PNum(1)];
hp2 = &locpoints[locelements.Get(i).PNum(2)];
hp3 = &locpoints[locelements.Get(i).PNum(3)];
hp4 = &locpoints[locelements.Get(i).PNum(4)];
tetvol += (1.0 / 6.0) * ( Cross ( *hp2 - *hp1, *hp3 - *hp1) * (*hp4 - *hp1) );
for (j = 1; j <= locelements.Get(i).NP(); j++)
locelements.Elem(i).PNum(j) =
adfront -> GetGlobalIndex (pindex[locelements.Get(i).PNum(j)]);
mesh.AddVolumeElement (locelements.Get(i));
stat.cntelem++;
}
for(int i = oldnf; i < locfaces.Size(); i++)
{
for (j = 1; j <= locfaces[i].GetNP(); j++)
locfaces[i].PNum(j) =
pindex[locfaces[i].PNum(j)];
// (*testout) << "add face " << locfaces.Get(i) << endl;
adfront->AddFace (locfaces[i]);
}
for(int i = 1; i <= delfaces.Size(); i++)
adfront->DeleteFace (findex[delfaces.Get(i)-1]);
}
else
{
adfront->IncrementClass (findex[0]);
if (impossible && mp.check_impossible)
{
(*testout) << "skip face since it is impossible" << endl;
for (j = 0; j < 100; j++)
adfront->IncrementClass (findex[0]);
}
}
locelements.SetSize (0);
delpoints.SetSize(0);
delfaces.SetSize(0);
if (stat.qualclass >= mp.giveuptol)
break;
}
PrintMessage (5, ""); // line feed after statistics
for(int i = 1; i <= ruleused.Size(); i++)
(*testout) << setw(4) << ruleused.Get(i)
<< " times used rule " << rules.Get(i) -> Name() << endl;
if (!mp.baseelnp && adfront->Empty())
return MESHING3_OK;
if (mp.baseelnp && adfront->Empty (mp.baseelnp))
return MESHING3_OK;
if (stat.vol < -1e-15)
return MESHING3_NEGVOL;
return MESHING3_NEGVOL;
}
enum blocktyp { BLOCKUNDEF, BLOCKINNER, BLOCKBOUND, BLOCKOUTER };
void Meshing3 :: BlockFill (Mesh & mesh, double gh)
{
static Timer t("Mesing3::BlockFill"); RegionTimer reg(t);
PrintMessage (3, "Block-filling called (obsolete) ");
int i, j(0), i1, i2, i3, j1, j2, j3;
int n1, n2, n3, n, min1, min2, min3, max1, max2, max3;
int changed, filled;
double xmin(0), xmax(0), ymin(0), ymax(0), zmin(0), zmax(0);
double xminb, xmaxb, yminb, ymaxb, zminb, zmaxb;
//double rad = 0.7 * gh;
for(int i = 1; i <= adfront->GetNP(); i++)
{
const Point3d & p = adfront->GetPoint(PointIndex(i));
if (i == 1)
{
xmin = xmax = p.X();
ymin = ymax = p.Y();
zmin = zmax = p.Z();
}
else
{
if (p.X() < xmin) xmin = p.X();
if (p.X() > xmax) xmax = p.X();
if (p.Y() < ymin) ymin = p.Y();
if (p.Y() > ymax) ymax = p.Y();
if (p.Z() < zmin) zmin = p.Z();
if (p.Z() > zmax) zmax = p.Z();
}
}
xmin -= 5 * gh;
ymin -= 5 * gh;
zmin -= 5 * gh;
n1 = int ((xmax-xmin) / gh + 5);
n2 = int ((ymax-ymin) / gh + 5);
n3 = int ((zmax-zmin) / gh + 5);
n = n1 * n2 * n3;
PrintMessage (5, "n1 = ", n1, " n2 = ", n2, " n3 = ", n3);
NgArray<blocktyp> inner(n);
NgArray<PointIndex> pointnr(n);
NgArray<int> frontpointnr(n);
// initialize inner to 1
for(int i = 1; i <= n; i++)
inner.Elem(i) = BLOCKUNDEF;
// set blocks cutting surfaces to 0
for(int i = 1; i <= adfront->GetNF(); i++)
{
const MiniElement2d & el = adfront->GetFace(i);
xminb = xmax; xmaxb = xmin;
yminb = ymax; ymaxb = ymin;
zminb = zmax; zmaxb = zmin;
for (j = 1; j <= 3; j++)
{
const Point3d & p = adfront->GetPoint (el.PNum(j));
if (p.X() < xminb) xminb = p.X();
if (p.X() > xmaxb) xmaxb = p.X();
if (p.Y() < yminb) yminb = p.Y();
if (p.Y() > ymaxb) ymaxb = p.Y();
if (p.Z() < zminb) zminb = p.Z();
if (p.Z() > zmaxb) zmaxb = p.Z();
}
double filldist = 0.2; // globflags.GetNumFlag ("filldist", 0.4);
xminb -= filldist * gh;
xmaxb += filldist * gh;
yminb -= filldist * gh;
ymaxb += filldist * gh;
zminb -= filldist * gh;
zmaxb += filldist * gh;
min1 = int ((xminb - xmin) / gh) + 1;
max1 = int ((xmaxb - xmin) / gh) + 1;
min2 = int ((yminb - ymin) / gh) + 1;
max2 = int ((ymaxb - ymin) / gh) + 1;
min3 = int ((zminb - zmin) / gh) + 1;
max3 = int ((zmaxb - zmin) / gh) + 1;
for (i1 = min1; i1 <= max1; i1++)
for (i2 = min2; i2 <= max2; i2++)
for (i3 = min3; i3 <= max3; i3++)
inner.Elem(i3 + (i2-1) * n3 + (i1-1) * n2 * n3) = BLOCKBOUND;
}
while (1)
{
int undefi = 0;
Point3d undefp;
for (i1 = 1; i1 <= n1 && !undefi; i1++)
for (i2 = 1; i2 <= n2 && !undefi; i2++)
for (i3 = 1; i3 <= n3 && !undefi; i3++)
{
i = i3 + (i2-1) * n3 + (i1-1) * n2 * n3;
if (inner.Elem(i) == BLOCKUNDEF)
{
undefi = i;
undefp.X() = xmin + (i1-0.5) * gh;
undefp.Y() = ymin + (i2-0.5) * gh;
undefp.Z() = zmin + (i3-0.5) * gh;
}
}
if (!undefi)
break;
// PrintMessage (5, "Test point: ", undefp);
if (adfront -> Inside (undefp))
{
// (*mycout) << "inner" << endl;
inner.Elem(undefi) = BLOCKINNER;
}
else
{
// (*mycout) << "outer" << endl;
inner.Elem(undefi) = BLOCKOUTER;
}
do
{
changed = 0;
for (i1 = 1; i1 <= n1; i1++)
for (i2 = 1; i2 <= n2; i2++)
for (i3 = 1; i3 <= n3; i3++)
{
i = i3 + (i2-1) * n3 + (i1-1) * n2 * n3;
for (int k = 1; k <= 3; k++)
{
switch (k)
{
case 1: j = i + n2 * n3; break;
case 2: j = i + n3; break;
case 3: j = i + 1; break;
}
if (j > n1 * n2 * n3) continue;
if (inner.Elem(i) == BLOCKOUTER && inner.Elem(j) == BLOCKUNDEF)
{
changed = 1;
inner.Elem(j) = BLOCKOUTER;
}
if (inner.Elem(j) == BLOCKOUTER && inner.Elem(i) == BLOCKUNDEF)
{
changed = 1;
inner.Elem(i) = BLOCKOUTER;
}
if (inner.Elem(i) == BLOCKINNER && inner.Elem(j) == BLOCKUNDEF)
{
changed = 1;
inner.Elem(j) = BLOCKINNER;
}
if (inner.Elem(j) == BLOCKINNER && inner.Elem(i) == BLOCKUNDEF)
{
changed = 1;
inner.Elem(i) = BLOCKINNER;
}
}
}
}
while (changed);
}
filled = 0;
for(int i = 1; i <= n; i++)
if (inner.Elem(i) == BLOCKINNER)
{
filled++;
}
PrintMessage (5, "Filled blocks: ", filled);
for(int i = 1; i <= n; i++)
{
pointnr.Elem(i) = PointIndex::INVALID;
frontpointnr.Elem(i) = 0;
}
for (i1 = 1; i1 <= n1-1; i1++)
for (i2 = 1; i2 <= n2-1; i2++)
for (i3 = 1; i3 <= n3-1; i3++)
{
i = i3 + (i2-1) * n3 + (i1-1) * n2 * n3;
if (inner.Elem(i) == BLOCKINNER)
{
for (j1 = i1; j1 <= i1+1; j1++)
for (j2 = i2; j2 <= i2+1; j2++)
for (j3 = i3; j3 <= i3+1; j3++)
{
j = j3 + (j2-1) * n3 + (j1-1) * n2 * n3;
if (!pointnr.Get(j).IsValid())
{
Point3d hp(xmin + (j1-1) * gh,
ymin + (j2-1) * gh,
zmin + (j3-1) * gh);
pointnr.Elem(j) = mesh.AddPoint (hp);
frontpointnr.Elem(j) =
AddPoint (hp, pointnr.Elem(j));
}
}
}
}
for (i1 = 2; i1 <= n1-1; i1++)
for (i2 = 2; i2 <= n2-1; i2++)
for (i3 = 2; i3 <= n3-1; i3++)
{
i = i3 + (i2-1) * n3 + (i1-1) * n2 * n3;
if (inner.Elem(i) == BLOCKINNER)
{
int pn[9];
pn[1] = pointnr.Get(i);
pn[2] = pointnr.Get(i+1);
pn[3] = pointnr.Get(i+n3);
pn[4] = pointnr.Get(i+n3+1);
pn[5] = pointnr.Get(i+n2*n3);
pn[6] = pointnr.Get(i+n2*n3+1);
pn[7] = pointnr.Get(i+n2*n3+n3);
pn[8] = pointnr.Get(i+n2*n3+n3+1);
static int elind[][4] =
{
{ 1, 8, 2, 4 },
{ 1, 8, 4, 3 },
{ 1, 8, 3, 7 },
{ 1, 8, 7, 5 },
{ 1, 8, 5, 6 },
{ 1, 8, 6, 2 }
};
for (j = 1; j <= 6; j++)
{
Element el(4);
for (int k = 1; k <= 4; k++)
el.PNum(k) = pn[elind[j-1][k-1]];
mesh.AddVolumeElement (el);
}
}
}
for (i1 = 2; i1 <= n1-1; i1++)
for (i2 = 2; i2 <= n2-1; i2++)
for (i3 = 2; i3 <= n3-1; i3++)
{
i = i3 + (i2-1) * n3 + (i1-1) * n2 * n3;
if (inner.Elem(i) == BLOCKINNER)
{
int pi1(0), pi2(0), pi3(0), pi4(0);
int pn1 = frontpointnr.Get(i);
int pn2 = frontpointnr.Get(i+1);
int pn3 = frontpointnr.Get(i+n3);
int pn4 = frontpointnr.Get(i+n3+1);
int pn5 = frontpointnr.Get(i+n2*n3);
int pn6 = frontpointnr.Get(i+n2*n3+1);
int pn7 = frontpointnr.Get(i+n2*n3+n3);
int pn8 = frontpointnr.Get(i+n2*n3+n3+1);
for (int k = 1; k <= 6; k++)
{
switch (k)
{
case 1: // j3 = i3+1
j = i + 1;
pi1 = pn2;
pi2 = pn6;
pi3 = pn4;
pi4 = pn8;
break;
case 2: // j3 = i3-1
j = i - 1;
pi1 = pn1;
pi2 = pn3;
pi3 = pn5;
pi4 = pn7;
break;
case 3: // j2 = i2+1
j = i + n3;
pi1 = pn3;
pi2 = pn4;
pi3 = pn7;
pi4 = pn8;
break;
case 4: // j2 = i2-1
j = i - n3;
pi1 = pn1;
pi2 = pn5;
pi3 = pn2;
pi4 = pn6;
break;
case 5: // j1 = i1+1
j = i + n3*n2;
pi1 = pn5;
pi2 = pn7;
pi3 = pn6;
pi4 = pn8;
break;
case 6: // j1 = i1-1
j = i - n3*n2;
pi1 = pn1;
pi2 = pn2;
pi3 = pn3;
pi4 = pn4;
break;
}
if (inner.Get(j) == BLOCKBOUND)
{
MiniElement2d face;
face.PNum(1) = pi4;
face.PNum(2) = pi1;
face.PNum(3) = pi3;
AddBoundaryElement (face);
face.PNum(1) = pi1;
face.PNum(2) = pi4;
face.PNum(3) = pi2;
AddBoundaryElement (face);
}
}
}
}
}
/*
static const AdFront3 * locadfront;
static int TestInner (const Point3d & p)
{
return locadfront->Inside (p);
}
static int TestSameSide (const Point3d & p1, const Point3d & p2)
{
return locadfront->SameSide (p1, p2);
}
*/
void Meshing3 :: BlockFillLocalH (Mesh & mesh,
const MeshingParameters & mp)
{
static Timer t("Mesing3::BlockFillLocalH"); RegionTimer reg(t);
double filldist = mp.filldist;
// (*testout) << "blockfill local h" << endl;
// (*testout) << "rel filldist = " << filldist << endl;
PrintMessage (3, "blockfill local h");
NgArray<Point<3> > npoints;
adfront -> CreateTrees();
Box<3> bbox ( Box<3>::EMPTY_BOX );
double maxh = 0;
for (int i = 1; i <= adfront->GetNF(); i++)
{
const MiniElement2d & el = adfront->GetFace(i);
for (int j = 1; j <= 3; j++)
{
const Point3d & p1 = adfront->GetPoint (el.PNumMod(j));
const Point3d & p2 = adfront->GetPoint (el.PNumMod(j+1));
double hi = Dist (p1, p2);
if (hi > maxh) maxh = hi;
bbox.Add (p1);
}
}
Point3d mpmin = bbox.PMin();
Point3d mpmax = bbox.PMax();
Point3d mpc = Center (mpmin, mpmax);
double d = max3(mpmax.X()-mpmin.X(),
mpmax.Y()-mpmin.Y(),
mpmax.Z()-mpmin.Z()) / 2;
mpmin = mpc - Vec3d (d, d, d);
mpmax = mpc + Vec3d (d, d, d);
Box3d meshbox (mpmin, mpmax);
LocalH loch2 (mpmin, mpmax, 1);
if (mp.maxh < maxh) maxh = mp.maxh;
auto loch_ptr = mesh.LocalHFunction().Copy(bbox);
auto & loch = *loch_ptr;
bool changed;
static Timer t1("loop1");
t1.Start();
do
{
loch.ClearFlags();
static Timer tbox("adfront-bbox");
tbox.Start();
for (int i = 1; i <= adfront->GetNF(); i++)
{
const MiniElement2d & el = adfront->GetFace(i);
Box<3> bbox (adfront->GetPoint (el[0]));
bbox.Add (adfront->GetPoint (el[1]));
bbox.Add (adfront->GetPoint (el[2]));
double filld = filldist * bbox.Diam();
bbox.Increase (filld);
loch.CutBoundary (bbox); // .PMin(), bbox.PMax());
}
tbox.Stop();
// locadfront = adfront;
loch.FindInnerBoxes (adfront, NULL);
npoints.SetSize(0);
loch.GetInnerPoints (npoints);
changed = false;
for (int i = 1; i <= npoints.Size(); i++)
{
if (loch.GetH(npoints.Get(i)) > 1.5 * maxh)
{
loch.SetH (npoints.Get(i), maxh);
changed = true;
}
}
}
while (changed);
t1.Stop();
if (debugparam.slowchecks)
(*testout) << "Blockfill with points: " << endl;
for (int i = 1; i <= npoints.Size(); i++)
{
if (meshbox.IsIn (npoints.Get(i)))
{
PointIndex gpnum = mesh.AddPoint (npoints.Get(i));
adfront->AddPoint (npoints.Get(i), gpnum);
if (debugparam.slowchecks)
{
(*testout) << npoints.Get(i) << endl;
if (!adfront->Inside(npoints.Get(i)))
{
cout << "add outside point" << endl;
(*testout) << "outside" << endl;
}
}
}
}
// find outer points
static Timer tloch2("build loch2");
tloch2.Start();
loch2.ClearFlags();
for (int i = 1; i <= adfront->GetNF(); i++)
{
const MiniElement2d & el = adfront->GetFace(i);
Point3d pmin = adfront->GetPoint (el.PNum(1));
Point3d pmax = pmin;
for (int j = 2; j <= 3; j++)
{
const Point3d & p = adfront->GetPoint (el.PNum(j));
pmin.SetToMin (p);
pmax.SetToMax (p);
}
loch2.SetH (Center (pmin, pmax), Dist (pmin, pmax));
}
for (int i = 1; i <= adfront->GetNF(); i++)
{
const MiniElement2d & el = adfront->GetFace(i);
Point3d pmin = adfront->GetPoint (el.PNum(1));
Point3d pmax = pmin;
for (int j = 2; j <= 3; j++)
{
const Point3d & p = adfront->GetPoint (el.PNum(j));
pmin.SetToMin (p);
pmax.SetToMax (p);
}
double filld = filldist * Dist (pmin, pmax);
pmin = pmin - Vec3d (filld, filld, filld);
pmax = pmax + Vec3d (filld, filld, filld);
// loch2.CutBoundary (pmin, pmax);
loch2.CutBoundary (Box<3> (pmin, pmax)); // pmin, pmax);
}
tloch2.Stop();
// locadfront = adfront;
loch2.FindInnerBoxes (adfront, NULL);
npoints.SetSize(0);
loch2.GetOuterPoints (npoints);
for (int i = 1; i <= npoints.Size(); i++)
{
if (meshbox.IsIn (npoints.Get(i)))
{
PointIndex gpnum = mesh.AddPoint (npoints.Get(i));
adfront->AddPoint (npoints.Get(i), gpnum);
}
}
}
}