Mmg
Simplicial remeshers (mesh adaptation, isovalue discretization, lagrangian movement)
mmg2.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*/
23
36#include "mmgcommon_private.h"
37#include "mmgexterns_private.h"
38
46static MMG5_int MMG5_InvMat_key(MMG5_pInvMat pim,int ref) {
47 return (ref - pim->offset);
48}
49
58static int MMG5_InvMat_code(int k,int attr) {
59 return 4*(k+1)+attr;
60}
61
69static int MMG5_InvMat_getIndex(MMG5_pInvMat pim,int ref) {
70 int key = MMG5_InvMat_key(pim,ref);
71 /* The parent index is stored as 4*(k+1) */
72 return (pim->lookup[key] / 4 - 1);
73}
74
84static int MMG5_InvMat_getAttrib(MMG5_pInvMat pim,int ref) {
85 int key = MMG5_InvMat_key(pim,ref);
86 /* The nosplit/split/plus/minus attribute is stored as the rest of the
87 * integer division. */
88 return (pim->lookup[key] % 4);
89}
90
98static int MMG5_InvMat_check(MMG5_pInvMat pim,int key) {
99 return pim->lookup[key] ? 0 : 1;
100}
101
111static void MMG5_InvMat_error(MMG5_pInvMat pim,int ref,int k) {
112 fprintf(stderr,"\n ## Warning: Overwrite material reference %d"
113 " (from LSReferences line %d) with another entry from LSReferences line %d."
114 ,ref,MMG5_InvMat_getIndex(pim,ref)+1,k+1);
115 fprintf(stderr,"\n Check your LSReferences table: if possible,"
116 " each material reference should be unique,\n"
117 " if not possible, you may"
118 " encounter unexpected issues (wrong domain mapping or erroneous"
119 " detection of non-manifold level-set)!\n");
120}
121
130 MMG5_pMat pm;
131 int key;
132
133 /* Get material */
134 pm = &mesh->info.mat[k];
135
137 key = MMG5_InvMat_key(pim,pm->ref);
138 if( !MMG5_InvMat_check(pim,key) ) {
139 MMG5_InvMat_error(pim,pm->ref,k);
140 }
141 pim->lookup[key] = MMG5_InvMat_code(k,pm->dospl);
142
150 if( pm->dospl ) {
151 key = MMG5_InvMat_key(pim,pm->rin);
152 if( !MMG5_InvMat_check(pim,key) ) {
153 MMG5_InvMat_error(pim,pm->rin,k);
154 }
155 pim->lookup[key] = MMG5_InvMat_code(k,MG_MINUS);
156
157 key = MMG5_InvMat_key(pim,pm->rex);
158 if( !MMG5_InvMat_check(pim,key) ) {
159 MMG5_InvMat_error(pim,pm->rex,k);
160 }
161 pim->lookup[key] = MMG5_InvMat_code(k,MG_PLUS);
162 }
163
164 return 1;
165}
166
177static int MMG5_InvMat_getParent(MMG5_pMesh mesh,MMG5_pInvMat pim,MMG5_int ref,MMG5_int *pref) {
178 MMG5_pMat pm;
179 int k;
180
181 /* The parent index */
182 k = MMG5_InvMat_getIndex(pim,ref);
183
184 /* Material not found in the table */
185 if( k == -1 ) {
186 fprintf(stderr,"\n ## Warning: %s: material %" MMG5_PRId " not found in table.\n",
187 __func__,ref);
188 fprintf(stderr," Please ensure that you provide all mesh"
189 " references in the material map\n"
190 " (that is, the whole list of"
191 " surface materials in lssurf mode,\n"
192 " and the whole list of domain"
193 " materials in ls mode).\n" );
194 return 0;
195 }
196
197 /* Get the material in the lookup table and return the parent reference */
198 pm = &mesh->info.mat[k];
199 *pref = pm->ref;
200 return 1;
201}
202
213int MMG5_getStartRef(MMG5_pMesh mesh,MMG5_int ref,MMG5_int *pref) {
214 MMG5_pInvMat pim;
215
216 /* No multi-materials nor single material reference preservation */
217 if( !mesh->info.nmat ) {
218 *pref = 0;
219 return 1;
220 }
221
222 /* Get parent of material */
223 pim = &mesh->info.invmat;
224
225 /* Return 0 if the material does not exist, 1 otherwise */
226 if( !MMG5_InvMat_getParent(mesh,pim,ref,pref) )
227 return 0;
228 else
229 return 1;
230}
231
239 MMG5_int ref,pref;
240
241 /* Scan all references in the table limits, some may not exist */
242 for( ref = pim->offset; ref < pim->offset + pim->size; ref++ ) {
243 if( !MMG5_InvMat_getParent(mesh,pim,ref,&pref) ) continue;
244 printf("%" MMG5_PRId " (%" MMG5_PRId "): %" MMG5_PRId " %d\n",ref,
245 MMG5_InvMat_key(pim,ref),pref,
246 MMG5_InvMat_getAttrib(pim,ref));
247 }
248}
249
327 MMG5_pMat pm;
328 MMG5_pInvMat pim;
329 int k;
330 MMG5_int refmax,refmin;
331
332 /* Nothing to do if no multi-material option */
333 if( !mesh->info.nmat ) return 1;
334
335 /* Error if all the materials have not been set */
336 if( mesh->info.nmati < mesh->info.nmat ) {
337 fprintf(stderr,"\n ## Error: %s: Only %d materials out of %d have been set.\n",
338 __func__,mesh->info.nmati,mesh->info.nmat);
339 return 0;
340 }
341
342 /* Get pointer to the structure */
343 pim = &mesh->info.invmat;
344
345 /* Initialize the max and min reference */
346 refmax = 0;
347
348 refmin = MMG5_INTMAX;
349
350 /* Look for the max/min reference provided in material table */
351 for( k = 0; k < mesh->info.nmat; k++ ) {
352 pm = &mesh->info.mat[k];
353 /* Update max and min val for original ref */
354 if( pm->ref > refmax ) refmax = pm->ref;
355 if( pm->ref < refmin ) refmin = pm->ref;
356 if( !pm->dospl ) continue;
357 /* Update max and min val with interior ref */
358 if( pm->rin > refmax ) refmax = pm->rin;
359 if( pm->rin < refmin ) refmin = pm->rin;
360 /* Update max and min val with exterior ref */
361 if( pm->rex > refmax ) refmax = pm->rex;
362 if( pm->rex < refmin ) refmin = pm->rex;
363 }
364
365 /* Look for the max/min reference of tetra, triangles and edges provided
366 * inside the mesh (worst case to avoid memory error when checking the
367 * the inverse map). Looking at vertices is useless as
368 * we will never check for the mapping of reference of vertices */
369 for ( k=1; k<=mesh->ne; ++k ) {
370 if( mesh->tetra[k].ref > refmax ) refmax = mesh->tetra[k].ref;
371 if( mesh->tetra[k].ref < refmin ) refmin = mesh->tetra[k].ref;
372 }
373 for ( k=1; k<=mesh->nt; ++k ) {
374 if( mesh->tria[k].ref > refmax ) refmax = mesh->tria[k].ref;
375 if( mesh->tria[k].ref < refmin ) refmin = mesh->tria[k].ref;
376 }
377 for ( k=1; k<=mesh->na; ++k ) {
378 if( mesh->edge[k].ref > refmax ) refmax = mesh->edge[k].ref;
379 if( mesh->edge[k].ref < refmin ) refmin = mesh->edge[k].ref;
380 }
381
382 /* Get span of the lookup table */
383 pim->offset = refmin;
384 pim->size = refmax - refmin + 1;
385 assert( pim->size > 0 );
386
387 /* Allocate lookup table */
388 MMG5_ADD_MEM(mesh,pim->size*sizeof(int),"materials lookup table",return 0);
389 MMG5_SAFE_CALLOC(pim->lookup,pim->size,int,return 0);
390
391 /* Fill lookup table */
392 for( k = 0; k < mesh->info.nmat; k++ ) {
393 if( !MMG5_InvMat_set(mesh,pim,k) )
394 return 0;
395 }
396
397 // MMG5_InvMat_print(mesh,pim);
398 return 1;
399}
400
412int MMG5_isSplit(MMG5_pMesh mesh,MMG5_int ref,MMG5_int *refint,MMG5_int *refext) {
413 MMG5_pInvMat pim;
414 MMG5_pMat pm;
415 int k;
416
417 /* Default case: split with references MG_MINUS, MG_PLUS */
418 if( !mesh->info.nmat ) {
419 *refint = MG_MINUS;
420 *refext = MG_PLUS;
421 return 1;
422 }
423
424 /* Check in the info->mat table if reference ref is supplied by the user */
425 pim = &mesh->info.invmat;
426 k = MMG5_InvMat_getIndex(pim,ref);
427
428 assert( k != -1 );
429 pm = &mesh->info.mat[k];
430
431 if ( !pm->dospl ) {
432 return 0;
433 } else {
434 *refint = pm->rin;
435 *refext = pm->rex;
436 return 1;
437 }
438}
439
448int MMG5_isNotSplit(MMG5_pMesh mesh,MMG5_int ref) {
449 MMG5_pInvMat pim;
450
451 /* Split material by default if not in multi-material mode */
452 if( !mesh->info.nmat ) return 0;
453
454 /* Look in the table otherwise */
455 pim = &mesh->info.invmat;
456 if( !MMG5_InvMat_getAttrib(pim,ref) )
457 return 0;
458 else
459 return 1;
460
461}
462
472int MMG5_isLevelSet(MMG5_pMesh mesh,MMG5_int ref0,MMG5_int ref1) {
473 MMG5_pInvMat pim;
474 int8_t found0,found1;
475
476 /* Check whether multimaterial case or not */
477 if( mesh->info.nmat ) {
478 /* Retrieve levelset information from the lookup table */
479 pim = &mesh->info.invmat;
480 found0 = MMG5_InvMat_getAttrib(pim,ref0);
481 found1 = MMG5_InvMat_getAttrib(pim,ref1);
482
483 if( (found0+found1) == (MG_MINUS+MG_PLUS) ) return 1;
484 else return 0;
485
486 } else {
487 /* Single material, check references directly */
488 if( ( ref0 == MG_MINUS && ref1 == MG_PLUS ) ||
489 ( ref1 == MG_MINUS && ref0 == MG_PLUS ) ) return 1;
490 else return 0;
491 }
492}
493
504 MMG5_pTria pt,pt1;
505 MMG5_pPoint p0;
506 double v1,v2,*tmp;
507 MMG5_int k,kk,iel,ns,nc,ip,ip1,ip2,npl,nmn;
508 int ilist;
509 int8_t i,j,j1,j2;
510 MMG5_int list[MMG5_TRIA_LMAX+2];
511
512 /* Allocate memory for tmp */
513 MMG5_ADD_MEM(mesh,(mesh->npmax+1)*sizeof(double),"temporary table",
514 fprintf(stderr," Exit program.\n");
515 return 0);
516 MMG5_SAFE_CALLOC(tmp,mesh->npmax+1,double,return 0);
517
518 /* Reset point flags */
519 for (k=1; k<=mesh->np; k++)
520 mesh->point[k].flag = 0;
521
522 /* Snap values of sol that are close to 0 to 0 exactly */
523 ns = nc = 0;
524 for (k=1; k<=mesh->np; k++) {
525 p0 = &mesh->point[k];
526 if ( !MG_VOK(p0) ) continue;
527 if ( fabs(sol->m[k]) < MMG5_EPS ) {
528 tmp[k] = sol->m[k];
529 p0->flag = 1;
530 sol->m[k] = 0.0;
531 ns++;
532 }
533 }
534
535 /* Check that the snapping process has not led to a nonmanifold situation */
536 for (k=1; k<=mesh->nt; k++) {
537 pt = &mesh->tria[k];
538 if ( !MG_EOK(pt) ) continue;
539 for (i=0; i<3; i++) {
540 ip = pt->v[i];
541 ip1 = pt->v[MMG5_inxt2[i]];
542 ip2 = pt->v[MMG5_iprv2[i]];
543
544 p0 = &mesh->point[ip];
545 v1 = sol->m[ip1];
546 v2 = sol->m[ip2];
547
548 /* Catch a snapped point by a triangle where there is a sign change: use
549 * the same convention than in ismaniball to evaluate sign changes. If
550 * travelled in direct sense from a triangle, an edge is considered
551 * without sign change if first vertex is 0. It has a sign change if
552 * second vertex is 0 or if we have 2 vertices with different signs
553 * (otherwise a 0 vertex leads to count 2 sign changes instead of one). */
554 int smsgn = ((fabs(v2) < MMG5_EPS) || MG_SMSGN(v1,v2)) ? 1 : 0;
555 if ( p0->flag && !smsgn ) {
556 if ( !MMG5_ismaniball(mesh,sol,k,i) ) {
557 if ( tmp[ip] < 0.0 )
558 sol->m[ip] = -100.0*MMG5_EPS;
559 else
560 sol->m[ip] = 100.0*MMG5_EPS;
561 nc++;
562 }
563 p0->flag = 0;
564 }
565 }
566 }
567
568 /* Check that the ls function does not show isolated spots with 0 values (without sign changes) */
569 for (k=1; k<=mesh->nt; k++) {
570 pt = &mesh->tria[k];
571 if ( !MG_EOK(pt) ) continue;
572 for (i=0; i<3; i++) {
573 ip = pt->v[i];
574 if ( fabs(sol->m[ip]) >= MMG5_EPS ) continue;
575 npl = nmn = 0;
576 int8_t opn; //unused
577 ilist = MMG5_boulet(mesh,k,i,list,1,&opn);
578 for(kk=0; kk<ilist; kk++) {
579 iel = list[kk] / 3;
580 j = list[kk] % 3;
581 j1 = MMG5_inxt2[j];
582 j2 = MMG5_iprv2[i];
583 pt1 = &mesh->tria[iel];
584 ip1 = pt1->v[j1];
585 ip2 = pt1->v[j2];
586 if ( sol->m[ip1] >= MMG5_EPS ) npl = 1;
587 else if ( sol->m[ip1] <= -MMG5_EPS ) nmn = 1;
588
589 if ( sol->m[ip2] >= MMG5_EPS ) npl = 1;
590 else if ( sol->m[ip2] <= -MMG5_EPS ) nmn = 1;
591 }
592
593 if ( npl == 1 && nmn == 0 )
594 sol->m[ip] = 100.0*MMG5_EPS;
595 else if ( npl == 0 && nmn == 1 )
596 sol->m[ip] = -100.0*MMG5_EPS;
597 }
598 }
599
600 if ( (abs(mesh->info.imprim) > 5 || mesh->info.ddebug) && ns+nc > 0 )
601 fprintf(stdout," %8" MMG5_PRId " points snapped, %" MMG5_PRId " corrected\n",ns,nc);
602
603 /* memory free */
605
606 return 1;
607}
608
623int MMG5_ismaniball(MMG5_pMesh mesh, MMG5_pSol sol, MMG5_int start, int8_t istart) {
624 MMG5_pTria pt;
625 double v1, v2;
626 MMG5_int refstart,*adja,k,ip1,ip2,end1;
627 int8_t i,i1,smsgn;
628 static int8_t mmgWarn=0;
629
630 k = start;
631 refstart = mesh->tria[k].ref;
632 i = MMG5_inxt2[istart];
633
634 /* First loop: stop if an external boundary, or a change in signs (or a 0) is met
635 recall that MG_SMGSGN(a,b) = 1 provided a*b >0 */
636 do{
637 adja = &mesh->adja[3*(k-1)+1];
638 k = adja[i] / 3;
639 i1 = adja[i] % 3;
640 i = MMG5_iprv2[i1];
641
642 if ( k==0 ) break;
643
644 pt = &mesh->tria[k];
645
646 ip1 = pt->v[i1];
647 ip2 = pt->v[i];
648
649 v1 = sol->m[ip1];
650 v2 = sol->m[ip2];
651
652 if ( (fabs(v1) < MMG5_EPS) && (fabs(v2) < MMG5_EPS) ) {
653 /* Do not authorize a snap that leads to a triangle with only 0 vertices */
654 return 0;
655 }
656
657 /* Authorize change of references only provided the boundary reference is mesh->info.isoref */
658 if ( pt->ref != refstart && pt->edg[i1] != mesh->info.isoref ) {
659 smsgn = 0;
660 k = 0;
661 } else {
662 /* Evaluation of sign change using following convention: If
663 * travelled in direct sense from a triangle, an edge is considered
664 * without sign change if first vertex is 0. It has a sign change if
665 * second vertex is 0 or if we have 2 vertices with different signs
666 * (otherwise a 0 vertex leads to count 2 sign changes instead of one). */
667 smsgn = (fabs(v1) < MMG5_EPS) || ( (fabs(v2) > MMG5_EPS) && MG_SMSGN(v1,v2) ) ? 1 : 0;
668 }
669 }
670 while ( smsgn && (k != start) );
671
672 if ( k==start ) {
673 /* Complete ball has been travelled without crossing a boundary or finding a
674 * sign change: we are in the special case where v1 = v2 = v[istart] = 0 in
675 * tria start. In this case, test MG_SMSGN(v1,v2) returns 0 while smsgn is
676 * computed to 1, which is non consistent. */
677 assert ( smsgn );
678 return 0;
679 }
680
681 end1 = k;
682 k = start;
683 i = MMG5_iprv2[istart];
684
685 /* Second loop: same travel in the opposite sense */
686 do{
687 adja = &mesh->adja[3*(k-1)+1];
688 k = adja[i] / 3;
689 i1 = adja[i] % 3;
690 i = MMG5_inxt2[i1];
691
692 if ( k==0 ) break;
693
694 pt = &mesh->tria[k];
695 ip1 = pt->v[i1];
696 ip2 = pt->v[i];
697
698 v1 = sol->m[ip1];
699 v2 = sol->m[ip2];
700
701 if ( (fabs(v1) < MMG5_EPS) && (fabs(v2) < MMG5_EPS) ) {
702 /* Do not authorize a snap that leads to a triangle with only 0 vertices */
703 return 0;
704 }
705
706 if ( pt->ref != refstart && pt->edg[i1] != mesh->info.isoref ) {
707 smsgn = 0;
708 k = 0;
709 } else {
710 /* Evaluation of sign change using following convention: If
711 * travelled in undirect sense from a triangle, an edge is considered
712 * without sign change if second vertex is 0. It has a sign change if
713 * first vertex is 0 or if we have 2 vertices with different signs
714 * (it allows to evaluate the same splitted edges than the first loop). */
715 smsgn = (fabs(v2) < MMG5_EPS) || ( (fabs(v1) > MMG5_EPS) && MG_SMSGN(v1,v2) ) ? 1 : 0;
716 }
717 }
718 while ( smsgn && (k != start) );
719
720 assert ( k!=start );
721
722 /* If first stop was due to an external boundary, the second one must too
723 (k==end1==0); else, the final triangle for the first travel must be that of
724 the second one */
725 if ( k != end1 ) {
726 if ( !mmgWarn ) {
727 mmgWarn = 1;
728 fprintf(stderr,"\n ## Warning: %s: unsnap at least 1 point "
729 "(point %" MMG5_PRId " in tri %" MMG5_PRId ").\n",__func__,
730 MMG5_indPt(mesh,mesh->tria[start].v[istart]),MMG5_indElt(mesh,start));
731 }
732 return 0;
733 }
734 return 1;
735}
736
747static inline
748double MMG5_voltri(MMG5_pMesh mesh,MMG5_int ip0,MMG5_int ip1,MMG5_int ip2) {
749 MMG5_pPoint p0,p1,p2;
750 double vol;
751
752 p0 = &mesh->point[ip0];
753 p1 = &mesh->point[ip1];
754 p2 = &mesh->point[ip2];
755
756 vol = (p1->c[0]-p0->c[0])*(p2->c[1]-p0->c[1]) - (p1->c[1]-p0->c[1])*(p2->c[0]-p0->c[0]);
757 vol = 0.5*fabs(vol);
758
759 return vol;
760}
761
774static inline
775double MMG5_vfrac(MMG5_pMesh mesh,MMG5_pSol sol,MMG5_int k,int pm) {
776 MMG5_pTria pt;
777 MMG5_pPoint ppt[3];
778 double v[3],vfp,vfm,lam,area,eps,o1[2],o2[2];
779 MMG5_int ip[3],nplus,nminus,nzero;
780 int8_t i,i0,i1,i2,imin1,iplus1,iz;
781
782 eps = MMG5_EPS*MMG5_EPS;
783 pt = &mesh->tria[k];
784
785 ip[0] = pt->v[0];
786 ip[1] = pt->v[1];
787 ip[2] = pt->v[2];
788
789 ppt[0] = &mesh->point[ip[0]];
790 ppt[1] = &mesh->point[ip[1]];
791 ppt[2] = &mesh->point[ip[2]];
792
793 v[0] = sol->m[ip[0]];
794 v[1] = sol->m[ip[1]];
795 v[2] = sol->m[ip[2]];
796
797 /* Identify number of zero, positive and negative vertices, and corresponding indices */
798 nplus = nminus = nzero = 0;
799 imin1 = iplus1 = iz = -1;
800
801 for (i=0; i<3; i++) {
802 if ( fabs(v[i]) < eps ) {
803 nzero++;
804 if ( iz < 0 ) iz = i;
805 }
806 else if ( v[i] >= eps ) {
807 nplus++;
808 if ( iplus1 < 0 ) iplus1 = i;
809 }
810 else {
811 nminus++;
812 if ( imin1 < 0 ) imin1 = i;
813 }
814 }
815
816 /* Degenerate case */
817 if ( nzero == 3 ) return 0.0;
818
819 /* Whole triangle is positive */
820 if ( nminus == 0 ) {
821 vfp = (ppt[1]->c[0]-ppt[0]->c[0])*(ppt[2]->c[1]-ppt[0]->c[1]) - (ppt[1]->c[1]-ppt[0]->c[1])*(ppt[2]->c[0]-ppt[0]->c[0]);
822 vfp = 0.5*fabs(vfp);
823 if ( pm == 1 ) return vfp;
824 else return 0.0;
825 }
826
827 /* Whole triangle is negative */
828 if ( nplus == 0 ) {
829 vfm = (ppt[1]->c[0]-ppt[0]->c[0])*(ppt[2]->c[1]-ppt[0]->c[1]) - (ppt[1]->c[1]-ppt[0]->c[1])*(ppt[2]->c[0]-ppt[0]->c[0]);
830 vfm = 0.5*fabs(vfm);
831 if ( pm == -1 ) return vfm;
832 else return 0.0;
833 }
834
835 /* Exactly one vertex is negative */
836 if ( nminus == 1 ) {
837 i0 = imin1;
838 i1 = MMG5_inxt2[i0];
839 i2 = MMG5_iprv2[i0];
840
841 lam = v[i0] / (v[i0]-v[i1]);
842 o1[0] = ppt[i0]->c[0] + lam*(ppt[i1]->c[0]-ppt[i0]->c[0]);
843 o1[1] = ppt[i0]->c[1] + lam*(ppt[i1]->c[1]-ppt[i0]->c[1]);
844
845 lam = v[i0] / (v[i0]-v[i2]);
846 o2[0] = ppt[i0]->c[0] + lam*(ppt[i2]->c[0]-ppt[i0]->c[0]);
847 o2[1] = ppt[i0]->c[1] + lam*(ppt[i2]->c[1]-ppt[i0]->c[1]);
848
849 vfm = (o1[0]-ppt[i0]->c[0])*(o2[1]-ppt[i0]->c[1]) - (o1[1]-ppt[i0]->c[1])*(o2[0]-ppt[i0]->c[0]);
850 vfm = 0.5*fabs(vfm);
851
852 if ( pm == -1 ) return vfm;
853 else {
854 area = (ppt[1]->c[0]-ppt[0]->c[0])*(ppt[2]->c[1]-ppt[0]->c[1]) - (ppt[1]->c[1]-ppt[0]->c[1])*(ppt[2]->c[0]-ppt[0]->c[0]);
855 area = 0.5*fabs(area);
856 vfp = area-vfm;
857 return vfp;
858 }
859 }
860
861 /* Exactly one vertex is positive */
862 if ( nplus == 1 ) {
863 i0 = iplus1;
864 i1 = MMG5_inxt2[i0];
865 i2 = MMG5_iprv2[i0];
866
867 lam = v[i0] / (v[i0]-v[i1]);
868 o1[0] = ppt[i0]->c[0] + lam*(ppt[i1]->c[0]-ppt[i0]->c[0]);
869 o1[1] = ppt[i0]->c[1] + lam*(ppt[i1]->c[1]-ppt[i0]->c[1]);
870
871 lam = v[i0] / (v[i0]-v[i2]);
872 o2[0] = ppt[i0]->c[0] + lam*(ppt[i2]->c[0]-ppt[i0]->c[0]);
873 o2[1] = ppt[i0]->c[1] + lam*(ppt[i2]->c[1]-ppt[i0]->c[1]);
874
875 vfp = (o1[0]-ppt[i0]->c[0])*(o2[1]-ppt[i0]->c[1]) - (o1[1]-ppt[i0]->c[1])*(o2[0]-ppt[i0]->c[0]);
876 vfp = 0.5*fabs(vfp);
877
878 if ( pm == 1 ) return vfp;
879 else {
880 area = (ppt[1]->c[0]-ppt[0]->c[0])*(ppt[2]->c[1]-ppt[0]->c[1]) - (ppt[1]->c[1]-ppt[0]->c[1])*(ppt[2]->c[0]-ppt[0]->c[0]);
881 area = 0.5*fabs(area);
882 vfm = area-vfp;
883 return vfm;
884 }
885 }
886
887 /* Should not pass here */
888 return 0.0;
889}
890
892static inline
893int MMG5_isbr(MMG5_pMesh mesh,MMG5_int ref) {
894 MMG5_int k;
895
896 for(k=0; k<mesh->info.nbr; k++)
897 if ( ref == mesh->info.br[k] ) return 1;
898
899 return 0;
900}
901
913 MMG5_pTria pt,pt1,pt2;
914 double volc,voltot,v0,v1,v2;
915 MMG5_int k,kk,l,ll,ncp,ncm,ip0,ip1,ip2,cur,ipile,*pile,*adja,base;
916 int8_t i,i1,i2,onbr;
917
918 ncp = 0;
919 ncm = 0;
920
921 /* Erase triangle flags */
922 for (k=1; k<=mesh->nt; k++) mesh->tria[k].flag = 0;
923
924 /* Calculate volume of the total mesh */
925 voltot = 0.0;
926 for (k=1; k<=mesh->nt; k++) {
927 pt = &mesh->tria[k];
928 if ( !MG_EOK(pt) ) continue;
929 ip0 = pt->v[0];
930 ip1 = pt->v[1];
931 ip2 = pt->v[2];
932 voltot += MMG5_voltri(mesh,ip0,ip1,ip2);
933 }
934
935 /* Memory allocation for pile */
936 MMG5_ADD_MEM(mesh,(mesh->nt+1)*sizeof(MMG5_int),"temporary table",
937 printf(" Exit program.\n");
938 return 0);
939 MMG5_SAFE_CALLOC(pile,mesh->nt+1,MMG5_int,return 0);
940
941 /* Investigate only positive connected components */
942 base = ++mesh->base;
943
944 for (k=1; k<=mesh->nt; k++) {
945 ipile = 0;
946 volc = 0.0;
947 pt = &mesh->tria[k];
948 if ( !MG_EOK(pt) ) continue;
949 if ( pt->flag == base ) continue;
950
951 /* Checks signs of the LS function at the 3 vertices of pt */
952 ip0 = pt->v[0];
953 ip1 = pt->v[1];
954 ip2 = pt->v[2];
955
956 v0 = sol->m[ip0];
957 v1 = sol->m[ip1];
958 v2 = sol->m[ip2];
959
960 if ( v0 <= 0.0 && v1 <= 0.0 && v2 <= 0.0 ) continue;
961
962 /* Add triangle to pile if one vertex is > 0 */
963 pt->flag = base;
964 pile[ipile] = k;
965 ipile++;
966 if ( ipile > mesh->nt ) {
967 fprintf(stderr,"\n ## Problem in length of pile; function rmc.\n"
968 " Check that the level-set intersect the mesh.\n"
969 " Exit program.\n");
970
971 return 0;
972 }
973
974 /* Pile up all the positive connected component attached to the first triangle */
975 cur = 0;
976 do {
977 kk = pile[cur];
978 pt1 = &mesh->tria[kk];
979
980 /* Add local volume fraction of the positive subdomain to volc */
981 volc += MMG5_vfrac(mesh,sol,kk,1);
982
983 /* Add adjacent triangles to kk via positive vertices to the pile, if need be */
984 adja = &mesh->adja[3*(kk-1)+1];
985 for (i=0; i<3; i++) {
986 ip0 = pt1->v[i];
987 if ( sol->m[ip0] <= 0.0 ) continue;
988
989 i1 = MMG5_inxt2[i];
990 i2 = MMG5_inxt2[i1];
991
992 /* First neighbor of positive vertex i */
993 ll = adja[i1] / 3;
994 if ( ll ) {
995 pt2 = &mesh->tria[ll];
996 if ( pt2->flag != base ) {
997 pt2->flag = base;
998 pile[ipile] = ll;
999 ipile++;
1000 if ( ipile > mesh->nt ) {
1001 fprintf(stderr,"\n ## Problem in length of pile; function rmc. Exit program.\n");
1002 return 0;
1003 }
1004 }
1005 }
1006
1007 /* Second neighbor of positive vertex i */
1008 ll = adja[i2] / 3;
1009 if ( ll ) {
1010 pt2 = &mesh->tria[ll];
1011 if ( pt2->flag != base ) {
1012 pt2->flag = base;
1013 pile[ipile] = ll;
1014 ipile++;
1015 if ( ipile > mesh->nt ) {
1016 fprintf(stderr,"\n ## Problem in length of pile; function rmc. Exit program.\n");
1017 return 0;
1018 }
1019 }
1020 }
1021 }
1022 }
1023 while ( ++cur < ipile );
1024
1025 /* Remove connected component if its volume is too small */
1026 if ( volc < mesh->info.rmc*voltot ) {
1027 for (l=0; l<ipile; l++) {
1028 pt1 = &mesh->tria[pile[l]];
1029 for (i=0; i<3; i++) {
1030 ip0 = pt1->v[i];
1031 if ( sol->m[ip0] > 0.0 ) sol->m[ip0] = -100*MMG5_EPS;
1032 }
1033 }
1034 ncp++;
1035 }
1036
1037 }
1038
1039 /* Investigate only negative connected components */
1040 base = ++mesh->base;
1041
1042 for (k=1; k<=mesh->nt; k++) {
1043 ipile = 0;
1044 volc = 0.0;
1045 pt = &mesh->tria[k];
1046 if ( !MG_EOK(pt) ) continue;
1047 if ( pt->flag == base ) continue;
1048
1049 /* Checks signs of the LS function at the 3 vertices of pt */
1050 ip0 = pt->v[0];
1051 ip1 = pt->v[1];
1052 ip2 = pt->v[2];
1053
1054 v0 = sol->m[ip0];
1055 v1 = sol->m[ip1];
1056 v2 = sol->m[ip2];
1057
1058 if ( v0 >= 0.0 && v1 >= 0.0 && v2 >= 0.0 ) continue;
1059
1060 /* Pile up all the negative connected component attached to the first triangle */
1061 pt->flag = base;
1062 pile[ipile] = k;
1063 ipile++;
1064 if ( ipile > mesh->nt ) {
1065 fprintf(stderr,"\n ## Problem in length of pile; function rmc. Exit program.\n");
1066 return 0;
1067 }
1068
1069 cur = 0;
1070 do {
1071 kk = pile[cur];
1072 pt1 = &mesh->tria[kk];
1073
1074 /* Add local volume fraction of the negative subdomain to volc */
1075 volc += MMG5_vfrac(mesh,sol,kk,-1);
1076
1077 /* Add adjacent triangles to kk via negative vertices to the pile, if need be */
1078 adja = &mesh->adja[3*(kk-1)+1];
1079 for (i=0; i<3; i++) {
1080 ip0 = pt1->v[i];
1081 if ( sol->m[ip0] >= 0.0 ) continue;
1082
1083 i1= MMG5_inxt2[i];
1084 i2 = MMG5_inxt2[i1];
1085
1086 /* First neighbor of negative vertex i */
1087 ll = adja[i1] / 3;
1088 if ( ll ) {
1089 pt2 = &mesh->tria[ll];
1090 if ( pt2->flag != base ) {
1091 pt2->flag = base;
1092 pile[ipile] = ll;
1093 ipile++;
1094 if ( ipile > mesh->nt ) {
1095 fprintf(stderr,"\n ## Problem in length of pile; function rmc. Exit program.\n");
1096 return 0;
1097 }
1098 }
1099 }
1100
1101 /* Second neighbor of negative vertex i */
1102 ll = adja[i2] / 3;
1103 if ( ll ) {
1104 pt2 = &mesh->tria[ll];
1105 if ( pt2->flag != base ) {
1106 pt2->flag = base;
1107 pile[ipile] = ll;
1108 ipile++;
1109 if ( ipile > mesh->nt ) {
1110 fprintf(stderr,"\n ## Problem in length of pile; function rmc. Exit program.\n");
1111 return 0;
1112 }
1113 }
1114 }
1115
1116 }
1117 }
1118 while ( ++cur < ipile );
1119
1120 /* Remove connected component if its volume is too small */
1121 if ( volc < mesh->info.rmc*voltot ) {
1122 for (l=0; l<ipile; l++) {
1123 pt1 = &mesh->tria[pile[l]];
1124 for (i=0; i<3; i++) {
1125 ip0 = pt1->v[i];
1126 if ( sol->m[ip0] < 0.0 ) sol->m[ip0] = 100*MMG5_EPS;
1127 }
1128 }
1129 ncm++;
1130 }
1131
1132 /* Remove connected component if it is not attached to one base reference */
1133 if ( mesh->info.nbr ) {
1134 onbr = 0;
1135 for (l=0; l<ipile; l++) {
1136 pt1 = &mesh->tria[pile[l]];
1137 for (i=0; i<3; i++) {
1138 if ( MMG5_isbr(mesh,pt1->edg[i]) ) {
1139 i1 = MMG5_inxt2[i];
1140 i2 = MMG5_inxt2[i1];
1141 ip1 = pt1->v[i1];
1142 if ( sol->m[ip1] < 0.0 ) {
1143 onbr = 1;
1144 break;
1145 }
1146 ip2 = pt1->v[i2];
1147 if ( sol->m[ip2] < 0.0 ) {
1148 onbr = 1;
1149 break;
1150 }
1151 }
1152 }
1153 if ( onbr ) break;
1154 }
1155
1156 if ( !onbr ) {
1157 for (l=0; l<ipile; l++) {
1158 pt1 = &mesh->tria[pile[l]];
1159 for (i=0; i<3; i++) {
1160 ip0 = pt1->v[i];
1161 if ( sol->m[ip0] < 0.0 ) sol->m[ip0] = 100*MMG5_EPS;
1162 }
1163 }
1164 ncm++;
1165 }
1166 }
1167
1168 }
1169
1170 /* Erase triangle flags */
1171 for (k=1; k<=mesh->nt; k++) mesh->tria[k].flag = 0;
1172
1173 /* Release memory */
1174 MMG5_DEL_MEM(mesh,pile);
1175
1176 if ( mesh->info.imprim > 0 || mesh->info.ddebug ) {
1177 printf("\n *** Removed %" MMG5_PRId " positive parasitic bubbles and %" MMG5_PRId " negative parasitic bubbles\n",ncp,ncm);
1178 }
1179
1180 return(1);
1181}
1182
1190 MMG5_pTria pt;
1191 MMG5_pPoint p0;
1192 MMG5_int ref,k;
1193 int8_t i;
1194
1195 for (k=1; k<=mesh->nt; k++) {
1196 pt = &mesh->tria[k];
1197 if ( !pt->v[0] ) continue;
1198
1199 for (i=0; i<3; i++) {
1200 p0 = &mesh->point[pt->v[i]];
1201 if ( pt->edg[i] == mesh->info.isoref ) pt->edg[i] = 0;
1202 if ( p0->ref == mesh->info.isoref ) p0->ref = 0;
1203 }
1204 }
1205
1206 /* Reset the triangle references to their initial distribution */
1207 for (k=1; k<=mesh->nt; k++) {
1208 pt = &mesh->tria[k];
1209 if ( !pt->v[0] ) continue;
1210 if( !MMG5_getStartRef(mesh,pt->ref,&ref) ) return 0;
1211 pt->ref = ref;
1212 }
1213
1214 return 1;
1215}
1216
1226 MMG5_pTria pt;
1227 double v,v1;
1228 int ier;
1229 MMG5_int k,ip,ip1,ref,refint,refext;
1230 int8_t i,i1,i2,nmn,npl,nz;
1231
1232 for (k=1; k<=mesh->nt; k++) {
1233 pt = &mesh->tria[k];
1234 if ( !MG_EOK(pt) ) continue;
1235
1236 ref = pt->ref;
1237 nmn = npl = nz = 0;
1238 for (i=0; i<3; i++) {
1239 ip = pt->v[i];
1240 v = sol->m[ip];
1241
1242 if ( v > 0.0 )
1243 npl++;
1244 else if ( v < 0.0 )
1245 nmn++;
1246 else
1247 nz++;
1248 }
1249
1250 assert(nz < 3);
1251
1252 /* Keep the initial triangle references of the mesh if iso==2, set
1253 * positive and negative ls refs otherwise */
1254 if ( mesh->info.iso != 2 ) {
1255
1256 /* find if current reference should be splitted and the new positive and negative refs */
1257 ier = MMG5_isSplit(mesh,ref,&refint,&refext);
1258 if ( ier ) {
1259 if ( npl ) {
1260 assert( !nmn );
1261 pt->ref = refext;
1262 }
1263 else {
1264 assert ( nmn );
1265 pt->ref = refint;
1266 }
1267 }
1268 }
1269
1270 /* Set mesh->info.isoref ref at ls edges and at the points of these edges */
1271 if ( nz == 2 ) {
1272 for (i=0; i<3; i++) {
1273 ip = pt->v[MMG5_inxt2[i]];
1274 ip1 = pt->v[MMG5_iprv2[i]];
1275 v = sol->m[ip];
1276 v1 = sol->m[ip1];
1277 if ( v == 0.0 && v1 == 0.0) {
1278 pt->edg[i] = mesh->info.isoref;
1279 pt->tag[i] |= MG_REF;
1280 i1 = MMG5_inxt2[i];
1281 i2 = MMG5_inxt2[i1];
1282 mesh->point[pt->v[i1]].ref = mesh->info.isoref;
1283 mesh->point[pt->v[i2]].ref = mesh->info.isoref;
1284 }
1285 }
1286 }
1287
1288 }
1289
1290 return 1;
1291}
1292
1304int MMG5_chkmaniball(MMG5_pMesh mesh, MMG5_int start, int8_t istart) {
1305 MMG5_int refstart,*adja,k;
1306 int8_t i,i1;
1307
1308 k = start;
1309 i = istart;
1310
1311 i1 = MMG5_iprv2[i];
1312
1313
1314 MMG5_pTria pt = &mesh->tria[start];
1315 assert( MG_EDG(pt->tag[i1]) && (pt->edg[i1]==mesh->info.isoref) );
1316
1318 refstart = pt->ref;
1319 do {
1320 adja = &mesh->adja[3*(k-1)+1];
1321 i1 = MMG5_inxt2[i];
1322
1323 k = adja[i1] / 3;
1324 i = adja[i1] % 3;
1325
1326 if ( !k ) break;
1327
1328 if ( mesh->info.iso !=2 ) {
1330 if ( mesh->tria[k].ref != refstart) break;
1331 }
1332 else {
1335 if ( mesh->tria[k].edg[i]==mesh->info.isoref ) break;
1336 }
1337 i = MMG5_inxt2[i];
1338 }
1339 while ( k!=start );
1340
1341 assert(k!=start); //unexpected case
1342
1346 if ( k == 0 ) {
1347 k = start;
1348 i = istart;
1349
1350 adja = &mesh->adja[3*(k-1)+1];
1351 i1 = MMG5_iprv2[i];
1352 k = adja[i1] / 3;
1353 i = adja[i1] % 3;
1354 i = MMG5_iprv2[i];
1355
1357 if ( k == 0 ) return 1;
1358
1359 do {
1360 adja = &mesh->adja[3*(k-1)+1];
1361 i1 = MMG5_iprv2[i];
1362
1363 k = adja[i1] / 3;
1364 i = adja[i1] % 3;
1365
1366 if ( !k ) break;
1367
1368 if ( mesh->info.iso !=2 ) {
1369 /* Normal or multi-material mode: check for change in triangle references */
1370 if ( mesh->tria[k].ref == refstart) break;
1371 }
1372 else {
1373 /* Input reference preservation mode (mmgs --keep-ref option): Check if
1374 * we cross an isoref edge */
1375 if ( mesh->tria[k].edg[i]==mesh->info.isoref ) break;
1376 }
1377 i = MMG5_iprv2[i];
1378 }
1379 while ( k!=start );
1380
1381 assert(k!=start); //unexpected case
1382
1383 return !k;
1384 }
1385
1387 i = MMG5_inxt2[i];
1388 do {
1389 adja = &mesh->adja[3*(k-1)+1];
1390 i1 = MMG5_inxt2[i];
1391
1392 k = adja[i1] / 3;
1393 i = adja[i1] % 3;
1394
1395 if ( !k ) break;
1396
1397 if ( mesh->info.iso !=2 ) {
1398 /* Check tria ref change */
1399 if ( mesh->tria[k].ref == refstart) break;
1400 }
1401 else {
1402 /* Check if we cross an isoref edge */
1403 if ( mesh->tria[k].edg[i]==mesh->info.isoref ) break;
1404 }
1405
1406 i = MMG5_inxt2[i];
1407 }
1408 while ( k!=start );
1409
1411 if ( k != start )
1412 return 0;
1413
1414 return 1;
1415}
1416
1425 MMG5_pTria pt,pt1;
1426 MMG5_int *adja,k;
1427 MMG5_int cnt,iel;
1428 int8_t i,i1;
1429 static int8_t mmgWarn = 0;
1430
1432 for (k=1; k<=mesh->nt; k++) {
1433 pt = &mesh->tria[k];
1434 if ( !MG_EOK(pt) ) continue;
1435
1436 adja = &mesh->adja[3*(k-1)+1];
1437 cnt = 0;
1438 for (i=0; i<3; i++) {
1439 iel = adja[i] / 3;
1440
1441 if (!iel ) {
1442 cnt++;
1443 continue;
1444 }
1445 else {
1446 if ( mesh->info.iso !=2 ) {
1447 /* Multi-material mode may lead to have only 1 isoref edge around a
1448 point (due to nosplit option): check tria ref change */
1449 pt1 = &mesh->tria[iel];
1450 if ( pt1->ref != pt->ref ) cnt++;
1451 }
1452 else {
1453 /* keep-ref mode: check if isoref edge */
1454 if ( pt->edg[i] == mesh->info.isoref ) cnt++;
1455 }
1456 }
1457 }
1458 if( cnt == 3 ) {
1459 if ( !mmgWarn ) {
1460 mmgWarn = 1;
1461 fprintf(stderr,"\n ## Warning: %s: at least 1 triangle with 3 boundary"
1462 " edges.\n",__func__);
1463 }
1464 }
1465 }
1466
1470 for (k=1; k<=mesh->nt; k++) {
1471 pt = &mesh->tria[k];
1472 if ( !MG_EOK(pt) ) continue;
1473
1474 for (i=0; i<3; i++) {
1475 adja = &mesh->adja[3*(k-1)+1];
1476 iel = adja[i] / 3;
1477
1478 if (! iel ) continue;
1479
1480 if ( mesh->info.iso !=2 ) {
1481 /* Check change of tria ref */
1482 pt1 = &mesh->tria[iel];
1483 if ( pt->ref == pt1->ref || pt->edg[i]!= mesh->info.isoref ) continue;
1484 }
1485 else {
1486 /* Check isoref edge only */
1487 if ( pt->edg[i] != mesh->info.isoref ) continue;
1488 }
1489
1490 i1 = MMG5_inxt2[i];
1491 if ( !MMG5_chkmaniball(mesh,k,i1) ) {
1492 fprintf(stderr," *** Topological problem\n");
1493 fprintf(stderr," non manifold curve at point %" MMG5_PRId "\n",pt->v[i1]);
1494 fprintf(stderr," non manifold curve at tria %" MMG5_PRId " (ip %d)\n", MMG5_indElt(mesh,k),i1);
1495 return 0;
1496 }
1497 }
1498 }
1499
1500 if ( mesh->info.imprim > 0 || mesh->info.ddebug )
1501 fprintf(stdout," *** Manifold implicit surface.\n");
1502
1503 return 1;
1504}
int ier
tmp[*strlen0]
MMG5_pMesh MMG5_pSol * sol
MMG5_pMesh * mesh
MMG5_Info info
int MMG5_boulet(MMG5_pMesh mesh, MMG5_int start, int ip, MMG5_int *list, int8_t s, int8_t *opn)
Definition: boulep.c:363
#define MMG5_EPS
#define MG_PLUS
Definition: libmmgtypes.h:77
#define MG_MINUS
Definition: libmmgtypes.h:82
static int MMG5_InvMat_code(int k, int attr)
Definition: mmg2.c:58
static void MMG5_InvMat_error(MMG5_pInvMat pim, int ref, int k)
Definition: mmg2.c:111
int MMG5_rmc(MMG5_pMesh mesh, MMG5_pSol sol)
Definition: mmg2.c:912
static int MMG5_InvMat_check(MMG5_pInvMat pim, int key)
Definition: mmg2.c:98
int MMG5_isLevelSet(MMG5_pMesh mesh, MMG5_int ref0, MMG5_int ref1)
Definition: mmg2.c:472
int MMG5_isNotSplit(MMG5_pMesh mesh, MMG5_int ref)
Definition: mmg2.c:448
int MMG5_ismaniball(MMG5_pMesh mesh, MMG5_pSol sol, MMG5_int start, int8_t istart)
Definition: mmg2.c:623
static int MMG5_isbr(MMG5_pMesh mesh, MMG5_int ref)
Definition: mmg2.c:893
static int MMG5_InvMat_getParent(MMG5_pMesh mesh, MMG5_pInvMat pim, MMG5_int ref, MMG5_int *pref)
Definition: mmg2.c:177
static MMG5_int MMG5_InvMat_key(MMG5_pInvMat pim, int ref)
Definition: mmg2.c:46
static int MMG5_InvMat_getAttrib(MMG5_pInvMat pim, int ref)
Definition: mmg2.c:84
int MMG5_snpval_ls(MMG5_pMesh mesh, MMG5_pSol sol)
Definition: mmg2.c:503
int MMG5_isSplit(MMG5_pMesh mesh, MMG5_int ref, MMG5_int *refint, MMG5_int *refext)
Definition: mmg2.c:412
static int MMG5_InvMat_set(MMG5_pMesh mesh, MMG5_pInvMat pim, int k)
Definition: mmg2.c:129
static void MMG5_InvMat_print(MMG5_pMesh mesh, MMG5_pInvMat pim)
Definition: mmg2.c:238
static double MMG5_voltri(MMG5_pMesh mesh, MMG5_int ip0, MMG5_int ip1, MMG5_int ip2)
Definition: mmg2.c:748
int MMG5_MultiMat_init(MMG5_pMesh mesh)
Definition: mmg2.c:326
int MMG5_getStartRef(MMG5_pMesh mesh, MMG5_int ref, MMG5_int *pref)
Definition: mmg2.c:213
static double MMG5_vfrac(MMG5_pMesh mesh, MMG5_pSol sol, MMG5_int k, int pm)
Definition: mmg2.c:775
int MMG5_resetRef_ls(MMG5_pMesh mesh)
Definition: mmg2.c:1189
int MMG5_chkmaniball(MMG5_pMesh mesh, MMG5_int start, int8_t istart)
Definition: mmg2.c:1304
int MMG5_chkmanimesh(MMG5_pMesh mesh)
Definition: mmg2.c:1424
static int MMG5_InvMat_getIndex(MMG5_pInvMat pim, int ref)
Definition: mmg2.c:69
int MMG5_setref_ls(MMG5_pMesh mesh, MMG5_pSol sol)
Definition: mmg2.c:1225
#define MMG5_SAFE_CALLOC(ptr, size, type, law)
#define MG_EOK(pt)
#define MMG5_ADD_MEM(mesh, size, message, law)
#define MG_EDG(tag)
static const uint8_t MMG5_iprv2[3]
#define MMG5_TRIA_LMAX
static const uint8_t MMG5_inxt2[6]
#define MG_VOK(ppt)
#define MG_SMSGN(a, b)
#define MG_REF
#define MMG5_DEL_MEM(mesh, ptr)
MMG5_int ref
Definition: libmmgtypes.h:313
int8_t iso
Definition: libmmgtypes.h:541
int8_t ddebug
Definition: libmmgtypes.h:539
MMG5_int * br
Definition: libmmgtypes.h:527
double rmc
Definition: libmmgtypes.h:526
MMG5_pMat mat
Definition: libmmgtypes.h:562
MMG5_int isoref
Definition: libmmgtypes.h:528
MMG5_InvMat invmat
Definition: libmmgtypes.h:563
To store lookup table for references in the mesh (useful in LS mode)
Definition: libmmgtypes.h:512
MMG5_int offset
Definition: libmmgtypes.h:513
MMG5_int size
Definition: libmmgtypes.h:514
int * lookup
Definition: libmmgtypes.h:515
To store user-defined references in the mesh (useful in LS mode)
Definition: libmmgtypes.h:502
MMG5_int ref
Definition: libmmgtypes.h:504
int8_t dospl
Definition: libmmgtypes.h:503
MMG5_int rex
Definition: libmmgtypes.h:504
MMG5_int rin
Definition: libmmgtypes.h:504
MMG mesh structure.
Definition: libmmgtypes.h:613
MMG5_Info info
Definition: libmmgtypes.h:659
MMG5_int ne
Definition: libmmgtypes.h:620
MMG5_pPoint point
Definition: libmmgtypes.h:649
MMG5_int * adja
Definition: libmmgtypes.h:632
MMG5_int npmax
Definition: libmmgtypes.h:620
MMG5_int base
Definition: libmmgtypes.h:624
MMG5_pTetra tetra
Definition: libmmgtypes.h:651
MMG5_int nt
Definition: libmmgtypes.h:620
MMG5_pTria tria
Definition: libmmgtypes.h:655
MMG5_int np
Definition: libmmgtypes.h:620
MMG5_pEdge edge
Definition: libmmgtypes.h:657
MMG5_int na
Definition: libmmgtypes.h:620
Structure to store vertices of an MMG mesh.
Definition: libmmgtypes.h:276
double c[3]
Definition: libmmgtypes.h:277
MMG5_int ref
Definition: libmmgtypes.h:284
MMG5_int flag
Definition: libmmgtypes.h:288
MMG5_int ref
Definition: libmmgtypes.h:410
Structure to store triangles of a MMG mesh.
Definition: libmmgtypes.h:338
MMG5_int edg[3]
Definition: libmmgtypes.h:345
MMG5_int ref
Definition: libmmgtypes.h:341
MMG5_int flag
Definition: libmmgtypes.h:347
uint16_t tag[3]
Definition: libmmgtypes.h:348
MMG5_int v[3]
Definition: libmmgtypes.h:340