#include <mystdlib.h> #include "meshing.hpp" namespace netgen { #include "hpref_segm.hpp" #include "hpref_trig.hpp" #include "hpref_quad.hpp" #include "hpref_tet.hpp" #include "hpref_prism.hpp" #include "hpref_hex.hpp" #include "hpref_pyramid.hpp" #include "classifyhpel.hpp" void HPRefElement :: Reset(void) { np = 8; for (int i = 0; i < 8; i++) { pnums[i] = -1; param[i][0] = param[i][1] = param[i][2] = 0; } domin=-1; domout=-1; // he: levelx = 0; levely = 0; levelz = 0; } HPRefElement :: HPRefElement () { Reset(); } HPRefElement :: HPRefElement(Element & el) : type(HP_NONE), index(el.GetIndex()), levelx(0), levely(0), levelz(0), np(el.GetNV()), domin(-1), domout(-1) //domin,out for segments { //Reset(); for (int i=0; i<np ; i++) pnums[i] = el[i]; const Point3d * points = MeshTopology :: GetVertices (el.GetType()); for(int i=0;i<np;i++) for(int l=0;l<3;l++) param[i][l] = points[i].X(l+1); } HPRefElement :: HPRefElement(Element2d & el) : type(HP_NONE), index(el.GetIndex()), levelx(0), levely(0), levelz(0), np(el.GetNV()), domin(-1), domout(-1) //domin,out for segments { //Reset(); for (int i=0; i<np ; i++) pnums[i] = el[i]; const Point3d * points = MeshTopology :: GetVertices (el.GetType()); for(int i=0;i<np;i++) for(int l=0;l<3;l++) param[i][l] = points[i].X(l+1); } HPRefElement :: HPRefElement(Segment & el) : type(HP_NONE), levelx(0), levely(0), levelz(0), np(2), domin(el.domin), domout(el.domout), singedge_left(el.singedge_left), singedge_right(el.singedge_right) { //Reset(); for (int i=0; i<np ; i++) pnums[i] = el[i]; const Point3d * points = MeshTopology :: GetVertices (SEGMENT); for(int i=0;i<np;i++) for(int l=0;l<3;l++) param[i][l] = points[i].X(l+1); /* for (int i=0; i<np; i++) { param[i][0] = i; param[i][1] = -1; param[i][2] = -1; } */ } HPRefElement :: HPRefElement(HPRefElement & el) : type(el.type), index(el.index), si(el.si), levelx(el.levelx), levely(el.levely), levelz(el.levelz), np(el.np), coarse_elnr(el.coarse_elnr), domin(el.domin), domout(el.domout), singedge_left(el.singedge_left), singedge_right(el.singedge_right) { //Reset(); for (int i=0; i<np ; i++) { pnums[i] = el[i]; for(int l=0; l<3; l++) param[i][l] = el.param[i][l]; } } void HPRefElement :: SetType (HPREF_ELEMENT_TYPE t) { type = t; switch(type) { case HP_SEGM: np=2; break; case HP_TRIG: np=3; break; case HP_QUAD: np=4; break; case HP_TET: np=4; break; case HP_PRISM: np=6; break; case HP_PYRAMID: np=5; break; case HP_HEX: np=8; break; default: cerr << "HPRefElement: illegal type " << type << endl; throw NgException ("HPRefElement::SetType: illegal type"); } for(int k = 0; k < 8;k++) { pnums[k]=0; for(int l = 0; l < 3; l++) param[k][l]=0.; } } std::map<HPREF_ELEMENT_TYPE, HPRef_Struct*> & GetHPRegistry()\ { static std::map<HPREF_ELEMENT_TYPE, HPRef_Struct*> registry; return registry; } HPRef_Struct * Get_HPRef_Struct (HPREF_ELEMENT_TYPE type) { HPRef_Struct * hps = NULL; /* if (type >= 100 && type < 1000) if (type != HP_TET_1F_0E_1VA) type = HP_NONETET; */ switch (type) { case HP_SEGM: hps = &refsegm; break; case HP_SEGM_SINGCORNERL: hps = &refsegm_scl; break; case HP_SEGM_SINGCORNERR: hps = &refsegm_scr; break; case HP_SEGM_SINGCORNERS: hps = &refsegm_sc2; break; case HP_TRIG: hps = &reftrig; break; case HP_TRIG_SINGCORNER: hps = &reftrig_singcorner; break; case HP_TRIG_SINGCORNER12: hps = &reftrig_singcorner12; break; case HP_TRIG_SINGCORNER123: hps = &reftrig_singcorner123; break; case HP_TRIG_SINGCORNER123_2D: hps = &reftrig_singcorner123_2D; break; case HP_TRIG_SINGEDGE: hps = &reftrig_singedge; break; case HP_TRIG_SINGEDGECORNER1: hps = &reftrig_singedgecorner1; break; case HP_TRIG_SINGEDGECORNER2: hps = &reftrig_singedgecorner2; break; case HP_TRIG_SINGEDGECORNER12: hps = &reftrig_singedgecorner12; break; case HP_TRIG_SINGEDGECORNER3: hps = &reftrig_singedgecorner3; break; case HP_TRIG_SINGEDGECORNER13: hps = &reftrig_singedgecorner13; break; case HP_TRIG_SINGEDGECORNER23: hps = &reftrig_singedgecorner23; break; case HP_TRIG_SINGEDGECORNER123: hps = &reftrig_singedgecorner123; break; case HP_TRIG_SINGEDGES: hps = &reftrig_singedges; break; case HP_TRIG_SINGEDGES2: hps = &reftrig_singedges2; break; case HP_TRIG_SINGEDGES3: hps = &reftrig_singedges3; break; case HP_TRIG_SINGEDGES23: hps = &reftrig_singedges23; break; case HP_TRIG_3SINGEDGES: hps = &reftrig_3singedges; break; case HP_TRIG_ALFELD: hps = &reftrig_Alfeld; break; case HP_TRIG_POWELL: hps = &reftrig_Powell; break; case HP_QUAD: hps = &refquad; break; case HP_DUMMY_QUAD_SINGCORNER: hps = &refdummyquad_singcorner; break; case HP_QUAD_SINGCORNER: hps = &refquad_singcorner; break; case HP_QUAD_SINGEDGE: hps = &refquad_singedge; break; case HP_QUAD_0E_2VA: hps = &refquad_0e_2va; break; case HP_QUAD_0E_2VB: hps = &refquad_0e_2vb; break; case HP_QUAD_0E_3V: hps = &refquad_0e_3v; break; case HP_QUAD_0E_4V: hps = &refquad_0e_4v; break; case HP_QUAD_1E_1VA: hps = &refquad_1e_1va; break; case HP_QUAD_1E_1VB: hps = &refquad_1e_1vb; break; case HP_QUAD_1E_1VC: hps = &refquad_1e_1vc; break; case HP_QUAD_1E_1VD: hps = &refquad_1e_1vd; break; case HP_QUAD_1E_2VA: hps = &refquad_1e_2va; break; case HP_QUAD_1E_2VB: hps = &refquad_1e_2vb; break; case HP_QUAD_1E_2VC: hps = &refquad_1e_2vc; break; case HP_QUAD_1E_2VD: hps = &refquad_1e_2vd; break; case HP_QUAD_1E_2VE: hps = &refquad_1e_2ve; break; case HP_QUAD_1E_2VF: hps = &refquad_1e_2vf; break; case HP_QUAD_1E_3VA: hps = &refquad_1e_3va; break; case HP_QUAD_1E_3VB: hps = &refquad_1e_3vb; break; case HP_QUAD_1E_3VC: hps = &refquad_1e_3vc; break; case HP_QUAD_1E_3VD: hps = &refquad_1e_3vd; break; case HP_QUAD_1E_4V: hps = &refquad_1e_4v; break; case HP_QUAD_2E: hps = &refquad_2e; break; case HP_QUAD_2E_1VA: hps = &refquad_2e_1va; break; case HP_QUAD_2E_1VB: hps = &refquad_2e_1vb; break; case HP_QUAD_2E_1VC: hps = &refquad_2e_1vc; break; case HP_QUAD_2E_2VA: hps = &refquad_2e_2va; break; case HP_QUAD_2E_2VB: hps = &refquad_2e_2vb; break; case HP_QUAD_2E_2VC: hps = &refquad_2e_2vc; break; case HP_QUAD_2E_3V: hps = &refquad_2e_3v; break; case HP_QUAD_2EB_0V: hps = &refquad_2eb_0v; break; case HP_QUAD_2EB_1VA: hps = &refquad_2eb_1va; break; case HP_QUAD_2EB_1VB: hps = &refquad_2eb_1vb; break; case HP_QUAD_2EB_2VA: hps = &refquad_2eb_2va; break; case HP_QUAD_2EB_2VB: hps = &refquad_2eb_2vb; break; case HP_QUAD_2EB_2VC: hps = &refquad_2eb_2vc; break; case HP_QUAD_2EB_2VD: hps = &refquad_2eb_2vd; break; case HP_QUAD_2EB_3VA: hps = &refquad_2eb_3va; break; case HP_QUAD_2EB_3VB: hps = &refquad_2eb_3vb; break; case HP_QUAD_2EB_4V: hps = &refquad_2eb_4v; break; case HP_QUAD_3E: hps = &refquad_3e; break; case HP_QUAD_3E_3VA: hps = &refquad_3e_3va; break; case HP_QUAD_3E_3VB: hps = &refquad_3e_3vb; break; case HP_QUAD_3E_4V: hps = &refquad_3e_4v; break; case HP_QUAD_4E: hps = &refquad_4e; break; case HP_TET: hps = &reftet; break; case HP_NONETET: hps = &refnonetet; break; case HP_TET_0E_1V: hps = &reftet_0e_1v; break; case HP_TET_0E_2V: hps = &reftet_0e_2v; break; case HP_TET_0E_3V: hps = &reftet_0e_3v; break; case HP_TET_0E_4V: hps = &reftet_0e_4v; break; case HP_TET_1E_0V: hps = &reftet_1e_0v; break; case HP_TET_1E_1VA: hps = &reftet_1e_1va; break; case HP_TET_1E_1VB: hps = &reftet_1e_1vb; break; case HP_TET_1E_2VA: hps = &reftet_1e_2va; break; case HP_TET_1E_2VB: hps = &reftet_1e_2vb; break; case HP_TET_1E_2VC: hps = &reftet_1e_2vc; break; case HP_TET_1E_2VD: hps = &reftet_1e_2vd; break; case HP_TET_1E_3VA: hps = &reftet_1e_3va; break; case HP_TET_1E_3VB: hps = &reftet_1e_3vb; break; case HP_TET_1E_4V: hps = &reftet_1e_4v; break; case HP_TET_2EA_0V: hps = &reftet_2ea_0v; break; case HP_TET_2EA_1VB: hps = &reftet_2ea_1vb; break; case HP_TET_2EA_1VC: hps = &reftet_2ea_1vc; break; case HP_TET_2EA_1VA: hps = &reftet_2ea_1va; break; case HP_TET_2EA_2VA: hps = &reftet_2ea_2va; break; case HP_TET_2EA_2VB: hps = &reftet_2ea_2vb; break; case HP_TET_2EA_2VC: hps = &reftet_2ea_2vc; break; case HP_TET_2EA_3V: hps = &reftet_2ea_3v; break; case HP_TET_2EB_0V: hps = &reftet_2eb_0v; break; case HP_TET_2EB_1V: hps = &reftet_2eb_1v; break; case HP_TET_2EB_2VA: hps = &reftet_2eb_2va; break; case HP_TET_2EB_2VB: hps = &reftet_2eb_2vb; break; case HP_TET_2EB_2VC: hps = &reftet_2eb_2vc; break; case HP_TET_2EB_3V: hps = &reftet_2eb_3v; break; case HP_TET_2EB_4V: hps = &reftet_2eb_4v; break; case HP_TET_3EA_0V: hps = &reftet_3ea_0v; break; case HP_TET_3EA_1V: hps = &reftet_3ea_1v; break; case HP_TET_3EA_2V: hps = &reftet_3ea_2v; break; case HP_TET_3EA_3V: hps = &reftet_3ea_3v; break; case HP_TET_3EB_0V: hps = &reftet_3eb_0v; break; case HP_TET_3EB_1V: hps = &reftet_3eb_1v; break; case HP_TET_3EB_2V: hps = &reftet_3eb_2v; break; case HP_TET_3EC_0V: hps = &reftet_3ec_0v; break; case HP_TET_3EC_1V: hps = &reftet_3ec_1v; break; case HP_TET_3EC_2V: hps = &reftet_3ec_2v; break; case HP_TET_3ED_3V: hps = &reftet_3ed_3v; break; case HP_TET_1F_0E_0V: hps = &reftet_1f_0e_0v; break; // case HP_TET_1F_0E_1VA: // hps = &reftet_1f_0e_1va; break; case HP_TET_1F_0E_1VB: hps = &reftet_1f_0e_1vb; break; case HP_TET_1F_0E_2V: hps = &reftet_1f_0e_2v; break; // case HP_TET_1F_0E_3V: // hps = &reftet_1f_0e_3v; break; case HP_TET_1F_1EA_0V: hps = &reftet_1f_1ea_0v; break; case HP_TET_1F_1EB_0V: hps = &reftet_1f_1eb_0v; break; case HP_TET_1F_1E_1VA: hps = &reftet_1f_1e_1va; break; case HP_TET_1F_1E_1VB: hps = &reftet_1f_1e_1vb; break; case HP_TET_1F_1E_2VA: hps = &reftet_1f_1e_2va; break; case HP_TET_1F_1E_2VB: hps = &reftet_1f_1e_2vb; break; case HP_TET_1F_2E_0VA: hps = &reftet_1f_2e_0va; break; case HP_TET_1F_2E_0VB: hps = &reftet_1f_2e_0vb; break; case HP_TET_2F_0E_0V: hps = &reftet_2f_0e_0v; break; case HP_TET_2F_0E_1V: hps = &reftet_2f_0e_1v; break; case HP_TET_2F_1E_0VA: hps = &reftet_2f_1e_0va; break; case HP_TET_2F_1E_0VB: hps = &reftet_2f_1e_0vb; break; case HP_TET_3F_0E_0V: hps = &reftet_3f_0e_0v; break; case HP_PRISM: hps = &refprism; break; case HP_PRISM_SINGEDGE: hps = &refprism_singedge; break; // case HP_PRISM_SINGEDGE_H1: // hps = &refprism_singedge_h1; break; // case HP_PRISM_SINGEDGE_H12: // hps = &refprism_singedge_h12; break; case HP_PRISM_SINGEDGE_V12: hps = &refprism_singedge_v12; break; case HP_PRISM_1FA_0E_0V: hps = &refprism_1fa_0e_0v; break; case HP_PRISM_2FA_0E_0V: hps = &refprism_2fa_0e_0v; break; case HP_PRISM_1FB_0E_0V: hps = &refprism_1fb_0e_0v; break; case HP_PRISM_1FB_1EA_0V: hps = &refprism_1fb_1ea_0v; break; case HP_PRISM_1FA_1E_0V: hps = &refprism_1fa_1e_0v; break; case HP_PRISM_2FA_1E_0V: hps = &refprism_2fa_1e_0v; break; case HP_PRISM_1FA_1FB_0E_0V: hps = &refprism_1fa_1fb_0e_0v; break; case HP_PRISM_2FA_1FB_0E_0V: hps = &refprism_2fa_1fb_0e_0v; break; case HP_PRISM_1FA_1FB_1EA_0V: hps = &refprism_1fa_1fb_1ea_0v; break; case HP_PRISM_1FA_1FB_1EB_0V: hps = &refprism_1fa_1fb_1eb_0v; break; case HP_PRISM_2FA_1FB_1EA_0V: hps = &refprism_2fa_1fb_1ea_0v; break; case HP_PRISM_1FB_1EC_0V: hps = &refprism_1fb_1ec_0v; break; case HP_PRISM_1FA_1FB_1EC_0V: hps = &refprism_1fa_1fb_1ec_0v; break; case HP_PRISM_2FA_1FB_1EC_0V: hps = &refprism_2fa_1fb_1ec_0v; break; case HP_PRISM_1FB_2EA_0V: hps = &refprism_1fb_2ea_0v; break; case HP_PRISM_1FA_1FB_2EA_0V: hps = &refprism_1fa_1fb_2ea_0v; break; case HP_PRISM_2FA_1FB_2EA_0V: hps = &refprism_2fa_1fb_2ea_0v; break; case HP_PRISM_1FB_2EB_0V: hps = &refprism_1fb_2eb_0v; break; case HP_PRISM_1FA_1FB_2EB_0V: hps = &refprism_1fa_1fb_2eb_0v; break; case HP_PRISM_1FA_1FB_2EC_0V: hps = &refprism_1fa_1fb_2ec_0v; break; case HP_PRISM_2FA_1FB_2EB_0V: hps = &refprism_2fa_1fb_2eb_0v; break; case HP_PRISM_1FB_3E_0V: hps = &refprism_1fb_3e_0v; break; case HP_PRISM_1FA_1FB_3E_0V: hps = &refprism_1fa_1fb_3e_0v; break; case HP_PRISM_2FA_1FB_3E_0V: hps = &refprism_2fa_1fb_3e_0v; break; case HP_PRISM_2FB_0E_0V: hps = &refprism_2fb_0e_0v; break; case HP_PRISM_1FA_2FB_0E_0V: hps = &refprism_1fa_2fb_0e_0v; break; case HP_PRISM_2FA_2FB_0E_0V: hps = &refprism_2fa_2fb_0e_0v; break; case HP_PRISM_2FB_1EC_0V: hps = &refprism_2fb_1ec_0v; break; case HP_PRISM_1FA_2FB_1EC_0V: hps = &refprism_1fa_2fb_1ec_0v; break; case HP_PRISM_2FA_2FB_1EC_0V: hps = &refprism_2fa_2fb_1ec_0v; break; case HP_PRISM_1FA_2FB_1EB_0V: hps = &refprism_1fa_2fb_1eb_0v; break; case HP_PRISM_2FB_3E_0V: hps = &refprism_2fb_3e_0v; break; case HP_PRISM_1FA_2FB_3E_0V: hps = &refprism_1fa_2fb_3e_0v; break; case HP_PRISM_2FA_2FB_3E_0V: hps = &refprism_2fa_2fb_3e_0v; break; case HP_PRISM_1FA_2E_0V: hps = &refprism_1fa_2e_0v; break; case HP_PRISM_2FA_2E_0V: hps = &refprism_2fa_2e_0v; break; case HP_PRISM_3E_0V: hps = &refprism_3e_0v; break; case HP_PRISM_1FA_3E_0V: hps = &refprism_1fa_3e_0v; break; case HP_PRISM_2FA_3E_0V: hps = &refprism_2fa_3e_0v; break; case HP_PRISM_3FB_0V: hps = &refprism_3fb_0v; break; case HP_PRISM_1FA_3FB_0V: hps = &refprism_1fa_3fb_0v; break; case HP_PRISM_2FA_3FB_0V: hps = &refprism_2fa_3fb_0v; break; // case HP_PRISM_3E_4EH: // hps = &refprism_3e_4eh; break; /*case HP_PRISM_1FB_1EB_0V: hps = &refprism_1fb_1eb_0v; break; case HP_PRISM_2F_0E_0V: hps = &refprism_2f_0e_0v; break; */ case HP_PYRAMID: hps = &refpyramid; break; case HP_PYRAMID_0E_1V: hps = &refpyramid_0e_1v; break; case HP_PYRAMID_EDGES: hps = &refpyramid_edges; break; case HP_PYRAMID_1FB_0E_0V: hps = &refpyramid_1fb_0e_0v; break; case HP_PYRAMID_1FB_0E_1VA: hps = &refpyramid_1fb_0e_1va; break; case HP_HEX: hps = &refhex; break; case HP_HEX_0E_1V: hps = &refhex_0e_1v; break; case HP_HEX_1E_1V: hps = &refhex_1e_1v; break; case HP_HEX_1E_0V: hps = &refhex_1e_0v; break; case HP_HEX_3E_0V: hps = &refhex_3e_0v; break; case HP_HEX_1F_0E_0V: hps = &refhex_1f_0e_0v; break; case HP_HEX_1FA_1FB_0E_0V: hps = &refhex_1fa_1fb_0e_0v; break; case HP_HEX7: hps = &refhex7; break; case HP_HEX7_1FA: hps = &refhex7_1fa; break; case HP_HEX7_1FB: hps = &refhex7_1fb; break; default: { if (GetHPRegistry().count(type)) hps = GetHPRegistry()[type]; else hps = NULL; } } /* if (type != HP_TET_1E_4V && type != HP_TET_1E_2VD) { if (hps->geom == HP_TET) hps = &reftet; if (hps->geom == HP_TRIG) hps = &reftrig; } */ if (!hps) { cout << "Attention hps : hp-refinement not implemented for case " << type << endl; PrintSysError ("hp-refinement not implemented for case ", type); } return hps; } bool CheckSingularities(Mesh & mesh, INDEX_2_HASHTABLE<int> & edges, INDEX_2_HASHTABLE<int> & edgepoiclt_dom, NgBitArray & cornerpoint, NgBitArray & edgepoint, INDEX_3_HASHTABLE<int> & faces, INDEX_2_HASHTABLE<int> & face_edges, INDEX_2_HASHTABLE<int> & surf_edges, NgArray<int, PointIndex::BASE> & facepoint, int & levels, int & act_ref); bool ClassifyHPElements (Mesh & mesh, NgArray<HPRefElement> & elements, SplittingType split, int & act_ref, int & levels); void InitHPElements(Mesh & mesh, NgArray<HPRefElement> & elements) { for(ElementIndex i = 0; i < mesh.GetNE(); i++) { HPRefElement hpel(mesh[i]); hpel.coarse_elnr = i; switch (mesh[i].GetType()) { case PRISM: hpel.type = HP_PRISM; break; case HEX: hpel.type = HP_HEX; break; case TET: hpel.type = HP_TET; break; case PYRAMID: hpel.type = HP_PYRAMID; break; default: cerr << "HPRefElement: illegal elementtype (1) " << mesh[i].GetType() << endl; throw NgException ("HPRefElement: illegal elementtype (1)"); } elements.Append(hpel); } for(SurfaceElementIndex i = 0; i < mesh.GetNSE(); i++) { HPRefElement hpel(mesh[i]); hpel.coarse_elnr = i; switch(mesh[i].GetType()) { case TRIG: hpel.type = HP_TRIG; break; case QUAD: hpel.type = HP_QUAD; break; default: cerr << "HPRefElement: illegal elementtype (1b) " << mesh[i].GetType() << endl; throw NgException ("HPRefElement: illegal elementtype (1b)"); } elements.Append(hpel); } for(SegmentIndex i = 0; i < mesh.GetNSeg(); i++) { Segment & seg = mesh[i]; HPRefElement hpel(mesh[i]); hpel.coarse_elnr = i; hpel.type = HP_SEGM; // hpel.index = seg.edgenr + 10000*seg.si; hpel.index = seg.edgenr; hpel.si = seg.si; /* if(seg.edgenr >= 10000) { throw NgException("assumption that seg.edgenr < 10000 is wrong"); } */ elements.Append(hpel); } } /* ******************************* DoRefinement *************************************** */ void DoRefinement (Mesh & mesh, NgArray<HPRefElement> & elements, Refinement * ref, double fac1) { elements.SetAllocSize (5 * elements.Size()); INDEX_2_HASHTABLE<int> newpts(elements.Size()+1); INDEX_3_HASHTABLE<int> newfacepts(elements.Size()+1); double fac2 = max(0.001,min(1.0/3,fac1)); // factor for face points PrintMessage(3, " in HP-REFINEMENT with fac1 ", fac1); *testout << " in HP-REFINEMENT with fac1 " << fac1 << endl; int oldelsize = elements.Size(); for (int i = 0; i < oldelsize; i++) { HPRefElement & el = elements[i]; HPRef_Struct * hprs = Get_HPRef_Struct (el.type); if (!hprs) { cout << "Refinementstruct not defined for element " << el.type << endl; continue; } int j = 0; while (hprs->splitedges[j][0]) { INDEX_2 i2(el.pnums[hprs->splitedges[j][0]-1], el.pnums[hprs->splitedges[j][1]-1]); if (fac1 == 0.5) i2.Sort(); if (!newpts.Used (i2)) { Point<3> np; for( int l=0;l<3;l++) np(l) = (1-fac1)*mesh.Point(i2.I1())(l) + fac1 * mesh.Point(i2.I2())(l); int npi = mesh.AddPoint (np); newpts.Set (i2, npi); } j++; } j = 0; if (hprs->splitfaces) while (hprs->splitfaces[j][0]) { INDEX_3 i3(el.pnums[hprs->splitfaces[j][0]-1], el.pnums[hprs->splitfaces[j][1]-1], el.pnums[hprs->splitfaces[j][2]-1]); if (i3.I2() > i3.I3()) Swap (i3.I2(), i3.I3()); if (!newfacepts.Used (i3)) { Point<3> np; for( int l=0;l<3;l++) np(l) = (1-2*fac2)*mesh.Point(i3.I1())(l) + fac2*mesh.Point(i3.I2())(l) + fac2*mesh.Point(i3.I3())(l); int npi = mesh.AddPoint (np); newfacepts.Set (i3, npi); } j++; } } for (int i = 0; i < oldelsize; i++) { HPRefElement el = elements[i]; HPRef_Struct * hprs = Get_HPRef_Struct (el.type); int newlevel = el.levelx; int oldnp = 0; switch (hprs->geom) { case HP_SEGM: oldnp = 2; break; case HP_TRIG: oldnp = 3; break; case HP_QUAD: oldnp = 4; break; case HP_TET: oldnp = 4; break; case HP_PYRAMID: oldnp = 5; break; case HP_PRISM: oldnp = 6; break; case HP_HEX: oldnp = 8; break; case HP_HEX7: oldnp = 7; break; default: cerr << "HPRefElement: illegal type (3) " << hprs->geom << endl; throw NgException ("HPRefElement::SetType: illegal type (3)"); } if (el.type == HP_SEGM || el.type == HP_TRIG || el.type == HP_QUAD || el.type == HP_TET || el.type == HP_PRISM || el.type == HP_HEX || el.type == HP_HEX7 || el.type == HP_PYRAMID) newlevel = el.levelx; if (!hprs) continue; int newpnums[64]; double newparam[64][3]; int j; for (j = 0; j < oldnp; j++) { newpnums[j] = el.pnums[j]; for (int l = 0; l < 3; l++) newparam[j][l] = el.param[j][l]; } // split edges, incl. transferring curvature j = 0; while (hprs->splitedges[j][0]) { INDEX_2 i2(el.pnums[hprs->splitedges[j][0]-1], el.pnums[hprs->splitedges[j][1]-1]); if (fac1 == 0.5) i2.Sort(); int npi = newpts.Get(i2); newpnums[hprs->splitedges[j][2]-1] = npi; for (int l = 0; l < 3; l++) newparam[hprs->splitedges[j][2]-1][l] = (1-fac1) * el.param[hprs->splitedges[j][0]-1][l] + fac1 * el.param[hprs->splitedges[j][1]-1][l]; j++; } // split faces j = 0; if (hprs->splitfaces) while (hprs->splitfaces[j][0]) { INDEX_3 i3(el.pnums[hprs->splitfaces[j][0]-1], el.pnums[hprs->splitfaces[j][1]-1], el.pnums[hprs->splitfaces[j][2]-1]); if (i3.I2() > i3.I3()) Swap (i3.I2(), i3.I3()); int npi = newfacepts.Get(i3); newpnums[hprs->splitfaces[j][3]-1] = npi; for (int l = 0; l < 3; l++) newparam[hprs->splitfaces[j][3]-1][l] = (1-2*fac2) * el.param[hprs->splitfaces[j][0]-1][l] + fac2 * el.param[hprs->splitfaces[j][1]-1][l] + fac2 * el.param[hprs->splitfaces[j][2]-1][l]; j++; } // split elements j = 0; if (hprs->splitelements) while (hprs->splitelements[j][0]) { //int pi1 = el.pnums[hprs->splitelements[j][0]-1]; Point<3> np; for( int l=0;l<3;l++) np(l) = (1-3*fac1)* mesh.Point(el.pnums[hprs->splitelements[j][0]-1])(l) + fac1* mesh.Point(el.pnums[hprs->splitelements[j][1]-1])(l) + fac1* mesh.Point(el.pnums[hprs->splitelements[j][2]-1])(l) + fac1* mesh.Point(el.pnums[hprs->splitelements[j][3]-1])(l); int npi = mesh.AddPoint (np); newpnums[hprs->splitelements[j][4]-1] = npi; for (int l = 0; l < 3; l++) newparam[hprs->splitelements[j][4]-1][l] = (1-3*fac1) * el.param[hprs->splitelements[j][0]-1][l] + fac1 * el.param[hprs->splitelements[j][1]-1][l] + fac1 * el.param[hprs->splitelements[j][2]-1][l] + fac1 * el.param[hprs->splitelements[j][3]-1][l]; j++; } j = 0; /* *testout << " newpnums = "; for (int hi = 0; hi < 64; hi++) *testout << newpnums[hi] << " "; *testout << endl; */ while (hprs->neweltypes[j]) { HPRef_Struct * hprsnew = Get_HPRef_Struct (hprs->neweltypes[j]); HPRefElement newel(el); newel.type = hprs->neweltypes[j]; // newel.index = elements[i].index; // newel.coarse_elnr = elements[i].coarse_elnr; if (newel.type == HP_SEGM || newel.type == HP_TRIG || newel.type == HP_QUAD || newel.type == HP_TET || newel.type == HP_PRISM || newel.type == HP_HEX || newel.type == HP_HEX7 || newel.type == HP_PYRAMID) newel.levelx = newel.levely = newel.levelz = newlevel; else newel.levelx = newel.levely = newel.levelz = newlevel+1; switch(hprsnew->geom) { case HP_SEGM: newel.np=2; break; case HP_QUAD: newel.np=4; break; case HP_TRIG: newel.np=3; break; case HP_HEX: newel.np=8; break; case HP_HEX7: newel.np=7; break; case HP_PRISM: newel.np=6; break; case HP_TET: newel.np=4; break; case HP_PYRAMID: newel.np=5; break; default: throw NgException (string("hprefinement.cpp: illegal type")); } for (int k = 0; k < newel.np; k++) newel.pnums[k] = newpnums[hprs->newels[j][k]-1]; /* *testout << " newel pnums " ; for (int k = 0; k < newel.np; k++) *testout << newel.pnums[k] << "\t"; *testout << endl; */ for (int k = 0; k < newel.np; k++) { for (int l = 0; l < 3; l++) { newel.param[k][l] = newparam[hprs->newels[j][k]-1][l]; // *testout << newel.param[k][l] << " \t "; } // *testout << endl; } if (j == 0) elements[i] = newel; // overwrite old element else elements.Append (newel); j++; } } } /* ************************** DoRefineDummies ******************************** */ void DoRefineDummies (Mesh & mesh, NgArray<HPRefElement> & elements, Refinement * ref) { int oldelsize = elements.Size(); for (int i = 0; i < oldelsize; i++) { HPRefElement el = elements[i]; HPRef_Struct * hprs = Get_HPRef_Struct (el.type); if (!hprs) continue; if (el.type != HP_DUMMY_QUAD_SINGCORNER && el.type != HP_PYRAMID_EDGES && el.type != HP_PYRAMID_0E_1V && el.type != HP_HEX_0E_1V && el.type != HP_HEX_1E_1V && el.type != HP_HEX_1E_0V && el.type != HP_HEX_3E_0V ) continue; int newlevel = el.levelx; int newpnums[8]; int j; for (j = 0; j < 8; j++) newpnums[j] = el.pnums[j]; double newparam[8][3]; for (j = 0; j < 8; j++) for (int k = 0; k < 3; k++) newparam[j][k] = el.param[j][k]; j = 0; while (hprs->neweltypes[j]) { HPRef_Struct * hprsnew = Get_HPRef_Struct (hprs->neweltypes[j]); HPRefElement newel(el); switch(hprsnew->geom) { case HP_SEGM: newel.np=2; break; case HP_QUAD: newel.np=4; break; case HP_TRIG: newel.np=3; break; case HP_HEX: newel.np=8; break; case HP_PRISM: newel.np=6; break; case HP_TET: newel.np=4; break; case HP_PYRAMID: newel.np=5; break; default: cerr << "HPRefElement: illegal type (4) " << hprsnew->geom << endl; throw NgException ("HPRefElement: illegal type (4)"); } newel.type = hprs->neweltypes[j]; for (int k = 0; k < 8; k++) newel.pnums[k] = newpnums[hprs->newels[j][k]-1]; newel.index = el.index; newel.si = el.si; newel.coarse_elnr = el.coarse_elnr; newel.levelx = newel.levely = newel.levelz = newlevel; for (int k = 0; k < 8; k++) for (int l = 0; l < 3; l++) newel.param[k][l] = newparam[hprs->newels[j][k]-1][l]; if (j == 0) elements[i] = newel; else elements.Append (newel); j++; } } } void SubdivideDegeneratedHexes (Mesh & mesh, NgArray<HPRefElement> & elements, double fac1) { int oldne = elements.Size(); for (int i = 0; i < oldne; i++) if (Get_HPRef_Struct (elements[i].type)->geom == HP_HEX) { bool common = 0; for (int j = 0; j < 8; j++) for (int k = 0; k < j; k++) if (elements[i].pnums[j] == elements[i].pnums[k]) common = 1; if (common) { cout << " Degenerate Hex found " << endl; *testout << " Degenerate Hex found " << endl; HPRefElement el = elements[i]; HPRefElement newel = el; Point<3> center(0,0,0); double newparam[3] = { 0, 0, 0 }; for (int j = 0; j < 8; j++) { center += 0.125 * Vec<3>(mesh[el.pnums[j]]); // 0.125 originates form 8 points not from fac1; for (int l = 0; l < 3; l++) newparam[l] += 0.125 * el.param[j][l]; } int npi = mesh.AddPoint (center); const ELEMENT_FACE * faces = MeshTopology::GetFaces1 (HEX); for (int j = 0; j < 6; j++) { NgArray<int> pts; for (int k = 0; k < 4; k++) { bool same = 0; for (int l = 0; l < pts.Size(); l++) if (el.pnums[pts[l]] == el.pnums[faces[j][k]-1]) same = 1; if (!same) pts.Append (faces[j][k]-1); } if (pts.Size() == 3) // TrigFace -> TET { for (int k = 0; k < 3; k++) { newel.pnums[k] = el.pnums[pts[2-k]]; for (int l = 0; l < 3; l++) newel.param[k][l] = el.param[pts[2-k]][l]; } newel.pnums[3] = npi; for (int l = 0; l < 3; l++) newel.param[3][l] = newparam[l]; newel.type = HP_TET; newel.np = 4; } else { for (int k = 0; k < 4; k++) { newel.pnums[k] = el.pnums[pts[3-k]]; for (int l = 0; l < 3; l++) newel.param[k][l] = el.param[pts[3-k]][l]; } newel.pnums[4] = npi; for (int l = 0; l < 3; l++) newel.param[4][l] = newparam[l]; newel.type = HP_PYRAMID; newel.np = 5; } if (j == 0) elements[i] = newel; else elements.Append (newel); } /* const ELEMENT_EDGE * edges = MeshTopology::GetEdges (HEX); for(int k=0;k<12;k++) { int e[2]; for(int l=0;l<2;l++) e[l] = edges[k][l]-1; if(el.PNum(e[0]+1)!=el.PNum(e[1]+1)) { newel.SetType(HP_SEGM); for(int l=0;l<2;l++) { newel.pnums[0] = el.PNum(e[l]+1); newel.pnums[1] = npi; for(int j=0;j<3;j++) { // newel.param[0][j] = el.param[e[l]][j]; // newel.param[1][j] = newparam[j]; } elements.Append(newel); } newel.SetType(HP_TRIG); newel.pnums[0] = el.PNum(e[0]+1); newel.pnums[1] = el.PNum(e[1]+1); newel.pnums[2] = npi; *testout << "DEGHEX TRIG :: newpnums " << newel.pnums[0] << "\t" << newel.pnums[1] << "\t" << newel.pnums[2] << endl; cout << "DEGHEX TRIG :: newpnums " << newel.pnums[0] << "\t" << newel.pnums[1] << "\t" << newel.pnums[2] << endl; for(int j=0;j<3;j++) { // newel.param[0][j] = el.param[e[0]][j]; // newel.param[1][j] = el.param[e[1]][j]; // newel.param[2][j] = newparam[j]; } elements.Append(newel); } }*/ } } } void CalcStatistics (NgArray<HPRefElement> & elements) { return; #ifdef ABC int i, p; int nsegm = 0, ntrig = 0, nquad = 0; int nhex = 0, nprism = 0, npyramid = 0, ntet = 0; int maxlevel = 0; for (i = 1; i <= elements.Size(); i++) { const HPRefElement & el = elements.Get(i); maxlevel = max2 (el.level, maxlevel); switch (Get_HPRef_Struct (el.type)->geom) { case HP_SEGM: { nsegm++; break; } case HP_TRIG: { ntrig ++; break; } case HP_QUAD: { nquad++; break; } case HP_TET: { ntet++; break; } case HP_PRISM: { nprism++; break; } case HP_PYRAMID: { npyramid++; break; } case HP_HEX: { nhex++; break; } default: { cerr << "statistics error, unknown element type" << endl; } } } cout << "level = " << maxlevel << endl; cout << "nsegm = " << nsegm << endl; cout << "ntrig = " << ntrig << ", nquad = " << nquad << endl; cout << "ntet = " << ntet << ", npyr = " << npyramid << ", nprism = " << nprism << ", nhex = " << nhex << endl; return; double memcost = 0, cpucost = 0; for (p = 1; p <= 20; p++) { memcost = (ntet + nprism + nhex) * pow (static_cast<double>(p), 6.0); cpucost = (ntet + nprism + nhex) * pow (static_cast<double>(p), 9.0); cout << "costs for p = " << p << ": mem = " << memcost << ", cpu = " << cpucost << endl; } double memcosttet = 0; double memcostprism = 0; double memcosthex = 0; double memcostsctet = 0; double memcostscprism = 0; double memcostschex = 0; double cpucosttet = 0; double cpucostprism = 0; double cpucosthex = 0; for (i = 1; i <= elements.Size(); i++) { const HPRefElement & el = elements.Get(i); switch (el.type) { case HP_TET: case HP_TET_0E_1V: case HP_TET_1E_0V: case HP_TET_1E_1VA: { int p1 = maxlevel - el.level + 1; (*testout) << "p1 = " << p1 << ", P1^6 = " << pow (static_cast<double>(p1), 6.0) << " (p1-3)^6 = " << pow ( static_cast<double>(max2(p1-3, 0)), 6.0) << " p1^3 = " << pow ( static_cast<double>(p1), 3.0) << " (p1-3)^3 = " << pow ( static_cast<double>(p1-3), 3.0) << " [p1^3-(p1-3)^3]^2 = " << sqr (pow (static_cast<double>(p1),3.0) - pow ( static_cast<double>(p1-3), 3.0)) << endl; p1 /= 2 +1; memcosttet += pow (static_cast<double>(p1), 6.0); memcostsctet += pow (static_cast<double>(p1), 6.0) - pow ( static_cast<double>(max2(p1-3, 1)), 6.0); cpucosttet += pow (static_cast<double>(p1), 9.0); break; } case HP_PRISM: case HP_PRISM_SINGEDGE: { int p1 = maxlevel - el.level + 1; p1 /= 2 +1; memcostprism += pow (static_cast<double>(p1), 6.0); memcostscprism += pow (static_cast<double>(p1), 6.0) - pow ( static_cast<double>(max2(p1-3, 1)), 6.0); cpucostprism += pow (static_cast<double>(p1), 9.0); break; } case HP_HEX: { int p1 = maxlevel - el.level + 1; int p2 = maxlevel; p1 /= 2 +1; p2 /= 2 +1; memcosthex += pow (static_cast<double>(p1), 4.0) * pow (static_cast<double>(p2), 2.0); memcostschex += pow (static_cast<double>(p1), 6.0) - pow ( static_cast<double>(max2(p1-2, 0)), 6.0); cpucosthex += pow (static_cast<double>(p1), 6.0) * pow (static_cast<double>(p2), 3.0); break; } default: ; } } cout << "TET: hp-memcost = " << memcosttet << ", scmemcost = " << memcostsctet << ", cpucost = " << cpucosttet << endl; cout << "PRI: hp-memcost = " << memcostprism << ", scmemcost = " << memcostscprism << ", cpucost = " << cpucostprism << endl; cout << "HEX: hp-memcost = " << memcosthex << ", scmemcost = " << memcostschex << ", cpucost = " << cpucosthex << endl; #endif } void ReorderPoints (Mesh & mesh, NgArray<HPRefElement> & hpelements) { NgArray<int, 1> map (mesh.GetNP()); for (int i = 1; i <= mesh.GetNP(); i++) map[i] = i; int nwrong(0), nright(0); for (int k = 0; k < 5; k++) { nwrong = nright = 0; for (int i = 0; i < hpelements.Size(); i++) { const HPRefElement & hpel = hpelements[i]; if (Get_HPRef_Struct (hpel.type) -> geom == HP_PRISM) { int minbot = 0, mintop = 0; for (int j = 0; j < 3; j++) { if (map[hpel.pnums[j]] < map[hpel.pnums[minbot]]) minbot = j; if (map[hpel.pnums[j+3]] < map[hpel.pnums[mintop+3]]) mintop = j; } if (minbot != mintop) nwrong++; else nright++; if (minbot != mintop) { if (map[hpel.pnums[minbot]] < map[hpel.pnums[mintop+3]]) swap (map[hpel.pnums[3+minbot]], map[hpel.pnums[3+mintop]]); else swap (map[hpel.pnums[minbot]], map[hpel.pnums[mintop]]); } } } // cout << nwrong << " wrong prisms, " << nright << " right prisms" << endl; } PrintMessage(3, nwrong, " wrong prisms, ", nright, " right prisms"); NgArray<MeshPoint, 1> hpts(mesh.GetNP()); for (int i = 1; i <= mesh.GetNP(); i++) hpts[map[i]] = mesh.Point(i); for (int i = 1; i <= mesh.GetNP(); i++) mesh.Point(i) = hpts[i]; for (int i = 0; i < hpelements.Size(); i++) { HPRefElement & hpel = hpelements[i]; for (int j = 0; j < hpel.np; j++) hpel.pnums[j] = map[hpel.pnums[j]]; } } /* ***************************** HPRefinement ********************************** */ void HPRefinement (Mesh & mesh, Refinement * ref, SplittingType split, int levels, double fac1, bool setorders, bool reflevels) { PrintMessage (1, "HP Refinement called, levels = ", levels); // NgLock mem_lock (mem_mutex,1); mesh.coarsemesh = make_unique<Mesh>(); *mesh.coarsemesh = mesh; // #ifdef CURVEDELEMS_NEW const_cast<CurvedElements&> (mesh.coarsemesh->GetCurvedElements() ). BuildCurvedElements (ref, mesh.GetCurvedElements().GetOrder()); // #endif // delete mesh.hpelements; mesh.hpelements = make_unique<NgArray<HPRefElement>>(); NgArray<HPRefElement> & hpelements = *mesh.hpelements; InitHPElements(mesh,hpelements); NgArray<int> nplevel; nplevel.Append (mesh.GetNP()); int act_ref=1; bool sing = ClassifyHPElements (mesh,hpelements, split, act_ref, levels); sing = true; // iterate at least once while(sing) { PrintMessage(3, " Start new hp-refinement: step ", act_ref); DoRefinement (mesh, hpelements, ref, fac1); DoRefineDummies (mesh, hpelements, ref); nplevel.Append (mesh.GetNP()); CalcStatistics (hpelements); SubdivideDegeneratedHexes (mesh, hpelements,fac1); ReorderPoints (mesh, hpelements); mesh.ClearSegments(); mesh.ClearSurfaceElements(); mesh.ClearVolumeElements(); for (int i = 0; i < hpelements.Size(); i++) { HPRefElement & hpel = hpelements[i]; if (Get_HPRef_Struct (hpel.type)) switch (Get_HPRef_Struct (hpel.type) -> geom) { case HP_SEGM: { Segment seg; seg[0] = hpel.pnums[0]; seg[1] = hpel.pnums[1]; // NOTE: only for less than 10000 elements (HACK) !!! // seg.edgenr = hpel.index % 10000; // seg.si = hpel.index / 10000; seg.edgenr = hpel.index; seg.si = hpel.si; /* seg.epgeominfo[0].dist = hpel.param[0][0]; // he: war hpel.param[0][0] seg.epgeominfo[1].dist = hpel.param[1][0]; // he: war hpel.param[1][0] */ const Segment & coarseseg = mesh.coarsemesh->LineSegment(hpel.coarse_elnr+1); double d1 = coarseseg.epgeominfo[0].dist; double d2 = coarseseg.epgeominfo[1].dist; // seg.epgeominfo[0].dist = hpel.param[0][0]; // he: war hpel.param[0][0] // seg.epgeominfo[1].dist = hpel.param[1][0]; // he: war hpel.param[1][0] seg.epgeominfo[0].dist = d1 + hpel.param[0][0] * (d2-d1); // JS, June 08 seg.epgeominfo[1].dist = d1 + hpel.param[1][0] * (d2-d1); seg.epgeominfo[0].edgenr = seg.edgenr; seg.epgeominfo[1].edgenr = seg.edgenr; seg.domin = hpel.domin; seg.domout=hpel.domout; // he: needed for segments! seg.hp_elnr = i; seg.singedge_left = hpel.singedge_left; seg.singedge_right = hpel.singedge_right; seg.SetCurved (coarseseg.IsCurved()); mesh.AddSegment (seg); break; } case HP_TRIG: case HP_QUAD: { Element2d el(hpel.np); for(int j=0;j<hpel.np;j++) el.PNum(j+1) = hpel.pnums[j]; el.SetHpElnr(i); el.SetIndex(hpel.index); if(setorders) el.SetOrder(act_ref+1,act_ref+1,0); if((*mesh.coarsemesh)[SurfaceElementIndex{hpel.coarse_elnr}].IsCurved()) el.SetCurved(true); mesh.AddSurfaceElement(el); break; } case HP_HEX: case HP_HEX7: case HP_TET: case HP_PRISM: case HP_PYRAMID: { Element el(hpel.np); for(int j=0;j<hpel.np;j++) el.PNum(j+1) = hpel.pnums[j]; el.SetIndex(hpel.index); el.SetHpElnr(i); if(setorders) el.SetOrder(act_ref+1,act_ref+1,act_ref+1); if((*mesh.coarsemesh)[ElementIndex{hpel.coarse_elnr}].IsCurved()) el.SetCurved(true); mesh.AddVolumeElement(el); break; } default: PrintSysError ("hpref, backconversion failed for element ", int(Get_HPRef_Struct (hpel.type) -> geom)); } } PrintMessage(5, " Start with Update Topology "); mesh.UpdateTopology(); PrintMessage(5, " Mesh Update Topology done "); act_ref++; sing = ClassifyHPElements(mesh,hpelements, split, act_ref, levels); } PrintMessage(3, " HP-Refinement done with ", --act_ref, " refinement steps."); if(act_ref>=1) { for(ElementIndex i=0;i<mesh.GetNE(); i++) { // Element el = mesh[i] ; HPRefElement & hpel = hpelements[mesh[i].GetHpElnr()]; const ELEMENT_EDGE * edges = MeshTopology::GetEdges1 (mesh[i].GetType()); double dist[3] = {0,0,0}; int ord_dir[3] = {0,0,0}; int edge_dir[12] = {0,0,0,0,0,0,0,0,0,0,0,0}; int ned = 4; switch (mesh[i].GetType()) { case TET: /* cout << " TET " ; for(int k=0;k<4;k++) cout << el[k] << "\t" ; cout << endl; */ break; case PRISM: /* cout << " PRISM " ; for(int k=0;k<6;k++) cout << el[k] << "\t" ; cout << endl; */ for(int l=6;l<9;l++) edge_dir[l] = 2; ord_dir[2] = 2; ned = 9; break; case HEX7: // ???? break; case HEX: /* cout << " HEX " ; for(int k=0;k<8;k++) cout << el[k] << "\t" ; cout << endl; */ for(int l=8;l<12; l++) edge_dir[l] = 2; edge_dir[2] = edge_dir[3] = edge_dir[6] = edge_dir[7] = 1; ord_dir[1] = 1; ord_dir[2] = 2; ned = 12; break; case PYRAMID: /* cout << " PYRAMID " ; for(int k=0;k<5;k++) cout << el[k] << "\t" ; cout << endl; */ for(int l=4;l<8;l++) edge_dir[l] = 2; edge_dir[2] = edge_dir[3] = 1; ord_dir[1] = 1; ord_dir[2] = 2; ned = 8; break; default: cerr << "HPRefElement: illegal elementtype (2) " << mesh[i].GetType() << endl; throw NgException ("HPRefElement: illegal elementtype (2)"); } for (int j=0;j<ned;j++) { Vec<3> v(hpel.param[edges[j][0]-1][0]-hpel.param[edges[j][1]-1][0], hpel.param[edges[j][0]-1][1]-hpel.param[edges[j][1]-1][1], hpel.param[edges[j][0]-1][2]-hpel.param[edges[j][1]-1][2]); dist[edge_dir[j]] = max(v.Length(),dist[edge_dir[j]]); } int refi[3]; for(int j=0;j<3;j++) refi[j] = int(max(double(floor(log(dist[ord_dir[j]]/sqrt(2.))/log(fac1))),0.)); // cout << " ref " << refi[0] << "\t" << refi[1] << "\t" << refi[2] << endl; // cout << " order " << act_ref +1 - refi[0] << "\t" << act_ref +1 - refi[1] << "\t" << act_ref +1 - refi[2] << endl; if(setorders) mesh[i].SetOrder(act_ref+1-refi[0],act_ref+1-refi[1],act_ref+1-refi[2]); } for(SurfaceElementIndex i=0;i<mesh.GetNSE(); i++) { // Element2d el = mesh[i] ; HPRefElement & hpel = hpelements[mesh[i].GetHpElnr()]; const ELEMENT_EDGE * edges = MeshTopology::GetEdges1 (mesh[i].GetType()); double dist[3] = {0,0,0}; int ord_dir[3] = {0,0,0}; int edge_dir[4] = {0,0,0,0} ; int ned = 3; if(mesh[i].GetType() == QUAD) { /* cout << " QUAD " ; for(int k=0;k<4;k++) cout << el[k] << "\t" ; cout << endl; */ edge_dir[2] = edge_dir[3] = 1; ord_dir[1] = 1; ned = 4; } /* else { cout << " TRIG " ; for(int k=0;k<3;k++) cout << el[k] << "\t" ; cout << endl; } */ for (int j=0;j<ned;j++) { Vec<3> v(hpel.param[edges[j][0]-1][0]-hpel.param[edges[j][1]-1][0], hpel.param[edges[j][0]-1][1]-hpel.param[edges[j][1]-1][1], hpel.param[edges[j][0]-1][2]-hpel.param[edges[j][1]-1][2]); dist[edge_dir[j]] = max(v.Length(),dist[edge_dir[j]]); } int refi[3]; for(int j=0;j<3;j++) refi[j] = int(max(double(floor(log(dist[ord_dir[j]]/sqrt(2.))/log(fac1))),0.)); if(setorders) mesh[i].SetOrder(act_ref+1-refi[0],act_ref+1-refi[1],act_ref+1-refi[2]); // cout << " ref " << refi[0] << "\t" << refi[1] << endl; // cout << " order " << act_ref +1 - refi[0] << "\t" << act_ref +1 - refi[1] << endl; } } } bool CheckSingularities(Mesh & mesh, INDEX_2_HASHTABLE<int> & edges, INDEX_2_HASHTABLE<int> & edgepoint_dom, NgBitArray & cornerpoint, NgBitArray & edgepoint, INDEX_3_HASHTABLE<int> & faces, INDEX_2_HASHTABLE<int> & face_edges, INDEX_2_HASHTABLE<int> & surf_edges, NgArray<int, PointIndex::BASE> & facepoint, int & levels, int & act_ref) { bool sing = 0; if (mesh.GetDimension() == 3) { /* // check, if point has as least 3 different surfs: NgArray<INDEX_3, PointIndex::BASE> surfonpoint(mesh.GetNP()); surfonpoint = INDEX_3(0,0,0); for (SurfaceElementIndex sei = 0; sei < mesh.GetNSE(); sei++) { const Element2d & el = mesh[sei]; int ind = el.GetIndex(); for (int j = 0; j < el.GetNP(); j++) { INDEX_3 & i3 = surfonpoint[el[j]]; if (ind != i3.I1() && ind != i3.I2() && ind != i3.I3()) { i3.I1() = i3.I2(); i3.I2() = i3.I3(); i3.I3() = ind; } } } for (int i = 1; i <= mesh.GetNP(); i++) if (surfonpoint.Get(i).I1()) cornerpoint.Set(i); */ cornerpoint.Clear(); for (int i = 1; i <= mesh.GetNP(); i++) { if (mesh.Point(i).Singularity() * levels >= act_ref) { cornerpoint.Set(i); sing = 1; } } // cout << endl; for (int i = 1; i <= mesh.GetNSeg(); i++) if (mesh.LineSegment(i).singedge_left * levels >= act_ref) { INDEX_2 i2 (mesh.LineSegment(i)[0], mesh.LineSegment(i)[1]); /* // before edges.Set (i2, 1); i2.Sort(); INDEX_2 i2s(i2.I2(), i2.I1()); edges.Set (i2s, 1); */ edges.Set (i2, 1); INDEX_2 i2s(i2.I2(), i2.I1()); edges.Set (i2s, 1); edgepoint.Set (i2.I1()); edgepoint.Set (i2.I2()); sing = 1; } // if 2 adjacent edges of an element are singular, the // common point must be a singular point for (int i = 1; i <= mesh.GetNE(); i++) { const Element & el = mesh.VolumeElement(i); const ELEMENT_EDGE * eledges = MeshTopology::GetEdges1 (el.GetType()); int nedges = MeshTopology::GetNEdges (el.GetType()); for (int j = 0; j < nedges; j++) for (int k = 0; k < nedges; k++) if (j != k) { INDEX_2 ej(el.PNum(eledges[j][0]), el.PNum(eledges[j][1])); ej.Sort(); INDEX_2 ek(el.PNum(eledges[k][0]), el.PNum(eledges[k][1])); ek.Sort(); if (edges.Used(ej) && edges.Used(ek)) { if (ej.I1() == ek.I1()) cornerpoint.Set (ek.I1()); if (ej.I1() == ek.I2()) cornerpoint.Set (ek.I2()); if (ej.I2() == ek.I1()) cornerpoint.Set (ek.I1()); if (ej.I2() == ek.I2()) cornerpoint.Set (ek.I2()); } } } edgepoint.Or (cornerpoint); (*testout) << "cornerpoint = " << endl << cornerpoint << endl; (*testout) << "edgepoint = " << endl << edgepoint << endl; facepoint = 0; for (SurfaceElementIndex sei = 0; sei < mesh.GetNSE(); sei++) { const Element2d & el = mesh[sei]; const FaceDescriptor & fd = mesh.GetFaceDescriptor (el.GetIndex()); int domnr = 0; if (fd.DomainInSingular() * levels < act_ref && fd.DomainOutSingular() * levels < act_ref) { domnr=0; continue;} if (fd.DomainInSingular() * levels >= act_ref) { domnr = fd.DomainIn(); sing = 1; } if (fd.DomainOutSingular() * levels >= act_ref) { domnr = fd.DomainOut(); sing = 1; } if (fd.DomainInSingular() * levels >= act_ref && fd.DomainOutSingular() * levels >= act_ref) { domnr = -1; sing = 1; } INDEX_3 i3; if (el.GetNP() == 3) i3 = INDEX_3::Sort (el[0], el[1], el[2]); else { INDEX_4 i4 (el[0], el[1], el[2], el[3]); i4.Sort(); i3 = INDEX_3(i4.I1(), i4.I2(), i4.I3()); } faces.Set (i3, domnr); *testout << "set face " << i3 << ", domnr = " << domnr << endl; for (int j = 0; j < el.GetNP(); j++) { face_edges.Set (INDEX_2::Sort (el[j], el[(j+1)%el.GetNP()]), domnr); surf_edges.Set (INDEX_2::Sort (el[j], el[(j+1)%el.GetNP()]), fd.SurfNr()+1); facepoint[el[j]] = domnr; } } // (*testout) << "singular edges = " << edges << endl; // (*testout) << "singular faces = " << faces << endl; // (*testout) << "singular faces_edges = " << face_edges << endl; } else { // 2D case // check, if point has as least 3 different surfs: NgArray<INDEX_3, PointIndex::BASE> surfonpoint(mesh.GetNP()); for (int i = 1; i <= mesh.GetNP(); i++) surfonpoint.Elem(i) = INDEX_3(0,0,0); for (int i = 1; i <= mesh.GetNSeg(); i++) { const Segment & seg = mesh.LineSegment(i); int ind = seg.edgenr; if (seg.singedge_left * levels >= act_ref) { INDEX_2 i2 = INDEX_2::Sort(mesh.LineSegment(i)[0], mesh.LineSegment(i)[1]); edges.Set(i2,1); edgepoint.Set(i2.I1()); edgepoint.Set(i2.I2()); *testout << " singleft " << endl; *testout << " mesh.LineSegment(i).domout " << mesh.LineSegment(i).domout << endl; *testout << " mesh.LineSegment(i).domin " << mesh.LineSegment(i).domin << endl; edgepoint_dom.Set (INDEX_2(mesh.LineSegment(i).domin, i2.I1()), 1); edgepoint_dom.Set (INDEX_2(mesh.LineSegment(i).domin, i2.I2()), 1); sing = 1; } if (seg.singedge_right * levels >= act_ref) { INDEX_2 i2 = INDEX_2::Sort(mesh.LineSegment(i)[1], mesh.LineSegment(i)[0]); edges.Set (i2, 1); edgepoint.Set(i2.I1()); edgepoint.Set(i2.I2()); *testout << " singright " << endl; *testout << " mesh.LineSegment(i).domout " << mesh.LineSegment(i).domout << endl; *testout << " mesh.LineSegment(i).domin " << mesh.LineSegment(i).domin << endl; edgepoint_dom.Set (INDEX_2(mesh.LineSegment(i).domout, i2.I1()), 1); edgepoint_dom.Set (INDEX_2(mesh.LineSegment(i).domout, i2.I2()), 1); sing = 1; } // (*testout) << "seg = " << ind << ", " << seg[0] << "-" << seg[1] << endl; if (seg.singedge_left * levels >= act_ref || seg.singedge_right* levels >= act_ref) { for (int j = 0; j < 2; j++) { int pi = (j == 0) ? seg[0] : seg[1]; INDEX_3 & i3 = surfonpoint.Elem(pi); if (ind != i3.I1() && ind != i3.I2()) { i3.I1() = i3.I2(); i3.I2() = ind; } } } } for (int i = 1; i <= mesh.GetNP(); i++) { // mark points for refinement that are in corners between two anisotropic edges if (surfonpoint.Get(i).I1()) { // cornerpoint.Set(i); // disabled by JS, Aug 2009 edgepoint.Set(i); } // mark points for refinement that are explicitly specified in input file if (mesh.Point(i).Singularity()*levels >= act_ref) { cornerpoint.Set(i); edgepoint.Set(i); sing = 1; } } edgepoint.Or (cornerpoint); (*testout) << "2d sing edges: " << endl << edges << endl; (*testout) << "2d cornerpoints: " << endl << cornerpoint << endl << "2d edgepoints: " << endl << edgepoint << endl; facepoint = 0; } if (!sing) PrintMessage(3, "PrepareElements no more to do for actual refinement ", act_ref); return(sing); } bool ClassifyHPElements (Mesh & mesh, NgArray<HPRefElement> & elements, SplittingType split, int & act_ref, int & levels) { INDEX_2_HASHTABLE<int> edges(mesh.GetNSeg()+1); NgBitArray edgepoint(mesh.GetNP()); INDEX_2_HASHTABLE<int> edgepoint_dom(mesh.GetNSeg()+1); edgepoint.Clear(); NgBitArray cornerpoint(mesh.GetNP()); cornerpoint.Clear(); // value = nr > 0 ... refine elements in domain nr // value = -1 ..... refine elements in any domain INDEX_3_HASHTABLE<int> faces(mesh.GetNSE()+1); INDEX_2_HASHTABLE<int> face_edges(mesh.GetNSE()+1); INDEX_2_HASHTABLE<int> surf_edges(mesh.GetNSE()+1); NgArray<int, PointIndex::BASE> facepoint(mesh.GetNP()); bool sing = CheckSingularities(mesh, edges, edgepoint_dom, cornerpoint, edgepoint, faces, face_edges, surf_edges, facepoint, levels, act_ref); if (act_ref == 1 && split == SPLIT_ALFELD) sing = true; if (act_ref == 1 && split == SPLIT_POWELL) sing = true; if(sing==0) return(sing); int cnt_undef = 0, cnt_nonimplement = 0; NgArray<int> misses(10000); misses = 0; (*testout) << "edgepoint_dom = " << endl << edgepoint_dom << endl; for( int i = 0; i<elements.Size(); i++) { HPRefElement & hpel = elements[i]; HPRef_Struct * hprs = Get_HPRef_Struct (hpel.type); HPRefElement old_el = elements[i]; int dd=3; if(act_ref !=1 && (hpel.type == HP_HEX || hpel.type == HP_PRISM || hpel.type == HP_TET || hpel.type == HP_PYRAMID || hpel.type == HP_QUAD || hpel.type == HP_TRIG || hpel.type == HP_SEGM)) continue; sing = 1; switch (hprs->geom) { case HP_TET: { hpel.type = ClassifyTet(hpel, edges, edgepoint_dom, cornerpoint, edgepoint, faces,face_edges, surf_edges, facepoint); /* // if (i != 182) if ( (!hpel.PNums().Contains(40)) || (!hpel.PNums().Contains(41)) ) hpel.type = HP_NONETET; else cout << "type = " << hpel.type << endl; */ break; } case HP_PRISM: { hpel.type = ClassifyPrism(hpel, edges, edgepoint_dom, cornerpoint, edgepoint, faces, face_edges, surf_edges, facepoint); break; } case HP_HEX7: { hpel.type = ClassifyHex7(hpel, edges, edgepoint_dom, cornerpoint, edgepoint, faces, face_edges, surf_edges, facepoint); break; } case HP_HEX: { hpel.type = ClassifyHex(hpel, edges, edgepoint_dom, cornerpoint, edgepoint, faces, face_edges, surf_edges, facepoint); break; } case HP_TRIG: { int dim = mesh.GetDimension(); const FaceDescriptor & fd = mesh.GetFaceDescriptor (hpel.GetIndex()); if (split == SPLIT_HP) hpel.type = ClassifyTrig(hpel, edges, edgepoint_dom, cornerpoint, edgepoint, faces, face_edges, surf_edges, facepoint, dim, fd); else if (split == SPLIT_ALFELD) hpel.type = HP_TRIG_ALFELD; else if (split == SPLIT_POWELL) hpel.type = HP_TRIG_POWELL; dd = 2; break; } case HP_QUAD: { int dim = mesh.GetDimension(); const FaceDescriptor & fd = mesh.GetFaceDescriptor (hpel.GetIndex()); hpel.type = ClassifyQuad(hpel, edges, edgepoint_dom, cornerpoint, edgepoint, faces, face_edges, surf_edges, facepoint, dim, fd); dd = 2; break; } case HP_SEGM: { if (split == SPLIT_HP) hpel.type = ClassifySegm(hpel, edges, edgepoint_dom, cornerpoint, edgepoint, faces, face_edges, surf_edges, facepoint); else if (split == SPLIT_POWELL) hpel.type = HP_SEGM_SINGCORNERL; dd = 1; break; } case HP_PYRAMID: { hpel.type = ClassifyPyramid(hpel, edges, edgepoint_dom, cornerpoint, edgepoint, faces, face_edges, surf_edges, facepoint); break; } default: { cout << "illegal element type for hp-prepare elements " << hpel.type << endl; throw NgException ("hprefinement.cpp: don't know how to set parameters"); } } if(hpel.type == HP_NONE) cnt_undef++; //else //cout << "elem " << i << " classified type " << hpel.type << endl; if (!Get_HPRef_Struct (hpel.type)) { (*testout) << "hp-element-type " << hpel.type << " not implemented " << endl; (*testout) << " elType " << hprs->geom << endl; (cout) << " elType " << hprs->geom << endl; cnt_nonimplement++; misses[hpel.type]++; } for(int j=0; j<hpel.np; j++) { for( int k=0; k<hpel.np; k++) if(hpel[j] == old_el.pnums[k]) { for(int l=0;l<dd;l++) hpel.param[j][l] = old_el.param[k][l]; break; } } } PrintMessage(3, "undefined elements update classification: ", cnt_undef); PrintMessage(3, "non-implemented in update classification: ", cnt_nonimplement); for (int i = 0; i < misses.Size(); i++) if (misses[i]) cout << " in update classification missing case " << i << " occurred " << misses[i] << " times" << endl; return(sing); } }