PostGIS 3.7.0dev-r@@SVN_REVISION@@
Loading...
Searching...
No Matches

◆ _lwt_AddEdge()

static LWT_ELEMID _lwt_AddEdge ( LWT_TOPOLOGY topo,
LWT_ELEMID  start_node,
LWT_ELEMID  end_node,
LWLINE geom,
int  skipChecks,
int  modFace 
)
static
Parameters
modFacecan be 0 - have two new faces replace a split face 1 - modify a split face, adding a new one -1 - do not check at all for face splitting

Definition at line 2420 of file lwgeom_topo.c.

2423{
2424 LWT_ISO_EDGE newedge;
2425 LWGEOM *cleangeom;
2426 edgeend span; /* start point analysis */
2427 edgeend epan; /* end point analysis */
2428 POINT2D p1, pn, p2;
2429 POINTARRAY *pa;
2430 LWT_ELEMID node_ids[2];
2431 const LWPOINT *start_node_geom = NULL;
2432 const LWPOINT *end_node_geom = NULL;
2433 uint64_t num_nodes;
2434 LWT_ISO_NODE *endpoints;
2435 uint64_t i;
2436 LWT_ELEMID prev_left;
2437 LWT_ELEMID prev_right;
2438 LWT_ISO_EDGE seledge;
2439 LWT_ISO_EDGE updedge;
2440
2441 if ( ! skipChecks )
2442 {
2443 /* curve must be simple */
2444 if ( ! lwgeom_is_simple(lwline_as_lwgeom(geom)) )
2445 {
2446 lwerror("SQL/MM Spatial exception - curve not simple");
2447 return -1;
2448 }
2449 }
2450
2451 newedge.start_node = start_node;
2452 newedge.end_node = end_node;
2453 newedge.geom = geom;
2454 newedge.face_left = -1;
2455 newedge.face_right = -1;
2456 /* TODO: should do the repeated points removal in 2D space */
2457 cleangeom = lwgeom_remove_repeated_points( lwline_as_lwgeom(geom), 0 );
2458
2459 pa = lwgeom_as_lwline(cleangeom)->points;
2460 if ( pa->npoints < 2 ) {
2461 lwgeom_free(cleangeom);
2462 lwerror("Invalid edge (no two distinct vertices exist)");
2463 return -1;
2464 }
2465
2466 /* Initialize endpoint info (some of that ) */
2467 span.cwFace = span.ccwFace =
2468 epan.cwFace = epan.ccwFace = -1;
2469
2470 /* Compute azimuth of first edge end on start node */
2471 getPoint2d_p(pa, 0, &p1);
2472 if ( ! _lwt_FirstDistinctVertex2D(pa, &p1, 0, 1, &pn) )
2473 {
2474 lwgeom_free(cleangeom);
2475 lwerror("Invalid edge (no two distinct vertices exist)");
2476 return -1;
2477 }
2478 if ( ! azimuth_pt_pt(&p1, &pn, &span.myaz) ) {
2479 lwgeom_free(cleangeom);
2480 lwerror("error computing azimuth of first edgeend [%.15g %.15g,%.15g %.15g]",
2481 p1.x, p1.y, pn.x, pn.y);
2482 return -1;
2483 }
2484 LWDEBUGF(1, "edge's start node is %g,%g", p1.x, p1.y);
2485
2486 /* Compute azimuth of last edge end on end node */
2487 getPoint2d_p(pa, pa->npoints-1, &p2);
2488 if ( ! _lwt_FirstDistinctVertex2D(pa, &p2, pa->npoints-1, -1, &pn) )
2489 {
2490 lwgeom_free(cleangeom);
2491 /* This should never happen as we checked the edge while computing first edgend */
2492 lwerror("Invalid clean edge (no two distinct vertices exist) - should not happen");
2493 return -1;
2494 }
2495 lwgeom_free(cleangeom);
2496 if ( ! azimuth_pt_pt(&p2, &pn, &epan.myaz) ) {
2497 lwerror("error computing azimuth of last edgeend [%.15g %.15g,%.15g %.15g]",
2498 p2.x, p2.y, pn.x, pn.y);
2499 return -1;
2500 }
2501 LWDEBUGF(1, "edge's end node is %g,%g", p2.x, p2.y);
2502
2503 /*
2504 * Check endpoints existence, match with Curve geometry
2505 * and get face information (if any)
2506 */
2507
2508 if ( start_node != end_node ) {
2509 num_nodes = 2;
2510 node_ids[0] = start_node;
2511 node_ids[1] = end_node;
2512 } else {
2513 num_nodes = 1;
2514 node_ids[0] = start_node;
2515 }
2516
2517 endpoints = lwt_be_getNodeById( topo, node_ids, &num_nodes, LWT_COL_NODE_ALL );
2518 if (num_nodes == UINT64_MAX)
2519 {
2521 return -1;
2522 }
2523 for ( i=0; i<num_nodes; ++i )
2524 {
2525 LWT_ISO_NODE* node = &(endpoints[i]);
2526 if ( modFace != -1 && node->containing_face != -1 )
2527 {
2528 if ( newedge.face_left == -1 )
2529 {
2530 newedge.face_left = newedge.face_right = node->containing_face;
2531 }
2532 else if ( newedge.face_left != node->containing_face )
2533 {
2534 _lwt_release_nodes(endpoints, num_nodes);
2535 lwerror("SQL/MM Spatial exception - geometry crosses an edge"
2536 " (endnodes in faces %" LWTFMT_ELEMID " and %" LWTFMT_ELEMID ")",
2537 newedge.face_left, node->containing_face);
2538 }
2539 }
2540
2541 LWDEBUGF(1, "Node %" LWTFMT_ELEMID ", with geom %p (looking for %"
2542 LWTFMT_ELEMID " and %" LWTFMT_ELEMID ")",
2543 node->node_id, node->geom, start_node, end_node);
2544 if ( node->node_id == start_node ) {
2545 start_node_geom = node->geom;
2546 }
2547 if ( node->node_id == end_node ) {
2548 end_node_geom = node->geom;
2549 }
2550 }
2551
2552 if ( ! skipChecks )
2553 {
2554 if ( ! start_node_geom )
2555 {
2556 if ( num_nodes ) _lwt_release_nodes(endpoints, num_nodes);
2557 lwerror("SQL/MM Spatial exception - non-existent node");
2558 return -1;
2559 }
2560 else
2561 {
2562 pa = start_node_geom->point;
2563 getPoint2d_p(pa, 0, &pn);
2564 if ( ! P2D_SAME_STRICT(&pn, &p1) )
2565 {
2566 if ( num_nodes ) _lwt_release_nodes(endpoints, num_nodes);
2567 lwerror("SQL/MM Spatial exception"
2568 " - start node not geometry start point."
2569 //" - start node not geometry start point (%g,%g != %g,%g).", pn.x, pn.y, p1.x, p1.y
2570 );
2571 return -1;
2572 }
2573 }
2574
2575 if ( ! end_node_geom )
2576 {
2577 if ( num_nodes ) _lwt_release_nodes(endpoints, num_nodes);
2578 lwerror("SQL/MM Spatial exception - non-existent node");
2579 return -1;
2580 }
2581 else
2582 {
2583 pa = end_node_geom->point;
2584 getPoint2d_p(pa, 0, &pn);
2585 if ( ! P2D_SAME_STRICT(&pn, &p2) )
2586 {
2587 if ( num_nodes ) _lwt_release_nodes(endpoints, num_nodes);
2588 lwerror("SQL/MM Spatial exception"
2589 " - end node not geometry end point."
2590 //" - end node not geometry end point (%g,%g != %g,%g).", pn.x, pn.y, p2.x, p2.y
2591 );
2592 return -1;
2593 }
2594 }
2595
2596 if ( num_nodes ) _lwt_release_nodes(endpoints, num_nodes);
2597
2598 if ( _lwt_CheckEdgeCrossing( topo, start_node, end_node, geom, 0 ) )
2599 return -1;
2600
2601 } /* ! skipChecks */
2602
2603 /*
2604 * All checks passed, time to prepare the new edge
2605 */
2606
2607 newedge.edge_id = lwt_be_getNextEdgeId( topo );
2608 if ( newedge.edge_id == -1 ) {
2610 return -1;
2611 }
2612
2613 /* Find adjacent edges to each endpoint */
2614 int isclosed = start_node == end_node;
2615 int found;
2616 found = _lwt_FindAdjacentEdges( topo, start_node, &span,
2617 isclosed ? &epan : NULL, -1 );
2618 if ( found ) {
2619 span.was_isolated = 0;
2620 newedge.next_right = span.nextCW ? span.nextCW : -newedge.edge_id;
2621 prev_left = span.nextCCW ? -span.nextCCW : newedge.edge_id;
2622 LWDEBUGF(1, "New edge %" LWTFMT_ELEMID " is connected on start node, "
2623 "next_right is %" LWTFMT_ELEMID
2624 ", prev_left is %" LWTFMT_ELEMID,
2625 newedge.edge_id, newedge.next_right, prev_left);
2626 if ( modFace != -1 )
2627 {
2628 if ( newedge.face_right == -1 ) {
2629 newedge.face_right = span.cwFace;
2630 }
2631 if ( newedge.face_left == -1 ) {
2632 newedge.face_left = span.ccwFace;
2633 }
2634 }
2635 } else {
2636 span.was_isolated = 1;
2637 newedge.next_right = isclosed ? -newedge.edge_id : newedge.edge_id;
2638 prev_left = isclosed ? newedge.edge_id : -newedge.edge_id;
2639 LWDEBUGF(1, "New edge %" LWTFMT_ELEMID " is isolated on start node, "
2640 "next_right is %" LWTFMT_ELEMID
2641 ", prev_left is %" LWTFMT_ELEMID,
2642 newedge.edge_id, newedge.next_right, prev_left);
2643 }
2644
2645 found = _lwt_FindAdjacentEdges( topo, end_node, &epan,
2646 isclosed ? &span : NULL, -1 );
2647 if ( found ) {
2648 epan.was_isolated = 0;
2649 newedge.next_left = epan.nextCW ? epan.nextCW : newedge.edge_id;
2650 prev_right = epan.nextCCW ? -epan.nextCCW : -newedge.edge_id;
2651 LWDEBUGF(1, "New edge %" LWTFMT_ELEMID " is connected on end node, "
2652 "next_left is %" LWTFMT_ELEMID
2653 ", prev_right is %" LWTFMT_ELEMID,
2654 newedge.edge_id, newedge.next_left, prev_right);
2655 if ( modFace != -1 )
2656 {
2657 if ( newedge.face_right == -1 ) {
2658 newedge.face_right = span.ccwFace;
2659 } else if ( newedge.face_right != epan.ccwFace ) {
2660 /* side-location conflict */
2661 lwerror("Side-location conflict: "
2662 "new edge starts in face"
2663 " %" LWTFMT_ELEMID " and ends in face"
2664 " %" LWTFMT_ELEMID,
2665 newedge.face_right, epan.ccwFace
2666 );
2667 return -1;
2668 }
2669 if ( newedge.face_left == -1 ) {
2670 newedge.face_left = span.cwFace;
2671 } else if ( newedge.face_left != epan.cwFace ) {
2672 /* side-location conflict */
2673 lwerror("Side-location conflict: "
2674 "new edge starts in face"
2675 " %" LWTFMT_ELEMID " and ends in face"
2676 " %" LWTFMT_ELEMID,
2677 newedge.face_left, epan.cwFace
2678 );
2679 return -1;
2680 }
2681 }
2682 } else {
2683 epan.was_isolated = 1;
2684 newedge.next_left = isclosed ? newedge.edge_id : -newedge.edge_id;
2685 prev_right = isclosed ? -newedge.edge_id : newedge.edge_id;
2686 LWDEBUGF(1, "New edge %" LWTFMT_ELEMID " is isolated on end node, "
2687 "next_left is %" LWTFMT_ELEMID
2688 ", prev_right is %" LWTFMT_ELEMID,
2689 newedge.edge_id, newedge.next_left, prev_right);
2690 }
2691
2692 /*
2693 * If we don't have faces setup by now we must have encountered
2694 * a malformed topology (no containing_face on isolated nodes, no
2695 * left/right faces on adjacent edges or mismatching values)
2696 */
2697 if ( modFace > -1 )
2698 {
2699 if ( newedge.face_left != newedge.face_right )
2700 {
2701 lwerror("Left(%" LWTFMT_ELEMID ")/right(%" LWTFMT_ELEMID ")"
2702 " faces mismatch: invalid topology ?",
2703 newedge.face_left, newedge.face_right);
2704 return -1;
2705 }
2706 else if ( newedge.face_left == -1 )
2707 {
2708 lwerror("Could not derive edge face from linked primitives:"
2709 " invalid topology ?");
2710 return -1;
2711 }
2712 }
2713
2714 /*
2715 * Insert the new edge, and update all linking
2716 */
2717
2718 int ret = lwt_be_insertEdges(topo, &newedge, 1);
2719 if ( ret == -1 ) {
2721 return -1;
2722 } else if ( ret == 0 ) {
2723 lwerror("Insertion of split edge failed (no reason)");
2724 return -1;
2725 }
2726
2727 int updfields;
2728
2729 /* Link prev_left to us
2730 * (if it's not us already) */
2731 if ( llabs(prev_left) != newedge.edge_id )
2732 {
2733 if ( prev_left > 0 )
2734 {
2735 /* its next_left_edge is us */
2736 updfields = LWT_COL_EDGE_NEXT_LEFT;
2737 updedge.next_left = newedge.edge_id;
2738 seledge.edge_id = prev_left;
2739 }
2740 else
2741 {
2742 /* its next_right_edge is us */
2743 updfields = LWT_COL_EDGE_NEXT_RIGHT;
2744 updedge.next_right = newedge.edge_id;
2745 seledge.edge_id = -prev_left;
2746 }
2747
2748 ret = lwt_be_updateEdges(topo,
2749 &seledge, LWT_COL_EDGE_EDGE_ID,
2750 &updedge, updfields,
2751 NULL, 0);
2752 if ( ret == -1 ) {
2754 return -1;
2755 }
2756 }
2757
2758 /* Link prev_right to us
2759 * (if it's not us already) */
2760 if ( llabs(prev_right) != newedge.edge_id )
2761 {
2762 if ( prev_right > 0 )
2763 {
2764 /* its next_left_edge is -us */
2765 updfields = LWT_COL_EDGE_NEXT_LEFT;
2766 updedge.next_left = -newedge.edge_id;
2767 seledge.edge_id = prev_right;
2768 }
2769 else
2770 {
2771 /* its next_right_edge is -us */
2772 updfields = LWT_COL_EDGE_NEXT_RIGHT;
2773 updedge.next_right = -newedge.edge_id;
2774 seledge.edge_id = -prev_right;
2775 }
2776
2777 ret = lwt_be_updateEdges(topo,
2778 &seledge, LWT_COL_EDGE_EDGE_ID,
2779 &updedge, updfields,
2780 NULL, 0);
2781 if ( ret == -1 ) {
2783 return -1;
2784 }
2785 }
2786
2787 /* NOT IN THE SPECS...
2788 * set containing_face = null for start_node and end_node
2789 * if they where isolated
2790 *
2791 */
2792 LWT_ISO_NODE updnode, selnode;
2793 updnode.containing_face = -1;
2794 if ( span.was_isolated )
2795 {
2796 selnode.node_id = start_node;
2797 ret = lwt_be_updateNodes(topo,
2798 &selnode, LWT_COL_NODE_NODE_ID,
2800 NULL, 0);
2801 if ( ret == -1 ) {
2803 return -1;
2804 }
2805 }
2806 if ( epan.was_isolated )
2807 {
2808 selnode.node_id = end_node;
2809 ret = lwt_be_updateNodes(topo,
2810 &selnode, LWT_COL_NODE_NODE_ID,
2812 NULL, 0);
2813 if ( ret == -1 ) {
2815 return -1;
2816 }
2817 }
2818
2819 /* Check face splitting, if required */
2820
2821 if ( modFace > -1 ) {
2822
2823 if ( ! isclosed && ( epan.was_isolated || span.was_isolated ) )
2824 {
2825 LWDEBUG(1, "New edge is dangling, so it cannot split any face");
2826 return newedge.edge_id; /* no split */
2827 }
2828
2829 int newface1 = -1;
2830
2831 /* IDEA: avoid building edge ring if input is closed, which means we
2832 * know in advance it splits a face */
2833
2834 if ( ! modFace )
2835 {
2836 newface1 = _lwt_AddFaceSplit( topo, -newedge.edge_id, newedge.face_left, 0 );
2837 if ( newface1 == 0 ) {
2838 LWDEBUG(1, "New edge does not split any face");
2839 return newedge.edge_id; /* no split */
2840 }
2841 }
2842
2843 int newface = _lwt_AddFaceSplit( topo, newedge.edge_id,
2844 newedge.face_left, 0 );
2845 if ( modFace )
2846 {
2847 if ( newface == 0 ) {
2848 LWDEBUG(1, "New edge does not split any face");
2849 return newedge.edge_id; /* no split */
2850 }
2851
2852 if ( newface < 0 )
2853 {
2854 /* face on the left is the universe face */
2855 /* must be forming a maximal ring in universal face */
2856 newface = _lwt_AddFaceSplit( topo, -newedge.edge_id,
2857 newedge.face_left, 0 );
2858 if ( newface < 0 ) return newedge.edge_id; /* no split */
2859 }
2860 else
2861 {
2862 _lwt_AddFaceSplit( topo, -newedge.edge_id, newedge.face_left, 1 );
2863 }
2864 }
2865
2866 /*
2867 * Update topogeometries, if needed
2868 */
2869 if ( newedge.face_left != 0 )
2870 {
2871 ret = lwt_be_updateTopoGeomFaceSplit(topo, newedge.face_left,
2872 newface, newface1);
2873 if ( ret == 0 ) {
2875 return -1;
2876 }
2877
2878 if ( ! modFace )
2879 {
2880 /* drop old face from the face table */
2881 ret = lwt_be_deleteFacesById(topo, &(newedge.face_left), 1);
2882 if ( ret == -1 ) {
2884 return -1;
2885 }
2886 }
2887 }
2888
2889 } // end of face split checking
2890
2891 return newedge.edge_id;
2892}
int azimuth_pt_pt(const POINT2D *p1, const POINT2D *p2, double *ret)
Compute the azimuth of segment AB in radians.
Definition measures.c:2408
void lwgeom_free(LWGEOM *geom)
Definition lwgeom.c:1246
int lwgeom_is_simple(const LWGEOM *lwgeom)
int getPoint2d_p(const POINTARRAY *pa, uint32_t n, POINT2D *point)
Definition lwgeom_api.c:342
LWGEOM * lwline_as_lwgeom(const LWLINE *obj)
Definition lwgeom.c:367
LWLINE * lwgeom_as_lwline(const LWGEOM *lwgeom)
Definition lwgeom.c:207
LWGEOM * lwgeom_remove_repeated_points(const LWGEOM *in, double tolerance)
Definition lwgeom.c:1562
#define P2D_SAME_STRICT(a, b)
LWT_INT64 LWT_ELEMID
Identifier of topology element.
#define LWT_COL_EDGE_NEXT_RIGHT
#define LWT_COL_NODE_CONTAINING_FACE
#define LWT_COL_EDGE_EDGE_ID
Edge fields.
#define LWT_COL_EDGE_NEXT_LEFT
#define LWT_COL_NODE_NODE_ID
Node fields.
#define LWT_COL_NODE_ALL
#define PGTOPO_BE_ERROR()
#define LWTFMT_ELEMID
#define LWDEBUG(level, msg)
Definition lwgeom_log.h:101
#define LWDEBUGF(level, msg,...)
Definition lwgeom_log.h:106
void void lwerror(const char *fmt,...) __attribute__((format(printf
Write a notice out to the error handler.
static int lwt_be_deleteFacesById(const LWT_TOPOLOGY *topo, const LWT_ELEMID *ids, uint64_t numelems)
LWT_ELEMID lwt_be_getNextEdgeId(LWT_TOPOLOGY *topo)
static int lwt_be_updateTopoGeomFaceSplit(LWT_TOPOLOGY *topo, LWT_ELEMID split_face, LWT_ELEMID new_face1, LWT_ELEMID new_face2)
static int _lwt_CheckEdgeCrossing(LWT_TOPOLOGY *topo, LWT_ELEMID start_node, LWT_ELEMID end_node, const LWLINE *geom, LWT_ELEMID myself)
static void _lwt_release_nodes(LWT_ISO_NODE *nodes, int num_nodes)
int lwt_be_updateEdges(LWT_TOPOLOGY *topo, const LWT_ISO_EDGE *sel_edge, int sel_fields, const LWT_ISO_EDGE *upd_edge, int upd_fields, const LWT_ISO_EDGE *exc_edge, int exc_fields)
static LWT_ELEMID _lwt_AddFaceSplit(LWT_TOPOLOGY *topo, LWT_ELEMID sedge, LWT_ELEMID face, int mbr_only)
int lwt_be_insertEdges(LWT_TOPOLOGY *topo, LWT_ISO_EDGE *edge, uint64_t numelems)
static int _lwt_FirstDistinctVertex2D(const POINTARRAY *pa, const POINT2D *ref, int from, int dir, POINT2D *op)
LWT_ISO_NODE * lwt_be_getNodeById(LWT_TOPOLOGY *topo, const LWT_ELEMID *ids, uint64_t *numelems, int fields)
static int lwt_be_updateNodes(LWT_TOPOLOGY *topo, const LWT_ISO_NODE *sel_node, int sel_fields, const LWT_ISO_NODE *upd_node, int upd_fields, const LWT_ISO_NODE *exc_node, int exc_fields)
static int _lwt_FindAdjacentEdges(LWT_TOPOLOGY *topo, LWT_ELEMID node, edgeend *data, edgeend *other, LWT_ELEMID myedge_id)
POINTARRAY * points
Definition liblwgeom.h:483
POINTARRAY * point
Definition liblwgeom.h:471
LWT_ELEMID face_right
LWT_ELEMID next_right
LWT_ELEMID end_node
LWT_ELEMID face_left
LWT_ELEMID next_left
LWT_ELEMID edge_id
LWT_ELEMID start_node
LWT_ELEMID node_id
LWT_ELEMID containing_face
LWPOINT * geom
double y
Definition liblwgeom.h:390
double x
Definition liblwgeom.h:390
uint32_t npoints
Definition liblwgeom.h:427
double myaz
LWT_ELEMID nextCCW
LWT_ELEMID ccwFace
LWT_ELEMID cwFace
LWT_ELEMID nextCW

References _lwt_AddFaceSplit(), _lwt_CheckEdgeCrossing(), _lwt_FindAdjacentEdges(), _lwt_FirstDistinctVertex2D(), _lwt_release_nodes(), azimuth_pt_pt(), edgeend_t::ccwFace, LWT_ISO_NODE::containing_face, edgeend_t::cwFace, LWT_ISO_EDGE::edge_id, LWT_ISO_EDGE::end_node, LWT_ISO_EDGE::face_left, LWT_ISO_EDGE::face_right, LWT_ISO_NODE::geom, LWT_ISO_EDGE::geom, getPoint2d_p(), LWDEBUG, LWDEBUGF, lwerror(), lwgeom_as_lwline(), lwgeom_free(), lwgeom_is_simple(), lwgeom_remove_repeated_points(), lwline_as_lwgeom(), lwt_be_deleteFacesById(), lwt_be_getNextEdgeId(), lwt_be_getNodeById(), lwt_be_insertEdges(), lwt_be_updateEdges(), lwt_be_updateNodes(), lwt_be_updateTopoGeomFaceSplit(), LWT_COL_EDGE_EDGE_ID, LWT_COL_EDGE_NEXT_LEFT, LWT_COL_EDGE_NEXT_RIGHT, LWT_COL_NODE_ALL, LWT_COL_NODE_CONTAINING_FACE, LWT_COL_NODE_NODE_ID, LWTFMT_ELEMID, edgeend_t::myaz, LWT_ISO_EDGE::next_left, LWT_ISO_EDGE::next_right, edgeend_t::nextCCW, edgeend_t::nextCW, LWT_ISO_NODE::node_id, POINTARRAY::npoints, P2D_SAME_STRICT, PGTOPO_BE_ERROR, LWPOINT::point, LWLINE::points, LWT_ISO_EDGE::start_node, edgeend_t::was_isolated, POINT2D::x, and POINT2D::y.

Referenced by _lwt_AddLineEdge(), lwt_AddEdgeModFace(), and lwt_AddEdgeNewFaces().

Here is the call graph for this function:
Here is the caller graph for this function: