"share/examples/parallel_plate/parallel_plate.geo" did not exist on "ad399724d242e24c466597cabef39c0fc6ec47eb"
Newer
Older
// Gmsh - Copyright (C) 1997-2009 C. Geuzaine, J.-F. Remacle
// See the LICENSE.txt file for license information. Please report all
// bugs and problems to <gmsh@geuz.org>.
#include "GmshConfig.h"

Christophe Geuzaine
committed
#include "GmshMessage.h"
#include "Numeric.h"
#include "Context.h"
#include "MLine.h"
#include "MTriangle.h"
#include "MQuadrangle.h"
#include "MTetrahedron.h"
#include "MHexahedron.h"
#include "MPrism.h"
#include "MPyramid.h"
#include "BoundaryLayers.h"

Christophe Geuzaine
committed
#if !defined(HAVE_NO_POST)
#include "PView.h"
#include "PViewData.h"

Christophe Geuzaine
committed
#endif
static MVertex* isEquivalentTo(std::multimap<MVertex*, MVertex*> &m, MVertex *v)
std::multimap<MVertex*, MVertex*>::iterator it = m.lower_bound(v);
std::multimap<MVertex*, MVertex*>::iterator ite = m.upper_bound(v);
if (it == ite) return v;
MVertex *res = it->second; ++it;
while (it !=ite){
if (res < v) return isEquivalentTo(m, res);
static void buildASetOfEquivalentMeshVertices(GFace *gf,
std::multimap<MVertex*, MVertex *> &equivalent,
std::map<GVertex*, MVertex*> &bm)
{
// an edge is degenerated when is length is considered to be
// zero. In some cases, a model edge can be considered as too
// small an is ignored.
// for taking that into account, we loop over the edges
// and create pairs of MVertices that are considered as
// equal.
std::list<GEdge*> edges = gf->edges();
std::list<GEdge*> emb_edges = gf->embeddedEdges();
std::list<GEdge*>::iterator it = edges.begin();
while(it != edges.end()){
if((*it)->isMeshDegenerated()){
MVertex *va = *((*it)->getBeginVertex()->mesh_vertices.begin());
MVertex *vb = *((*it)->getEndVertex()->mesh_vertices.begin());
if (va != vb){
equivalent.insert(std::make_pair(va, vb));
equivalent.insert(std::make_pair(vb, va));
bm[(*it)->getBeginVertex()] = va;
bm[(*it)->getEndVertex()] = vb;
printf("%d equivalent to %d\n", va->getNum(), vb->getNum());
}
}
++it;
}
it = emb_edges.begin();
while(it != emb_edges.end()){
if((*it)->isMeshDegenerated()){
MVertex *va = *((*it)->getBeginVertex()->mesh_vertices.begin());
MVertex *vb = *((*it)->getEndVertex()->mesh_vertices.begin());
if (va != vb){
equivalent.insert(std::make_pair(va, vb));
equivalent.insert(std::make_pair(vb, va));
bm[(*it)->getBeginVertex()] = va;
bm[(*it)->getEndVertex()] = vb;
}
}
++it;
}
}
struct geomTresholdVertexEquivalence
{
// Initial MVertex associated to one given MVertex
std::map<GVertex*, MVertex*> backward_map;
// initiate the forward and backward maps
geomTresholdVertexEquivalence(GModel *g);
// restores the initial state
~geomTresholdVertexEquivalence ();
};
geomTresholdVertexEquivalence::geomTresholdVertexEquivalence(GModel *g)
std::multimap<MVertex*, MVertex*> equivalenceMap;
for (GModel::fiter it = g->firstFace(); it != g->lastFace(); ++it)
buildASetOfEquivalentMeshVertices(*it, equivalenceMap, backward_map);
// build the structure that identifiate geometrically equivalent
// mesh vertices.
for (std::map<GVertex*, MVertex*>::iterator it = backward_map.begin();
it != backward_map.end(); ++it){
GVertex *g = it->first;
MVertex *v = it->second;
MVertex *other = isEquivalentTo(equivalenceMap, v);
printf("Finally : %d equivalent to %d\n", v->getNum(), other->getNum());
g->mesh_vertices.clear();
g->mesh_vertices.push_back(other);
}
}
}
geomTresholdVertexEquivalence::~geomTresholdVertexEquivalence()
for (std::map<GVertex*, MVertex*>::iterator it = backward_map.begin();
it != backward_map.end() ; ++it){
GVertex *g = it->first;
MVertex *v = it->second;
g->mesh_vertices.clear();
g->mesh_vertices.push_back(v);
}
}
double &gamma, double &gammaMin, double &gammaMax,
double &eta, double &etaMin, double &etaMax,
double &rho, double &rhoMin, double &rhoMax,
double &disto, double &distoMin, double &distoMax,
double quality[4][100])
for(unsigned int i = 0; i < ele.size(); i++){
double g = ele[i]->gammaShapeMeasure();
gamma += g;
gammaMin = std::min(gammaMin, g);
gammaMax = std::max(gammaMax, g);
double e = ele[i]->etaShapeMeasure();
eta += e;
etaMin = std::min(etaMin, e);
etaMax = std::max(etaMax, e);
double r = ele[i]->rhoShapeMeasure();
rho += r;
rhoMin = std::min(rhoMin, r);
rhoMax = std::max(rhoMax, r);
double d = ele[i]->distoShapeMeasure();
disto += d;
distoMin = std::min(distoMin, d);
distoMax = std::max(distoMax, d);
for(int j = 0; j < 100; j++){
if(g > j / 100. && g <= (j + 1) / 100.) quality[0][j]++;
if(e > j / 100. && e <= (j + 1) / 100.) quality[1][j]++;
if(r > j / 100. && r <= (j + 1) / 100.) quality[2][j]++;
if(d > j / 100. && d <= (j + 1) / 100.) quality[3][j]++;
void GetStatistics(double stat[50], double quality[4][100])
for(int i = 0; i < 50; i++) stat[i] = 0.;
stat[0] = m->getNumVertices();
stat[1] = m->getNumEdges();
stat[2] = m->getNumFaces();
stat[3] = m->getNumRegions();
std::map<int, std::vector<GEntity*> > physicals[4];
stat[45] = physicals[0].size() + physicals[1].size() +
physicals[2].size() + physicals[3].size();
for(GModel::eiter it = m->firstEdge(); it != m->lastEdge(); ++it)
stat[4] += (*it)->mesh_vertices.size();
for(GModel::fiter it = m->firstFace(); it != m->lastFace(); ++it){
stat[5] += (*it)->mesh_vertices.size();
stat[7] += (*it)->triangles.size();
stat[8] += (*it)->quadrangles.size();
}
for(GModel::riter it = m->firstRegion(); it != m->lastRegion(); ++it){
stat[6] += (*it)->mesh_vertices.size();
stat[9] += (*it)->tetrahedra.size();
stat[10] += (*it)->hexahedra.size();
stat[11] += (*it)->prisms.size();
stat[12] += (*it)->pyramids.size();
}
stat[13] = CTX::instance()->meshTimer[0];
stat[14] = CTX::instance()->meshTimer[1];
stat[15] = CTX::instance()->meshTimer[2];
if(quality){
for(int i = 0; i < 3; i++)
for(int j = 0; j < 100; j++)
double gamma=0., gammaMin=1., gammaMax=0.;
double eta=0., etaMin=1., etaMax=0.;
double rho=0., rhoMin=1., rhoMax=0.;
double disto=0., distoMin=1., distoMax=0.;
if (m->firstRegion() == m->lastRegion()){
for(GModel::fiter it = m->firstFace(); it != m->lastFace(); ++it){
GetQualityMeasure((*it)->quadrangles, gamma, gammaMin, gammaMax,
eta, etaMin, etaMax, rho, rhoMin, rhoMax,
disto, distoMin, distoMax, quality);
GetQualityMeasure((*it)->triangles, gamma, gammaMin, gammaMax,
eta, etaMin, etaMax, rho, rhoMin, rhoMax,
disto, distoMin, distoMax, quality);
}
}
else{
for(GModel::riter it = m->firstRegion(); it != m->lastRegion(); ++it){
GetQualityMeasure((*it)->tetrahedra, gamma, gammaMin, gammaMax,
eta, etaMin, etaMax, rho, rhoMin, rhoMax,
disto, distoMin, distoMax, quality);
GetQualityMeasure((*it)->hexahedra, gamma, gammaMin, gammaMax,
eta, etaMin, etaMax, rho, rhoMin, rhoMax,
disto, distoMin, distoMax, quality);
GetQualityMeasure((*it)->prisms, gamma, gammaMin, gammaMax,
eta, etaMin, etaMax, rho, rhoMin, rhoMax,
disto, distoMin, distoMax,quality);
GetQualityMeasure((*it)->pyramids, gamma, gammaMin, gammaMax,
eta, etaMin, etaMax, rho, rhoMin, rhoMax,
disto, distoMin, distoMax, quality);
double N = stat[9] + stat[10] + stat[11] + stat[12];
stat[17] = N ? gamma / N : 0.;
stat[18] = gammaMin;
stat[19] = gammaMax;
stat[20] = N ? eta / N : 0.;
stat[21] = etaMin;
stat[22] = etaMax;
stat[23] = N ? rho / N : 0;
stat[24] = rhoMin;
stat[25] = rhoMax;
stat[46] = N ? disto / N : 0;
stat[47] = distoMin;
stat[48] = distoMax;

Christophe Geuzaine
committed
#if !defined(HAVE_NO_POST)
stat[26] = PView::list.size();
for(unsigned int i = 0; i < PView::list.size(); i++) {
PViewData *data = PView::list[i]->getData(true);
stat[27] += data->getNumPoints();
stat[28] += data->getNumLines();
stat[29] += data->getNumTriangles();
stat[30] += data->getNumQuadrangles();
stat[31] += data->getNumTetrahedra();
stat[32] += data->getNumHexahedra();
stat[33] += data->getNumPrisms();
stat[34] += data->getNumPyramids();

Christophe Geuzaine
committed
#endif
if(CTX::instance()->expertMode || !m->getNumVertices()) return false;
// try to detect obvious mistakes in characteristic lenghts (one of
// the most common cause for erroneous bug reports on the mailing
// list)
for(GModel::viter it = m->firstVertex(); it != m->lastVertex(); ++it)
sumAllLc += (*it)->prescribedMeshSizeAtVertex() * CTX::instance()->mesh.lcFactor;
sumAllLc /= (double)m->getNumVertices();
if(!sumAllLc || pow(CTX::instance()->lc / sumAllLc, dim) > 1.e10)
return !Msg::GetBinaryAnswer

Christophe Geuzaine
committed
("Your choice of characteristic lengths will likely produce a very\n"
"large mesh. Do you really want to continue?\n\n"
"(To disable this warning in the future, select `Enable expert mode'\n"
"in the option dialog.)",
"Continue", "Cancel");
return false;
}
static bool CancelDelaunayHybrid(GModel *m)
{
if(CTX::instance()->expertMode) return false;

Christophe Geuzaine
committed
int n = 0;
for(GModel::riter it = m->firstRegion(); it != m->lastRegion(); ++it)
n += (*it)->getNumMeshElements();
if(n)
return !Msg::GetBinaryAnswer
("You are trying to generate a mixed structured/unstructured grid using\n"
"the 3D Delaunay algorithm. This algorithm cannot garantee that the\n"
"final mesh will be conforming. You should probably use the 3D Frontal\n"
"algorithm instead. Do you really want to continue?\n\n"

Christophe Geuzaine
committed
"(To disable this warning in the future, select `Enable expert mode'\n"
"in the option dialog.)",
"Continue", "Cancel");
}
Msg::StatusBar(1, true, "Meshing 1D...");
double t1 = Cpu();
std::for_each(m->firstEdge(), m->lastEdge(), meshGEdge());
double t2 = Cpu();
CTX::instance()->meshTimer[0] = t2 - t1;
Msg::Info("Mesh 1D complete (%g s)", CTX::instance()->meshTimer[0]);
}
if(CTX::instance()->createAppendMeshStatReport == 1)
statreport = fopen(CTX::instance()->meshStatReportFileName.c_str(), "w");
else if(CTX::instance()->createAppendMeshStatReport == 2)
statreport = fopen(CTX::instance()->meshStatReportFileName.c_str(), "a");
double worst = 1, best = 0, avg = 0;
double e_long = 0, e_short = 1.e22, e_avg = 0;
int nTotT = 0, nTotE = 0, nTotGoodLength = 0, nTotGoodQuality = 0;
int nUnmeshed = 0, numFaces = 0;
Msg::Info("2D Mesh Statistics :");
for(GModel::fiter it = m->firstFace() ; it != m->lastFace(); ++it){
worst = std::min((*it)->meshStatistics.worst_element_shape, worst);
best = std::max((*it)->meshStatistics.best_element_shape, best);
avg += (*it)->meshStatistics.average_element_shape * (*it)->meshStatistics.nbTriangle;
e_avg += (*it)->meshStatistics.efficiency_index;//* (*it)->meshStatistics.nbEdge;
e_long = std::max((*it)->meshStatistics.longest_edge_length, e_long);
e_short = std::min((*it)->meshStatistics.smallest_edge_length, e_short);
if ((*it)->meshStatistics.status == GFace::FAILED ||
nTotT += (*it)->meshStatistics.nbTriangle;
nTotE += (*it)->meshStatistics.nbEdge;
nTotGoodLength += (*it)->meshStatistics.nbGoodLength;
nTotGoodQuality += (*it)->meshStatistics.nbGoodQuality;
numFaces++;
}
if(CTX::instance()->createAppendMeshStatReport == 1){
"#t\t\tQavg\t\tQbest\t\tQworst\t\t#Q>90\t\t#Q>90/#t\t"
"#e\t\ttau\t\t#Egood\t\t#Egood/#e\tCPU\n");
fprintf(statreport,"\t%16s\t%d\t\t%d\t\t", m->getName().c_str(), numFaces, nUnmeshed);
fprintf(statreport,"%d\t\t%8.7f\t%8.7f\t%8.7f\t%d\t\t%8.7f\t",
nTotT, avg / (double)nTotT, best, worst, nTotGoodQuality,
(double)nTotGoodQuality / nTotT);
fprintf(statreport,"%d\t\t%8.7f\t%d\t\t%8.7f\t%8.1f\n",
(double)nTotGoodLength / nTotE, CTX::instance()->meshTimer[1]);
Msg::StatusBar(1, true, "Meshing 2D...");
double t1 = Cpu();
// skip short mesh edges
geomTresholdVertexEquivalence inst (m);
// boundary layers are special: their generation (including vertices
// and curve meshes) is global as it depends on a smooth normal
// field generated from the surface mesh of the source surfaces
if(!Mesh2DWithBoundaryLayers(m)){
std::for_each(m->firstFace(), m->lastFace(), meshGFace());
int nIter = 0;
while(1){
meshGFace mesher;
int nbPending = 0;
for(GModel::fiter it = m->firstFace() ; it!=m->lastFace(); ++it){
if ((*it)->meshStatistics.status == GFace::PENDING){
mesher(*it);
nbPending++;
}
double t2 = Cpu();
CTX::instance()->meshTimer[1] = t2 - t1;
Msg::Info("Mesh 2D complete (%g s)", CTX::instance()->meshTimer[1]);
}
std::vector<std::vector<GRegion*> > &connected)
{
// FIXME: need to split region vector into connected components here!
connected.push_back(delaunay);
}
Msg::StatusBar(1, true, "Meshing 3D...");
double t1 = Cpu();
std::for_each(m->firstRegion(), m->lastRegion(), meshGRegionExtruded());
// then subdivide if necessary (unfortunately the subdivision is a
// global operation, which can require changing the surface mesh!)
// then mesh all the non-delaunay regions
std::vector<GRegion*> delaunay;
std::for_each(m->firstRegion(), m->lastRegion(), meshGRegion(delaunay));

Christophe Geuzaine
committed
// warn if attempting to use Delaunay for mixed meshes
if(delaunay.size() && CancelDelaunayHybrid(m)) return;
// and finally mesh the delaunay regions (again, this is global; but
// we mesh each connected part separately for performance and mesh
// quality reasons)
std::vector<std::vector<GRegion*> > connected;
FindConnectedRegions(delaunay, connected);
for(unsigned int i = 0; i < connected.size(); i++){
double t2 = Cpu();
CTX::instance()->meshTimer[2] = t2 - t1;
Msg::Info("Mesh 3D complete (%g s)", CTX::instance()->meshTimer[2]);
}
void OptimizeMeshNetgen(GModel *m)
{
Msg::StatusBar(1, true, "Optimizing 3D with Netgen...");
double t1 = Cpu();
std::for_each(m->firstRegion(), m->lastRegion(), optimizeMeshGRegionNetgen());
double t2 = Cpu();
Msg::Info("Mesh 3D optimization with Netgen complete (%g s)", t2 - t1);
Msg::StatusBar(1, true, "Optimizing 3D...");
std::for_each(m->firstRegion(), m->lastRegion(), optimizeMeshGRegionGmsh());
Msg::Info("Mesh 3D optimization complete (%g s)", t2 - t1);
if(CTX::instance()->lock) {
Msg::Info("I'm busy! Ask me that later...");
CTX::instance()->lock = 1;
std::for_each(m->firstRegion(), m->lastRegion(), adaptMeshGRegion());
std::for_each(m->firstRegion(), m->lastRegion(), adaptMeshGRegion());
std::for_each(m->firstRegion(), m->lastRegion(), adaptMeshGRegion());
std::for_each(m->firstRegion(), m->lastRegion(), adaptMeshGRegion());
std::for_each(m->firstRegion(), m->lastRegion(), adaptMeshGRegion());
std::for_each(m->firstRegion(), m->lastRegion(), adaptMeshGRegion());
std::for_each(m->firstRegion(), m->lastRegion(), adaptMeshGRegion());
std::for_each(m->firstRegion(), m->lastRegion(), adaptMeshGRegion());
std::for_each(m->firstRegion(), m->lastRegion(), adaptMeshGRegion());
std::for_each(m->firstRegion(), m->lastRegion(), adaptMeshGRegion());
double t2 = Cpu();
Msg::Info("Mesh Adaptation complete (%g s)", t2 - t1);
if(CTX::instance()->lock) {
Msg::Info("I'm busy! Ask me that later...");
CTX::instance()->lock = 1;
Msg::ResetErrorCounter();
// Initialize pseudo random mesh generator with the same seed
srand(1);
std::for_each(m->firstRegion(), m->lastRegion(), deMeshGRegion());
std::for_each(m->firstFace(), m->lastFace(), deMeshGFace());
}
std::for_each(m->firstRegion(), m->lastRegion(), deMeshGRegion());
}
}
if(m->getMeshStatus() >= 2)
std::for_each(m->firstFace(), m->lastFace(), orientMeshGFace());
// Optimize quality of 3D tet mesh
if(m->getMeshStatus() == 3){
for(int i = 0; i < std::max(CTX::instance()->mesh.optimize,
CTX::instance()->mesh.optimizeNetgen); i++){
if(CTX::instance()->mesh.optimize > i) OptimizeMesh(m);
if(CTX::instance()->mesh.optimizeNetgen > i) OptimizeMeshNetgen(m);
if(m->getMeshStatus() == 2 && CTX::instance()->mesh.algoSubdivide == 1)
RefineMesh(m, CTX::instance()->mesh.secondOrderLinear, true);
else if(m->getMeshStatus() == 3 && CTX::instance()->mesh.algoSubdivide == 2)
RefineMesh(m, CTX::instance()->mesh.secondOrderLinear, false, true);
if(m->getMeshStatus() && CTX::instance()->mesh.order > 1)
SetOrderN(m, CTX::instance()->mesh.order, CTX::instance()->mesh.secondOrderLinear,
CTX::instance()->mesh.secondOrderIncomplete);
Msg::Info("%d vertices %d elements",
m->getNumMeshVertices(), m->getNumMeshElements());
Msg::PrintErrorCounter("Mesh generation error summary");
CTX::instance()->lock = 0;
CTX::instance()->mesh.changed = ENT_ALL;