From 0b3bfed5472c09536662827b8d33a2eb5f273953 Mon Sep 17 00:00:00 2001
From: Emilie Marchandise <emilie.marchandise@uclouvain.be>
Date: Wed, 19 May 2010 15:31:35 +0000
Subject: [PATCH] Spectral conformal faster + createTopo for connected faces
 done !

---
 Geo/GFaceCompound.cpp        |  45 +++------
 Geo/GModel.cpp               | 185 ++++++++++++++++++++++++-----------
 Geo/GModelIO_Mesh.cpp        |   4 +-
 Mesh/multiscalePartition.cpp |   2 +-
 Solver/multiscaleLaplace.cpp |   1 -
 benchmarks/boolean/nurbs.lua |   2 +-
 6 files changed, 147 insertions(+), 92 deletions(-)

diff --git a/Geo/GFaceCompound.cpp b/Geo/GFaceCompound.cpp
index 460d0099b0..1c4ae412ba 100644
--- a/Geo/GFaceCompound.cpp
+++ b/Geo/GFaceCompound.cpp
@@ -363,7 +363,7 @@ bool GFaceCompound::trivial() const
 //of triangles
 bool GFaceCompound::checkFolding(std::vector<MVertex*> &ordered) const
 {
- 
+
   bool has_no_folding = true;
 
   for(unsigned int i = 0; i < ordered.size()-1; ++i){
@@ -519,10 +519,10 @@ bool GFaceCompound::parametrize() const
   else if (_mapping == CONFORMAL){
     Msg::Debug("Parametrizing surface %d with 'conformal map'", tag());
     fillNeumannBCS();
-    //bool withoutFolding = parametrize_conformal_spectral() ;
-    bool withoutFolding = parametrize_conformal();
+    bool withoutFolding = parametrize_conformal_spectral() ;
+    //bool withoutFolding = parametrize_conformal();
     if ( withoutFolding == false ){
-      //printStuff(); exit(1);
+      printStuff(); //exit(1);
       Msg::Warning("$$$ Parametrization switched to harmonic map");
       parametrize(ITERU,HARMONIC); 
       parametrize(ITERV,HARMONIC);
@@ -1129,6 +1129,7 @@ bool GFaceCompound::parametrize_conformal_spectral() const
   linearSystem <double> *lsysB  = new linearSystemPETSc<double>;
   dofManager<double> myAssembler(lsysA, lsysB);
 
+  //printf("nbNodes = %d in spectral param \n", allNodes.size());
   //-------------------------------
   myAssembler.setCurrentMatrix("A");
   for(std::set<MVertex *>::iterator itv = allNodes.begin(); itv !=allNodes.end() ; ++itv){
@@ -1164,26 +1165,6 @@ bool GFaceCompound::parametrize_conformal_spectral() const
 
   //-------------------------------
    myAssembler.setCurrentMatrix("B");
-      
-//     massTerm mass1(model(), 1, &MONE);
-//     massTerm mass2(model(), 2, &MONE);
-//    it = _compound.begin();
-//    for( ; it != _compound.end() ; ++it){
-//      for(unsigned int i = 0; i < (*it)->triangles.size(); ++i){
-//        SElement se((*it)->triangles[i]);
-//        mass1.addToMatrix(myAssembler, &se);
-//        mass2.addToMatrix(myAssembler, &se);
-//      }
-//    }
-
-//    std::list<GEdge*>::const_iterator it0 = _U0.begin();
-//    for( ; it0 != _U0.end(); ++it0 ){
-//      for(unsigned int i = 0; i < (*it0)->lines.size(); i++ ){
-//        SElement se((*it0)->lines[i]);
-//        mass1.addToMatrix(myAssembler, &se);
-//        mass2.addToMatrix(myAssembler, &se);
-//      }
-//    }
 
    double small = 0.0;
    for(std::set<MVertex *>::iterator itv = allNodes.begin(); itv !=allNodes.end() ; ++itv){
@@ -1198,19 +1179,25 @@ bool GFaceCompound::parametrize_conformal_spectral() const
      }
    } 
 
+   //mettre max NC contraintes par bord
    int NB = ordered.size();
-   for (int i = 0; i< NB; i++){
-     MVertex *v1 = ordered[i];
-     for (int j = 0; j< NB; j++){
-       MVertex *v2 = ordered[j];
+   int NC = std::min(100,NB);
+   int jump = (int) NB/NC;
+   int nbLoop = (int) NB/jump ;
+   //printf("nb bound nodes=%d jump =%d \n", NB, jump);
+   for (int i = 0; i< nbLoop; i++){
+     MVertex *v1 = ordered[i*jump];
+     for (int j = 0; j< nbLoop; j++){
+       MVertex *v2 = ordered[j*jump];
        myAssembler.assemble(v1, 0, 1, v2, 0, 1,  -1./NB);
        myAssembler.assemble(v1, 0, 2, v2, 0, 2,  -1./NB);
      }
    }
 
    //-------------------------------
+   //printf("Solve eigensystem \n");
    eigenSolver eig(&myAssembler, "B" , "A", true);
-   int nb = 1;
+   int nb = 2;
    bool converged = eig.solve(nb, "largest");
     
    if(converged) {
diff --git a/Geo/GModel.cpp b/Geo/GModel.cpp
index 01f8e88eb8..d2997b089a 100644
--- a/Geo/GModel.cpp
+++ b/Geo/GModel.cpp
@@ -40,6 +40,90 @@
 std::vector<GModel*> GModel::list;
 int GModel::_current = -1;
 
+static void recur_connect(MVertex *v,
+                          std::multimap<MVertex*,MEdge> &v2e,
+                          std::set<MEdge,Less_Edge> &group,
+                          std::set<MVertex*> &touched)
+{
+  if (touched.find(v) != touched.end()) return;
+
+  touched.insert(v);
+  for (std::multimap <MVertex*,MEdge>::iterator it = v2e.lower_bound(v); 
+       it != v2e.upper_bound(v) ; ++it){
+    group.insert(it->second);
+    for (int i = 0; i < it->second.getNumVertices(); ++i){
+      recur_connect (it->second.getVertex(i), v2e, group, touched);
+    }
+  }
+}
+
+// starting form a list of elements, returns lists of lists that are
+// all simply connected
+static void recur_connect_e(const MEdge &e,
+                            std::multimap<MEdge,MElement*, Less_Edge> &e2e,
+                            std::set<MElement*> &group,
+                            std::set<MEdge, Less_Edge> &touched)
+{
+  if (touched.find(e) != touched.end())return;
+  touched.insert(e);
+  for (std::multimap <MEdge,MElement*,Less_Edge>::iterator it = e2e.lower_bound(e);
+         it != e2e.upper_bound(e) ; ++it){
+    group.insert(it->second);
+    for (int i = 0; i < it->second->getNumEdges(); ++i){
+      recur_connect_e(it->second->getEdge(i), e2e, group, touched);
+    }
+  }
+}
+
+static int connected_bounds(std::set<MEdge, Less_Edge> &edges, 
+                            std::vector<std::vector<MEdge> > &boundaries)
+{
+  std::multimap<MVertex*,MEdge> v2e;
+  for(std::set<MEdge, Less_Edge>::iterator it = edges.begin(); it != edges.end(); it++){
+    for (int j=0;j<it->getNumVertices();j++){
+      v2e.insert(std::make_pair(it->getVertex(j),*it));
+    }
+  }
+
+  while (!v2e.empty()){
+    std::set<MEdge, Less_Edge> group;
+    std::set<MVertex*> touched;
+    recur_connect(v2e.begin()->first, v2e, group, touched);
+    std::vector<MEdge> temp;
+    temp.insert(temp.begin(), group.begin(), group.end());
+    boundaries.push_back(temp);
+    for (std::set<MVertex*>::iterator it = touched.begin() ; it != touched.end();++it)
+      v2e.erase(*it);
+  }
+
+  return boundaries.size();
+}
+
+//--------------------------------------------------------------
+static int connectedRegions (std::vector<MElement*> &elements,
+			     std::vector<std::vector<MElement*> > &regions)
+{
+  std::multimap<MEdge,MElement*,Less_Edge> e2e;
+  for (unsigned int i = 0; i < elements.size(); ++i){
+    for (int j = 0; j < elements[i]->getNumEdges(); j++){
+      e2e.insert(std::make_pair(elements[i]->getEdge(j),elements[i]));
+    }
+  }
+  while (!e2e.empty()){
+    std::set<MElement*> group;
+    std::set<MEdge,Less_Edge> touched;
+    recur_connect_e (e2e.begin()->first,e2e,group,touched);
+    std::vector<MElement*> temp;
+    temp.insert(temp.begin(), group.begin(), group.end());
+    regions.push_back(temp);
+    for ( std::set<MEdge,Less_Edge>::iterator it = touched.begin() ; it != touched.end();++it)
+      e2e.erase(*it);
+  }
+
+  return regions.size();
+}
+//-----------------------------------------------------------------------------
+
 GModel::GModel(std::string name)
   : _name(name), _visible(1), _octree(0),
     _geo_internals(0), _occ_internals(0), _acis_internals(0), _fm_internals(0),_factory(0), _fields(0), _currentMeshEntity(0), normals(0)
@@ -1079,7 +1163,7 @@ void GModel::createTopologyFromMesh()
       discRegions.push_back((discreteRegion*) *it);
 
   //EMI-FIX in case of createTopology for Volumes
-  //all faces are set to aeach volume
+  //all faces are set to each volume
   for (std::vector<discreteRegion*>::iterator it = discRegions.begin();
        it != discRegions.end(); it++)
     (*it)->setBoundFaces();
@@ -1090,8 +1174,48 @@ void GModel::createTopologyFromMesh()
     if((*it)->geomType() == GEntity::DiscreteSurface)
       discFaces.push_back((discreteFace*) *it);
   
+  //--------
   //EMI TODO
   //check for closed faces
+
+  printf("discr faces size=%d \n", discFaces.size());
+  std::vector<discreteFace*> newDiscFaces;
+  for (std::vector<discreteFace*>::iterator itF = discFaces.begin(); 
+         itF != discFaces.end(); itF++){
+
+    std::vector<MElement*> allElements;
+    for (unsigned int i = 0; i < (*itF)->getNumMeshElements(); i++)
+      allElements.push_back((*itF)->getMeshElement(i));
+
+    std::vector<std::vector<MElement*> >  conRegions;
+    int nbFaces = connectedRegions (allElements, conRegions);
+    
+    remove(*itF);
+    for (int ifa  = 0; ifa < nbFaces; ifa++){
+      std::vector<MElement*> myElements = conRegions[ifa];
+
+      int numF = maxFaceNum()+1;
+      discreteFace *f = new discreteFace(this, numF);
+      //printf("*** Created discreteFace %d \n", numF);
+      add(f);
+      newDiscFaces.push_back(f);
+
+      //fill new face with mesh vertices
+      std::set<MVertex*> allV;
+      for(unsigned int i = 0; i < myElements.size(); i++) {
+        MVertex *v0 = myElements[i]->getVertex(0);
+	MVertex *v1 = myElements[i]->getVertex(1);
+	MVertex *v2 = myElements[i]->getVertex(2);
+        f->triangles.push_back(new MTriangle( v0, v1, v2));
+        allV.insert(v0);  allV.insert(v1);  allV.insert(v2);
+        v0->setEntity(f); v1->setEntity(f); v2->setEntity(f);
+      }
+      f->mesh_vertices.insert(f->mesh_vertices.begin(), allV.begin(),allV.end());
+    }
+
+  }
+  discFaces = newDiscFaces;
+  //------
  
   createTopologyFromFaces(discFaces);
   
@@ -1102,64 +1226,7 @@ void GModel::createTopologyFromMesh()
   exportDiscreteGEOInternals();
 }
 
-static void recur_connect(MVertex *v,
-                          std::multimap<MVertex*,MEdge> &v2e,
-                          std::set<MEdge,Less_Edge> &group,
-                          std::set<MVertex*> &touched)
-{
-  if (touched.find(v) != touched.end()) return;
-
-  touched.insert(v);
-  for (std::multimap <MVertex*,MEdge>::iterator it = v2e.lower_bound(v); 
-       it != v2e.upper_bound(v) ; ++it){
-    group.insert(it->second);
-    for (int i = 0; i < it->second.getNumVertices(); ++i){
-      recur_connect (it->second.getVertex(i), v2e, group, touched);
-    }
-  }
-}
-
-// starting form a list of elements, returns lists of lists that are
-// all simply connected
-static void recur_connect_e(const MEdge &e,
-                            std::multimap<MEdge,MElement*, Less_Edge> &e2e,
-                            std::set<MElement*> &group,
-                            std::set<MEdge, Less_Edge> &touched)
-{
-  if (touched.find(e) != touched.end())return;
-  touched.insert(e);
-  for (std::multimap <MEdge,MElement*,Less_Edge>::iterator it = e2e.lower_bound(e);
-         it != e2e.upper_bound(e) ; ++it){
-    group.insert(it->second);
-    for (int i = 0; i < it->second->getNumEdges(); ++i){
-      recur_connect_e(it->second->getEdge(i), e2e, group, touched);
-    }
-  }
-}
-
-static int connected_bounds(std::set<MEdge, Less_Edge> &edges, 
-                            std::vector<std::vector<MEdge> > &boundaries)
-{
-  std::multimap<MVertex*,MEdge> v2e;
-  for(std::set<MEdge, Less_Edge>::iterator it = edges.begin(); it != edges.end(); it++){
-    for (int j=0;j<it->getNumVertices();j++){
-      v2e.insert(std::make_pair(it->getVertex(j),*it));
-    }
-  }
-
-  while (!v2e.empty()){
-    std::set<MEdge, Less_Edge> group;
-    std::set<MVertex*> touched;
-    recur_connect(v2e.begin()->first, v2e, group, touched);
-    std::vector<MEdge> temp;
-    temp.insert(temp.begin(), group.begin(), group.end());
-    boundaries.push_back(temp);
-    for (std::set<MVertex*>::iterator it = touched.begin() ; it != touched.end();++it)
-      v2e.erase(*it);
-  }
 
-  return boundaries.size();
-}
 
 void GModel::createTopologyFromFaces(std::vector<discreteFace*> &discFaces)
 {
@@ -1243,7 +1310,7 @@ void GModel::createTopologyFromFaces(std::vector<discreteFace*> &discFaces)
     }
 
     std::vector<std::vector<MEdge> > boundaries;
-    double nbBounds = connected_bounds(myEdges, boundaries);   
+    int nbBounds = connected_bounds(myEdges, boundaries);   
 
     for (int ib  = 0; ib < nbBounds; ib++){
       std::vector<MEdge> myLines = boundaries[ib];
diff --git a/Geo/GModelIO_Mesh.cpp b/Geo/GModelIO_Mesh.cpp
index f6771098d1..f91c9873e2 100644
--- a/Geo/GModelIO_Mesh.cpp
+++ b/Geo/GModelIO_Mesh.cpp
@@ -535,7 +535,9 @@ static void writeElementMSH(FILE *fp, GModel *model, T *ele, bool saveAll,
     for(unsigned int j = 0; j < physicals.size(); j++)
       ele->writeMSH(fp, version, binary, ++num, elementary, physicals[j],
                     parentNum, &ghosts);
-  (*newElemNumbers)(ele->getNum()) = num;
+
+  //(*newElemNumbers)(ele->getNum()) = num;
+
 }
 
 template<class T>
diff --git a/Mesh/multiscalePartition.cpp b/Mesh/multiscalePartition.cpp
index 2da3da6241..25003d7098 100644
--- a/Mesh/multiscalePartition.cpp
+++ b/Mesh/multiscalePartition.cpp
@@ -348,7 +348,7 @@ void multiscalePartition::partition(partitionLevel & level, int nbParts,
     int genus, AR, NB;
     getGenusAndRatio(regions[i], genus, AR, NB);
 
-    printLevel (nextLevel->elements, nextLevel->recur,nextLevel->region);  
+    //printLevel (nextLevel->elements, nextLevel->recur,nextLevel->region);  
 
     if (genus < 0) {
       Msg::Error("Genus partition is negative G=%d!", genus);
diff --git a/Solver/multiscaleLaplace.cpp b/Solver/multiscaleLaplace.cpp
index a0df0d798d..d23372a6b7 100644
--- a/Solver/multiscaleLaplace.cpp
+++ b/Solver/multiscaleLaplace.cpp
@@ -1006,7 +1006,6 @@ void multiscaleLaplace::parametrize (multiscaleLaplaceLevel & level){
       else  tooSmall.insert(tooSmall.begin(), regGoodSize[i].begin(),  regGoodSize[i].end());
     }
   }
- 
 
   //Add the not too small regions to the level.elements 
   std::vector<std::vector<MElement*> >  regions_, regions ;
diff --git a/benchmarks/boolean/nurbs.lua b/benchmarks/boolean/nurbs.lua
index ef04763fd8..97c3efa30d 100644
--- a/benchmarks/boolean/nurbs.lua
+++ b/benchmarks/boolean/nurbs.lua
@@ -21,7 +21,7 @@ g:addLine(v7,v2);
 
 g:addBezier (v1,v2, {{v3:x(),v3:y(),0},{v4:x(),v4:y(),0},{v5:x(),v5:y(),0},{v6:x(),v6:y(),0},{v7:x(),v7:y(),0}}); 
 g:addNURBS (v1,v2, {{v3:x(),v3:y(),0},{v4:x(),v4:y(),0},{v5:x(),v5:y(),0},{v6:x(),v6:y(),0},{v7:x(),v7:y(),0}},	   
-	   {0,0.25,0.5,0.75,1},{1,1,1,1,1,1,1},{4,1,1,1,4}); 							   
+	    {0,0.25,0.5,0.75,1}}},{1,1,1,1,1,1,1},{4,1,1,1,4}); 							   
 g:addNURBS (v1,v2, {{v3:x(),v3:y(),0},{v4:x(),v4:y(),0},{v5:x(),v5:y(),0},{v6:x(),v6:y(),0},{v7:x(),v7:y(),0}},	   
 	   {0,0.5,1},{1,1,1,1,1,1,1},{4,3,4}); 							   
 
-- 
GitLab