LCOV - code coverage report
Current view: top level - alg/internal_libqhull - poly_r.c (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 369 580 63.6 %
Date: 2025-10-24 00:53:13 Functions: 22 23 95.7 %

          Line data    Source code
       1             : /*<html><pre>  -<a                             href="qh-poly_r.htm"
       2             :   >-------------------------------</a><a name="TOP">-</a>
       3             : 
       4             :    poly_r.c
       5             :    implements polygons and simplices
       6             : 
       7             :    see qh-poly_r.htm, poly_r.h and libqhull_r.h
       8             : 
       9             :    infrequent code is in poly2_r.c
      10             :    (all but top 50 and their callers 12/3/95)
      11             : 
      12             :    Copyright (c) 1993-2020 The Geometry Center.
      13             :    $Id: //main/2019/qhull/src/libqhull_r/poly_r.c#8 $$Change: 2953 $
      14             :    $DateTime: 2020/05/21 22:05:32 $$Author: bbarber $
      15             : */
      16             : 
      17             : #include "qhull_ra.h"
      18             : 
      19             : /*======== functions in alphabetical order ==========*/
      20             : 
      21             : /*-<a                             href="qh-poly_r.htm#TOC"
      22             :   >-------------------------------</a><a name="appendfacet">-</a>
      23             : 
      24             :   qh_appendfacet(qh, facet )
      25             :     appends facet to end of qh.facet_list,
      26             : 
      27             :   returns:
      28             :     updates qh.newfacet_list, facet_next, facet_list
      29             :     increments qh.numfacets
      30             : 
      31             :   notes:
      32             :     assumes qh.facet_list/facet_tail is defined (createsimplex)
      33             : 
      34             :   see:
      35             :     qh_removefacet()
      36             : 
      37             : */
      38      190535 : void qh_appendfacet(qhT *qh, facetT *facet) {
      39      190535 :   facetT *tail= qh->facet_tail;
      40             : 
      41      190535 :   if (tail == qh->newfacet_list) {
      42       29084 :     qh->newfacet_list= facet;
      43       29084 :     if (tail == qh->visible_list) /* visible_list is at or before newfacet_list */
      44       14405 :       qh->visible_list= facet;
      45             :   }
      46      190535 :   if (tail == qh->facet_next)
      47           9 :     qh->facet_next= facet;
      48      190535 :   facet->previous= tail->previous;
      49      190535 :   facet->next= tail;
      50      190535 :   if (tail->previous)
      51      190526 :     tail->previous->next= facet;
      52             :   else
      53           9 :     qh->facet_list= facet;
      54      190535 :   tail->previous= facet;
      55      190535 :   qh->num_facets++;
      56      190535 :   trace4((qh, qh->ferr, 4044, "qh_appendfacet: append f%d to facet_list\n", facet->id));
      57      190535 : } /* appendfacet */
      58             : 
      59             : 
      60             : /*-<a                             href="qh-poly_r.htm#TOC"
      61             :   >-------------------------------</a><a name="appendvertex">-</a>
      62             : 
      63             :   qh_appendvertex(qh, vertex )
      64             :     appends vertex to end of qh.vertex_list,
      65             : 
      66             :   returns:
      67             :     sets vertex->newfacet
      68             :     updates qh.vertex_list, newvertex_list
      69             :     increments qh.num_vertices
      70             : 
      71             :   notes:
      72             :     assumes qh.vertex_list/vertex_tail is defined (createsimplex)
      73             : 
      74             : */
      75      151037 : void qh_appendvertex(qhT *qh, vertexT *vertex) {
      76      151037 :   vertexT *tail= qh->vertex_tail;
      77             : 
      78      151037 :   if (tail == qh->newvertex_list)
      79       14681 :     qh->newvertex_list= vertex;
      80      151037 :   vertex->newfacet= True;
      81      151037 :   vertex->previous= tail->previous;
      82      151037 :   vertex->next= tail;
      83      151037 :   if (tail->previous)
      84      151028 :     tail->previous->next= vertex;
      85             :   else
      86           9 :     qh->vertex_list= vertex;
      87      151037 :   tail->previous= vertex;
      88      151037 :   qh->num_vertices++;
      89      151037 :   trace4((qh, qh->ferr, 4045, "qh_appendvertex: append v%d to qh.newvertex_list and set v.newfacet\n", vertex->id));
      90      151037 : } /* appendvertex */
      91             : 
      92             : 
      93             : /*-<a                             href="qh-poly_r.htm#TOC"
      94             :   >-------------------------------</a><a name="attachnewfacets">-</a>
      95             : 
      96             :   qh_attachnewfacets(qh)
      97             :     attach horizon facets to new facets in qh.newfacet_list
      98             :     newfacets have neighbor and ridge links to horizon but not vice versa
      99             : 
     100             :   returns:
     101             :     clears qh.NEWtentative
     102             :     set qh.NEWfacets
     103             :     horizon facets linked to new facets
     104             :       ridges changed from visible facets to new facets
     105             :       simplicial ridges deleted
     106             :     qh.visible_list, no ridges valid
     107             :     facet->f.replace is a newfacet (if any)
     108             : 
     109             :   notes:
     110             :     used for qh.NEWtentative, otherwise see qh_makenew_nonsimplicial and qh_makenew_simplicial
     111             :     qh_delridge_merge not needed (as tested by qh_checkdelridge)
     112             : 
     113             :   design:
     114             :     delete interior ridges and neighbor sets by
     115             :       for each visible, non-simplicial facet
     116             :         for each ridge
     117             :           if last visit or if neighbor is simplicial
     118             :             if horizon neighbor
     119             :               delete ridge for horizon's ridge set
     120             :             delete ridge
     121             :         erase neighbor set
     122             :     attach horizon facets and new facets by
     123             :       for all new facets
     124             :         if corresponding horizon facet is simplicial
     125             :           locate corresponding visible facet {may be more than one}
     126             :           link visible facet to new facet
     127             :           replace visible facet with new facet in horizon
     128             :         else it is non-simplicial
     129             :           for all visible neighbors of the horizon facet
     130             :             link visible neighbor to new facet
     131             :             delete visible neighbor from horizon facet
     132             :           append new facet to horizon's neighbors
     133             :           the first ridge of the new facet is the horizon ridge
     134             :           link the new facet into the horizon ridge
     135             : */
     136           0 : void qh_attachnewfacets(qhT *qh /* qh.visible_list, qh.newfacet_list */) {
     137           0 :   facetT *newfacet= NULL, *neighbor, **neighborp, *horizon, *visible;
     138             :   ridgeT *ridge, **ridgep;
     139             : 
     140           0 :   trace3((qh, qh->ferr, 3012, "qh_attachnewfacets: delete interior ridges\n"));
     141           0 :   if (qh->CHECKfrequently) {
     142           0 :     qh_checkdelridge(qh);
     143             :   }
     144           0 :   qh->visit_id++;
     145           0 :   FORALLvisible_facets {
     146           0 :     visible->visitid= qh->visit_id;
     147           0 :     if (visible->ridges) {
     148           0 :       FOREACHridge_(visible->ridges) {
     149           0 :         neighbor= otherfacet_(ridge, visible);
     150           0 :         if (neighbor->visitid == qh->visit_id
     151           0 :             || (!neighbor->visible && neighbor->simplicial)) {
     152           0 :           if (!neighbor->visible)  /* delete ridge for simplicial horizon */
     153           0 :             qh_setdel(neighbor->ridges, ridge);
     154           0 :           qh_delridge(qh, ridge); /* delete on second visit */
     155             :         }
     156             :       }
     157             :     }
     158             :   }
     159           0 :   trace1((qh, qh->ferr, 1017, "qh_attachnewfacets: attach horizon facets to new facets\n"));
     160           0 :   FORALLnew_facets {
     161           0 :     horizon= SETfirstt_(newfacet->neighbors, facetT);
     162           0 :     if (horizon->simplicial) {
     163           0 :       visible= NULL;
     164           0 :       FOREACHneighbor_(horizon) {   /* may have more than one horizon ridge */
     165           0 :         if (neighbor->visible) {
     166           0 :           if (visible) {
     167           0 :             if (qh_setequal_skip(newfacet->vertices, 0, horizon->vertices,
     168           0 :                                   SETindex_(horizon->neighbors, neighbor))) {
     169           0 :               visible= neighbor;
     170           0 :               break;
     171             :             }
     172             :           }else
     173           0 :             visible= neighbor;
     174             :         }
     175             :       }
     176           0 :       if (visible) {
     177           0 :         visible->f.replace= newfacet;
     178           0 :         qh_setreplace(qh, horizon->neighbors, visible, newfacet);
     179             :       }else {
     180           0 :         qh_fprintf(qh, qh->ferr, 6102, "qhull internal error (qh_attachnewfacets): could not find visible facet for horizon f%d of newfacet f%d\n",
     181             :                  horizon->id, newfacet->id);
     182           0 :         qh_errexit2(qh, qh_ERRqhull, horizon, newfacet);
     183             :       }
     184             :     }else { /* non-simplicial, with a ridge for newfacet */
     185           0 :       FOREACHneighbor_(horizon) {    /* may hold for many new facets */
     186           0 :         if (neighbor->visible) {
     187           0 :           neighbor->f.replace= newfacet;
     188           0 :           qh_setdelnth(qh, horizon->neighbors, SETindex_(horizon->neighbors, neighbor));
     189           0 :           neighborp--; /* repeat */
     190             :         }
     191             :       }
     192           0 :       qh_setappend(qh, &horizon->neighbors, newfacet);
     193           0 :       ridge= SETfirstt_(newfacet->ridges, ridgeT);
     194           0 :       if (ridge->top == horizon) {
     195           0 :         ridge->bottom= newfacet;
     196           0 :         ridge->simplicialbot= True;
     197             :       }else {
     198           0 :         ridge->top= newfacet;
     199           0 :         ridge->simplicialtop= True;
     200             :       }
     201             :     }
     202             :   } /* newfacets */
     203           0 :   trace4((qh, qh->ferr, 4094, "qh_attachnewfacets: clear f.ridges and f.neighbors for visible facets, may become invalid before qh_deletevisible\n"));
     204           0 :   FORALLvisible_facets {
     205           0 :     if (visible->ridges)
     206           0 :       SETfirst_(visible->ridges)= NULL;
     207           0 :     SETfirst_(visible->neighbors)= NULL;
     208             :   }
     209           0 :   qh->NEWtentative= False;
     210           0 :   qh->NEWfacets= True;
     211           0 :   if (qh->PRINTstatistics) {
     212           0 :     FORALLvisible_facets {
     213           0 :       if (!visible->f.replace)
     214           0 :         zinc_(Zinsidevisible);
     215             :     }
     216             :   }
     217           0 : } /* attachnewfacets */
     218             : 
     219             : /*-<a                             href="qh-poly_r.htm#TOC"
     220             :   >-------------------------------</a><a name="checkflipped">-</a>
     221             : 
     222             :   qh_checkflipped(qh, facet, dist, allerror )
     223             :     checks facet orientation to interior point
     224             : 
     225             :     if allerror set,
     226             :       tests against -qh.DISTround
     227             :     else
     228             :       tests against 0.0 since tested against -qh.DISTround before
     229             : 
     230             :   returns:
     231             :     False if it flipped orientation (sets facet->flipped)
     232             :     distance if non-NULL
     233             : 
     234             :   notes:
     235             :     called by qh_setfacetplane, qh_initialhull, and qh_checkflipped_all
     236             : */
     237       53461 : boolT qh_checkflipped(qhT *qh, facetT *facet, realT *distp, boolT allerror) {
     238             :   realT dist;
     239             : 
     240       53461 :   if (facet->flipped && !distp)
     241           0 :     return False;
     242       53461 :   zzinc_(Zdistcheck);
     243       53461 :   qh_distplane(qh, qh->interior_point, facet, &dist);
     244       53461 :   if (distp)
     245           0 :     *distp= dist;
     246       53461 :   if ((allerror && dist >= -qh->DISTround) || (!allerror && dist > 0.0)) {
     247          10 :     facet->flipped= True;
     248          10 :     trace0((qh, qh->ferr, 19, "qh_checkflipped: facet f%d flipped, allerror? %d, distance= %6.12g during p%d\n",
     249             :               facet->id, allerror, dist, qh->furthest_id));
     250          10 :     if (qh->num_facets > qh->hull_dim+1) { /* qh_initialhull reverses orientation if !qh_checkflipped */
     251           0 :       zzinc_(Zflippedfacets);
     252           0 :       qh_joggle_restart(qh, "flipped facet");
     253             :     }
     254          10 :     return False;
     255             :   }
     256       53451 :   return True;
     257             : } /* checkflipped */
     258             : 
     259             : /*-<a                             href="qh-poly_r.htm#TOC"
     260             :   >-------------------------------</a><a name="delfacet">-</a>
     261             : 
     262             :   qh_delfacet(qh, facet )
     263             :     removes facet from facet_list and frees up its memory
     264             : 
     265             :   notes:
     266             :     assumes vertices and ridges already freed or referenced elsewhere
     267             : */
     268      102187 : void qh_delfacet(qhT *qh, facetT *facet) {
     269             :   void **freelistp; /* used if !qh_NOmem by qh_memfree_() */
     270             : 
     271      102187 :   trace3((qh, qh->ferr, 3057, "qh_delfacet: delete f%d\n", facet->id));
     272      102187 :   if (qh->CHECKfrequently || qh->VERIFYoutput) {
     273           0 :     if (!qh->NOerrexit) {
     274           0 :       qh_checkdelfacet(qh, facet, qh->facet_mergeset);
     275           0 :       qh_checkdelfacet(qh, facet, qh->degen_mergeset);
     276           0 :       qh_checkdelfacet(qh, facet, qh->vertex_mergeset);
     277             :     }
     278             :   }
     279      102187 :   if (facet == qh->tracefacet)
     280           0 :     qh->tracefacet= NULL;
     281      102187 :   if (facet == qh->GOODclosest)
     282           0 :     qh->GOODclosest= NULL;
     283      102187 :   qh_removefacet(qh, facet);
     284      102187 :   if (!facet->tricoplanar || facet->keepcentrum) {
     285       73377 :     qh_memfree_(qh, facet->normal, qh->normal_size, freelistp);
     286       73377 :     if (qh->CENTERtype == qh_ASvoronoi) {   /* braces for macro calls */
     287           0 :       qh_memfree_(qh, facet->center, qh->center_size, freelistp);
     288             :     }else /* AScentrum */ {
     289       73377 :       qh_memfree_(qh, facet->center, qh->normal_size, freelistp);
     290             :     }
     291             :   }
     292      102187 :   qh_setfree(qh, &(facet->neighbors));
     293      102187 :   if (facet->ridges)
     294       52133 :     qh_setfree(qh, &(facet->ridges));
     295      102187 :   qh_setfree(qh, &(facet->vertices));
     296      102187 :   if (facet->outsideset)
     297       26529 :     qh_setfree(qh, &(facet->outsideset));
     298      102187 :   if (facet->coplanarset)
     299           3 :     qh_setfree(qh, &(facet->coplanarset));
     300      102187 :   qh_memfree_(qh, facet, (int)sizeof(facetT), freelistp);
     301      102187 : } /* delfacet */
     302             : 
     303             : 
     304             : /*-<a                             href="qh-poly_r.htm#TOC"
     305             :   >-------------------------------</a><a name="deletevisible">-</a>
     306             : 
     307             :   qh_deletevisible()
     308             :     delete visible facets and vertices
     309             : 
     310             :   returns:
     311             :     deletes each facet and removes from facetlist
     312             :     deletes vertices on qh.del_vertices and ridges in qh.del_ridges
     313             :     at exit, qh.visible_list empty (== qh.newfacet_list)
     314             : 
     315             :   notes:
     316             :     called by qh_all_vertexmerges, qh_addpoint, and qh_qhull
     317             :     ridges already deleted or moved elsewhere
     318             :     deleted vertices on qh.del_vertices
     319             :     horizon facets do not reference facets on qh.visible_list
     320             :     new facets in qh.newfacet_list
     321             :     uses   qh.visit_id;
     322             : */
     323       14670 : void qh_deletevisible(qhT *qh /* qh.visible_list */) {
     324             :   facetT *visible, *nextfacet;
     325             :   vertexT *vertex, **vertexp;
     326       14670 :   int numvisible= 0, numdel= qh_setsize(qh, qh->del_vertices);
     327             : 
     328       14670 :   trace1((qh, qh->ferr, 1018, "qh_deletevisible: delete %d visible facets and %d vertices\n",
     329             :          qh->num_visible, numdel));
     330       73642 :   for (visible=qh->visible_list; visible && visible->visible;
     331       58972 :                 visible= nextfacet) { /* deleting current */
     332       58972 :     nextfacet= visible->next;
     333       58972 :     numvisible++;
     334       58972 :     qh_delfacet(qh, visible);  /* f.ridges deleted or moved elsewhere, deleted f.vertices on qh.del_vertices */
     335             :   }
     336       14670 :   if (numvisible != qh->num_visible) {
     337           0 :     qh_fprintf(qh, qh->ferr, 6103, "qhull internal error (qh_deletevisible): qh->num_visible %d is not number of visible facets %d\n",
     338             :              qh->num_visible, numvisible);
     339           0 :     qh_errexit(qh, qh_ERRqhull, NULL, NULL);
     340             :   }
     341       14670 :   qh->num_visible= 0;
     342       14670 :   zadd_(Zvisfacettot, numvisible);
     343       14670 :   zmax_(Zvisfacetmax, numvisible);
     344       14670 :   zzadd_(Zdelvertextot, numdel);
     345       14670 :   zmax_(Zdelvertexmax, numdel);
     346       14670 :   FOREACHvertex_(qh->del_vertices)
     347           0 :     qh_delvertex(qh, vertex);
     348       14670 :   qh_settruncate(qh, qh->del_vertices, 0);
     349       14670 : } /* deletevisible */
     350             : 
     351             : /*-<a                             href="qh-poly_r.htm#TOC"
     352             :   >-------------------------------</a><a name="facetintersect">-</a>
     353             : 
     354             :   qh_facetintersect(qh, facetA, facetB, skipa, skipB, prepend )
     355             :     return vertices for intersection of two simplicial facets
     356             :     may include 1 prepended entry (if more, need to settemppush)
     357             : 
     358             :   returns:
     359             :     returns set of qh.hull_dim-1 + prepend vertices
     360             :     returns skipped index for each test and checks for exactly one
     361             : 
     362             :   notes:
     363             :     does not need settemp since set in quick memory
     364             : 
     365             :   see also:
     366             :     qh_vertexintersect and qh_vertexintersect_new
     367             :     use qh_setnew_delnthsorted to get nth ridge (no skip information)
     368             : 
     369             :   design:
     370             :     locate skipped vertex by scanning facet A's neighbors
     371             :     locate skipped vertex by scanning facet B's neighbors
     372             :     intersect the vertex sets
     373             : */
     374       49268 : setT *qh_facetintersect(qhT *qh, facetT *facetA, facetT *facetB,
     375             :                          int *skipA,int *skipB, int prepend) {
     376             :   setT *intersect;
     377       49268 :   int dim= qh->hull_dim, i, j;
     378             :   facetT **neighborsA, **neighborsB;
     379             : 
     380       49268 :   neighborsA= SETaddr_(facetA->neighbors, facetT);
     381       49268 :   neighborsB= SETaddr_(facetB->neighbors, facetT);
     382       49268 :   i= j= 0;
     383       49268 :   if (facetB == *neighborsA++)
     384       20564 :     *skipA= 0;
     385             :   /* coverity[overrun-local] */
     386       28704 :   else if (facetB == *neighborsA++)
     387       15063 :     *skipA= 1;
     388             :   /* coverity[overrun-local] */
     389       13641 :   else if (facetB == *neighborsA++)
     390       13641 :     *skipA= 2;
     391             :   else {
     392           0 :     for (i=3; i < dim; i++) {
     393           0 :       if (facetB == *neighborsA++) {
     394           0 :         *skipA= i;
     395           0 :         break;
     396             :       }
     397             :     }
     398             :   }
     399       49268 :   if (facetA == *neighborsB++)
     400       12810 :     *skipB= 0;
     401             :   /* coverity[overrun-local] */
     402       36458 :   else if (facetA == *neighborsB++)
     403       19774 :     *skipB= 1;
     404             :   /* coverity[overrun-local] */
     405       16684 :   else if (facetA == *neighborsB++)
     406       16684 :     *skipB= 2;
     407             :   else {
     408           0 :     for (j=3; j < dim; j++) {
     409           0 :       if (facetA == *neighborsB++) {
     410           0 :         *skipB= j;
     411           0 :         break;
     412             :       }
     413             :     }
     414             :   }
     415       49268 :   if (i >= dim || j >= dim) {
     416           0 :     qh_fprintf(qh, qh->ferr, 6104, "qhull internal error (qh_facetintersect): f%d or f%d not in other's neighbors\n",
     417             :             facetA->id, facetB->id);
     418           0 :     qh_errexit2(qh, qh_ERRqhull, facetA, facetB);
     419             :   }
     420       49268 :   intersect= qh_setnew_delnthsorted(qh, facetA->vertices, qh->hull_dim, *skipA, prepend);
     421       49268 :   trace4((qh, qh->ferr, 4047, "qh_facetintersect: f%d skip %d matches f%d skip %d\n",
     422             :           facetA->id, *skipA, facetB->id, *skipB));
     423       49268 :   return(intersect);
     424             : } /* facetintersect */
     425             : 
     426             : /*-<a                             href="qh-poly_r.htm#TOC"
     427             :   >-------------------------------</a><a name="gethash">-</a>
     428             : 
     429             :   qh_gethash(qh, hashsize, set, size, firstindex, skipelem )
     430             :     return hashvalue for a set with firstindex and skipelem
     431             : 
     432             :   notes:
     433             :     returned hash is in [0,hashsize)
     434             :     assumes at least firstindex+1 elements
     435             :     assumes skipelem is NULL, in set, or part of hash
     436             : 
     437             :     hashes memory addresses which may change over different runs of the same data
     438             :     using sum for hash does badly in high d
     439             : */
     440      263054 : int qh_gethash(qhT *qh, int hashsize, setT *set, int size, int firstindex, void *skipelem) {
     441      263054 :   void **elemp= SETelemaddr_(set, firstindex, void);
     442      263054 :   ptr_intT hash= 0, elem;
     443             :   unsigned int uresult;
     444             :   int i;
     445             : #ifdef _MSC_VER                   /* Microsoft Visual C++ -- warn about 64-bit issues */
     446             : #pragma warning( push)            /* WARN64 -- ptr_intT holds a 64-bit pointer */
     447             : #pragma warning( disable : 4311)  /* 'type cast': pointer truncation from 'void*' to 'ptr_intT' */
     448             : #endif
     449             : 
     450      263054 :   switch (size-firstindex) {
     451           0 :   case 1:
     452           0 :     hash= (ptr_intT)((long long)(ptr_intT)(*elemp) - (ptr_intT) skipelem);
     453           0 :     break;
     454      263054 :   case 2:
     455             :     /* coverity[overrun-local] */
     456      263054 :     hash= (ptr_intT)((long long)(ptr_intT)(*elemp) + (ptr_intT)elemp[1] - (ptr_intT) skipelem);
     457      263054 :     break;
     458           0 :   case 3:
     459           0 :     hash= (ptr_intT)((long long)(ptr_intT)(*elemp) + (ptr_intT)elemp[1] + (ptr_intT)elemp[2]
     460           0 :       - (ptr_intT) skipelem);
     461           0 :     break;
     462           0 :   case 4:
     463           0 :     hash= (ptr_intT)((long long)(ptr_intT)(*elemp) + (ptr_intT)elemp[1] + (ptr_intT)elemp[2]
     464           0 :       + (ptr_intT)elemp[3] - (ptr_intT) skipelem);
     465           0 :     break;
     466           0 :   case 5:
     467           0 :     hash= (ptr_intT)((long long)(ptr_intT)(*elemp) + (ptr_intT)elemp[1] + (ptr_intT)elemp[2]
     468           0 :       + (ptr_intT)elemp[3] + (ptr_intT)elemp[4] - (ptr_intT) skipelem);
     469           0 :     break;
     470           0 :   case 6:
     471           0 :     hash= (ptr_intT)((long long)(ptr_intT)(*elemp) + (ptr_intT)elemp[1] + (ptr_intT)elemp[2]
     472           0 :       + (ptr_intT)elemp[3] + (ptr_intT)elemp[4]+ (ptr_intT)elemp[5]
     473           0 :       - (ptr_intT) skipelem);
     474           0 :     break;
     475           0 :   default:
     476           0 :     hash= 0;
     477           0 :     i= 3;
     478             :     do {     /* this is about 10% in 10-d */
     479           0 :       if ((elem= (ptr_intT)*elemp++) != (ptr_intT)skipelem) {
     480           0 :         hash ^= (elem << i) + (elem >> (32-i));
     481           0 :         i += 3;
     482           0 :         if (i >= 32)
     483           0 :           i -= 32;
     484             :       }
     485           0 :     }while (*elemp);
     486           0 :     break;
     487             :   }
     488      263054 :   if (hashsize<0) {
     489           0 :     qh_fprintf(qh, qh->ferr, 6202, "qhull internal error: negative hashsize %d passed to qh_gethash [poly_r.c]\n", hashsize);
     490           0 :     qh_errexit2(qh, qh_ERRqhull, NULL, NULL);
     491             :   }
     492      263054 :   uresult= (unsigned int)hash;
     493      263054 :   uresult %= (unsigned int)hashsize;
     494             :   /* result= 0; for debugging */
     495      263054 :   return (int)uresult;
     496             : #ifdef _MSC_VER
     497             : #pragma warning( pop)
     498             : #endif
     499             : } /* gethash */
     500             : 
     501             : /*-<a                             href="qh-poly_r.htm#TOC"
     502             :   >-------------------------------</a><a name="getreplacement">-</a>
     503             : 
     504             :   qh_getreplacement(qh, visible )
     505             :     get replacement for visible facet
     506             : 
     507             :   returns:
     508             :     valid facet from visible.replace (may be chained)
     509             : */
     510       26532 : facetT *qh_getreplacement(qhT *qh, facetT *visible) {
     511       26532 :   unsigned int count= 0;
     512             : 
     513       26532 :   facetT *result= visible;
     514       58627 :   while (result && result->visible) {
     515       32095 :     result= result->f.replace;
     516       32095 :     if (count++ > qh->facet_id)
     517           0 :       qh_infiniteloop(qh, visible);
     518             :   }
     519       26532 :   return result;
     520             : }
     521             : 
     522             : /*-<a                             href="qh-poly_r.htm#TOC"
     523             :   >-------------------------------</a><a name="makenewfacet">-</a>
     524             : 
     525             :   qh_makenewfacet(qh, vertices, toporient, horizon )
     526             :     creates a toporient? facet from vertices
     527             : 
     528             :   returns:
     529             :     returns newfacet
     530             :       adds newfacet to qh.facet_list
     531             :       newfacet->vertices= vertices
     532             :       if horizon
     533             :         newfacet->neighbor= horizon, but not vice versa
     534             :     newvertex_list updated with vertices
     535             : */
     536      131527 : facetT *qh_makenewfacet(qhT *qh, setT *vertices, boolT toporient, facetT *horizon) {
     537             :   facetT *newfacet;
     538             :   vertexT *vertex, **vertexp;
     539             : 
     540      526108 :   FOREACHvertex_(vertices) {
     541      394581 :     if (!vertex->newfacet) {
     542       88084 :       qh_removevertex(qh, vertex);
     543       88084 :       qh_appendvertex(qh, vertex);
     544             :     }
     545             :   }
     546      131527 :   newfacet= qh_newfacet(qh);
     547      131527 :   newfacet->vertices= vertices;
     548      131527 :   if (toporient)
     549       65706 :     newfacet->toporient= True;
     550      131527 :   if (horizon)
     551      131527 :     qh_setappend(qh, &(newfacet->neighbors), horizon);
     552      131527 :   qh_appendfacet(qh, newfacet);
     553      131527 :   return(newfacet);
     554             : } /* makenewfacet */
     555             : 
     556             : 
     557             : /*-<a                             href="qh-poly_r.htm#TOC"
     558             :   >-------------------------------</a><a name="makenewplanes">-</a>
     559             : 
     560             :   qh_makenewplanes()
     561             :     make new hyperplanes for facets on qh.newfacet_list
     562             : 
     563             :   returns:
     564             :     all facets have hyperplanes or are marked for   merging
     565             :     doesn't create hyperplane if horizon is coplanar (will merge)
     566             :     updates qh.min_vertex if qh.JOGGLEmax
     567             : 
     568             :   notes:
     569             :     facet->f.samecycle is defined for facet->mergehorizon facets
     570             : */
     571       14670 : void qh_makenewplanes(qhT *qh /* qh.newfacet_list */) {
     572             :   facetT *newfacet;
     573             : 
     574       14670 :   trace4((qh, qh->ferr, 4074, "qh_makenewplanes: make new hyperplanes for facets on qh.newfacet_list f%d\n",
     575             :     qh->newfacet_list->id));
     576       88109 :   FORALLnew_facets {
     577       73439 :     if (!newfacet->mergehorizon)
     578       53387 :       qh_setfacetplane(qh, newfacet); /* updates Wnewvertexmax */
     579             :   }
     580       14670 :   if (qh->JOGGLEmax < REALmax/2)
     581           0 :     minimize_(qh->min_vertex, -wwval_(Wnewvertexmax));
     582       14670 : } /* makenewplanes */
     583             : 
     584             : #ifndef qh_NOmerge
     585             : /*-<a                             href="qh-poly_r.htm#TOC"
     586             :   >-------------------------------</a><a name="makenew_nonsimplicial">-</a>
     587             : 
     588             :   qh_makenew_nonsimplicial(qh, visible, apex, numnew )
     589             :     make new facets for ridges of a visible facet
     590             : 
     591             :   returns:
     592             :     first newfacet, bumps numnew as needed
     593             :     attaches new facets if !qh->NEWtentative
     594             :     marks ridge neighbors for simplicial visible
     595             :     if (qh.NEWtentative)
     596             :       ridges on newfacet, horizon, and visible
     597             :     else
     598             :       ridge and neighbors between newfacet and horizon
     599             :       visible facet's ridges are deleted
     600             :       visible facet's f.neighbors is empty
     601             : 
     602             :   notes:
     603             :     called by qh_makenewfacets and qh_triangulatefacet
     604             :     qh.visit_id if visible has already been processed
     605             :     sets neighbor->seen for building f.samecycle
     606             :       assumes all 'seen' flags initially false
     607             :     qh_delridge_merge not needed (as tested by qh_checkdelridge in qh_makenewfacets)
     608             : 
     609             :   design:
     610             :     for each ridge of visible facet
     611             :       get neighbor of visible facet
     612             :       if neighbor was already processed
     613             :         delete the ridge (will delete all visible facets later)
     614             :       if neighbor is a horizon facet
     615             :         create a new facet
     616             :         if neighbor coplanar
     617             :           adds newfacet to f.samecycle for later merging
     618             :         else
     619             :           updates neighbor's neighbor set
     620             :           (checks for non-simplicial facet with multiple ridges to visible facet)
     621             :         updates neighbor's ridge set
     622             :         (checks for simplicial neighbor to non-simplicial visible facet)
     623             :         (deletes ridge if neighbor is simplicial)
     624             : 
     625             : */
     626       32080 : facetT *qh_makenew_nonsimplicial(qhT *qh, facetT *visible, vertexT *apex, int *numnew) {
     627             :   void **freelistp; /* used if !qh_NOmem by qh_memfree_() */
     628             :   ridgeT *ridge, **ridgep;
     629       32080 :   facetT *neighbor, *newfacet= NULL, *samecycle;
     630             :   setT *vertices;
     631             :   boolT toporient;
     632             :   unsigned int ridgeid;
     633             : 
     634      121773 :   FOREACHridge_(visible->ridges) {
     635       89693 :     ridgeid= ridge->id;
     636       89693 :     neighbor= otherfacet_(ridge, visible);
     637       89693 :     if (neighbor->visible) {
     638        7434 :       if (!qh->NEWtentative) {
     639        7434 :         if (neighbor->visitid == qh->visit_id) {
     640        3717 :           if (qh->traceridge == ridge)
     641           0 :             qh->traceridge= NULL;
     642        3717 :           qh_setfree(qh, &(ridge->vertices));  /* delete on 2nd visit */
     643        3717 :           qh_memfree_(qh, ridge, (int)sizeof(ridgeT), freelistp);
     644             :         }
     645             :       }
     646             :     }else {  /* neighbor is an horizon facet */
     647       82259 :       toporient= (ridge->top == visible);
     648       82259 :       vertices= qh_setnew(qh, qh->hull_dim); /* makes sure this is quick */
     649       82259 :       qh_setappend(qh, &vertices, apex);
     650       82259 :       qh_setappend_set(qh, &vertices, ridge->vertices);
     651       82259 :       newfacet= qh_makenewfacet(qh, vertices, toporient, neighbor);
     652       82259 :       (*numnew)++;
     653       82259 :       if (neighbor->coplanarhorizon) {
     654        6675 :         newfacet->mergehorizon= True;
     655        6675 :         if (!neighbor->seen) {
     656        6675 :           newfacet->f.samecycle= newfacet;
     657        6675 :           neighbor->f.newcycle= newfacet;
     658             :         }else {
     659           0 :           samecycle= neighbor->f.newcycle;
     660           0 :           newfacet->f.samecycle= samecycle->f.samecycle;
     661           0 :           samecycle->f.samecycle= newfacet;
     662             :         }
     663             :       }
     664       82259 :       if (qh->NEWtentative) {
     665           0 :         if (!neighbor->simplicial)
     666           0 :           qh_setappend(qh, &(newfacet->ridges), ridge);
     667             :       }else {  /* qh_attachnewfacets */
     668       82259 :         if (neighbor->seen) {
     669           0 :           if (neighbor->simplicial) {
     670           0 :             qh_fprintf(qh, qh->ferr, 6105, "qhull internal error (qh_makenew_nonsimplicial): simplicial f%d sharing two ridges with f%d\n",
     671             :                    neighbor->id, visible->id);
     672           0 :             qh_errexit2(qh, qh_ERRqhull, neighbor, visible);
     673             :           }
     674           0 :           qh_setappend(qh, &(neighbor->neighbors), newfacet);
     675             :         }else
     676       82259 :           qh_setreplace(qh, neighbor->neighbors, visible, newfacet);
     677       82259 :         if (neighbor->simplicial) {
     678       40785 :           qh_setdel(neighbor->ridges, ridge);
     679       40785 :           qh_delridge(qh, ridge);
     680             :         }else {
     681       41474 :           qh_setappend(qh, &(newfacet->ridges), ridge);
     682       41474 :           if (toporient) {
     683       20815 :             ridge->top= newfacet;
     684       20815 :             ridge->simplicialtop= True;
     685             :           }else {
     686       20659 :             ridge->bottom= newfacet;
     687       20659 :             ridge->simplicialbot= True;
     688             :           }
     689             :         }
     690             :       }
     691       82259 :       trace4((qh, qh->ferr, 4048, "qh_makenew_nonsimplicial: created facet f%d from v%d and r%d of horizon f%d\n",
     692             :           newfacet->id, apex->id, ridgeid, neighbor->id));
     693             :     }
     694       89693 :     neighbor->seen= True;
     695             :   } /* for each ridge */
     696       32080 :   return newfacet;
     697             : } /* makenew_nonsimplicial */
     698             : 
     699             : #else /* qh_NOmerge */
     700             : facetT *qh_makenew_nonsimplicial(qhT *qh, facetT *visible, vertexT *apex, int *numnew) {
     701             :   QHULL_UNUSED(qh)
     702             :   QHULL_UNUSED(visible)
     703             :   QHULL_UNUSED(apex)
     704             :   QHULL_UNUSED(numnew)
     705             : 
     706             :   return NULL;
     707             : }
     708             : #endif /* qh_NOmerge */
     709             : 
     710             : /*-<a                             href="qh-poly_r.htm#TOC"
     711             :   >-------------------------------</a><a name="makenew_simplicial">-</a>
     712             : 
     713             :   qh_makenew_simplicial(qh, visible, apex, numnew )
     714             :     make new facets for simplicial visible facet and apex
     715             : 
     716             :   returns:
     717             :     attaches new facets if !qh.NEWtentative
     718             :       neighbors between newfacet and horizon
     719             : 
     720             :   notes:
     721             :     nop if neighbor->seen or neighbor->visible(see qh_makenew_nonsimplicial)
     722             : 
     723             :   design:
     724             :     locate neighboring horizon facet for visible facet
     725             :     determine vertices and orientation
     726             :     create new facet
     727             :     if coplanar,
     728             :       add new facet to f.samecycle
     729             :     update horizon facet's neighbor list
     730             : */
     731       34075 : facetT *qh_makenew_simplicial(qhT *qh, facetT *visible, vertexT *apex, int *numnew) {
     732       34075 :   facetT *neighbor, **neighborp, *newfacet= NULL;
     733             :   setT *vertices;
     734             :   boolT flip, toporient;
     735       34075 :   int horizonskip= 0, visibleskip= 0;
     736             : 
     737      136300 :   FOREACHneighbor_(visible) {
     738      102225 :     if (!neighbor->seen && !neighbor->visible) {
     739       49268 :       vertices= qh_facetintersect(qh, neighbor,visible, &horizonskip, &visibleskip, 1);
     740       49268 :       SETfirst_(vertices)= apex;
     741       49268 :       flip= ((horizonskip & 0x1) ^ (visibleskip & 0x1));
     742       49268 :       if (neighbor->toporient)
     743       24773 :         toporient= horizonskip & 0x1;
     744             :       else
     745       24495 :         toporient= (horizonskip & 0x1) ^ 0x1;
     746       49268 :       newfacet= qh_makenewfacet(qh, vertices, toporient, neighbor);
     747       49268 :       (*numnew)++;
     748       49268 :       if (neighbor->coplanarhorizon && (qh->PREmerge || qh->MERGEexact)) {
     749             : #ifndef qh_NOmerge
     750       13377 :         newfacet->f.samecycle= newfacet;
     751       13377 :         newfacet->mergehorizon= True;
     752             : #endif
     753             :       }
     754       49268 :       if (!qh->NEWtentative)
     755       49268 :         SETelem_(neighbor->neighbors, horizonskip)= newfacet;
     756       49268 :       trace4((qh, qh->ferr, 4049, "qh_makenew_simplicial: create facet f%d top %d from v%d and horizon f%d skip %d top %d and visible f%d skip %d, flip? %d\n",
     757             :             newfacet->id, toporient, apex->id, neighbor->id, horizonskip,
     758             :               neighbor->toporient, visible->id, visibleskip, flip));
     759             :     }
     760             :   }
     761       34075 :   return newfacet;
     762             : } /* makenew_simplicial */
     763             : 
     764             : /*-<a                             href="qh-poly_r.htm#TOC"
     765             :   >-------------------------------</a><a name="matchneighbor">-</a>
     766             : 
     767             :   qh_matchneighbor(qh, newfacet, newskip, hashsize, hashcount )
     768             :     either match subridge of newfacet with neighbor or add to hash_table
     769             : 
     770             :   returns:
     771             :     matched ridges of newfacet, except for duplicate ridges
     772             :     duplicate ridges marked by qh_DUPLICATEridge for qh_matchdupridge
     773             : 
     774             :   notes:
     775             :     called by qh_matchnewfacets
     776             :     assumes newfacet is simplicial
     777             :     ridge is newfacet->vertices w/o newskip vertex
     778             :     do not allocate memory (need to free hash_table cleanly)
     779             :     uses linear hash chains
     780             :     see qh_matchdupridge (poly2_r.c)
     781             : 
     782             :   design:
     783             :     for each possible matching facet in qh.hash_table
     784             :       if vertices match
     785             :         set ismatch, if facets have opposite orientation
     786             :         if ismatch and matching facet doesn't have a match
     787             :           match the facets by updating their neighbor sets
     788             :         else
     789             :           note: dupridge detected when a match 'f&d skip %d' has already been seen
     790             :                 need to mark all of the dupridges for qh_matchdupridge
     791             :           indicate a duplicate ridge by qh_DUPLICATEridge and f.dupridge
     792             :           add facet to hashtable
     793             :           unless the other facet was already a duplicate ridge
     794             :             mark both facets with a duplicate ridge
     795             :             add other facet (if defined) to hash table
     796             : 
     797             :   state at "indicate a duplicate ridge":
     798             :     newfacet@newskip= the argument
     799             :     facet= the hashed facet@skip that has the same vertices as newfacet@newskip
     800             :     same= true if matched vertices have the same orientation
     801             :     matchfacet= neighbor at facet@skip
     802             :     matchfacet=qh_DUPLICATEridge, matchfacet was previously detected as a dupridge of facet@skip
     803             :     ismatch if 'vertex orientation (same) matches facet/newfacet orientation (toporient)
     804             :     unknown facet will match later
     805             : 
     806             :   details at "indicate a duplicate ridge":
     807             :     if !ismatch and matchfacet,
     808             :       dupridge is between hashed facet@skip/matchfacet@matchskip and arg newfacet@newskip/unknown
     809             :       set newfacet@newskip, facet@skip, and matchfacet@matchskip to qh_DUPLICATEridge
     810             :       add newfacet and matchfacet to hash_table
     811             :       if ismatch and matchfacet,
     812             :         same as !ismatch and matchfacet -- it matches facet instead of matchfacet
     813             :       if !ismatch and !matchfacet
     814             :         dupridge between hashed facet@skip/unknown and arg newfacet@newskip/unknown
     815             :         set newfacet@newskip and facet@skip to qh_DUPLICATEridge
     816             :         add newfacet to hash_table
     817             :       if ismatch and matchfacet==qh_DUPLICATEridge
     818             :         dupridge with already duplicated hashed facet@skip and arg newfacet@newskip/unknown
     819             :         set newfacet@newskip to qh_DUPLICATEridge
     820             :         add newfacet to hash_table
     821             :         facet's hyperplane already set
     822             : */
     823      263054 : void qh_matchneighbor(qhT *qh, facetT *newfacet, int newskip, int hashsize, int *hashcount) {
     824      263054 :   boolT newfound= False;   /* True, if new facet is already in hash chain */
     825             :   boolT same, ismatch;
     826             :   int hash, scan;
     827             :   facetT *facet, *matchfacet;
     828             :   int skip, matchskip;
     829             : 
     830      263054 :   hash= qh_gethash(qh, hashsize, newfacet->vertices, qh->hull_dim, 1,
     831      263054 :                      SETelem_(newfacet->vertices, newskip));
     832      263054 :   trace4((qh, qh->ferr, 4050, "qh_matchneighbor: newfacet f%d skip %d hash %d hashcount %d\n",
     833             :           newfacet->id, newskip, hash, *hashcount));
     834      263054 :   zinc_(Zhashlookup);
     835      284889 :   for (scan=hash; (facet= SETelemt_(qh->hash_table, scan, facetT));
     836       21835 :        scan= (++scan >= hashsize ? 0 : scan)) {
     837      153362 :     if (facet == newfacet) {
     838        1595 :       newfound= True;
     839        1595 :       continue;
     840             :     }
     841      151767 :     zinc_(Zhashtests);
     842      151767 :     if (qh_matchvertices(qh, 1, newfacet->vertices, newskip, facet->vertices, &skip, &same)) {
     843      131527 :       if (SETelem_(newfacet->vertices, newskip) == SETelem_(facet->vertices, skip)) {
     844           0 :         qh_joggle_restart(qh, "two new facets with the same vertices");
     845             :         /* duplicated for multiple skips, not easily avoided */
     846           0 :         qh_fprintf(qh, qh->ferr, 7084, "qhull topology warning (qh_matchneighbor): will merge vertices to undo new facets -- f%d and f%d have the same vertices (skip %d, skip %d) and same horizon ridges to f%d and f%d\n",
     847           0 :           facet->id, newfacet->id, skip, newskip, SETfirstt_(facet->neighbors, facetT)->id, SETfirstt_(newfacet->neighbors, facetT)->id);
     848             :         /* will rename a vertex (QH3053).  The fault was duplicate ridges (same vertices) in different facets due to a previous rename.  Expensive to detect beforehand */
     849             :       }
     850      131527 :       ismatch= (same == (boolT)((newfacet->toporient ^ facet->toporient)));
     851      131527 :       matchfacet= SETelemt_(facet->neighbors, skip, facetT);
     852      131527 :       if (ismatch && !matchfacet) {
     853      131527 :         SETelem_(facet->neighbors, skip)= newfacet;
     854      131527 :         SETelem_(newfacet->neighbors, newskip)= facet;
     855      131527 :         (*hashcount)--;
     856      131527 :         trace4((qh, qh->ferr, 4051, "qh_matchneighbor: f%d skip %d matched with new f%d skip %d\n",
     857             :            facet->id, skip, newfacet->id, newskip));
     858      131527 :         return;
     859             :       }
     860           0 :       if (!qh->PREmerge && !qh->MERGEexact) {
     861           0 :         qh_joggle_restart(qh, "a ridge with more than two neighbors");
     862           0 :         qh_fprintf(qh, qh->ferr, 6107, "qhull topology error: facets f%d, f%d and f%d meet at a ridge with more than 2 neighbors.  Can not continue due to no qh.PREmerge and no 'Qx' (MERGEexact)\n",
     863           0 :                  facet->id, newfacet->id, getid_(matchfacet));
     864           0 :         qh_errexit2(qh, qh_ERRtopology, facet, newfacet);
     865             :       }
     866           0 :       SETelem_(newfacet->neighbors, newskip)= qh_DUPLICATEridge;
     867           0 :       newfacet->dupridge= True;
     868           0 :       qh_addhash(newfacet, qh->hash_table, hashsize, hash);
     869           0 :       (*hashcount)++;
     870           0 :       if (matchfacet != qh_DUPLICATEridge) {
     871           0 :         SETelem_(facet->neighbors, skip)= qh_DUPLICATEridge;
     872           0 :         facet->dupridge= True;
     873           0 :         if (matchfacet) {
     874           0 :           matchskip= qh_setindex(matchfacet->neighbors, facet);
     875           0 :           if (matchskip<0) {
     876           0 :               qh_fprintf(qh, qh->ferr, 6260, "qhull topology error (qh_matchneighbor): matchfacet f%d is in f%d neighbors but not vice versa.  Can not continue.\n",
     877             :                   matchfacet->id, facet->id);
     878           0 :               qh_errexit2(qh, qh_ERRtopology, matchfacet, facet);
     879             :           }
     880             :           else {
     881           0 :           SETelem_(matchfacet->neighbors, matchskip)= qh_DUPLICATEridge; /* matchskip>=0 by QH6260 */
     882           0 :           matchfacet->dupridge= True;
     883           0 :           qh_addhash(matchfacet, qh->hash_table, hashsize, hash);
     884           0 :           *hashcount += 2;
     885             :           }
     886             :         }
     887             :       }
     888           0 :       trace4((qh, qh->ferr, 4052, "qh_matchneighbor: new f%d skip %d duplicates ridge for f%d skip %d matching f%d ismatch %d at hash %d\n",
     889             :            newfacet->id, newskip, facet->id, skip,
     890             :            (matchfacet == qh_DUPLICATEridge ? -2 : getid_(matchfacet)),
     891             :            ismatch, hash));
     892           0 :       return; /* end of duplicate ridge */
     893             :     }
     894             :   }
     895      131527 :   if (!newfound)
     896      129932 :     SETelem_(qh->hash_table, scan)= newfacet;  /* same as qh_addhash */
     897      131527 :   (*hashcount)++;
     898      131527 :   trace4((qh, qh->ferr, 4053, "qh_matchneighbor: no match for f%d skip %d at hash %d\n",
     899             :            newfacet->id, newskip, hash));
     900             : } /* matchneighbor */
     901             : 
     902             : 
     903             : /*-<a                             href="qh-poly_r.htm#TOC"
     904             :   >-------------------------------</a><a name="matchnewfacets">-</a>
     905             : 
     906             :   qh_matchnewfacets(qh )
     907             :     match new facets in qh.newfacet_list to their newfacet neighbors
     908             :     all facets are simplicial
     909             : 
     910             :   returns:
     911             :     if dupridges and merging
     912             :       returns maxdupdist (>=0.0) from vertex to opposite facet
     913             :       sets facet->dupridge
     914             :       missing neighbor links identify dupridges to be merged (qh_DUPLICATEridge)
     915             :     else
     916             :       qh.newfacet_list with full neighbor sets
     917             :         vertices for the nth neighbor match all but the nth vertex
     918             :     if not merging and qh.FORCEoutput
     919             :       for facets with normals (i.e., with dupridges)
     920             :       sets facet->flippped for flipped normals, also prevents point partitioning
     921             : 
     922             :   notes:
     923             :     called by qh_buildcone* and qh_triangulate_facet
     924             :     neighbor[0] of new facets is the horizon facet
     925             :     if NEWtentative, new facets not attached to the horizon
     926             :     assumes qh.hash_table is NULL
     927             :     vertex->neighbors has not been updated yet
     928             :     do not allocate memory after qh.hash_table (need to free it cleanly)
     929             : 
     930             :   design:
     931             :     truncate neighbor sets to horizon facet for all new facets
     932             :     initialize a hash table
     933             :     for all new facets
     934             :       match facet with neighbors
     935             :     if unmatched facets (due to duplicate ridges)
     936             :       for each new facet with a duplicate ridge
     937             :         try to match facets with the same coplanar horizon
     938             :     if not all matched
     939             :       for each new facet with a duplicate ridge
     940             :         match it with a coplanar facet, or identify a pinched vertex
     941             :     if not merging and qh.FORCEoutput
     942             :       check for flipped facets
     943             : */
     944       29075 : coordT qh_matchnewfacets(qhT *qh /* qh.newfacet_list */) {
     945       29075 :   int numnew=0, hashcount=0, newskip;
     946             :   facetT *newfacet, *neighbor;
     947       29075 :   coordT maxdupdist= 0.0, maxdist2;
     948       29075 :   int dim= qh->hull_dim, hashsize, neighbor_i, neighbor_n;
     949             :   setT *neighbors;
     950             : #ifndef qh_NOtrace
     951       29075 :   int facet_i, facet_n, numunused= 0;
     952             :   facetT *facet;
     953             : #endif
     954             : 
     955       29075 :   trace1((qh, qh->ferr, 1019, "qh_matchnewfacets: match neighbors for new facets.\n"));
     956      160602 :   FORALLnew_facets {
     957      131527 :     numnew++;
     958             :     {  /* inline qh_setzero(qh, newfacet->neighbors, 1, qh->hull_dim); */
     959      131527 :       neighbors= newfacet->neighbors;
     960      131527 :       neighbors->e[neighbors->maxsize].i= dim+1; /*may be overwritten*/
     961      131527 :       memset((char *)SETelemaddr_(neighbors, 1, void), 0, (size_t)(dim * SETelemsize));
     962             :     }
     963             :   }
     964             : 
     965       29075 :   qh_newhashtable(qh, numnew*(qh->hull_dim-1)); /* twice what is normally needed,
     966             :                                      but every ridge could be DUPLICATEridge */
     967       29075 :   hashsize= qh_setsize(qh, qh->hash_table);
     968       29075 :   if( hashsize == 0 ) return 0.0; /* make coverity happy */
     969      160602 :   FORALLnew_facets {
     970      131527 :     if (!newfacet->simplicial) {
     971           0 :       qh_fprintf(qh, qh->ferr, 6377, "qhull internal error (qh_matchnewfacets): expecting simplicial facets on qh.newfacet_list f%d for qh_matchneighbors, qh_matchneighbor, and qh_matchdupridge.  Got non-simplicial f%d\n",
     972           0 :         qh->newfacet_list->id, newfacet->id);
     973           0 :       qh_errexit2(qh, qh_ERRqhull, newfacet, qh->newfacet_list);
     974             :     }
     975      394581 :     for (newskip=1; newskip<qh->hull_dim; newskip++) /* furthest/horizon already matched */
     976             :       /* hashsize>0 because hull_dim>1 and numnew>0 */
     977      263054 :       qh_matchneighbor(qh, newfacet, newskip, hashsize, &hashcount);
     978             : #if 0   /* use the following to trap hashcount errors */
     979             :     {
     980             :       int count= 0, k;
     981             :       facetT *facet, *neighbor;
     982             : 
     983             :       count= 0;
     984             :       FORALLfacet_(qh->newfacet_list) {  /* newfacet already in use */
     985             :         for (k=1; k < qh->hull_dim; k++) {
     986             :           neighbor= SETelemt_(facet->neighbors, k, facetT);
     987             :           if (!neighbor || neighbor == qh_DUPLICATEridge)
     988             :             count++;
     989             :         }
     990             :         if (facet == newfacet)
     991             :           break;
     992             :       }
     993             :       if (count != hashcount) {
     994             :         qh_fprintf(qh, qh->ferr, 6266, "qhull error (qh_matchnewfacets): after adding facet %d, hashcount %d != count %d\n",
     995             :                  newfacet->id, hashcount, count);
     996             :         qh_errexit(qh, qh_ERRdebug, newfacet, NULL);
     997             :       }
     998             :     }
     999             : #endif  /* end of trap code */
    1000             :   } /* end FORALLnew_facets */
    1001       29075 :   if (hashcount) { /* all neighbors matched, except for qh_DUPLICATEridge neighbors */
    1002           0 :     qh_joggle_restart(qh, "ridge with multiple neighbors");
    1003           0 :     if (hashcount) {
    1004           0 :       FORALLnew_facets {
    1005           0 :         if (newfacet->dupridge) {
    1006           0 :           FOREACHneighbor_i_(qh, newfacet) {
    1007           0 :             if (neighbor == qh_DUPLICATEridge) {
    1008           0 :               maxdist2= qh_matchdupridge(qh, newfacet, neighbor_i, hashsize, &hashcount);
    1009           0 :               maximize_(maxdupdist, maxdist2);
    1010             :             }
    1011             :           }
    1012             :         }
    1013             :       }
    1014             :     }
    1015             :   }
    1016       29075 :   if (hashcount) {
    1017           0 :     qh_fprintf(qh, qh->ferr, 6108, "qhull internal error (qh_matchnewfacets): %d neighbors did not match up\n",
    1018             :         hashcount);
    1019           0 :     qh_printhashtable(qh, qh->ferr);
    1020           0 :     qh_errexit(qh, qh_ERRqhull, NULL, NULL);
    1021             :   }
    1022             : #ifndef qh_NOtrace
    1023       29075 :   if (qh->IStracing >= 3) {
    1024           0 :     FOREACHfacet_i_(qh, qh->hash_table) {
    1025           0 :       if (!facet)
    1026           0 :         numunused++;
    1027             :     }
    1028           0 :     qh_fprintf(qh, qh->ferr, 3063, "qh_matchnewfacets: maxdupdist %2.2g, new facets %d, unused hash entries %d, hashsize %d\n",
    1029             :              maxdupdist, numnew, numunused, qh_setsize(qh, qh->hash_table));
    1030             :   }
    1031             : #endif /* !qh_NOtrace */
    1032       29075 :   qh_setfree(qh, &qh->hash_table);
    1033       29075 :   if (qh->PREmerge || qh->MERGEexact) {
    1034       29075 :     if (qh->IStracing >= 4)
    1035           0 :       qh_printfacetlist(qh, qh->newfacet_list, NULL, qh_ALL);
    1036             :   }
    1037       29075 :   return maxdupdist;
    1038             : } /* matchnewfacets */
    1039             : 
    1040             : 
    1041             : /*-<a                             href="qh-poly_r.htm#TOC"
    1042             :   >-------------------------------</a><a name="matchvertices">-</a>
    1043             : 
    1044             :   qh_matchvertices(qh, firstindex, verticesA, skipA, verticesB, skipB, same )
    1045             :     tests whether vertices match with a single skip
    1046             :     starts match at firstindex since all new facets have a common vertex
    1047             : 
    1048             :   returns:
    1049             :     true if matched vertices
    1050             :     skip index for skipB
    1051             :     sets same iff vertices have the same orientation
    1052             : 
    1053             :   notes:
    1054             :     called by qh_matchneighbor and qh_matchdupridge
    1055             :     assumes skipA is in A and both sets are the same size
    1056             : 
    1057             :   design:
    1058             :     set up pointers
    1059             :     scan both sets checking for a match
    1060             :     test orientation
    1061             : */
    1062      151767 : boolT qh_matchvertices(qhT *qh, int firstindex, setT *verticesA, int skipA,
    1063             :        setT *verticesB, int *skipB, boolT *same) {
    1064      151767 :   vertexT **elemAp, **elemBp, **skipBp=NULL, **skipAp;
    1065             : 
    1066      151767 :   elemAp= SETelemaddr_(verticesA, firstindex, vertexT);
    1067      151767 :   elemBp= SETelemaddr_(verticesB, firstindex, vertexT);
    1068      151767 :   skipAp= SETelemaddr_(verticesA, skipA, vertexT);
    1069      291425 :   do if (elemAp != skipAp) {
    1070      227176 :     while (*elemAp != *elemBp++) {
    1071       95649 :       if (skipBp)
    1072       20240 :         return False;
    1073       75409 :       skipBp= elemBp;  /* one extra like FOREACH */
    1074             :     }
    1075             :     /* coverity[overrun-local] */
    1076      271185 :   }while (*(++elemAp));
    1077      131527 :   if (!skipBp)
    1078       76358 :     skipBp= ++elemBp;
    1079      131527 :   *skipB= SETindex_(verticesB, skipB); /* i.e., skipBp - verticesB
    1080             :                                        verticesA and verticesB are the same size, otherwise trace4 may segfault */
    1081      131527 :   *same= !((skipA & 0x1) ^ (*skipB & 0x1)); /* result is 0 or 1 */
    1082             :   /* coverity[overrun-local] */
    1083      131527 :   trace4((qh, qh->ferr, 4054, "qh_matchvertices: matched by skip %d(v%d) and skip %d(v%d) same? %d\n",
    1084             :           skipA, (*skipAp)->id, *skipB, (*(skipBp-1))->id, *same));
    1085      131527 :   return(True);
    1086             : } /* matchvertices */
    1087             : 
    1088             : /*-<a                             href="qh-poly_r.htm#TOC"
    1089             :   >-------------------------------</a><a name="newfacet">-</a>
    1090             : 
    1091             :   qh_newfacet(qh)
    1092             :     return a new facet
    1093             : 
    1094             :   returns:
    1095             :     all fields initialized or cleared   (NULL)
    1096             :     preallocates neighbors set
    1097             : */
    1098      131572 : facetT *qh_newfacet(qhT *qh) {
    1099             :   facetT *facet;
    1100             :   void **freelistp; /* used if !qh_NOmem by qh_memalloc_() */
    1101             : 
    1102      131572 :   qh_memalloc_(qh, (int)sizeof(facetT), freelistp, facet, facetT);
    1103      131572 :   memset((char *)facet, (size_t)0, sizeof(facetT));
    1104      131572 :   if (qh->facet_id == qh->tracefacet_id)
    1105           0 :     qh->tracefacet= facet;
    1106      131572 :   facet->id= qh->facet_id++;
    1107      131572 :   facet->neighbors= qh_setnew(qh, qh->hull_dim);
    1108             : #if !qh_COMPUTEfurthest
    1109      131572 :   facet->furthestdist= 0.0;
    1110             : #endif
    1111             : #if qh_MAXoutside
    1112      131572 :   if (qh->FORCEoutput && qh->APPROXhull)
    1113           0 :     facet->maxoutside= qh->MINoutside;
    1114             :   else
    1115      131572 :     facet->maxoutside= qh->DISTround; /* same value as test for QH7082 */
    1116             : #endif
    1117      131572 :   facet->simplicial= True;
    1118      131572 :   facet->good= True;
    1119      131572 :   facet->newfacet= True;
    1120      131572 :   trace4((qh, qh->ferr, 4055, "qh_newfacet: created facet f%d\n", facet->id));
    1121      131572 :   return(facet);
    1122             : } /* newfacet */
    1123             : 
    1124             : 
    1125             : /*-<a                             href="qh-poly_r.htm#TOC"
    1126             :   >-------------------------------</a><a name="newridge">-</a>
    1127             : 
    1128             :   qh_newridge()
    1129             :     return a new ridge
    1130             :   notes:
    1131             :     caller sets qh.traceridge
    1132             : */
    1133       64555 : ridgeT *qh_newridge(qhT *qh) {
    1134             :   ridgeT *ridge;
    1135             :   void **freelistp;   /* used if !qh_NOmem by qh_memalloc_() */
    1136             : 
    1137       64555 :   qh_memalloc_(qh, (int)sizeof(ridgeT), freelistp, ridge, ridgeT);
    1138       64555 :   memset((char *)ridge, (size_t)0, sizeof(ridgeT));
    1139       64555 :   zinc_(Ztotridges);
    1140       64555 :   ridge->id= qh->ridge_id;
    1141       64555 :   if (qh->ridge_id == UINT_MAX) {
    1142           0 :     qh_fprintf(qh, qh->ferr, 7074, "qhull warning: more than 2^32 ridges.  Qhull results are OK.  Since the ridge ID wraps around to 0, two ridges may have the same identifier.\n");
    1143           0 :     qh->ridge_id = 0;
    1144             :   } else {
    1145       64555 :     qh->ridge_id++;
    1146             :   }
    1147       64555 :   trace4((qh, qh->ferr, 4056, "qh_newridge: created ridge r%d\n", ridge->id));
    1148       64555 :   return(ridge);
    1149             : } /* newridge */
    1150             : 
    1151             : 
    1152             : /*-<a                             href="qh-poly_r.htm#TOC"
    1153             :   >-------------------------------</a><a name="pointid">-</a>
    1154             : 
    1155             :   qh_pointid(qh, point )
    1156             :     return id for a point,
    1157             :     returns qh_IDnone(-3) if null, qh_IDinterior(-2) if interior, or qh_IDunknown(-1) if not known
    1158             : 
    1159             :   alternative code if point is in qh.first_point...
    1160             :     unsigned long id;
    1161             :     id= ((unsigned long)point - (unsigned long)qh.first_point)/qh.normal_size;
    1162             : 
    1163             :   notes:
    1164             :     Valid points are non-negative
    1165             :     WARN64 -- id truncated to 32-bits, at most 2G points
    1166             :     NOerrors returned (QhullPoint::id)
    1167             :     if point not in point array
    1168             :       the code does a comparison of unrelated pointers.
    1169             : */
    1170      160123 : int qh_pointid(qhT *qh, pointT *point) {
    1171             :   ptr_intT offset, id;
    1172             : 
    1173      160123 :   if (!point || !qh)
    1174           8 :     return qh_IDnone;
    1175      160115 :   else if (point == qh->interior_point)
    1176           0 :     return qh_IDinterior;
    1177      160115 :   else if (point >= qh->first_point
    1178      160115 :   && point < qh->first_point + qh->num_points * qh->hull_dim) {
    1179      160115 :     offset= (ptr_intT)(point - qh->first_point);
    1180             :     /* coverity[divide_arg] */
    1181      160115 :     id= offset / qh->hull_dim;
    1182             :   } else {
    1183           0 :     const int idx = qh_setindex(qh->other_points, point);
    1184           0 :     if (idx >= 0) {
    1185           0 :       id = (ptr_intT)idx + qh->num_points;
    1186             :     } else {
    1187           0 :       return qh_IDunknown;
    1188             :     }
    1189             :   }
    1190      160115 :   return (int)id;
    1191             : } /* pointid */
    1192             : 
    1193             : /*-<a                             href="qh-poly_r.htm#TOC"
    1194             :   >-------------------------------</a><a name="removefacet">-</a>
    1195             : 
    1196             :   qh_removefacet(qh, facet )
    1197             :     unlinks facet from qh.facet_list,
    1198             : 
    1199             :   returns:
    1200             :     updates qh.facet_list .newfacet_list .facet_next visible_list
    1201             :     decrements qh.num_facets
    1202             : 
    1203             :   see:
    1204             :     qh_appendfacet
    1205             : */
    1206      224435 : void qh_removefacet(qhT *qh, facetT *facet) {
    1207      224435 :   facetT *next= facet->next, *previous= facet->previous; /* next is always defined */
    1208             : 
    1209      224435 :   if (facet == qh->newfacet_list)
    1210       11395 :     qh->newfacet_list= next;
    1211      224435 :   if (facet == qh->facet_next)
    1212       20369 :     qh->facet_next= next;
    1213      224435 :   if (facet == qh->visible_list)
    1214       58972 :     qh->visible_list= next;
    1215      224435 :   if (previous) {
    1216      224416 :     previous->next= next;
    1217      224416 :     next->previous= previous;
    1218             :   }else {  /* 1st facet in qh->facet_list */
    1219          19 :     qh->facet_list= next;
    1220          19 :     qh->facet_list->previous= NULL;
    1221             :   }
    1222      224435 :   qh->num_facets--;
    1223      224435 :   trace4((qh, qh->ferr, 4057, "qh_removefacet: removed f%d from facet_list, newfacet_list, and visible_list\n", facet->id));
    1224      224435 : } /* removefacet */
    1225             : 
    1226             : 
    1227             : /*-<a                             href="qh-poly_r.htm#TOC"
    1228             :   >-------------------------------</a><a name="removevertex">-</a>
    1229             : 
    1230             :   qh_removevertex(qh, vertex )
    1231             :     unlinks vertex from qh.vertex_list,
    1232             : 
    1233             :   returns:
    1234             :     updates qh.vertex_list .newvertex_list
    1235             :     decrements qh.num_vertices
    1236             : */
    1237      136331 : void qh_removevertex(qhT *qh, vertexT *vertex) {
    1238      136331 :   vertexT *next= vertex->next, *previous= vertex->previous; /* next is always defined */
    1239             : 
    1240      136331 :   trace4((qh, qh->ferr, 4058, "qh_removevertex: remove v%d from qh.vertex_list\n", vertex->id));
    1241      136331 :   if (vertex == qh->newvertex_list)
    1242           0 :     qh->newvertex_list= next;
    1243      136331 :   if (previous) {
    1244      136299 :     previous->next= next;
    1245      136299 :     next->previous= previous;
    1246             :   }else {  /* 1st vertex in qh->vertex_list */
    1247          32 :     qh->vertex_list= next;
    1248          32 :     qh->vertex_list->previous= NULL;
    1249             :   }
    1250      136331 :   qh->num_vertices--;
    1251      136331 : } /* removevertex */
    1252             : 
    1253             : 
    1254             : /*-<a                             href="qh-poly_r.htm#TOC"
    1255             :   >-------------------------------</a><a name="update_vertexneighbors">-</a>
    1256             : 
    1257             :   qh_update_vertexneighbors(qh )
    1258             :     update vertex neighbors and delete interior vertices
    1259             : 
    1260             :   returns:
    1261             :     if qh.VERTEXneighbors,
    1262             :       if qh.newvertex_list,
    1263             :          removes visible neighbors from vertex neighbors
    1264             :       if qh.newfacet_list
    1265             :          adds new facets to vertex neighbors
    1266             :       if qh.visible_list
    1267             :          interior vertices added to qh.del_vertices for later partitioning as coplanar points
    1268             :     if not qh.VERTEXneighbors (not merging)
    1269             :       interior vertices of visible facets added to qh.del_vertices for later partitioning as coplanar points
    1270             : 
    1271             :   notes
    1272             :     [jan'19] split off qh_update_vertexneighbors_cone.  Optimize the remaining cases in a future release
    1273             :     called by qh_triangulate_facet after triangulating a non-simplicial facet, followed by reset_lists
    1274             :     called by qh_triangulate after triangulating null and mirror facets
    1275             :     called by qh_all_vertexmerges after calling qh_merge_pinchedvertices
    1276             : 
    1277             :   design:
    1278             :     if qh.VERTEXneighbors
    1279             :       for each vertex on newvertex_list (i.e., new vertices and vertices of new facets)
    1280             :         delete visible facets from vertex neighbors
    1281             :       for each new facet on newfacet_list
    1282             :         for each vertex of facet
    1283             :           append facet to vertex neighbors
    1284             :       for each visible facet on qh.visible_list
    1285             :         for each vertex of facet
    1286             :           if the vertex is not on a new facet and not itself deleted
    1287             :             if the vertex has a not-visible neighbor (due to merging)
    1288             :                remove the visible facet from the vertex's neighbors
    1289             :             otherwise
    1290             :                add the vertex to qh.del_vertices for later deletion
    1291             : 
    1292             :     if not qh.VERTEXneighbors (not merging)
    1293             :       for each vertex of a visible facet
    1294             :         if the vertex is not on a new facet and not itself deleted
    1295             :            add the vertex to qh.del_vertices for later deletion
    1296             : */
    1297       14413 : void qh_update_vertexneighbors(qhT *qh /* qh.newvertex_list, newfacet_list, visible_list */) {
    1298       14413 :   facetT *newfacet= NULL, *neighbor, **neighborp, *visible;
    1299             :   vertexT *vertex, **vertexp;
    1300       14413 :   int neighborcount= 0;
    1301             : 
    1302       14413 :   if (qh->VERTEXneighbors) {
    1303       14413 :     trace3((qh, qh->ferr, 3013, "qh_update_vertexneighbors: update v.neighbors for qh.newvertex_list (v%d) and qh.newfacet_list (f%d)\n",
    1304             :          getid_(qh->newvertex_list), getid_(qh->newfacet_list)));
    1305       29058 :     FORALLvertex_(qh->newvertex_list) {
    1306       14645 :       neighborcount= 0;
    1307      247019 :       FOREACHneighbor_(vertex) {
    1308      232374 :         if (neighbor->visible) {
    1309      144518 :           neighborcount++;
    1310      144518 :           SETref_(neighbor)= NULL;
    1311             :         }
    1312             :       }
    1313       14645 :       if (neighborcount) {
    1314       14645 :         trace4((qh, qh->ferr, 4046, "qh_update_vertexneighbors: delete %d of %d vertex neighbors for v%d.  Removes to-be-deleted, visible facets\n",
    1315             :           neighborcount, qh_setsize(qh, vertex->neighbors), vertex->id));
    1316       14645 :         qh_setcompact(qh, vertex->neighbors);
    1317             :       }
    1318             :     }
    1319       72501 :     FORALLnew_facets {
    1320       58088 :       if (qh->first_newfacet && newfacet->id >= qh->first_newfacet) {
    1321      232352 :         FOREACHvertex_(newfacet->vertices)
    1322      174264 :           qh_setappend(qh, &vertex->neighbors, newfacet);
    1323             :       }else {  /* called after qh_merge_pinchedvertices.  In 7-D, many more neighbors than new facets.  qh_setin is expensive */
    1324           0 :         FOREACHvertex_(newfacet->vertices)
    1325           0 :           qh_setunique(qh, &vertex->neighbors, newfacet);
    1326             :       }
    1327             :     }
    1328       14413 :     trace3((qh, qh->ferr, 3058, "qh_update_vertexneighbors: delete interior vertices for qh.visible_list (f%d)\n",
    1329             :         getid_(qh->visible_list)));
    1330       14413 :     FORALLvisible_facets {
    1331           0 :       FOREACHvertex_(visible->vertices) {
    1332           0 :         if (!vertex->newfacet && !vertex->deleted) {
    1333           0 :           FOREACHneighbor_(vertex) { /* this can happen under merging */
    1334           0 :             if (!neighbor->visible)
    1335           0 :               break;
    1336             :           }
    1337           0 :           if (neighbor)
    1338           0 :             qh_setdel(vertex->neighbors, visible);
    1339             :           else {
    1340           0 :             vertex->deleted= True;
    1341           0 :             qh_setappend(qh, &qh->del_vertices, vertex);
    1342           0 :             trace2((qh, qh->ferr, 2041, "qh_update_vertexneighbors: delete interior vertex p%d(v%d) of visible f%d\n",
    1343             :                   qh_pointid(qh, vertex->point), vertex->id, visible->id));
    1344             :           }
    1345             :         }
    1346             :       }
    1347             :     }
    1348             :   }else {  /* !VERTEXneighbors */
    1349           0 :     trace3((qh, qh->ferr, 3058, "qh_update_vertexneighbors: delete old vertices for qh.visible_list (f%d)\n",
    1350             :       getid_(qh->visible_list)));
    1351           0 :     FORALLvisible_facets {
    1352           0 :       FOREACHvertex_(visible->vertices) {
    1353           0 :         if (!vertex->newfacet && !vertex->deleted) {
    1354           0 :           vertex->deleted= True;
    1355           0 :           qh_setappend(qh, &qh->del_vertices, vertex);
    1356           0 :           trace2((qh, qh->ferr, 2042, "qh_update_vertexneighbors: will delete interior vertex p%d(v%d) of visible f%d\n",
    1357             :                   qh_pointid(qh, vertex->point), vertex->id, visible->id));
    1358             :         }
    1359             :       }
    1360             :     }
    1361             :   }
    1362       14413 : } /* update_vertexneighbors */
    1363             : 
    1364             : /*-<a                             href="qh-poly_r.htm#TOC"
    1365             :   >-------------------------------</a><a name="update_vertexneighbors_cone">-</a>
    1366             : 
    1367             :   qh_update_vertexneighbors_cone(qh )
    1368             :     update vertex neighbors for a cone of new facets and delete interior vertices
    1369             : 
    1370             :   returns:
    1371             :     if qh.VERTEXneighbors,
    1372             :       if qh.newvertex_list,
    1373             :          removes visible neighbors from vertex neighbors
    1374             :       if qh.newfacet_list
    1375             :          adds new facets to vertex neighbors
    1376             :       if qh.visible_list
    1377             :          interior vertices added to qh.del_vertices for later partitioning as coplanar points
    1378             :     if not qh.VERTEXneighbors (not merging)
    1379             :       interior vertices of visible facets added to qh.del_vertices for later partitioning as coplanar points
    1380             : 
    1381             :   notes
    1382             :     called by qh_addpoint after create cone and before premerge
    1383             : 
    1384             :   design:
    1385             :     if qh.VERTEXneighbors
    1386             :       for each vertex on newvertex_list (i.e., new vertices and vertices of new facets)
    1387             :         delete visible facets from vertex neighbors
    1388             :       for each new facet on newfacet_list
    1389             :         for each vertex of facet
    1390             :           append facet to vertex neighbors
    1391             :       for each visible facet on qh.visible_list
    1392             :         for each vertex of facet
    1393             :           if the vertex is not on a new facet and not itself deleted
    1394             :             if the vertex has a not-visible neighbor (due to merging)
    1395             :                remove the visible facet from the vertex's neighbors
    1396             :             otherwise
    1397             :                add the vertex to qh.del_vertices for later deletion
    1398             : 
    1399             :     if not qh.VERTEXneighbors (not merging)
    1400             :       for each vertex of a visible facet
    1401             :         if the vertex is not on a new facet and not itself deleted
    1402             :            add the vertex to qh.del_vertices for later deletion
    1403             : 
    1404             : */
    1405       14670 : void qh_update_vertexneighbors_cone(qhT *qh /* qh.newvertex_list, newfacet_list, visible_list */) {
    1406       14670 :   facetT *newfacet= NULL, *neighbor, **neighborp, *visible;
    1407             :   vertexT *vertex, **vertexp;
    1408       14670 :   int delcount= 0;
    1409             : 
    1410       14670 :   if (qh->VERTEXneighbors) {
    1411       14655 :     trace3((qh, qh->ferr, 3059, "qh_update_vertexneighbors_cone: update v.neighbors for qh.newvertex_list (v%d) and qh.newfacet_list (f%d)\n",
    1412             :          getid_(qh->newvertex_list), getid_(qh->newfacet_list)));
    1413      102692 :     FORALLvertex_(qh->newvertex_list) {
    1414       88037 :       delcount= 0;
    1415      503805 :       FOREACHneighbor_(vertex) {
    1416      415768 :         if (neighbor->visible) { /* alternative design is a loop over visible facets, but needs qh_setdel() */
    1417      121856 :           delcount++;
    1418      121856 :           qh_setdelnth(qh, vertex->neighbors, SETindex_(vertex->neighbors, neighbor));
    1419      121856 :           neighborp--; /* repeat */
    1420             :         }
    1421             :       }
    1422       88037 :       if (delcount) {
    1423       73382 :         trace4((qh, qh->ferr, 4021, "qh_update_vertexneighbors_cone: deleted %d visible vertexneighbors of v%d\n",
    1424             :           delcount, vertex->id));
    1425             :       }
    1426             :     }
    1427       88037 :     FORALLnew_facets {
    1428      293528 :       FOREACHvertex_(newfacet->vertices)
    1429      220146 :         qh_setappend(qh, &vertex->neighbors, newfacet);
    1430             :     }
    1431       14655 :     trace3((qh, qh->ferr, 3065, "qh_update_vertexneighbors_cone: delete interior vertices, if any, for qh.visible_list (f%d)\n",
    1432             :         getid_(qh->visible_list)));
    1433       53547 :     FORALLvisible_facets {
    1434      160748 :       FOREACHvertex_(visible->vertices) {
    1435      121856 :         if (!vertex->newfacet && !vertex->deleted) {
    1436           0 :           FOREACHneighbor_(vertex) { /* this can happen under merging, qh_checkfacet QH4025 */
    1437           0 :             if (!neighbor->visible)
    1438           0 :               break;
    1439             :           }
    1440           0 :           if (neighbor)
    1441           0 :             qh_setdel(vertex->neighbors, visible);
    1442             :           else {
    1443           0 :             vertex->deleted= True;
    1444           0 :             qh_setappend(qh, &qh->del_vertices, vertex);
    1445           0 :             trace2((qh, qh->ferr, 2102, "qh_update_vertexneighbors_cone: will delete interior vertex p%d(v%d) of visible f%d\n",
    1446             :               qh_pointid(qh, vertex->point), vertex->id, visible->id));
    1447             :           }
    1448             :         }
    1449             :       }
    1450             :     }
    1451             :   }else {  /* !VERTEXneighbors */
    1452          15 :     trace3((qh, qh->ferr, 3066, "qh_update_vertexneighbors_cone: delete interior vertices for qh.visible_list (f%d)\n",
    1453             :       getid_(qh->visible_list)));
    1454          42 :     FORALLvisible_facets {
    1455         108 :       FOREACHvertex_(visible->vertices) {
    1456          81 :         if (!vertex->newfacet && !vertex->deleted) {
    1457           0 :           vertex->deleted= True;
    1458           0 :           qh_setappend(qh, &qh->del_vertices, vertex);
    1459           0 :           trace2((qh, qh->ferr, 2059, "qh_update_vertexneighbors_cone: will delete interior vertex p%d(v%d) of visible f%d\n",
    1460             :                   qh_pointid(qh, vertex->point), vertex->id, visible->id));
    1461             :         }
    1462             :       }
    1463             :     }
    1464             :   }
    1465       14670 : } /* update_vertexneighbors_cone */
    1466             : 

Generated by: LCOV version 1.14