PostGIS  3.2.2dev-r@@SVN_REVISION@@

◆ LWGEOM_angle()

Datum LWGEOM_angle ( PG_FUNCTION_ARGS  )

Definition at line 2583 of file lwgeom_functions_basic.c.

2584 {
2585  GSERIALIZED *seri_geoms[4];
2586  LWGEOM *geom_unser;
2587  LWPOINT *lwpoint;
2588  POINT2D points[4];
2589  double az1, az2;
2590  double result;
2591  int32_t srids[4];
2592  int i = 0;
2593  int j = 0;
2594  int err_code = 0;
2595  int n_args = PG_NARGS();
2596 
2597  /* no deserialize, checking for common error first*/
2598  for (i = 0; i < n_args; i++)
2599  {
2600  seri_geoms[i] = PG_GETARG_GSERIALIZED_P(i);
2601  if (gserialized_is_empty(seri_geoms[i]))
2602  { /* empty geom */
2603  if (i == 3)
2604  {
2605  n_args = 3;
2606  }
2607  else
2608  {
2609  err_code = 1;
2610  break;
2611  }
2612  }
2613  else
2614  {
2615  if (gserialized_get_type(seri_geoms[i]) != POINTTYPE)
2616  { /* geom type */
2617  err_code = 2;
2618  break;
2619  }
2620  else
2621  {
2622  srids[i] = gserialized_get_srid(seri_geoms[i]);
2623  if (srids[0] != srids[i])
2624  { /* error on srid*/
2625  err_code = 3;
2626  break;
2627  }
2628  }
2629  }
2630  }
2631  if (err_code > 0)
2632  switch (err_code)
2633  {
2634  default: /*always executed*/
2635  for (j = 0; j <= i; j++)
2636  PG_FREE_IF_COPY(seri_geoms[j], j);
2637  /*FALLTHROUGH*/
2638  case 1:
2639  lwpgerror("Empty geometry");
2640  PG_RETURN_NULL();
2641  break;
2642 
2643  case 2:
2644  lwpgerror("Argument must be POINT geometries");
2645  PG_RETURN_NULL();
2646  break;
2647 
2648  case 3:
2649  lwpgerror("Operation on mixed SRID geometries");
2650  PG_RETURN_NULL();
2651  break;
2652  }
2653  /* extract points */
2654  for (i = 0; i < n_args; i++)
2655  {
2656  geom_unser = lwgeom_from_gserialized(seri_geoms[i]);
2657  lwpoint = lwgeom_as_lwpoint(geom_unser);
2658  if (!lwpoint)
2659  {
2660  for (j = 0; j < n_args; j++)
2661  PG_FREE_IF_COPY(seri_geoms[j], j);
2662  lwpgerror("Error unserializing geometry");
2663  PG_RETURN_NULL();
2664  }
2665 
2666  if (!getPoint2d_p(lwpoint->point, 0, &points[i]))
2667  {
2668  /* // can't free serialized geom, it might be needed by lw
2669  for (j=0;j<n_args;j++)
2670  PG_FREE_IF_COPY(seri_geoms[j], j); */
2671  lwpgerror("Error extracting point");
2672  PG_RETURN_NULL();
2673  }
2674  /* lwfree(geom_unser);don't do, lw may rely on this memory
2675  lwpoint_free(lwpoint); dont do , this memory is needed ! */
2676  }
2677  /* // can't free serialized geom, it might be needed by lw
2678  for (j=0;j<n_args;j++)
2679  PG_FREE_IF_COPY(seri_geoms[j], j); */
2680 
2681  /* compute azimuth for the 2 pairs of points
2682  * note that angle is not defined identically for 3 points or 4 points*/
2683  if (n_args == 3)
2684  { /* we rely on azimuth to complain if points are identical */
2685  if (!azimuth_pt_pt(&points[0], &points[1], &az1))
2686  PG_RETURN_NULL();
2687  if (!azimuth_pt_pt(&points[2], &points[1], &az2))
2688  PG_RETURN_NULL();
2689  }
2690  else
2691  {
2692  if (!azimuth_pt_pt(&points[0], &points[1], &az1))
2693  PG_RETURN_NULL();
2694  if (!azimuth_pt_pt(&points[2], &points[3], &az2))
2695  PG_RETURN_NULL();
2696  }
2697  result = az2 - az1;
2698  result += (result < 0) * 2 * M_PI; /* we dont want negative angle*/
2699  PG_RETURN_FLOAT8(result);
2700 }
char result[OUT_DOUBLE_BUFFER_SIZE]
Definition: cu_print.c:267
int32_t gserialized_get_srid(const GSERIALIZED *g)
Extract the SRID from the serialized form (it is packed into three bytes so this is a handy function)...
Definition: gserialized.c:126
LWGEOM * lwgeom_from_gserialized(const GSERIALIZED *g)
Allocate a new LWGEOM from a GSERIALIZED.
Definition: gserialized.c:239
int gserialized_is_empty(const GSERIALIZED *g)
Check if a GSERIALIZED is empty without deserializing first.
Definition: gserialized.c:152
uint32_t gserialized_get_type(const GSERIALIZED *g)
Extract the geometry type from the serialized form (it hides in the anonymous data area,...
Definition: gserialized.c:89
int azimuth_pt_pt(const POINT2D *p1, const POINT2D *p2, double *ret)
Compute the azimuth of segment AB in radians.
Definition: measures.c:2461
int getPoint2d_p(const POINTARRAY *pa, uint32_t n, POINT2D *point)
Definition: lwgeom_api.c:343
#define POINTTYPE
LWTYPE numbers, used internally by PostGIS.
Definition: liblwgeom.h:116
static LWPOINT * lwgeom_as_lwpoint(const LWGEOM *lwgeom)
Definition: lwinline.h:131
POINTARRAY * point
Definition: liblwgeom.h:485

References azimuth_pt_pt(), getPoint2d_p(), gserialized_get_srid(), gserialized_get_type(), gserialized_is_empty(), lwgeom_as_lwpoint(), lwgeom_from_gserialized(), LWPOINT::point, POINTTYPE, and result.

Here is the call graph for this function: