Line data Source code
1 : /******************************************************************************
2 : *
3 : * Project: OpenGIS Simple Features Reference Implementation
4 : * Purpose: Implements a few base methods on OGRGeometry.
5 : * Author: Frank Warmerdam, warmerdam@pobox.com
6 : *
7 : ******************************************************************************
8 : * Copyright (c) 1999, Frank Warmerdam
9 : * Copyright (c) 2008-2013, Even Rouault <even dot rouault at spatialys.com>
10 : *
11 : * SPDX-License-Identifier: MIT
12 : ****************************************************************************/
13 :
14 : #include "cpl_port.h"
15 : #include "ogr_geometry.h"
16 :
17 : #include <climits>
18 : #include <cstdarg>
19 : #include <cstddef>
20 : #include <cstdio>
21 : #include <cstdlib>
22 : #include <cstring>
23 : #include <limits>
24 : #include <memory>
25 : #include <stdexcept>
26 : #include <string>
27 :
28 : #include "cpl_conv.h"
29 : #include "cpl_error.h"
30 : #include "cpl_multiproc.h"
31 : #include "cpl_string.h"
32 : #include "ogr_api.h"
33 : #include "ogr_core.h"
34 : #include "ogr_geos.h"
35 : #include "ogr_sfcgal.h"
36 : #include "ogr_libs.h"
37 : #include "ogr_p.h"
38 : #include "ogr_spatialref.h"
39 : #include "ogr_srs_api.h"
40 : #include "ogr_wkb.h"
41 :
42 : #ifndef HAVE_GEOS
43 : #define UNUSED_IF_NO_GEOS CPL_UNUSED
44 : #else
45 : #define UNUSED_IF_NO_GEOS
46 : #endif
47 :
48 : //! @cond Doxygen_Suppress
49 : int OGRGeometry::bGenerate_DB2_V72_BYTE_ORDER = FALSE;
50 : //! @endcond
51 :
52 : #ifdef HAVE_GEOS
53 122 : static void OGRGEOSErrorHandler(const char *fmt, ...)
54 : {
55 : va_list args;
56 :
57 122 : va_start(args, fmt);
58 122 : CPLErrorV(CE_Failure, CPLE_AppDefined, fmt, args);
59 122 : va_end(args);
60 122 : }
61 :
62 91 : static void OGRGEOSWarningHandler(const char *fmt, ...)
63 : {
64 : va_list args;
65 :
66 91 : va_start(args, fmt);
67 91 : CPLErrorV(CE_Warning, CPLE_AppDefined, fmt, args);
68 91 : va_end(args);
69 91 : }
70 : #endif
71 :
72 : /************************************************************************/
73 : /* OGRWktOptions() */
74 : /************************************************************************/
75 :
76 10988 : int OGRWktOptions::getDefaultPrecision()
77 : {
78 10988 : return atoi(CPLGetConfigOption("OGR_WKT_PRECISION", "15"));
79 : }
80 :
81 11084 : bool OGRWktOptions::getDefaultRound()
82 : {
83 11084 : return CPLTestBool(CPLGetConfigOption("OGR_WKT_ROUND", "TRUE"));
84 : }
85 :
86 : /************************************************************************/
87 : /* OGRGeometry() */
88 : /************************************************************************/
89 :
90 : OGRGeometry::OGRGeometry() = default;
91 :
92 : /************************************************************************/
93 : /* OGRGeometry( const OGRGeometry& ) */
94 : /************************************************************************/
95 :
96 : /**
97 : * \brief Copy constructor.
98 : *
99 : * Note: before GDAL 2.1, only the default implementation of the constructor
100 : * existed, which could be unsafe to use.
101 : *
102 : * @since GDAL 2.1
103 : */
104 :
105 2132780 : OGRGeometry::OGRGeometry(const OGRGeometry &other)
106 2132780 : : poSRS(other.poSRS), flags(other.flags)
107 : {
108 2132780 : if (poSRS != nullptr)
109 484776 : const_cast<OGRSpatialReference *>(poSRS)->Reference();
110 2132780 : }
111 :
112 : /************************************************************************/
113 : /* OGRGeometry( OGRGeometry&& ) */
114 : /************************************************************************/
115 :
116 : /**
117 : * \brief Move constructor.
118 : *
119 : * @since GDAL 3.11
120 : */
121 :
122 112371 : OGRGeometry::OGRGeometry(OGRGeometry &&other)
123 112371 : : poSRS(other.poSRS), flags(other.flags)
124 : {
125 112371 : other.poSRS = nullptr;
126 112371 : }
127 :
128 : /************************************************************************/
129 : /* ~OGRGeometry() */
130 : /************************************************************************/
131 :
132 25638000 : OGRGeometry::~OGRGeometry()
133 :
134 : {
135 12819000 : if (poSRS != nullptr)
136 3972490 : const_cast<OGRSpatialReference *>(poSRS)->Release();
137 12819000 : }
138 :
139 : /************************************************************************/
140 : /* operator=( const OGRGeometry&) */
141 : /************************************************************************/
142 :
143 : /**
144 : * \brief Assignment operator.
145 : *
146 : * Note: before GDAL 2.1, only the default implementation of the operator
147 : * existed, which could be unsafe to use.
148 : *
149 : * @since GDAL 2.1
150 : */
151 :
152 1575 : OGRGeometry &OGRGeometry::operator=(const OGRGeometry &other)
153 : {
154 1575 : if (this != &other)
155 : {
156 1575 : empty();
157 1575 : assignSpatialReference(other.getSpatialReference());
158 1575 : flags = other.flags;
159 : }
160 1575 : return *this;
161 : }
162 :
163 : /************************************************************************/
164 : /* operator=( OGRGeometry&&) */
165 : /************************************************************************/
166 :
167 : /**
168 : * \brief Move assignment operator.
169 : *
170 : * @since GDAL 3.11
171 : */
172 :
173 85461 : OGRGeometry &OGRGeometry::operator=(OGRGeometry &&other)
174 : {
175 85461 : if (this != &other)
176 : {
177 85461 : poSRS = other.poSRS;
178 85461 : other.poSRS = nullptr;
179 85461 : flags = other.flags;
180 : }
181 85461 : return *this;
182 : }
183 :
184 : /************************************************************************/
185 : /* dumpReadable() */
186 : /************************************************************************/
187 :
188 : /**
189 : * \brief Dump geometry in well known text format to indicated output file.
190 : *
191 : * A few options can be defined to change the default dump :
192 : * <ul>
193 : * <li>DISPLAY_GEOMETRY=NO : to hide the dump of the geometry</li>
194 : * <li>DISPLAY_GEOMETRY=WKT or YES (default) : dump the geometry as a WKT</li>
195 : * <li>DISPLAY_GEOMETRY=SUMMARY : to get only a summary of the geometry</li>
196 : * </ul>
197 : *
198 : * This method is the same as the C function OGR_G_DumpReadable().
199 : *
200 : * @param fp the text file to write the geometry to.
201 : * @param pszPrefix the prefix to put on each line of output.
202 : * @param papszOptions NULL terminated list of options (may be NULL)
203 : */
204 :
205 0 : void OGRGeometry::dumpReadable(FILE *fp, const char *pszPrefix,
206 : CSLConstList papszOptions) const
207 :
208 : {
209 0 : if (fp == nullptr)
210 0 : fp = stdout;
211 :
212 0 : const auto osStr = dumpReadable(pszPrefix, papszOptions);
213 0 : fprintf(fp, "%s", osStr.c_str());
214 0 : }
215 :
216 : /************************************************************************/
217 : /* dumpReadable() */
218 : /************************************************************************/
219 :
220 : /**
221 : * \brief Dump geometry in well known text format to indicated output file.
222 : *
223 : * A few options can be defined to change the default dump :
224 : * <ul>
225 : * <li>DISPLAY_GEOMETRY=NO : to hide the dump of the geometry</li>
226 : * <li>DISPLAY_GEOMETRY=WKT or YES (default) : dump the geometry as a WKT</li>
227 : * <li>DISPLAY_GEOMETRY=SUMMARY : to get only a summary of the geometry</li>
228 : * <li>XY_COORD_PRECISION=integer: number of decimal figures for X,Y coordinates
229 : * in WKT (added in GDAL 3.9)</li>
230 : * <li>Z_COORD_PRECISION=integer: number of decimal figures for Z coordinates in
231 : * WKT (added in GDAL 3.9)</li>
232 : * </ul>
233 : *
234 : * @param pszPrefix the prefix to put on each line of output.
235 : * @param papszOptions NULL terminated list of options (may be NULL)
236 : * @return a string with the geometry representation.
237 : * @since GDAL 3.7
238 : */
239 :
240 254 : std::string OGRGeometry::dumpReadable(const char *pszPrefix,
241 : CSLConstList papszOptions) const
242 :
243 : {
244 254 : if (pszPrefix == nullptr)
245 254 : pszPrefix = "";
246 :
247 254 : std::string osRet;
248 :
249 : const auto exportToWktWithOpts =
250 1701 : [this, pszPrefix, papszOptions, &osRet](bool bIso)
251 : {
252 243 : OGRErr err(OGRERR_NONE);
253 243 : OGRWktOptions opts;
254 243 : if (const char *pszXYPrecision =
255 243 : CSLFetchNameValue(papszOptions, "XY_COORD_PRECISION"))
256 : {
257 1 : opts.format = OGRWktFormat::F;
258 1 : opts.xyPrecision = atoi(pszXYPrecision);
259 : }
260 243 : if (const char *pszZPrecision =
261 243 : CSLFetchNameValue(papszOptions, "Z_COORD_PRECISION"))
262 : {
263 1 : opts.format = OGRWktFormat::F;
264 1 : opts.zPrecision = atoi(pszZPrecision);
265 : }
266 243 : if (bIso)
267 243 : opts.variant = wkbVariantIso;
268 486 : std::string wkt = exportToWkt(opts, &err);
269 243 : if (err == OGRERR_NONE)
270 : {
271 243 : osRet = pszPrefix;
272 243 : osRet += wkt.data();
273 243 : osRet += '\n';
274 : }
275 243 : };
276 :
277 : const char *pszDisplayGeometry =
278 254 : CSLFetchNameValue(papszOptions, "DISPLAY_GEOMETRY");
279 254 : if (pszDisplayGeometry != nullptr && EQUAL(pszDisplayGeometry, "SUMMARY"))
280 : {
281 11 : osRet += CPLOPrintf("%s%s : ", pszPrefix, getGeometryName());
282 11 : switch (getGeometryType())
283 : {
284 1 : case wkbUnknown:
285 : case wkbNone:
286 : case wkbPoint:
287 : case wkbPoint25D:
288 : case wkbPointM:
289 : case wkbPointZM:
290 1 : break;
291 0 : case wkbPolyhedralSurface:
292 : case wkbTIN:
293 : case wkbPolyhedralSurfaceZ:
294 : case wkbTINZ:
295 : case wkbPolyhedralSurfaceM:
296 : case wkbTINM:
297 : case wkbPolyhedralSurfaceZM:
298 : case wkbTINZM:
299 : {
300 0 : const OGRPolyhedralSurface *poPS = toPolyhedralSurface();
301 : osRet +=
302 0 : CPLOPrintf("%d geometries:\n", poPS->getNumGeometries());
303 0 : for (auto &&poSubGeom : *poPS)
304 : {
305 0 : osRet += pszPrefix;
306 0 : osRet += poSubGeom->dumpReadable(pszPrefix, papszOptions);
307 : }
308 0 : break;
309 : }
310 0 : case wkbLineString:
311 : case wkbLineString25D:
312 : case wkbLineStringM:
313 : case wkbLineStringZM:
314 : case wkbCircularString:
315 : case wkbCircularStringZ:
316 : case wkbCircularStringM:
317 : case wkbCircularStringZM:
318 : {
319 0 : const OGRSimpleCurve *poSC = toSimpleCurve();
320 0 : osRet += CPLOPrintf("%d points\n", poSC->getNumPoints());
321 0 : break;
322 : }
323 10 : case wkbPolygon:
324 : case wkbTriangle:
325 : case wkbTriangleZ:
326 : case wkbTriangleM:
327 : case wkbTriangleZM:
328 : case wkbPolygon25D:
329 : case wkbPolygonM:
330 : case wkbPolygonZM:
331 : case wkbCurvePolygon:
332 : case wkbCurvePolygonZ:
333 : case wkbCurvePolygonM:
334 : case wkbCurvePolygonZM:
335 : {
336 10 : const OGRCurvePolygon *poPoly = toCurvePolygon();
337 10 : const OGRCurve *poRing = poPoly->getExteriorRingCurve();
338 10 : const int nRings = poPoly->getNumInteriorRings();
339 10 : if (poRing == nullptr)
340 : {
341 0 : osRet += "empty";
342 : }
343 : else
344 : {
345 10 : osRet += CPLOPrintf("%d points", poRing->getNumPoints());
346 10 : if (wkbFlatten(poRing->getGeometryType()) ==
347 : wkbCompoundCurve)
348 : {
349 0 : osRet += " (";
350 0 : osRet += poRing->dumpReadable(nullptr, papszOptions);
351 0 : osRet += ")";
352 : }
353 10 : if (nRings)
354 : {
355 0 : osRet += CPLOPrintf(", %d inner rings (", nRings);
356 0 : for (int ir = 0; ir < nRings; ir++)
357 : {
358 0 : poRing = poPoly->getInteriorRingCurve(ir);
359 0 : if (ir)
360 0 : osRet += ", ";
361 : osRet +=
362 0 : CPLOPrintf("%d points", poRing->getNumPoints());
363 0 : if (wkbFlatten(poRing->getGeometryType()) ==
364 : wkbCompoundCurve)
365 : {
366 0 : osRet += " (";
367 : osRet +=
368 0 : poRing->dumpReadable(nullptr, papszOptions);
369 0 : osRet += ")";
370 : }
371 : }
372 0 : osRet += ")";
373 : }
374 : }
375 10 : osRet += "\n";
376 10 : break;
377 : }
378 0 : case wkbCompoundCurve:
379 : case wkbCompoundCurveZ:
380 : case wkbCompoundCurveM:
381 : case wkbCompoundCurveZM:
382 : {
383 0 : const OGRCompoundCurve *poCC = toCompoundCurve();
384 0 : if (poCC->getNumCurves() == 0)
385 : {
386 0 : osRet += "empty";
387 : }
388 : else
389 : {
390 0 : for (int i = 0; i < poCC->getNumCurves(); i++)
391 : {
392 0 : if (i)
393 0 : osRet += ", ";
394 : osRet +=
395 0 : CPLOPrintf("%s (%d points)",
396 0 : poCC->getCurve(i)->getGeometryName(),
397 0 : poCC->getCurve(i)->getNumPoints());
398 : }
399 : }
400 0 : break;
401 : }
402 :
403 0 : case wkbMultiPoint:
404 : case wkbMultiLineString:
405 : case wkbMultiPolygon:
406 : case wkbMultiCurve:
407 : case wkbMultiSurface:
408 : case wkbGeometryCollection:
409 : case wkbMultiPoint25D:
410 : case wkbMultiLineString25D:
411 : case wkbMultiPolygon25D:
412 : case wkbMultiCurveZ:
413 : case wkbMultiSurfaceZ:
414 : case wkbGeometryCollection25D:
415 : case wkbMultiPointM:
416 : case wkbMultiLineStringM:
417 : case wkbMultiPolygonM:
418 : case wkbMultiCurveM:
419 : case wkbMultiSurfaceM:
420 : case wkbGeometryCollectionM:
421 : case wkbMultiPointZM:
422 : case wkbMultiLineStringZM:
423 : case wkbMultiPolygonZM:
424 : case wkbMultiCurveZM:
425 : case wkbMultiSurfaceZM:
426 : case wkbGeometryCollectionZM:
427 : {
428 0 : const OGRGeometryCollection *poColl = toGeometryCollection();
429 : osRet +=
430 0 : CPLOPrintf("%d geometries:\n", poColl->getNumGeometries());
431 0 : for (auto &&poSubGeom : *poColl)
432 : {
433 0 : osRet += pszPrefix;
434 0 : osRet += poSubGeom->dumpReadable(pszPrefix, papszOptions);
435 : }
436 0 : break;
437 : }
438 0 : case wkbLinearRing:
439 : case wkbCurve:
440 : case wkbSurface:
441 : case wkbCurveZ:
442 : case wkbSurfaceZ:
443 : case wkbCurveM:
444 : case wkbSurfaceM:
445 : case wkbCurveZM:
446 : case wkbSurfaceZM:
447 0 : break;
448 11 : }
449 : }
450 243 : else if (pszDisplayGeometry != nullptr && EQUAL(pszDisplayGeometry, "WKT"))
451 : {
452 0 : exportToWktWithOpts(/* bIso=*/false);
453 : }
454 243 : else if (pszDisplayGeometry == nullptr || CPLTestBool(pszDisplayGeometry) ||
455 0 : EQUAL(pszDisplayGeometry, "ISO_WKT"))
456 : {
457 243 : exportToWktWithOpts(/* bIso=*/true);
458 : }
459 :
460 508 : return osRet;
461 : }
462 :
463 : /************************************************************************/
464 : /* OGR_G_DumpReadable() */
465 : /************************************************************************/
466 : /**
467 : * \brief Dump geometry in well known text format to indicated output file.
468 : *
469 : * This method is the same as the CPP method OGRGeometry::dumpReadable.
470 : *
471 : * @param hGeom handle on the geometry to dump.
472 : * @param fp the text file to write the geometry to.
473 : * @param pszPrefix the prefix to put on each line of output.
474 : */
475 :
476 0 : void OGR_G_DumpReadable(OGRGeometryH hGeom, FILE *fp, const char *pszPrefix)
477 :
478 : {
479 0 : VALIDATE_POINTER0(hGeom, "OGR_G_DumpReadable");
480 :
481 0 : OGRGeometry::FromHandle(hGeom)->dumpReadable(fp, pszPrefix);
482 : }
483 :
484 : /************************************************************************/
485 : /* assignSpatialReference() */
486 : /************************************************************************/
487 :
488 : /**
489 : * \brief Assign spatial reference to this object.
490 : *
491 : * Any existing spatial reference
492 : * is replaced, but under no circumstances does this result in the object
493 : * being reprojected. It is just changing the interpretation of the existing
494 : * geometry. Note that assigning a spatial reference increments the
495 : * reference count on the OGRSpatialReference, but does not copy it.
496 : *
497 : * Starting with GDAL 2.3, this will also assign the spatial reference to
498 : * potential sub-geometries of the geometry (OGRGeometryCollection,
499 : * OGRCurvePolygon/OGRPolygon, OGRCompoundCurve, OGRPolyhedralSurface and their
500 : * derived classes).
501 : *
502 : * This is similar to the SFCOM IGeometry::put_SpatialReference() method.
503 : *
504 : * This method is the same as the C function OGR_G_AssignSpatialReference().
505 : *
506 : * @param poSR new spatial reference system to apply.
507 : */
508 :
509 5491840 : void OGRGeometry::assignSpatialReference(const OGRSpatialReference *poSR)
510 :
511 : {
512 : // Do in that order to properly handle poSR == poSRS
513 5491840 : if (poSR != nullptr)
514 3512590 : const_cast<OGRSpatialReference *>(poSR)->Reference();
515 5491840 : if (poSRS != nullptr)
516 24879 : const_cast<OGRSpatialReference *>(poSRS)->Release();
517 :
518 5491840 : poSRS = poSR;
519 5491840 : }
520 :
521 : /************************************************************************/
522 : /* OGR_G_AssignSpatialReference() */
523 : /************************************************************************/
524 : /**
525 : * \brief Assign spatial reference to this object.
526 : *
527 : * Any existing spatial reference
528 : * is replaced, but under no circumstances does this result in the object
529 : * being reprojected. It is just changing the interpretation of the existing
530 : * geometry. Note that assigning a spatial reference increments the
531 : * reference count on the OGRSpatialReference, but does not copy it.
532 : *
533 : * Starting with GDAL 2.3, this will also assign the spatial reference to
534 : * potential sub-geometries of the geometry (OGRGeometryCollection,
535 : * OGRCurvePolygon/OGRPolygon, OGRCompoundCurve, OGRPolyhedralSurface and their
536 : * derived classes).
537 : *
538 : * This is similar to the SFCOM IGeometry::put_SpatialReference() method.
539 : *
540 : * This function is the same as the CPP method
541 : * OGRGeometry::assignSpatialReference.
542 : *
543 : * @param hGeom handle on the geometry to apply the new spatial reference
544 : * system.
545 : * @param hSRS handle on the new spatial reference system to apply.
546 : */
547 :
548 146 : void OGR_G_AssignSpatialReference(OGRGeometryH hGeom, OGRSpatialReferenceH hSRS)
549 :
550 : {
551 146 : VALIDATE_POINTER0(hGeom, "OGR_G_AssignSpatialReference");
552 :
553 292 : OGRGeometry::FromHandle(hGeom)->assignSpatialReference(
554 146 : OGRSpatialReference::FromHandle(hSRS));
555 : }
556 :
557 : /************************************************************************/
558 : /* Intersects() */
559 : /************************************************************************/
560 :
561 : /**
562 : * \brief Do these features intersect?
563 : *
564 : * Determines whether two geometries intersect. If GEOS is enabled, then
565 : * this is done in rigorous fashion otherwise TRUE is returned if the
566 : * envelopes (bounding boxes) of the two geometries overlap.
567 : *
568 : * The poOtherGeom argument may be safely NULL, but in this case the method
569 : * will always return TRUE. That is, a NULL geometry is treated as being
570 : * everywhere.
571 : *
572 : * This method is the same as the C function OGR_G_Intersects().
573 : *
574 : * @param poOtherGeom the other geometry to test against.
575 : *
576 : * @return TRUE if the geometries intersect, otherwise FALSE.
577 : */
578 :
579 35 : OGRBoolean OGRGeometry::Intersects(const OGRGeometry *poOtherGeom) const
580 :
581 : {
582 35 : if (poOtherGeom == nullptr)
583 0 : return TRUE;
584 :
585 35 : OGREnvelope oEnv1;
586 35 : getEnvelope(&oEnv1);
587 :
588 35 : OGREnvelope oEnv2;
589 35 : poOtherGeom->getEnvelope(&oEnv2);
590 :
591 35 : if (oEnv1.MaxX < oEnv2.MinX || oEnv1.MaxY < oEnv2.MinY ||
592 19 : oEnv2.MaxX < oEnv1.MinX || oEnv2.MaxY < oEnv1.MinY)
593 16 : return FALSE;
594 :
595 : #ifndef HAVE_GEOS
596 : // Without GEOS we assume that envelope overlap is equivalent to
597 : // actual intersection.
598 : return TRUE;
599 : #else
600 :
601 19 : GEOSContextHandle_t hGEOSCtxt = createGEOSContext();
602 19 : GEOSGeom hThisGeosGeom = exportToGEOS(hGEOSCtxt);
603 19 : GEOSGeom hOtherGeosGeom = poOtherGeom->exportToGEOS(hGEOSCtxt);
604 :
605 19 : OGRBoolean bResult = FALSE;
606 19 : if (hThisGeosGeom != nullptr && hOtherGeosGeom != nullptr)
607 : {
608 19 : bResult =
609 19 : GEOSIntersects_r(hGEOSCtxt, hThisGeosGeom, hOtherGeosGeom) != 0;
610 : }
611 :
612 19 : GEOSGeom_destroy_r(hGEOSCtxt, hThisGeosGeom);
613 19 : GEOSGeom_destroy_r(hGEOSCtxt, hOtherGeosGeom);
614 19 : freeGEOSContext(hGEOSCtxt);
615 :
616 19 : return bResult;
617 : #endif // HAVE_GEOS
618 : }
619 :
620 : // Old API compatibility function.
621 :
622 : //! @cond Doxygen_Suppress
623 0 : OGRBoolean OGRGeometry::Intersect(OGRGeometry *poOtherGeom) const
624 :
625 : {
626 0 : return Intersects(poOtherGeom);
627 : }
628 :
629 : //! @endcond
630 :
631 : /************************************************************************/
632 : /* OGR_G_Intersects() */
633 : /************************************************************************/
634 : /**
635 : * \brief Do these features intersect?
636 : *
637 : * Determines whether two geometries intersect. If GEOS is enabled, then
638 : * this is done in rigorous fashion otherwise TRUE is returned if the
639 : * envelopes (bounding boxes) of the two geometries overlap.
640 : *
641 : * This function is the same as the CPP method OGRGeometry::Intersects.
642 : *
643 : * @param hGeom handle on the first geometry.
644 : * @param hOtherGeom handle on the other geometry to test against.
645 : *
646 : * @return TRUE if the geometries intersect, otherwise FALSE.
647 : */
648 :
649 11 : int OGR_G_Intersects(OGRGeometryH hGeom, OGRGeometryH hOtherGeom)
650 :
651 : {
652 11 : VALIDATE_POINTER1(hGeom, "OGR_G_Intersects", FALSE);
653 11 : VALIDATE_POINTER1(hOtherGeom, "OGR_G_Intersects", FALSE);
654 :
655 22 : return OGRGeometry::FromHandle(hGeom)->Intersects(
656 11 : OGRGeometry::FromHandle(hOtherGeom));
657 : }
658 :
659 : //! @cond Doxygen_Suppress
660 0 : int OGR_G_Intersect(OGRGeometryH hGeom, OGRGeometryH hOtherGeom)
661 :
662 : {
663 0 : VALIDATE_POINTER1(hGeom, "OGR_G_Intersect", FALSE);
664 0 : VALIDATE_POINTER1(hOtherGeom, "OGR_G_Intersect", FALSE);
665 :
666 0 : return OGRGeometry::FromHandle(hGeom)->Intersects(
667 0 : OGRGeometry::FromHandle(hOtherGeom));
668 : }
669 :
670 : //! @endcond
671 :
672 : /************************************************************************/
673 : /* transformTo() */
674 : /************************************************************************/
675 :
676 : /**
677 : * \brief Transform geometry to new spatial reference system.
678 : *
679 : * This method will transform the coordinates of a geometry from
680 : * their current spatial reference system to a new target spatial
681 : * reference system. Normally this means reprojecting the vectors,
682 : * but it could include datum shifts, and changes of units.
683 : *
684 : * This method will only work if the geometry already has an assigned
685 : * spatial reference system, and if it is transformable to the target
686 : * coordinate system.
687 : *
688 : * Because this method requires internal creation and initialization of an
689 : * OGRCoordinateTransformation object it is significantly more expensive to
690 : * use this method to transform many geometries than it is to create the
691 : * OGRCoordinateTransformation in advance, and call transform() with that
692 : * transformation. This method exists primarily for convenience when only
693 : * transforming a single geometry.
694 : *
695 : * This method is the same as the C function OGR_G_TransformTo().
696 : *
697 : * @param poSR spatial reference system to transform to.
698 : *
699 : * @return OGRERR_NONE on success, or an error code.
700 : */
701 :
702 19 : OGRErr OGRGeometry::transformTo(const OGRSpatialReference *poSR)
703 :
704 : {
705 19 : if (getSpatialReference() == nullptr)
706 : {
707 1 : CPLError(CE_Failure, CPLE_AppDefined, "Geometry has no SRS");
708 1 : return OGRERR_FAILURE;
709 : }
710 :
711 18 : if (poSR == nullptr)
712 : {
713 0 : CPLError(CE_Failure, CPLE_AppDefined, "Target SRS is NULL");
714 0 : return OGRERR_FAILURE;
715 : }
716 :
717 : OGRCoordinateTransformation *poCT =
718 18 : OGRCreateCoordinateTransformation(getSpatialReference(), poSR);
719 18 : if (poCT == nullptr)
720 0 : return OGRERR_FAILURE;
721 :
722 18 : const OGRErr eErr = transform(poCT);
723 :
724 18 : delete poCT;
725 :
726 18 : return eErr;
727 : }
728 :
729 : /************************************************************************/
730 : /* OGR_G_TransformTo() */
731 : /************************************************************************/
732 : /**
733 : * \brief Transform geometry to new spatial reference system.
734 : *
735 : * This function will transform the coordinates of a geometry from
736 : * their current spatial reference system to a new target spatial
737 : * reference system. Normally this means reprojecting the vectors,
738 : * but it could include datum shifts, and changes of units.
739 : *
740 : * This function will only work if the geometry already has an assigned
741 : * spatial reference system, and if it is transformable to the target
742 : * coordinate system.
743 : *
744 : * Because this function requires internal creation and initialization of an
745 : * OGRCoordinateTransformation object it is significantly more expensive to
746 : * use this function to transform many geometries than it is to create the
747 : * OGRCoordinateTransformation in advance, and call transform() with that
748 : * transformation. This function exists primarily for convenience when only
749 : * transforming a single geometry.
750 : *
751 : * This function is the same as the CPP method OGRGeometry::transformTo.
752 : *
753 : * @param hGeom handle on the geometry to apply the transform to.
754 : * @param hSRS handle on the spatial reference system to apply.
755 : *
756 : * @return OGRERR_NONE on success, or an error code.
757 : */
758 :
759 9 : OGRErr OGR_G_TransformTo(OGRGeometryH hGeom, OGRSpatialReferenceH hSRS)
760 :
761 : {
762 9 : VALIDATE_POINTER1(hGeom, "OGR_G_TransformTo", OGRERR_FAILURE);
763 :
764 18 : return OGRGeometry::FromHandle(hGeom)->transformTo(
765 18 : OGRSpatialReference::FromHandle(hSRS));
766 : }
767 :
768 : /**
769 : * \fn OGRErr OGRGeometry::transform( OGRCoordinateTransformation *poCT );
770 : *
771 : * \brief Apply arbitrary coordinate transformation to geometry.
772 : *
773 : * This method will transform the coordinates of a geometry from
774 : * their current spatial reference system to a new target spatial
775 : * reference system. Normally this means reprojecting the vectors,
776 : * but it could include datum shifts, and changes of units.
777 : *
778 : * Note that this method does not require that the geometry already
779 : * have a spatial reference system. It will be assumed that they can
780 : * be treated as having the source spatial reference system of the
781 : * OGRCoordinateTransformation object, and the actual SRS of the geometry
782 : * will be ignored. On successful completion the output OGRSpatialReference
783 : * of the OGRCoordinateTransformation will be assigned to the geometry.
784 : *
785 : * This method only does reprojection on a point-by-point basis. It does not
786 : * include advanced logic to deal with discontinuities at poles or antimeridian.
787 : * For that, use the OGRGeometryFactory::transformWithOptions() method.
788 : *
789 : * This method is the same as the C function OGR_G_Transform().
790 : *
791 : * @param poCT the transformation to apply.
792 : *
793 : * @return OGRERR_NONE on success or an error code.
794 : */
795 :
796 : /************************************************************************/
797 : /* OGR_G_Transform() */
798 : /************************************************************************/
799 : /**
800 : * \brief Apply arbitrary coordinate transformation to geometry.
801 : *
802 : * This function will transform the coordinates of a geometry from
803 : * their current spatial reference system to a new target spatial
804 : * reference system. Normally this means reprojecting the vectors,
805 : * but it could include datum shifts, and changes of units.
806 : *
807 : * Note that this function does not require that the geometry already
808 : * have a spatial reference system. It will be assumed that they can
809 : * be treated as having the source spatial reference system of the
810 : * OGRCoordinateTransformation object, and the actual SRS of the geometry
811 : * will be ignored. On successful completion the output OGRSpatialReference
812 : * of the OGRCoordinateTransformation will be assigned to the geometry.
813 : *
814 : * This function only does reprojection on a point-by-point basis. It does not
815 : * include advanced logic to deal with discontinuities at poles or antimeridian.
816 : * For that, use the OGR_GeomTransformer_Create() and
817 : * OGR_GeomTransformer_Transform() functions.
818 : *
819 : * This function is the same as the CPP method OGRGeometry::transform.
820 : *
821 : * @param hGeom handle on the geometry to apply the transform to.
822 : * @param hTransform handle on the transformation to apply.
823 : *
824 : * @return OGRERR_NONE on success or an error code.
825 : */
826 :
827 17 : OGRErr OGR_G_Transform(OGRGeometryH hGeom,
828 : OGRCoordinateTransformationH hTransform)
829 :
830 : {
831 17 : VALIDATE_POINTER1(hGeom, "OGR_G_Transform", OGRERR_FAILURE);
832 :
833 34 : return OGRGeometry::FromHandle(hGeom)->transform(
834 17 : OGRCoordinateTransformation::FromHandle(hTransform));
835 : }
836 :
837 : /**
838 : * \fn int OGRGeometry::getDimension() const;
839 : *
840 : * \brief Get the dimension of this object.
841 : *
842 : * This method corresponds to the SFCOM IGeometry::GetDimension() method.
843 : * It indicates the dimension of the object, but does not indicate the
844 : * dimension of the underlying space (as indicated by
845 : * OGRGeometry::getCoordinateDimension()).
846 : *
847 : * This method is the same as the C function OGR_G_GetDimension().
848 : *
849 : * @return 0 for points, 1 for lines and 2 for surfaces.
850 : */
851 :
852 : /**
853 : * \brief Get the geometry type that conforms with ISO SQL/MM Part3
854 : *
855 : * @return the geometry type that conforms with ISO SQL/MM Part3
856 : */
857 401254 : OGRwkbGeometryType OGRGeometry::getIsoGeometryType() const
858 : {
859 401254 : OGRwkbGeometryType nGType = wkbFlatten(getGeometryType());
860 :
861 401254 : if (flags & OGR_G_3D)
862 137131 : nGType = static_cast<OGRwkbGeometryType>(nGType + 1000);
863 401254 : if (flags & OGR_G_MEASURED)
864 25512 : nGType = static_cast<OGRwkbGeometryType>(nGType + 2000);
865 :
866 401254 : return nGType;
867 : }
868 :
869 : /************************************************************************/
870 : /* OGRGeometry::segmentize() */
871 : /************************************************************************/
872 : /**
873 : *
874 : * \brief Modify the geometry such it has no segment longer then the
875 : * given distance.
876 : *
877 : * This method modifies the geometry to add intermediate vertices if necessary
878 : * so that the maximum length between 2 consecutive vertices is lower than
879 : * dfMaxLength.
880 : *
881 : * Interpolated points will have Z and M values (if needed) set to 0.
882 : * Distance computation is performed in 2d only
883 : *
884 : * This function is the same as the C function OGR_G_Segmentize()
885 : *
886 : * @param dfMaxLength the maximum distance between 2 points after segmentization
887 : * @return (since 3.10) true in case of success, false in case of error.
888 : */
889 :
890 0 : bool OGRGeometry::segmentize(CPL_UNUSED double dfMaxLength)
891 : {
892 : // Do nothing.
893 0 : return true;
894 : }
895 :
896 : /************************************************************************/
897 : /* OGR_G_Segmentize() */
898 : /************************************************************************/
899 :
900 : /**
901 : *
902 : * \brief Modify the geometry such it has no segment longer then the given
903 : * distance.
904 : *
905 : * Interpolated points will have Z and M values (if needed) set to 0.
906 : * Distance computation is performed in 2d only.
907 : *
908 : * This function is the same as the CPP method OGRGeometry::segmentize().
909 : *
910 : * @param hGeom handle on the geometry to segmentize
911 : * @param dfMaxLength the maximum distance between 2 points after segmentization
912 : */
913 :
914 21 : void CPL_DLL OGR_G_Segmentize(OGRGeometryH hGeom, double dfMaxLength)
915 : {
916 21 : VALIDATE_POINTER0(hGeom, "OGR_G_Segmentize");
917 :
918 21 : if (dfMaxLength <= 0)
919 : {
920 0 : CPLError(CE_Failure, CPLE_AppDefined,
921 : "dfMaxLength must be strictly positive");
922 0 : return;
923 : }
924 21 : OGRGeometry::FromHandle(hGeom)->segmentize(dfMaxLength);
925 : }
926 :
927 : /************************************************************************/
928 : /* OGR_G_GetDimension() */
929 : /************************************************************************/
930 : /**
931 : *
932 : * \brief Get the dimension of this geometry.
933 : *
934 : * This function corresponds to the SFCOM IGeometry::GetDimension() method.
935 : * It indicates the dimension of the geometry, but does not indicate the
936 : * dimension of the underlying space (as indicated by
937 : * OGR_G_GetCoordinateDimension() function).
938 : *
939 : * This function is the same as the CPP method OGRGeometry::getDimension().
940 : *
941 : * @param hGeom handle on the geometry to get the dimension from.
942 : * @return 0 for points, 1 for lines and 2 for surfaces.
943 : */
944 :
945 21 : int OGR_G_GetDimension(OGRGeometryH hGeom)
946 :
947 : {
948 21 : VALIDATE_POINTER1(hGeom, "OGR_G_GetDimension", 0);
949 :
950 21 : return OGRGeometry::FromHandle(hGeom)->getDimension();
951 : }
952 :
953 : /************************************************************************/
954 : /* getCoordinateDimension() */
955 : /************************************************************************/
956 : /**
957 : * \brief Get the dimension of the coordinates in this object.
958 : *
959 : * This method is the same as the C function OGR_G_GetCoordinateDimension().
960 : *
961 : * @deprecated use CoordinateDimension().
962 : *
963 : * @return this will return 2 or 3.
964 : */
965 :
966 1062060 : int OGRGeometry::getCoordinateDimension() const
967 :
968 : {
969 1062060 : return (flags & OGR_G_3D) ? 3 : 2;
970 : }
971 :
972 : /************************************************************************/
973 : /* CoordinateDimension() */
974 : /************************************************************************/
975 : /**
976 : * \brief Get the dimension of the coordinates in this object.
977 : *
978 : * This method is the same as the C function OGR_G_CoordinateDimension().
979 : *
980 : * @return this will return 2 for XY, 3 for XYZ and XYM, and 4 for XYZM data.
981 : *
982 : * @since GDAL 2.1
983 : */
984 :
985 99239 : int OGRGeometry::CoordinateDimension() const
986 :
987 : {
988 99239 : if ((flags & OGR_G_3D) && (flags & OGR_G_MEASURED))
989 7221 : return 4;
990 92018 : else if ((flags & OGR_G_3D) || (flags & OGR_G_MEASURED))
991 6304 : return 3;
992 : else
993 85714 : return 2;
994 : }
995 :
996 : /************************************************************************/
997 : /* OGR_G_GetCoordinateDimension() */
998 : /************************************************************************/
999 : /**
1000 : *
1001 : * \brief Get the dimension of the coordinates in this geometry.
1002 : *
1003 : * This function is the same as the CPP method
1004 : * OGRGeometry::getCoordinateDimension().
1005 : *
1006 : * @param hGeom handle on the geometry to get the dimension of the
1007 : * coordinates from.
1008 : *
1009 : * @deprecated use OGR_G_CoordinateDimension(), OGR_G_Is3D(), or
1010 : * OGR_G_IsMeasured().
1011 : *
1012 : * @return this will return 2 or 3.
1013 : */
1014 :
1015 872 : int OGR_G_GetCoordinateDimension(OGRGeometryH hGeom)
1016 :
1017 : {
1018 872 : VALIDATE_POINTER1(hGeom, "OGR_G_GetCoordinateDimension", 0);
1019 :
1020 872 : return OGRGeometry::FromHandle(hGeom)->getCoordinateDimension();
1021 : }
1022 :
1023 : /************************************************************************/
1024 : /* OGR_G_CoordinateDimension() */
1025 : /************************************************************************/
1026 : /**
1027 : *
1028 : * \brief Get the dimension of the coordinates in this geometry.
1029 : *
1030 : * This function is the same as the CPP method
1031 : * OGRGeometry::CoordinateDimension().
1032 : *
1033 : * @param hGeom handle on the geometry to get the dimension of the
1034 : * coordinates from.
1035 : *
1036 : * @return this will return 2 for XY, 3 for XYZ and XYM, and 4 for XYZM data.
1037 : *
1038 : * @since GDAL 2.1
1039 : */
1040 :
1041 4 : int OGR_G_CoordinateDimension(OGRGeometryH hGeom)
1042 :
1043 : {
1044 4 : VALIDATE_POINTER1(hGeom, "OGR_G_CoordinateDimension", 0);
1045 :
1046 4 : return OGRGeometry::FromHandle(hGeom)->CoordinateDimension();
1047 : }
1048 :
1049 : /**
1050 : *
1051 : * \brief See whether this geometry has Z coordinates.
1052 : *
1053 : * This function is the same as the CPP method
1054 : * OGRGeometry::Is3D().
1055 : *
1056 : * @param hGeom handle on the geometry to check whether it has Z coordinates.
1057 : *
1058 : * @return TRUE if the geometry has Z coordinates.
1059 : * @since GDAL 2.1
1060 : */
1061 :
1062 34497 : int OGR_G_Is3D(OGRGeometryH hGeom)
1063 :
1064 : {
1065 34497 : VALIDATE_POINTER1(hGeom, "OGR_G_Is3D", 0);
1066 :
1067 34497 : return OGRGeometry::FromHandle(hGeom)->Is3D();
1068 : }
1069 :
1070 : /**
1071 : *
1072 : * \brief See whether this geometry is measured.
1073 : *
1074 : * This function is the same as the CPP method
1075 : * OGRGeometry::IsMeasured().
1076 : *
1077 : * @param hGeom handle on the geometry to check whether it is measured.
1078 : *
1079 : * @return TRUE if the geometry has M coordinates.
1080 : * @since GDAL 2.1
1081 : */
1082 :
1083 37412 : int OGR_G_IsMeasured(OGRGeometryH hGeom)
1084 :
1085 : {
1086 37412 : VALIDATE_POINTER1(hGeom, "OGR_G_IsMeasured", 0);
1087 :
1088 37412 : return OGRGeometry::FromHandle(hGeom)->IsMeasured();
1089 : }
1090 :
1091 : /************************************************************************/
1092 : /* setCoordinateDimension() */
1093 : /************************************************************************/
1094 :
1095 : /**
1096 : * \brief Set the coordinate dimension.
1097 : *
1098 : * This method sets the explicit coordinate dimension. Setting the coordinate
1099 : * dimension of a geometry to 2 should zero out any existing Z values. Setting
1100 : * the dimension of a geometry collection, a compound curve, a polygon, etc.
1101 : * will affect the children geometries.
1102 : * This will also remove the M dimension if present before this call.
1103 : *
1104 : * @deprecated use set3D() or setMeasured().
1105 : *
1106 : * @param nNewDimension New coordinate dimension value, either 2 or 3.
1107 : * @return (since 3.10) true in case of success, false in case of memory allocation error
1108 : */
1109 :
1110 63414 : bool OGRGeometry::setCoordinateDimension(int nNewDimension)
1111 :
1112 : {
1113 63414 : if (nNewDimension == 2)
1114 62946 : flags &= ~OGR_G_3D;
1115 : else
1116 468 : flags |= OGR_G_3D;
1117 63414 : return setMeasured(FALSE);
1118 : }
1119 :
1120 : /**
1121 : * \brief Add or remove the Z coordinate dimension.
1122 : *
1123 : * This method adds or removes the explicit Z coordinate dimension.
1124 : * Removing the Z coordinate dimension of a geometry will remove any
1125 : * existing Z values. Adding the Z dimension to a geometry
1126 : * collection, a compound curve, a polygon, etc. will affect the
1127 : * children geometries.
1128 : *
1129 : * @param bIs3D Should the geometry have a Z dimension, either TRUE or FALSE.
1130 : * @return (since 3.10) true in case of success, false in case of memory allocation error
1131 : * @since GDAL 2.1
1132 : */
1133 :
1134 1581610 : bool OGRGeometry::set3D(OGRBoolean bIs3D)
1135 :
1136 : {
1137 1581610 : if (bIs3D)
1138 1577000 : flags |= OGR_G_3D;
1139 : else
1140 4611 : flags &= ~OGR_G_3D;
1141 1581610 : return true;
1142 : }
1143 :
1144 : /**
1145 : * \brief Add or remove the M coordinate dimension.
1146 : *
1147 : * This method adds or removes the explicit M coordinate dimension.
1148 : * Removing the M coordinate dimension of a geometry will remove any
1149 : * existing M values. Adding the M dimension to a geometry
1150 : * collection, a compound curve, a polygon, etc. will affect the
1151 : * children geometries.
1152 : *
1153 : * @param bIsMeasured Should the geometry have a M dimension, either
1154 : * TRUE or FALSE.
1155 : * @return (since 3.10) true in case of success, false in case of memory allocation error
1156 : * @since GDAL 2.1
1157 : */
1158 :
1159 407773 : bool OGRGeometry::setMeasured(OGRBoolean bIsMeasured)
1160 :
1161 : {
1162 407773 : if (bIsMeasured)
1163 137604 : flags |= OGR_G_MEASURED;
1164 : else
1165 270169 : flags &= ~OGR_G_MEASURED;
1166 407773 : return true;
1167 : }
1168 :
1169 : /************************************************************************/
1170 : /* OGR_G_SetCoordinateDimension() */
1171 : /************************************************************************/
1172 :
1173 : /**
1174 : * \brief Set the coordinate dimension.
1175 : *
1176 : * This method sets the explicit coordinate dimension. Setting the coordinate
1177 : * dimension of a geometry to 2 should zero out any existing Z values. Setting
1178 : * the dimension of a geometry collection, a compound curve, a polygon, etc.
1179 : * will affect the children geometries.
1180 : * This will also remove the M dimension if present before this call.
1181 : *
1182 : * @deprecated use OGR_G_Set3D() or OGR_G_SetMeasured().
1183 : *
1184 : * @param hGeom handle on the geometry to set the dimension of the
1185 : * coordinates.
1186 : * @param nNewDimension New coordinate dimension value, either 2 or 3.
1187 : */
1188 :
1189 131 : void OGR_G_SetCoordinateDimension(OGRGeometryH hGeom, int nNewDimension)
1190 :
1191 : {
1192 131 : VALIDATE_POINTER0(hGeom, "OGR_G_SetCoordinateDimension");
1193 :
1194 131 : OGRGeometry::FromHandle(hGeom)->setCoordinateDimension(nNewDimension);
1195 : }
1196 :
1197 : /************************************************************************/
1198 : /* OGR_G_Set3D() */
1199 : /************************************************************************/
1200 :
1201 : /**
1202 : * \brief Add or remove the Z coordinate dimension.
1203 : *
1204 : * This method adds or removes the explicit Z coordinate dimension.
1205 : * Removing the Z coordinate dimension of a geometry will remove any
1206 : * existing Z values. Adding the Z dimension to a geometry
1207 : * collection, a compound curve, a polygon, etc. will affect the
1208 : * children geometries.
1209 : *
1210 : * @param hGeom handle on the geometry to set or unset the Z dimension.
1211 : * @param bIs3D Should the geometry have a Z dimension, either TRUE or FALSE.
1212 : * @since GDAL 2.1
1213 : */
1214 :
1215 154 : void OGR_G_Set3D(OGRGeometryH hGeom, int bIs3D)
1216 :
1217 : {
1218 154 : VALIDATE_POINTER0(hGeom, "OGR_G_Set3D");
1219 :
1220 154 : OGRGeometry::FromHandle(hGeom)->set3D(bIs3D);
1221 : }
1222 :
1223 : /************************************************************************/
1224 : /* OGR_G_SetMeasured() */
1225 : /************************************************************************/
1226 :
1227 : /**
1228 : * \brief Add or remove the M coordinate dimension.
1229 : *
1230 : * This method adds or removes the explicit M coordinate dimension.
1231 : * Removing the M coordinate dimension of a geometry will remove any
1232 : * existing M values. Adding the M dimension to a geometry
1233 : * collection, a compound curve, a polygon, etc. will affect the
1234 : * children geometries.
1235 : *
1236 : * @param hGeom handle on the geometry to set or unset the M dimension.
1237 : * @param bIsMeasured Should the geometry have a M dimension, either
1238 : * TRUE or FALSE.
1239 : * @since GDAL 2.1
1240 : */
1241 :
1242 154 : void OGR_G_SetMeasured(OGRGeometryH hGeom, int bIsMeasured)
1243 :
1244 : {
1245 154 : VALIDATE_POINTER0(hGeom, "OGR_G_SetMeasured");
1246 :
1247 154 : OGRGeometry::FromHandle(hGeom)->setMeasured(bIsMeasured);
1248 : }
1249 :
1250 : /**
1251 : * \fn int OGRGeometry::Equals( OGRGeometry *poOtherGeom ) const;
1252 : *
1253 : * \brief Returns TRUE if two geometries are equivalent.
1254 : *
1255 : * This operation implements the SQL/MM ST_OrderingEquals() operation.
1256 : *
1257 : * The comparison is done in a structural way, that is to say that the geometry
1258 : * types must be identical, as well as the number and ordering of sub-geometries
1259 : * and vertices.
1260 : * Or equivalently, two geometries are considered equal by this method if their
1261 : * WKT/WKB representation is equal.
1262 : * Note: this must be distinguished for equality in a spatial way (which is
1263 : * the purpose of the ST_Equals() operation).
1264 : *
1265 : * This method is the same as the C function OGR_G_Equals().
1266 : *
1267 : * @return TRUE if equivalent or FALSE otherwise.
1268 : */
1269 :
1270 : // Backward compatibility method.
1271 :
1272 : //! @cond Doxygen_Suppress
1273 0 : int OGRGeometry::Equal(OGRGeometry *poOtherGeom) const
1274 : {
1275 0 : return Equals(poOtherGeom);
1276 : }
1277 :
1278 : //! @endcond
1279 :
1280 : /************************************************************************/
1281 : /* OGR_G_Equals() */
1282 : /************************************************************************/
1283 :
1284 : /**
1285 : * \brief Returns TRUE if two geometries are equivalent.
1286 : *
1287 : * This operation implements the SQL/MM ST_OrderingEquals() operation.
1288 : *
1289 : * The comparison is done in a structural way, that is to say that the geometry
1290 : * types must be identical, as well as the number and ordering of sub-geometries
1291 : * and vertices.
1292 : * Or equivalently, two geometries are considered equal by this method if their
1293 : * WKT/WKB representation is equal.
1294 : * Note: this must be distinguished for equality in a spatial way (which is
1295 : * the purpose of the ST_Equals() operation).
1296 : *
1297 : * This function is the same as the CPP method OGRGeometry::Equals() method.
1298 : *
1299 : * @param hGeom handle on the first geometry.
1300 : * @param hOther handle on the other geometry to test against.
1301 : * @return TRUE if equivalent or FALSE otherwise.
1302 : */
1303 :
1304 28206 : int OGR_G_Equals(OGRGeometryH hGeom, OGRGeometryH hOther)
1305 :
1306 : {
1307 28206 : VALIDATE_POINTER1(hGeom, "OGR_G_Equals", FALSE);
1308 :
1309 28206 : if (hOther == nullptr)
1310 : {
1311 0 : CPLError(CE_Failure, CPLE_ObjectNull,
1312 : "hOther was NULL in OGR_G_Equals");
1313 0 : return 0;
1314 : }
1315 :
1316 56412 : return OGRGeometry::FromHandle(hGeom)->Equals(
1317 28206 : OGRGeometry::FromHandle(hOther));
1318 : }
1319 :
1320 : //! @cond Doxygen_Suppress
1321 0 : int OGR_G_Equal(OGRGeometryH hGeom, OGRGeometryH hOther)
1322 :
1323 : {
1324 0 : if (hGeom == nullptr)
1325 : {
1326 0 : CPLError(CE_Failure, CPLE_ObjectNull, "hGeom was NULL in OGR_G_Equal");
1327 0 : return 0;
1328 : }
1329 :
1330 0 : if (hOther == nullptr)
1331 : {
1332 0 : CPLError(CE_Failure, CPLE_ObjectNull, "hOther was NULL in OGR_G_Equal");
1333 0 : return 0;
1334 : }
1335 :
1336 0 : return OGRGeometry::FromHandle(hGeom)->Equals(
1337 0 : OGRGeometry::FromHandle(hOther));
1338 : }
1339 :
1340 : //! @endcond
1341 :
1342 : /**
1343 : * \fn int OGRGeometry::WkbSize() const;
1344 : *
1345 : * \brief Returns size of related binary representation.
1346 : *
1347 : * This method returns the exact number of bytes required to hold the
1348 : * well known binary representation of this geometry object. Its computation
1349 : * may be slightly expensive for complex geometries.
1350 : *
1351 : * This method relates to the SFCOM IWks::WkbSize() method.
1352 : *
1353 : * This method is the same as the C function OGR_G_WkbSize().
1354 : *
1355 : * @return size of binary representation in bytes.
1356 : */
1357 :
1358 : /************************************************************************/
1359 : /* OGR_G_WkbSize() */
1360 : /************************************************************************/
1361 : /**
1362 : * \brief Returns size of related binary representation.
1363 : *
1364 : * This function returns the exact number of bytes required to hold the
1365 : * well known binary representation of this geometry object. Its computation
1366 : * may be slightly expensive for complex geometries.
1367 : *
1368 : * This function relates to the SFCOM IWks::WkbSize() method.
1369 : *
1370 : * This function is the same as the CPP method OGRGeometry::WkbSize().
1371 : *
1372 : * Use OGR_G_WkbSizeEx() if called on huge geometries (> 2 GB serialized)
1373 : *
1374 : * @param hGeom handle on the geometry to get the binary size from.
1375 : * @return size of binary representation in bytes.
1376 : */
1377 :
1378 1 : int OGR_G_WkbSize(OGRGeometryH hGeom)
1379 :
1380 : {
1381 1 : VALIDATE_POINTER1(hGeom, "OGR_G_WkbSize", 0);
1382 :
1383 1 : const size_t nSize = OGRGeometry::FromHandle(hGeom)->WkbSize();
1384 1 : if (nSize > static_cast<size_t>(std::numeric_limits<int>::max()))
1385 : {
1386 0 : CPLError(CE_Failure, CPLE_AppDefined,
1387 : "OGR_G_WkbSize() would return a value beyond int range. "
1388 : "Use OGR_G_WkbSizeEx() instead");
1389 0 : return 0;
1390 : }
1391 1 : return static_cast<int>(nSize);
1392 : }
1393 :
1394 : /************************************************************************/
1395 : /* OGR_G_WkbSizeEx() */
1396 : /************************************************************************/
1397 : /**
1398 : * \brief Returns size of related binary representation.
1399 : *
1400 : * This function returns the exact number of bytes required to hold the
1401 : * well known binary representation of this geometry object. Its computation
1402 : * may be slightly expensive for complex geometries.
1403 : *
1404 : * This function relates to the SFCOM IWks::WkbSize() method.
1405 : *
1406 : * This function is the same as the CPP method OGRGeometry::WkbSize().
1407 : *
1408 : * @param hGeom handle on the geometry to get the binary size from.
1409 : * @return size of binary representation in bytes.
1410 : * @since GDAL 3.3
1411 : */
1412 :
1413 10675 : size_t OGR_G_WkbSizeEx(OGRGeometryH hGeom)
1414 :
1415 : {
1416 10675 : VALIDATE_POINTER1(hGeom, "OGR_G_WkbSizeEx", 0);
1417 :
1418 10675 : return OGRGeometry::FromHandle(hGeom)->WkbSize();
1419 : }
1420 :
1421 : /**
1422 : * \fn void OGRGeometry::getEnvelope(OGREnvelope *psEnvelope) const;
1423 : *
1424 : * \brief Computes and returns the bounding envelope for this geometry
1425 : * in the passed psEnvelope structure.
1426 : *
1427 : * This method is the same as the C function OGR_G_GetEnvelope().
1428 : *
1429 : * @param psEnvelope the structure in which to place the results.
1430 : */
1431 :
1432 : /************************************************************************/
1433 : /* OGR_G_GetEnvelope() */
1434 : /************************************************************************/
1435 : /**
1436 : * \brief Computes and returns the bounding envelope for this geometry
1437 : * in the passed psEnvelope structure.
1438 : *
1439 : * This function is the same as the CPP method OGRGeometry::getEnvelope().
1440 : *
1441 : * @param hGeom handle of the geometry to get envelope from.
1442 : * @param psEnvelope the structure in which to place the results.
1443 : */
1444 :
1445 13356 : void OGR_G_GetEnvelope(OGRGeometryH hGeom, OGREnvelope *psEnvelope)
1446 :
1447 : {
1448 13356 : VALIDATE_POINTER0(hGeom, "OGR_G_GetEnvelope");
1449 :
1450 13356 : OGRGeometry::FromHandle(hGeom)->getEnvelope(psEnvelope);
1451 : }
1452 :
1453 : /**
1454 : * \fn void OGRGeometry::getEnvelope(OGREnvelope3D *psEnvelope) const;
1455 : *
1456 : * \brief Computes and returns the bounding envelope (3D) for this
1457 : * geometry in the passed psEnvelope structure.
1458 : *
1459 : * This method is the same as the C function OGR_G_GetEnvelope3D().
1460 : *
1461 : * @param psEnvelope the structure in which to place the results.
1462 : *
1463 : * @since OGR 1.9.0
1464 : */
1465 :
1466 : /************************************************************************/
1467 : /* OGR_G_GetEnvelope3D() */
1468 : /************************************************************************/
1469 : /**
1470 : * \brief Computes and returns the bounding envelope (3D) for this
1471 : * geometry in the passed psEnvelope structure.
1472 : *
1473 : * This function is the same as the CPP method OGRGeometry::getEnvelope().
1474 : *
1475 : * @param hGeom handle of the geometry to get envelope from.
1476 : * @param psEnvelope the structure in which to place the results.
1477 : *
1478 : * @since OGR 1.9.0
1479 : */
1480 :
1481 10 : void OGR_G_GetEnvelope3D(OGRGeometryH hGeom, OGREnvelope3D *psEnvelope)
1482 :
1483 : {
1484 10 : VALIDATE_POINTER0(hGeom, "OGR_G_GetEnvelope3D");
1485 :
1486 10 : OGRGeometry::FromHandle(hGeom)->getEnvelope(psEnvelope);
1487 : }
1488 :
1489 : /************************************************************************/
1490 : /* importFromWkb() */
1491 : /************************************************************************/
1492 :
1493 : /**
1494 : * \brief Assign geometry from well known binary data.
1495 : *
1496 : * The object must have already been instantiated as the correct derived
1497 : * type of geometry object to match the binaries type. This method is used
1498 : * by the OGRGeometryFactory class, but not normally called by application
1499 : * code.
1500 : *
1501 : * This method relates to the SFCOM IWks::ImportFromWKB() method.
1502 : *
1503 : * This method is the same as the C function OGR_G_ImportFromWkb().
1504 : *
1505 : * @param pabyData the binary input data.
1506 : * @param nSize the size of pabyData in bytes, or -1 if not known.
1507 : * @param eWkbVariant if wkbVariantPostGIS1, special interpretation is
1508 : * done for curve geometries code
1509 : *
1510 : * @return OGRERR_NONE if all goes well, otherwise any of
1511 : * OGRERR_NOT_ENOUGH_DATA, OGRERR_UNSUPPORTED_GEOMETRY_TYPE, or
1512 : * OGRERR_CORRUPT_DATA may be returned.
1513 : */
1514 :
1515 492 : OGRErr OGRGeometry::importFromWkb(const GByte *pabyData, size_t nSize,
1516 : OGRwkbVariant eWkbVariant)
1517 : {
1518 492 : size_t nBytesConsumedOutIgnored = 0;
1519 492 : return importFromWkb(pabyData, nSize, eWkbVariant,
1520 984 : nBytesConsumedOutIgnored);
1521 : }
1522 :
1523 : /**
1524 : * \fn OGRErr OGRGeometry::importFromWkb( const unsigned char * pabyData,
1525 : * size_t nSize, OGRwkbVariant eWkbVariant, size_t& nBytesConsumedOut );
1526 : *
1527 : * \brief Assign geometry from well known binary data.
1528 : *
1529 : * The object must have already been instantiated as the correct derived
1530 : * type of geometry object to match the binaries type. This method is used
1531 : * by the OGRGeometryFactory class, but not normally called by application
1532 : * code.
1533 : *
1534 : * This method relates to the SFCOM IWks::ImportFromWKB() method.
1535 : *
1536 : * This method is the same as the C function OGR_G_ImportFromWkb().
1537 : *
1538 : * @param pabyData the binary input data.
1539 : * @param nSize the size of pabyData in bytes, or -1 if not known.
1540 : * @param eWkbVariant if wkbVariantPostGIS1, special interpretation is
1541 : * done for curve geometries code
1542 : * @param nBytesConsumedOut output parameter. Number of bytes consumed.
1543 : *
1544 : * @return OGRERR_NONE if all goes well, otherwise any of
1545 : * OGRERR_NOT_ENOUGH_DATA, OGRERR_UNSUPPORTED_GEOMETRY_TYPE, or
1546 : * OGRERR_CORRUPT_DATA may be returned.
1547 : *
1548 : * @since GDAL 2.3
1549 : */
1550 :
1551 : /************************************************************************/
1552 : /* OGR_G_ImportFromWkb() */
1553 : /************************************************************************/
1554 : /**
1555 : * \brief Assign geometry from well known binary data.
1556 : *
1557 : * The object must have already been instantiated as the correct derived
1558 : * type of geometry object to match the binaries type.
1559 : *
1560 : * This function relates to the SFCOM IWks::ImportFromWKB() method.
1561 : *
1562 : * This function is the same as the CPP method OGRGeometry::importFromWkb().
1563 : *
1564 : * @param hGeom handle on the geometry to assign the well know binary data to.
1565 : * @param pabyData the binary input data.
1566 : * @param nSize the size of pabyData in bytes, or -1 if not known.
1567 : *
1568 : * @return OGRERR_NONE if all goes well, otherwise any of
1569 : * OGRERR_NOT_ENOUGH_DATA, OGRERR_UNSUPPORTED_GEOMETRY_TYPE, or
1570 : * OGRERR_CORRUPT_DATA may be returned.
1571 : */
1572 :
1573 0 : OGRErr OGR_G_ImportFromWkb(OGRGeometryH hGeom, const void *pabyData, int nSize)
1574 :
1575 : {
1576 0 : VALIDATE_POINTER1(hGeom, "OGR_G_ImportFromWkb", OGRERR_FAILURE);
1577 :
1578 0 : return OGRGeometry::FromHandle(hGeom)->importFromWkb(
1579 0 : static_cast<const GByte *>(pabyData), nSize);
1580 : }
1581 :
1582 : /************************************************************************/
1583 : /* OGRGeometry::exportToWkb() */
1584 : /************************************************************************/
1585 :
1586 : /* clang-format off */
1587 : /**
1588 : * \brief Convert a geometry into well known binary format.
1589 : *
1590 : * This method relates to the SFCOM IWks::ExportToWKB() method.
1591 : *
1592 : * This method is the same as the C function OGR_G_ExportToWkb() or
1593 : * OGR_G_ExportToIsoWkb(), depending on the value of eWkbVariant.
1594 : *
1595 : * @param eByteOrder One of wkbXDR or wkbNDR indicating MSB or LSB byte order
1596 : * respectively.
1597 : * @param pabyData a buffer into which the binary representation is
1598 : * written. This buffer must be at least
1599 : * OGRGeometry::WkbSize() byte in size.
1600 : * @param eWkbVariant What standard to use when exporting geometries
1601 : * with three dimensions (or more). The default
1602 : * wkbVariantOldOgc is the historical OGR
1603 : * variant. wkbVariantIso is the variant defined
1604 : * in ISO SQL/MM and adopted by OGC for SFSQL
1605 : * 1.2.
1606 : *
1607 : * @return Currently OGRERR_NONE is always returned.
1608 : */
1609 : /* clang-format on */
1610 251661 : OGRErr OGRGeometry::exportToWkb(OGRwkbByteOrder eByteOrder,
1611 : unsigned char *pabyData,
1612 : OGRwkbVariant eWkbVariant) const
1613 : {
1614 251661 : OGRwkbExportOptions sOptions;
1615 251661 : sOptions.eByteOrder = eByteOrder;
1616 251661 : sOptions.eWkbVariant = eWkbVariant;
1617 503298 : return exportToWkb(pabyData, &sOptions);
1618 : }
1619 :
1620 : /************************************************************************/
1621 : /* OGR_G_ExportToWkb() */
1622 : /************************************************************************/
1623 : /**
1624 : * \brief Convert a geometry well known binary format
1625 : *
1626 : * This function relates to the SFCOM IWks::ExportToWKB() method.
1627 : *
1628 : * For backward compatibility purposes, it exports the Old-style 99-402
1629 : * extended dimension (Z) WKB types for types Point, LineString, Polygon,
1630 : * MultiPoint, MultiLineString, MultiPolygon and GeometryCollection.
1631 : * For other geometry types, it is equivalent to OGR_G_ExportToIsoWkb().
1632 : *
1633 : * This function is the same as the CPP method
1634 : * OGRGeometry::exportToWkb(OGRwkbByteOrder, unsigned char *,
1635 : * OGRwkbVariant) with eWkbVariant = wkbVariantOldOgc.
1636 : *
1637 : * @param hGeom handle on the geometry to convert to a well know binary
1638 : * data from.
1639 : * @param eOrder One of wkbXDR or wkbNDR indicating MSB or LSB byte order
1640 : * respectively.
1641 : * @param pabyDstBuffer a buffer into which the binary representation is
1642 : * written. This buffer must be at least
1643 : * OGR_G_WkbSize() byte in size.
1644 : *
1645 : * @return Currently OGRERR_NONE is always returned.
1646 : */
1647 :
1648 105 : OGRErr OGR_G_ExportToWkb(OGRGeometryH hGeom, OGRwkbByteOrder eOrder,
1649 : unsigned char *pabyDstBuffer)
1650 :
1651 : {
1652 105 : VALIDATE_POINTER1(hGeom, "OGR_G_ExportToWkb", OGRERR_FAILURE);
1653 :
1654 105 : return OGRGeometry::FromHandle(hGeom)->exportToWkb(eOrder, pabyDstBuffer);
1655 : }
1656 :
1657 : /************************************************************************/
1658 : /* OGR_G_ExportToIsoWkb() */
1659 : /************************************************************************/
1660 : /**
1661 : * \brief Convert a geometry into SFSQL 1.2 / ISO SQL/MM Part 3 well known
1662 : * binary format
1663 : *
1664 : * This function relates to the SFCOM IWks::ExportToWKB() method.
1665 : * It exports the SFSQL 1.2 and ISO SQL/MM Part 3 extended dimension (Z&M) WKB
1666 : * types.
1667 : *
1668 : * This function is the same as the CPP method
1669 : * OGRGeometry::exportToWkb(OGRwkbByteOrder, unsigned char *, OGRwkbVariant)
1670 : * with eWkbVariant = wkbVariantIso.
1671 : *
1672 : * @param hGeom handle on the geometry to convert to a well know binary
1673 : * data from.
1674 : * @param eOrder One of wkbXDR or wkbNDR indicating MSB or LSB byte order
1675 : * respectively.
1676 : * @param pabyDstBuffer a buffer into which the binary representation is
1677 : * written. This buffer must be at least
1678 : * OGR_G_WkbSize() byte in size.
1679 : *
1680 : * @return Currently OGRERR_NONE is always returned.
1681 : *
1682 : * @since GDAL 2.0
1683 : */
1684 :
1685 10571 : OGRErr OGR_G_ExportToIsoWkb(OGRGeometryH hGeom, OGRwkbByteOrder eOrder,
1686 : unsigned char *pabyDstBuffer)
1687 :
1688 : {
1689 10571 : VALIDATE_POINTER1(hGeom, "OGR_G_ExportToIsoWkb", OGRERR_FAILURE);
1690 :
1691 10571 : return OGRGeometry::FromHandle(hGeom)->exportToWkb(eOrder, pabyDstBuffer,
1692 10571 : wkbVariantIso);
1693 : }
1694 :
1695 : /************************************************************************/
1696 : /* OGR_G_ExportToWkbEx() */
1697 : /************************************************************************/
1698 :
1699 : /* clang-format off */
1700 : /**
1701 : * \fn OGRErr OGRGeometry::exportToWkb(unsigned char *pabyDstBuffer, const OGRwkbExportOptions *psOptions=nullptr) const
1702 : *
1703 : * \brief Convert a geometry into well known binary format
1704 : *
1705 : * This function relates to the SFCOM IWks::ExportToWKB() method.
1706 : *
1707 : * This function is the same as the C function OGR_G_ExportToWkbEx().
1708 : *
1709 : * @param pabyDstBuffer a buffer into which the binary representation is
1710 : * written. This buffer must be at least
1711 : * OGR_G_WkbSize() byte in size.
1712 : * @param psOptions WKB export options.
1713 :
1714 : * @return Currently OGRERR_NONE is always returned.
1715 : *
1716 : * @since GDAL 3.9
1717 : */
1718 : /* clang-format on */
1719 :
1720 : /**
1721 : * \brief Convert a geometry into well known binary format
1722 : *
1723 : * This function relates to the SFCOM IWks::ExportToWKB() method.
1724 : *
1725 : * This function is the same as the CPP method
1726 : * OGRGeometry::exportToWkb(unsigned char *, const OGRwkbExportOptions*)
1727 : *
1728 : * @param hGeom handle on the geometry to convert to a well know binary
1729 : * data from.
1730 : * @param pabyDstBuffer a buffer into which the binary representation is
1731 : * written. This buffer must be at least
1732 : * OGR_G_WkbSize() byte in size.
1733 : * @param psOptions WKB export options.
1734 :
1735 : * @return Currently OGRERR_NONE is always returned.
1736 : *
1737 : * @since GDAL 3.9
1738 : */
1739 :
1740 2 : OGRErr OGR_G_ExportToWkbEx(OGRGeometryH hGeom, unsigned char *pabyDstBuffer,
1741 : const OGRwkbExportOptions *psOptions)
1742 : {
1743 2 : VALIDATE_POINTER1(hGeom, "OGR_G_ExportToWkbEx", OGRERR_FAILURE);
1744 :
1745 4 : return OGRGeometry::FromHandle(hGeom)->exportToWkb(pabyDstBuffer,
1746 2 : psOptions);
1747 : }
1748 :
1749 : /**
1750 : * \fn OGRErr OGRGeometry::importFromWkt( const char ** ppszInput );
1751 : *
1752 : * \brief Assign geometry from well known text data.
1753 : *
1754 : * The object must have already been instantiated as the correct derived
1755 : * type of geometry object to match the text type. This method is used
1756 : * by the OGRGeometryFactory class, but not normally called by application
1757 : * code.
1758 : *
1759 : * This method relates to the SFCOM IWks::ImportFromWKT() method.
1760 : *
1761 : * This method is the same as the C function OGR_G_ImportFromWkt().
1762 : *
1763 : * @param ppszInput pointer to a pointer to the source text. The pointer is
1764 : * updated to pointer after the consumed text.
1765 : *
1766 : * @return OGRERR_NONE if all goes well, otherwise any of
1767 : * OGRERR_NOT_ENOUGH_DATA, OGRERR_UNSUPPORTED_GEOMETRY_TYPE, or
1768 : * OGRERR_CORRUPT_DATA may be returned.
1769 : */
1770 :
1771 : /************************************************************************/
1772 : /* OGR_G_ImportFromWkt() */
1773 : /************************************************************************/
1774 : /**
1775 : * \brief Assign geometry from well known text data.
1776 : *
1777 : * The object must have already been instantiated as the correct derived
1778 : * type of geometry object to match the text type.
1779 : *
1780 : * This function relates to the SFCOM IWks::ImportFromWKT() method.
1781 : *
1782 : * This function is the same as the CPP method OGRGeometry::importFromWkt().
1783 : *
1784 : * @param hGeom handle on the geometry to assign well know text data to.
1785 : * @param ppszSrcText pointer to a pointer to the source text. The pointer is
1786 : * updated to pointer after the consumed text.
1787 : *
1788 : * @return OGRERR_NONE if all goes well, otherwise any of
1789 : * OGRERR_NOT_ENOUGH_DATA, OGRERR_UNSUPPORTED_GEOMETRY_TYPE, or
1790 : * OGRERR_CORRUPT_DATA may be returned.
1791 : */
1792 :
1793 0 : OGRErr OGR_G_ImportFromWkt(OGRGeometryH hGeom, char **ppszSrcText)
1794 :
1795 : {
1796 0 : VALIDATE_POINTER1(hGeom, "OGR_G_ImportFromWkt", OGRERR_FAILURE);
1797 :
1798 0 : return OGRGeometry::FromHandle(hGeom)->importFromWkt(
1799 0 : const_cast<const char **>(ppszSrcText));
1800 : }
1801 :
1802 : /************************************************************************/
1803 : /* importPreambleFromWkt() */
1804 : /************************************************************************/
1805 :
1806 : // Returns -1 if processing must continue.
1807 : //! @cond Doxygen_Suppress
1808 123228 : OGRErr OGRGeometry::importPreambleFromWkt(const char **ppszInput, int *pbHasZ,
1809 : int *pbHasM, bool *pbIsEmpty)
1810 : {
1811 123228 : const char *pszInput = *ppszInput;
1812 :
1813 : /* -------------------------------------------------------------------- */
1814 : /* Clear existing Geoms. */
1815 : /* -------------------------------------------------------------------- */
1816 123228 : empty();
1817 123228 : *pbIsEmpty = false;
1818 :
1819 : /* -------------------------------------------------------------------- */
1820 : /* Read and verify the type keyword, and ensure it matches the */
1821 : /* actual type of this container. */
1822 : /* -------------------------------------------------------------------- */
1823 123228 : bool bHasM = false;
1824 123228 : bool bHasZ = false;
1825 123228 : bool bAlreadyGotDimension = false;
1826 :
1827 123228 : char szToken[OGR_WKT_TOKEN_MAX] = {};
1828 123228 : pszInput = OGRWktReadToken(pszInput, szToken);
1829 123228 : if (szToken[0] != '\0')
1830 : {
1831 : // Postgis EWKT: POINTM instead of POINT M.
1832 : // Current QGIS versions (at least <= 3.38) also export POINTZ.
1833 123228 : const size_t nTokenLen = strlen(szToken);
1834 123228 : if (szToken[nTokenLen - 1] == 'M' || szToken[nTokenLen - 1] == 'm')
1835 : {
1836 11 : szToken[nTokenLen - 1] = '\0';
1837 11 : bHasM = true;
1838 11 : bAlreadyGotDimension = true;
1839 :
1840 11 : if (nTokenLen > 2 && (szToken[nTokenLen - 2] == 'Z' ||
1841 9 : szToken[nTokenLen - 2] == 'z'))
1842 : {
1843 4 : bHasZ = true;
1844 4 : szToken[nTokenLen - 2] = '\0';
1845 : }
1846 : }
1847 123217 : else if (szToken[nTokenLen - 1] == 'Z' || szToken[nTokenLen - 1] == 'z')
1848 : {
1849 5 : szToken[nTokenLen - 1] = '\0';
1850 5 : bHasZ = true;
1851 5 : bAlreadyGotDimension = true;
1852 : }
1853 : }
1854 :
1855 123228 : if (!EQUAL(szToken, getGeometryName()))
1856 0 : return OGRERR_CORRUPT_DATA;
1857 :
1858 : /* -------------------------------------------------------------------- */
1859 : /* Check for Z, M or ZM */
1860 : /* -------------------------------------------------------------------- */
1861 123228 : if (!bAlreadyGotDimension)
1862 : {
1863 123212 : const char *pszNewInput = OGRWktReadToken(pszInput, szToken);
1864 123212 : if (EQUAL(szToken, "Z"))
1865 : {
1866 1041 : pszInput = pszNewInput;
1867 1041 : bHasZ = true;
1868 : }
1869 122171 : else if (EQUAL(szToken, "M"))
1870 : {
1871 320 : pszInput = pszNewInput;
1872 320 : bHasM = true;
1873 : }
1874 121851 : else if (EQUAL(szToken, "ZM"))
1875 : {
1876 473 : pszInput = pszNewInput;
1877 473 : bHasZ = true;
1878 473 : bHasM = true;
1879 : }
1880 : }
1881 123228 : *pbHasZ = bHasZ;
1882 123228 : *pbHasM = bHasM;
1883 :
1884 : /* -------------------------------------------------------------------- */
1885 : /* Check for EMPTY ... */
1886 : /* -------------------------------------------------------------------- */
1887 123228 : const char *pszNewInput = OGRWktReadToken(pszInput, szToken);
1888 123228 : if (EQUAL(szToken, "EMPTY"))
1889 : {
1890 1560 : *ppszInput = pszNewInput;
1891 1560 : *pbIsEmpty = true;
1892 1560 : if (bHasZ)
1893 129 : set3D(TRUE);
1894 1560 : if (bHasM)
1895 83 : setMeasured(TRUE);
1896 1560 : return OGRERR_NONE;
1897 : }
1898 :
1899 121668 : if (!EQUAL(szToken, "("))
1900 35 : return OGRERR_CORRUPT_DATA;
1901 :
1902 121633 : if (!bHasZ && !bHasM)
1903 : {
1904 : // Test for old-style XXXXXXXXX(EMPTY).
1905 119955 : pszNewInput = OGRWktReadToken(pszNewInput, szToken);
1906 119955 : if (EQUAL(szToken, "EMPTY"))
1907 : {
1908 69 : pszNewInput = OGRWktReadToken(pszNewInput, szToken);
1909 :
1910 69 : if (EQUAL(szToken, ","))
1911 : {
1912 : // This is OK according to SFSQL SPEC.
1913 : }
1914 44 : else if (!EQUAL(szToken, ")"))
1915 : {
1916 9 : return OGRERR_CORRUPT_DATA;
1917 : }
1918 : else
1919 : {
1920 35 : *ppszInput = pszNewInput;
1921 35 : empty();
1922 35 : *pbIsEmpty = true;
1923 35 : return OGRERR_NONE;
1924 : }
1925 : }
1926 : }
1927 :
1928 121589 : *ppszInput = pszInput;
1929 :
1930 121589 : return OGRERR_NONE;
1931 : }
1932 :
1933 : //! @endcond
1934 :
1935 : /************************************************************************/
1936 : /* wktTypeString() */
1937 : /************************************************************************/
1938 :
1939 : //! @cond Doxygen_Suppress
1940 : /** Get a type string for WKT, padded with a space at the end.
1941 : *
1942 : * @param variant OGR type variant
1943 : * @return "Z " for 3D, "M " for measured, "ZM " for both, or the empty string.
1944 : */
1945 13820 : std::string OGRGeometry::wktTypeString(OGRwkbVariant variant) const
1946 : {
1947 13820 : std::string s(" ");
1948 :
1949 13820 : if (variant == wkbVariantIso)
1950 : {
1951 8984 : if (flags & OGR_G_3D)
1952 1845 : s += "Z";
1953 8984 : if (flags & OGR_G_MEASURED)
1954 1087 : s += "M";
1955 : }
1956 13820 : if (s.size() > 1)
1957 2352 : s += " ";
1958 13820 : return s;
1959 : }
1960 :
1961 : //! @endcond
1962 :
1963 : /**
1964 : * \fn OGRErr OGRGeometry::exportToWkt( char ** ppszDstText,
1965 : * OGRwkbVariant variant = wkbVariantOldOgc ) const;
1966 : *
1967 : * \brief Convert a geometry into well known text format.
1968 : *
1969 : * This method relates to the SFCOM IWks::ExportToWKT() method.
1970 : *
1971 : * This method is the same as the C function OGR_G_ExportToWkt().
1972 : *
1973 : * @param ppszDstText a text buffer is allocated by the program, and assigned
1974 : * to the passed pointer. After use, *ppszDstText should be
1975 : * freed with CPLFree().
1976 : * @param variant the specification that must be conformed too :
1977 : * - wkbVariantOgc for old-style 99-402 extended
1978 : * dimension (Z) WKB types
1979 : * - wkbVariantIso for SFSQL 1.2 and ISO SQL/MM Part 3
1980 : *
1981 : * @return Currently OGRERR_NONE is always returned.
1982 : */
1983 8745 : OGRErr OGRGeometry::exportToWkt(char **ppszDstText, OGRwkbVariant variant) const
1984 : {
1985 8745 : OGRWktOptions opts;
1986 8745 : opts.variant = variant;
1987 8745 : OGRErr err(OGRERR_NONE);
1988 :
1989 8745 : std::string wkt = exportToWkt(opts, &err);
1990 8745 : *ppszDstText = CPLStrdup(wkt.data());
1991 17490 : return err;
1992 : }
1993 :
1994 : /************************************************************************/
1995 : /* OGR_G_ExportToWkt() */
1996 : /************************************************************************/
1997 :
1998 : /**
1999 : * \brief Convert a geometry into well known text format.
2000 : *
2001 : * This function relates to the SFCOM IWks::ExportToWKT() method.
2002 : *
2003 : * For backward compatibility purposes, it exports the Old-style 99-402
2004 : * extended dimension (Z) WKB types for types Point, LineString, Polygon,
2005 : * MultiPoint, MultiLineString, MultiPolygon and GeometryCollection.
2006 : * For other geometry types, it is equivalent to OGR_G_ExportToIsoWkt().
2007 : *
2008 : * This function is the same as the CPP method OGRGeometry::exportToWkt().
2009 : *
2010 : * @param hGeom handle on the geometry to convert to a text format from.
2011 : * @param ppszSrcText a text buffer is allocated by the program, and assigned
2012 : * to the passed pointer. After use, *ppszDstText should be
2013 : * freed with CPLFree().
2014 : *
2015 : * @return Currently OGRERR_NONE is always returned.
2016 : */
2017 :
2018 2399 : OGRErr OGR_G_ExportToWkt(OGRGeometryH hGeom, char **ppszSrcText)
2019 :
2020 : {
2021 2399 : VALIDATE_POINTER1(hGeom, "OGR_G_ExportToWkt", OGRERR_FAILURE);
2022 :
2023 2399 : return OGRGeometry::FromHandle(hGeom)->exportToWkt(ppszSrcText);
2024 : }
2025 :
2026 : /************************************************************************/
2027 : /* OGR_G_ExportToIsoWkt() */
2028 : /************************************************************************/
2029 :
2030 : /**
2031 : * \brief Convert a geometry into SFSQL 1.2 / ISO SQL/MM Part 3 well
2032 : * known text format.
2033 : *
2034 : * This function relates to the SFCOM IWks::ExportToWKT() method.
2035 : * It exports the SFSQL 1.2 and ISO SQL/MM Part 3 extended dimension
2036 : * (Z&M) WKB types.
2037 : *
2038 : * This function is the same as the CPP method
2039 : * OGRGeometry::exportToWkt(wkbVariantIso).
2040 : *
2041 : * @param hGeom handle on the geometry to convert to a text format from.
2042 : * @param ppszSrcText a text buffer is allocated by the program, and assigned
2043 : * to the passed pointer. After use, *ppszDstText should be
2044 : * freed with CPLFree().
2045 : *
2046 : * @return Currently OGRERR_NONE is always returned.
2047 : *
2048 : * @since GDAL 2.0
2049 : */
2050 :
2051 5408 : OGRErr OGR_G_ExportToIsoWkt(OGRGeometryH hGeom, char **ppszSrcText)
2052 :
2053 : {
2054 5408 : VALIDATE_POINTER1(hGeom, "OGR_G_ExportToIsoWkt", OGRERR_FAILURE);
2055 :
2056 5408 : return OGRGeometry::FromHandle(hGeom)->exportToWkt(ppszSrcText,
2057 5408 : wkbVariantIso);
2058 : }
2059 :
2060 : /**
2061 : * \fn OGRwkbGeometryType OGRGeometry::getGeometryType() const;
2062 : *
2063 : * \brief Fetch geometry type.
2064 : *
2065 : * Note that the geometry type may include the 2.5D flag. To get a 2D
2066 : * flattened version of the geometry type apply the wkbFlatten() macro
2067 : * to the return result.
2068 : *
2069 : * This method is the same as the C function OGR_G_GetGeometryType().
2070 : *
2071 : * @return the geometry type code.
2072 : */
2073 :
2074 : /************************************************************************/
2075 : /* OGR_G_GetGeometryType() */
2076 : /************************************************************************/
2077 : /**
2078 : * \brief Fetch geometry type.
2079 : *
2080 : * Note that the geometry type may include the 2.5D flag. To get a 2D
2081 : * flattened version of the geometry type apply the wkbFlatten() macro
2082 : * to the return result.
2083 : *
2084 : * This function is the same as the CPP method OGRGeometry::getGeometryType().
2085 : *
2086 : * @param hGeom handle on the geometry to get type from.
2087 : * @return the geometry type code.
2088 : */
2089 :
2090 5154 : OGRwkbGeometryType OGR_G_GetGeometryType(OGRGeometryH hGeom)
2091 :
2092 : {
2093 5154 : VALIDATE_POINTER1(hGeom, "OGR_G_GetGeometryType", wkbUnknown);
2094 :
2095 5154 : return OGRGeometry::FromHandle(hGeom)->getGeometryType();
2096 : }
2097 :
2098 : /**
2099 : * \fn const char * OGRGeometry::getGeometryName() const;
2100 : *
2101 : * \brief Fetch WKT name for geometry type.
2102 : *
2103 : * There is no SFCOM analog to this method.
2104 : *
2105 : * This method is the same as the C function OGR_G_GetGeometryName().
2106 : *
2107 : * @return name used for this geometry type in well known text format. The
2108 : * returned pointer is to a static internal string and should not be modified
2109 : * or freed.
2110 : */
2111 :
2112 : /************************************************************************/
2113 : /* OGR_G_GetGeometryName() */
2114 : /************************************************************************/
2115 : /**
2116 : * \brief Fetch WKT name for geometry type.
2117 : *
2118 : * There is no SFCOM analog to this function.
2119 : *
2120 : * This function is the same as the CPP method OGRGeometry::getGeometryName().
2121 : *
2122 : * @param hGeom handle on the geometry to get name from.
2123 : * @return name used for this geometry type in well known text format.
2124 : */
2125 :
2126 16093 : const char *OGR_G_GetGeometryName(OGRGeometryH hGeom)
2127 :
2128 : {
2129 16093 : VALIDATE_POINTER1(hGeom, "OGR_G_GetGeometryName", "");
2130 :
2131 16093 : return OGRGeometry::FromHandle(hGeom)->getGeometryName();
2132 : }
2133 :
2134 : /**
2135 : * \fn OGRGeometry *OGRGeometry::clone() const;
2136 : *
2137 : * \brief Make a copy of this object.
2138 : *
2139 : * This method relates to the SFCOM IGeometry::clone() method.
2140 : *
2141 : * This method is the same as the C function OGR_G_Clone().
2142 : *
2143 : * @return a new object instance with the same geometry, and spatial
2144 : * reference system as the original.
2145 : */
2146 :
2147 : /************************************************************************/
2148 : /* OGR_G_Clone() */
2149 : /************************************************************************/
2150 : /**
2151 : * \brief Make a copy of this object.
2152 : *
2153 : * This function relates to the SFCOM IGeometry::clone() method.
2154 : *
2155 : * This function is the same as the CPP method OGRGeometry::clone().
2156 : *
2157 : * @param hGeom handle on the geometry to clone from.
2158 : * @return a handle on the copy of the geometry with the spatial
2159 : * reference system as the original.
2160 : */
2161 :
2162 13516 : OGRGeometryH OGR_G_Clone(OGRGeometryH hGeom)
2163 :
2164 : {
2165 13516 : VALIDATE_POINTER1(hGeom, "OGR_G_Clone", nullptr);
2166 :
2167 13516 : return OGRGeometry::ToHandle(OGRGeometry::FromHandle(hGeom)->clone());
2168 : }
2169 :
2170 : /**
2171 : * \fn OGRSpatialReference *OGRGeometry::getSpatialReference();
2172 : *
2173 : * \brief Returns spatial reference system for object.
2174 : *
2175 : * This method relates to the SFCOM IGeometry::get_SpatialReference() method.
2176 : *
2177 : * This method is the same as the C function OGR_G_GetSpatialReference().
2178 : *
2179 : * @return a reference to the spatial reference object. The object may be
2180 : * shared with many geometry objects, and should not be modified.
2181 : */
2182 :
2183 : /************************************************************************/
2184 : /* OGR_G_GetSpatialReference() */
2185 : /************************************************************************/
2186 : /**
2187 : * \brief Returns spatial reference system for geometry.
2188 : *
2189 : * This function relates to the SFCOM IGeometry::get_SpatialReference() method.
2190 : *
2191 : * This function is the same as the CPP method
2192 : * OGRGeometry::getSpatialReference().
2193 : *
2194 : * @param hGeom handle on the geometry to get spatial reference from.
2195 : * @return a reference to the spatial reference geometry, which should not be
2196 : * modified.
2197 : */
2198 :
2199 49 : OGRSpatialReferenceH OGR_G_GetSpatialReference(OGRGeometryH hGeom)
2200 :
2201 : {
2202 49 : VALIDATE_POINTER1(hGeom, "OGR_G_GetSpatialReference", nullptr);
2203 :
2204 49 : return OGRSpatialReference::ToHandle(const_cast<OGRSpatialReference *>(
2205 98 : OGRGeometry::FromHandle(hGeom)->getSpatialReference()));
2206 : }
2207 :
2208 : /**
2209 : * \fn void OGRGeometry::empty();
2210 : *
2211 : * \brief Clear geometry information.
2212 : * This restores the geometry to its initial
2213 : * state after construction, and before assignment of actual geometry.
2214 : *
2215 : * This method relates to the SFCOM IGeometry::Empty() method.
2216 : *
2217 : * This method is the same as the C function OGR_G_Empty().
2218 : */
2219 :
2220 : /************************************************************************/
2221 : /* OGR_G_Empty() */
2222 : /************************************************************************/
2223 : /**
2224 : * \brief Clear geometry information.
2225 : * This restores the geometry to its initial
2226 : * state after construction, and before assignment of actual geometry.
2227 : *
2228 : * This function relates to the SFCOM IGeometry::Empty() method.
2229 : *
2230 : * This function is the same as the CPP method OGRGeometry::empty().
2231 : *
2232 : * @param hGeom handle on the geometry to empty.
2233 : */
2234 :
2235 4 : void OGR_G_Empty(OGRGeometryH hGeom)
2236 :
2237 : {
2238 4 : VALIDATE_POINTER0(hGeom, "OGR_G_Empty");
2239 :
2240 4 : OGRGeometry::FromHandle(hGeom)->empty();
2241 : }
2242 :
2243 : /**
2244 : * \fn OGRBoolean OGRGeometry::IsEmpty() const;
2245 : *
2246 : * \brief Returns TRUE (non-zero) if the object has no points.
2247 : *
2248 : * Normally this
2249 : * returns FALSE except between when an object is instantiated and points
2250 : * have been assigned.
2251 : *
2252 : * This method relates to the SFCOM IGeometry::IsEmpty() method.
2253 : *
2254 : * @return TRUE if object is empty, otherwise FALSE.
2255 : */
2256 :
2257 : /************************************************************************/
2258 : /* OGR_G_IsEmpty() */
2259 : /************************************************************************/
2260 :
2261 : /**
2262 : * \brief Test if the geometry is empty.
2263 : *
2264 : * This method is the same as the CPP method OGRGeometry::IsEmpty().
2265 : *
2266 : * @param hGeom The Geometry to test.
2267 : *
2268 : * @return TRUE if the geometry has no points, otherwise FALSE.
2269 : */
2270 :
2271 2141 : int OGR_G_IsEmpty(OGRGeometryH hGeom)
2272 :
2273 : {
2274 2141 : VALIDATE_POINTER1(hGeom, "OGR_G_IsEmpty", TRUE);
2275 :
2276 2141 : return OGRGeometry::FromHandle(hGeom)->IsEmpty();
2277 : }
2278 :
2279 : /************************************************************************/
2280 : /* IsValid() */
2281 : /************************************************************************/
2282 :
2283 : /**
2284 : * \brief Test if the geometry is valid.
2285 : *
2286 : * This method is the same as the C function OGR_G_IsValid().
2287 : *
2288 : * This method is built on the GEOS library, check it for the definition
2289 : * of the geometry operation.
2290 : * If OGR is built without the GEOS library, this method will always return
2291 : * FALSE.
2292 : *
2293 : *
2294 : * @return TRUE if the geometry has no points, otherwise FALSE.
2295 : */
2296 :
2297 1684 : OGRBoolean OGRGeometry::IsValid() const
2298 :
2299 : {
2300 1684 : if (IsSFCGALCompatible())
2301 : {
2302 : #ifndef HAVE_SFCGAL
2303 :
2304 : #ifdef HAVE_GEOS
2305 2 : if (wkbFlatten(getGeometryType()) == wkbTriangle)
2306 : {
2307 : // go on
2308 : }
2309 : else
2310 : #endif
2311 : {
2312 1 : CPLError(CE_Failure, CPLE_NotSupported,
2313 : "SFCGAL support not enabled.");
2314 1 : return FALSE;
2315 : }
2316 : #else
2317 : sfcgal_init();
2318 : sfcgal_geometry_t *poThis = OGRGeometry::OGRexportToSFCGAL(this);
2319 : if (poThis == nullptr)
2320 : {
2321 : CPLError(CE_Failure, CPLE_IllegalArg,
2322 : "SFCGAL geometry returned is NULL");
2323 : return FALSE;
2324 : }
2325 :
2326 : const int res = sfcgal_geometry_is_valid(poThis);
2327 : sfcgal_geometry_delete(poThis);
2328 : return res == 1;
2329 : #endif
2330 : }
2331 :
2332 : {
2333 : #ifndef HAVE_GEOS
2334 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
2335 : return FALSE;
2336 :
2337 : #else
2338 1655 : OGRBoolean bResult = FALSE;
2339 :
2340 : // Some invalid geometries, such as lines with one point, or
2341 : // rings that do not close, cannot be converted to GEOS.
2342 : // For validity checking we initialize the GEOS context with
2343 : // the warning handler as the error handler to avoid emitting
2344 : // CE_Failure when a geometry cannot be converted to GEOS.
2345 : GEOSContextHandle_t hGEOSCtxt =
2346 1655 : initGEOS_r(OGRGEOSWarningHandler, OGRGEOSWarningHandler);
2347 :
2348 1653 : GEOSGeom hThisGeosGeom = exportToGEOS(hGEOSCtxt);
2349 :
2350 1655 : if (hThisGeosGeom != nullptr)
2351 : {
2352 1645 : bResult = GEOSisValid_r(hGEOSCtxt, hThisGeosGeom);
2353 : #ifdef DEBUG_VERBOSE
2354 : if (!bResult)
2355 : {
2356 : char *pszReason = GEOSisValidReason_r(hGEOSCtxt, hThisGeosGeom);
2357 : CPLDebug("OGR", "%s", pszReason);
2358 : GEOSFree_r(hGEOSCtxt, pszReason);
2359 : }
2360 : #endif
2361 1649 : GEOSGeom_destroy_r(hGEOSCtxt, hThisGeosGeom);
2362 : }
2363 1673 : freeGEOSContext(hGEOSCtxt);
2364 :
2365 1648 : return bResult;
2366 :
2367 : #endif // HAVE_GEOS
2368 : }
2369 : }
2370 :
2371 : /************************************************************************/
2372 : /* OGR_G_IsValid() */
2373 : /************************************************************************/
2374 :
2375 : /**
2376 : * \brief Test if the geometry is valid.
2377 : *
2378 : * This function is the same as the C++ method OGRGeometry::IsValid().
2379 : *
2380 : * This function is built on the GEOS library, check it for the definition
2381 : * of the geometry operation.
2382 : * If OGR is built without the GEOS library, this function will always return
2383 : * FALSE.
2384 : *
2385 : * @param hGeom The Geometry to test.
2386 : *
2387 : * @return TRUE if the geometry has no points, otherwise FALSE.
2388 : */
2389 :
2390 19 : int OGR_G_IsValid(OGRGeometryH hGeom)
2391 :
2392 : {
2393 19 : VALIDATE_POINTER1(hGeom, "OGR_G_IsValid", FALSE);
2394 :
2395 19 : return OGRGeometry::FromHandle(hGeom)->IsValid();
2396 : }
2397 :
2398 : /************************************************************************/
2399 : /* IsSimple() */
2400 : /************************************************************************/
2401 :
2402 : /**
2403 : * \brief Test if the geometry is simple.
2404 : *
2405 : * This method is the same as the C function OGR_G_IsSimple().
2406 : *
2407 : * This method is built on the GEOS library, check it for the definition
2408 : * of the geometry operation.
2409 : * If OGR is built without the GEOS library, this method will always return
2410 : * FALSE.
2411 : *
2412 : *
2413 : * @return TRUE if the geometry has no points, otherwise FALSE.
2414 : */
2415 :
2416 5 : OGRBoolean OGRGeometry::IsSimple() const
2417 :
2418 : {
2419 : #ifndef HAVE_GEOS
2420 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
2421 : return FALSE;
2422 :
2423 : #else
2424 :
2425 5 : OGRBoolean bResult = FALSE;
2426 :
2427 5 : GEOSContextHandle_t hGEOSCtxt = createGEOSContext();
2428 5 : GEOSGeom hThisGeosGeom = exportToGEOS(hGEOSCtxt);
2429 :
2430 5 : if (hThisGeosGeom != nullptr)
2431 : {
2432 5 : bResult = GEOSisSimple_r(hGEOSCtxt, hThisGeosGeom);
2433 5 : GEOSGeom_destroy_r(hGEOSCtxt, hThisGeosGeom);
2434 : }
2435 5 : freeGEOSContext(hGEOSCtxt);
2436 :
2437 5 : return bResult;
2438 :
2439 : #endif // HAVE_GEOS
2440 : }
2441 :
2442 : /**
2443 : * \brief Returns TRUE if the geometry is simple.
2444 : *
2445 : * Returns TRUE if the geometry has no anomalous geometric points, such
2446 : * as self intersection or self tangency. The description of each
2447 : * instantiable geometric class will include the specific conditions that
2448 : * cause an instance of that class to be classified as not simple.
2449 : *
2450 : * This function is the same as the C++ method OGRGeometry::IsSimple() method.
2451 : *
2452 : * If OGR is built without the GEOS library, this function will always return
2453 : * FALSE.
2454 : *
2455 : * @param hGeom The Geometry to test.
2456 : *
2457 : * @return TRUE if object is simple, otherwise FALSE.
2458 : */
2459 :
2460 5 : int OGR_G_IsSimple(OGRGeometryH hGeom)
2461 :
2462 : {
2463 5 : VALIDATE_POINTER1(hGeom, "OGR_G_IsSimple", TRUE);
2464 :
2465 5 : return OGRGeometry::FromHandle(hGeom)->IsSimple();
2466 : }
2467 :
2468 : /************************************************************************/
2469 : /* IsRing() */
2470 : /************************************************************************/
2471 :
2472 : /**
2473 : * \brief Test if the geometry is a ring
2474 : *
2475 : * This method is the same as the C function OGR_G_IsRing().
2476 : *
2477 : * This method is built on the GEOS library, check it for the definition
2478 : * of the geometry operation.
2479 : * If OGR is built without the GEOS library, this method will always return
2480 : * FALSE.
2481 : *
2482 : *
2483 : * @return TRUE if the geometry has no points, otherwise FALSE.
2484 : */
2485 :
2486 1 : OGRBoolean OGRGeometry::IsRing() const
2487 :
2488 : {
2489 : #ifndef HAVE_GEOS
2490 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
2491 : return FALSE;
2492 :
2493 : #else
2494 :
2495 1 : OGRBoolean bResult = FALSE;
2496 :
2497 1 : GEOSContextHandle_t hGEOSCtxt = createGEOSContext();
2498 1 : GEOSGeom hThisGeosGeom = exportToGEOS(hGEOSCtxt);
2499 :
2500 1 : if (hThisGeosGeom != nullptr)
2501 : {
2502 1 : bResult = GEOSisRing_r(hGEOSCtxt, hThisGeosGeom);
2503 1 : GEOSGeom_destroy_r(hGEOSCtxt, hThisGeosGeom);
2504 : }
2505 1 : freeGEOSContext(hGEOSCtxt);
2506 :
2507 1 : return bResult;
2508 :
2509 : #endif // HAVE_GEOS
2510 : }
2511 :
2512 : /************************************************************************/
2513 : /* OGR_G_IsRing() */
2514 : /************************************************************************/
2515 :
2516 : /**
2517 : * \brief Test if the geometry is a ring
2518 : *
2519 : * This function is the same as the C++ method OGRGeometry::IsRing().
2520 : *
2521 : * This function is built on the GEOS library, check it for the definition
2522 : * of the geometry operation.
2523 : * If OGR is built without the GEOS library, this function will always return
2524 : * FALSE.
2525 : *
2526 : * @param hGeom The Geometry to test.
2527 : *
2528 : * @return TRUE if the geometry has no points, otherwise FALSE.
2529 : */
2530 :
2531 1 : int OGR_G_IsRing(OGRGeometryH hGeom)
2532 :
2533 : {
2534 1 : VALIDATE_POINTER1(hGeom, "OGR_G_IsRing", FALSE);
2535 :
2536 1 : return OGRGeometry::FromHandle(hGeom)->IsRing();
2537 : }
2538 :
2539 : /************************************************************************/
2540 : /* OGRFromOGCGeomType() */
2541 : /************************************************************************/
2542 :
2543 : /** Map OGCgeometry format type to corresponding OGR constants.
2544 : * @param pszGeomType POINT[ ][Z][M], LINESTRING[ ][Z][M], etc...
2545 : * @return OGR constant.
2546 : */
2547 2405 : OGRwkbGeometryType OGRFromOGCGeomType(const char *pszGeomType)
2548 : {
2549 2405 : OGRwkbGeometryType eType = wkbUnknown;
2550 2405 : bool bConvertTo3D = false;
2551 2405 : bool bIsMeasured = false;
2552 2405 : if (*pszGeomType != '\0')
2553 : {
2554 2399 : char ch = pszGeomType[strlen(pszGeomType) - 1];
2555 2399 : if (ch == 'm' || ch == 'M')
2556 : {
2557 2 : bIsMeasured = true;
2558 2 : if (strlen(pszGeomType) > 1)
2559 2 : ch = pszGeomType[strlen(pszGeomType) - 2];
2560 : }
2561 2399 : if (ch == 'z' || ch == 'Z')
2562 : {
2563 16 : bConvertTo3D = true;
2564 : }
2565 : }
2566 :
2567 2405 : if (STARTS_WITH_CI(pszGeomType, "POINT"))
2568 553 : eType = wkbPoint;
2569 1852 : else if (STARTS_WITH_CI(pszGeomType, "LINESTRING"))
2570 172 : eType = wkbLineString;
2571 1680 : else if (STARTS_WITH_CI(pszGeomType, "POLYGON"))
2572 573 : eType = wkbPolygon;
2573 1107 : else if (STARTS_WITH_CI(pszGeomType, "MULTIPOINT"))
2574 21 : eType = wkbMultiPoint;
2575 1086 : else if (STARTS_WITH_CI(pszGeomType, "MULTILINESTRING"))
2576 13 : eType = wkbMultiLineString;
2577 1073 : else if (STARTS_WITH_CI(pszGeomType, "MULTIPOLYGON"))
2578 64 : eType = wkbMultiPolygon;
2579 1009 : else if (STARTS_WITH_CI(pszGeomType, "GEOMETRYCOLLECTION"))
2580 4 : eType = wkbGeometryCollection;
2581 1005 : else if (STARTS_WITH_CI(pszGeomType, "CIRCULARSTRING"))
2582 331 : eType = wkbCircularString;
2583 674 : else if (STARTS_WITH_CI(pszGeomType, "COMPOUNDCURVE"))
2584 0 : eType = wkbCompoundCurve;
2585 674 : else if (STARTS_WITH_CI(pszGeomType, "CURVEPOLYGON"))
2586 16 : eType = wkbCurvePolygon;
2587 658 : else if (STARTS_WITH_CI(pszGeomType, "MULTICURVE"))
2588 2 : eType = wkbMultiCurve;
2589 656 : else if (STARTS_WITH_CI(pszGeomType, "MULTISURFACE"))
2590 0 : eType = wkbMultiSurface;
2591 656 : else if (STARTS_WITH_CI(pszGeomType, "TRIANGLE"))
2592 0 : eType = wkbTriangle;
2593 656 : else if (STARTS_WITH_CI(pszGeomType, "POLYHEDRALSURFACE"))
2594 1 : eType = wkbPolyhedralSurface;
2595 655 : else if (STARTS_WITH_CI(pszGeomType, "TIN"))
2596 5 : eType = wkbTIN;
2597 650 : else if (STARTS_WITH_CI(pszGeomType, "CURVE"))
2598 3 : eType = wkbCurve;
2599 647 : else if (STARTS_WITH_CI(pszGeomType, "SURFACE"))
2600 3 : eType = wkbSurface;
2601 : else
2602 644 : eType = wkbUnknown;
2603 :
2604 2405 : if (bConvertTo3D)
2605 16 : eType = wkbSetZ(eType);
2606 2405 : if (bIsMeasured)
2607 2 : eType = wkbSetM(eType);
2608 :
2609 2405 : return eType;
2610 : }
2611 :
2612 : /************************************************************************/
2613 : /* OGRToOGCGeomType() */
2614 : /************************************************************************/
2615 :
2616 : /** Map OGR geometry format constants to corresponding OGC geometry type.
2617 : * @param eGeomType OGR geometry type
2618 : * @param bCamelCase Whether the return should be like "MultiPoint"
2619 : * (bCamelCase=true) or "MULTIPOINT" (bCamelCase=false, default)
2620 : * @param bAddZM Whether to include Z, M or ZM suffix for non-2D geometries.
2621 : * Default is false.
2622 : * @param bSpaceBeforeZM Whether to include a space character before the Z/M/ZM
2623 : * suffix. Default is false.
2624 : * @return string with OGC geometry type (without dimensionality)
2625 : */
2626 2461 : const char *OGRToOGCGeomType(OGRwkbGeometryType eGeomType, bool bCamelCase,
2627 : bool bAddZM, bool bSpaceBeforeZM)
2628 : {
2629 2461 : const char *pszRet = "";
2630 2461 : switch (wkbFlatten(eGeomType))
2631 : {
2632 1377 : case wkbUnknown:
2633 1377 : pszRet = "Geometry";
2634 1377 : break;
2635 354 : case wkbPoint:
2636 354 : pszRet = "Point";
2637 354 : break;
2638 141 : case wkbLineString:
2639 141 : pszRet = "LineString";
2640 141 : break;
2641 263 : case wkbPolygon:
2642 263 : pszRet = "Polygon";
2643 263 : break;
2644 50 : case wkbMultiPoint:
2645 50 : pszRet = "MultiPoint";
2646 50 : break;
2647 53 : case wkbMultiLineString:
2648 53 : pszRet = "MultiLineString";
2649 53 : break;
2650 68 : case wkbMultiPolygon:
2651 68 : pszRet = "MultiPolygon";
2652 68 : break;
2653 54 : case wkbGeometryCollection:
2654 54 : pszRet = "GeometryCollection";
2655 54 : break;
2656 9 : case wkbCircularString:
2657 9 : pszRet = "CircularString";
2658 9 : break;
2659 3 : case wkbCompoundCurve:
2660 3 : pszRet = "CompoundCurve";
2661 3 : break;
2662 12 : case wkbCurvePolygon:
2663 12 : pszRet = "CurvePolygon";
2664 12 : break;
2665 2 : case wkbMultiCurve:
2666 2 : pszRet = "MultiCurve";
2667 2 : break;
2668 3 : case wkbMultiSurface:
2669 3 : pszRet = "MultiSurface";
2670 3 : break;
2671 3 : case wkbTriangle:
2672 3 : pszRet = "Triangle";
2673 3 : break;
2674 3 : case wkbPolyhedralSurface:
2675 3 : pszRet = "PolyhedralSurface";
2676 3 : break;
2677 1 : case wkbTIN:
2678 1 : pszRet = "Tin";
2679 1 : break;
2680 3 : case wkbCurve:
2681 3 : pszRet = "Curve";
2682 3 : break;
2683 3 : case wkbSurface:
2684 3 : pszRet = "Surface";
2685 3 : break;
2686 59 : default:
2687 59 : break;
2688 : }
2689 2461 : if (bAddZM)
2690 : {
2691 42 : const bool bHasZ = CPL_TO_BOOL(OGR_GT_HasZ(eGeomType));
2692 42 : const bool bHasM = CPL_TO_BOOL(OGR_GT_HasM(eGeomType));
2693 42 : if (bHasZ || bHasM)
2694 : {
2695 11 : if (bSpaceBeforeZM)
2696 1 : pszRet = CPLSPrintf("%s ", pszRet);
2697 11 : if (bHasZ)
2698 10 : pszRet = CPLSPrintf("%sZ", pszRet);
2699 11 : if (bHasM)
2700 6 : pszRet = CPLSPrintf("%sM", pszRet);
2701 : }
2702 : }
2703 2461 : if (!bCamelCase)
2704 2418 : pszRet = CPLSPrintf("%s", CPLString(pszRet).toupper().c_str());
2705 2461 : return pszRet;
2706 : }
2707 :
2708 : /************************************************************************/
2709 : /* OGRGeometryTypeToName() */
2710 : /************************************************************************/
2711 :
2712 : /**
2713 : * \brief Fetch a human readable name corresponding to an OGRwkbGeometryType
2714 : * value. The returned value should not be modified, or freed by the
2715 : * application.
2716 : *
2717 : * This function is C callable.
2718 : *
2719 : * @param eType the geometry type.
2720 : *
2721 : * @return internal human readable string, or NULL on failure.
2722 : */
2723 :
2724 373 : const char *OGRGeometryTypeToName(OGRwkbGeometryType eType)
2725 :
2726 : {
2727 373 : bool b3D = wkbHasZ(eType);
2728 373 : bool bMeasured = wkbHasM(eType);
2729 :
2730 373 : switch (wkbFlatten(eType))
2731 : {
2732 34 : case wkbUnknown:
2733 34 : if (b3D && bMeasured)
2734 0 : return "3D Measured Unknown (any)";
2735 34 : else if (b3D)
2736 1 : return "3D Unknown (any)";
2737 33 : else if (bMeasured)
2738 0 : return "Measured Unknown (any)";
2739 : else
2740 33 : return "Unknown (any)";
2741 :
2742 50 : case wkbPoint:
2743 50 : if (b3D && bMeasured)
2744 2 : return "3D Measured Point";
2745 48 : else if (b3D)
2746 11 : return "3D Point";
2747 37 : else if (bMeasured)
2748 2 : return "Measured Point";
2749 : else
2750 35 : return "Point";
2751 :
2752 29 : case wkbLineString:
2753 29 : if (b3D && bMeasured)
2754 0 : return "3D Measured Line String";
2755 29 : else if (b3D)
2756 9 : return "3D Line String";
2757 20 : else if (bMeasured)
2758 0 : return "Measured Line String";
2759 : else
2760 20 : return "Line String";
2761 :
2762 56 : case wkbPolygon:
2763 56 : if (b3D && bMeasured)
2764 0 : return "3D Measured Polygon";
2765 56 : else if (b3D)
2766 8 : return "3D Polygon";
2767 48 : else if (bMeasured)
2768 0 : return "Measured Polygon";
2769 : else
2770 48 : return "Polygon";
2771 :
2772 23 : case wkbMultiPoint:
2773 23 : if (b3D && bMeasured)
2774 0 : return "3D Measured Multi Point";
2775 23 : else if (b3D)
2776 9 : return "3D Multi Point";
2777 14 : else if (bMeasured)
2778 0 : return "Measured Multi Point";
2779 : else
2780 14 : return "Multi Point";
2781 :
2782 14 : case wkbMultiLineString:
2783 14 : if (b3D && bMeasured)
2784 0 : return "3D Measured Multi Line String";
2785 14 : else if (b3D)
2786 6 : return "3D Multi Line String";
2787 8 : else if (bMeasured)
2788 0 : return "Measured Multi Line String";
2789 : else
2790 8 : return "Multi Line String";
2791 :
2792 19 : case wkbMultiPolygon:
2793 19 : if (b3D && bMeasured)
2794 0 : return "3D Measured Multi Polygon";
2795 19 : else if (b3D)
2796 8 : return "3D Multi Polygon";
2797 11 : else if (bMeasured)
2798 0 : return "Measured Multi Polygon";
2799 : else
2800 11 : return "Multi Polygon";
2801 :
2802 25 : case wkbGeometryCollection:
2803 25 : if (b3D && bMeasured)
2804 0 : return "3D Measured Geometry Collection";
2805 25 : else if (b3D)
2806 10 : return "3D Geometry Collection";
2807 15 : else if (bMeasured)
2808 0 : return "Measured Geometry Collection";
2809 : else
2810 15 : return "Geometry Collection";
2811 :
2812 0 : case wkbCircularString:
2813 0 : if (b3D && bMeasured)
2814 0 : return "3D Measured Circular String";
2815 0 : else if (b3D)
2816 0 : return "3D Circular String";
2817 0 : else if (bMeasured)
2818 0 : return "Measured Circular String";
2819 : else
2820 0 : return "Circular String";
2821 :
2822 1 : case wkbCompoundCurve:
2823 1 : if (b3D && bMeasured)
2824 0 : return "3D Measured Compound Curve";
2825 1 : else if (b3D)
2826 0 : return "3D Compound Curve";
2827 1 : else if (bMeasured)
2828 0 : return "Measured Compound Curve";
2829 : else
2830 1 : return "Compound Curve";
2831 :
2832 0 : case wkbCurvePolygon:
2833 0 : if (b3D && bMeasured)
2834 0 : return "3D Measured Curve Polygon";
2835 0 : else if (b3D)
2836 0 : return "3D Curve Polygon";
2837 0 : else if (bMeasured)
2838 0 : return "Measured Curve Polygon";
2839 : else
2840 0 : return "Curve Polygon";
2841 :
2842 0 : case wkbMultiCurve:
2843 0 : if (b3D && bMeasured)
2844 0 : return "3D Measured Multi Curve";
2845 0 : else if (b3D)
2846 0 : return "3D Multi Curve";
2847 0 : else if (bMeasured)
2848 0 : return "Measured Multi Curve";
2849 : else
2850 0 : return "Multi Curve";
2851 :
2852 0 : case wkbMultiSurface:
2853 0 : if (b3D && bMeasured)
2854 0 : return "3D Measured Multi Surface";
2855 0 : else if (b3D)
2856 0 : return "3D Multi Surface";
2857 0 : else if (bMeasured)
2858 0 : return "Measured Multi Surface";
2859 : else
2860 0 : return "Multi Surface";
2861 :
2862 4 : case wkbCurve:
2863 4 : if (b3D && bMeasured)
2864 1 : return "3D Measured Curve";
2865 3 : else if (b3D)
2866 1 : return "3D Curve";
2867 2 : else if (bMeasured)
2868 1 : return "Measured Curve";
2869 : else
2870 1 : return "Curve";
2871 :
2872 4 : case wkbSurface:
2873 4 : if (b3D && bMeasured)
2874 1 : return "3D Measured Surface";
2875 3 : else if (b3D)
2876 1 : return "3D Surface";
2877 2 : else if (bMeasured)
2878 1 : return "Measured Surface";
2879 : else
2880 1 : return "Surface";
2881 :
2882 0 : case wkbTriangle:
2883 0 : if (b3D && bMeasured)
2884 0 : return "3D Measured Triangle";
2885 0 : else if (b3D)
2886 0 : return "3D Triangle";
2887 0 : else if (bMeasured)
2888 0 : return "Measured Triangle";
2889 : else
2890 0 : return "Triangle";
2891 :
2892 0 : case wkbPolyhedralSurface:
2893 0 : if (b3D && bMeasured)
2894 0 : return "3D Measured PolyhedralSurface";
2895 0 : else if (b3D)
2896 0 : return "3D PolyhedralSurface";
2897 0 : else if (bMeasured)
2898 0 : return "Measured PolyhedralSurface";
2899 : else
2900 0 : return "PolyhedralSurface";
2901 :
2902 2 : case wkbTIN:
2903 2 : if (b3D && bMeasured)
2904 0 : return "3D Measured TIN";
2905 2 : else if (b3D)
2906 0 : return "3D TIN";
2907 2 : else if (bMeasured)
2908 0 : return "Measured TIN";
2909 : else
2910 2 : return "TIN";
2911 :
2912 111 : case wkbNone:
2913 111 : return "None";
2914 :
2915 1 : default:
2916 : {
2917 1 : return CPLSPrintf("Unrecognized: %d", static_cast<int>(eType));
2918 : }
2919 : }
2920 : }
2921 :
2922 : /************************************************************************/
2923 : /* OGRMergeGeometryTypes() */
2924 : /************************************************************************/
2925 :
2926 : /**
2927 : * \brief Find common geometry type.
2928 : *
2929 : * Given two geometry types, find the most specific common
2930 : * type. Normally used repeatedly with the geometries in a
2931 : * layer to try and establish the most specific geometry type
2932 : * that can be reported for the layer.
2933 : *
2934 : * NOTE: wkbUnknown is the "worst case" indicating a mixture of
2935 : * geometry types with nothing in common but the base geometry
2936 : * type. wkbNone should be used to indicate that no geometries
2937 : * have been encountered yet, and means the first geometry
2938 : * encountered will establish the preliminary type.
2939 : *
2940 : * @param eMain the first input geometry type.
2941 : * @param eExtra the second input geometry type.
2942 : *
2943 : * @return the merged geometry type.
2944 : */
2945 :
2946 0 : OGRwkbGeometryType OGRMergeGeometryTypes(OGRwkbGeometryType eMain,
2947 : OGRwkbGeometryType eExtra)
2948 :
2949 : {
2950 0 : return OGRMergeGeometryTypesEx(eMain, eExtra, FALSE);
2951 : }
2952 :
2953 : /**
2954 : * \brief Find common geometry type.
2955 : *
2956 : * Given two geometry types, find the most specific common
2957 : * type. Normally used repeatedly with the geometries in a
2958 : * layer to try and establish the most specific geometry type
2959 : * that can be reported for the layer.
2960 : *
2961 : * NOTE: wkbUnknown is the "worst case" indicating a mixture of
2962 : * geometry types with nothing in common but the base geometry
2963 : * type. wkbNone should be used to indicate that no geometries
2964 : * have been encountered yet, and means the first geometry
2965 : * encountered will establish the preliminary type.
2966 : *
2967 : * If bAllowPromotingToCurves is set to TRUE, mixing Polygon and CurvePolygon
2968 : * will return CurvePolygon. Mixing LineString, CircularString, CompoundCurve
2969 : * will return CompoundCurve. Mixing MultiPolygon and MultiSurface will return
2970 : * MultiSurface. Mixing MultiCurve and MultiLineString will return MultiCurve.
2971 : *
2972 : * @param eMain the first input geometry type.
2973 : * @param eExtra the second input geometry type.
2974 : * @param bAllowPromotingToCurves determine if promotion to curve type
2975 : * must be done.
2976 : *
2977 : * @return the merged geometry type.
2978 : *
2979 : * @since GDAL 2.0
2980 : */
2981 :
2982 398 : OGRwkbGeometryType OGRMergeGeometryTypesEx(OGRwkbGeometryType eMain,
2983 : OGRwkbGeometryType eExtra,
2984 : int bAllowPromotingToCurves)
2985 :
2986 : {
2987 398 : OGRwkbGeometryType eFMain = wkbFlatten(eMain);
2988 398 : OGRwkbGeometryType eFExtra = wkbFlatten(eExtra);
2989 :
2990 398 : const bool bHasZ = (wkbHasZ(eMain) || wkbHasZ(eExtra));
2991 398 : const bool bHasM = (wkbHasM(eMain) || wkbHasM(eExtra));
2992 :
2993 398 : if (eFMain == wkbUnknown || eFExtra == wkbUnknown)
2994 17 : return OGR_GT_SetModifier(wkbUnknown, bHasZ, bHasM);
2995 :
2996 381 : if (eFMain == wkbNone)
2997 2 : return eExtra;
2998 :
2999 379 : if (eFExtra == wkbNone)
3000 0 : return eMain;
3001 :
3002 379 : if (eFMain == eFExtra)
3003 : {
3004 362 : return OGR_GT_SetModifier(eFMain, bHasZ, bHasM);
3005 : }
3006 :
3007 17 : if (bAllowPromotingToCurves)
3008 : {
3009 17 : if (OGR_GT_IsCurve(eFMain) && OGR_GT_IsCurve(eFExtra))
3010 4 : return OGR_GT_SetModifier(wkbCompoundCurve, bHasZ, bHasM);
3011 :
3012 13 : if (OGR_GT_IsSubClassOf(eFMain, eFExtra))
3013 3 : return OGR_GT_SetModifier(eFExtra, bHasZ, bHasM);
3014 :
3015 10 : if (OGR_GT_IsSubClassOf(eFExtra, eFMain))
3016 6 : return OGR_GT_SetModifier(eFMain, bHasZ, bHasM);
3017 : }
3018 :
3019 : // One is subclass of the other one
3020 4 : if (OGR_GT_IsSubClassOf(eFMain, eFExtra))
3021 : {
3022 0 : return OGR_GT_SetModifier(eFExtra, bHasZ, bHasM);
3023 : }
3024 4 : else if (OGR_GT_IsSubClassOf(eFExtra, eFMain))
3025 : {
3026 0 : return OGR_GT_SetModifier(eFMain, bHasZ, bHasM);
3027 : }
3028 :
3029 : // Nothing apparently in common.
3030 4 : return OGR_GT_SetModifier(wkbUnknown, bHasZ, bHasM);
3031 : }
3032 :
3033 : /**
3034 : * \fn void OGRGeometry::flattenTo2D();
3035 : *
3036 : * \brief Convert geometry to strictly 2D.
3037 : * In a sense this converts all Z coordinates
3038 : * to 0.0.
3039 : *
3040 : * This method is the same as the C function OGR_G_FlattenTo2D().
3041 : */
3042 :
3043 : /************************************************************************/
3044 : /* OGR_G_FlattenTo2D() */
3045 : /************************************************************************/
3046 : /**
3047 : * \brief Convert geometry to strictly 2D.
3048 : * In a sense this converts all Z coordinates
3049 : * to 0.0.
3050 : *
3051 : * This function is the same as the CPP method OGRGeometry::flattenTo2D().
3052 : *
3053 : * @param hGeom handle on the geometry to convert.
3054 : */
3055 :
3056 31 : void OGR_G_FlattenTo2D(OGRGeometryH hGeom)
3057 :
3058 : {
3059 31 : OGRGeometry::FromHandle(hGeom)->flattenTo2D();
3060 31 : }
3061 :
3062 : /************************************************************************/
3063 : /* exportToGML() */
3064 : /************************************************************************/
3065 :
3066 : /**
3067 : * \fn char *OGRGeometry::exportToGML( const char* const *
3068 : * papszOptions = NULL ) const;
3069 : *
3070 : * \brief Convert a geometry into GML format.
3071 : *
3072 : * The GML geometry is expressed directly in terms of GML basic data
3073 : * types assuming the this is available in the gml namespace. The returned
3074 : * string should be freed with CPLFree() when no longer required.
3075 : *
3076 : * The supported options are :
3077 : * <ul>
3078 : * <li> FORMAT=GML2/GML3/GML32 (GML2 or GML32 added in GDAL 2.1).
3079 : * If not set, it will default to GML 2.1.2 output.
3080 : * </li>
3081 : * <li> GML3_LINESTRING_ELEMENT=curve. (Only valid for FORMAT=GML3)
3082 : * To use gml:Curve element for linestrings.
3083 : * Otherwise gml:LineString will be used .
3084 : * </li>
3085 : * <li> GML3_LONGSRS=YES/NO. (Only valid for FORMAT=GML3, deprecated by
3086 : * SRSNAME_FORMAT in GDAL >=2.2). Defaults to YES.
3087 : * If YES, SRS with EPSG authority will be written with the
3088 : * "urn:ogc:def:crs:EPSG::" prefix.
3089 : * In the case the SRS should be treated as lat/long or
3090 : * northing/easting, then the function will take care of coordinate order
3091 : * swapping if the data axis to CRS axis mapping indicates it.
3092 : * If set to NO, SRS with EPSG authority will be written with the "EPSG:"
3093 : * prefix, even if they are in lat/long order.
3094 : * </li>
3095 : * <li> SRSNAME_FORMAT=SHORT/OGC_URN/OGC_URL (Only valid for FORMAT=GML3, added
3096 : * in GDAL 2.2). Defaults to OGC_URN. If SHORT, then srsName will be in
3097 : * the form AUTHORITY_NAME:AUTHORITY_CODE. If OGC_URN, then srsName will be
3098 : * in the form urn:ogc:def:crs:AUTHORITY_NAME::AUTHORITY_CODE. If OGC_URL,
3099 : * then srsName will be in the form
3100 : * http://www.opengis.net/def/crs/AUTHORITY_NAME/0/AUTHORITY_CODE. For
3101 : * OGC_URN and OGC_URL, in the case the SRS should be treated as lat/long
3102 : * or northing/easting, then the function will take care of coordinate
3103 : * order swapping if the data axis to CRS axis mapping indicates it.
3104 : * </li>
3105 : * <li> GMLID=astring. If specified, a gml:id attribute will be written in the
3106 : * top-level geometry element with the provided value.
3107 : * Required for GML 3.2 compatibility.
3108 : * </li>
3109 : * <li> SRSDIMENSION_LOC=POSLIST/GEOMETRY/GEOMETRY,POSLIST. (Only valid for
3110 : * FORMAT=GML3/GML32, GDAL >= 2.0) Default to POSLIST.
3111 : * For 2.5D geometries, define the location where to attach the
3112 : * srsDimension attribute.
3113 : * There are diverging implementations. Some put in on the
3114 : * <gml:posList> element, other on the top geometry element.
3115 : * </li>
3116 : * <li> NAMESPACE_DECL=YES/NO. If set to YES,
3117 : * xmlns:gml="http://www.opengis.net/gml" will be added to the root node
3118 : * for GML < 3.2 or xmlns:gml="http://www.opengis.net/gml/3.2" for GML 3.2
3119 : * </li>
3120 : * <li> XY_COORD_RESOLUTION=double (added in GDAL 3.9):
3121 : * Resolution for the coordinate precision of the X and Y coordinates.
3122 : * Expressed in the units of the X and Y axis of the SRS. eg 1e-5 for up
3123 : * to 5 decimal digits. 0 for the default behavior.
3124 : * </li>
3125 : * <li> Z_COORD_RESOLUTION=double (added in GDAL 3.9):
3126 : * Resolution for the coordinate precision of the Z coordinates.
3127 : * Expressed in the units of the Z axis of the SRS.
3128 : * 0 for the default behavior.
3129 : * </li>
3130 : * </ul>
3131 : *
3132 : * This method is the same as the C function OGR_G_ExportToGMLEx().
3133 : *
3134 : * @param papszOptions NULL-terminated list of options.
3135 : * @return A GML fragment to be freed with CPLFree() or NULL in case of error.
3136 : */
3137 :
3138 251 : char *OGRGeometry::exportToGML(const char *const *papszOptions) const
3139 : {
3140 251 : return OGR_G_ExportToGMLEx(
3141 : OGRGeometry::ToHandle(const_cast<OGRGeometry *>(this)),
3142 251 : const_cast<char **>(papszOptions));
3143 : }
3144 :
3145 : /************************************************************************/
3146 : /* exportToKML() */
3147 : /************************************************************************/
3148 :
3149 : /**
3150 : * \fn char *OGRGeometry::exportToKML() const;
3151 : *
3152 : * \brief Convert a geometry into KML format.
3153 : *
3154 : * The returned string should be freed with CPLFree() when no longer required.
3155 : *
3156 : * This method is the same as the C function OGR_G_ExportToKML().
3157 : *
3158 : * @return A KML fragment to be freed with CPLFree() or NULL in case of error.
3159 : */
3160 :
3161 0 : char *OGRGeometry::exportToKML() const
3162 : {
3163 0 : return OGR_G_ExportToKML(
3164 0 : OGRGeometry::ToHandle(const_cast<OGRGeometry *>(this)), nullptr);
3165 : }
3166 :
3167 : /************************************************************************/
3168 : /* exportToJson() */
3169 : /************************************************************************/
3170 :
3171 : /**
3172 : * \fn char *OGRGeometry::exportToJson() const;
3173 : *
3174 : * \brief Convert a geometry into GeoJSON format.
3175 : *
3176 : * The returned string should be freed with CPLFree() when no longer required.
3177 : *
3178 : * The following options are supported :
3179 : * <ul>
3180 : * <li>XY_COORD_PRECISION=integer: number of decimal figures for X,Y coordinates
3181 : * (added in GDAL 3.9)</li>
3182 : * <li>Z_COORD_PRECISION=integer: number of decimal figures for Z coordinates
3183 : * (added in GDAL 3.9)</li>
3184 : * </ul>
3185 : *
3186 : * This method is the same as the C function OGR_G_ExportToJson().
3187 : *
3188 : * @param papszOptions Null terminated list of options, or null (added in 3.9)
3189 : * @return A GeoJSON fragment to be freed with CPLFree() or NULL in case of error.
3190 : */
3191 :
3192 40 : char *OGRGeometry::exportToJson(CSLConstList papszOptions) const
3193 : {
3194 40 : OGRGeometry *poGeometry = const_cast<OGRGeometry *>(this);
3195 40 : return OGR_G_ExportToJsonEx(OGRGeometry::ToHandle(poGeometry),
3196 40 : const_cast<char **>(papszOptions));
3197 : }
3198 :
3199 : /************************************************************************/
3200 : /* OGRSetGenerate_DB2_V72_BYTE_ORDER() */
3201 : /************************************************************************/
3202 :
3203 : /**
3204 : * \brief Special entry point to enable the hack for generating DB2 V7.2 style
3205 : * WKB.
3206 : *
3207 : * DB2 seems to have placed (and require) an extra 0x30 or'ed with the byte
3208 : * order in WKB. This entry point is used to turn on or off the generation of
3209 : * such WKB.
3210 : */
3211 4 : OGRErr OGRSetGenerate_DB2_V72_BYTE_ORDER(int bGenerate_DB2_V72_BYTE_ORDER)
3212 :
3213 : {
3214 : #if defined(HACK_FOR_IBM_DB2_V72)
3215 4 : OGRGeometry::bGenerate_DB2_V72_BYTE_ORDER = bGenerate_DB2_V72_BYTE_ORDER;
3216 4 : return OGRERR_NONE;
3217 : #else
3218 : if (bGenerate_DB2_V72_BYTE_ORDER)
3219 : return OGRERR_FAILURE;
3220 : else
3221 : return OGRERR_NONE;
3222 : #endif
3223 : }
3224 :
3225 : /************************************************************************/
3226 : /* OGRGetGenerate_DB2_V72_BYTE_ORDER() */
3227 : /* */
3228 : /* This is a special entry point to get the value of static flag */
3229 : /* OGRGeometry::bGenerate_DB2_V72_BYTE_ORDER. */
3230 : /************************************************************************/
3231 0 : int OGRGetGenerate_DB2_V72_BYTE_ORDER()
3232 : {
3233 0 : return OGRGeometry::bGenerate_DB2_V72_BYTE_ORDER;
3234 : }
3235 :
3236 : /************************************************************************/
3237 : /* createGEOSContext() */
3238 : /************************************************************************/
3239 :
3240 : /** Create a new GEOS context.
3241 : * @return a new GEOS context (to be freed with freeGEOSContext())
3242 : */
3243 75311 : GEOSContextHandle_t OGRGeometry::createGEOSContext()
3244 : {
3245 : #ifndef HAVE_GEOS
3246 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
3247 : return nullptr;
3248 : #else
3249 75311 : return initGEOS_r(OGRGEOSWarningHandler, OGRGEOSErrorHandler);
3250 : #endif
3251 : }
3252 :
3253 : /************************************************************************/
3254 : /* freeGEOSContext() */
3255 : /************************************************************************/
3256 :
3257 : /** Destroy a GEOS context.
3258 : * @param hGEOSCtxt GEOS context
3259 : */
3260 76992 : void OGRGeometry::freeGEOSContext(
3261 : UNUSED_IF_NO_GEOS GEOSContextHandle_t hGEOSCtxt)
3262 : {
3263 : #ifdef HAVE_GEOS
3264 76992 : if (hGEOSCtxt != nullptr)
3265 : {
3266 76968 : finishGEOS_r(hGEOSCtxt);
3267 : }
3268 : #endif
3269 76995 : }
3270 :
3271 : #ifdef HAVE_GEOS
3272 :
3273 : /************************************************************************/
3274 : /* convertToGEOSGeom() */
3275 : /************************************************************************/
3276 :
3277 222603 : static GEOSGeom convertToGEOSGeom(GEOSContextHandle_t hGEOSCtxt,
3278 : OGRGeometry *poGeom)
3279 : {
3280 222603 : GEOSGeom hGeom = nullptr;
3281 222603 : const size_t nDataSize = poGeom->WkbSize();
3282 : unsigned char *pabyData =
3283 222573 : static_cast<unsigned char *>(CPLMalloc(nDataSize));
3284 : #if GEOS_VERSION_MAJOR > 3 || \
3285 : (GEOS_VERSION_MAJOR == 3 && GEOS_VERSION_MINOR >= 12)
3286 : OGRwkbVariant eWkbVariant = wkbVariantIso;
3287 : #else
3288 222588 : OGRwkbVariant eWkbVariant = wkbVariantOldOgc;
3289 : #endif
3290 222588 : if (poGeom->exportToWkb(wkbNDR, pabyData, eWkbVariant) == OGRERR_NONE)
3291 221708 : hGeom = GEOSGeomFromWKB_buf_r(hGEOSCtxt, pabyData, nDataSize);
3292 222610 : CPLFree(pabyData);
3293 222616 : return hGeom;
3294 : }
3295 : #endif
3296 :
3297 : /************************************************************************/
3298 : /* exportToGEOS() */
3299 : /************************************************************************/
3300 :
3301 : /** Returns a GEOSGeom object corresponding to the geometry.
3302 : *
3303 : * @param hGEOSCtxt GEOS context
3304 : * @param bRemoveEmptyParts Whether empty parts of the geometry should be
3305 : * removed before exporting to GEOS (GDAL >= 3.10)
3306 : * @return a GEOSGeom object corresponding to the geometry (to be freed with
3307 : * GEOSGeom_destroy_r()), or NULL in case of error
3308 : */
3309 : GEOSGeom
3310 222697 : OGRGeometry::exportToGEOS(UNUSED_IF_NO_GEOS GEOSContextHandle_t hGEOSCtxt,
3311 : UNUSED_IF_NO_GEOS bool bRemoveEmptyParts) const
3312 :
3313 : {
3314 : #ifndef HAVE_GEOS
3315 :
3316 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
3317 : return nullptr;
3318 :
3319 : #else
3320 :
3321 222697 : if (hGEOSCtxt == nullptr)
3322 0 : return nullptr;
3323 :
3324 222697 : const OGRwkbGeometryType eType = wkbFlatten(getGeometryType());
3325 : #if (GEOS_VERSION_MAJOR == 3 && GEOS_VERSION_MINOR < 12)
3326 : // POINT EMPTY is exported to WKB as if it were POINT(0 0),
3327 : // so that particular case is necessary.
3328 222674 : if (eType == wkbPoint && IsEmpty())
3329 : {
3330 98 : return GEOSGeomFromWKT_r(hGEOSCtxt, "POINT EMPTY");
3331 : }
3332 : #endif
3333 :
3334 222576 : GEOSGeom hGeom = nullptr;
3335 :
3336 222576 : OGRGeometry *poLinearGeom = nullptr;
3337 222576 : if (hasCurveGeometry())
3338 : {
3339 836 : poLinearGeom = getLinearGeometry();
3340 836 : if (bRemoveEmptyParts)
3341 0 : poLinearGeom->removeEmptyParts();
3342 : #if (GEOS_VERSION_MAJOR == 3 && GEOS_VERSION_MINOR < 12)
3343 : // GEOS < 3.12 doesn't support M dimension
3344 836 : if (poLinearGeom->IsMeasured())
3345 64 : poLinearGeom->setMeasured(FALSE);
3346 : #endif
3347 : }
3348 : else
3349 : {
3350 221761 : poLinearGeom = const_cast<OGRGeometry *>(this);
3351 : #if (GEOS_VERSION_MAJOR == 3 && GEOS_VERSION_MINOR < 12)
3352 : // GEOS < 3.12 doesn't support M dimension
3353 221761 : if (IsMeasured())
3354 : {
3355 290 : poLinearGeom = clone();
3356 295 : if (bRemoveEmptyParts)
3357 0 : poLinearGeom->removeEmptyParts();
3358 295 : poLinearGeom->setMeasured(FALSE);
3359 : }
3360 : else
3361 : #endif
3362 221445 : if (bRemoveEmptyParts && hasEmptyParts())
3363 : {
3364 1 : poLinearGeom = clone();
3365 1 : poLinearGeom->removeEmptyParts();
3366 : }
3367 : }
3368 222575 : if (eType == wkbTriangle)
3369 : {
3370 85 : OGRPolygon oPolygon(*(poLinearGeom->toPolygon()));
3371 85 : hGeom = convertToGEOSGeom(hGEOSCtxt, &oPolygon);
3372 : }
3373 222490 : else if (eType == wkbPolyhedralSurface || eType == wkbTIN)
3374 : {
3375 1707 : OGRGeometry *poGC = OGRGeometryFactory::forceTo(
3376 848 : poLinearGeom->clone(),
3377 : OGR_GT_SetModifier(wkbGeometryCollection, poLinearGeom->Is3D(),
3378 : poLinearGeom->IsMeasured()),
3379 : nullptr);
3380 848 : hGeom = convertToGEOSGeom(hGEOSCtxt, poGC);
3381 848 : delete poGC;
3382 : }
3383 221631 : else if (eType == wkbGeometryCollection)
3384 : {
3385 124 : bool bCanConvertToMultiPoly = true;
3386 : // bool bMustConvertToMultiPoly = true;
3387 : const OGRGeometryCollection *poGC =
3388 124 : poLinearGeom->toGeometryCollection();
3389 274 : for (int iGeom = 0; iGeom < poGC->getNumGeometries(); iGeom++)
3390 : {
3391 : const OGRwkbGeometryType eSubGeomType =
3392 220 : wkbFlatten(poGC->getGeometryRef(iGeom)->getGeometryType());
3393 220 : if (eSubGeomType == wkbPolyhedralSurface || eSubGeomType == wkbTIN)
3394 : {
3395 : // bMustConvertToMultiPoly = true;
3396 : }
3397 135 : else if (eSubGeomType != wkbMultiPolygon &&
3398 : eSubGeomType != wkbPolygon)
3399 : {
3400 70 : bCanConvertToMultiPoly = false;
3401 70 : break;
3402 : }
3403 : }
3404 124 : if (bCanConvertToMultiPoly /* && bMustConvertToMultiPoly */)
3405 : {
3406 108 : OGRGeometry *poMultiPolygon = OGRGeometryFactory::forceTo(
3407 54 : poLinearGeom->clone(),
3408 : OGR_GT_SetModifier(wkbMultiPolygon, poLinearGeom->Is3D(),
3409 : poLinearGeom->IsMeasured()),
3410 : nullptr);
3411 54 : OGRGeometry *poGCDest = OGRGeometryFactory::forceTo(
3412 : poMultiPolygon,
3413 : OGR_GT_SetModifier(wkbGeometryCollection, poLinearGeom->Is3D(),
3414 : poLinearGeom->IsMeasured()),
3415 : nullptr);
3416 54 : hGeom = convertToGEOSGeom(hGEOSCtxt, poGCDest);
3417 54 : delete poGCDest;
3418 : }
3419 : else
3420 : {
3421 70 : hGeom = convertToGEOSGeom(hGEOSCtxt, poLinearGeom);
3422 : }
3423 : }
3424 : else
3425 : {
3426 221507 : hGeom = convertToGEOSGeom(hGEOSCtxt, poLinearGeom);
3427 : }
3428 :
3429 222605 : if (poLinearGeom != this)
3430 1132 : delete poLinearGeom;
3431 :
3432 222570 : return hGeom;
3433 :
3434 : #endif // HAVE_GEOS
3435 : }
3436 :
3437 : /************************************************************************/
3438 : /* hasCurveGeometry() */
3439 : /************************************************************************/
3440 :
3441 : /**
3442 : * \brief Returns if this geometry is or has curve geometry.
3443 : *
3444 : * Returns if a geometry is, contains or may contain a CIRCULARSTRING,
3445 : * COMPOUNDCURVE, CURVEPOLYGON, MULTICURVE or MULTISURFACE.
3446 : *
3447 : * If bLookForNonLinear is set to TRUE, it will be actually looked if
3448 : * the geometry or its subgeometries are or contain a non-linear
3449 : * geometry in them. In which case, if the method returns TRUE, it
3450 : * means that getLinearGeometry() would return an approximate version
3451 : * of the geometry. Otherwise, getLinearGeometry() would do a
3452 : * conversion, but with just converting container type, like
3453 : * COMPOUNDCURVE -> LINESTRING, MULTICURVE -> MULTILINESTRING or
3454 : * MULTISURFACE -> MULTIPOLYGON, resulting in a "loss-less"
3455 : * conversion.
3456 : *
3457 : * This method is the same as the C function OGR_G_HasCurveGeometry().
3458 : *
3459 : * @param bLookForNonLinear set it to TRUE to check if the geometry is
3460 : * or contains a CIRCULARSTRING.
3461 : *
3462 : * @return TRUE if this geometry is or has curve geometry.
3463 : *
3464 : * @since GDAL 2.0
3465 : */
3466 :
3467 273694 : OGRBoolean OGRGeometry::hasCurveGeometry(CPL_UNUSED int bLookForNonLinear) const
3468 : {
3469 273694 : return FALSE;
3470 : }
3471 :
3472 : /************************************************************************/
3473 : /* getLinearGeometry() */
3474 : /************************************************************************/
3475 :
3476 : /**
3477 : * \brief Return, possibly approximate, non-curve version of this geometry.
3478 : *
3479 : * Returns a geometry that has no CIRCULARSTRING, COMPOUNDCURVE, CURVEPOLYGON,
3480 : * MULTICURVE or MULTISURFACE in it, by approximating curve geometries.
3481 : *
3482 : * The ownership of the returned geometry belongs to the caller.
3483 : *
3484 : * The reverse method is OGRGeometry::getCurveGeometry().
3485 : *
3486 : * This method is the same as the C function OGR_G_GetLinearGeometry().
3487 : *
3488 : * @param dfMaxAngleStepSizeDegrees the largest step in degrees along the
3489 : * arc, zero to use the default setting.
3490 : * @param papszOptions options as a null-terminated list of strings.
3491 : * See OGRGeometryFactory::curveToLineString() for
3492 : * valid options.
3493 : *
3494 : * @return a new geometry to be freed by the caller, or NULL if an error occurs.
3495 : *
3496 : * @since GDAL 2.0
3497 : */
3498 :
3499 : OGRGeometry *
3500 79 : OGRGeometry::getLinearGeometry(CPL_UNUSED double dfMaxAngleStepSizeDegrees,
3501 : CPL_UNUSED const char *const *papszOptions) const
3502 : {
3503 79 : return clone();
3504 : }
3505 :
3506 : /************************************************************************/
3507 : /* getCurveGeometry() */
3508 : /************************************************************************/
3509 :
3510 : /**
3511 : * \brief Return curve version of this geometry.
3512 : *
3513 : * Returns a geometry that has possibly CIRCULARSTRING, COMPOUNDCURVE,
3514 : * CURVEPOLYGON, MULTICURVE or MULTISURFACE in it, by de-approximating
3515 : * curve geometries.
3516 : *
3517 : * If the geometry has no curve portion, the returned geometry will be a clone
3518 : * of it.
3519 : *
3520 : * The ownership of the returned geometry belongs to the caller.
3521 : *
3522 : * The reverse method is OGRGeometry::getLinearGeometry().
3523 : *
3524 : * This function is the same as C function OGR_G_GetCurveGeometry().
3525 : *
3526 : * @param papszOptions options as a null-terminated list of strings.
3527 : * Unused for now. Must be set to NULL.
3528 : *
3529 : * @return a new geometry to be freed by the caller, or NULL if an error occurs.
3530 : *
3531 : * @since GDAL 2.0
3532 : */
3533 :
3534 : OGRGeometry *
3535 5 : OGRGeometry::getCurveGeometry(CPL_UNUSED const char *const *papszOptions) const
3536 : {
3537 5 : return clone();
3538 : }
3539 :
3540 : /************************************************************************/
3541 : /* Distance() */
3542 : /************************************************************************/
3543 :
3544 : /**
3545 : * \brief Compute distance between two geometries.
3546 : *
3547 : * Returns the shortest distance between the two geometries. The distance is
3548 : * expressed into the same unit as the coordinates of the geometries.
3549 : *
3550 : * This method is the same as the C function OGR_G_Distance().
3551 : *
3552 : * This method is built on the GEOS library, check it for the definition
3553 : * of the geometry operation.
3554 : * If OGR is built without the GEOS library, this method will always fail,
3555 : * issuing a CPLE_NotSupported error.
3556 : *
3557 : * @param poOtherGeom the other geometry to compare against.
3558 : *
3559 : * @return the distance between the geometries or -1 if an error occurs.
3560 : */
3561 :
3562 24 : double OGRGeometry::Distance(const OGRGeometry *poOtherGeom) const
3563 :
3564 : {
3565 24 : if (nullptr == poOtherGeom)
3566 : {
3567 0 : CPLDebug("OGR",
3568 : "OGRGeometry::Distance called with NULL geometry pointer");
3569 0 : return -1.0;
3570 : }
3571 :
3572 24 : if (IsSFCGALCompatible() || poOtherGeom->IsSFCGALCompatible())
3573 : {
3574 : #ifndef HAVE_SFCGAL
3575 :
3576 0 : CPLError(CE_Failure, CPLE_NotSupported, "SFCGAL support not enabled.");
3577 0 : return -1.0;
3578 :
3579 : #else
3580 :
3581 : sfcgal_geometry_t *poThis = OGRGeometry::OGRexportToSFCGAL(this);
3582 : if (poThis == nullptr)
3583 : return -1.0;
3584 :
3585 : sfcgal_geometry_t *poOther =
3586 : OGRGeometry::OGRexportToSFCGAL(poOtherGeom);
3587 : if (poOther == nullptr)
3588 : {
3589 : sfcgal_geometry_delete(poThis);
3590 : return -1.0;
3591 : }
3592 :
3593 : const double dfDistance = sfcgal_geometry_distance(poThis, poOther);
3594 :
3595 : sfcgal_geometry_delete(poThis);
3596 : sfcgal_geometry_delete(poOther);
3597 :
3598 : return dfDistance > 0.0 ? dfDistance : -1.0;
3599 :
3600 : #endif
3601 : }
3602 :
3603 : else
3604 : {
3605 : #ifndef HAVE_GEOS
3606 :
3607 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
3608 : return -1.0;
3609 :
3610 : #else
3611 :
3612 24 : GEOSContextHandle_t hGEOSCtxt = createGEOSContext();
3613 : // GEOSGeom is a pointer
3614 24 : GEOSGeom hOther = poOtherGeom->exportToGEOS(hGEOSCtxt);
3615 24 : GEOSGeom hThis = exportToGEOS(hGEOSCtxt);
3616 :
3617 24 : int bIsErr = 0;
3618 24 : double dfDistance = 0.0;
3619 :
3620 24 : if (hThis != nullptr && hOther != nullptr)
3621 : {
3622 24 : bIsErr = GEOSDistance_r(hGEOSCtxt, hThis, hOther, &dfDistance);
3623 : }
3624 :
3625 24 : GEOSGeom_destroy_r(hGEOSCtxt, hThis);
3626 24 : GEOSGeom_destroy_r(hGEOSCtxt, hOther);
3627 24 : freeGEOSContext(hGEOSCtxt);
3628 :
3629 24 : if (bIsErr > 0)
3630 : {
3631 24 : return dfDistance;
3632 : }
3633 :
3634 : /* Calculations error */
3635 0 : return -1.0;
3636 :
3637 : #endif /* HAVE_GEOS */
3638 : }
3639 : }
3640 :
3641 : /************************************************************************/
3642 : /* OGR_G_Distance() */
3643 : /************************************************************************/
3644 : /**
3645 : * \brief Compute distance between two geometries.
3646 : *
3647 : * Returns the shortest distance between the two geometries. The distance is
3648 : * expressed into the same unit as the coordinates of the geometries.
3649 : *
3650 : * This function is the same as the C++ method OGRGeometry::Distance().
3651 : *
3652 : * This function is built on the GEOS library, check it for the definition
3653 : * of the geometry operation.
3654 : * If OGR is built without the GEOS library, this function will always fail,
3655 : * issuing a CPLE_NotSupported error.
3656 : *
3657 : * @param hFirst the first geometry to compare against.
3658 : * @param hOther the other geometry to compare against.
3659 : *
3660 : * @return the distance between the geometries or -1 if an error occurs.
3661 : */
3662 :
3663 1 : double OGR_G_Distance(OGRGeometryH hFirst, OGRGeometryH hOther)
3664 :
3665 : {
3666 1 : VALIDATE_POINTER1(hFirst, "OGR_G_Distance", 0.0);
3667 :
3668 2 : return OGRGeometry::FromHandle(hFirst)->Distance(
3669 1 : OGRGeometry::FromHandle(hOther));
3670 : }
3671 :
3672 : /************************************************************************/
3673 : /* Distance3D() */
3674 : /************************************************************************/
3675 :
3676 : /**
3677 : * \brief Returns the 3D distance between two geometries
3678 : *
3679 : * The distance is expressed into the same unit as the coordinates of the
3680 : * geometries.
3681 : *
3682 : * This method is built on the SFCGAL library, check it for the definition
3683 : * of the geometry operation.
3684 : * If OGR is built without the SFCGAL library, this method will always return
3685 : * -1.0
3686 : *
3687 : * This function is the same as the C function OGR_G_Distance3D().
3688 : *
3689 : * @return distance between the two geometries
3690 : * @since GDAL 2.2
3691 : */
3692 :
3693 1 : double OGRGeometry::Distance3D(
3694 : UNUSED_IF_NO_SFCGAL const OGRGeometry *poOtherGeom) const
3695 : {
3696 1 : if (poOtherGeom == nullptr)
3697 : {
3698 0 : CPLDebug("OGR",
3699 : "OGRTriangle::Distance3D called with NULL geometry pointer");
3700 0 : return -1.0;
3701 : }
3702 :
3703 1 : if (!(poOtherGeom->Is3D() && Is3D()))
3704 : {
3705 0 : CPLDebug("OGR", "OGRGeometry::Distance3D called with two dimensional "
3706 : "geometry(geometries)");
3707 0 : return -1.0;
3708 : }
3709 :
3710 : #ifndef HAVE_SFCGAL
3711 :
3712 1 : CPLError(CE_Failure, CPLE_NotSupported, "SFCGAL support not enabled.");
3713 1 : return -1.0;
3714 :
3715 : #else
3716 :
3717 : sfcgal_init();
3718 : sfcgal_geometry_t *poThis = OGRGeometry::OGRexportToSFCGAL(this);
3719 : if (poThis == nullptr)
3720 : return -1.0;
3721 :
3722 : sfcgal_geometry_t *poOther = OGRGeometry::OGRexportToSFCGAL(poOtherGeom);
3723 : if (poOther == nullptr)
3724 : {
3725 : sfcgal_geometry_delete(poThis);
3726 : return -1.0;
3727 : }
3728 :
3729 : const double dfDistance = sfcgal_geometry_distance_3d(poThis, poOther);
3730 :
3731 : sfcgal_geometry_delete(poThis);
3732 : sfcgal_geometry_delete(poOther);
3733 :
3734 : return dfDistance > 0 ? dfDistance : -1.0;
3735 :
3736 : #endif
3737 : }
3738 :
3739 : /************************************************************************/
3740 : /* OGR_G_Distance3D() */
3741 : /************************************************************************/
3742 : /**
3743 : * \brief Returns the 3D distance between two geometries
3744 : *
3745 : * The distance is expressed into the same unit as the coordinates of the
3746 : * geometries.
3747 : *
3748 : * This method is built on the SFCGAL library, check it for the definition
3749 : * of the geometry operation.
3750 : * If OGR is built without the SFCGAL library, this method will always return
3751 : * -1.0
3752 : *
3753 : * This function is the same as the C++ method OGRGeometry::Distance3D().
3754 : *
3755 : * @param hFirst the first geometry to compare against.
3756 : * @param hOther the other geometry to compare against.
3757 : * @return distance between the two geometries
3758 : * @since GDAL 2.2
3759 : *
3760 : * @return the distance between the geometries or -1 if an error occurs.
3761 : */
3762 :
3763 1 : double OGR_G_Distance3D(OGRGeometryH hFirst, OGRGeometryH hOther)
3764 :
3765 : {
3766 1 : VALIDATE_POINTER1(hFirst, "OGR_G_Distance3D", 0.0);
3767 :
3768 2 : return OGRGeometry::FromHandle(hFirst)->Distance3D(
3769 1 : OGRGeometry::FromHandle(hOther));
3770 : }
3771 :
3772 : /************************************************************************/
3773 : /* OGRGeometryRebuildCurves() */
3774 : /************************************************************************/
3775 :
3776 : #ifdef HAVE_GEOS
3777 1951 : static OGRGeometry *OGRGeometryRebuildCurves(const OGRGeometry *poGeom,
3778 : const OGRGeometry *poOtherGeom,
3779 : OGRGeometry *poOGRProduct)
3780 : {
3781 3902 : if (poOGRProduct != nullptr &&
3782 3843 : wkbFlatten(poOGRProduct->getGeometryType()) != wkbPoint &&
3783 1892 : (poGeom->hasCurveGeometry(true) ||
3784 1018 : (poOtherGeom && poOtherGeom->hasCurveGeometry(true))))
3785 : {
3786 8 : OGRGeometry *poCurveGeom = poOGRProduct->getCurveGeometry();
3787 8 : delete poOGRProduct;
3788 8 : return poCurveGeom;
3789 : }
3790 1943 : return poOGRProduct;
3791 : }
3792 :
3793 : /************************************************************************/
3794 : /* BuildGeometryFromGEOS() */
3795 : /************************************************************************/
3796 :
3797 1868 : static OGRGeometry *BuildGeometryFromGEOS(GEOSContextHandle_t hGEOSCtxt,
3798 : GEOSGeom hGeosProduct,
3799 : const OGRGeometry *poSelf,
3800 : const OGRGeometry *poOtherGeom)
3801 : {
3802 1868 : OGRGeometry *poOGRProduct = nullptr;
3803 1868 : if (hGeosProduct != nullptr)
3804 : {
3805 : poOGRProduct =
3806 1867 : OGRGeometryFactory::createFromGEOS(hGEOSCtxt, hGeosProduct);
3807 1867 : if (poOGRProduct != nullptr &&
3808 3808 : poSelf->getSpatialReference() != nullptr &&
3809 74 : (poOtherGeom == nullptr ||
3810 74 : (poOtherGeom->getSpatialReference() != nullptr &&
3811 5 : poOtherGeom->getSpatialReference()->IsSame(
3812 : poSelf->getSpatialReference()))))
3813 : {
3814 8 : poOGRProduct->assignSpatialReference(poSelf->getSpatialReference());
3815 : }
3816 : poOGRProduct =
3817 1867 : OGRGeometryRebuildCurves(poSelf, poOtherGeom, poOGRProduct);
3818 1867 : GEOSGeom_destroy_r(hGEOSCtxt, hGeosProduct);
3819 : }
3820 1868 : return poOGRProduct;
3821 : }
3822 :
3823 : /************************************************************************/
3824 : /* BuildGeometryFromTwoGeoms() */
3825 : /************************************************************************/
3826 :
3827 1075 : static OGRGeometry *BuildGeometryFromTwoGeoms(
3828 : const OGRGeometry *poSelf, const OGRGeometry *poOtherGeom,
3829 : GEOSGeometry *(*pfnGEOSFunction_r)(GEOSContextHandle_t,
3830 : const GEOSGeometry *,
3831 : const GEOSGeometry *))
3832 : {
3833 1075 : OGRGeometry *poOGRProduct = nullptr;
3834 :
3835 1075 : GEOSContextHandle_t hGEOSCtxt = poSelf->createGEOSContext();
3836 1075 : GEOSGeom hThisGeosGeom = poSelf->exportToGEOS(hGEOSCtxt);
3837 1075 : GEOSGeom hOtherGeosGeom = poOtherGeom->exportToGEOS(hGEOSCtxt);
3838 1075 : if (hThisGeosGeom != nullptr && hOtherGeosGeom != nullptr)
3839 : {
3840 : GEOSGeom hGeosProduct =
3841 1075 : pfnGEOSFunction_r(hGEOSCtxt, hThisGeosGeom, hOtherGeosGeom);
3842 :
3843 : poOGRProduct =
3844 1075 : BuildGeometryFromGEOS(hGEOSCtxt, hGeosProduct, poSelf, poOtherGeom);
3845 : }
3846 1075 : GEOSGeom_destroy_r(hGEOSCtxt, hThisGeosGeom);
3847 1075 : GEOSGeom_destroy_r(hGEOSCtxt, hOtherGeosGeom);
3848 1075 : poSelf->freeGEOSContext(hGEOSCtxt);
3849 :
3850 1075 : return poOGRProduct;
3851 : }
3852 :
3853 : /************************************************************************/
3854 : /* OGRGEOSBooleanPredicate() */
3855 : /************************************************************************/
3856 :
3857 21268 : static OGRBoolean OGRGEOSBooleanPredicate(
3858 : const OGRGeometry *poSelf, const OGRGeometry *poOtherGeom,
3859 : char (*pfnGEOSFunction_r)(GEOSContextHandle_t, const GEOSGeometry *,
3860 : const GEOSGeometry *))
3861 : {
3862 21268 : OGRBoolean bResult = FALSE;
3863 :
3864 21268 : GEOSContextHandle_t hGEOSCtxt = poSelf->createGEOSContext();
3865 21268 : GEOSGeom hThisGeosGeom = poSelf->exportToGEOS(hGEOSCtxt);
3866 21268 : GEOSGeom hOtherGeosGeom = poOtherGeom->exportToGEOS(hGEOSCtxt);
3867 21268 : if (hThisGeosGeom != nullptr && hOtherGeosGeom != nullptr)
3868 : {
3869 20783 : bResult = pfnGEOSFunction_r(hGEOSCtxt, hThisGeosGeom, hOtherGeosGeom);
3870 : }
3871 21268 : GEOSGeom_destroy_r(hGEOSCtxt, hThisGeosGeom);
3872 21268 : GEOSGeom_destroy_r(hGEOSCtxt, hOtherGeosGeom);
3873 21268 : poSelf->freeGEOSContext(hGEOSCtxt);
3874 :
3875 21268 : return bResult;
3876 : }
3877 :
3878 : #endif // HAVE_GEOS
3879 :
3880 : /************************************************************************/
3881 : /* MakeValid() */
3882 : /************************************************************************/
3883 :
3884 : /**
3885 : * \brief Attempts to make an invalid geometry valid without losing vertices.
3886 : *
3887 : * Already-valid geometries are cloned without further intervention
3888 : * for default MODE=LINEWORK. Already-valid geometries with MODE=STRUCTURE
3889 : * may be subject to non-significant transformations, such as duplicated point
3890 : * removal, change in ring winding order, etc. (before GDAL 3.10, single-part
3891 : * geometry collections could be returned a single geometry. GDAL 3.10
3892 : * returns the same type of geometry).
3893 : *
3894 : * Running OGRGeometryFactory::removeLowerDimensionSubGeoms() as a
3895 : * post-processing step is often desired.
3896 : *
3897 : * This method is the same as the C function OGR_G_MakeValid().
3898 : *
3899 : * This function is built on the GEOS >= 3.8 library, check it for the
3900 : * definition of the geometry operation. If OGR is built without the GEOS >= 3.8
3901 : * library, this function will return a clone of the input geometry if it is
3902 : * valid, or NULL if it is invalid
3903 : *
3904 : * @param papszOptions NULL terminated list of options, or NULL. The following
3905 : * options are available:
3906 : * <ul>
3907 : * <li>METHOD=LINEWORK/STRUCTURE.
3908 : * LINEWORK is the default method, which combines all rings into a set of
3909 : * noded lines and then extracts valid polygons from that linework.
3910 : * The STRUCTURE method (requires GEOS >= 3.10 and GDAL >= 3.4) first makes
3911 : * all rings valid, then merges shells and
3912 : * subtracts holes from shells to generate valid result. Assumes that
3913 : * holes and shells are correctly categorized.</li>
3914 : * <li>KEEP_COLLAPSED=YES/NO. Only for METHOD=STRUCTURE.
3915 : * NO (default): collapses are converted to empty geometries
3916 : * YES: collapses are converted to a valid geometry of lower dimension.</li>
3917 : * </ul>
3918 : * @return a new geometry to be freed by the caller, or NULL if an error occurs.
3919 : *
3920 : * @since GDAL 3.0
3921 : */
3922 86 : OGRGeometry *OGRGeometry::MakeValid(CSLConstList papszOptions) const
3923 : {
3924 : (void)papszOptions;
3925 : #ifndef HAVE_GEOS
3926 : if (IsValid())
3927 : return clone();
3928 :
3929 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
3930 : return nullptr;
3931 : #else
3932 86 : if (IsSFCGALCompatible())
3933 : {
3934 0 : if (IsValid())
3935 0 : return clone();
3936 : }
3937 86 : else if (wkbFlatten(getGeometryType()) == wkbCurvePolygon)
3938 : {
3939 2 : GEOSContextHandle_t hGEOSCtxt = initGEOS_r(nullptr, nullptr);
3940 2 : OGRBoolean bIsValid = FALSE;
3941 2 : GEOSGeom hGeosGeom = exportToGEOS(hGEOSCtxt);
3942 2 : if (hGeosGeom)
3943 : {
3944 2 : bIsValid = GEOSisValid_r(hGEOSCtxt, hGeosGeom);
3945 2 : GEOSGeom_destroy_r(hGEOSCtxt, hGeosGeom);
3946 : }
3947 2 : freeGEOSContext(hGEOSCtxt);
3948 2 : if (bIsValid)
3949 1 : return clone();
3950 : }
3951 :
3952 85 : const bool bStructureMethod = EQUAL(
3953 : CSLFetchNameValueDef(papszOptions, "METHOD", "LINEWORK"), "STRUCTURE");
3954 85 : CPL_IGNORE_RET_VAL(bStructureMethod);
3955 : #if !(GEOS_VERSION_MAJOR > 3 || \
3956 : (GEOS_VERSION_MAJOR == 3 && GEOS_VERSION_MINOR >= 10))
3957 : if (bStructureMethod)
3958 : {
3959 : CPLError(CE_Failure, CPLE_NotSupported,
3960 : "GEOS 3.10 or later needed for METHOD=STRUCTURE.");
3961 : return nullptr;
3962 : }
3963 : #endif
3964 :
3965 85 : OGRGeometry *poOGRProduct = nullptr;
3966 :
3967 85 : GEOSContextHandle_t hGEOSCtxt = createGEOSContext();
3968 85 : GEOSGeom hGeosGeom = exportToGEOS(hGEOSCtxt);
3969 85 : if (hGeosGeom != nullptr)
3970 : {
3971 : GEOSGeom hGEOSRet;
3972 : #if GEOS_VERSION_MAJOR > 3 || \
3973 : (GEOS_VERSION_MAJOR == 3 && GEOS_VERSION_MINOR >= 10)
3974 84 : if (bStructureMethod)
3975 : {
3976 : GEOSMakeValidParams *params =
3977 5 : GEOSMakeValidParams_create_r(hGEOSCtxt);
3978 5 : CPLAssert(params);
3979 5 : GEOSMakeValidParams_setMethod_r(hGEOSCtxt, params,
3980 : GEOS_MAKE_VALID_STRUCTURE);
3981 5 : GEOSMakeValidParams_setKeepCollapsed_r(
3982 : hGEOSCtxt, params,
3983 5 : CPLFetchBool(papszOptions, "KEEP_COLLAPSED", false));
3984 5 : hGEOSRet = GEOSMakeValidWithParams_r(hGEOSCtxt, hGeosGeom, params);
3985 5 : GEOSMakeValidParams_destroy_r(hGEOSCtxt, params);
3986 : }
3987 : else
3988 : #endif
3989 : {
3990 79 : hGEOSRet = GEOSMakeValid_r(hGEOSCtxt, hGeosGeom);
3991 : }
3992 84 : GEOSGeom_destroy_r(hGEOSCtxt, hGeosGeom);
3993 :
3994 84 : if (hGEOSRet != nullptr)
3995 : {
3996 : poOGRProduct =
3997 84 : OGRGeometryFactory::createFromGEOS(hGEOSCtxt, hGEOSRet);
3998 84 : if (poOGRProduct != nullptr && getSpatialReference() != nullptr)
3999 3 : poOGRProduct->assignSpatialReference(getSpatialReference());
4000 : poOGRProduct =
4001 84 : OGRGeometryRebuildCurves(this, nullptr, poOGRProduct);
4002 84 : GEOSGeom_destroy_r(hGEOSCtxt, hGEOSRet);
4003 :
4004 : #if GEOS_VERSION_MAJOR > 3 || \
4005 : (GEOS_VERSION_MAJOR == 3 && GEOS_VERSION_MINOR >= 10)
4006 : // METHOD=STRUCTURE is not guaranteed to return a multiple geometry
4007 : // if the input is a multiple geometry
4008 84 : if (poOGRProduct && bStructureMethod &&
4009 172 : OGR_GT_IsSubClassOf(getGeometryType(), wkbGeometryCollection) &&
4010 4 : !OGR_GT_IsSubClassOf(poOGRProduct->getGeometryType(),
4011 : wkbGeometryCollection))
4012 : {
4013 3 : poOGRProduct = OGRGeometryFactory::forceTo(poOGRProduct,
4014 3 : getGeometryType());
4015 : }
4016 : #endif
4017 : }
4018 : }
4019 85 : freeGEOSContext(hGEOSCtxt);
4020 :
4021 85 : return poOGRProduct;
4022 : #endif
4023 : }
4024 :
4025 : /************************************************************************/
4026 : /* OGR_G_MakeValid() */
4027 : /************************************************************************/
4028 :
4029 : /**
4030 : * \brief Attempts to make an invalid geometry valid without losing vertices.
4031 : *
4032 : * Already-valid geometries are cloned without further intervention.
4033 : *
4034 : * This function is the same as the C++ method OGRGeometry::MakeValid().
4035 : *
4036 : * This function is built on the GEOS >= 3.8 library, check it for the
4037 : * definition of the geometry operation. If OGR is built without the GEOS >= 3.8
4038 : * library, this function will return a clone of the input geometry if it is
4039 : * valid, or NULL if it is invalid
4040 : *
4041 : * @param hGeom The Geometry to make valid.
4042 : *
4043 : * @return a new geometry to be freed by the caller with OGR_G_DestroyGeometry,
4044 : * or NULL if an error occurs.
4045 : *
4046 : * @since GDAL 3.0
4047 : */
4048 :
4049 0 : OGRGeometryH OGR_G_MakeValid(OGRGeometryH hGeom)
4050 :
4051 : {
4052 0 : VALIDATE_POINTER1(hGeom, "OGR_G_MakeValid", nullptr);
4053 :
4054 0 : return OGRGeometry::ToHandle(OGRGeometry::FromHandle(hGeom)->MakeValid());
4055 : }
4056 :
4057 : /************************************************************************/
4058 : /* OGR_G_MakeValidEx() */
4059 : /************************************************************************/
4060 :
4061 : /**
4062 : * \brief Attempts to make an invalid geometry valid without losing vertices.
4063 : *
4064 : * Already-valid geometries are cloned without further intervention.
4065 : *
4066 : * This function is the same as the C++ method OGRGeometry::MakeValid().
4067 : *
4068 : * See documentation of that method for possible options.
4069 : *
4070 : * @param hGeom The Geometry to make valid.
4071 : * @param papszOptions Options.
4072 : *
4073 : * @return a new geometry to be freed by the caller with OGR_G_DestroyGeometry,
4074 : * or NULL if an error occurs.
4075 : *
4076 : * @since GDAL 3.4
4077 : */
4078 :
4079 11 : OGRGeometryH OGR_G_MakeValidEx(OGRGeometryH hGeom, CSLConstList papszOptions)
4080 :
4081 : {
4082 11 : VALIDATE_POINTER1(hGeom, "OGR_G_MakeValidEx", nullptr);
4083 :
4084 11 : return OGRGeometry::ToHandle(
4085 22 : OGRGeometry::FromHandle(hGeom)->MakeValid(papszOptions));
4086 : }
4087 :
4088 : /************************************************************************/
4089 : /* Normalize() */
4090 : /************************************************************************/
4091 :
4092 : /**
4093 : * \brief Attempts to bring geometry into normalized/canonical form.
4094 : *
4095 : * This method is the same as the C function OGR_G_Normalize().
4096 : *
4097 : * This function is built on the GEOS library; check it for the definition
4098 : * of the geometry operation.
4099 : * If OGR is built without the GEOS library, this function will always fail,
4100 : * issuing a CPLE_NotSupported error.
4101 : *
4102 : * @return a new geometry to be freed by the caller, or NULL if an error occurs.
4103 : *
4104 : * @since GDAL 3.3
4105 : */
4106 35 : OGRGeometry *OGRGeometry::Normalize() const
4107 : {
4108 : #ifndef HAVE_GEOS
4109 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
4110 : return nullptr;
4111 : #else
4112 35 : OGRGeometry *poOGRProduct = nullptr;
4113 :
4114 35 : GEOSContextHandle_t hGEOSCtxt = createGEOSContext();
4115 35 : GEOSGeom hGeosGeom = exportToGEOS(hGEOSCtxt);
4116 35 : if (hGeosGeom != nullptr)
4117 : {
4118 :
4119 35 : int hGEOSRet = GEOSNormalize_r(hGEOSCtxt, hGeosGeom);
4120 :
4121 35 : if (hGEOSRet == 0)
4122 : {
4123 : poOGRProduct =
4124 35 : BuildGeometryFromGEOS(hGEOSCtxt, hGeosGeom, this, nullptr);
4125 : }
4126 : else
4127 : {
4128 0 : GEOSGeom_destroy_r(hGEOSCtxt, hGeosGeom);
4129 : }
4130 : }
4131 35 : freeGEOSContext(hGEOSCtxt);
4132 :
4133 35 : return poOGRProduct;
4134 : #endif
4135 : }
4136 :
4137 : /************************************************************************/
4138 : /* OGR_G_Normalize() */
4139 : /************************************************************************/
4140 :
4141 : /**
4142 : * \brief Attempts to bring geometry into normalized/canonical form.
4143 : *
4144 : * This function is the same as the C++ method OGRGeometry::Normalize().
4145 : *
4146 : * This function is built on the GEOS library; check it for the definition
4147 : * of the geometry operation.
4148 : * If OGR is built without the GEOS library, this function will always fail,
4149 : * issuing a CPLE_NotSupported error.
4150 : * @param hGeom The Geometry to normalize.
4151 : *
4152 : * @return a new geometry to be freed by the caller with OGR_G_DestroyGeometry,
4153 : * or NULL if an error occurs.
4154 : *
4155 : * @since GDAL 3.3
4156 : */
4157 :
4158 5 : OGRGeometryH OGR_G_Normalize(OGRGeometryH hGeom)
4159 :
4160 : {
4161 5 : VALIDATE_POINTER1(hGeom, "OGR_G_Normalize", nullptr);
4162 :
4163 5 : return OGRGeometry::ToHandle(OGRGeometry::FromHandle(hGeom)->Normalize());
4164 : }
4165 :
4166 : /************************************************************************/
4167 : /* ConvexHull() */
4168 : /************************************************************************/
4169 :
4170 : /**
4171 : * \brief Compute convex hull.
4172 : *
4173 : * A new geometry object is created and returned containing the convex
4174 : * hull of the geometry on which the method is invoked.
4175 : *
4176 : * This method is the same as the C function OGR_G_ConvexHull().
4177 : *
4178 : * This method is built on the GEOS library, check it for the definition
4179 : * of the geometry operation.
4180 : * If OGR is built without the GEOS library, this method will always fail,
4181 : * issuing a CPLE_NotSupported error.
4182 : *
4183 : * @return a new geometry to be freed by the caller, or NULL if an error occurs.
4184 : */
4185 :
4186 2 : OGRGeometry *OGRGeometry::ConvexHull() const
4187 :
4188 : {
4189 2 : if (IsSFCGALCompatible())
4190 : {
4191 : #ifndef HAVE_SFCGAL
4192 :
4193 0 : CPLError(CE_Failure, CPLE_NotSupported, "SFCGAL support not enabled.");
4194 0 : return nullptr;
4195 :
4196 : #else
4197 :
4198 : sfcgal_geometry_t *poThis = OGRGeometry::OGRexportToSFCGAL(this);
4199 : if (poThis == nullptr)
4200 : return nullptr;
4201 :
4202 : sfcgal_geometry_t *poRes = sfcgal_geometry_convexhull_3d(poThis);
4203 : OGRGeometry *h_prodGeom = SFCGALexportToOGR(poRes);
4204 : if (h_prodGeom)
4205 : h_prodGeom->assignSpatialReference(getSpatialReference());
4206 :
4207 : sfcgal_geometry_delete(poThis);
4208 : sfcgal_geometry_delete(poRes);
4209 :
4210 : return h_prodGeom;
4211 :
4212 : #endif
4213 : }
4214 :
4215 : else
4216 : {
4217 : #ifndef HAVE_GEOS
4218 :
4219 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
4220 : return nullptr;
4221 :
4222 : #else
4223 :
4224 2 : OGRGeometry *poOGRProduct = nullptr;
4225 :
4226 2 : GEOSContextHandle_t hGEOSCtxt = createGEOSContext();
4227 2 : GEOSGeom hGeosGeom = exportToGEOS(hGEOSCtxt);
4228 2 : if (hGeosGeom != nullptr)
4229 : {
4230 2 : GEOSGeom hGeosHull = GEOSConvexHull_r(hGEOSCtxt, hGeosGeom);
4231 2 : GEOSGeom_destroy_r(hGEOSCtxt, hGeosGeom);
4232 :
4233 : poOGRProduct =
4234 2 : BuildGeometryFromGEOS(hGEOSCtxt, hGeosHull, this, nullptr);
4235 : }
4236 2 : freeGEOSContext(hGEOSCtxt);
4237 :
4238 2 : return poOGRProduct;
4239 :
4240 : #endif /* HAVE_GEOS */
4241 : }
4242 : }
4243 :
4244 : /************************************************************************/
4245 : /* OGR_G_ConvexHull() */
4246 : /************************************************************************/
4247 : /**
4248 : * \brief Compute convex hull.
4249 : *
4250 : * A new geometry object is created and returned containing the convex
4251 : * hull of the geometry on which the method is invoked.
4252 : *
4253 : * This function is the same as the C++ method OGRGeometry::ConvexHull().
4254 : *
4255 : * This function is built on the GEOS library, check it for the definition
4256 : * of the geometry operation.
4257 : * If OGR is built without the GEOS library, this function will always fail,
4258 : * issuing a CPLE_NotSupported error.
4259 : *
4260 : * @param hTarget The Geometry to calculate the convex hull of.
4261 : *
4262 : * @return a new geometry to be freed by the caller with OGR_G_DestroyGeometry,
4263 : * or NULL if an error occurs.
4264 : */
4265 :
4266 1 : OGRGeometryH OGR_G_ConvexHull(OGRGeometryH hTarget)
4267 :
4268 : {
4269 1 : VALIDATE_POINTER1(hTarget, "OGR_G_ConvexHull", nullptr);
4270 :
4271 1 : return OGRGeometry::ToHandle(
4272 2 : OGRGeometry::FromHandle(hTarget)->ConvexHull());
4273 : }
4274 :
4275 : /************************************************************************/
4276 : /* ConcaveHull() */
4277 : /************************************************************************/
4278 :
4279 : /**
4280 : * \brief Compute "concave hull" of a geometry.
4281 : *
4282 : * The concave hull is fully contained within the convex hull and also
4283 : * contains all the points of the input, but in a smaller area.
4284 : * The area ratio is the ratio of the area of the convex hull and the concave
4285 : * hull. Frequently used to convert a multi-point into a polygonal area.
4286 : * that contains all the points in the input Geometry.
4287 : *
4288 : * A new geometry object is created and returned containing the concave
4289 : * hull of the geometry on which the method is invoked.
4290 : *
4291 : * This method is the same as the C function OGR_G_ConcaveHull().
4292 : *
4293 : * This method is built on the GEOS >= 3.11 library
4294 : * If OGR is built without the GEOS >= 3.11 librray, this method will always
4295 : * fail, issuing a CPLE_NotSupported error.
4296 : *
4297 : * @param dfRatio Ratio of the area of the convex hull and the concave hull.
4298 : * @param bAllowHoles Whether holes are allowed.
4299 : *
4300 : * @return a new geometry to be freed by the caller, or NULL if an error occurs.
4301 : *
4302 : * @since GDAL 3.6
4303 : */
4304 :
4305 2 : OGRGeometry *OGRGeometry::ConcaveHull(double dfRatio, bool bAllowHoles) const
4306 : {
4307 : #ifndef HAVE_GEOS
4308 : (void)dfRatio;
4309 : (void)bAllowHoles;
4310 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
4311 : return nullptr;
4312 : #elif GEOS_VERSION_MAJOR == 3 && GEOS_VERSION_MINOR < 11
4313 : (void)dfRatio;
4314 : (void)bAllowHoles;
4315 : CPLError(CE_Failure, CPLE_NotSupported,
4316 : "GEOS 3.11 or later needed for ConcaveHull.");
4317 : return nullptr;
4318 : #else
4319 2 : OGRGeometry *poOGRProduct = nullptr;
4320 :
4321 2 : GEOSContextHandle_t hGEOSCtxt = createGEOSContext();
4322 2 : GEOSGeom hGeosGeom = exportToGEOS(hGEOSCtxt);
4323 2 : if (hGeosGeom != nullptr)
4324 : {
4325 : GEOSGeom hGeosHull =
4326 2 : GEOSConcaveHull_r(hGEOSCtxt, hGeosGeom, dfRatio, bAllowHoles);
4327 2 : GEOSGeom_destroy_r(hGEOSCtxt, hGeosGeom);
4328 :
4329 : poOGRProduct =
4330 2 : BuildGeometryFromGEOS(hGEOSCtxt, hGeosHull, this, nullptr);
4331 : }
4332 2 : freeGEOSContext(hGEOSCtxt);
4333 :
4334 2 : return poOGRProduct;
4335 : #endif /* HAVE_GEOS */
4336 : }
4337 :
4338 : /************************************************************************/
4339 : /* OGR_G_ConcaveHull() */
4340 : /************************************************************************/
4341 : /**
4342 : * \brief Compute "concave hull" of a geometry.
4343 : *
4344 : * The concave hull is fully contained within the convex hull and also
4345 : * contains all the points of the input, but in a smaller area.
4346 : * The area ratio is the ratio of the area of the convex hull and the concave
4347 : * hull. Frequently used to convert a multi-point into a polygonal area.
4348 : * that contains all the points in the input Geometry.
4349 : *
4350 : * A new geometry object is created and returned containing the convex
4351 : * hull of the geometry on which the function is invoked.
4352 : *
4353 : * This function is the same as the C++ method OGRGeometry::ConcaveHull().
4354 : *
4355 : * This function is built on the GEOS >= 3.11 library
4356 : * If OGR is built without the GEOS >= 3.11 librray, this function will always
4357 : * fail, issuing a CPLE_NotSupported error.
4358 : *
4359 : * @param hTarget The Geometry to calculate the concave hull of.
4360 : * @param dfRatio Ratio of the area of the convex hull and the concave hull.
4361 : * @param bAllowHoles Whether holes are allowed.
4362 : *
4363 : * @return a new geometry to be freed by the caller with OGR_G_DestroyGeometry,
4364 : * or NULL if an error occurs.
4365 : *
4366 : * @since GDAL 3.6
4367 : */
4368 :
4369 2 : OGRGeometryH OGR_G_ConcaveHull(OGRGeometryH hTarget, double dfRatio,
4370 : bool bAllowHoles)
4371 :
4372 : {
4373 2 : VALIDATE_POINTER1(hTarget, "OGR_G_ConcaveHull", nullptr);
4374 :
4375 2 : return OGRGeometry::ToHandle(
4376 4 : OGRGeometry::FromHandle(hTarget)->ConcaveHull(dfRatio, bAllowHoles));
4377 : }
4378 :
4379 : /************************************************************************/
4380 : /* Boundary() */
4381 : /************************************************************************/
4382 :
4383 : /**
4384 : * \brief Compute boundary.
4385 : *
4386 : * A new geometry object is created and returned containing the boundary
4387 : * of the geometry on which the method is invoked.
4388 : *
4389 : * This method is the same as the C function OGR_G_Boundary().
4390 : *
4391 : * This method is built on the GEOS library, check it for the definition
4392 : * of the geometry operation.
4393 : * If OGR is built without the GEOS library, this method will always fail,
4394 : * issuing a CPLE_NotSupported error.
4395 : *
4396 : * @return a new geometry to be freed by the caller, or NULL if an error occurs.
4397 : *
4398 : * @since OGR 1.8.0
4399 : */
4400 :
4401 6 : OGRGeometry *OGRGeometry::Boundary() const
4402 :
4403 : {
4404 : #ifndef HAVE_GEOS
4405 :
4406 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
4407 : return nullptr;
4408 :
4409 : #else
4410 :
4411 6 : OGRGeometry *poOGRProduct = nullptr;
4412 :
4413 6 : GEOSContextHandle_t hGEOSCtxt = createGEOSContext();
4414 6 : GEOSGeom hGeosGeom = exportToGEOS(hGEOSCtxt);
4415 6 : if (hGeosGeom != nullptr)
4416 : {
4417 6 : GEOSGeom hGeosProduct = GEOSBoundary_r(hGEOSCtxt, hGeosGeom);
4418 6 : GEOSGeom_destroy_r(hGEOSCtxt, hGeosGeom);
4419 :
4420 : poOGRProduct =
4421 6 : BuildGeometryFromGEOS(hGEOSCtxt, hGeosProduct, this, nullptr);
4422 : }
4423 6 : freeGEOSContext(hGEOSCtxt);
4424 :
4425 6 : return poOGRProduct;
4426 :
4427 : #endif // HAVE_GEOS
4428 : }
4429 :
4430 : //! @cond Doxygen_Suppress
4431 : /**
4432 : * \brief Compute boundary (deprecated)
4433 : *
4434 : * @deprecated
4435 : *
4436 : * @see Boundary()
4437 : */
4438 0 : OGRGeometry *OGRGeometry::getBoundary() const
4439 :
4440 : {
4441 0 : return Boundary();
4442 : }
4443 :
4444 : //! @endcond
4445 :
4446 : /************************************************************************/
4447 : /* OGR_G_Boundary() */
4448 : /************************************************************************/
4449 : /**
4450 : * \brief Compute boundary.
4451 : *
4452 : * A new geometry object is created and returned containing the boundary
4453 : * of the geometry on which the method is invoked.
4454 : *
4455 : * This function is the same as the C++ method OGR_G_Boundary().
4456 : *
4457 : * This function is built on the GEOS library, check it for the definition
4458 : * of the geometry operation.
4459 : * If OGR is built without the GEOS library, this function will always fail,
4460 : * issuing a CPLE_NotSupported error.
4461 : *
4462 : * @param hTarget The Geometry to calculate the boundary of.
4463 : *
4464 : * @return a new geometry to be freed by the caller with OGR_G_DestroyGeometry,
4465 : * or NULL if an error occurs.
4466 : *
4467 : * @since OGR 1.8.0
4468 : */
4469 6 : OGRGeometryH OGR_G_Boundary(OGRGeometryH hTarget)
4470 :
4471 : {
4472 6 : VALIDATE_POINTER1(hTarget, "OGR_G_Boundary", nullptr);
4473 :
4474 6 : return OGRGeometry::ToHandle(OGRGeometry::FromHandle(hTarget)->Boundary());
4475 : }
4476 :
4477 : /**
4478 : * \brief Compute boundary (deprecated)
4479 : *
4480 : * @deprecated
4481 : *
4482 : * @see OGR_G_Boundary()
4483 : */
4484 0 : OGRGeometryH OGR_G_GetBoundary(OGRGeometryH hTarget)
4485 :
4486 : {
4487 0 : VALIDATE_POINTER1(hTarget, "OGR_G_GetBoundary", nullptr);
4488 :
4489 0 : return OGRGeometry::ToHandle(OGRGeometry::FromHandle(hTarget)->Boundary());
4490 : }
4491 :
4492 : /************************************************************************/
4493 : /* Buffer() */
4494 : /************************************************************************/
4495 :
4496 : /**
4497 : * \brief Compute buffer of geometry.
4498 : *
4499 : * Builds a new geometry containing the buffer region around the geometry
4500 : * on which it is invoked. The buffer is a polygon containing the region within
4501 : * the buffer distance of the original geometry.
4502 : *
4503 : * Some buffer sections are properly described as curves, but are converted to
4504 : * approximate polygons. The nQuadSegs parameter can be used to control how
4505 : * many segments should be used to define a 90 degree curve - a quadrant of a
4506 : * circle. A value of 30 is a reasonable default. Large values result in
4507 : * large numbers of vertices in the resulting buffer geometry while small
4508 : * numbers reduce the accuracy of the result.
4509 : *
4510 : * This method is the same as the C function OGR_G_Buffer().
4511 : *
4512 : * This method is built on the GEOS library, check it for the definition
4513 : * of the geometry operation.
4514 : * If OGR is built without the GEOS library, this method will always fail,
4515 : * issuing a CPLE_NotSupported error.
4516 : *
4517 : * @param dfDist the buffer distance to be applied. Should be expressed into
4518 : * the same unit as the coordinates of the geometry.
4519 : *
4520 : * @param nQuadSegs the number of segments used to approximate a 90
4521 : * degree (quadrant) of curvature.
4522 : *
4523 : * @return a new geometry to be freed by the caller, or NULL if an error occurs.
4524 : */
4525 :
4526 4 : OGRGeometry *OGRGeometry::Buffer(UNUSED_IF_NO_GEOS double dfDist,
4527 : UNUSED_IF_NO_GEOS int nQuadSegs) const
4528 :
4529 : {
4530 : #ifndef HAVE_GEOS
4531 :
4532 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
4533 : return nullptr;
4534 :
4535 : #else
4536 :
4537 4 : OGRGeometry *poOGRProduct = nullptr;
4538 :
4539 4 : GEOSContextHandle_t hGEOSCtxt = createGEOSContext();
4540 4 : GEOSGeom hGeosGeom = exportToGEOS(hGEOSCtxt);
4541 4 : if (hGeosGeom != nullptr)
4542 : {
4543 : GEOSGeom hGeosProduct =
4544 4 : GEOSBuffer_r(hGEOSCtxt, hGeosGeom, dfDist, nQuadSegs);
4545 4 : GEOSGeom_destroy_r(hGEOSCtxt, hGeosGeom);
4546 :
4547 : poOGRProduct =
4548 4 : BuildGeometryFromGEOS(hGEOSCtxt, hGeosProduct, this, nullptr);
4549 : }
4550 4 : freeGEOSContext(hGEOSCtxt);
4551 :
4552 4 : return poOGRProduct;
4553 :
4554 : #endif // HAVE_GEOS
4555 : }
4556 :
4557 : /************************************************************************/
4558 : /* OGR_G_Buffer() */
4559 : /************************************************************************/
4560 :
4561 : /**
4562 : * \brief Compute buffer of geometry.
4563 : *
4564 : * Builds a new geometry containing the buffer region around the geometry
4565 : * on which it is invoked. The buffer is a polygon containing the region within
4566 : * the buffer distance of the original geometry.
4567 : *
4568 : * Some buffer sections are properly described as curves, but are converted to
4569 : * approximate polygons. The nQuadSegs parameter can be used to control how
4570 : * many segments should be used to define a 90 degree curve - a quadrant of a
4571 : * circle. A value of 30 is a reasonable default. Large values result in
4572 : * large numbers of vertices in the resulting buffer geometry while small
4573 : * numbers reduce the accuracy of the result.
4574 : *
4575 : * This function is the same as the C++ method OGRGeometry::Buffer().
4576 : *
4577 : * This function is built on the GEOS library, check it for the definition
4578 : * of the geometry operation.
4579 : * If OGR is built without the GEOS library, this function will always fail,
4580 : * issuing a CPLE_NotSupported error.
4581 : *
4582 : * @param hTarget the geometry.
4583 : * @param dfDist the buffer distance to be applied. Should be expressed into
4584 : * the same unit as the coordinates of the geometry.
4585 : *
4586 : * @param nQuadSegs the number of segments used to approximate a 90 degree
4587 : * (quadrant) of curvature.
4588 : *
4589 : * @return a new geometry to be freed by the caller with OGR_G_DestroyGeometry,
4590 : * or NULL if an error occurs.
4591 : */
4592 :
4593 4 : OGRGeometryH OGR_G_Buffer(OGRGeometryH hTarget, double dfDist, int nQuadSegs)
4594 :
4595 : {
4596 4 : VALIDATE_POINTER1(hTarget, "OGR_G_Buffer", nullptr);
4597 :
4598 4 : return OGRGeometry::ToHandle(
4599 8 : OGRGeometry::FromHandle(hTarget)->Buffer(dfDist, nQuadSegs));
4600 : }
4601 :
4602 : /**
4603 : * \brief Compute buffer of geometry.
4604 : *
4605 : * Builds a new geometry containing the buffer region around the geometry
4606 : * on which it is invoked. The buffer is a polygon containing the region within
4607 : * the buffer distance of the original geometry.
4608 : *
4609 : * This function is built on the GEOS library, check it for the definition
4610 : * of the geometry operation.
4611 : * If OGR is built without the GEOS library, this function will always fail,
4612 : * issuing a CPLE_NotSupported error.
4613 : *
4614 : * The following options are supported. See the GEOS library for more detailed
4615 : * descriptions.
4616 : *
4617 : * <ul>
4618 : * <li>ENDCAP_STYLE=ROUND/FLAT/SQUARE</li>
4619 : * <li>JOIN_STYLE=ROUND/MITRE/BEVEL</li>
4620 : * <li>MITRE_LIMIT=double</li>
4621 : * <li>QUADRANT_SEGMENTS=double</li>
4622 : * <li>SINGLE_SIDED=YES/NO</li>
4623 : * </ul>
4624 : *
4625 : * This function is the same as the C function OGR_G_BufferEx().
4626 : *
4627 : * @param dfDist the buffer distance to be applied. Should be expressed into
4628 : * the same unit as the coordinates of the geometry.
4629 : * @param papszOptions NULL terminated list of options (may be NULL)
4630 : *
4631 : * @return a new geometry to be freed by the caller, or NULL if an error occurs.
4632 : *
4633 : * @since GDAL 3.10
4634 : */
4635 :
4636 : OGRGeometry *
4637 12 : OGRGeometry::BufferEx(UNUSED_IF_NO_GEOS double dfDist,
4638 : UNUSED_IF_NO_GEOS CSLConstList papszOptions) const
4639 : {
4640 : #ifndef HAVE_GEOS
4641 :
4642 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
4643 : return nullptr;
4644 :
4645 : #else
4646 12 : OGRGeometry *poOGRProduct = nullptr;
4647 12 : GEOSContextHandle_t hGEOSCtxt = createGEOSContext();
4648 :
4649 12 : auto hParams = GEOSBufferParams_create_r(hGEOSCtxt);
4650 12 : bool bParamsAreValid = true;
4651 :
4652 28 : for (const auto &[pszParam, pszValue] : cpl::IterateNameValue(papszOptions))
4653 : {
4654 16 : if (EQUAL(pszParam, "ENDCAP_STYLE"))
4655 : {
4656 : int nStyle;
4657 2 : if (EQUAL(pszValue, "ROUND"))
4658 : {
4659 0 : nStyle = GEOSBUF_CAP_ROUND;
4660 : }
4661 2 : else if (EQUAL(pszValue, "FLAT"))
4662 : {
4663 1 : nStyle = GEOSBUF_CAP_FLAT;
4664 : }
4665 1 : else if (EQUAL(pszValue, "SQUARE"))
4666 : {
4667 0 : nStyle = GEOSBUF_CAP_SQUARE;
4668 : }
4669 : else
4670 : {
4671 1 : bParamsAreValid = false;
4672 1 : CPLError(CE_Failure, CPLE_NotSupported,
4673 : "Invalid value for ENDCAP_STYLE: %s", pszValue);
4674 2 : break;
4675 : }
4676 :
4677 1 : if (!GEOSBufferParams_setEndCapStyle_r(hGEOSCtxt, hParams, nStyle))
4678 : {
4679 0 : bParamsAreValid = false;
4680 : }
4681 : }
4682 14 : else if (EQUAL(pszParam, "JOIN_STYLE"))
4683 : {
4684 : int nStyle;
4685 2 : if (EQUAL(pszValue, "ROUND"))
4686 : {
4687 0 : nStyle = GEOSBUF_JOIN_ROUND;
4688 : }
4689 2 : else if (EQUAL(pszValue, "MITRE"))
4690 : {
4691 1 : nStyle = GEOSBUF_JOIN_MITRE;
4692 : }
4693 1 : else if (EQUAL(pszValue, "BEVEL"))
4694 : {
4695 0 : nStyle = GEOSBUF_JOIN_BEVEL;
4696 : }
4697 : else
4698 : {
4699 1 : bParamsAreValid = false;
4700 1 : CPLError(CE_Failure, CPLE_NotSupported,
4701 : "Invalid value for JOIN_STYLE: %s", pszValue);
4702 1 : break;
4703 : }
4704 :
4705 1 : if (!GEOSBufferParams_setJoinStyle_r(hGEOSCtxt, hParams, nStyle))
4706 : {
4707 0 : bParamsAreValid = false;
4708 0 : break;
4709 : }
4710 : }
4711 12 : else if (EQUAL(pszParam, "MITRE_LIMIT"))
4712 : {
4713 : try
4714 : {
4715 : std::size_t end;
4716 7 : double dfLimit = std::stod(pszValue, &end);
4717 :
4718 1 : if (end != strlen(pszValue))
4719 : {
4720 0 : throw std::invalid_argument("");
4721 : }
4722 :
4723 1 : if (!GEOSBufferParams_setMitreLimit_r(hGEOSCtxt, hParams,
4724 : dfLimit))
4725 : {
4726 0 : bParamsAreValid = false;
4727 0 : break;
4728 : }
4729 : }
4730 4 : catch (const std::invalid_argument &)
4731 : {
4732 2 : bParamsAreValid = false;
4733 2 : CPLError(CE_Failure, CPLE_IllegalArg,
4734 : "Invalid value for MITRE_LIMIT: %s", pszValue);
4735 : }
4736 0 : catch (const std::out_of_range &)
4737 : {
4738 0 : bParamsAreValid = false;
4739 0 : CPLError(CE_Failure, CPLE_IllegalArg,
4740 : "Invalid value for MITRE_LIMIT: %s", pszValue);
4741 : }
4742 : }
4743 9 : else if (EQUAL(pszParam, "QUADRANT_SEGMENTS"))
4744 : {
4745 : try
4746 : {
4747 : std::size_t end;
4748 15 : int nQuadSegs = std::stoi(pszValue, &end, 10);
4749 :
4750 3 : if (end != strlen(pszValue))
4751 : {
4752 0 : throw std::invalid_argument("");
4753 : }
4754 :
4755 3 : if (!GEOSBufferParams_setQuadrantSegments_r(hGEOSCtxt, hParams,
4756 : nQuadSegs))
4757 : {
4758 0 : bParamsAreValid = false;
4759 0 : break;
4760 : }
4761 : }
4762 6 : catch (const std::invalid_argument &)
4763 : {
4764 3 : bParamsAreValid = false;
4765 3 : CPLError(CE_Failure, CPLE_IllegalArg,
4766 : "Invalid value for QUADRANT_SEGMENTS: %s", pszValue);
4767 : }
4768 2 : catch (const std::out_of_range &)
4769 : {
4770 1 : bParamsAreValid = false;
4771 1 : CPLError(CE_Failure, CPLE_IllegalArg,
4772 : "Invalid value for QUADRANT_SEGMENTS: %s", pszValue);
4773 : }
4774 : }
4775 2 : else if (EQUAL(pszParam, "SINGLE_SIDED"))
4776 : {
4777 1 : bool bSingleSided = CPLTestBool(pszValue);
4778 :
4779 1 : if (!GEOSBufferParams_setSingleSided_r(hGEOSCtxt, hParams,
4780 : bSingleSided))
4781 : {
4782 0 : bParamsAreValid = false;
4783 0 : break;
4784 : }
4785 : }
4786 : else
4787 : {
4788 1 : bParamsAreValid = false;
4789 1 : CPLError(CE_Failure, CPLE_NotSupported,
4790 : "Unsupported buffer option: %s", pszValue);
4791 : }
4792 : }
4793 :
4794 12 : if (bParamsAreValid)
4795 : {
4796 3 : GEOSGeom hGeosGeom = exportToGEOS(hGEOSCtxt);
4797 3 : if (hGeosGeom != nullptr)
4798 : {
4799 : GEOSGeom hGeosProduct =
4800 3 : GEOSBufferWithParams_r(hGEOSCtxt, hGeosGeom, hParams, dfDist);
4801 3 : GEOSGeom_destroy_r(hGEOSCtxt, hGeosGeom);
4802 :
4803 3 : if (hGeosProduct != nullptr)
4804 : {
4805 3 : poOGRProduct = BuildGeometryFromGEOS(hGEOSCtxt, hGeosProduct,
4806 : this, nullptr);
4807 : }
4808 : }
4809 : }
4810 :
4811 12 : GEOSBufferParams_destroy_r(hGEOSCtxt, hParams);
4812 12 : freeGEOSContext(hGEOSCtxt);
4813 12 : return poOGRProduct;
4814 : #endif
4815 : }
4816 :
4817 : /**
4818 : * \brief Compute buffer of geometry.
4819 : *
4820 : * Builds a new geometry containing the buffer region around the geometry
4821 : * on which it is invoked. The buffer is a polygon containing the region within
4822 : * the buffer distance of the original geometry.
4823 : *
4824 : * This function is built on the GEOS library, check it for the definition
4825 : * of the geometry operation.
4826 : * If OGR is built without the GEOS library, this function will always fail,
4827 : * issuing a CPLE_NotSupported error.
4828 : *
4829 : * The following options are supported. See the GEOS library for more detailed
4830 : * descriptions.
4831 : *
4832 : * <ul>
4833 : * <li>ENDCAP_STYLE=ROUND/FLAT/SQUARE</li>
4834 : * <li>JOIN_STYLE=ROUND/MITRE/BEVEL</li>
4835 : * <li>MITRE_LIMIT=double</li>
4836 : * <li>QUADRANT_SEGMENTS=double</li>
4837 : * <li>SINGLE_SIDED=YES/NO</li>
4838 : * </ul>
4839 : *
4840 : * This function is the same as the C++ method OGRGeometry::BufferEx().
4841 : *
4842 : * @param hTarget the geometry.
4843 : * @param dfDist the buffer distance to be applied. Should be expressed into
4844 : * the same unit as the coordinates of the geometry.
4845 : * @param papszOptions NULL terminated list of options (may be NULL)
4846 : *
4847 : * @return a new geometry to be freed by the caller with OGR_G_DestroyGeometry,
4848 : * or NULL if an error occurs.
4849 : *
4850 : * @since GDAL 3.10
4851 : */
4852 :
4853 12 : OGRGeometryH OGR_G_BufferEx(OGRGeometryH hTarget, double dfDist,
4854 : CSLConstList papszOptions)
4855 :
4856 : {
4857 12 : VALIDATE_POINTER1(hTarget, "OGR_G_BufferEx", nullptr);
4858 :
4859 12 : return OGRGeometry::ToHandle(
4860 24 : OGRGeometry::FromHandle(hTarget)->BufferEx(dfDist, papszOptions));
4861 : }
4862 :
4863 : /************************************************************************/
4864 : /* Intersection() */
4865 : /************************************************************************/
4866 :
4867 : /**
4868 : * \brief Compute intersection.
4869 : *
4870 : * Generates a new geometry which is the region of intersection of the
4871 : * two geometries operated on. The Intersects() method can be used to test if
4872 : * two geometries intersect.
4873 : *
4874 : * Geometry validity is not checked. In case you are unsure of the validity
4875 : * of the input geometries, call IsValid() before, otherwise the result might
4876 : * be wrong.
4877 : *
4878 : * This method is the same as the C function OGR_G_Intersection().
4879 : *
4880 : * This method is built on the GEOS library, check it for the definition
4881 : * of the geometry operation.
4882 : * If OGR is built without the GEOS library, this method will always fail,
4883 : * issuing a CPLE_NotSupported error.
4884 : *
4885 : * @param poOtherGeom the other geometry intersected with "this" geometry.
4886 : *
4887 : * @return a new geometry to be freed by the caller, or NULL if there is no
4888 : * intersection or if an error occurs.
4889 : *
4890 : */
4891 :
4892 : OGRGeometry *
4893 901 : OGRGeometry::Intersection(UNUSED_PARAMETER const OGRGeometry *poOtherGeom) const
4894 :
4895 : {
4896 901 : if (IsSFCGALCompatible() || poOtherGeom->IsSFCGALCompatible())
4897 : {
4898 : #ifndef HAVE_SFCGAL
4899 :
4900 0 : CPLError(CE_Failure, CPLE_NotSupported, "SFCGAL support not enabled.");
4901 0 : return nullptr;
4902 :
4903 : #else
4904 :
4905 : sfcgal_geometry_t *poThis = OGRGeometry::OGRexportToSFCGAL(this);
4906 : if (poThis == nullptr)
4907 : return nullptr;
4908 :
4909 : sfcgal_geometry_t *poOther =
4910 : OGRGeometry::OGRexportToSFCGAL(poOtherGeom);
4911 : if (poOther == nullptr)
4912 : {
4913 : sfcgal_geometry_delete(poThis);
4914 : return nullptr;
4915 : }
4916 :
4917 : sfcgal_geometry_t *poRes =
4918 : sfcgal_geometry_intersection_3d(poThis, poOther);
4919 : OGRGeometry *h_prodGeom = SFCGALexportToOGR(poRes);
4920 : if (h_prodGeom != nullptr && getSpatialReference() != nullptr &&
4921 : poOtherGeom->getSpatialReference() != nullptr &&
4922 : poOtherGeom->getSpatialReference()->IsSame(getSpatialReference()))
4923 : h_prodGeom->assignSpatialReference(getSpatialReference());
4924 :
4925 : sfcgal_geometry_delete(poThis);
4926 : sfcgal_geometry_delete(poOther);
4927 : sfcgal_geometry_delete(poRes);
4928 :
4929 : return h_prodGeom;
4930 :
4931 : #endif
4932 : }
4933 :
4934 : else
4935 : {
4936 : #ifndef HAVE_GEOS
4937 :
4938 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
4939 : return nullptr;
4940 :
4941 : #else
4942 901 : return BuildGeometryFromTwoGeoms(this, poOtherGeom, GEOSIntersection_r);
4943 : #endif /* HAVE_GEOS */
4944 : }
4945 : }
4946 :
4947 : /************************************************************************/
4948 : /* OGR_G_Intersection() */
4949 : /************************************************************************/
4950 :
4951 : /**
4952 : * \brief Compute intersection.
4953 : *
4954 : * Generates a new geometry which is the region of intersection of the
4955 : * two geometries operated on. The OGR_G_Intersects() function can be used to
4956 : * test if two geometries intersect.
4957 : *
4958 : * Geometry validity is not checked. In case you are unsure of the validity
4959 : * of the input geometries, call IsValid() before, otherwise the result might
4960 : * be wrong.
4961 : *
4962 : * This function is the same as the C++ method OGRGeometry::Intersection().
4963 : *
4964 : * This function is built on the GEOS library, check it for the definition
4965 : * of the geometry operation.
4966 : * If OGR is built without the GEOS library, this function will always fail,
4967 : * issuing a CPLE_NotSupported error.
4968 : *
4969 : * @param hThis the geometry.
4970 : * @param hOther the other geometry.
4971 : *
4972 : * @return a new geometry to be freed by the caller with OGR_G_DestroyGeometry,
4973 : * or NULL if there is not intersection of if an error occurs.
4974 : */
4975 :
4976 12 : OGRGeometryH OGR_G_Intersection(OGRGeometryH hThis, OGRGeometryH hOther)
4977 :
4978 : {
4979 12 : VALIDATE_POINTER1(hThis, "OGR_G_Intersection", nullptr);
4980 :
4981 24 : return OGRGeometry::ToHandle(OGRGeometry::FromHandle(hThis)->Intersection(
4982 24 : OGRGeometry::FromHandle(hOther)));
4983 : }
4984 :
4985 : /************************************************************************/
4986 : /* Union() */
4987 : /************************************************************************/
4988 :
4989 : /**
4990 : * \brief Compute union.
4991 : *
4992 : * Generates a new geometry which is the region of union of the
4993 : * two geometries operated on.
4994 : *
4995 : * Geometry validity is not checked. In case you are unsure of the validity
4996 : * of the input geometries, call IsValid() before, otherwise the result might
4997 : * be wrong.
4998 : *
4999 : * This method is the same as the C function OGR_G_Union().
5000 : *
5001 : * This method is built on the GEOS library, check it for the definition
5002 : * of the geometry operation.
5003 : * If OGR is built without the GEOS library, this method will always fail,
5004 : * issuing a CPLE_NotSupported error.
5005 : *
5006 : * @param poOtherGeom the other geometry unioned with "this" geometry.
5007 : *
5008 : * @return a new geometry to be freed by the caller, or NULL if an error occurs.
5009 : */
5010 :
5011 : OGRGeometry *
5012 43 : OGRGeometry::Union(UNUSED_PARAMETER const OGRGeometry *poOtherGeom) const
5013 :
5014 : {
5015 43 : if (IsSFCGALCompatible() || poOtherGeom->IsSFCGALCompatible())
5016 : {
5017 : #ifndef HAVE_SFCGAL
5018 :
5019 0 : CPLError(CE_Failure, CPLE_NotSupported, "SFCGAL support not enabled.");
5020 0 : return nullptr;
5021 :
5022 : #else
5023 :
5024 : sfcgal_geometry_t *poThis = OGRGeometry::OGRexportToSFCGAL(this);
5025 : if (poThis == nullptr)
5026 : return nullptr;
5027 :
5028 : sfcgal_geometry_t *poOther =
5029 : OGRGeometry::OGRexportToSFCGAL(poOtherGeom);
5030 : if (poOther == nullptr)
5031 : {
5032 : sfcgal_geometry_delete(poThis);
5033 : return nullptr;
5034 : }
5035 :
5036 : sfcgal_geometry_t *poRes = sfcgal_geometry_union_3d(poThis, poOther);
5037 : OGRGeometry *h_prodGeom = OGRGeometry::SFCGALexportToOGR(poRes);
5038 : if (h_prodGeom != nullptr && getSpatialReference() != nullptr &&
5039 : poOtherGeom->getSpatialReference() != nullptr &&
5040 : poOtherGeom->getSpatialReference()->IsSame(getSpatialReference()))
5041 : h_prodGeom->assignSpatialReference(getSpatialReference());
5042 :
5043 : sfcgal_geometry_delete(poThis);
5044 : sfcgal_geometry_delete(poOther);
5045 : sfcgal_geometry_delete(poRes);
5046 :
5047 : return h_prodGeom;
5048 :
5049 : #endif
5050 : }
5051 :
5052 : else
5053 : {
5054 : #ifndef HAVE_GEOS
5055 :
5056 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
5057 : return nullptr;
5058 :
5059 : #else
5060 43 : return BuildGeometryFromTwoGeoms(this, poOtherGeom, GEOSUnion_r);
5061 : #endif /* HAVE_GEOS */
5062 : }
5063 : }
5064 :
5065 : /************************************************************************/
5066 : /* OGR_G_Union() */
5067 : /************************************************************************/
5068 :
5069 : /**
5070 : * \brief Compute union.
5071 : *
5072 : * Generates a new geometry which is the region of union of the
5073 : * two geometries operated on.
5074 : *
5075 : * Geometry validity is not checked. In case you are unsure of the validity
5076 : * of the input geometries, call IsValid() before, otherwise the result might
5077 : * be wrong.
5078 : *
5079 : * This function is the same as the C++ method OGRGeometry::Union().
5080 : *
5081 : * This function is built on the GEOS library, check it for the definition
5082 : * of the geometry operation.
5083 : * If OGR is built without the GEOS library, this function will always fail,
5084 : * issuing a CPLE_NotSupported error.
5085 : *
5086 : * @param hThis the geometry.
5087 : * @param hOther the other geometry.
5088 : *
5089 : * @return a new geometry to be freed by the caller with OGR_G_DestroyGeometry,
5090 : * or NULL if an error occurs.
5091 : */
5092 :
5093 8 : OGRGeometryH OGR_G_Union(OGRGeometryH hThis, OGRGeometryH hOther)
5094 :
5095 : {
5096 8 : VALIDATE_POINTER1(hThis, "OGR_G_Union", nullptr);
5097 :
5098 8 : return OGRGeometry::ToHandle(
5099 16 : OGRGeometry::FromHandle(hThis)->Union(OGRGeometry::FromHandle(hOther)));
5100 : }
5101 :
5102 : /************************************************************************/
5103 : /* UnionCascaded() */
5104 : /************************************************************************/
5105 :
5106 : /**
5107 : * \brief Compute union using cascading.
5108 : *
5109 : * Geometry validity is not checked. In case you are unsure of the validity
5110 : * of the input geometries, call IsValid() before, otherwise the result might
5111 : * be wrong.
5112 : *
5113 : * The input geometry must be a MultiPolygon.
5114 : *
5115 : * This method is the same as the C function OGR_G_UnionCascaded().
5116 : *
5117 : * This method is built on the GEOS library, check it for the definition
5118 : * of the geometry operation.
5119 : * If OGR is built without the GEOS library, this method will always fail,
5120 : * issuing a CPLE_NotSupported error.
5121 : *
5122 : * @return a new geometry to be freed by the caller, or NULL if an error occurs.
5123 : *
5124 : * @since OGR 1.8.0
5125 : *
5126 : * @deprecated Use UnaryUnion() instead
5127 : */
5128 :
5129 2 : OGRGeometry *OGRGeometry::UnionCascaded() const
5130 :
5131 : {
5132 : #ifndef HAVE_GEOS
5133 :
5134 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
5135 : return nullptr;
5136 : #else
5137 :
5138 : #if GEOS_VERSION_MAJOR == 3 && GEOS_VERSION_MINOR < 11
5139 : if (wkbFlatten(getGeometryType()) == wkbMultiPolygon && IsEmpty())
5140 : {
5141 : // GEOS < 3.11 crashes on an empty multipolygon input
5142 : auto poRet = new OGRGeometryCollection();
5143 : poRet->assignSpatialReference(getSpatialReference());
5144 : return poRet;
5145 : }
5146 : #endif
5147 2 : OGRGeometry *poOGRProduct = nullptr;
5148 :
5149 2 : GEOSContextHandle_t hGEOSCtxt = createGEOSContext();
5150 2 : GEOSGeom hThisGeosGeom = exportToGEOS(hGEOSCtxt);
5151 2 : if (hThisGeosGeom != nullptr)
5152 : {
5153 2 : GEOSGeom hGeosProduct = GEOSUnionCascaded_r(hGEOSCtxt, hThisGeosGeom);
5154 2 : GEOSGeom_destroy_r(hGEOSCtxt, hThisGeosGeom);
5155 :
5156 : poOGRProduct =
5157 2 : BuildGeometryFromGEOS(hGEOSCtxt, hGeosProduct, this, nullptr);
5158 : }
5159 2 : freeGEOSContext(hGEOSCtxt);
5160 :
5161 2 : return poOGRProduct;
5162 :
5163 : #endif // HAVE_GEOS
5164 : }
5165 :
5166 : /************************************************************************/
5167 : /* OGR_G_UnionCascaded() */
5168 : /************************************************************************/
5169 :
5170 : /**
5171 : * \brief Compute union using cascading.
5172 : *
5173 : * Geometry validity is not checked. In case you are unsure of the validity
5174 : * of the input geometries, call IsValid() before, otherwise the result might
5175 : * be wrong.
5176 : *
5177 : * The input geometry must be a MultiPolygon.
5178 : *
5179 : * This function is the same as the C++ method OGRGeometry::UnionCascaded().
5180 : *
5181 : * This function is built on the GEOS library, check it for the definition
5182 : * of the geometry operation.
5183 : * If OGR is built without the GEOS library, this function will always fail,
5184 : * issuing a CPLE_NotSupported error.
5185 : *
5186 : * @param hThis the geometry.
5187 : *
5188 : * @return a new geometry to be freed by the caller with OGR_G_DestroyGeometry,
5189 : * or NULL if an error occurs.
5190 : *
5191 : * @deprecated Use OGR_G_UnaryUnion() instead
5192 : */
5193 :
5194 2 : OGRGeometryH OGR_G_UnionCascaded(OGRGeometryH hThis)
5195 :
5196 : {
5197 2 : VALIDATE_POINTER1(hThis, "OGR_G_UnionCascaded", nullptr);
5198 :
5199 2 : return OGRGeometry::ToHandle(
5200 4 : OGRGeometry::FromHandle(hThis)->UnionCascaded());
5201 : }
5202 :
5203 : /************************************************************************/
5204 : /* UnaryUnion() */
5205 : /************************************************************************/
5206 :
5207 : /**
5208 : * \brief Returns the union of all components of a single geometry.
5209 : *
5210 : * Usually used to convert a collection into the smallest set of polygons that
5211 : * cover the same area.
5212 : *
5213 : * See https://postgis.net/docs/ST_UnaryUnion.html for more details.
5214 : *
5215 : * This method is the same as the C function OGR_G_UnaryUnion().
5216 : *
5217 : * This method is built on the GEOS library, check it for the definition
5218 : * of the geometry operation.
5219 : * If OGR is built without the GEOS library, this method will always fail,
5220 : * issuing a CPLE_NotSupported error.
5221 : *
5222 : * @return a new geometry to be freed by the caller, or NULL if an error occurs.
5223 : *
5224 : * @since GDAL 3.7
5225 : */
5226 :
5227 618 : OGRGeometry *OGRGeometry::UnaryUnion() const
5228 :
5229 : {
5230 : #ifndef HAVE_GEOS
5231 :
5232 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
5233 : return nullptr;
5234 : #else
5235 :
5236 : #if GEOS_VERSION_MAJOR == 3 && GEOS_VERSION_MINOR < 11
5237 : if (IsEmpty())
5238 : {
5239 : // GEOS < 3.11 crashes on an empty geometry
5240 : auto poRet = new OGRGeometryCollection();
5241 : poRet->assignSpatialReference(getSpatialReference());
5242 : return poRet;
5243 : }
5244 : #endif
5245 618 : OGRGeometry *poOGRProduct = nullptr;
5246 :
5247 618 : GEOSContextHandle_t hGEOSCtxt = createGEOSContext();
5248 618 : GEOSGeom hThisGeosGeom = exportToGEOS(hGEOSCtxt);
5249 618 : if (hThisGeosGeom != nullptr)
5250 : {
5251 618 : GEOSGeom hGeosProduct = GEOSUnaryUnion_r(hGEOSCtxt, hThisGeosGeom);
5252 618 : GEOSGeom_destroy_r(hGEOSCtxt, hThisGeosGeom);
5253 :
5254 : poOGRProduct =
5255 618 : BuildGeometryFromGEOS(hGEOSCtxt, hGeosProduct, this, nullptr);
5256 : }
5257 618 : freeGEOSContext(hGEOSCtxt);
5258 :
5259 618 : return poOGRProduct;
5260 :
5261 : #endif // HAVE_GEOS
5262 : }
5263 :
5264 : /************************************************************************/
5265 : /* OGR_G_UnaryUnion() */
5266 : /************************************************************************/
5267 :
5268 : /**
5269 : * \brief Returns the union of all components of a single geometry.
5270 : *
5271 : * Usually used to convert a collection into the smallest set of polygons that
5272 : * cover the same area.
5273 : *
5274 : * See https://postgis.net/docs/ST_UnaryUnion.html for more details.
5275 : *
5276 : * Geometry validity is not checked. In case you are unsure of the validity
5277 : * of the input geometries, call IsValid() before, otherwise the result might
5278 : * be wrong.
5279 : *
5280 : * This function is the same as the C++ method OGRGeometry::UnaryUnion().
5281 : *
5282 : * This function is built on the GEOS library, check it for the definition
5283 : * of the geometry operation.
5284 : * If OGR is built without the GEOS library, this function will always fail,
5285 : * issuing a CPLE_NotSupported error.
5286 : *
5287 : * @param hThis the geometry.
5288 : *
5289 : * @return a new geometry to be freed by the caller with OGR_G_DestroyGeometry,
5290 : * or NULL if an error occurs.
5291 : *
5292 : * @since GDAL 3.7
5293 : */
5294 :
5295 3 : OGRGeometryH OGR_G_UnaryUnion(OGRGeometryH hThis)
5296 :
5297 : {
5298 3 : VALIDATE_POINTER1(hThis, "OGR_G_UnaryUnion", nullptr);
5299 :
5300 3 : return OGRGeometry::ToHandle(OGRGeometry::FromHandle(hThis)->UnaryUnion());
5301 : }
5302 :
5303 : /************************************************************************/
5304 : /* Difference() */
5305 : /************************************************************************/
5306 :
5307 : /**
5308 : * \brief Compute difference.
5309 : *
5310 : * Generates a new geometry which is the region of this geometry with the
5311 : * region of the second geometry removed.
5312 : *
5313 : * Geometry validity is not checked. In case you are unsure of the validity
5314 : * of the input geometries, call IsValid() before, otherwise the result might
5315 : * be wrong.
5316 : *
5317 : * This method is the same as the C function OGR_G_Difference().
5318 : *
5319 : * This method is built on the GEOS library, check it for the definition
5320 : * of the geometry operation.
5321 : * If OGR is built without the GEOS library, this method will always fail,
5322 : * issuing a CPLE_NotSupported error.
5323 : *
5324 : * @param poOtherGeom the other geometry removed from "this" geometry.
5325 : *
5326 : * @return a new geometry to be freed by the caller, or NULL if the difference
5327 : * is empty or if an error occurs.
5328 : */
5329 :
5330 : OGRGeometry *
5331 124 : OGRGeometry::Difference(UNUSED_PARAMETER const OGRGeometry *poOtherGeom) const
5332 :
5333 : {
5334 124 : if (IsSFCGALCompatible() || poOtherGeom->IsSFCGALCompatible())
5335 : {
5336 : #ifndef HAVE_SFCGAL
5337 :
5338 0 : CPLError(CE_Failure, CPLE_NotSupported, "SFCGAL support not enabled.");
5339 0 : return nullptr;
5340 :
5341 : #else
5342 :
5343 : sfcgal_geometry_t *poThis = OGRGeometry::OGRexportToSFCGAL(this);
5344 : if (poThis == nullptr)
5345 : return nullptr;
5346 :
5347 : sfcgal_geometry_t *poOther =
5348 : OGRGeometry::OGRexportToSFCGAL(poOtherGeom);
5349 : if (poOther == nullptr)
5350 : {
5351 : sfcgal_geometry_delete(poThis);
5352 : return nullptr;
5353 : }
5354 :
5355 : sfcgal_geometry_t *poRes =
5356 : sfcgal_geometry_difference_3d(poThis, poOther);
5357 : OGRGeometry *h_prodGeom = OGRGeometry::SFCGALexportToOGR(poRes);
5358 : if (h_prodGeom != nullptr && getSpatialReference() != nullptr &&
5359 : poOtherGeom->getSpatialReference() != nullptr &&
5360 : poOtherGeom->getSpatialReference()->IsSame(getSpatialReference()))
5361 : h_prodGeom->assignSpatialReference(getSpatialReference());
5362 :
5363 : sfcgal_geometry_delete(poThis);
5364 : sfcgal_geometry_delete(poOther);
5365 : sfcgal_geometry_delete(poRes);
5366 :
5367 : return h_prodGeom;
5368 :
5369 : #endif
5370 : }
5371 :
5372 : else
5373 : {
5374 : #ifndef HAVE_GEOS
5375 :
5376 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
5377 : return nullptr;
5378 :
5379 : #else
5380 124 : return BuildGeometryFromTwoGeoms(this, poOtherGeom, GEOSDifference_r);
5381 : #endif /* HAVE_GEOS */
5382 : }
5383 : }
5384 :
5385 : /************************************************************************/
5386 : /* OGR_G_Difference() */
5387 : /************************************************************************/
5388 :
5389 : /**
5390 : * \brief Compute difference.
5391 : *
5392 : * Generates a new geometry which is the region of this geometry with the
5393 : * region of the other geometry removed.
5394 : *
5395 : * Geometry validity is not checked. In case you are unsure of the validity
5396 : * of the input geometries, call IsValid() before, otherwise the result might
5397 : * be wrong.
5398 : *
5399 : * This function is the same as the C++ method OGRGeometry::Difference().
5400 : *
5401 : * This function is built on the GEOS library, check it for the definition
5402 : * of the geometry operation.
5403 : * If OGR is built without the GEOS library, this function will always fail,
5404 : * issuing a CPLE_NotSupported error.
5405 : *
5406 : * @param hThis the geometry.
5407 : * @param hOther the other geometry.
5408 : *
5409 : * @return a new geometry to be freed by the caller with OGR_G_DestroyGeometry,
5410 : * or NULL if the difference is empty or if an error occurs.
5411 : */
5412 :
5413 6 : OGRGeometryH OGR_G_Difference(OGRGeometryH hThis, OGRGeometryH hOther)
5414 :
5415 : {
5416 6 : VALIDATE_POINTER1(hThis, "OGR_G_Difference", nullptr);
5417 :
5418 12 : return OGRGeometry::ToHandle(OGRGeometry::FromHandle(hThis)->Difference(
5419 12 : OGRGeometry::FromHandle(hOther)));
5420 : }
5421 :
5422 : /************************************************************************/
5423 : /* SymDifference() */
5424 : /************************************************************************/
5425 :
5426 : /**
5427 : * \brief Compute symmetric difference.
5428 : *
5429 : * Generates a new geometry which is the symmetric difference of this
5430 : * geometry and the second geometry passed into the method.
5431 : *
5432 : * Geometry validity is not checked. In case you are unsure of the validity
5433 : * of the input geometries, call IsValid() before, otherwise the result might
5434 : * be wrong.
5435 : *
5436 : * This method is the same as the C function OGR_G_SymDifference().
5437 : *
5438 : * This method is built on the GEOS library, check it for the definition
5439 : * of the geometry operation.
5440 : * If OGR is built without the GEOS library, this method will always fail,
5441 : * issuing a CPLE_NotSupported error.
5442 : *
5443 : * @param poOtherGeom the other geometry.
5444 : *
5445 : * @return a new geometry to be freed by the caller, or NULL if the difference
5446 : * is empty or if an error occurs.
5447 : *
5448 : * @since OGR 1.8.0
5449 : */
5450 :
5451 7 : OGRGeometry *OGRGeometry::SymDifference(
5452 : UNUSED_IF_NO_GEOS const OGRGeometry *poOtherGeom) const
5453 :
5454 : {
5455 7 : if (IsSFCGALCompatible() || poOtherGeom->IsSFCGALCompatible())
5456 : {
5457 : #ifndef HAVE_SFCGAL
5458 0 : CPLError(CE_Failure, CPLE_NotSupported, "SFCGAL support not enabled.");
5459 0 : return nullptr;
5460 : #else
5461 : OGRGeometry *poFirstDifference = Difference(poOtherGeom);
5462 : if (poFirstDifference == nullptr)
5463 : return nullptr;
5464 :
5465 : OGRGeometry *poOtherDifference = poOtherGeom->Difference(this);
5466 : if (poOtherDifference == nullptr)
5467 : {
5468 : delete poFirstDifference;
5469 : return nullptr;
5470 : }
5471 :
5472 : OGRGeometry *poSymDiff = poFirstDifference->Union(poOtherDifference);
5473 : delete poFirstDifference;
5474 : delete poOtherDifference;
5475 : return poSymDiff;
5476 : #endif
5477 : }
5478 :
5479 : #ifndef HAVE_GEOS
5480 :
5481 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
5482 : return nullptr;
5483 :
5484 : #else
5485 7 : return BuildGeometryFromTwoGeoms(this, poOtherGeom, GEOSSymDifference_r);
5486 : #endif // HAVE_GEOS
5487 : }
5488 :
5489 : //! @cond Doxygen_Suppress
5490 : /**
5491 : * \brief Compute symmetric difference (deprecated)
5492 : *
5493 : * @deprecated
5494 : *
5495 : * @see OGRGeometry::SymDifference()
5496 : */
5497 : OGRGeometry *
5498 0 : OGRGeometry::SymmetricDifference(const OGRGeometry *poOtherGeom) const
5499 :
5500 : {
5501 0 : return SymDifference(poOtherGeom);
5502 : }
5503 :
5504 : //! @endcond
5505 :
5506 : /************************************************************************/
5507 : /* OGR_G_SymDifference() */
5508 : /************************************************************************/
5509 :
5510 : /**
5511 : * \brief Compute symmetric difference.
5512 : *
5513 : * Generates a new geometry which is the symmetric difference of this
5514 : * geometry and the other geometry.
5515 : *
5516 : * Geometry validity is not checked. In case you are unsure of the validity
5517 : * of the input geometries, call IsValid() before, otherwise the result might
5518 : * be wrong.
5519 : *
5520 : * This function is the same as the C++ method
5521 : * OGRGeometry::SymmetricDifference().
5522 : *
5523 : * This function is built on the GEOS library, check it for the definition
5524 : * of the geometry operation.
5525 : * If OGR is built without the GEOS library, this function will always fail,
5526 : * issuing a CPLE_NotSupported error.
5527 : *
5528 : * @param hThis the geometry.
5529 : * @param hOther the other geometry.
5530 : *
5531 : * @return a new geometry to be freed by the caller with OGR_G_DestroyGeometry,
5532 : * or NULL if the difference is empty or if an error occurs.
5533 : *
5534 : * @since OGR 1.8.0
5535 : */
5536 :
5537 7 : OGRGeometryH OGR_G_SymDifference(OGRGeometryH hThis, OGRGeometryH hOther)
5538 :
5539 : {
5540 7 : VALIDATE_POINTER1(hThis, "OGR_G_SymDifference", nullptr);
5541 :
5542 14 : return OGRGeometry::ToHandle(OGRGeometry::FromHandle(hThis)->SymDifference(
5543 14 : OGRGeometry::FromHandle(hOther)));
5544 : }
5545 :
5546 : /**
5547 : * \brief Compute symmetric difference (deprecated)
5548 : *
5549 : * @deprecated
5550 : *
5551 : * @see OGR_G_SymmetricDifference()
5552 : */
5553 0 : OGRGeometryH OGR_G_SymmetricDifference(OGRGeometryH hThis, OGRGeometryH hOther)
5554 :
5555 : {
5556 0 : VALIDATE_POINTER1(hThis, "OGR_G_SymmetricDifference", nullptr);
5557 :
5558 0 : return OGRGeometry::ToHandle(OGRGeometry::FromHandle(hThis)->SymDifference(
5559 0 : OGRGeometry::FromHandle(hOther)));
5560 : }
5561 :
5562 : /************************************************************************/
5563 : /* Disjoint() */
5564 : /************************************************************************/
5565 :
5566 : /**
5567 : * \brief Test for disjointness.
5568 : *
5569 : * Tests if this geometry and the other passed into the method are disjoint.
5570 : *
5571 : * Geometry validity is not checked. In case you are unsure of the validity
5572 : * of the input geometries, call IsValid() before, otherwise the result might
5573 : * be wrong.
5574 : *
5575 : * This method is the same as the C function OGR_G_Disjoint().
5576 : *
5577 : * This method is built on the GEOS library, check it for the definition
5578 : * of the geometry operation.
5579 : * If OGR is built without the GEOS library, this method will always fail,
5580 : * issuing a CPLE_NotSupported error.
5581 : *
5582 : * @param poOtherGeom the geometry to compare to this geometry.
5583 : *
5584 : * @return TRUE if they are disjoint, otherwise FALSE.
5585 : */
5586 :
5587 : OGRBoolean
5588 8 : OGRGeometry::Disjoint(UNUSED_IF_NO_GEOS const OGRGeometry *poOtherGeom) const
5589 :
5590 : {
5591 : #ifndef HAVE_GEOS
5592 :
5593 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
5594 : return FALSE;
5595 :
5596 : #else
5597 8 : return OGRGEOSBooleanPredicate(this, poOtherGeom, GEOSDisjoint_r);
5598 : #endif // HAVE_GEOS
5599 : }
5600 :
5601 : /************************************************************************/
5602 : /* OGR_G_Disjoint() */
5603 : /************************************************************************/
5604 :
5605 : /**
5606 : * \brief Test for disjointness.
5607 : *
5608 : * Tests if this geometry and the other geometry are disjoint.
5609 : *
5610 : * Geometry validity is not checked. In case you are unsure of the validity
5611 : * of the input geometries, call IsValid() before, otherwise the result might
5612 : * be wrong.
5613 : *
5614 : * This function is the same as the C++ method OGRGeometry::Disjoint().
5615 : *
5616 : * This function is built on the GEOS library, check it for the definition
5617 : * of the geometry operation.
5618 : * If OGR is built without the GEOS library, this function will always fail,
5619 : * issuing a CPLE_NotSupported error.
5620 : *
5621 : * @param hThis the geometry to compare.
5622 : * @param hOther the other geometry to compare.
5623 : *
5624 : * @return TRUE if they are disjoint, otherwise FALSE.
5625 : */
5626 8 : int OGR_G_Disjoint(OGRGeometryH hThis, OGRGeometryH hOther)
5627 :
5628 : {
5629 8 : VALIDATE_POINTER1(hThis, "OGR_G_Disjoint", FALSE);
5630 :
5631 16 : return OGRGeometry::FromHandle(hThis)->Disjoint(
5632 8 : OGRGeometry::FromHandle(hOther));
5633 : }
5634 :
5635 : /************************************************************************/
5636 : /* Touches() */
5637 : /************************************************************************/
5638 :
5639 : /**
5640 : * \brief Test for touching.
5641 : *
5642 : * Tests if this geometry and the other passed into the method are touching.
5643 : *
5644 : * Geometry validity is not checked. In case you are unsure of the validity
5645 : * of the input geometries, call IsValid() before, otherwise the result might
5646 : * be wrong.
5647 : *
5648 : * This method is the same as the C function OGR_G_Touches().
5649 : *
5650 : * This method is built on the GEOS library, check it for the definition
5651 : * of the geometry operation.
5652 : * If OGR is built without the GEOS library, this method will always fail,
5653 : * issuing a CPLE_NotSupported error.
5654 : *
5655 : * @param poOtherGeom the geometry to compare to this geometry.
5656 : *
5657 : * @return TRUE if they are touching, otherwise FALSE.
5658 : */
5659 :
5660 : OGRBoolean
5661 11 : OGRGeometry::Touches(UNUSED_IF_NO_GEOS const OGRGeometry *poOtherGeom) const
5662 :
5663 : {
5664 : #ifndef HAVE_GEOS
5665 :
5666 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
5667 : return FALSE;
5668 :
5669 : #else
5670 11 : return OGRGEOSBooleanPredicate(this, poOtherGeom, GEOSTouches_r);
5671 : #endif // HAVE_GEOS
5672 : }
5673 :
5674 : /************************************************************************/
5675 : /* OGR_G_Touches() */
5676 : /************************************************************************/
5677 : /**
5678 : * \brief Test for touching.
5679 : *
5680 : * Tests if this geometry and the other geometry are touching.
5681 : *
5682 : * Geometry validity is not checked. In case you are unsure of the validity
5683 : * of the input geometries, call IsValid() before, otherwise the result might
5684 : * be wrong.
5685 : *
5686 : * This function is the same as the C++ method OGRGeometry::Touches().
5687 : *
5688 : * This function is built on the GEOS library, check it for the definition
5689 : * of the geometry operation.
5690 : * If OGR is built without the GEOS library, this function will always fail,
5691 : * issuing a CPLE_NotSupported error.
5692 : *
5693 : * @param hThis the geometry to compare.
5694 : * @param hOther the other geometry to compare.
5695 : *
5696 : * @return TRUE if they are touching, otherwise FALSE.
5697 : */
5698 :
5699 8 : int OGR_G_Touches(OGRGeometryH hThis, OGRGeometryH hOther)
5700 :
5701 : {
5702 8 : VALIDATE_POINTER1(hThis, "OGR_G_Touches", FALSE);
5703 :
5704 16 : return OGRGeometry::FromHandle(hThis)->Touches(
5705 8 : OGRGeometry::FromHandle(hOther));
5706 : }
5707 :
5708 : /************************************************************************/
5709 : /* Crosses() */
5710 : /************************************************************************/
5711 :
5712 : /**
5713 : * \brief Test for crossing.
5714 : *
5715 : * Tests if this geometry and the other passed into the method are crossing.
5716 : *
5717 : * Geometry validity is not checked. In case you are unsure of the validity
5718 : * of the input geometries, call IsValid() before, otherwise the result might
5719 : * be wrong.
5720 : *
5721 : * This method is the same as the C function OGR_G_Crosses().
5722 : *
5723 : * This method is built on the GEOS library, check it for the definition
5724 : * of the geometry operation.
5725 : * If OGR is built without the GEOS library, this method will always fail,
5726 : * issuing a CPLE_NotSupported error.
5727 : *
5728 : * @param poOtherGeom the geometry to compare to this geometry.
5729 : *
5730 : * @return TRUE if they are crossing, otherwise FALSE.
5731 : */
5732 :
5733 : OGRBoolean
5734 8 : OGRGeometry::Crosses(UNUSED_PARAMETER const OGRGeometry *poOtherGeom) const
5735 :
5736 : {
5737 8 : if (IsSFCGALCompatible() || poOtherGeom->IsSFCGALCompatible())
5738 : {
5739 : #ifndef HAVE_SFCGAL
5740 :
5741 0 : CPLError(CE_Failure, CPLE_NotSupported, "SFCGAL support not enabled.");
5742 0 : return FALSE;
5743 :
5744 : #else
5745 :
5746 : sfcgal_geometry_t *poThis = OGRGeometry::OGRexportToSFCGAL(this);
5747 : if (poThis == nullptr)
5748 : return FALSE;
5749 :
5750 : sfcgal_geometry_t *poOther =
5751 : OGRGeometry::OGRexportToSFCGAL(poOtherGeom);
5752 : if (poOther == nullptr)
5753 : {
5754 : sfcgal_geometry_delete(poThis);
5755 : return FALSE;
5756 : }
5757 :
5758 : int res = sfcgal_geometry_intersects_3d(poThis, poOther);
5759 :
5760 : sfcgal_geometry_delete(poThis);
5761 : sfcgal_geometry_delete(poOther);
5762 :
5763 : return (res == 1) ? TRUE : FALSE;
5764 :
5765 : #endif
5766 : }
5767 :
5768 : else
5769 : {
5770 :
5771 : #ifndef HAVE_GEOS
5772 :
5773 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
5774 : return FALSE;
5775 :
5776 : #else
5777 8 : return OGRGEOSBooleanPredicate(this, poOtherGeom, GEOSCrosses_r);
5778 : #endif /* HAVE_GEOS */
5779 : }
5780 : }
5781 :
5782 : /************************************************************************/
5783 : /* OGR_G_Crosses() */
5784 : /************************************************************************/
5785 : /**
5786 : * \brief Test for crossing.
5787 : *
5788 : * Tests if this geometry and the other geometry are crossing.
5789 : *
5790 : * Geometry validity is not checked. In case you are unsure of the validity
5791 : * of the input geometries, call IsValid() before, otherwise the result might
5792 : * be wrong.
5793 : *
5794 : * This function is the same as the C++ method OGRGeometry::Crosses().
5795 : *
5796 : * This function is built on the GEOS library, check it for the definition
5797 : * of the geometry operation.
5798 : * If OGR is built without the GEOS library, this function will always fail,
5799 : * issuing a CPLE_NotSupported error.
5800 : *
5801 : * @param hThis the geometry to compare.
5802 : * @param hOther the other geometry to compare.
5803 : *
5804 : * @return TRUE if they are crossing, otherwise FALSE.
5805 : */
5806 :
5807 8 : int OGR_G_Crosses(OGRGeometryH hThis, OGRGeometryH hOther)
5808 :
5809 : {
5810 8 : VALIDATE_POINTER1(hThis, "OGR_G_Crosses", FALSE);
5811 :
5812 16 : return OGRGeometry::FromHandle(hThis)->Crosses(
5813 8 : OGRGeometry::FromHandle(hOther));
5814 : }
5815 :
5816 : /************************************************************************/
5817 : /* Within() */
5818 : /************************************************************************/
5819 :
5820 : /**
5821 : * \brief Test for containment.
5822 : *
5823 : * Tests if actual geometry object is within the passed geometry.
5824 : *
5825 : * Geometry validity is not checked. In case you are unsure of the validity
5826 : * of the input geometries, call IsValid() before, otherwise the result might
5827 : * be wrong.
5828 : *
5829 : * This method is the same as the C function OGR_G_Within().
5830 : *
5831 : * This method is built on the GEOS library, check it for the definition
5832 : * of the geometry operation.
5833 : * If OGR is built without the GEOS library, this method will always fail,
5834 : * issuing a CPLE_NotSupported error.
5835 : *
5836 : * @param poOtherGeom the geometry to compare to this geometry.
5837 : *
5838 : * @return TRUE if poOtherGeom is within this geometry, otherwise FALSE.
5839 : */
5840 :
5841 : OGRBoolean
5842 21190 : OGRGeometry::Within(UNUSED_IF_NO_GEOS const OGRGeometry *poOtherGeom) const
5843 :
5844 : {
5845 : #ifndef HAVE_GEOS
5846 :
5847 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
5848 : return FALSE;
5849 :
5850 : #else
5851 21190 : return OGRGEOSBooleanPredicate(this, poOtherGeom, GEOSWithin_r);
5852 : #endif // HAVE_GEOS
5853 : }
5854 :
5855 : /************************************************************************/
5856 : /* OGR_G_Within() */
5857 : /************************************************************************/
5858 :
5859 : /**
5860 : * \brief Test for containment.
5861 : *
5862 : * Tests if this geometry is within the other geometry.
5863 : *
5864 : * Geometry validity is not checked. In case you are unsure of the validity
5865 : * of the input geometries, call IsValid() before, otherwise the result might
5866 : * be wrong.
5867 : *
5868 : * This function is the same as the C++ method OGRGeometry::Within().
5869 : *
5870 : * This function is built on the GEOS library, check it for the definition
5871 : * of the geometry operation.
5872 : * If OGR is built without the GEOS library, this function will always fail,
5873 : * issuing a CPLE_NotSupported error.
5874 : *
5875 : * @param hThis the geometry to compare.
5876 : * @param hOther the other geometry to compare.
5877 : *
5878 : * @return TRUE if hThis is within hOther, otherwise FALSE.
5879 : */
5880 6154 : int OGR_G_Within(OGRGeometryH hThis, OGRGeometryH hOther)
5881 :
5882 : {
5883 6154 : VALIDATE_POINTER1(hThis, "OGR_G_Within", FALSE);
5884 :
5885 12308 : return OGRGeometry::FromHandle(hThis)->Within(
5886 6154 : OGRGeometry::FromHandle(hOther));
5887 : }
5888 :
5889 : /************************************************************************/
5890 : /* Contains() */
5891 : /************************************************************************/
5892 :
5893 : /**
5894 : * \brief Test for containment.
5895 : *
5896 : * Tests if actual geometry object contains the passed geometry.
5897 : *
5898 : * Geometry validity is not checked. In case you are unsure of the validity
5899 : * of the input geometries, call IsValid() before, otherwise the result might
5900 : * be wrong.
5901 : *
5902 : * This method is the same as the C function OGR_G_Contains().
5903 : *
5904 : * This method is built on the GEOS library, check it for the definition
5905 : * of the geometry operation.
5906 : * If OGR is built without the GEOS library, this method will always fail,
5907 : * issuing a CPLE_NotSupported error.
5908 : *
5909 : * @param poOtherGeom the geometry to compare to this geometry.
5910 : *
5911 : * @return TRUE if poOtherGeom contains this geometry, otherwise FALSE.
5912 : */
5913 :
5914 : OGRBoolean
5915 44 : OGRGeometry::Contains(UNUSED_IF_NO_GEOS const OGRGeometry *poOtherGeom) const
5916 :
5917 : {
5918 : #ifndef HAVE_GEOS
5919 :
5920 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
5921 : return FALSE;
5922 :
5923 : #else
5924 44 : return OGRGEOSBooleanPredicate(this, poOtherGeom, GEOSContains_r);
5925 : #endif // HAVE_GEOS
5926 : }
5927 :
5928 : /************************************************************************/
5929 : /* OGR_G_Contains() */
5930 : /************************************************************************/
5931 :
5932 : /**
5933 : * \brief Test for containment.
5934 : *
5935 : * Tests if this geometry contains the other geometry.
5936 : *
5937 : * Geometry validity is not checked. In case you are unsure of the validity
5938 : * of the input geometries, call IsValid() before, otherwise the result might
5939 : * be wrong.
5940 : *
5941 : * This function is the same as the C++ method OGRGeometry::Contains().
5942 : *
5943 : * This function is built on the GEOS library, check it for the definition
5944 : * of the geometry operation.
5945 : * If OGR is built without the GEOS library, this function will always fail,
5946 : * issuing a CPLE_NotSupported error.
5947 : *
5948 : * @param hThis the geometry to compare.
5949 : * @param hOther the other geometry to compare.
5950 : *
5951 : * @return TRUE if hThis contains hOther geometry, otherwise FALSE.
5952 : */
5953 15 : int OGR_G_Contains(OGRGeometryH hThis, OGRGeometryH hOther)
5954 :
5955 : {
5956 15 : VALIDATE_POINTER1(hThis, "OGR_G_Contains", FALSE);
5957 :
5958 30 : return OGRGeometry::FromHandle(hThis)->Contains(
5959 15 : OGRGeometry::FromHandle(hOther));
5960 : }
5961 :
5962 : /************************************************************************/
5963 : /* Overlaps() */
5964 : /************************************************************************/
5965 :
5966 : /**
5967 : * \brief Test for overlap.
5968 : *
5969 : * Tests if this geometry and the other passed into the method overlap, that is
5970 : * their intersection has a non-zero area.
5971 : *
5972 : * Geometry validity is not checked. In case you are unsure of the validity
5973 : * of the input geometries, call IsValid() before, otherwise the result might
5974 : * be wrong.
5975 : *
5976 : * This method is the same as the C function OGR_G_Overlaps().
5977 : *
5978 : * This method is built on the GEOS library, check it for the definition
5979 : * of the geometry operation.
5980 : * If OGR is built without the GEOS library, this method will always fail,
5981 : * issuing a CPLE_NotSupported error.
5982 : *
5983 : * @param poOtherGeom the geometry to compare to this geometry.
5984 : *
5985 : * @return TRUE if they are overlapping, otherwise FALSE.
5986 : */
5987 :
5988 : OGRBoolean
5989 7 : OGRGeometry::Overlaps(UNUSED_IF_NO_GEOS const OGRGeometry *poOtherGeom) const
5990 :
5991 : {
5992 : #ifndef HAVE_GEOS
5993 :
5994 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
5995 : return FALSE;
5996 :
5997 : #else
5998 7 : return OGRGEOSBooleanPredicate(this, poOtherGeom, GEOSOverlaps_r);
5999 : #endif // HAVE_GEOS
6000 : }
6001 :
6002 : /************************************************************************/
6003 : /* OGR_G_Overlaps() */
6004 : /************************************************************************/
6005 : /**
6006 : * \brief Test for overlap.
6007 : *
6008 : * Tests if this geometry and the other geometry overlap, that is their
6009 : * intersection has a non-zero area.
6010 : *
6011 : * Geometry validity is not checked. In case you are unsure of the validity
6012 : * of the input geometries, call IsValid() before, otherwise the result might
6013 : * be wrong.
6014 : *
6015 : * This function is the same as the C++ method OGRGeometry::Overlaps().
6016 : *
6017 : * This function is built on the GEOS library, check it for the definition
6018 : * of the geometry operation.
6019 : * If OGR is built without the GEOS library, this function will always fail,
6020 : * issuing a CPLE_NotSupported error.
6021 : *
6022 : * @param hThis the geometry to compare.
6023 : * @param hOther the other geometry to compare.
6024 : *
6025 : * @return TRUE if they are overlapping, otherwise FALSE.
6026 : */
6027 :
6028 7 : int OGR_G_Overlaps(OGRGeometryH hThis, OGRGeometryH hOther)
6029 :
6030 : {
6031 7 : VALIDATE_POINTER1(hThis, "OGR_G_Overlaps", FALSE);
6032 :
6033 14 : return OGRGeometry::FromHandle(hThis)->Overlaps(
6034 7 : OGRGeometry::FromHandle(hOther));
6035 : }
6036 :
6037 : /************************************************************************/
6038 : /* closeRings() */
6039 : /************************************************************************/
6040 :
6041 : /**
6042 : * \brief Force rings to be closed.
6043 : *
6044 : * If this geometry, or any contained geometries has polygon rings that
6045 : * are not closed, they will be closed by adding the starting point at
6046 : * the end.
6047 : */
6048 :
6049 1265 : void OGRGeometry::closeRings()
6050 : {
6051 1265 : }
6052 :
6053 : /************************************************************************/
6054 : /* OGR_G_CloseRings() */
6055 : /************************************************************************/
6056 :
6057 : /**
6058 : * \brief Force rings to be closed.
6059 : *
6060 : * If this geometry, or any contained geometries has polygon rings that
6061 : * are not closed, they will be closed by adding the starting point at
6062 : * the end.
6063 : *
6064 : * @param hGeom handle to the geometry.
6065 : */
6066 :
6067 6 : void OGR_G_CloseRings(OGRGeometryH hGeom)
6068 :
6069 : {
6070 6 : VALIDATE_POINTER0(hGeom, "OGR_G_CloseRings");
6071 :
6072 6 : OGRGeometry::FromHandle(hGeom)->closeRings();
6073 : }
6074 :
6075 : /************************************************************************/
6076 : /* Centroid() */
6077 : /************************************************************************/
6078 :
6079 : /**
6080 : * \brief Compute the geometry centroid.
6081 : *
6082 : * The centroid location is applied to the passed in OGRPoint object.
6083 : * The centroid is not necessarily within the geometry.
6084 : *
6085 : * This method relates to the SFCOM ISurface::get_Centroid() method
6086 : * however the current implementation based on GEOS can operate on other
6087 : * geometry types such as multipoint, linestring, geometrycollection such as
6088 : * multipolygons.
6089 : * OGC SF SQL 1.1 defines the operation for surfaces (polygons).
6090 : * SQL/MM-Part 3 defines the operation for surfaces and multisurfaces
6091 : * (multipolygons).
6092 : *
6093 : * This function is the same as the C function OGR_G_Centroid().
6094 : *
6095 : * This function is built on the GEOS library, check it for the definition
6096 : * of the geometry operation.
6097 : * If OGR is built without the GEOS library, this function will always fail,
6098 : * issuing a CPLE_NotSupported error.
6099 : *
6100 : * @return OGRERR_NONE on success or OGRERR_FAILURE on error.
6101 : *
6102 : * @since OGR 1.8.0 as a OGRGeometry method (previously was restricted
6103 : * to OGRPolygon)
6104 : */
6105 :
6106 5 : OGRErr OGRGeometry::Centroid(OGRPoint *poPoint) const
6107 :
6108 : {
6109 5 : if (poPoint == nullptr)
6110 0 : return OGRERR_FAILURE;
6111 :
6112 : #ifndef HAVE_GEOS
6113 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
6114 : return OGRERR_FAILURE;
6115 :
6116 : #else
6117 :
6118 5 : GEOSContextHandle_t hGEOSCtxt = createGEOSContext();
6119 : GEOSGeom hThisGeosGeom =
6120 5 : exportToGEOS(hGEOSCtxt, /* bRemoveEmptyParts = */ true);
6121 :
6122 5 : if (hThisGeosGeom != nullptr)
6123 : {
6124 5 : GEOSGeom hOtherGeosGeom = GEOSGetCentroid_r(hGEOSCtxt, hThisGeosGeom);
6125 5 : GEOSGeom_destroy_r(hGEOSCtxt, hThisGeosGeom);
6126 :
6127 5 : if (hOtherGeosGeom == nullptr)
6128 : {
6129 0 : freeGEOSContext(hGEOSCtxt);
6130 0 : return OGRERR_FAILURE;
6131 : }
6132 :
6133 : OGRGeometry *poCentroidGeom =
6134 5 : OGRGeometryFactory::createFromGEOS(hGEOSCtxt, hOtherGeosGeom);
6135 :
6136 5 : GEOSGeom_destroy_r(hGEOSCtxt, hOtherGeosGeom);
6137 :
6138 5 : if (poCentroidGeom == nullptr)
6139 : {
6140 0 : freeGEOSContext(hGEOSCtxt);
6141 0 : return OGRERR_FAILURE;
6142 : }
6143 5 : if (wkbFlatten(poCentroidGeom->getGeometryType()) != wkbPoint)
6144 : {
6145 0 : delete poCentroidGeom;
6146 0 : freeGEOSContext(hGEOSCtxt);
6147 0 : return OGRERR_FAILURE;
6148 : }
6149 :
6150 5 : if (getSpatialReference() != nullptr)
6151 0 : poCentroidGeom->assignSpatialReference(getSpatialReference());
6152 :
6153 5 : OGRPoint *poCentroid = poCentroidGeom->toPoint();
6154 :
6155 5 : if (!poCentroid->IsEmpty())
6156 : {
6157 4 : poPoint->setX(poCentroid->getX());
6158 4 : poPoint->setY(poCentroid->getY());
6159 : }
6160 : else
6161 : {
6162 1 : poPoint->empty();
6163 : }
6164 :
6165 5 : delete poCentroidGeom;
6166 :
6167 5 : freeGEOSContext(hGEOSCtxt);
6168 5 : return OGRERR_NONE;
6169 : }
6170 : else
6171 : {
6172 0 : freeGEOSContext(hGEOSCtxt);
6173 0 : return OGRERR_FAILURE;
6174 : }
6175 :
6176 : #endif // HAVE_GEOS
6177 : }
6178 :
6179 : /************************************************************************/
6180 : /* OGR_G_Centroid() */
6181 : /************************************************************************/
6182 :
6183 : /**
6184 : * \brief Compute the geometry centroid.
6185 : *
6186 : * The centroid location is applied to the passed in OGRPoint object.
6187 : * The centroid is not necessarily within the geometry.
6188 : *
6189 : * This method relates to the SFCOM ISurface::get_Centroid() method
6190 : * however the current implementation based on GEOS can operate on other
6191 : * geometry types such as multipoint, linestring, geometrycollection such as
6192 : * multipolygons.
6193 : * OGC SF SQL 1.1 defines the operation for surfaces (polygons).
6194 : * SQL/MM-Part 3 defines the operation for surfaces and multisurfaces
6195 : * (multipolygons).
6196 : *
6197 : * This function is the same as the C++ method OGRGeometry::Centroid().
6198 : *
6199 : * This function is built on the GEOS library, check it for the definition
6200 : * of the geometry operation.
6201 : * If OGR is built without the GEOS library, this function will always fail,
6202 : * issuing a CPLE_NotSupported error.
6203 : *
6204 : * @return OGRERR_NONE on success or OGRERR_FAILURE on error.
6205 : */
6206 :
6207 5 : int OGR_G_Centroid(OGRGeometryH hGeom, OGRGeometryH hCentroidPoint)
6208 :
6209 : {
6210 5 : VALIDATE_POINTER1(hGeom, "OGR_G_Centroid", OGRERR_FAILURE);
6211 :
6212 5 : OGRGeometry *poCentroidGeom = OGRGeometry::FromHandle(hCentroidPoint);
6213 5 : if (poCentroidGeom == nullptr)
6214 0 : return OGRERR_FAILURE;
6215 5 : if (wkbFlatten(poCentroidGeom->getGeometryType()) != wkbPoint)
6216 : {
6217 0 : CPLError(CE_Failure, CPLE_AppDefined,
6218 : "Passed wrong geometry type as centroid argument.");
6219 0 : return OGRERR_FAILURE;
6220 : }
6221 :
6222 5 : return OGRGeometry::FromHandle(hGeom)->Centroid(poCentroidGeom->toPoint());
6223 : }
6224 :
6225 : /************************************************************************/
6226 : /* OGR_G_PointOnSurface() */
6227 : /************************************************************************/
6228 :
6229 : /**
6230 : * \brief Returns a point guaranteed to lie on the surface.
6231 : *
6232 : * This method relates to the SFCOM ISurface::get_PointOnSurface() method
6233 : * however the current implementation based on GEOS can operate on other
6234 : * geometry types than the types that are supported by SQL/MM-Part 3 :
6235 : * surfaces (polygons) and multisurfaces (multipolygons).
6236 : *
6237 : * This method is built on the GEOS library, check it for the definition
6238 : * of the geometry operation.
6239 : * If OGR is built without the GEOS library, this method will always fail,
6240 : * issuing a CPLE_NotSupported error.
6241 : *
6242 : * @param hGeom the geometry to operate on.
6243 : * @return a new geometry to be freed by the caller with OGR_G_DestroyGeometry,
6244 : * or NULL if an error occurs.
6245 : *
6246 : * @since OGR 1.10
6247 : */
6248 :
6249 4 : OGRGeometryH OGR_G_PointOnSurface(OGRGeometryH hGeom)
6250 :
6251 : {
6252 4 : VALIDATE_POINTER1(hGeom, "OGR_G_PointOnSurface", nullptr);
6253 :
6254 : #ifndef HAVE_GEOS
6255 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
6256 : return nullptr;
6257 : #else
6258 :
6259 4 : OGRGeometry *poThis = OGRGeometry::FromHandle(hGeom);
6260 :
6261 4 : GEOSContextHandle_t hGEOSCtxt = OGRGeometry::createGEOSContext();
6262 4 : GEOSGeom hThisGeosGeom = poThis->exportToGEOS(hGEOSCtxt);
6263 :
6264 4 : if (hThisGeosGeom != nullptr)
6265 : {
6266 : GEOSGeom hOtherGeosGeom =
6267 4 : GEOSPointOnSurface_r(hGEOSCtxt, hThisGeosGeom);
6268 4 : GEOSGeom_destroy_r(hGEOSCtxt, hThisGeosGeom);
6269 :
6270 4 : if (hOtherGeosGeom == nullptr)
6271 : {
6272 0 : OGRGeometry::freeGEOSContext(hGEOSCtxt);
6273 0 : return nullptr;
6274 : }
6275 :
6276 : OGRGeometry *poInsidePointGeom =
6277 4 : OGRGeometryFactory::createFromGEOS(hGEOSCtxt, hOtherGeosGeom);
6278 :
6279 4 : GEOSGeom_destroy_r(hGEOSCtxt, hOtherGeosGeom);
6280 :
6281 4 : if (poInsidePointGeom == nullptr)
6282 : {
6283 0 : OGRGeometry::freeGEOSContext(hGEOSCtxt);
6284 0 : return nullptr;
6285 : }
6286 4 : if (wkbFlatten(poInsidePointGeom->getGeometryType()) != wkbPoint)
6287 : {
6288 0 : delete poInsidePointGeom;
6289 0 : OGRGeometry::freeGEOSContext(hGEOSCtxt);
6290 0 : return nullptr;
6291 : }
6292 :
6293 4 : if (poThis->getSpatialReference() != nullptr)
6294 0 : poInsidePointGeom->assignSpatialReference(
6295 0 : poThis->getSpatialReference());
6296 :
6297 4 : OGRGeometry::freeGEOSContext(hGEOSCtxt);
6298 4 : return OGRGeometry::ToHandle(poInsidePointGeom);
6299 : }
6300 :
6301 0 : OGRGeometry::freeGEOSContext(hGEOSCtxt);
6302 0 : return nullptr;
6303 : #endif
6304 : }
6305 :
6306 : /************************************************************************/
6307 : /* PointOnSurfaceInternal() */
6308 : /************************************************************************/
6309 :
6310 : //! @cond Doxygen_Suppress
6311 0 : OGRErr OGRGeometry::PointOnSurfaceInternal(OGRPoint *poPoint) const
6312 : {
6313 0 : if (poPoint == nullptr || poPoint->IsEmpty())
6314 0 : return OGRERR_FAILURE;
6315 :
6316 0 : OGRGeometryH hInsidePoint = OGR_G_PointOnSurface(
6317 : OGRGeometry::ToHandle(const_cast<OGRGeometry *>(this)));
6318 0 : if (hInsidePoint == nullptr)
6319 0 : return OGRERR_FAILURE;
6320 :
6321 0 : OGRPoint *poInsidePoint = OGRGeometry::FromHandle(hInsidePoint)->toPoint();
6322 0 : if (poInsidePoint->IsEmpty())
6323 : {
6324 0 : poPoint->empty();
6325 : }
6326 : else
6327 : {
6328 0 : poPoint->setX(poInsidePoint->getX());
6329 0 : poPoint->setY(poInsidePoint->getY());
6330 : }
6331 :
6332 0 : OGR_G_DestroyGeometry(hInsidePoint);
6333 :
6334 0 : return OGRERR_NONE;
6335 : }
6336 :
6337 : //! @endcond
6338 :
6339 : /************************************************************************/
6340 : /* Simplify() */
6341 : /************************************************************************/
6342 :
6343 : /**
6344 : * \brief Simplify the geometry.
6345 : *
6346 : * This function is the same as the C function OGR_G_Simplify().
6347 : *
6348 : * This function is built on the GEOS library, check it for the definition
6349 : * of the geometry operation.
6350 : * If OGR is built without the GEOS library, this function will always fail,
6351 : * issuing a CPLE_NotSupported error.
6352 : *
6353 : * @param dTolerance the distance tolerance for the simplification.
6354 : *
6355 : * @return a new geometry to be freed by the caller, or NULL if an error occurs.
6356 : *
6357 : * @since OGR 1.8.0
6358 : */
6359 :
6360 33 : OGRGeometry *OGRGeometry::Simplify(UNUSED_IF_NO_GEOS double dTolerance) const
6361 :
6362 : {
6363 : #ifndef HAVE_GEOS
6364 :
6365 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
6366 : return nullptr;
6367 :
6368 : #else
6369 33 : OGRGeometry *poOGRProduct = nullptr;
6370 :
6371 33 : GEOSContextHandle_t hGEOSCtxt = createGEOSContext();
6372 33 : GEOSGeom hThisGeosGeom = exportToGEOS(hGEOSCtxt);
6373 33 : if (hThisGeosGeom != nullptr)
6374 : {
6375 : GEOSGeom hGeosProduct =
6376 33 : GEOSSimplify_r(hGEOSCtxt, hThisGeosGeom, dTolerance);
6377 33 : GEOSGeom_destroy_r(hGEOSCtxt, hThisGeosGeom);
6378 : poOGRProduct =
6379 33 : BuildGeometryFromGEOS(hGEOSCtxt, hGeosProduct, this, nullptr);
6380 : }
6381 33 : freeGEOSContext(hGEOSCtxt);
6382 33 : return poOGRProduct;
6383 :
6384 : #endif // HAVE_GEOS
6385 : }
6386 :
6387 : /************************************************************************/
6388 : /* OGR_G_Simplify() */
6389 : /************************************************************************/
6390 :
6391 : /**
6392 : * \brief Compute a simplified geometry.
6393 : *
6394 : * This function is the same as the C++ method OGRGeometry::Simplify().
6395 : *
6396 : * This function is built on the GEOS library, check it for the definition
6397 : * of the geometry operation.
6398 : * If OGR is built without the GEOS library, this function will always fail,
6399 : * issuing a CPLE_NotSupported error.
6400 : *
6401 : * @param hThis the geometry.
6402 : * @param dTolerance the distance tolerance for the simplification.
6403 : *
6404 : * @return a new geometry to be freed by the caller with OGR_G_DestroyGeometry,
6405 : * or NULL if an error occurs.
6406 : *
6407 : * @since OGR 1.8.0
6408 : */
6409 :
6410 1 : OGRGeometryH OGR_G_Simplify(OGRGeometryH hThis, double dTolerance)
6411 :
6412 : {
6413 1 : VALIDATE_POINTER1(hThis, "OGR_G_Simplify", nullptr);
6414 1 : return OGRGeometry::ToHandle(
6415 2 : OGRGeometry::FromHandle(hThis)->Simplify(dTolerance));
6416 : }
6417 :
6418 : /************************************************************************/
6419 : /* SimplifyPreserveTopology() */
6420 : /************************************************************************/
6421 :
6422 : /**
6423 : * \brief Simplify the geometry while preserving topology.
6424 : *
6425 : * This function is the same as the C function OGR_G_SimplifyPreserveTopology().
6426 : *
6427 : * This function is built on the GEOS library, check it for the definition
6428 : * of the geometry operation.
6429 : * If OGR is built without the GEOS library, this function will always fail,
6430 : * issuing a CPLE_NotSupported error.
6431 : *
6432 : * @param dTolerance the distance tolerance for the simplification.
6433 : *
6434 : * @return a new geometry to be freed by the caller, or NULL if an error occurs.
6435 : *
6436 : * @since OGR 1.9.0
6437 : */
6438 :
6439 : OGRGeometry *
6440 14 : OGRGeometry::SimplifyPreserveTopology(UNUSED_IF_NO_GEOS double dTolerance) const
6441 :
6442 : {
6443 : #ifndef HAVE_GEOS
6444 :
6445 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
6446 : return nullptr;
6447 :
6448 : #else
6449 14 : OGRGeometry *poOGRProduct = nullptr;
6450 :
6451 14 : GEOSContextHandle_t hGEOSCtxt = createGEOSContext();
6452 14 : GEOSGeom hThisGeosGeom = exportToGEOS(hGEOSCtxt);
6453 14 : if (hThisGeosGeom != nullptr)
6454 : {
6455 14 : GEOSGeom hGeosProduct = GEOSTopologyPreserveSimplify_r(
6456 : hGEOSCtxt, hThisGeosGeom, dTolerance);
6457 14 : GEOSGeom_destroy_r(hGEOSCtxt, hThisGeosGeom);
6458 : poOGRProduct =
6459 14 : BuildGeometryFromGEOS(hGEOSCtxt, hGeosProduct, this, nullptr);
6460 : }
6461 14 : freeGEOSContext(hGEOSCtxt);
6462 14 : return poOGRProduct;
6463 :
6464 : #endif // HAVE_GEOS
6465 : }
6466 :
6467 : /************************************************************************/
6468 : /* OGR_G_SimplifyPreserveTopology() */
6469 : /************************************************************************/
6470 :
6471 : /**
6472 : * \brief Simplify the geometry while preserving topology.
6473 : *
6474 : * This function is the same as the C++ method
6475 : * OGRGeometry::SimplifyPreserveTopology().
6476 : *
6477 : * This function is built on the GEOS library, check it for the definition
6478 : * of the geometry operation.
6479 : * If OGR is built without the GEOS library, this function will always fail,
6480 : * issuing a CPLE_NotSupported error.
6481 : *
6482 : * @param hThis the geometry.
6483 : * @param dTolerance the distance tolerance for the simplification.
6484 : *
6485 : * @return a new geometry to be freed by the caller with OGR_G_DestroyGeometry,
6486 : * or NULL if an error occurs.
6487 : *
6488 : * @since OGR 1.9.0
6489 : */
6490 :
6491 1 : OGRGeometryH OGR_G_SimplifyPreserveTopology(OGRGeometryH hThis,
6492 : double dTolerance)
6493 :
6494 : {
6495 1 : VALIDATE_POINTER1(hThis, "OGR_G_SimplifyPreserveTopology", nullptr);
6496 1 : return OGRGeometry::ToHandle(
6497 1 : OGRGeometry::FromHandle(hThis)->SimplifyPreserveTopology(dTolerance));
6498 : }
6499 :
6500 : /************************************************************************/
6501 : /* roundCoordinates() */
6502 : /************************************************************************/
6503 :
6504 : /** Round coordinates of the geometry to the specified precision.
6505 : *
6506 : * Note that this is not the same as OGRGeometry::SetPrecision(). The later
6507 : * will return valid geometries, whereas roundCoordinates() does not make
6508 : * such guarantee and may return geometries with invalidities, if they are
6509 : * not compatible of the specified precision. roundCoordinates() supports
6510 : * curve geometries, whereas SetPrecision() does not currently.
6511 : *
6512 : * One use case for roundCoordinates() is to undo the effect of
6513 : * quantizeCoordinates().
6514 : *
6515 : * @param sPrecision Contains the precision requirements.
6516 : * @since GDAL 3.9
6517 : */
6518 39 : void OGRGeometry::roundCoordinates(const OGRGeomCoordinatePrecision &sPrecision)
6519 : {
6520 : struct Rounder : public OGRDefaultGeometryVisitor
6521 : {
6522 : const OGRGeomCoordinatePrecision &m_precision;
6523 : const double m_invXYResolution;
6524 : const double m_invZResolution;
6525 : const double m_invMResolution;
6526 :
6527 39 : explicit Rounder(const OGRGeomCoordinatePrecision &sPrecisionIn)
6528 39 : : m_precision(sPrecisionIn),
6529 39 : m_invXYResolution(m_precision.dfXYResolution !=
6530 : OGRGeomCoordinatePrecision::UNKNOWN
6531 39 : ? 1.0 / m_precision.dfXYResolution
6532 : : 0.0),
6533 39 : m_invZResolution(m_precision.dfZResolution !=
6534 : OGRGeomCoordinatePrecision::UNKNOWN
6535 39 : ? 1.0 / m_precision.dfZResolution
6536 : : 0.0),
6537 39 : m_invMResolution(m_precision.dfMResolution !=
6538 : OGRGeomCoordinatePrecision::UNKNOWN
6539 39 : ? 1.0 / m_precision.dfMResolution
6540 117 : : 0.0)
6541 : {
6542 39 : }
6543 :
6544 : using OGRDefaultGeometryVisitor::visit;
6545 :
6546 379 : void visit(OGRPoint *poPoint) override
6547 : {
6548 379 : if (m_precision.dfXYResolution !=
6549 : OGRGeomCoordinatePrecision::UNKNOWN)
6550 : {
6551 379 : poPoint->setX(std::round(poPoint->getX() * m_invXYResolution) *
6552 379 : m_precision.dfXYResolution);
6553 379 : poPoint->setY(std::round(poPoint->getY() * m_invXYResolution) *
6554 379 : m_precision.dfXYResolution);
6555 : }
6556 758 : if (m_precision.dfZResolution !=
6557 383 : OGRGeomCoordinatePrecision::UNKNOWN &&
6558 4 : poPoint->Is3D())
6559 : {
6560 4 : poPoint->setZ(std::round(poPoint->getZ() * m_invZResolution) *
6561 4 : m_precision.dfZResolution);
6562 : }
6563 758 : if (m_precision.dfMResolution !=
6564 383 : OGRGeomCoordinatePrecision::UNKNOWN &&
6565 4 : poPoint->IsMeasured())
6566 : {
6567 4 : poPoint->setM(std::round(poPoint->getM() * m_invMResolution) *
6568 4 : m_precision.dfMResolution);
6569 : }
6570 379 : }
6571 : };
6572 :
6573 78 : Rounder rounder(sPrecision);
6574 39 : accept(&rounder);
6575 39 : }
6576 :
6577 : /************************************************************************/
6578 : /* SetPrecision() */
6579 : /************************************************************************/
6580 :
6581 : /** Set the geometry's precision, rounding all its coordinates to the precision
6582 : * grid, and making sure the geometry is still valid.
6583 : *
6584 : * This is a stronger version of roundCoordinates().
6585 : *
6586 : * Note that at time of writing GEOS does no supported curve geometries. So
6587 : * currently if this function is called on such a geometry, OGR will first call
6588 : * getLinearGeometry() on the input and getCurveGeometry() on the output, but
6589 : * that it is unlikely to yield to the expected result.
6590 : *
6591 : * This function is the same as the C function OGR_G_SetPrecision().
6592 : *
6593 : * This function is built on the GEOSGeom_setPrecision_r() function of the
6594 : * GEOS library. Check it for the definition of the geometry operation.
6595 : * If OGR is built without the GEOS library, this function will always fail,
6596 : * issuing a CPLE_NotSupported error.
6597 : *
6598 : * @param dfGridSize size of the precision grid, or 0 for FLOATING
6599 : * precision.
6600 : * @param nFlags The bitwise OR of zero, one or several of OGR_GEOS_PREC_NO_TOPO
6601 : * and OGR_GEOS_PREC_KEEP_COLLAPSED
6602 : *
6603 : * @return a new geometry to be freed by the caller, or NULL if an error occurs.
6604 : *
6605 : * @since GDAL 3.9
6606 : */
6607 :
6608 6 : OGRGeometry *OGRGeometry::SetPrecision(UNUSED_IF_NO_GEOS double dfGridSize,
6609 : UNUSED_IF_NO_GEOS int nFlags) const
6610 : {
6611 : #ifndef HAVE_GEOS
6612 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
6613 : return nullptr;
6614 :
6615 : #else
6616 6 : OGRGeometry *poOGRProduct = nullptr;
6617 :
6618 6 : GEOSContextHandle_t hGEOSCtxt = createGEOSContext();
6619 6 : GEOSGeom hThisGeosGeom = exportToGEOS(hGEOSCtxt);
6620 6 : if (hThisGeosGeom != nullptr)
6621 : {
6622 6 : GEOSGeom hGeosProduct = GEOSGeom_setPrecision_r(
6623 : hGEOSCtxt, hThisGeosGeom, dfGridSize, nFlags);
6624 6 : GEOSGeom_destroy_r(hGEOSCtxt, hThisGeosGeom);
6625 : poOGRProduct =
6626 6 : BuildGeometryFromGEOS(hGEOSCtxt, hGeosProduct, this, nullptr);
6627 : }
6628 6 : freeGEOSContext(hGEOSCtxt);
6629 6 : return poOGRProduct;
6630 :
6631 : #endif // HAVE_GEOS
6632 : }
6633 :
6634 : /************************************************************************/
6635 : /* OGR_G_SetPrecision() */
6636 : /************************************************************************/
6637 :
6638 : /** Set the geometry's precision, rounding all its coordinates to the precision
6639 : * grid, and making sure the geometry is still valid.
6640 : *
6641 : * This is a stronger version of roundCoordinates().
6642 : *
6643 : * Note that at time of writing GEOS does no supported curve geometries. So
6644 : * currently if this function is called on such a geometry, OGR will first call
6645 : * getLinearGeometry() on the input and getCurveGeometry() on the output, but
6646 : * that it is unlikely to yield to the expected result.
6647 : *
6648 : * This function is the same as the C++ method OGRGeometry::SetPrecision().
6649 : *
6650 : * This function is built on the GEOSGeom_setPrecision_r() function of the
6651 : * GEOS library. Check it for the definition of the geometry operation.
6652 : * If OGR is built without the GEOS library, this function will always fail,
6653 : * issuing a CPLE_NotSupported error.
6654 : *
6655 : * @param hThis the geometry.
6656 : * @param dfGridSize size of the precision grid, or 0 for FLOATING
6657 : * precision.
6658 : * @param nFlags The bitwise OR of zero, one or several of OGR_GEOS_PREC_NO_TOPO
6659 : * and OGR_GEOS_PREC_KEEP_COLLAPSED
6660 : *
6661 : * @return a new geometry to be freed by the caller with OGR_G_DestroyGeometry,
6662 : * or NULL if an error occurs.
6663 : *
6664 : * @since GDAL 3.9
6665 : */
6666 1 : OGRGeometryH OGR_G_SetPrecision(OGRGeometryH hThis, double dfGridSize,
6667 : int nFlags)
6668 : {
6669 1 : VALIDATE_POINTER1(hThis, "OGR_G_SetPrecision", nullptr);
6670 1 : return OGRGeometry::ToHandle(
6671 1 : OGRGeometry::FromHandle(hThis)->SetPrecision(dfGridSize, nFlags));
6672 : }
6673 :
6674 : /************************************************************************/
6675 : /* DelaunayTriangulation() */
6676 : /************************************************************************/
6677 :
6678 : /**
6679 : * \brief Return a Delaunay triangulation of the vertices of the geometry.
6680 : *
6681 : * This function is the same as the C function OGR_G_DelaunayTriangulation().
6682 : *
6683 : * This function is built on the GEOS library, v3.4 or above.
6684 : * If OGR is built without the GEOS library, this function will always fail,
6685 : * issuing a CPLE_NotSupported error.
6686 : *
6687 : * @param dfTolerance optional snapping tolerance to use for improved robustness
6688 : * @param bOnlyEdges if TRUE, will return a MULTILINESTRING, otherwise it will
6689 : * return a GEOMETRYCOLLECTION containing triangular POLYGONs.
6690 : *
6691 : * @return a new geometry to be freed by the caller, or NULL if an error occurs.
6692 : *
6693 : * @since OGR 2.1
6694 : */
6695 :
6696 : #ifndef HAVE_GEOS
6697 : OGRGeometry *OGRGeometry::DelaunayTriangulation(double /*dfTolerance*/,
6698 : int /*bOnlyEdges*/) const
6699 : {
6700 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
6701 : return nullptr;
6702 : }
6703 : #else
6704 1 : OGRGeometry *OGRGeometry::DelaunayTriangulation(double dfTolerance,
6705 : int bOnlyEdges) const
6706 : {
6707 1 : OGRGeometry *poOGRProduct = nullptr;
6708 :
6709 1 : GEOSContextHandle_t hGEOSCtxt = createGEOSContext();
6710 1 : GEOSGeom hThisGeosGeom = exportToGEOS(hGEOSCtxt);
6711 1 : if (hThisGeosGeom != nullptr)
6712 : {
6713 1 : GEOSGeom hGeosProduct = GEOSDelaunayTriangulation_r(
6714 : hGEOSCtxt, hThisGeosGeom, dfTolerance, bOnlyEdges);
6715 1 : GEOSGeom_destroy_r(hGEOSCtxt, hThisGeosGeom);
6716 : poOGRProduct =
6717 1 : BuildGeometryFromGEOS(hGEOSCtxt, hGeosProduct, this, nullptr);
6718 : }
6719 1 : freeGEOSContext(hGEOSCtxt);
6720 1 : return poOGRProduct;
6721 : }
6722 : #endif
6723 :
6724 : /************************************************************************/
6725 : /* OGR_G_DelaunayTriangulation() */
6726 : /************************************************************************/
6727 :
6728 : /**
6729 : * \brief Return a Delaunay triangulation of the vertices of the geometry.
6730 : *
6731 : * This function is the same as the C++ method
6732 : * OGRGeometry::DelaunayTriangulation().
6733 : *
6734 : * This function is built on the GEOS library, v3.4 or above.
6735 : * If OGR is built without the GEOS library, this function will always fail,
6736 : * issuing a CPLE_NotSupported error.
6737 : *
6738 : * @param hThis the geometry.
6739 : * @param dfTolerance optional snapping tolerance to use for improved robustness
6740 : * @param bOnlyEdges if TRUE, will return a MULTILINESTRING, otherwise it will
6741 : * return a GEOMETRYCOLLECTION containing triangular POLYGONs.
6742 : *
6743 : * @return a new geometry to be freed by the caller with OGR_G_DestroyGeometry,
6744 : * or NULL if an error occurs.
6745 : *
6746 : * @since OGR 2.1
6747 : */
6748 :
6749 1 : OGRGeometryH OGR_G_DelaunayTriangulation(OGRGeometryH hThis, double dfTolerance,
6750 : int bOnlyEdges)
6751 :
6752 : {
6753 1 : VALIDATE_POINTER1(hThis, "OGR_G_DelaunayTriangulation", nullptr);
6754 :
6755 1 : return OGRGeometry::ToHandle(
6756 1 : OGRGeometry::FromHandle(hThis)->DelaunayTriangulation(dfTolerance,
6757 2 : bOnlyEdges));
6758 : }
6759 :
6760 : /************************************************************************/
6761 : /* Polygonize() */
6762 : /************************************************************************/
6763 : /* Contributor: Alessandro Furieri, a.furieri@lqt.it */
6764 : /* Developed for Faunalia (http://www.faunalia.it) with funding from */
6765 : /* Regione Toscana - Settore SISTEMA INFORMATIVO TERRITORIALE ED */
6766 : /* AMBIENTALE */
6767 : /************************************************************************/
6768 :
6769 : /**
6770 : * \brief Polygonizes a set of sparse edges.
6771 : *
6772 : * A new geometry object is created and returned containing a collection
6773 : * of reassembled Polygons: NULL will be returned if the input collection
6774 : * doesn't corresponds to a MultiLinestring, or when reassembling Edges
6775 : * into Polygons is impossible due to topological inconsistencies.
6776 : *
6777 : * This method is the same as the C function OGR_G_Polygonize().
6778 : *
6779 : * This method is built on the GEOS library, check it for the definition
6780 : * of the geometry operation.
6781 : * If OGR is built without the GEOS library, this method will always fail,
6782 : * issuing a CPLE_NotSupported error.
6783 : *
6784 : * @return a new geometry to be freed by the caller, or NULL if an error occurs.
6785 : *
6786 : * @since OGR 1.9.0
6787 : */
6788 :
6789 69 : OGRGeometry *OGRGeometry::Polygonize() const
6790 :
6791 : {
6792 : #ifndef HAVE_GEOS
6793 :
6794 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
6795 : return nullptr;
6796 :
6797 : #else
6798 :
6799 69 : const OGRGeometryCollection *poColl = nullptr;
6800 137 : if (wkbFlatten(getGeometryType()) == wkbGeometryCollection ||
6801 68 : wkbFlatten(getGeometryType()) == wkbMultiLineString)
6802 68 : poColl = toGeometryCollection();
6803 : else
6804 1 : return nullptr;
6805 :
6806 68 : const int nCount = poColl->getNumGeometries();
6807 :
6808 68 : OGRGeometry *poPolygsOGRGeom = nullptr;
6809 68 : bool bError = false;
6810 :
6811 68 : GEOSContextHandle_t hGEOSCtxt = createGEOSContext();
6812 :
6813 68 : GEOSGeom *pahGeosGeomList = new GEOSGeom[nCount];
6814 425 : for (int ig = 0; ig < nCount; ig++)
6815 : {
6816 357 : GEOSGeom hGeosGeom = nullptr;
6817 357 : const OGRGeometry *poChild = poColl->getGeometryRef(ig);
6818 714 : if (poChild == nullptr ||
6819 357 : wkbFlatten(poChild->getGeometryType()) != wkbLineString)
6820 1 : bError = true;
6821 : else
6822 : {
6823 356 : hGeosGeom = poChild->exportToGEOS(hGEOSCtxt);
6824 356 : if (hGeosGeom == nullptr)
6825 0 : bError = true;
6826 : }
6827 357 : pahGeosGeomList[ig] = hGeosGeom;
6828 : }
6829 :
6830 68 : if (!bError)
6831 : {
6832 : GEOSGeom hGeosPolygs =
6833 67 : GEOSPolygonize_r(hGEOSCtxt, pahGeosGeomList, nCount);
6834 :
6835 : poPolygsOGRGeom =
6836 67 : BuildGeometryFromGEOS(hGEOSCtxt, hGeosPolygs, this, nullptr);
6837 : }
6838 :
6839 425 : for (int ig = 0; ig < nCount; ig++)
6840 : {
6841 357 : GEOSGeom hGeosGeom = pahGeosGeomList[ig];
6842 357 : if (hGeosGeom != nullptr)
6843 356 : GEOSGeom_destroy_r(hGEOSCtxt, hGeosGeom);
6844 : }
6845 68 : delete[] pahGeosGeomList;
6846 68 : freeGEOSContext(hGEOSCtxt);
6847 :
6848 68 : return poPolygsOGRGeom;
6849 :
6850 : #endif // HAVE_GEOS
6851 : }
6852 :
6853 : /************************************************************************/
6854 : /* OGR_G_Polygonize() */
6855 : /************************************************************************/
6856 : /**
6857 : * \brief Polygonizes a set of sparse edges.
6858 : *
6859 : * A new geometry object is created and returned containing a collection
6860 : * of reassembled Polygons: NULL will be returned if the input collection
6861 : * doesn't corresponds to a MultiLinestring, or when reassembling Edges
6862 : * into Polygons is impossible due to topological inconsistencies.
6863 : *
6864 : * This function is the same as the C++ method OGRGeometry::Polygonize().
6865 : *
6866 : * This function is built on the GEOS library, check it for the definition
6867 : * of the geometry operation.
6868 : * If OGR is built without the GEOS library, this function will always fail,
6869 : * issuing a CPLE_NotSupported error.
6870 : *
6871 : * @param hTarget The Geometry to be polygonized.
6872 : *
6873 : * @return a new geometry to be freed by the caller with OGR_G_DestroyGeometry,
6874 : * or NULL if an error occurs.
6875 : *
6876 : * @since OGR 1.9.0
6877 : */
6878 :
6879 3 : OGRGeometryH OGR_G_Polygonize(OGRGeometryH hTarget)
6880 :
6881 : {
6882 3 : VALIDATE_POINTER1(hTarget, "OGR_G_Polygonize", nullptr);
6883 :
6884 3 : return OGRGeometry::ToHandle(
6885 6 : OGRGeometry::FromHandle(hTarget)->Polygonize());
6886 : }
6887 :
6888 : /************************************************************************/
6889 : /* swapXY() */
6890 : /************************************************************************/
6891 :
6892 : /**
6893 : * \brief Swap x and y coordinates.
6894 : *
6895 : * @since OGR 1.8.0
6896 : */
6897 :
6898 0 : void OGRGeometry::swapXY()
6899 :
6900 : {
6901 0 : }
6902 :
6903 : /************************************************************************/
6904 : /* swapXY() */
6905 : /************************************************************************/
6906 :
6907 : /**
6908 : * \brief Swap x and y coordinates.
6909 : *
6910 : * @param hGeom geometry.
6911 : * @since OGR 2.3.0
6912 : */
6913 :
6914 2 : void OGR_G_SwapXY(OGRGeometryH hGeom)
6915 : {
6916 2 : VALIDATE_POINTER0(hGeom, "OGR_G_SwapXY");
6917 :
6918 2 : OGRGeometry::FromHandle(hGeom)->swapXY();
6919 : }
6920 :
6921 : /************************************************************************/
6922 : /* Prepared geometry API */
6923 : /************************************************************************/
6924 :
6925 : #if defined(HAVE_GEOS)
6926 : struct _OGRPreparedGeometry
6927 : {
6928 : GEOSContextHandle_t hGEOSCtxt;
6929 : GEOSGeom hGEOSGeom;
6930 : const GEOSPreparedGeometry *poPreparedGEOSGeom;
6931 : };
6932 : #endif
6933 :
6934 : /************************************************************************/
6935 : /* OGRHasPreparedGeometrySupport() */
6936 : /************************************************************************/
6937 :
6938 : /** Returns if GEOS has prepared geometry support.
6939 : * @return TRUE or FALSE
6940 : */
6941 1 : int OGRHasPreparedGeometrySupport()
6942 : {
6943 : #if defined(HAVE_GEOS)
6944 1 : return TRUE;
6945 : #else
6946 : return FALSE;
6947 : #endif
6948 : }
6949 :
6950 : /************************************************************************/
6951 : /* OGRCreatePreparedGeometry() */
6952 : /************************************************************************/
6953 :
6954 : /** Creates a prepared geometry.
6955 : *
6956 : * To free with OGRDestroyPreparedGeometry()
6957 : *
6958 : * @param hGeom input geometry to prepare.
6959 : * @return handle to a prepared geometry.
6960 : * @since GDAL 3.3
6961 : */
6962 : OGRPreparedGeometryH
6963 51954 : OGRCreatePreparedGeometry(UNUSED_IF_NO_GEOS OGRGeometryH hGeom)
6964 : {
6965 : #if defined(HAVE_GEOS)
6966 51954 : OGRGeometry *poGeom = OGRGeometry::FromHandle(hGeom);
6967 51954 : GEOSContextHandle_t hGEOSCtxt = OGRGeometry::createGEOSContext();
6968 51954 : GEOSGeom hGEOSGeom = poGeom->exportToGEOS(hGEOSCtxt);
6969 51954 : if (hGEOSGeom == nullptr)
6970 : {
6971 0 : OGRGeometry::freeGEOSContext(hGEOSCtxt);
6972 0 : return nullptr;
6973 : }
6974 : const GEOSPreparedGeometry *poPreparedGEOSGeom =
6975 51954 : GEOSPrepare_r(hGEOSCtxt, hGEOSGeom);
6976 51954 : if (poPreparedGEOSGeom == nullptr)
6977 : {
6978 0 : GEOSGeom_destroy_r(hGEOSCtxt, hGEOSGeom);
6979 0 : OGRGeometry::freeGEOSContext(hGEOSCtxt);
6980 0 : return nullptr;
6981 : }
6982 :
6983 51954 : OGRPreparedGeometry *poPreparedGeom = new OGRPreparedGeometry;
6984 51954 : poPreparedGeom->hGEOSCtxt = hGEOSCtxt;
6985 51954 : poPreparedGeom->hGEOSGeom = hGEOSGeom;
6986 51954 : poPreparedGeom->poPreparedGEOSGeom = poPreparedGEOSGeom;
6987 :
6988 51954 : return poPreparedGeom;
6989 : #else
6990 : return nullptr;
6991 : #endif
6992 : }
6993 :
6994 : /************************************************************************/
6995 : /* OGRDestroyPreparedGeometry() */
6996 : /************************************************************************/
6997 :
6998 : /** Destroys a prepared geometry.
6999 : * @param hPreparedGeom prepared geometry.
7000 : * @since GDAL 3.3
7001 : */
7002 51999 : void OGRDestroyPreparedGeometry(
7003 : UNUSED_IF_NO_GEOS OGRPreparedGeometryH hPreparedGeom)
7004 : {
7005 : #if defined(HAVE_GEOS)
7006 51999 : if (hPreparedGeom != nullptr)
7007 : {
7008 51954 : GEOSPreparedGeom_destroy_r(hPreparedGeom->hGEOSCtxt,
7009 : hPreparedGeom->poPreparedGEOSGeom);
7010 51954 : GEOSGeom_destroy_r(hPreparedGeom->hGEOSCtxt, hPreparedGeom->hGEOSGeom);
7011 51954 : OGRGeometry::freeGEOSContext(hPreparedGeom->hGEOSCtxt);
7012 51954 : delete hPreparedGeom;
7013 : }
7014 : #endif
7015 51999 : }
7016 :
7017 : /************************************************************************/
7018 : /* OGRPreparedGeometryIntersects() */
7019 : /************************************************************************/
7020 :
7021 : /** Returns whether a prepared geometry intersects with a geometry.
7022 : * @param hPreparedGeom prepared geometry.
7023 : * @param hOtherGeom other geometry.
7024 : * @return TRUE or FALSE.
7025 : * @since GDAL 3.3
7026 : */
7027 2862 : int OGRPreparedGeometryIntersects(
7028 : UNUSED_IF_NO_GEOS const OGRPreparedGeometryH hPreparedGeom,
7029 : UNUSED_IF_NO_GEOS const OGRGeometryH hOtherGeom)
7030 : {
7031 : #if defined(HAVE_GEOS)
7032 2862 : OGRGeometry *poOtherGeom = OGRGeometry::FromHandle(hOtherGeom);
7033 2862 : if (hPreparedGeom == nullptr ||
7034 : poOtherGeom == nullptr
7035 : // The check for IsEmpty() is for buggy GEOS versions.
7036 : // See https://github.com/libgeos/geos/pull/423
7037 5724 : || poOtherGeom->IsEmpty())
7038 : {
7039 1 : return FALSE;
7040 : }
7041 :
7042 : GEOSGeom hGEOSOtherGeom =
7043 2861 : poOtherGeom->exportToGEOS(hPreparedGeom->hGEOSCtxt);
7044 2861 : if (hGEOSOtherGeom == nullptr)
7045 0 : return FALSE;
7046 :
7047 2861 : const bool bRet = CPL_TO_BOOL(GEOSPreparedIntersects_r(
7048 : hPreparedGeom->hGEOSCtxt, hPreparedGeom->poPreparedGEOSGeom,
7049 : hGEOSOtherGeom));
7050 2861 : GEOSGeom_destroy_r(hPreparedGeom->hGEOSCtxt, hGEOSOtherGeom);
7051 :
7052 2861 : return bRet;
7053 : #else
7054 : return FALSE;
7055 : #endif
7056 : }
7057 :
7058 : /** Returns whether a prepared geometry contains a geometry.
7059 : * @param hPreparedGeom prepared geometry.
7060 : * @param hOtherGeom other geometry.
7061 : * @return TRUE or FALSE.
7062 : */
7063 120116 : int OGRPreparedGeometryContains(UNUSED_IF_NO_GEOS const OGRPreparedGeometryH
7064 : hPreparedGeom,
7065 : UNUSED_IF_NO_GEOS const OGRGeometryH hOtherGeom)
7066 : {
7067 : #if defined(HAVE_GEOS)
7068 120116 : OGRGeometry *poOtherGeom = OGRGeometry::FromHandle(hOtherGeom);
7069 120116 : if (hPreparedGeom == nullptr ||
7070 : poOtherGeom == nullptr
7071 : // The check for IsEmpty() is for buggy GEOS versions.
7072 : // See https://github.com/libgeos/geos/pull/423
7073 240232 : || poOtherGeom->IsEmpty())
7074 : {
7075 1 : return FALSE;
7076 : }
7077 :
7078 : GEOSGeom hGEOSOtherGeom =
7079 120115 : poOtherGeom->exportToGEOS(hPreparedGeom->hGEOSCtxt);
7080 120115 : if (hGEOSOtherGeom == nullptr)
7081 0 : return FALSE;
7082 :
7083 120115 : const bool bRet = CPL_TO_BOOL(GEOSPreparedContains_r(
7084 : hPreparedGeom->hGEOSCtxt, hPreparedGeom->poPreparedGEOSGeom,
7085 : hGEOSOtherGeom));
7086 120115 : GEOSGeom_destroy_r(hPreparedGeom->hGEOSCtxt, hGEOSOtherGeom);
7087 :
7088 120115 : return bRet;
7089 : #else
7090 : return FALSE;
7091 : #endif
7092 : }
7093 :
7094 : /************************************************************************/
7095 : /* OGRGeometryFromEWKB() */
7096 : /************************************************************************/
7097 :
7098 1429 : OGRGeometry *OGRGeometryFromEWKB(GByte *pabyEWKB, int nLength, int *pnSRID,
7099 : int bIsPostGIS1_EWKB)
7100 :
7101 : {
7102 1429 : OGRGeometry *poGeometry = nullptr;
7103 :
7104 1429 : size_t nWKBSize = 0;
7105 1429 : const GByte *pabyWKB = WKBFromEWKB(pabyEWKB, nLength, nWKBSize, pnSRID);
7106 1429 : if (pabyWKB == nullptr)
7107 0 : return nullptr;
7108 :
7109 : /* -------------------------------------------------------------------- */
7110 : /* Try to ingest the geometry. */
7111 : /* -------------------------------------------------------------------- */
7112 1429 : (void)OGRGeometryFactory::createFromWkb(
7113 : pabyWKB, nullptr, &poGeometry, nWKBSize,
7114 : (bIsPostGIS1_EWKB) ? wkbVariantPostGIS1 : wkbVariantOldOgc);
7115 :
7116 1429 : return poGeometry;
7117 : }
7118 :
7119 : /************************************************************************/
7120 : /* OGRGeometryFromHexEWKB() */
7121 : /************************************************************************/
7122 :
7123 1427 : OGRGeometry *OGRGeometryFromHexEWKB(const char *pszBytea, int *pnSRID,
7124 : int bIsPostGIS1_EWKB)
7125 :
7126 : {
7127 1427 : if (pszBytea == nullptr)
7128 0 : return nullptr;
7129 :
7130 1427 : int nWKBLength = 0;
7131 1427 : GByte *pabyWKB = CPLHexToBinary(pszBytea, &nWKBLength);
7132 :
7133 : OGRGeometry *poGeometry =
7134 1427 : OGRGeometryFromEWKB(pabyWKB, nWKBLength, pnSRID, bIsPostGIS1_EWKB);
7135 :
7136 1427 : CPLFree(pabyWKB);
7137 :
7138 1427 : return poGeometry;
7139 : }
7140 :
7141 : /************************************************************************/
7142 : /* OGRGeometryToHexEWKB() */
7143 : /************************************************************************/
7144 :
7145 1056 : char *OGRGeometryToHexEWKB(OGRGeometry *poGeometry, int nSRSId,
7146 : int nPostGISMajor, int nPostGISMinor)
7147 : {
7148 1056 : const size_t nWkbSize = poGeometry->WkbSize();
7149 1056 : GByte *pabyWKB = static_cast<GByte *>(VSI_MALLOC_VERBOSE(nWkbSize));
7150 1056 : if (pabyWKB == nullptr)
7151 0 : return CPLStrdup("");
7152 :
7153 115 : if ((nPostGISMajor > 2 || (nPostGISMajor == 2 && nPostGISMinor >= 2)) &&
7154 1786 : wkbFlatten(poGeometry->getGeometryType()) == wkbPoint &&
7155 615 : poGeometry->IsEmpty())
7156 : {
7157 2 : if (poGeometry->exportToWkb(wkbNDR, pabyWKB, wkbVariantIso) !=
7158 : OGRERR_NONE)
7159 : {
7160 0 : CPLFree(pabyWKB);
7161 0 : return CPLStrdup("");
7162 : }
7163 : }
7164 1054 : else if (poGeometry->exportToWkb(wkbNDR, pabyWKB,
7165 : (nPostGISMajor < 2)
7166 : ? wkbVariantPostGIS1
7167 1054 : : wkbVariantOldOgc) != OGRERR_NONE)
7168 : {
7169 0 : CPLFree(pabyWKB);
7170 0 : return CPLStrdup("");
7171 : }
7172 :
7173 : // When converting to hex, each byte takes 2 hex characters. In addition
7174 : // we add in 8 characters to represent the SRID integer in hex, and
7175 : // one for a null terminator.
7176 : // The limit of INT_MAX = 2 GB is a bit artificial, but at time of writing
7177 : // (2024), PostgreSQL by default cannot handle objects larger than 1 GB:
7178 : // https://github.com/postgres/postgres/blob/5d39becf8ba0080c98fee4b63575552f6800b012/src/include/utils/memutils.h#L40
7179 1056 : if (nWkbSize >
7180 1056 : static_cast<size_t>(std::numeric_limits<int>::max() - 8 - 1) / 2)
7181 : {
7182 0 : CPLFree(pabyWKB);
7183 0 : return CPLStrdup("");
7184 : }
7185 1056 : const size_t nTextSize = nWkbSize * 2 + 8 + 1;
7186 1056 : char *pszTextBuf = static_cast<char *>(VSI_MALLOC_VERBOSE(nTextSize));
7187 1056 : if (pszTextBuf == nullptr)
7188 : {
7189 0 : CPLFree(pabyWKB);
7190 0 : return CPLStrdup("");
7191 : }
7192 1056 : char *pszTextBufCurrent = pszTextBuf;
7193 :
7194 : // Convert the 1st byte, which is the endianness flag, to hex.
7195 1056 : char *pszHex = CPLBinaryToHex(1, pabyWKB);
7196 1056 : strcpy(pszTextBufCurrent, pszHex);
7197 1056 : CPLFree(pszHex);
7198 1056 : pszTextBufCurrent += 2;
7199 :
7200 : // Next, get the geom type which is bytes 2 through 5.
7201 : GUInt32 geomType;
7202 1056 : memcpy(&geomType, pabyWKB + 1, 4);
7203 :
7204 : // Now add the SRID flag if an SRID is provided.
7205 1056 : if (nSRSId > 0)
7206 : {
7207 : // Change the flag to wkbNDR (little) endianness.
7208 529 : constexpr GUInt32 WKBSRIDFLAG = 0x20000000;
7209 529 : GUInt32 nGSrsFlag = CPL_LSBWORD32(WKBSRIDFLAG);
7210 : // Apply the flag.
7211 529 : geomType = geomType | nGSrsFlag;
7212 : }
7213 :
7214 : // Now write the geom type which is 4 bytes.
7215 1056 : pszHex = CPLBinaryToHex(4, reinterpret_cast<const GByte *>(&geomType));
7216 1056 : strcpy(pszTextBufCurrent, pszHex);
7217 1056 : CPLFree(pszHex);
7218 1056 : pszTextBufCurrent += 8;
7219 :
7220 : // Now include SRID if provided.
7221 1056 : if (nSRSId > 0)
7222 : {
7223 : // Force the srsid to wkbNDR (little) endianness.
7224 529 : const GUInt32 nGSRSId = CPL_LSBWORD32(nSRSId);
7225 529 : pszHex = CPLBinaryToHex(sizeof(nGSRSId),
7226 : reinterpret_cast<const GByte *>(&nGSRSId));
7227 529 : strcpy(pszTextBufCurrent, pszHex);
7228 529 : CPLFree(pszHex);
7229 529 : pszTextBufCurrent += 8;
7230 : }
7231 :
7232 : // Copy the rest of the data over - subtract
7233 : // 5 since we already copied 5 bytes above.
7234 1056 : pszHex = CPLBinaryToHex(static_cast<int>(nWkbSize - 5), pabyWKB + 5);
7235 1056 : CPLFree(pabyWKB);
7236 1056 : if (!pszHex || pszHex[0] == 0)
7237 : {
7238 0 : CPLFree(pszTextBuf);
7239 0 : return pszHex;
7240 : }
7241 1056 : strcpy(pszTextBufCurrent, pszHex);
7242 1056 : CPLFree(pszHex);
7243 :
7244 1056 : return pszTextBuf;
7245 : }
7246 :
7247 : /************************************************************************/
7248 : /* importPreambleFromWkb() */
7249 : /************************************************************************/
7250 :
7251 : //! @cond Doxygen_Suppress
7252 153380 : OGRErr OGRGeometry::importPreambleFromWkb(const unsigned char *pabyData,
7253 : size_t nSize,
7254 : OGRwkbByteOrder &eByteOrder,
7255 : OGRwkbVariant eWkbVariant)
7256 : {
7257 153380 : if (nSize < 9 && nSize != static_cast<size_t>(-1))
7258 0 : return OGRERR_NOT_ENOUGH_DATA;
7259 :
7260 : /* -------------------------------------------------------------------- */
7261 : /* Get the byte order byte. */
7262 : /* -------------------------------------------------------------------- */
7263 153380 : int nByteOrder = DB2_V72_FIX_BYTE_ORDER(*pabyData);
7264 153380 : if (!(nByteOrder == wkbXDR || nByteOrder == wkbNDR))
7265 0 : return OGRERR_CORRUPT_DATA;
7266 153380 : eByteOrder = static_cast<OGRwkbByteOrder>(nByteOrder);
7267 :
7268 : /* -------------------------------------------------------------------- */
7269 : /* Get the geometry feature type. */
7270 : /* -------------------------------------------------------------------- */
7271 : OGRwkbGeometryType eGeometryType;
7272 : const OGRErr err =
7273 153380 : OGRReadWKBGeometryType(pabyData, eWkbVariant, &eGeometryType);
7274 153381 : if (wkbHasZ(eGeometryType))
7275 62304 : flags |= OGR_G_3D;
7276 153381 : if (wkbHasM(eGeometryType))
7277 59689 : flags |= OGR_G_MEASURED;
7278 :
7279 153381 : if (err != OGRERR_NONE || eGeometryType != getGeometryType())
7280 0 : return OGRERR_CORRUPT_DATA;
7281 :
7282 153381 : return OGRERR_NONE;
7283 : }
7284 :
7285 : /************************************************************************/
7286 : /* importPreambleOfCollectionFromWkb() */
7287 : /* */
7288 : /* Utility method for OGRSimpleCurve, OGRCompoundCurve, */
7289 : /* OGRCurvePolygon and OGRGeometryCollection. */
7290 : /************************************************************************/
7291 :
7292 70783 : OGRErr OGRGeometry::importPreambleOfCollectionFromWkb(
7293 : const unsigned char *pabyData, size_t &nSize, size_t &nDataOffset,
7294 : OGRwkbByteOrder &eByteOrder, size_t nMinSubGeomSize, int &nGeomCount,
7295 : OGRwkbVariant eWkbVariant)
7296 : {
7297 70783 : nGeomCount = 0;
7298 :
7299 : OGRErr eErr =
7300 70783 : importPreambleFromWkb(pabyData, nSize, eByteOrder, eWkbVariant);
7301 70783 : if (eErr != OGRERR_NONE)
7302 0 : return eErr;
7303 :
7304 : /* -------------------------------------------------------------------- */
7305 : /* Clear existing Geoms. */
7306 : /* -------------------------------------------------------------------- */
7307 70783 : int _flags = flags; // flags set in importPreambleFromWkb
7308 70783 : empty(); // may reset flags etc.
7309 :
7310 : // restore
7311 70784 : if (_flags & OGR_G_3D)
7312 59263 : set3D(TRUE);
7313 70784 : if (_flags & OGR_G_MEASURED)
7314 56768 : setMeasured(TRUE);
7315 :
7316 : /* -------------------------------------------------------------------- */
7317 : /* Get the sub-geometry count. */
7318 : /* -------------------------------------------------------------------- */
7319 70784 : memcpy(&nGeomCount, pabyData + 5, 4);
7320 :
7321 70784 : if (OGR_SWAP(eByteOrder))
7322 383 : nGeomCount = CPL_SWAP32(nGeomCount);
7323 :
7324 141434 : if (nGeomCount < 0 ||
7325 70650 : static_cast<size_t>(nGeomCount) >
7326 70650 : std::numeric_limits<size_t>::max() / nMinSubGeomSize)
7327 : {
7328 134 : nGeomCount = 0;
7329 134 : return OGRERR_CORRUPT_DATA;
7330 : }
7331 70650 : const size_t nBufferMinSize = nGeomCount * nMinSubGeomSize;
7332 :
7333 : // Each ring has a minimum of nMinSubGeomSize bytes.
7334 70650 : if (nSize != static_cast<size_t>(-1) && nSize - 9 < nBufferMinSize)
7335 : {
7336 910 : CPLError(CE_Failure, CPLE_AppDefined,
7337 : "Length of input WKB is too small");
7338 910 : nGeomCount = 0;
7339 910 : return OGRERR_NOT_ENOUGH_DATA;
7340 : }
7341 :
7342 69740 : nDataOffset = 9;
7343 69740 : if (nSize != static_cast<size_t>(-1))
7344 : {
7345 69720 : CPLAssert(nSize >= nDataOffset);
7346 69720 : nSize -= nDataOffset;
7347 : }
7348 :
7349 69740 : return OGRERR_NONE;
7350 : }
7351 :
7352 : /************************************************************************/
7353 : /* importCurveCollectionFromWkt() */
7354 : /* */
7355 : /* Utility method for OGRCompoundCurve, OGRCurvePolygon and */
7356 : /* OGRMultiCurve. */
7357 : /************************************************************************/
7358 :
7359 1431 : OGRErr OGRGeometry::importCurveCollectionFromWkt(
7360 : const char **ppszInput, int bAllowEmptyComponent, int bAllowLineString,
7361 : int bAllowCurve, int bAllowCompoundCurve,
7362 : OGRErr (*pfnAddCurveDirectly)(OGRGeometry *poSelf, OGRCurve *poCurve))
7363 :
7364 : {
7365 1431 : int bHasZ = FALSE;
7366 1431 : int bHasM = FALSE;
7367 1431 : bool bIsEmpty = false;
7368 1431 : OGRErr eErr = importPreambleFromWkt(ppszInput, &bHasZ, &bHasM, &bIsEmpty);
7369 1431 : flags = 0;
7370 1431 : if (eErr != OGRERR_NONE)
7371 14 : return eErr;
7372 1417 : if (bHasZ)
7373 189 : flags |= OGR_G_3D;
7374 1417 : if (bHasM)
7375 120 : flags |= OGR_G_MEASURED;
7376 1417 : if (bIsEmpty)
7377 109 : return OGRERR_NONE;
7378 :
7379 : char szToken[OGR_WKT_TOKEN_MAX];
7380 1308 : const char *pszInput = *ppszInput;
7381 1308 : eErr = OGRERR_NONE;
7382 :
7383 : // Skip first '('.
7384 1308 : pszInput = OGRWktReadToken(pszInput, szToken);
7385 :
7386 : /* ==================================================================== */
7387 : /* Read each curve in turn. Note that we try to reuse the same */
7388 : /* point list buffer from curve to curve to cut down on */
7389 : /* allocate/deallocate overhead. */
7390 : /* ==================================================================== */
7391 1308 : OGRRawPoint *paoPoints = nullptr;
7392 1308 : int nMaxPoints = 0;
7393 1308 : double *padfZ = nullptr;
7394 :
7395 624 : do
7396 : {
7397 :
7398 : /* --------------------------------------------------------------------
7399 : */
7400 : /* Get the first token, which should be the geometry type. */
7401 : /* --------------------------------------------------------------------
7402 : */
7403 1932 : const char *pszInputBefore = pszInput;
7404 1932 : pszInput = OGRWktReadToken(pszInput, szToken);
7405 :
7406 : /* --------------------------------------------------------------------
7407 : */
7408 : /* Do the import. */
7409 : /* --------------------------------------------------------------------
7410 : */
7411 1932 : OGRCurve *poCurve = nullptr;
7412 1932 : if (EQUAL(szToken, "("))
7413 : {
7414 1413 : OGRLineString *poLine = new OGRLineString();
7415 1413 : poCurve = poLine;
7416 1413 : pszInput = pszInputBefore;
7417 1413 : eErr = poLine->importFromWKTListOnly(&pszInput, bHasZ, bHasM,
7418 : paoPoints, nMaxPoints, padfZ);
7419 : }
7420 519 : else if (bAllowEmptyComponent && EQUAL(szToken, "EMPTY"))
7421 : {
7422 16 : poCurve = new OGRLineString();
7423 : }
7424 : // Accept LINESTRING(), but this is an extension to the BNF, also
7425 : // accepted by PostGIS.
7426 503 : else if ((bAllowLineString && STARTS_WITH_CI(szToken, "LINESTRING")) ||
7427 488 : (bAllowCurve && !STARTS_WITH_CI(szToken, "LINESTRING") &&
7428 488 : !STARTS_WITH_CI(szToken, "COMPOUNDCURVE") &&
7429 1155 : OGR_GT_IsCurve(OGRFromOGCGeomType(szToken))) ||
7430 149 : (bAllowCompoundCurve &&
7431 149 : STARTS_WITH_CI(szToken, "COMPOUNDCURVE")))
7432 : {
7433 465 : OGRGeometry *poGeom = nullptr;
7434 465 : pszInput = pszInputBefore;
7435 : eErr =
7436 465 : OGRGeometryFactory::createFromWkt(&pszInput, nullptr, &poGeom);
7437 465 : if (poGeom == nullptr)
7438 : {
7439 1 : eErr = OGRERR_CORRUPT_DATA;
7440 : }
7441 : else
7442 : {
7443 464 : poCurve = poGeom->toCurve();
7444 : }
7445 : }
7446 : else
7447 : {
7448 38 : CPLError(CE_Failure, CPLE_AppDefined, "Unexpected token : %s",
7449 : szToken);
7450 38 : eErr = OGRERR_CORRUPT_DATA;
7451 : }
7452 :
7453 : // If this has M it is an error if poGeom does not have M.
7454 1932 : if (poCurve && !Is3D() && IsMeasured() && !poCurve->IsMeasured())
7455 0 : eErr = OGRERR_CORRUPT_DATA;
7456 :
7457 1932 : if (eErr == OGRERR_NONE)
7458 1887 : eErr = pfnAddCurveDirectly(this, poCurve);
7459 1932 : if (eErr != OGRERR_NONE)
7460 : {
7461 55 : delete poCurve;
7462 55 : break;
7463 : }
7464 :
7465 : /* --------------------------------------------------------------------
7466 : */
7467 : /* Read the delimiter following the surface. */
7468 : /* --------------------------------------------------------------------
7469 : */
7470 1877 : pszInput = OGRWktReadToken(pszInput, szToken);
7471 1877 : } while (szToken[0] == ',' && eErr == OGRERR_NONE);
7472 :
7473 1308 : CPLFree(paoPoints);
7474 1308 : CPLFree(padfZ);
7475 :
7476 : /* -------------------------------------------------------------------- */
7477 : /* freak if we don't get a closing bracket. */
7478 : /* -------------------------------------------------------------------- */
7479 :
7480 1308 : if (eErr != OGRERR_NONE)
7481 55 : return eErr;
7482 :
7483 1253 : if (szToken[0] != ')')
7484 9 : return OGRERR_CORRUPT_DATA;
7485 :
7486 1244 : *ppszInput = pszInput;
7487 1244 : return OGRERR_NONE;
7488 : }
7489 :
7490 : //! @endcond
7491 :
7492 : /************************************************************************/
7493 : /* OGR_GT_Flatten() */
7494 : /************************************************************************/
7495 : /**
7496 : * \brief Returns the 2D geometry type corresponding to the passed geometry
7497 : * type.
7498 : *
7499 : * This function is intended to work with geometry types as old-style 99-402
7500 : * extended dimension (Z) WKB types, as well as with newer SFSQL 1.2 and
7501 : * ISO SQL/MM Part 3 extended dimension (Z&M) WKB types.
7502 : *
7503 : * @param eType Input geometry type
7504 : *
7505 : * @return 2D geometry type corresponding to the passed geometry type.
7506 : *
7507 : * @since GDAL 2.0
7508 : */
7509 :
7510 7152440 : OGRwkbGeometryType OGR_GT_Flatten(OGRwkbGeometryType eType)
7511 : {
7512 7152440 : eType = static_cast<OGRwkbGeometryType>(eType & (~wkb25DBitInternalUse));
7513 7152440 : if (eType >= 1000 && eType < 2000) // ISO Z.
7514 2710510 : return static_cast<OGRwkbGeometryType>(eType - 1000);
7515 4441930 : if (eType >= 2000 && eType < 3000) // ISO M.
7516 5560 : return static_cast<OGRwkbGeometryType>(eType - 2000);
7517 4436370 : if (eType >= 3000 && eType < 4000) // ISO ZM.
7518 135257 : return static_cast<OGRwkbGeometryType>(eType - 3000);
7519 4301110 : return eType;
7520 : }
7521 :
7522 : /************************************************************************/
7523 : /* OGR_GT_HasZ() */
7524 : /************************************************************************/
7525 : /**
7526 : * \brief Return if the geometry type is a 3D geometry type.
7527 : *
7528 : * @param eType Input geometry type
7529 : *
7530 : * @return TRUE if the geometry type is a 3D geometry type.
7531 : *
7532 : * @since GDAL 2.0
7533 : */
7534 :
7535 1984750 : int OGR_GT_HasZ(OGRwkbGeometryType eType)
7536 : {
7537 1984750 : if (eType & wkb25DBitInternalUse)
7538 156306 : return TRUE;
7539 1828440 : if (eType >= 1000 && eType < 2000) // Accept 1000 for wkbUnknownZ.
7540 221 : return TRUE;
7541 1828220 : if (eType >= 3000 && eType < 4000) // Accept 3000 for wkbUnknownZM.
7542 121221 : return TRUE;
7543 1707000 : return FALSE;
7544 : }
7545 :
7546 : /************************************************************************/
7547 : /* OGR_GT_HasM() */
7548 : /************************************************************************/
7549 : /**
7550 : * \brief Return if the geometry type is a measured type.
7551 : *
7552 : * @param eType Input geometry type
7553 : *
7554 : * @return TRUE if the geometry type is a measured type.
7555 : *
7556 : * @since GDAL 2.1
7557 : */
7558 :
7559 2040400 : int OGR_GT_HasM(OGRwkbGeometryType eType)
7560 : {
7561 2040400 : if (eType >= 2000 && eType < 3000) // Accept 2000 for wkbUnknownM.
7562 2542 : return TRUE;
7563 2037860 : if (eType >= 3000 && eType < 4000) // Accept 3000 for wkbUnknownZM.
7564 120882 : return TRUE;
7565 1916970 : return FALSE;
7566 : }
7567 :
7568 : /************************************************************************/
7569 : /* OGR_GT_SetZ() */
7570 : /************************************************************************/
7571 : /**
7572 : * \brief Returns the 3D geometry type corresponding to the passed geometry
7573 : * type.
7574 : *
7575 : * @param eType Input geometry type
7576 : *
7577 : * @return 3D geometry type corresponding to the passed geometry type.
7578 : *
7579 : * @since GDAL 2.0
7580 : */
7581 :
7582 5275 : OGRwkbGeometryType OGR_GT_SetZ(OGRwkbGeometryType eType)
7583 : {
7584 5275 : if (OGR_GT_HasZ(eType) || eType == wkbNone)
7585 478 : return eType;
7586 4797 : if (eType <= wkbGeometryCollection)
7587 4742 : return static_cast<OGRwkbGeometryType>(eType | wkb25DBitInternalUse);
7588 : else
7589 55 : return static_cast<OGRwkbGeometryType>(eType + 1000);
7590 : }
7591 :
7592 : /************************************************************************/
7593 : /* OGR_GT_SetM() */
7594 : /************************************************************************/
7595 : /**
7596 : * \brief Returns the measured geometry type corresponding to the passed
7597 : * geometry type.
7598 : *
7599 : * @param eType Input geometry type
7600 : *
7601 : * @return measured geometry type corresponding to the passed geometry type.
7602 : *
7603 : * @since GDAL 2.1
7604 : */
7605 :
7606 1913 : OGRwkbGeometryType OGR_GT_SetM(OGRwkbGeometryType eType)
7607 : {
7608 1913 : if (OGR_GT_HasM(eType) || eType == wkbNone)
7609 257 : return eType;
7610 1656 : if (eType & wkb25DBitInternalUse)
7611 : {
7612 699 : eType = static_cast<OGRwkbGeometryType>(eType & ~wkb25DBitInternalUse);
7613 699 : eType = static_cast<OGRwkbGeometryType>(eType + 1000);
7614 : }
7615 1656 : return static_cast<OGRwkbGeometryType>(eType + 2000);
7616 : }
7617 :
7618 : /************************************************************************/
7619 : /* OGR_GT_SetModifier() */
7620 : /************************************************************************/
7621 : /**
7622 : * \brief Returns a XY, XYZ, XYM or XYZM geometry type depending on parameter.
7623 : *
7624 : * @param eType Input geometry type
7625 : * @param bHasZ TRUE if the output geometry type must be 3D.
7626 : * @param bHasM TRUE if the output geometry type must be measured.
7627 : *
7628 : * @return Output geometry type.
7629 : *
7630 : * @since GDAL 2.0
7631 : */
7632 :
7633 4817 : OGRwkbGeometryType OGR_GT_SetModifier(OGRwkbGeometryType eType, int bHasZ,
7634 : int bHasM)
7635 : {
7636 4817 : if (bHasZ && bHasM)
7637 342 : return OGR_GT_SetM(OGR_GT_SetZ(eType));
7638 4475 : else if (bHasM)
7639 331 : return OGR_GT_SetM(wkbFlatten(eType));
7640 4144 : else if (bHasZ)
7641 1747 : return OGR_GT_SetZ(wkbFlatten(eType));
7642 : else
7643 2397 : return wkbFlatten(eType);
7644 : }
7645 :
7646 : /************************************************************************/
7647 : /* OGR_GT_IsSubClassOf) */
7648 : /************************************************************************/
7649 : /**
7650 : * \brief Returns if a type is a subclass of another one
7651 : *
7652 : * @param eType Type.
7653 : * @param eSuperType Super type
7654 : *
7655 : * @return TRUE if eType is a subclass of eSuperType.
7656 : *
7657 : * @since GDAL 2.0
7658 : */
7659 :
7660 105456 : int OGR_GT_IsSubClassOf(OGRwkbGeometryType eType, OGRwkbGeometryType eSuperType)
7661 : {
7662 105456 : eSuperType = wkbFlatten(eSuperType);
7663 105456 : eType = wkbFlatten(eType);
7664 :
7665 105456 : if (eSuperType == eType || eSuperType == wkbUnknown)
7666 7631 : return TRUE;
7667 :
7668 97825 : if (eSuperType == wkbGeometryCollection)
7669 25732 : return eType == wkbMultiPoint || eType == wkbMultiLineString ||
7670 52887 : eType == wkbMultiPolygon || eType == wkbMultiCurve ||
7671 27155 : eType == wkbMultiSurface;
7672 :
7673 70670 : if (eSuperType == wkbCurvePolygon)
7674 19316 : return eType == wkbPolygon || eType == wkbTriangle;
7675 :
7676 51354 : if (eSuperType == wkbMultiCurve)
7677 210 : return eType == wkbMultiLineString;
7678 :
7679 51144 : if (eSuperType == wkbMultiSurface)
7680 264 : return eType == wkbMultiPolygon;
7681 :
7682 50880 : if (eSuperType == wkbCurve)
7683 21048 : return eType == wkbLineString || eType == wkbCircularString ||
7684 21048 : eType == wkbCompoundCurve;
7685 :
7686 29832 : if (eSuperType == wkbSurface)
7687 3363 : return eType == wkbCurvePolygon || eType == wkbPolygon ||
7688 6864 : eType == wkbTriangle || eType == wkbPolyhedralSurface ||
7689 3501 : eType == wkbTIN;
7690 :
7691 26331 : if (eSuperType == wkbPolygon)
7692 216 : return eType == wkbTriangle;
7693 :
7694 26115 : if (eSuperType == wkbPolyhedralSurface)
7695 13334 : return eType == wkbTIN;
7696 :
7697 12781 : return FALSE;
7698 : }
7699 :
7700 : /************************************************************************/
7701 : /* OGR_GT_GetCollection() */
7702 : /************************************************************************/
7703 : /**
7704 : * \brief Returns the collection type that can contain the passed geometry type
7705 : *
7706 : * Handled conversions are : wkbNone->wkbNone, wkbPoint -> wkbMultiPoint,
7707 : * wkbLineString->wkbMultiLineString,
7708 : * wkbPolygon/wkbTriangle/wkbPolyhedralSurface/wkbTIN->wkbMultiPolygon,
7709 : * wkbCircularString->wkbMultiCurve, wkbCompoundCurve->wkbMultiCurve,
7710 : * wkbCurvePolygon->wkbMultiSurface.
7711 : * In other cases, wkbUnknown is returned
7712 : *
7713 : * Passed Z, M, ZM flag is preserved.
7714 : *
7715 : *
7716 : * @param eType Input geometry type
7717 : *
7718 : * @return the collection type that can contain the passed geometry type or
7719 : * wkbUnknown
7720 : *
7721 : * @since GDAL 2.0
7722 : */
7723 :
7724 2231 : OGRwkbGeometryType OGR_GT_GetCollection(OGRwkbGeometryType eType)
7725 : {
7726 2231 : const bool bHasZ = wkbHasZ(eType);
7727 2231 : const bool bHasM = wkbHasM(eType);
7728 2231 : if (eType == wkbNone)
7729 0 : return wkbNone;
7730 2231 : OGRwkbGeometryType eFGType = wkbFlatten(eType);
7731 2231 : if (eFGType == wkbPoint)
7732 41 : eType = wkbMultiPoint;
7733 :
7734 2190 : else if (eFGType == wkbLineString)
7735 152 : eType = wkbMultiLineString;
7736 :
7737 2038 : else if (eFGType == wkbPolygon)
7738 634 : eType = wkbMultiPolygon;
7739 :
7740 1404 : else if (eFGType == wkbTriangle)
7741 7 : eType = wkbTIN;
7742 :
7743 1397 : else if (OGR_GT_IsCurve(eFGType))
7744 165 : eType = wkbMultiCurve;
7745 :
7746 1232 : else if (OGR_GT_IsSurface(eFGType))
7747 951 : eType = wkbMultiSurface;
7748 :
7749 : else
7750 281 : return wkbUnknown;
7751 :
7752 1950 : if (bHasZ)
7753 2 : eType = wkbSetZ(eType);
7754 1950 : if (bHasM)
7755 2 : eType = wkbSetM(eType);
7756 :
7757 1950 : return eType;
7758 : }
7759 :
7760 : /************************************************************************/
7761 : /* OGR_GT_GetCurve() */
7762 : /************************************************************************/
7763 : /**
7764 : * \brief Returns the curve geometry type that can contain the passed geometry
7765 : * type
7766 : *
7767 : * Handled conversions are : wkbPolygon -> wkbCurvePolygon,
7768 : * wkbLineString->wkbCompoundCurve, wkbMultiPolygon->wkbMultiSurface
7769 : * and wkbMultiLineString->wkbMultiCurve.
7770 : * In other cases, the passed geometry is returned.
7771 : *
7772 : * Passed Z, M, ZM flag is preserved.
7773 : *
7774 : * @param eType Input geometry type
7775 : *
7776 : * @return the curve type that can contain the passed geometry type
7777 : *
7778 : * @since GDAL 2.0
7779 : */
7780 :
7781 33 : OGRwkbGeometryType OGR_GT_GetCurve(OGRwkbGeometryType eType)
7782 : {
7783 33 : const bool bHasZ = wkbHasZ(eType);
7784 33 : const bool bHasM = wkbHasM(eType);
7785 33 : OGRwkbGeometryType eFGType = wkbFlatten(eType);
7786 :
7787 33 : if (eFGType == wkbLineString)
7788 2 : eType = wkbCompoundCurve;
7789 :
7790 31 : else if (eFGType == wkbPolygon)
7791 1 : eType = wkbCurvePolygon;
7792 :
7793 30 : else if (eFGType == wkbTriangle)
7794 0 : eType = wkbCurvePolygon;
7795 :
7796 30 : else if (eFGType == wkbMultiLineString)
7797 6 : eType = wkbMultiCurve;
7798 :
7799 24 : else if (eFGType == wkbMultiPolygon)
7800 4 : eType = wkbMultiSurface;
7801 :
7802 33 : if (bHasZ)
7803 4 : eType = wkbSetZ(eType);
7804 33 : if (bHasM)
7805 4 : eType = wkbSetM(eType);
7806 :
7807 33 : return eType;
7808 : }
7809 :
7810 : /************************************************************************/
7811 : /* OGR_GT_GetLinear() */
7812 : /************************************************************************/
7813 : /**
7814 : * \brief Returns the non-curve geometry type that can contain the passed
7815 : * geometry type
7816 : *
7817 : * Handled conversions are : wkbCurvePolygon -> wkbPolygon,
7818 : * wkbCircularString->wkbLineString, wkbCompoundCurve->wkbLineString,
7819 : * wkbMultiSurface->wkbMultiPolygon and wkbMultiCurve->wkbMultiLineString.
7820 : * In other cases, the passed geometry is returned.
7821 : *
7822 : * Passed Z, M, ZM flag is preserved.
7823 : *
7824 : * @param eType Input geometry type
7825 : *
7826 : * @return the non-curve type that can contain the passed geometry type
7827 : *
7828 : * @since GDAL 2.0
7829 : */
7830 :
7831 693 : OGRwkbGeometryType OGR_GT_GetLinear(OGRwkbGeometryType eType)
7832 : {
7833 693 : const bool bHasZ = wkbHasZ(eType);
7834 693 : const bool bHasM = wkbHasM(eType);
7835 693 : OGRwkbGeometryType eFGType = wkbFlatten(eType);
7836 :
7837 693 : if (OGR_GT_IsCurve(eFGType))
7838 36 : eType = wkbLineString;
7839 :
7840 657 : else if (OGR_GT_IsSurface(eFGType))
7841 36 : eType = wkbPolygon;
7842 :
7843 621 : else if (eFGType == wkbMultiCurve)
7844 177 : eType = wkbMultiLineString;
7845 :
7846 444 : else if (eFGType == wkbMultiSurface)
7847 156 : eType = wkbMultiPolygon;
7848 :
7849 693 : if (bHasZ)
7850 122 : eType = wkbSetZ(eType);
7851 693 : if (bHasM)
7852 85 : eType = wkbSetM(eType);
7853 :
7854 693 : return eType;
7855 : }
7856 :
7857 : /************************************************************************/
7858 : /* OGR_GT_IsCurve() */
7859 : /************************************************************************/
7860 :
7861 : /**
7862 : * \brief Return if a geometry type is an instance of Curve
7863 : *
7864 : * Such geometry type are wkbLineString, wkbCircularString, wkbCompoundCurve
7865 : * and their Z/M/ZM variant.
7866 : *
7867 : * @param eGeomType the geometry type
7868 : * @return TRUE if the geometry type is an instance of Curve
7869 : *
7870 : * @since GDAL 2.0
7871 : */
7872 :
7873 21039 : int OGR_GT_IsCurve(OGRwkbGeometryType eGeomType)
7874 : {
7875 21039 : return OGR_GT_IsSubClassOf(eGeomType, wkbCurve);
7876 : }
7877 :
7878 : /************************************************************************/
7879 : /* OGR_GT_IsSurface() */
7880 : /************************************************************************/
7881 :
7882 : /**
7883 : * \brief Return if a geometry type is an instance of Surface
7884 : *
7885 : * Such geometry type are wkbCurvePolygon and wkbPolygon
7886 : * and their Z/M/ZM variant.
7887 : *
7888 : * @param eGeomType the geometry type
7889 : * @return TRUE if the geometry type is an instance of Surface
7890 : *
7891 : * @since GDAL 2.0
7892 : */
7893 :
7894 3495 : int OGR_GT_IsSurface(OGRwkbGeometryType eGeomType)
7895 : {
7896 3495 : return OGR_GT_IsSubClassOf(eGeomType, wkbSurface);
7897 : }
7898 :
7899 : /************************************************************************/
7900 : /* OGR_GT_IsNonLinear() */
7901 : /************************************************************************/
7902 :
7903 : /**
7904 : * \brief Return if a geometry type is a non-linear geometry type.
7905 : *
7906 : * Such geometry type are wkbCurve, wkbCircularString, wkbCompoundCurve,
7907 : * wkbSurface, wkbCurvePolygon, wkbMultiCurve, wkbMultiSurface and their
7908 : * Z/M variants.
7909 : *
7910 : * @param eGeomType the geometry type
7911 : * @return TRUE if the geometry type is a non-linear geometry type.
7912 : *
7913 : * @since GDAL 2.0
7914 : */
7915 :
7916 111283 : int OGR_GT_IsNonLinear(OGRwkbGeometryType eGeomType)
7917 : {
7918 111283 : OGRwkbGeometryType eFGeomType = wkbFlatten(eGeomType);
7919 111275 : return eFGeomType == wkbCurve || eFGeomType == wkbSurface ||
7920 111221 : eFGeomType == wkbCircularString || eFGeomType == wkbCompoundCurve ||
7921 222558 : eFGeomType == wkbCurvePolygon || eFGeomType == wkbMultiCurve ||
7922 111283 : eFGeomType == wkbMultiSurface;
7923 : }
7924 :
7925 : /************************************************************************/
7926 : /* CastToError() */
7927 : /************************************************************************/
7928 :
7929 : //! @cond Doxygen_Suppress
7930 0 : OGRGeometry *OGRGeometry::CastToError(OGRGeometry *poGeom)
7931 : {
7932 0 : CPLError(CE_Failure, CPLE_AppDefined, "%s found. Conversion impossible",
7933 0 : poGeom->getGeometryName());
7934 0 : delete poGeom;
7935 0 : return nullptr;
7936 : }
7937 :
7938 : //! @endcond
7939 :
7940 : /************************************************************************/
7941 : /* OGRexportToSFCGAL() */
7942 : /************************************************************************/
7943 :
7944 : //! @cond Doxygen_Suppress
7945 : sfcgal_geometry_t *
7946 0 : OGRGeometry::OGRexportToSFCGAL(UNUSED_IF_NO_SFCGAL const OGRGeometry *poGeom)
7947 : {
7948 : #ifdef HAVE_SFCGAL
7949 :
7950 : sfcgal_init();
7951 : #if SFCGAL_VERSION >= SFCGAL_MAKE_VERSION(1, 5, 2)
7952 :
7953 : const auto exportToSFCGALViaWKB =
7954 : [](const OGRGeometry *geom) -> sfcgal_geometry_t *
7955 : {
7956 : if (!geom)
7957 : return nullptr;
7958 :
7959 : // Get WKB size and allocate buffer
7960 : size_t nSize = geom->WkbSize();
7961 : unsigned char *pabyWkb = static_cast<unsigned char *>(CPLMalloc(nSize));
7962 :
7963 : // Set export options with NDR byte order
7964 : OGRwkbExportOptions oOptions;
7965 : oOptions.eByteOrder = wkbNDR;
7966 :
7967 : // Export to WKB
7968 : sfcgal_geometry_t *sfcgalGeom = nullptr;
7969 : if (geom->exportToWkb(pabyWkb, &oOptions) == OGRERR_NONE)
7970 : {
7971 : sfcgalGeom = sfcgal_io_read_wkb(
7972 : reinterpret_cast<const char *>(pabyWkb), nSize);
7973 : }
7974 :
7975 : CPLFree(pabyWkb);
7976 : return sfcgalGeom;
7977 : };
7978 :
7979 : // Handle special cases
7980 : if (EQUAL(poGeom->getGeometryName(), "LINEARRING"))
7981 : {
7982 : std::unique_ptr<OGRLineString> poLS(
7983 : OGRCurve::CastToLineString(poGeom->clone()->toCurve()));
7984 : return exportToSFCGALViaWKB(poLS.get());
7985 : }
7986 : else if (EQUAL(poGeom->getGeometryName(), "CIRCULARSTRING") ||
7987 : EQUAL(poGeom->getGeometryName(), "COMPOUNDCURVE"))
7988 : {
7989 : std::unique_ptr<OGRLineString> poLS(
7990 : OGRGeometryFactory::forceToLineString(poGeom->clone())
7991 : ->toLineString());
7992 : return exportToSFCGALViaWKB(poLS.get());
7993 : }
7994 : else if (EQUAL(poGeom->getGeometryName(), "CURVEPOLYGON"))
7995 : {
7996 : std::unique_ptr<OGRPolygon> poPolygon(
7997 : OGRGeometryFactory::forceToPolygon(
7998 : poGeom->clone()->toCurvePolygon())
7999 : ->toPolygon());
8000 : return exportToSFCGALViaWKB(poPolygon.get());
8001 : }
8002 : else
8003 : {
8004 : // Default case - direct export
8005 : return exportToSFCGALViaWKB(poGeom);
8006 : }
8007 : #else
8008 : char *buffer = nullptr;
8009 :
8010 : // special cases - LinearRing, Circular String, Compound Curve, Curve
8011 : // Polygon
8012 :
8013 : if (EQUAL(poGeom->getGeometryName(), "LINEARRING"))
8014 : {
8015 : // cast it to LineString and get the WKT
8016 : std::unique_ptr<OGRLineString> poLS(
8017 : OGRCurve::CastToLineString(poGeom->clone()->toCurve()));
8018 : if (poLS->exportToWkt(&buffer) == OGRERR_NONE)
8019 : {
8020 : sfcgal_geometry_t *_geometry =
8021 : sfcgal_io_read_wkt(buffer, strlen(buffer));
8022 : CPLFree(buffer);
8023 : return _geometry;
8024 : }
8025 : else
8026 : {
8027 : CPLFree(buffer);
8028 : return nullptr;
8029 : }
8030 : }
8031 : else if (EQUAL(poGeom->getGeometryName(), "CIRCULARSTRING") ||
8032 : EQUAL(poGeom->getGeometryName(), "COMPOUNDCURVE"))
8033 : {
8034 : // convert it to LineString and get the WKT
8035 : std::unique_ptr<OGRLineString> poLS(
8036 : OGRGeometryFactory::forceToLineString(poGeom->clone())
8037 : ->toLineString());
8038 : if (poLS->exportToWkt(&buffer) == OGRERR_NONE)
8039 : {
8040 : sfcgal_geometry_t *_geometry =
8041 : sfcgal_io_read_wkt(buffer, strlen(buffer));
8042 : CPLFree(buffer);
8043 : return _geometry;
8044 : }
8045 : else
8046 : {
8047 : CPLFree(buffer);
8048 : return nullptr;
8049 : }
8050 : }
8051 : else if (EQUAL(poGeom->getGeometryName(), "CURVEPOLYGON"))
8052 : {
8053 : // convert it to Polygon and get the WKT
8054 : std::unique_ptr<OGRPolygon> poPolygon(
8055 : OGRGeometryFactory::forceToPolygon(
8056 : poGeom->clone()->toCurvePolygon())
8057 : ->toPolygon());
8058 : if (poPolygon->exportToWkt(&buffer) == OGRERR_NONE)
8059 : {
8060 : sfcgal_geometry_t *_geometry =
8061 : sfcgal_io_read_wkt(buffer, strlen(buffer));
8062 : CPLFree(buffer);
8063 : return _geometry;
8064 : }
8065 : else
8066 : {
8067 : CPLFree(buffer);
8068 : return nullptr;
8069 : }
8070 : }
8071 : else if (poGeom->exportToWkt(&buffer) == OGRERR_NONE)
8072 : {
8073 : sfcgal_geometry_t *_geometry =
8074 : sfcgal_io_read_wkt(buffer, strlen(buffer));
8075 : CPLFree(buffer);
8076 : return _geometry;
8077 : }
8078 : else
8079 : {
8080 : CPLFree(buffer);
8081 : return nullptr;
8082 : }
8083 : #endif
8084 : #else
8085 0 : CPLError(CE_Failure, CPLE_NotSupported, "SFCGAL support not enabled.");
8086 0 : return nullptr;
8087 : #endif
8088 : }
8089 :
8090 : //! @endcond
8091 :
8092 : /************************************************************************/
8093 : /* SFCGALexportToOGR() */
8094 : /************************************************************************/
8095 :
8096 : //! @cond Doxygen_Suppress
8097 0 : OGRGeometry *OGRGeometry::SFCGALexportToOGR(
8098 : UNUSED_IF_NO_SFCGAL const sfcgal_geometry_t *geometry)
8099 : {
8100 : #ifdef HAVE_SFCGAL
8101 : if (geometry == nullptr)
8102 : return nullptr;
8103 :
8104 : sfcgal_init();
8105 : char *pabySFCGAL = nullptr;
8106 : size_t nLength = 0;
8107 : #if SFCGAL_VERSION >= SFCGAL_MAKE_VERSION(1, 5, 2)
8108 :
8109 : sfcgal_geometry_as_wkb(geometry, &pabySFCGAL, &nLength);
8110 :
8111 : if (pabySFCGAL == nullptr || nLength == 0)
8112 : return nullptr;
8113 :
8114 : OGRGeometry *poGeom = nullptr;
8115 : OGRErr eErr = OGRGeometryFactory::createFromWkb(
8116 : reinterpret_cast<unsigned char *>(pabySFCGAL), nullptr, &poGeom,
8117 : nLength);
8118 :
8119 : free(pabySFCGAL);
8120 :
8121 : if (eErr == OGRERR_NONE)
8122 : {
8123 : return poGeom;
8124 : }
8125 : else
8126 : {
8127 : return nullptr;
8128 : }
8129 : #else
8130 : sfcgal_geometry_as_text_decim(geometry, 19, &pabySFCGAL, &nLength);
8131 : char *pszWKT = static_cast<char *>(CPLMalloc(nLength + 1));
8132 : memcpy(pszWKT, pabySFCGAL, nLength);
8133 : pszWKT[nLength] = 0;
8134 : free(pabySFCGAL);
8135 :
8136 : sfcgal_geometry_type_t geom_type = sfcgal_geometry_type_id(geometry);
8137 :
8138 : OGRGeometry *poGeom = nullptr;
8139 : if (geom_type == SFCGAL_TYPE_POINT)
8140 : {
8141 : poGeom = new OGRPoint();
8142 : }
8143 : else if (geom_type == SFCGAL_TYPE_LINESTRING)
8144 : {
8145 : poGeom = new OGRLineString();
8146 : }
8147 : else if (geom_type == SFCGAL_TYPE_POLYGON)
8148 : {
8149 : poGeom = new OGRPolygon();
8150 : }
8151 : else if (geom_type == SFCGAL_TYPE_MULTIPOINT)
8152 : {
8153 : poGeom = new OGRMultiPoint();
8154 : }
8155 : else if (geom_type == SFCGAL_TYPE_MULTILINESTRING)
8156 : {
8157 : poGeom = new OGRMultiLineString();
8158 : }
8159 : else if (geom_type == SFCGAL_TYPE_MULTIPOLYGON)
8160 : {
8161 : poGeom = new OGRMultiPolygon();
8162 : }
8163 : else if (geom_type == SFCGAL_TYPE_GEOMETRYCOLLECTION)
8164 : {
8165 : poGeom = new OGRGeometryCollection();
8166 : }
8167 : else if (geom_type == SFCGAL_TYPE_TRIANGLE)
8168 : {
8169 : poGeom = new OGRTriangle();
8170 : }
8171 : else if (geom_type == SFCGAL_TYPE_POLYHEDRALSURFACE)
8172 : {
8173 : poGeom = new OGRPolyhedralSurface();
8174 : }
8175 : else if (geom_type == SFCGAL_TYPE_TRIANGULATEDSURFACE)
8176 : {
8177 : poGeom = new OGRTriangulatedSurface();
8178 : }
8179 : else
8180 : {
8181 : CPLFree(pszWKT);
8182 : return nullptr;
8183 : }
8184 :
8185 : const char *pszWKTTmp = pszWKT;
8186 : if (poGeom->importFromWkt(&pszWKTTmp) == OGRERR_NONE)
8187 : {
8188 : CPLFree(pszWKT);
8189 : return poGeom;
8190 : }
8191 : else
8192 : {
8193 : delete poGeom;
8194 : CPLFree(pszWKT);
8195 : return nullptr;
8196 : }
8197 : #endif
8198 : #else
8199 0 : CPLError(CE_Failure, CPLE_NotSupported, "SFCGAL support not enabled.");
8200 0 : return nullptr;
8201 : #endif
8202 : }
8203 :
8204 : //! @endcond
8205 :
8206 : //! @cond Doxygen_Suppress
8207 3971 : OGRBoolean OGRGeometry::IsSFCGALCompatible() const
8208 : {
8209 3971 : const OGRwkbGeometryType eGType = wkbFlatten(getGeometryType());
8210 3943 : if (eGType == wkbTriangle || eGType == wkbPolyhedralSurface ||
8211 : eGType == wkbTIN)
8212 : {
8213 2 : return TRUE;
8214 : }
8215 3941 : if (eGType == wkbGeometryCollection || eGType == wkbMultiSurface)
8216 : {
8217 15 : const OGRGeometryCollection *poGC = toGeometryCollection();
8218 8 : bool bIsSFCGALCompatible = false;
8219 8 : for (auto &&poSubGeom : *poGC)
8220 : {
8221 : OGRwkbGeometryType eSubGeomType =
8222 8 : wkbFlatten(poSubGeom->getGeometryType());
8223 8 : if (eSubGeomType == wkbTIN || eSubGeomType == wkbPolyhedralSurface)
8224 : {
8225 0 : bIsSFCGALCompatible = true;
8226 : }
8227 8 : else if (eSubGeomType != wkbMultiPolygon)
8228 : {
8229 8 : bIsSFCGALCompatible = false;
8230 8 : break;
8231 : }
8232 : }
8233 8 : return bIsSFCGALCompatible;
8234 : }
8235 3926 : return FALSE;
8236 : }
8237 :
8238 : //! @endcond
8239 :
8240 : /************************************************************************/
8241 : /* roundCoordinatesIEEE754() */
8242 : /************************************************************************/
8243 :
8244 : /** Round coordinates of a geometry, exploiting characteristics of the IEEE-754
8245 : * double-precision binary representation.
8246 : *
8247 : * Determines the number of bits (N) required to represent a coordinate value
8248 : * with a specified number of digits after the decimal point, and then sets all
8249 : * but the N most significant bits to zero. The resulting coordinate value will
8250 : * still round to the original value (e.g. after roundCoordinates()), but will
8251 : * have improved compressiblity.
8252 : *
8253 : * @param options Contains the precision requirements.
8254 : * @since GDAL 3.9
8255 : */
8256 1 : void OGRGeometry::roundCoordinatesIEEE754(
8257 : const OGRGeomCoordinateBinaryPrecision &options)
8258 : {
8259 : struct Quantizer : public OGRDefaultGeometryVisitor
8260 : {
8261 : const OGRGeomCoordinateBinaryPrecision &m_options;
8262 :
8263 1 : explicit Quantizer(const OGRGeomCoordinateBinaryPrecision &optionsIn)
8264 1 : : m_options(optionsIn)
8265 : {
8266 1 : }
8267 :
8268 : using OGRDefaultGeometryVisitor::visit;
8269 :
8270 3 : void visit(OGRPoint *poPoint) override
8271 : {
8272 3 : if (m_options.nXYBitPrecision != INT_MIN)
8273 : {
8274 : uint64_t i;
8275 : double d;
8276 3 : d = poPoint->getX();
8277 3 : memcpy(&i, &d, sizeof(i));
8278 3 : i = OGRRoundValueIEEE754(i, m_options.nXYBitPrecision);
8279 3 : memcpy(&d, &i, sizeof(i));
8280 3 : poPoint->setX(d);
8281 3 : d = poPoint->getY();
8282 3 : memcpy(&i, &d, sizeof(i));
8283 3 : i = OGRRoundValueIEEE754(i, m_options.nXYBitPrecision);
8284 3 : memcpy(&d, &i, sizeof(i));
8285 3 : poPoint->setY(d);
8286 : }
8287 3 : if (m_options.nZBitPrecision != INT_MIN && poPoint->Is3D())
8288 : {
8289 : uint64_t i;
8290 : double d;
8291 3 : d = poPoint->getZ();
8292 3 : memcpy(&i, &d, sizeof(i));
8293 3 : i = OGRRoundValueIEEE754(i, m_options.nZBitPrecision);
8294 3 : memcpy(&d, &i, sizeof(i));
8295 3 : poPoint->setZ(d);
8296 : }
8297 3 : if (m_options.nMBitPrecision != INT_MIN && poPoint->IsMeasured())
8298 : {
8299 : uint64_t i;
8300 : double d;
8301 3 : d = poPoint->getM();
8302 3 : memcpy(&i, &d, sizeof(i));
8303 3 : i = OGRRoundValueIEEE754(i, m_options.nMBitPrecision);
8304 3 : memcpy(&d, &i, sizeof(i));
8305 3 : poPoint->setM(d);
8306 : }
8307 3 : }
8308 : };
8309 :
8310 2 : Quantizer quantizer(options);
8311 1 : accept(&quantizer);
8312 1 : }
8313 :
8314 : /************************************************************************/
8315 : /* visit() */
8316 : /************************************************************************/
8317 :
8318 105 : void OGRDefaultGeometryVisitor::_visit(OGRSimpleCurve *poGeom)
8319 : {
8320 1248 : for (auto &&oPoint : *poGeom)
8321 : {
8322 1143 : oPoint.accept(this);
8323 : }
8324 105 : }
8325 :
8326 104 : void OGRDefaultGeometryVisitor::visit(OGRLineString *poGeom)
8327 : {
8328 104 : _visit(poGeom);
8329 104 : }
8330 :
8331 80 : void OGRDefaultGeometryVisitor::visit(OGRLinearRing *poGeom)
8332 : {
8333 80 : visit(poGeom->toUpperClass());
8334 80 : }
8335 :
8336 1 : void OGRDefaultGeometryVisitor::visit(OGRCircularString *poGeom)
8337 : {
8338 1 : _visit(poGeom);
8339 1 : }
8340 :
8341 78 : void OGRDefaultGeometryVisitor::visit(OGRCurvePolygon *poGeom)
8342 : {
8343 159 : for (auto &&poSubGeom : *poGeom)
8344 81 : poSubGeom->accept(this);
8345 78 : }
8346 :
8347 77 : void OGRDefaultGeometryVisitor::visit(OGRPolygon *poGeom)
8348 : {
8349 77 : visit(poGeom->toUpperClass());
8350 77 : }
8351 :
8352 1 : void OGRDefaultGeometryVisitor::visit(OGRMultiPoint *poGeom)
8353 : {
8354 1 : visit(poGeom->toUpperClass());
8355 1 : }
8356 :
8357 8 : void OGRDefaultGeometryVisitor::visit(OGRMultiLineString *poGeom)
8358 : {
8359 8 : visit(poGeom->toUpperClass());
8360 8 : }
8361 :
8362 14 : void OGRDefaultGeometryVisitor::visit(OGRMultiPolygon *poGeom)
8363 : {
8364 14 : visit(poGeom->toUpperClass());
8365 14 : }
8366 :
8367 26 : void OGRDefaultGeometryVisitor::visit(OGRGeometryCollection *poGeom)
8368 : {
8369 75 : for (auto &&poSubGeom : *poGeom)
8370 49 : poSubGeom->accept(this);
8371 26 : }
8372 :
8373 1 : void OGRDefaultGeometryVisitor::visit(OGRCompoundCurve *poGeom)
8374 : {
8375 2 : for (auto &&poSubGeom : *poGeom)
8376 1 : poSubGeom->accept(this);
8377 1 : }
8378 :
8379 1 : void OGRDefaultGeometryVisitor::visit(OGRMultiCurve *poGeom)
8380 : {
8381 1 : visit(poGeom->toUpperClass());
8382 1 : }
8383 :
8384 1 : void OGRDefaultGeometryVisitor::visit(OGRMultiSurface *poGeom)
8385 : {
8386 1 : visit(poGeom->toUpperClass());
8387 1 : }
8388 :
8389 2 : void OGRDefaultGeometryVisitor::visit(OGRTriangle *poGeom)
8390 : {
8391 2 : visit(poGeom->toUpperClass());
8392 2 : }
8393 :
8394 2 : void OGRDefaultGeometryVisitor::visit(OGRPolyhedralSurface *poGeom)
8395 : {
8396 4 : for (auto &&poSubGeom : *poGeom)
8397 2 : poSubGeom->accept(this);
8398 2 : }
8399 :
8400 1 : void OGRDefaultGeometryVisitor::visit(OGRTriangulatedSurface *poGeom)
8401 : {
8402 1 : visit(poGeom->toUpperClass());
8403 1 : }
8404 :
8405 14 : void OGRDefaultConstGeometryVisitor::_visit(const OGRSimpleCurve *poGeom)
8406 : {
8407 66 : for (auto &&oPoint : *poGeom)
8408 : {
8409 52 : oPoint.accept(this);
8410 : }
8411 14 : }
8412 :
8413 13 : void OGRDefaultConstGeometryVisitor::visit(const OGRLineString *poGeom)
8414 : {
8415 13 : _visit(poGeom);
8416 13 : }
8417 :
8418 8 : void OGRDefaultConstGeometryVisitor::visit(const OGRLinearRing *poGeom)
8419 : {
8420 8 : visit(poGeom->toUpperClass());
8421 8 : }
8422 :
8423 1 : void OGRDefaultConstGeometryVisitor::visit(const OGRCircularString *poGeom)
8424 : {
8425 1 : _visit(poGeom);
8426 1 : }
8427 :
8428 8 : void OGRDefaultConstGeometryVisitor::visit(const OGRCurvePolygon *poGeom)
8429 : {
8430 17 : for (auto &&poSubGeom : *poGeom)
8431 9 : poSubGeom->accept(this);
8432 8 : }
8433 :
8434 7 : void OGRDefaultConstGeometryVisitor::visit(const OGRPolygon *poGeom)
8435 : {
8436 7 : visit(poGeom->toUpperClass());
8437 7 : }
8438 :
8439 40 : void OGRDefaultConstGeometryVisitor::visit(const OGRMultiPoint *poGeom)
8440 : {
8441 40 : visit(poGeom->toUpperClass());
8442 40 : }
8443 :
8444 1 : void OGRDefaultConstGeometryVisitor::visit(const OGRMultiLineString *poGeom)
8445 : {
8446 1 : visit(poGeom->toUpperClass());
8447 1 : }
8448 :
8449 1 : void OGRDefaultConstGeometryVisitor::visit(const OGRMultiPolygon *poGeom)
8450 : {
8451 1 : visit(poGeom->toUpperClass());
8452 1 : }
8453 :
8454 45 : void OGRDefaultConstGeometryVisitor::visit(const OGRGeometryCollection *poGeom)
8455 : {
8456 217 : for (auto &&poSubGeom : *poGeom)
8457 172 : poSubGeom->accept(this);
8458 45 : }
8459 :
8460 1 : void OGRDefaultConstGeometryVisitor::visit(const OGRCompoundCurve *poGeom)
8461 : {
8462 2 : for (auto &&poSubGeom : *poGeom)
8463 1 : poSubGeom->accept(this);
8464 1 : }
8465 :
8466 1 : void OGRDefaultConstGeometryVisitor::visit(const OGRMultiCurve *poGeom)
8467 : {
8468 1 : visit(poGeom->toUpperClass());
8469 1 : }
8470 :
8471 1 : void OGRDefaultConstGeometryVisitor::visit(const OGRMultiSurface *poGeom)
8472 : {
8473 1 : visit(poGeom->toUpperClass());
8474 1 : }
8475 :
8476 2 : void OGRDefaultConstGeometryVisitor::visit(const OGRTriangle *poGeom)
8477 : {
8478 2 : visit(poGeom->toUpperClass());
8479 2 : }
8480 :
8481 2 : void OGRDefaultConstGeometryVisitor::visit(const OGRPolyhedralSurface *poGeom)
8482 : {
8483 4 : for (auto &&poSubGeom : *poGeom)
8484 2 : poSubGeom->accept(this);
8485 2 : }
8486 :
8487 1 : void OGRDefaultConstGeometryVisitor::visit(const OGRTriangulatedSurface *poGeom)
8488 : {
8489 1 : visit(poGeom->toUpperClass());
8490 1 : }
8491 :
8492 : /************************************************************************/
8493 : /* OGRGeometryUniquePtrDeleter */
8494 : /************************************************************************/
8495 :
8496 : //! @cond Doxygen_Suppress
8497 112 : void OGRGeometryUniquePtrDeleter::operator()(OGRGeometry *poGeom) const
8498 : {
8499 112 : delete poGeom;
8500 112 : }
8501 :
8502 : //! @endcond
8503 :
8504 : /************************************************************************/
8505 : /* OGRPreparedGeometryUniquePtrDeleter */
8506 : /************************************************************************/
8507 :
8508 : //! @cond Doxygen_Suppress
8509 39 : void OGRPreparedGeometryUniquePtrDeleter::operator()(
8510 : OGRPreparedGeometry *poPreparedGeom) const
8511 : {
8512 39 : OGRDestroyPreparedGeometry(poPreparedGeom);
8513 39 : }
8514 :
8515 : //! @endcond
8516 :
8517 : /************************************************************************/
8518 : /* HomogenizeDimensionalityWith() */
8519 : /************************************************************************/
8520 :
8521 : //! @cond Doxygen_Suppress
8522 3139440 : void OGRGeometry::HomogenizeDimensionalityWith(OGRGeometry *poOtherGeom)
8523 : {
8524 3139440 : if (poOtherGeom->Is3D() && !Is3D())
8525 1293900 : set3D(TRUE);
8526 :
8527 3139420 : if (poOtherGeom->IsMeasured() && !IsMeasured())
8528 792 : setMeasured(TRUE);
8529 :
8530 3139410 : if (!poOtherGeom->Is3D() && Is3D())
8531 295 : poOtherGeom->set3D(TRUE);
8532 :
8533 3139420 : if (!poOtherGeom->IsMeasured() && IsMeasured())
8534 28 : poOtherGeom->setMeasured(TRUE);
8535 3139420 : }
8536 :
8537 : //! @endcond
8538 :
8539 : /************************************************************************/
8540 : /* OGRGeomCoordinateBinaryPrecision::SetFrom() */
8541 : /************************************************************************/
8542 :
8543 : /** Set binary precision options from resolution.
8544 : *
8545 : * @since GDAL 3.9
8546 : */
8547 16 : void OGRGeomCoordinateBinaryPrecision::SetFrom(
8548 : const OGRGeomCoordinatePrecision &prec)
8549 : {
8550 16 : if (prec.dfXYResolution != 0)
8551 : {
8552 16 : nXYBitPrecision =
8553 16 : static_cast<int>(ceil(log2(1. / prec.dfXYResolution)));
8554 : }
8555 16 : if (prec.dfZResolution != 0)
8556 : {
8557 12 : nZBitPrecision = static_cast<int>(ceil(log2(1. / prec.dfZResolution)));
8558 : }
8559 16 : if (prec.dfMResolution != 0)
8560 : {
8561 12 : nMBitPrecision = static_cast<int>(ceil(log2(1. / prec.dfMResolution)));
8562 : }
8563 16 : }
8564 :
8565 : /************************************************************************/
8566 : /* OGRwkbExportOptionsCreate() */
8567 : /************************************************************************/
8568 :
8569 : /**
8570 : * \brief Create geometry WKB export options.
8571 : *
8572 : * The default is Intel order, old-OGC wkb variant and 0 discarded lsb bits.
8573 : *
8574 : * @return object to be freed with OGRwkbExportOptionsDestroy().
8575 : * @since GDAL 3.9
8576 : */
8577 2 : OGRwkbExportOptions *OGRwkbExportOptionsCreate()
8578 : {
8579 2 : return new OGRwkbExportOptions;
8580 : }
8581 :
8582 : /************************************************************************/
8583 : /* OGRwkbExportOptionsDestroy() */
8584 : /************************************************************************/
8585 :
8586 : /**
8587 : * \brief Destroy object returned by OGRwkbExportOptionsCreate()
8588 : *
8589 : * @param psOptions WKB export options
8590 : * @since GDAL 3.9
8591 : */
8592 :
8593 2 : void OGRwkbExportOptionsDestroy(OGRwkbExportOptions *psOptions)
8594 : {
8595 2 : delete psOptions;
8596 2 : }
8597 :
8598 : /************************************************************************/
8599 : /* OGRwkbExportOptionsSetByteOrder() */
8600 : /************************************************************************/
8601 :
8602 : /**
8603 : * \brief Set the WKB byte order.
8604 : *
8605 : * @param psOptions WKB export options
8606 : * @param eByteOrder Byte order: wkbXDR (big-endian) or wkbNDR (little-endian,
8607 : * Intel)
8608 : * @since GDAL 3.9
8609 : */
8610 :
8611 1 : void OGRwkbExportOptionsSetByteOrder(OGRwkbExportOptions *psOptions,
8612 : OGRwkbByteOrder eByteOrder)
8613 : {
8614 1 : psOptions->eByteOrder = eByteOrder;
8615 1 : }
8616 :
8617 : /************************************************************************/
8618 : /* OGRwkbExportOptionsSetVariant() */
8619 : /************************************************************************/
8620 :
8621 : /**
8622 : * \brief Set the WKB variant
8623 : *
8624 : * @param psOptions WKB export options
8625 : * @param eWkbVariant variant: wkbVariantOldOgc, wkbVariantIso,
8626 : * wkbVariantPostGIS1
8627 : * @since GDAL 3.9
8628 : */
8629 :
8630 1 : void OGRwkbExportOptionsSetVariant(OGRwkbExportOptions *psOptions,
8631 : OGRwkbVariant eWkbVariant)
8632 : {
8633 1 : psOptions->eWkbVariant = eWkbVariant;
8634 1 : }
8635 :
8636 : /************************************************************************/
8637 : /* OGRwkbExportOptionsSetPrecision() */
8638 : /************************************************************************/
8639 :
8640 : /**
8641 : * \brief Set precision options
8642 : *
8643 : * @param psOptions WKB export options
8644 : * @param hPrecisionOptions Precision options (might be null to reset them)
8645 : * @since GDAL 3.9
8646 : */
8647 :
8648 1 : void OGRwkbExportOptionsSetPrecision(
8649 : OGRwkbExportOptions *psOptions,
8650 : OGRGeomCoordinatePrecisionH hPrecisionOptions)
8651 : {
8652 1 : psOptions->sPrecision = OGRGeomCoordinateBinaryPrecision();
8653 1 : if (hPrecisionOptions)
8654 1 : psOptions->sPrecision.SetFrom(*hPrecisionOptions);
8655 1 : }
8656 :
8657 : /************************************************************************/
8658 : /* IsRectangle() */
8659 : /************************************************************************/
8660 :
8661 : /**
8662 : * \brief Returns whether the geometry is a polygon with 4 corners forming
8663 : * a rectangle.
8664 : *
8665 : * @since GDAL 3.10
8666 : */
8667 52040 : bool OGRGeometry::IsRectangle() const
8668 : {
8669 52040 : if (wkbFlatten(getGeometryType()) != wkbPolygon)
8670 253 : return false;
8671 :
8672 51787 : const OGRPolygon *poPoly = toPolygon();
8673 :
8674 51787 : if (poPoly->getNumInteriorRings() != 0)
8675 19 : return false;
8676 :
8677 51768 : const OGRLinearRing *poRing = poPoly->getExteriorRing();
8678 51768 : if (!poRing)
8679 4 : return false;
8680 :
8681 51764 : if (poRing->getNumPoints() > 5 || poRing->getNumPoints() < 4)
8682 6 : return false;
8683 :
8684 : // If the ring has 5 points, the last should be the first.
8685 103479 : if (poRing->getNumPoints() == 5 && (poRing->getX(0) != poRing->getX(4) ||
8686 51721 : poRing->getY(0) != poRing->getY(4)))
8687 1 : return false;
8688 :
8689 : // Polygon with first segment in "y" direction.
8690 102884 : if (poRing->getX(0) == poRing->getX(1) &&
8691 102253 : poRing->getY(1) == poRing->getY(2) &&
8692 154010 : poRing->getX(2) == poRing->getX(3) &&
8693 51126 : poRing->getY(3) == poRing->getY(0))
8694 51126 : return true;
8695 :
8696 : // Polygon with first segment in "x" direction.
8697 1221 : if (poRing->getY(0) == poRing->getY(1) &&
8698 1180 : poRing->getX(1) == poRing->getX(2) &&
8699 1811 : poRing->getY(2) == poRing->getY(3) &&
8700 590 : poRing->getX(3) == poRing->getX(0))
8701 590 : return true;
8702 :
8703 41 : return false;
8704 : }
8705 :
8706 : /************************************************************************/
8707 : /* hasEmptyParts() */
8708 : /************************************************************************/
8709 :
8710 : /**
8711 : * \brief Returns whether a geometry has empty parts/rings.
8712 : *
8713 : * Returns true if removeEmptyParts() will modify the geometry.
8714 : *
8715 : * This is different from IsEmpty().
8716 : *
8717 : * @since GDAL 3.10
8718 : */
8719 19 : bool OGRGeometry::hasEmptyParts() const
8720 : {
8721 19 : return false;
8722 : }
8723 :
8724 : /************************************************************************/
8725 : /* removeEmptyParts() */
8726 : /************************************************************************/
8727 :
8728 : /**
8729 : * \brief Remove empty parts/rings from this geometry.
8730 : *
8731 : * @since GDAL 3.10
8732 : */
8733 17 : void OGRGeometry::removeEmptyParts()
8734 : {
8735 17 : }
|