Mmg
Simplicial remeshers (mesh adaptation, isovalue discretization, lagrangian movement)
quality_2d.c
Go to the documentation of this file.
1/* =============================================================================
2** This file is part of the mmg software package for the tetrahedral
3** mesh modification.
4** Copyright (c) Bx INP/CNRS/Inria/UBordeaux/UPMC, 2004-
5**
6** mmg is free software: you can redistribute it and/or modify it
7** under the terms of the GNU Lesser General Public License as published
8** by the Free Software Foundation, either version 3 of the License, or
9** (at your option) any later version.
10**
11** mmg is distributed in the hope that it will be useful, but WITHOUT
12** ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13** FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
14** License for more details.
15**
16** You should have received a copy of the GNU Lesser General Public
17** License and of the GNU General Public License along with mmg (in
18** files COPYING.LESSER and COPYING). If not, see
19** <http://www.gnu.org/licenses/>. Please read their terms carefully and
20** use this copy of the mmg distribution only if you accept them.
21** =============================================================================
22*/
34#include "libmmg2d_private.h"
36
47 MMG5_pPoint p0,p1,p2;
48 double cal;
49
50 p0 = &mesh->point[pt->v[0]];
51 p1 = &mesh->point[pt->v[1]];
52 p2 = &mesh->point[pt->v[2]];
53
54 cal = MMG2D_quickarea(p0->c,p1->c,p2->c);
55 return cal;
56}
57
68double MMG2D_caltri_iso_3pt(double *a,double *b,double *c) {
69 double abx,aby,acx,acy,bcx,bcy,area,h1,h2,h3,hm;
70
71 abx = b[0] - a[0];
72 aby = b[1] - a[1];
73 acx = c[0] - a[0];
74 acy = c[1] - a[1];
75 bcx = c[0] - b[0];
76 bcy = c[1] - b[1];
77
78 /* orientation */
79 area = abx*acy - aby*acx;
80 if ( area <= 0.0 ) return 0.0;
81
82 /* edge lengths */
83 h1 = abx*abx + aby*aby;
84 h2 = acx*acx + acy*acy;
85 h3 = bcx*bcx + bcy*bcy;
86
87 hm = h1 + h2 + h3;
88
89 if ( hm > 0. ) {
90 return area / hm;
91 }
92 else {
93 return 0.0;
94 }
95}
96
108 double *a,*b,*c;
109
110 a = mesh->point[pt->v[0]].c;
111 b = mesh->point[pt->v[1]].c;
112 c = mesh->point[pt->v[2]].c;
113
114 return MMG2D_caltri_iso_3pt(a,b,c);
115}
116
117
118
119/* Compute quality of the triangle pt when the supplied metric is anisotropic;
120 return 0 in the case that the triangle has inverted orientation */
122 double abx,aby,acx,acy,bcx,bcy;
123 double *a,*b,*c,*ma,*mb,*mc;
124 double area,aream,hm,m[6],h1,h2,h3;
125 MMG5_int ipa,ipb,ipc;
126 int i;
127
128 ipa = pt->v[0];
129 ipb = pt->v[1];
130 ipc = pt->v[2];
131
132 a = mesh->point[ipa].c;
133 b = mesh->point[ipb].c;
134 c = mesh->point[ipc].c;
135
136 ma = &met->m[3*ipa];
137 mb = &met->m[3*ipb];
138 mc = &met->m[3*ipc];
139
140 /* Isotropic area of pt */
141 abx = b[0] - a[0];
142 aby = b[1] - a[1];
143 acx = c[0] - a[0];
144 acy = c[1] - a[1];
145 bcx = c[0] - b[0];
146 bcy = c[1] - b[1];
147 area = abx*acy - aby*acx;
148 if ( area <= 0.0 ) return 0.0;
149
150 for (i=0; i<3; i++) m[i] = (ma[i]+mb[i]+mc[i]) / 3.0;
151
152 /* Anisotropic edge lengths */
153 h1 = m[0]*abx*abx + m[2]*aby*aby + 2.0*m[1]*abx*aby;
154 h1 = h1 > 0.0 ? sqrt(h1) : 0.0;
155 h2 = m[0]*acx*acx + m[2]*acy*acy + 2.0*m[1]*acx*acy;
156 h2 = h2 > 0.0 ? sqrt(h2) : 0.0;
157 h3 = m[0]*bcx*bcx + m[2]*bcy*bcy + 2.0*m[1]*bcx*bcy;
158 h3 = h3 > 0.0 ? sqrt(h3) : 0.0;
159
160 hm = h1*h1 + h2*h2 + h3*h3;
161
162 /* Anisotropic volume of pt */
163 aream = sqrt(m[0]*m[2]-m[1]*m[1])*area;
164
165 /* Quality measure = (Vol_M(T) / (l(ab)^2+l(ac)^2+l(bc)^2)) */
166 if ( hm > 0. ) {
167 return aream/hm;
168 }
169 else {
170 return (0.0);
171 }
172}
173
184 MMG5_pTria pt;
185 double rap,rapmin,rapmax,rapavg,med,good;
186 int i,ir,imax,his[5];
187 MMG5_int k,iel,ok,nex;
188 static int8_t mmgWarn0;
189
190 if ( !mesh->nt ) {
191 return 1;
192 }
193
194 /* Compute triangle quality*/
195 for (k=1; k<=mesh->nt; k++) {
196 pt = &mesh->tria[k];
197 if( !MG_EOK(pt) ) continue;
198
199 if ( !met->m ) {
200 pt->qual = MMG2D_caltri_iso(mesh,met,pt);
201 }
202 else
203 pt->qual = MMG2D_caltri(mesh,met,pt);
204 }
205 if ( mesh->info.imprim <= 0 ) return 1;
206
207 rapmin = 2.0;
208 rapmax = 0.0;
209 rapavg = med = good = 0.0;
210 iel = 0;
211
212 for (k=0; k<5; k++) his[k] = 0;
213
214 nex = ok = 0;
215 for (k=1; k<=mesh->nt; k++) {
216 pt = &mesh->tria[k];
217 if( !MG_EOK(pt) ) {
218 nex++;
219 continue;
220 }
221 ok++;
222 if ( (!mmgWarn0) && (MMG2D_quickcal(mesh,pt) < 0.0) ) {
223 mmgWarn0 = 1;
224 fprintf(stderr," ## Warning: %s: at least 1 negative area\n",__func__);
225 }
226
227 if ( !met->m ) {
228 rap = MMG2D_ALPHAD * MMG2D_caltri_iso(mesh,met,pt);
229 }
230 else
231 rap = MMG2D_ALPHAD * MMG2D_caltri(mesh,met,pt);
232
233 if ( rap < rapmin ) {
234 rapmin = rap;
235 iel = ok;
236 }
237 if ( rap > 0.5 ) med++;
238 if ( rap > 0.12 ) good++;
239 if ( rap < MMG2D_BADKAL ) mesh->info.badkal = 1;
240 rapavg += rap;
241 rapmax = MG_MAX(rapmax,rap);
242 ir = MG_MIN(4,(int)(5.0*rap));
243 his[ir] += 1;
244 }
245
246#ifndef DEBUG
247 fprintf(stdout,"\n -- MESH QUALITY %" MMG5_PRId "\n",mesh->nt - nex);
248 fprintf(stdout," BEST %8.6f AVRG. %8.6f WRST. %8.6f (%" MMG5_PRId ")\n",
249 rapmax,rapavg / (mesh->nt-nex),rapmin,iel);
250#else
251 fprintf(stdout," BEST %e AVRG. %e WRST. %e (%" MMG5_PRId ")\n => %" MMG5_PRId " %" MMG5_PRId " %" MMG5_PRId "\n",
252 rapmax,rapavg / (mesh->nt-nex),rapmin,iel,
253 MMG5_indPt(mesh,mesh->tria[iel].v[0]),MMG5_indPt(mesh,mesh->tria[iel].v[1]),
254 MMG5_indPt(mesh,mesh->tria[iel].v[2]));
255#endif
256
257 /* print histo */
258 fprintf(stdout," HISTOGRAMM:");
259 fprintf(stdout," %6.2f %% > 0.12\n",100.0*(good/(float)(mesh->nt-nex)));
260 if ( abs(mesh->info.imprim) > 3 ) {
261 fprintf(stdout," %6.2f %% > 0.5\n",100.0*( med/(float)(mesh->nt-nex)));
262 imax = MG_MIN(4,(int)(5.*rapmax));
263 for (i=imax; i>=(int)(5*rapmin); i--) {
264 fprintf(stdout," %5.1f < Q < %5.1f %7d %6.2f %%\n",
265 i/5.,i/5.+0.2,his[i],100.*(his[i]/(float)(mesh->nt-nex)));
266 }
267 }
268
269 return MMG5_minQualCheck(iel,rapmin,1.);
270}
MMG5_pMesh * mesh
#define MMG2D_BADKAL
#define MMG2D_ALPHAD
int MMG5_minQualCheck(MMG5_int iel, double minqual, double alpha)
Definition: quality.c:343
double MMG2D_quickarea(double a[2], double b[2], double c[2])
Definition: tools.c:971
#define MG_EOK(pt)
#define MG_MIN(a, b)
#define MG_MAX(a, b)
double MMG2D_caltri_iso(MMG5_pMesh mesh, MMG5_pSol met, MMG5_pTria pt)
Definition: quality_2d.c:107
double MMG2D_caltri_iso_3pt(double *a, double *b, double *c)
Definition: quality_2d.c:68
double MMG2D_caltri_ani(MMG5_pMesh mesh, MMG5_pSol met, MMG5_pTria pt)
Definition: quality_2d.c:121
int MMG2D_outqua(MMG5_pMesh mesh, MMG5_pSol met)
Definition: quality_2d.c:183
double MMG2D_quickcal(MMG5_pMesh mesh, MMG5_pTria pt)
Definition: quality_2d.c:46
int8_t badkal
Definition: libmmgtypes.h:540
MMG mesh structure.
Definition: libmmgtypes.h:613
MMG5_Info info
Definition: libmmgtypes.h:659
MMG5_pPoint point
Definition: libmmgtypes.h:649
MMG5_int nt
Definition: libmmgtypes.h:620
MMG5_pTria tria
Definition: libmmgtypes.h:655
Structure to store vertices of an MMG mesh.
Definition: libmmgtypes.h:276
double c[3]
Definition: libmmgtypes.h:277
double * m
Definition: libmmgtypes.h:680
Structure to store triangles of a MMG mesh.
Definition: libmmgtypes.h:338
double qual
Definition: libmmgtypes.h:339
MMG5_int v[3]
Definition: libmmgtypes.h:340