diff --git a/Geo/GEdge.cpp b/Geo/GEdge.cpp index 9669bf6e36e73d291bcedcb850e993c7b33a3f09..8c43b0db8690a4643ab43811f9f932b4601ab4ac 100644 --- a/Geo/GEdge.cpp +++ b/Geo/GEdge.cpp @@ -1,4 +1,4 @@ -// $Id: GEdge.cpp,v 1.23 2007-03-05 09:10:54 geuzaine Exp $ +// $Id: GEdge.cpp,v 1.24 2007-03-09 14:57:06 remacle Exp $ // // Copyright (C) 1997-2007 C. Geuzaine, J.-F. Remacle // @@ -29,7 +29,7 @@ GEdge::GEdge(GModel *model, int tag, GVertex *_v0, GVertex *_v1) : GEntity(model, tag), v0(_v0), v1(_v1) { if(v0) v0->addEdge(this); - if(v1) v1->addEdge(this); + if(v1 && v1 != v0) v1->addEdge(this); resetMeshAttributes(); } diff --git a/Mesh/meshGEdge.cpp b/Mesh/meshGEdge.cpp index 60f2940fb9a21746e868b33e2d40aa4823ea65c4..7a33610cd5969aad048b41cf18acbe7381eacf19 100644 --- a/Mesh/meshGEdge.cpp +++ b/Mesh/meshGEdge.cpp @@ -1,4 +1,4 @@ -// $Id: meshGEdge.cpp,v 1.29 2007-02-26 08:25:39 geuzaine Exp $ +// $Id: meshGEdge.cpp,v 1.30 2007-03-09 14:57:06 remacle Exp $ // // Copyright (C) 1997-2007 C. Geuzaine, J.-F. Remacle // @@ -262,4 +262,22 @@ void meshGEdge::operator() (GEdge *ge) ge->getEndVertex()->mesh_vertices[0] : ge->mesh_vertices[i]; ge->lines.push_back(new MLine(v0, v1)); } + + // if the curve is periodic and if the begin vertex is identical to the end vertex + // and if this vertex has only one model curve adjacent to it, then the vertex is + // not connecting any other curve. So, the mesh vertex and its associated geom vertex + // are not necessary at the same location + + + // printf("%p %p %d\n",ge->getBeginVertex(),ge->getEndVertex(),ge->getBeginVertex()->edges().size()); + + if (ge->getBeginVertex() == ge->getEndVertex() && ge->getBeginVertex()->edges().size() == 1) + { + MVertex *v0 = ge->getBeginVertex()->mesh_vertices[0]; + GPoint gp = ge->point (t_begin); + v0->x() = gp.x(); + v0->y() = gp.y(); + v0->z() = gp.z(); + } + } diff --git a/Mesh/meshGFace.cpp b/Mesh/meshGFace.cpp index 3ca4642b50eb1feeb5165056a2dc7c8768553937..dadd19dd1f8371289b46bcbd366d024777480b1f 100644 --- a/Mesh/meshGFace.cpp +++ b/Mesh/meshGFace.cpp @@ -1,4 +1,4 @@ -// $Id: meshGFace.cpp,v 1.63 2007-03-02 09:55:01 remacle Exp $ +// $Id: meshGFace.cpp,v 1.64 2007-03-09 14:57:06 remacle Exp $ // // Copyright (C) 1997-2007 C. Geuzaine, J.-F. Remacle // @@ -38,6 +38,8 @@ extern Context_T CTX; +static double SCALINGU=1,SCALINGV=1; + void computeEdgeLoops(const GFace *gf, std::vector<MVertex*> &all_mvertices, std::vector<int> &indices) @@ -114,7 +116,7 @@ inline double computeEdgeLinearLength(BDS_Point *p1, BDS_Point *p2) inline double computeEdgeLinearLength(BDS_Point *p1, BDS_Point *p2, GFace *f) { - GPoint GP = f->point (SPoint2(0.5*(p1->u+p2->u),0.5*(p1->v+p2->v))); + GPoint GP = f->point (SPoint2(0.5*(p1->u+p2->u)*SCALINGU,0.5*(p1->v+p2->v)*SCALINGV)); const double dx1 = p1->X-GP.x(); const double dy1 = p1->Y-GP.y(); @@ -217,15 +219,17 @@ void OptimizeMesh(GFace *gf, BDS_Mesh &m, const int NIT) { std::list < BDS_Face * >t; (*itp)->getTriangles(t); - if ((t.size()==3 && (*itp)->edges.size() == 3)|| - (t.size()==4 && (*itp)->edges.size() == 4)) - m.collapse_edge_parametric ( *(*itp)->edges.begin(), (*itp)); + if (t.size()==(*itp)->edges.size() && t.size() < 5) + for (std::list<BDS_Edge*>::iterator ite = (*itp)->edges.begin();ite!=(*itp)->edges.end();++ite) + { + if(m.collapse_edge_parametric ( (*ite), (*itp)))break; + } else - m.smooth_point_parametric(*itp,gf); + m.smooth_point_parametric(*itp,gf); ++itp; } } - { + if(0){ // swap edges that provide a better configuration int NN1 = m.edges.size(); int NN2 = 0; @@ -235,10 +239,23 @@ void OptimizeMesh(GFace *gf, BDS_Mesh &m, const int NIT) if (NN2++ >= NN1)break; if (!(*it)->deleted && (*it)->numfaces() == 2) { + BDS_Point *op[2]; (*it)->oppositeof(op); - if ( edgeSwapTestQuality(*it) == 1) + BDS_Point *p1 = (*it)->p1; + BDS_Point *p2 = (*it)->p2; + std::list < BDS_Face * >t1,t2,to1,to2; + p1->getTriangles(t1); + p2->getTriangles(t2); + op[0]->getTriangles(to1); + op[1]->getTriangles(to2); + + if (t1.size() == 7 && t2.size() == 7 && to1.size() == 5 && to2.size() == 5) m.swap_edge ( *it , BDS_SwapEdgeTestParametric()); + else if (t1.size() == 5 && t2.size() == 5 && to1.size() == 7 && to2.size() == 7) + m.swap_edge ( *it , BDS_SwapEdgeTestParametric()); + // else if ( edgeSwapTestQuality(*it) == 1) + // m.swap_edge ( *it , BDS_SwapEdgeTestParametric()); } ++it; } @@ -257,7 +274,7 @@ void RefineMesh ( GFace *gf, BDS_Mesh &m , const int NIT) // computecharacteristic lengths using mesh edge spacing // separate attractors & - if (!Attractor::size()) + if (NIT > 0 && !Attractor::size()) { std::set<BDS_Point*,PointLessThan>::iterator itp = m.points.begin(); while (itp != m.points.end()) @@ -276,6 +293,8 @@ void RefineMesh ( GFace *gf, BDS_Mesh &m , const int NIT) } double OLDminL=1.E22,OLDmaxL=0; + + const double MINE_ = 0.7, MAXE_=1.4; while (1) { // double stot = 0; @@ -318,8 +337,8 @@ void RefineMesh ( GFace *gf, BDS_Mesh &m , const int NIT) if (OLDminL == minL && OLDmaxL == maxL)break; OLDminL = minL;OLDmaxL = maxL; - - if ((minL > 1./sqrt(2.0) && maxL < sqrt(2.0)) || IT > NIT)break; + + if ((minL > MINE_ && maxL < MAXE_) || IT > (abs(NIT)))break; NN1 = m.edges.size(); @@ -332,11 +351,13 @@ void RefineMesh ( GFace *gf, BDS_Mesh &m , const int NIT) { double lone = NewGetLc ( *it,gf); - const double coord = 0.5; // take care with seams : // if (lone < 1.e-10 && computeParametricEdgeLength((*it)->p1,(*it)->p2) > 1.e-5) lone = 2; - if ((*it)->numfaces() == 2 && (lone > 1.3)) + if ((*it)->numfaces() == 2 && (lone > MAXE_)) { + + const double coord = 0.5; + BDS_Point *mid ; mid = m.add_point(++m.MAXPOINTNUMBER, coord * (*it)->p1->u + (1 - coord) * (*it)->p2->u, @@ -388,7 +409,7 @@ void RefineMesh ( GFace *gf, BDS_Mesh &m , const int NIT) double lone = NewGetLc ( *it,gf); // if (lone < 1.e-10 && computeParametricEdgeLength((*it)->p1,(*it)->p2) > 1.e-5) lone = 2; - if (!(*it)->deleted && (*it)->numfaces() == 2 && lone < 0.6 ) + if (!(*it)->deleted && (*it)->numfaces() == 2 && lone < MINE_ ) { bool res = false; if ( (*it)->p1->iD > MAXNP ) @@ -464,8 +485,8 @@ void RefineMesh ( GFace *gf, BDS_Mesh &m , const int NIT) } } IT++; - Msg(DEBUG1," iter %3d minL %8.3f maxL %8.3f : %6d splits, %6d swaps, %6d collapses, %6d moves",IT,minL,maxL,nb_split,nb_swap,nb_collaps,nb_smooth); + if (nb_split==0 && nb_collaps == 0)break; } } @@ -843,7 +864,9 @@ bool gmsh2DMeshGenerator ( GFace *gf ) // start mesh generation // if (CTX.mesh.algo2d == ALGO_2D_MESHADAPT) { - RefineMesh (gf,*m,20); + RefineMesh (gf,*m,10); + OptimizeMesh(gf, *m, 3); + RefineMesh (gf,*m,-10); if (gf->meshAttributes.recombine) { m->recombineIntoQuads (gf->meshAttributes.recombineAngle,gf); @@ -1143,7 +1166,9 @@ bool buildConsecutiveListOfVertices ( GFace *gf, } -bool gmsh2DMeshGeneratorPeriodic ( GFace *gf ) + + +bool gmsh2DMeshGeneratorPeriodic ( GFace *gf , bool debug = true) { // if(gf->tag() != 6)return true; @@ -1162,8 +1187,10 @@ bool gmsh2DMeshGeneratorPeriodic ( GFace *gf ) // Buid a BDS_Mesh structure that is convenient for doing the actual meshing procedure BDS_Mesh *m = new BDS_Mesh; - // m->scalingU = fabs(du); - // m->scalingV = fabs(dv); + m->scalingU = fabs(du); + m->scalingV = fabs(dv); + SCALINGU = m->scalingU; + SCALINGV = m->scalingV; std::vector< std::vector<BDS_Point* > > edgeLoops_BDS; SBoundingBox3d bbox; int nbPointsTotal = 0; @@ -1250,6 +1277,16 @@ bool gmsh2DMeshGeneratorPeriodic ( GFace *gf ) BDS_GeomEntity CLASS_E (1,1); + if (debug) + { + char name[245]; + sprintf(name,"surface%d-initial-real.pos",gf->tag()); + outputScalarField(m->triangles, name,0); + sprintf(name,"surface%d-initial-param.pos",gf->tag()); + outputScalarField(m->triangles, name,1); + } + + for (unsigned int i=0;i<edgeLoops_BDS.size();i++) { std::vector<BDS_Point*> &edgeLoop_BDS = edgeLoops_BDS[i]; @@ -1259,6 +1296,7 @@ bool gmsh2DMeshGeneratorPeriodic ( GFace *gf ) if (!e) { Msg(GERROR,"impossible to recover the edge %d %d\n",edgeLoop_BDS[j]->iD,edgeLoop_BDS[(j+1)%edgeLoop_BDS.size()]->iD); + SCALINGU = SCALINGV = 1; return false; } else e->g = &CLASS_E; @@ -1335,13 +1373,25 @@ bool gmsh2DMeshGeneratorPeriodic ( GFace *gf ) m->del_point(m->find_point(-4)); + if (debug) + { + char name[245]; + sprintf(name,"surface%d-recovered-real.pos",gf->tag()); + outputScalarField(m->triangles, name,0); + sprintf(name,"surface%d-recovered-param.pos",gf->tag()); + outputScalarField(m->triangles, name,1); + } // goto hhh; // start mesh generation // if (CTX.mesh.algo2d == ALGO_2D_MESHADAPT) { - RefineMesh (gf,*m,100); + RefineMesh (gf,*m,10); + OptimizeMesh(gf, *m, 5); + RefineMesh (gf,*m,-10); + OptimizeMesh(gf, *m, 1); + if (gf->meshAttributes.recombine) { m->recombineIntoQuads (gf->meshAttributes.recombineAngle,gf); @@ -1349,6 +1399,7 @@ bool gmsh2DMeshGeneratorPeriodic ( GFace *gf ) } // fill the small gmsh structures + { std::set<BDS_Point*,PointLessThan>::iterator itp = m->points.begin(); while (itp != m->points.end()) @@ -1395,11 +1446,14 @@ bool gmsh2DMeshGeneratorPeriodic ( GFace *gf ) // delete the mesh - // sprintf(name,"real%d.pos",gf->tag()); - //outputScalarField(m->triangles, name,0); -// char name[245]; -// sprintf(name,"param%d.pos",gf->tag()); -// outputScalarField(m->triangles, name,1); + if (debug) + { + char name[245]; + sprintf(name,"surface%d-final-real.pos",gf->tag()); + outputScalarField(m->triangles, name,0); + sprintf(name,"surface%d-final-param.pos",gf->tag()); + outputScalarField(m->triangles, name,1); + } // if (CTX.mesh.algo2d == ALGO_2D_DELAUNAY) // { @@ -1408,6 +1462,7 @@ bool gmsh2DMeshGeneratorPeriodic ( GFace *gf ) delete m; + SCALINGU = SCALINGV = 1; return true; } diff --git a/Parser/Gmsh.tab.cpp b/Parser/Gmsh.tab.cpp index ddd036c2433bd9689b946fb48de2b92cb8f9fc58..03384fba52e317ac4b3f96ff375e548e640f6022 100644 --- a/Parser/Gmsh.tab.cpp +++ b/Parser/Gmsh.tab.cpp @@ -125,7 +125,7 @@ #line 1 "Gmsh.y" -// $Id: Gmsh.tab.cpp,v 1.311 2007-03-05 11:01:21 geuzaine Exp $ +// $Id: Gmsh.tab.cpp,v 1.312 2007-03-09 14:57:06 remacle Exp $ // // Copyright (C) 1997-2007 C. Geuzaine, J.-F. Remacle // diff --git a/Parser/Gmsh.yy.cpp b/Parser/Gmsh.yy.cpp index cf6ecc4fe2c20243c32f744ed7f7751febc024bb..89abe07bb41135eda8d584befc0cd58f682ccba2 100644 --- a/Parser/Gmsh.yy.cpp +++ b/Parser/Gmsh.yy.cpp @@ -2,7 +2,7 @@ /* A lexical scanner generated by flex */ /* Scanner skeleton version: - * $Header: /cvsroot/gmsh/Parser/Gmsh.yy.cpp,v 1.310 2007-03-05 11:01:21 geuzaine Exp $ + * $Header: /cvsroot/gmsh/Parser/Gmsh.yy.cpp,v 1.311 2007-03-09 14:57:08 remacle Exp $ */ #define FLEX_SCANNER @@ -740,7 +740,7 @@ char *yytext; #line 1 "Gmsh.l" #define INITIAL 0 #line 2 "Gmsh.l" -// $Id: Gmsh.yy.cpp,v 1.310 2007-03-05 11:01:21 geuzaine Exp $ +// $Id: Gmsh.yy.cpp,v 1.311 2007-03-09 14:57:08 remacle Exp $ // // Copyright (C) 1997-2007 C. Geuzaine, J.-F. Remacle // diff --git a/benchmarks/2d/world.geo b/benchmarks/2d/world.geo index f5b816bcfc782ebc0e42ec970d7ac91caaa7ebf5..82db5ecf812e19e50536eddf28d8b2b6b4b3d549 100755 --- a/benchmarks/2d/world.geo +++ b/benchmarks/2d/world.geo @@ -673,6 +673,6 @@ Point(604) = {2.18974,-20.8801,0,lc}; CatmullRom (48) = {600, 601, 602, 603, 604, 600}; Line Loop (49) = {48}; Physical Line (1336) = {48}; -//Plane Surface (3) = {43,5, 7, 9, 11, 13, 15, 17, 19, 21, 23, 25, 27, 29, 31, 33, 35, 37, 39, 41, 45, 47, 49}; -Plane Surface (3) = {43}; +Plane Surface (3) = {43,5, 7, 9, 11, 13, 15, 17, 19, 21, 23, 25, 27, 29, 31, 33, 35, 37, 39, 41, 45, 47, 49}; +//Plane Surface (3) = {43}; //Attractor Line (2) = {4, 6, 8, 10, 12, 14, 16, 18, 20, 22, 24, 26, 28, 30, 32, 34, 36, 38, 40, 42, 44, 46, 48};