diff --git a/Geo/GModelIO_Mesh.cpp b/Geo/GModelIO_Mesh.cpp index e837fb00a95e6c9e0a403fefda3f7cde4ceaea81..004bcf76e3710a1ece054793597b688ef891f661 100644 --- a/Geo/GModelIO_Mesh.cpp +++ b/Geo/GModelIO_Mesh.cpp @@ -10,6 +10,7 @@ #include <string> #include <sstream> #include <iomanip> +#include <cassert> #include "GModel.h" #include "GmshDefines.h" #include "MPoint.h" @@ -38,6 +39,8 @@ #include "PViewDataList.h" #endif +#define FAST_ELEMENTS 1 + void GModel::_storePhysicalTagsInEntities(int dim, std::map<int, std::map<int, std::string> > &map) { @@ -122,7 +125,8 @@ static MElement *createElementMSH(GModel *m, int num, int typeMSH, int physical, if(CTX::instance()->mesh.switchElementTags) { int tmp = reg; reg = physical; - physical = reg; +// physical = reg; + physical = tmp; } MElementFactory factory; @@ -132,7 +136,8 @@ static MElement *createElementMSH(GModel *m, int num, int typeMSH, int physical, Msg::Error("Unknown type of element %d", typeMSH); return NULL; } - + +#if (FAST_ELEMENTS!=1) switch(e->getType()){ case TYPE_PNT : elements[0][reg].push_back(e); break; @@ -156,6 +161,7 @@ static MElement *createElementMSH(GModel *m, int num, int typeMSH, int physical, elements[9][reg].push_back(e); break; default : Msg::Error("Wrong type of element"); return NULL; } +#endif int dim = e->getDim(); if(physical && (!physicals[dim].count(reg) || !physicals[dim][reg].count(physical))) @@ -397,7 +403,13 @@ int GModel::readMSH(const std::string &name) if(!fgets(str, sizeof(str), fp)) return 0; int numElements; + + std::map<int, MElement*> elems; std::map<int, MElement*> parents; + std::map<int, MElement*> domains; + std::map<int, int> elemreg; + std::map<int, int> elemphy; + std::set<MElement*> parentsOwned; sscanf(str, "%d", &numElements); Msg::Info("%d elements", numElements); @@ -451,33 +463,147 @@ int GModel::readMSH(const std::string &name) } MElement *p = NULL; bool own = false; - if(parent) { - if(parents.find(parent) == parents.end()){ - p = getParent(parent, type, elements); - if(p) parents[parent] = p; - else Msg::Error("Parent element %d not found for element %d", parent, num); - } - else p = parents.find(parent)->second; - if(parentsOwned.find(p) == parentsOwned.end()) { - own = true; - parentsOwned.insert(p); - } - } + + // search parent element + if (parent) + { +#if (FAST_ELEMENTS==1) + std::map<int, MElement* >::iterator ite = elems.find(parent); + if (ite == elems.end()) + Msg::Error("Parent element %d not found for element %d", parent, num); + else + { + p = ite->second; + parents[parent] = p; + } + std::set<MElement* >::iterator itpo = parentsOwned.find(p); + if (itpo == parentsOwned.end()) + { + own = true; + parentsOwned.insert(p); + } + assert(p != NULL); +#else + std::map<int, MElement* >::iterator itp = parents.find(parent); + if (itp == parents.end()) + { + p = getParent(parent, type, elements); + if (p) + parents[parent] = p; + else Msg::Error("Parent element %d not found for element %d", parent, num); + } + else + p = itp->second; + std::set<MElement* >::iterator itpo = parentsOwned.find(p); + if (itpo == parentsOwned.end()) + { + own = true; + parentsOwned.insert(p); + } +#endif + } + + // search domains MElement *doms[2] = {NULL, NULL}; - if(dom1) { + if(dom1) + { +#if (FAST_ELEMENTS==1) + std::map<int, MElement* >::iterator ite = elems.find(dom1); + if (ite == elems.end()) + Msg::Error("Domain element %d not found for element %d", dom1, num); + else + doms[0] = ite->second; + + ite = elems.find(dom2); + if (ite != elems.end()) + doms[1] = ite->second; + + if(!doms[0]) Msg::Error("Domain element %d not found for element %d", dom1, num); + if(dom2 && !doms[1]) Msg::Error("Domain element %d not found for element %d", dom2, num); +#else getDomains(dom1, dom2, type, elements, doms); if(!doms[0]) Msg::Error("Domain element %d not found for element %d", dom1, num); if(dom2 && !doms[1]) Msg::Error("Domain element %d not found for element %d", dom2, num); - } +#endif + } + MElement *e = createElementMSH(this, num, type, physical, elementary, partition, vertices, elements, physicals, own, p, doms[0], doms[1]); + +#if (FAST_ELEMENTS==1) + elems[num] = e; + elemreg[num] = elementary; + elemphy[num] = physical; +#endif + for(unsigned int j = 0; j < ghosts.size(); j++) _ghostCells.insert(std::pair<MElement*, short>(e, ghosts[j])); if(numElements > 100000) Msg::ProgressMeter(i + 1, numElements, "Reading elements"); delete [] indices; } + +#if (FAST_ELEMENTS==1) + for(int i = 0; i < 10; i++) + elements[i].clear(); + + std::map<int, MElement* >::iterator ite; + for (ite = elems.begin(); ite != elems.end(); ite++) + { + int num = ite->first; + MElement *e = ite->second; + if (parents.find(num) == parents.end()) + { + int reg; + + if (CTX::instance()->mesh.switchElementTags) { + reg = elemphy[num]; + } + else + { + reg = elemreg[num]; + } + + switch (e->getType()) + { + case TYPE_PNT : + elements[0][reg].push_back(e); + break; + case TYPE_LIN : + elements[1][reg].push_back(e); + break; + case TYPE_TRI : + elements[2][reg].push_back(e); + break; + case TYPE_QUA : + elements[3][reg].push_back(e); + break; + case TYPE_TET : + elements[4][reg].push_back(e); + break; + case TYPE_HEX : + elements[5][reg].push_back(e); + break; + case TYPE_PRI : + elements[6][reg].push_back(e); + break; + case TYPE_PYR : + elements[7][reg].push_back(e); + break; + case TYPE_POLYG : + elements[8][reg].push_back(e); + break; + case TYPE_POLYH : + elements[9][reg].push_back(e); + break; + default : + Msg::Error("Wrong type of element"); + exit(1); + } + } + } +#endif } else{ int numElementsPartial = 0; @@ -583,7 +709,7 @@ int GModel::readMSH(const std::string &name) // store the physical tags for(int i = 0; i < 4; i++) _storePhysicalTagsInEntities(i, physicals[i]); - + fclose(fp); return postpro ? 2 : 1;