Line data Source code
1 : /******************************************************************************
2 : *
3 : * Project: OpenGIS Simple Features Reference Implementation
4 : * Purpose: The OGRSimpleCurve and OGRLineString geometry classes.
5 : * Author: Frank Warmerdam, warmerdam@pobox.com
6 : *
7 : ******************************************************************************
8 : * Copyright (c) 1999, Frank Warmerdam
9 : * Copyright (c) 2008-2014, Even Rouault <even dot rouault at spatialys.com>
10 : *
11 : * SPDX-License-Identifier: MIT
12 : ****************************************************************************/
13 :
14 : #include "ogr_geometry.h"
15 : #include "ogr_geos.h"
16 : #include "ogr_p.h"
17 :
18 : #include "geodesic.h" // from PROJ
19 :
20 : #include <cmath>
21 : #include <cstdlib>
22 : #include <algorithm>
23 : #include <limits>
24 : #include <new>
25 :
26 : namespace
27 : {
28 :
29 567 : int DoubleToIntClamp(double dfValue)
30 : {
31 567 : if (std::isnan(dfValue))
32 0 : return 0;
33 567 : if (dfValue >= std::numeric_limits<int>::max())
34 1 : return std::numeric_limits<int>::max();
35 566 : if (dfValue <= std::numeric_limits<int>::min())
36 0 : return std::numeric_limits<int>::min();
37 566 : return static_cast<int>(dfValue);
38 : }
39 :
40 : } // namespace
41 :
42 : /************************************************************************/
43 : /* OGRSimpleCurve( const OGRSimpleCurve& ) */
44 : /************************************************************************/
45 :
46 : /**
47 : * \brief Copy constructor.
48 : *
49 : * Note: before GDAL 2.1, only the default implementation of the constructor
50 : * existed, which could be unsafe to use.
51 : *
52 : * @since GDAL 2.1
53 : */
54 :
55 301286 : OGRSimpleCurve::OGRSimpleCurve(const OGRSimpleCurve &other)
56 : : OGRCurve(other), nPointCount(0), paoPoints(nullptr), padfZ(nullptr),
57 301286 : padfM(nullptr)
58 : {
59 301286 : if (other.nPointCount > 0)
60 301077 : setPoints(other.nPointCount, other.paoPoints, other.padfZ, other.padfM);
61 301286 : }
62 :
63 : /************************************************************************/
64 : /* OGRSimpleCurve( OGRSimpleCurve&& ) */
65 : /************************************************************************/
66 :
67 : /**
68 : * \brief Move constructor.
69 : *
70 : * @since GDAL 3.11
71 : */
72 :
73 : // cppcheck-suppress-begin accessMoved
74 3 : OGRSimpleCurve::OGRSimpleCurve(OGRSimpleCurve &&other)
75 6 : : OGRCurve(std::move(other)), nPointCount(other.nPointCount),
76 3 : m_nPointCapacity(other.m_nPointCapacity), paoPoints(other.paoPoints),
77 3 : padfZ(other.padfZ), padfM(other.padfM)
78 : {
79 3 : other.nPointCount = 0;
80 3 : other.m_nPointCapacity = 0;
81 3 : other.paoPoints = nullptr;
82 3 : other.padfZ = nullptr;
83 3 : other.padfM = nullptr;
84 3 : }
85 :
86 : // cppcheck-suppress-end accessMoved
87 :
88 : /************************************************************************/
89 : /* ~OGRSimpleCurve() */
90 : /************************************************************************/
91 :
92 3772600 : OGRSimpleCurve::~OGRSimpleCurve()
93 :
94 : {
95 3772640 : CPLFree(paoPoints);
96 3772640 : CPLFree(padfZ);
97 3772600 : CPLFree(padfM);
98 3772610 : }
99 :
100 : /************************************************************************/
101 : /* operator=(const OGRSimpleCurve &other) */
102 : /************************************************************************/
103 :
104 : /**
105 : * \brief Assignment operator.
106 : *
107 : * Note: before GDAL 2.1, only the default implementation of the operator
108 : * existed, which could be unsafe to use.
109 : *
110 : * @since GDAL 2.1
111 : */
112 :
113 12 : OGRSimpleCurve &OGRSimpleCurve::operator=(const OGRSimpleCurve &other)
114 : {
115 12 : if (this == &other)
116 0 : return *this;
117 :
118 12 : OGRCurve::operator=(other);
119 :
120 12 : setPoints(other.nPointCount, other.paoPoints, other.padfZ, other.padfM);
121 12 : flags = other.flags;
122 :
123 12 : return *this;
124 : }
125 :
126 : /************************************************************************/
127 : /* operator=(OGRSimpleCurve &&other) */
128 : /************************************************************************/
129 :
130 : /**
131 : * \brief Move assignment operator.
132 : *
133 : * @since GDAL 3.11
134 : */
135 :
136 6 : OGRSimpleCurve &OGRSimpleCurve::operator=(OGRSimpleCurve &&other)
137 : {
138 6 : if (this != &other)
139 : {
140 : // cppcheck-suppress-begin accessMoved
141 6 : OGRCurve::operator=(std::move(other));
142 :
143 6 : nPointCount = other.nPointCount;
144 6 : m_nPointCapacity = other.m_nPointCapacity;
145 6 : CPLFree(paoPoints);
146 6 : paoPoints = other.paoPoints;
147 6 : CPLFree(padfZ);
148 6 : padfZ = other.padfZ;
149 6 : CPLFree(padfM);
150 6 : padfM = other.padfM;
151 6 : flags = other.flags;
152 6 : other.nPointCount = 0;
153 6 : other.m_nPointCapacity = 0;
154 6 : other.paoPoints = nullptr;
155 6 : other.padfZ = nullptr;
156 6 : other.padfM = nullptr;
157 : // cppcheck-suppress-end accessMoved
158 : }
159 :
160 6 : return *this;
161 : }
162 :
163 : /************************************************************************/
164 : /* flattenTo2D() */
165 : /************************************************************************/
166 :
167 1065 : void OGRSimpleCurve::flattenTo2D()
168 :
169 : {
170 1065 : Make2D();
171 1065 : setMeasured(FALSE);
172 1065 : }
173 :
174 : /************************************************************************/
175 : /* empty() */
176 : /************************************************************************/
177 :
178 14297 : void OGRSimpleCurve::empty()
179 :
180 : {
181 14297 : setNumPoints(0);
182 14297 : }
183 :
184 : /************************************************************************/
185 : /* setCoordinateDimension() */
186 : /************************************************************************/
187 :
188 2180 : bool OGRSimpleCurve::setCoordinateDimension(int nNewDimension)
189 :
190 : {
191 2180 : setMeasured(FALSE);
192 2180 : if (nNewDimension == 2)
193 1172 : Make2D();
194 1008 : else if (nNewDimension == 3)
195 1008 : return Make3D();
196 1172 : return true;
197 : }
198 :
199 119711 : bool OGRSimpleCurve::set3D(OGRBoolean bIs3D)
200 :
201 : {
202 119711 : if (bIs3D)
203 117239 : return Make3D();
204 : else
205 2472 : Make2D();
206 2472 : return true;
207 : }
208 :
209 119372 : bool OGRSimpleCurve::setMeasured(OGRBoolean bIsMeasured)
210 :
211 : {
212 119372 : if (bIsMeasured)
213 8192 : return AddM();
214 : else
215 111180 : RemoveM();
216 111180 : return true;
217 : }
218 :
219 : /************************************************************************/
220 : /* WkbSize() */
221 : /* */
222 : /* Return the size of this object in well known binary */
223 : /* representation including the byte order, and type information. */
224 : /************************************************************************/
225 :
226 92258 : size_t OGRSimpleCurve::WkbSize() const
227 :
228 : {
229 92258 : return 5 + 4 + 8 * static_cast<size_t>(nPointCount) * CoordinateDimension();
230 : }
231 :
232 : //! @cond Doxygen_Suppress
233 :
234 : /************************************************************************/
235 : /* Make2D() */
236 : /************************************************************************/
237 :
238 49683 : void OGRSimpleCurve::Make2D()
239 :
240 : {
241 49683 : if (padfZ != nullptr)
242 : {
243 1872 : CPLFree(padfZ);
244 1872 : padfZ = nullptr;
245 : }
246 49683 : flags &= ~OGR_G_3D;
247 49683 : }
248 :
249 : /************************************************************************/
250 : /* Make3D() */
251 : /************************************************************************/
252 :
253 2776450 : bool OGRSimpleCurve::Make3D()
254 :
255 : {
256 2776450 : if (padfZ == nullptr)
257 : {
258 2661660 : padfZ = static_cast<double *>(
259 2661660 : VSI_CALLOC_VERBOSE(sizeof(double), std::max(1, m_nPointCapacity)));
260 2661660 : if (padfZ == nullptr)
261 : {
262 0 : flags &= ~OGR_G_3D;
263 0 : CPLError(CE_Failure, CPLE_AppDefined,
264 : "OGRSimpleCurve::Make3D() failed");
265 0 : return false;
266 : }
267 : }
268 2776450 : flags |= OGR_G_3D;
269 2776450 : return true;
270 : }
271 :
272 : /************************************************************************/
273 : /* RemoveM() */
274 : /************************************************************************/
275 :
276 125606 : void OGRSimpleCurve::RemoveM()
277 :
278 : {
279 125606 : if (padfM != nullptr)
280 : {
281 521 : CPLFree(padfM);
282 521 : padfM = nullptr;
283 : }
284 125606 : flags &= ~OGR_G_MEASURED;
285 125606 : }
286 :
287 : /************************************************************************/
288 : /* AddM() */
289 : /************************************************************************/
290 :
291 39218 : bool OGRSimpleCurve::AddM()
292 :
293 : {
294 39218 : if (padfM == nullptr)
295 : {
296 34705 : padfM = static_cast<double *>(
297 34705 : VSI_CALLOC_VERBOSE(sizeof(double), std::max(1, m_nPointCapacity)));
298 34705 : if (padfM == nullptr)
299 : {
300 0 : flags &= ~OGR_G_MEASURED;
301 0 : CPLError(CE_Failure, CPLE_AppDefined,
302 : "OGRSimpleCurve::AddM() failed");
303 0 : return false;
304 : }
305 : }
306 39218 : flags |= OGR_G_MEASURED;
307 39218 : return true;
308 : }
309 :
310 : //! @endcond
311 :
312 : /************************************************************************/
313 : /* getPoint() */
314 : /************************************************************************/
315 :
316 : /**
317 : * \brief Fetch a point in line string.
318 : *
319 : * This method relates to the SFCOM ILineString::get_Point() method.
320 : *
321 : * @param i the vertex to fetch, from 0 to getNumPoints()-1.
322 : * @param poPoint a point to initialize with the fetched point.
323 : */
324 :
325 1165040 : void OGRSimpleCurve::getPoint(int i, OGRPoint *poPoint) const
326 :
327 : {
328 1165040 : CPLAssert(i >= 0);
329 1165040 : CPLAssert(i < nPointCount);
330 1165040 : CPLAssert(poPoint != nullptr);
331 :
332 1165040 : poPoint->setX(paoPoints[i].x);
333 1165040 : poPoint->setY(paoPoints[i].y);
334 :
335 1165040 : if ((flags & OGR_G_3D) && padfZ != nullptr)
336 46025 : poPoint->setZ(padfZ[i]);
337 1165040 : if ((flags & OGR_G_MEASURED) && padfM != nullptr)
338 28415 : poPoint->setM(padfM[i]);
339 1165040 : }
340 :
341 : /**
342 : * \fn int OGRSimpleCurve::getNumPoints() const;
343 : *
344 : * \brief Fetch vertex count.
345 : *
346 : * Returns the number of vertices in the line string.
347 : *
348 : * @return vertex count.
349 : */
350 :
351 : /**
352 : * \fn double OGRSimpleCurve::getX( int iVertex ) const;
353 : *
354 : * \brief Get X at vertex.
355 : *
356 : * Returns the X value at the indicated vertex. If iVertex is out of range a
357 : * crash may occur, no internal range checking is performed.
358 : *
359 : * @param iVertex the vertex to return, between 0 and getNumPoints()-1.
360 : *
361 : * @return X value.
362 : */
363 :
364 : /**
365 : * \fn double OGRSimpleCurve::getY( int iVertex ) const;
366 : *
367 : * \brief Get Y at vertex.
368 : *
369 : * Returns the Y value at the indicated vertex. If iVertex is out of range a
370 : * crash may occur, no internal range checking is performed.
371 : *
372 : * @param iVertex the vertex to return, between 0 and getNumPoints()-1.
373 : *
374 : * @return X value.
375 : */
376 :
377 : /************************************************************************/
378 : /* getZ() */
379 : /************************************************************************/
380 :
381 : /**
382 : * \brief Get Z at vertex.
383 : *
384 : * Returns the Z (elevation) value at the indicated vertex. If no Z
385 : * value is available, 0.0 is returned. If iVertex is out of range a
386 : * crash may occur, no internal range checking is performed.
387 : *
388 : * @param iVertex the vertex to return, between 0 and getNumPoints()-1.
389 : *
390 : * @return Z value.
391 : */
392 :
393 692261 : double OGRSimpleCurve::getZ(int iVertex) const
394 :
395 : {
396 692261 : if (padfZ != nullptr && iVertex >= 0 && iVertex < nPointCount &&
397 253422 : (flags & OGR_G_3D))
398 253422 : return (padfZ[iVertex]);
399 : else
400 438839 : return 0.0;
401 : }
402 :
403 : /************************************************************************/
404 : /* getM() */
405 : /************************************************************************/
406 :
407 : /**
408 : * \brief Get measure at vertex.
409 : *
410 : * Returns the M (measure) value at the indicated vertex. If no M
411 : * value is available, 0.0 is returned.
412 : *
413 : * @param iVertex the vertex to return, between 0 and getNumPoints()-1.
414 : *
415 : * @return M value.
416 : */
417 :
418 2660 : double OGRSimpleCurve::getM(int iVertex) const
419 :
420 : {
421 2660 : if (padfM != nullptr && iVertex >= 0 && iVertex < nPointCount &&
422 2660 : (flags & OGR_G_MEASURED))
423 2660 : return (padfM[iVertex]);
424 : else
425 0 : return 0.0;
426 : }
427 :
428 : /************************************************************************/
429 : /* setNumPoints() */
430 : /************************************************************************/
431 :
432 : /**
433 : * \brief Set number of points in geometry.
434 : *
435 : * This method primary exists to preset the number of points in a linestring
436 : * geometry before setPoint() is used to assign them to avoid reallocating
437 : * the array larger with each call to addPoint().
438 : *
439 : * This method has no SFCOM analog.
440 : *
441 : * @param nNewPointCount the new number of points for geometry.
442 : * @param bZeroizeNewContent whether to set to zero the new elements of arrays
443 : * that are extended.
444 : * @return (since 3.10) true in case of success, false in case of memory allocation error
445 : */
446 :
447 21591200 : bool OGRSimpleCurve::setNumPoints(int nNewPointCount, int bZeroizeNewContent)
448 :
449 : {
450 21591200 : CPLAssert(nNewPointCount >= 0);
451 :
452 21591200 : if (nNewPointCount > m_nPointCapacity)
453 : {
454 : // Overflow of sizeof(OGRRawPoint) * nNewPointCount can only occur on
455 : // 32 bit, but we don't really want to allocate 2 billion points even on
456 : // 64 bit...
457 8670620 : if (nNewPointCount > std::numeric_limits<int>::max() /
458 : static_cast<int>(sizeof(OGRRawPoint)))
459 : {
460 2 : CPLError(CE_Failure, CPLE_IllegalArg,
461 : "Too many points on line/curve (%d points exceeds the "
462 : "limit of %d points)",
463 : nNewPointCount,
464 0 : std::numeric_limits<int>::max() /
465 : static_cast<int>(sizeof(OGRRawPoint)));
466 2 : return false;
467 : }
468 :
469 : // If first allocation, just aim for nNewPointCount
470 : // Otherwise aim for nNewPointCount + nNewPointCount / 3 to have
471 : // exponential growth.
472 : const int nNewCapacity =
473 13576700 : (nPointCount == 0 ||
474 4906080 : nNewPointCount > std::numeric_limits<int>::max() /
475 4906080 : static_cast<int>(sizeof(OGRRawPoint)) -
476 4906080 : nNewPointCount / 3)
477 8670610 : ? nNewPointCount
478 4906080 : : nNewPointCount + nNewPointCount / 3;
479 :
480 8670610 : if (nPointCount == 0 && paoPoints)
481 : {
482 : // If there was an allocated array, but the old number of points is
483 : // 0, then free the arrays before allocating them, to avoid
484 : // potential costly recopy of useless data.
485 10 : VSIFree(paoPoints);
486 10 : paoPoints = nullptr;
487 10 : VSIFree(padfZ);
488 10 : padfZ = nullptr;
489 10 : VSIFree(padfM);
490 10 : padfM = nullptr;
491 10 : m_nPointCapacity = 0;
492 : }
493 :
494 : OGRRawPoint *paoNewPoints = static_cast<OGRRawPoint *>(
495 8670610 : VSI_REALLOC_VERBOSE(paoPoints, sizeof(OGRRawPoint) * nNewCapacity));
496 8670620 : if (paoNewPoints == nullptr)
497 : {
498 0 : return false;
499 : }
500 8670620 : paoPoints = paoNewPoints;
501 :
502 8670620 : if (flags & OGR_G_3D)
503 : {
504 : double *padfNewZ = static_cast<double *>(
505 3865280 : VSI_REALLOC_VERBOSE(padfZ, sizeof(double) * nNewCapacity));
506 3865280 : if (padfNewZ == nullptr)
507 : {
508 0 : return false;
509 : }
510 3865280 : padfZ = padfNewZ;
511 : }
512 :
513 8670620 : if (flags & OGR_G_MEASURED)
514 : {
515 : double *padfNewM = static_cast<double *>(
516 4522 : VSI_REALLOC_VERBOSE(padfM, sizeof(double) * nNewCapacity));
517 4522 : if (padfNewM == nullptr)
518 : {
519 0 : return false;
520 : }
521 4522 : padfM = padfNewM;
522 : }
523 :
524 8670620 : m_nPointCapacity = nNewCapacity;
525 : }
526 :
527 21591200 : if (nNewPointCount > nPointCount && bZeroizeNewContent)
528 : {
529 : // gcc 8.0 (dev) complains about -Wclass-memaccess since
530 : // OGRRawPoint() has a constructor. So use a void* pointer. Doing
531 : // the memset() here is correct since the constructor sets to 0. We
532 : // could instead use a std::fill(), but at every other place, we
533 : // treat this class as a regular POD (see above use of realloc())
534 19515100 : void *dest = static_cast<void *>(paoPoints + nPointCount);
535 19515100 : memset(dest, 0, sizeof(OGRRawPoint) * (nNewPointCount - nPointCount));
536 :
537 19515100 : if ((flags & OGR_G_3D) && padfZ)
538 5503410 : memset(padfZ + nPointCount, 0,
539 5503410 : sizeof(double) * (nNewPointCount - nPointCount));
540 :
541 19515100 : if ((flags & OGR_G_MEASURED) && padfM)
542 409 : memset(padfM + nPointCount, 0,
543 409 : sizeof(double) * (nNewPointCount - nPointCount));
544 : }
545 :
546 21591200 : nPointCount = nNewPointCount;
547 21591200 : return true;
548 : }
549 :
550 : /************************************************************************/
551 : /* setPoint() */
552 : /************************************************************************/
553 :
554 : /**
555 : * \brief Set the location of a vertex in line string.
556 : *
557 : * If iPoint is larger than the number of necessary the number of existing
558 : * points in the line string, the point count will be increased to
559 : * accommodate the request.
560 : *
561 : * There is no SFCOM analog to this method.
562 : *
563 : * @param iPoint the index of the vertex to assign (zero based).
564 : * @param poPoint the value to assign to the vertex.
565 : * @return (since 3.10) true in case of success, false in case of memory allocation error
566 : */
567 :
568 1605 : bool OGRSimpleCurve::setPoint(int iPoint, OGRPoint *poPoint)
569 :
570 : {
571 1605 : if ((flags & OGR_G_3D) && (flags & OGR_G_MEASURED))
572 59 : return setPoint(iPoint, poPoint->getX(), poPoint->getY(),
573 59 : poPoint->getZ(), poPoint->getM());
574 1546 : else if (flags & OGR_G_3D)
575 863 : return setPoint(iPoint, poPoint->getX(), poPoint->getY(),
576 863 : poPoint->getZ());
577 683 : else if (flags & OGR_G_MEASURED)
578 3 : return setPointM(iPoint, poPoint->getX(), poPoint->getY(),
579 3 : poPoint->getM());
580 : else
581 680 : return setPoint(iPoint, poPoint->getX(), poPoint->getY());
582 : }
583 :
584 : /************************************************************************/
585 : /* CheckPointCount() */
586 : /************************************************************************/
587 :
588 19495300 : static inline bool CheckPointCount(int iPoint)
589 : {
590 19495300 : if (iPoint == std::numeric_limits<int>::max())
591 : {
592 1 : CPLError(CE_Failure, CPLE_AppDefined, "Too big point count.");
593 1 : return false;
594 : }
595 19495300 : return true;
596 : }
597 :
598 : /************************************************************************/
599 : /* setPoint() */
600 : /************************************************************************/
601 :
602 : /**
603 : * \brief Set the location of a vertex in line string.
604 : *
605 : * If iPoint is larger than the number of necessary the number of existing
606 : * points in the line string, the point count will be increased to
607 : * accommodate the request.
608 : *
609 : * There is no SFCOM analog to this method.
610 : *
611 : * @param iPoint the index of the vertex to assign (zero based).
612 : * @param xIn input X coordinate to assign.
613 : * @param yIn input Y coordinate to assign.
614 : * @param zIn input Z coordinate to assign (defaults to zero).
615 : * @return (since 3.10) true in case of success, false in case of memory allocation error
616 : */
617 :
618 5562730 : bool OGRSimpleCurve::setPoint(int iPoint, double xIn, double yIn, double zIn)
619 :
620 : {
621 5562730 : if (!(flags & OGR_G_3D))
622 : {
623 1263480 : if (!Make3D())
624 0 : return false;
625 : }
626 :
627 5562730 : if (iPoint >= nPointCount)
628 : {
629 5503150 : if (!CheckPointCount(iPoint) || !setNumPoints(iPoint + 1))
630 1 : return false;
631 : }
632 : #ifdef DEBUG
633 5562730 : if (paoPoints == nullptr)
634 0 : return false;
635 : #endif
636 :
637 5562730 : paoPoints[iPoint].x = xIn;
638 5562730 : paoPoints[iPoint].y = yIn;
639 :
640 5562730 : if (padfZ != nullptr)
641 : {
642 5562730 : padfZ[iPoint] = zIn;
643 : }
644 5562730 : return true;
645 : }
646 :
647 : /**
648 : * \brief Set the location of a vertex in line string.
649 : *
650 : * If iPoint is larger than the number of necessary the number of existing
651 : * points in the line string, the point count will be increased to
652 : * accommodate the request.
653 : *
654 : * There is no SFCOM analog to this method.
655 : *
656 : * @param iPoint the index of the vertex to assign (zero based).
657 : * @param xIn input X coordinate to assign.
658 : * @param yIn input Y coordinate to assign.
659 : * @param mIn input M coordinate to assign (defaults to zero).
660 : * @return (since 3.10) true in case of success, false in case of memory allocation error
661 : */
662 :
663 443 : bool OGRSimpleCurve::setPointM(int iPoint, double xIn, double yIn, double mIn)
664 :
665 : {
666 443 : if (!(flags & OGR_G_MEASURED))
667 : {
668 108 : if (!AddM())
669 0 : return false;
670 : }
671 :
672 443 : if (iPoint >= nPointCount)
673 : {
674 6 : if (!CheckPointCount(iPoint) || !setNumPoints(iPoint + 1))
675 0 : return false;
676 : }
677 : #ifdef DEBUG
678 443 : if (paoPoints == nullptr)
679 0 : return false;
680 : #endif
681 :
682 443 : paoPoints[iPoint].x = xIn;
683 443 : paoPoints[iPoint].y = yIn;
684 :
685 443 : if (padfM != nullptr)
686 : {
687 443 : padfM[iPoint] = mIn;
688 : }
689 443 : return true;
690 : }
691 :
692 : /**
693 : * \brief Set the location of a vertex in line string.
694 : *
695 : * If iPoint is larger than the number of necessary the number of existing
696 : * points in the line string, the point count will be increased to
697 : * accommodate the request.
698 : *
699 : * There is no SFCOM analog to this method.
700 : *
701 : * @param iPoint the index of the vertex to assign (zero based).
702 : * @param xIn input X coordinate to assign.
703 : * @param yIn input Y coordinate to assign.
704 : * @param zIn input Z coordinate to assign (defaults to zero).
705 : * @param mIn input M coordinate to assign (defaults to zero).
706 : * @return (since 3.10) true in case of success, false in case of memory allocation error
707 : */
708 :
709 852 : bool OGRSimpleCurve::setPoint(int iPoint, double xIn, double yIn, double zIn,
710 : double mIn)
711 :
712 : {
713 852 : if (!(flags & OGR_G_3D))
714 : {
715 271 : if (!Make3D())
716 0 : return false;
717 : }
718 852 : if (!(flags & OGR_G_MEASURED))
719 : {
720 284 : if (!AddM())
721 0 : return false;
722 : }
723 :
724 852 : if (iPoint >= nPointCount)
725 : {
726 261 : if (!CheckPointCount(iPoint) || !setNumPoints(iPoint + 1))
727 0 : return false;
728 : }
729 : #ifdef DEBUG
730 852 : if (paoPoints == nullptr)
731 0 : return false;
732 : #endif
733 :
734 852 : paoPoints[iPoint].x = xIn;
735 852 : paoPoints[iPoint].y = yIn;
736 :
737 852 : if (padfZ != nullptr)
738 : {
739 852 : padfZ[iPoint] = zIn;
740 : }
741 852 : if (padfM != nullptr)
742 : {
743 852 : padfM[iPoint] = mIn;
744 : }
745 852 : return true;
746 : }
747 :
748 : /**
749 : * \brief Set the location of a vertex in line string.
750 : *
751 : * If iPoint is larger than the number of necessary the number of existing
752 : * points in the line string, the point count will be increased to
753 : * accommodate the request.
754 : *
755 : * There is no SFCOM analog to this method.
756 : *
757 : * @param iPoint the index of the vertex to assign (zero based).
758 : * @param xIn input X coordinate to assign.
759 : * @param yIn input Y coordinate to assign.
760 : * @return (since 3.10) true in case of success, false in case of memory allocation error
761 : */
762 :
763 27683000 : bool OGRSimpleCurve::setPoint(int iPoint, double xIn, double yIn)
764 :
765 : {
766 27683000 : if (iPoint >= nPointCount)
767 : {
768 13991900 : if (!CheckPointCount(iPoint) || !setNumPoints(iPoint + 1) || !paoPoints)
769 2 : return false;
770 : }
771 :
772 27683000 : paoPoints[iPoint].x = xIn;
773 27683000 : paoPoints[iPoint].y = yIn;
774 27683000 : return true;
775 : }
776 :
777 : /************************************************************************/
778 : /* setZ() */
779 : /************************************************************************/
780 :
781 : /**
782 : * \brief Set the Z of a vertex in line string.
783 : *
784 : * If iPoint is larger than the number of necessary the number of existing
785 : * points in the line string, the point count will be increased to
786 : * accommodate the request.
787 : *
788 : * There is no SFCOM analog to this method.
789 : *
790 : * @param iPoint the index of the vertex to assign (zero based).
791 : * @param zIn input Z coordinate to assign.
792 : * @return (since 3.10) true in case of success, false in case of memory allocation error
793 : */
794 :
795 7010 : bool OGRSimpleCurve::setZ(int iPoint, double zIn)
796 : {
797 7010 : if (getCoordinateDimension() == 2)
798 : {
799 840 : if (!Make3D())
800 0 : return false;
801 : }
802 :
803 7010 : if (iPoint >= nPointCount)
804 : {
805 0 : if (!CheckPointCount(iPoint) || !setNumPoints(iPoint + 1))
806 0 : return false;
807 : }
808 :
809 7010 : if (padfZ != nullptr)
810 7010 : padfZ[iPoint] = zIn;
811 7010 : return true;
812 : }
813 :
814 : /************************************************************************/
815 : /* setM() */
816 : /************************************************************************/
817 :
818 : /**
819 : * \brief Set the M of a vertex in line string.
820 : *
821 : * If iPoint is larger than the number of necessary the number of existing
822 : * points in the line string, the point count will be increased to
823 : * accommodate the request.
824 : *
825 : * There is no SFCOM analog to this method.
826 : *
827 : * @param iPoint the index of the vertex to assign (zero based).
828 : * @param mIn input M coordinate to assign.
829 : * @return (since 3.10) true in case of success, false in case of memory allocation error
830 : */
831 :
832 835 : bool OGRSimpleCurve::setM(int iPoint, double mIn)
833 : {
834 835 : if (!(flags & OGR_G_MEASURED))
835 : {
836 103 : if (!AddM())
837 0 : return false;
838 : }
839 :
840 835 : if (iPoint >= nPointCount)
841 : {
842 0 : if (!CheckPointCount(iPoint) || !setNumPoints(iPoint + 1))
843 0 : return false;
844 : }
845 :
846 835 : if (padfM != nullptr)
847 835 : padfM[iPoint] = mIn;
848 835 : return true;
849 : }
850 :
851 : /************************************************************************/
852 : /* addPoint() */
853 : /************************************************************************/
854 :
855 : /**
856 : * \brief Add a point to a line string.
857 : *
858 : * The vertex count of the line string is increased by one, and assigned from
859 : * the passed location value.
860 : *
861 : * There is no SFCOM analog to this method.
862 : *
863 : * @param poPoint the point to assign to the new vertex.
864 : * @return (since 3.10) true in case of success, false in case of memory allocation error
865 : */
866 :
867 5021160 : bool OGRSimpleCurve::addPoint(const OGRPoint *poPoint)
868 :
869 : {
870 5021160 : if (poPoint->Is3D() && poPoint->IsMeasured())
871 250 : return setPoint(nPointCount, poPoint->getX(), poPoint->getY(),
872 250 : poPoint->getZ(), poPoint->getM());
873 5020910 : else if (poPoint->Is3D())
874 4986920 : return setPoint(nPointCount, poPoint->getX(), poPoint->getY(),
875 4986920 : poPoint->getZ());
876 33994 : else if (poPoint->IsMeasured())
877 0 : return setPointM(nPointCount, poPoint->getX(), poPoint->getY(),
878 0 : poPoint->getM());
879 : else
880 33995 : return setPoint(nPointCount, poPoint->getX(), poPoint->getY());
881 : }
882 :
883 : /************************************************************************/
884 : /* addPoint() */
885 : /************************************************************************/
886 :
887 : /**
888 : * \brief Add a point to a line string.
889 : *
890 : * The vertex count of the line string is increased by one, and assigned from
891 : * the passed location value.
892 : *
893 : * There is no SFCOM analog to this method.
894 : *
895 : * @param x the X coordinate to assign to the new point.
896 : * @param y the Y coordinate to assign to the new point.
897 : * @param z the Z coordinate to assign to the new point (defaults to zero).
898 : * @param m the M coordinate to assign to the new point (defaults to zero).
899 : * @return (since 3.10) true in case of success, false in case of memory allocation error
900 : */
901 :
902 11 : bool OGRSimpleCurve::addPoint(double x, double y, double z, double m)
903 :
904 : {
905 11 : return setPoint(nPointCount, x, y, z, m);
906 : }
907 :
908 : /**
909 : * \brief Add a point to a line string.
910 : *
911 : * The vertex count of the line string is increased by one, and assigned from
912 : * the passed location value.
913 : *
914 : * There is no SFCOM analog to this method.
915 : *
916 : * @param x the X coordinate to assign to the new point.
917 : * @param y the Y coordinate to assign to the new point.
918 : * @param z the Z coordinate to assign to the new point (defaults to zero).
919 : * @return (since 3.10) true in case of success, false in case of memory allocation error
920 : */
921 :
922 507151 : bool OGRSimpleCurve::addPoint(double x, double y, double z)
923 :
924 : {
925 507151 : return setPoint(nPointCount, x, y, z);
926 : }
927 :
928 : /**
929 : * \brief Add a point to a line string.
930 : *
931 : * The vertex count of the line string is increased by one, and assigned from
932 : * the passed location value.
933 : *
934 : * There is no SFCOM analog to this method.
935 : *
936 : * @param x the X coordinate to assign to the new point.
937 : * @param y the Y coordinate to assign to the new point.
938 : * @return (since 3.10) true in case of success, false in case of memory allocation error
939 : */
940 :
941 13896700 : bool OGRSimpleCurve::addPoint(double x, double y)
942 :
943 : {
944 13896700 : return setPoint(nPointCount, x, y);
945 : }
946 :
947 : /**
948 : * \brief Add a point to a line string.
949 : *
950 : * The vertex count of the line string is increased by one, and assigned from
951 : * the passed location value.
952 : *
953 : * There is no SFCOM analog to this method.
954 : *
955 : * @param x the X coordinate to assign to the new point.
956 : * @param y the Y coordinate to assign to the new point.
957 : * @param m the M coordinate to assign to the new point.
958 : * @return (since 3.10) true in case of success, false in case of memory allocation error
959 : */
960 :
961 6 : bool OGRSimpleCurve::addPointM(double x, double y, double m)
962 :
963 : {
964 6 : return setPointM(nPointCount, x, y, m);
965 : }
966 :
967 : /************************************************************************/
968 : /* removePoint() */
969 : /************************************************************************/
970 :
971 : /**
972 : * \brief Remove a point from a line string.
973 : *
974 : * There is no SFCOM analog to this method.
975 : *
976 : * @param nIndex Point index
977 : * @since GDAL 3.3
978 : */
979 :
980 15 : bool OGRSimpleCurve::removePoint(int nIndex)
981 : {
982 15 : if (nIndex < 0 || nIndex >= nPointCount)
983 4 : return false;
984 11 : if (nIndex < nPointCount - 1)
985 : {
986 7 : memmove(paoPoints + nIndex, paoPoints + nIndex + 1,
987 7 : sizeof(OGRRawPoint) * (nPointCount - 1 - nIndex));
988 7 : if (padfZ)
989 : {
990 1 : memmove(padfZ + nIndex, padfZ + nIndex + 1,
991 1 : sizeof(double) * (nPointCount - 1 - nIndex));
992 : }
993 7 : if (padfM)
994 : {
995 1 : memmove(padfM + nIndex, padfM + nIndex + 1,
996 1 : sizeof(double) * (nPointCount - 1 - nIndex));
997 : }
998 : }
999 11 : nPointCount--;
1000 11 : return true;
1001 : }
1002 :
1003 : /************************************************************************/
1004 : /* setPointsM() */
1005 : /************************************************************************/
1006 :
1007 : /**
1008 : * \brief Assign all points in a line string.
1009 : *
1010 : * This method clears any existing points assigned to this line string,
1011 : * and assigns a whole new set. It is the most efficient way of assigning
1012 : * the value of a line string.
1013 : *
1014 : * There is no SFCOM analog to this method.
1015 : *
1016 : * @param nPointsIn number of points being passed in paoPointsIn
1017 : * @param paoPointsIn list of points being assigned.
1018 : * @param padfMIn the M values that go with the points.
1019 : * @return (since 3.10) true in case of success, false in case of memory allocation error
1020 : */
1021 :
1022 190 : bool OGRSimpleCurve::setPointsM(int nPointsIn, const OGRRawPoint *paoPointsIn,
1023 : const double *padfMIn)
1024 :
1025 : {
1026 190 : if (!setNumPoints(nPointsIn, FALSE)
1027 : #ifdef DEBUG
1028 190 : || paoPoints == nullptr
1029 : #endif
1030 : )
1031 0 : return false;
1032 :
1033 190 : if (nPointsIn)
1034 190 : memcpy(paoPoints, paoPointsIn, sizeof(OGRRawPoint) * nPointsIn);
1035 :
1036 : /* -------------------------------------------------------------------- */
1037 : /* Check measures. */
1038 : /* -------------------------------------------------------------------- */
1039 190 : if (padfMIn == nullptr && (flags & OGR_G_MEASURED))
1040 : {
1041 0 : RemoveM();
1042 : }
1043 190 : else if (padfMIn)
1044 : {
1045 190 : if (!AddM())
1046 0 : return false;
1047 190 : if (padfM && nPointsIn)
1048 190 : memcpy(padfM, padfMIn, sizeof(double) * nPointsIn);
1049 : }
1050 190 : return true;
1051 : }
1052 :
1053 : /************************************************************************/
1054 : /* setPoints() */
1055 : /************************************************************************/
1056 :
1057 : /**
1058 : * \brief Assign all points in a line string.
1059 : *
1060 : * This method clears any existing points assigned to this line string,
1061 : * and assigns a whole new set. It is the most efficient way of assigning
1062 : * the value of a line string.
1063 : *
1064 : * There is no SFCOM analog to this method.
1065 : *
1066 : * @param nPointsIn number of points being passed in paoPointsIn
1067 : * @param paoPointsIn list of points being assigned.
1068 : * @param padfZIn the Z values that go with the points.
1069 : * @param padfMIn the M values that go with the points.
1070 : * @return (since 3.10) true in case of success, false in case of memory allocation error
1071 : */
1072 :
1073 1900850 : bool OGRSimpleCurve::setPoints(int nPointsIn, const OGRRawPoint *paoPointsIn,
1074 : const double *padfZIn, const double *padfMIn)
1075 :
1076 : {
1077 1900850 : if (!setNumPoints(nPointsIn, FALSE)
1078 : #ifdef DEBUG
1079 1900850 : || paoPoints == nullptr
1080 : #endif
1081 : )
1082 89 : return false;
1083 :
1084 1900760 : if (nPointsIn)
1085 1900760 : memcpy(paoPoints, paoPointsIn, sizeof(OGRRawPoint) * nPointsIn);
1086 :
1087 : /* -------------------------------------------------------------------- */
1088 : /* Check 2D/3D. */
1089 : /* -------------------------------------------------------------------- */
1090 1900760 : if (padfZIn == nullptr && getCoordinateDimension() > 2)
1091 : {
1092 0 : Make2D();
1093 : }
1094 1900760 : else if (padfZIn)
1095 : {
1096 1352750 : if (!Make3D())
1097 0 : return false;
1098 1352750 : if (padfZ && nPointsIn)
1099 1352750 : memcpy(padfZ, padfZIn, sizeof(double) * nPointsIn);
1100 : }
1101 :
1102 : /* -------------------------------------------------------------------- */
1103 : /* Check measures. */
1104 : /* -------------------------------------------------------------------- */
1105 1900760 : if (padfMIn == nullptr && (flags & OGR_G_MEASURED))
1106 : {
1107 0 : RemoveM();
1108 : }
1109 1900760 : else if (padfMIn)
1110 : {
1111 804 : if (!AddM())
1112 0 : return false;
1113 804 : if (padfM && nPointsIn)
1114 804 : memcpy(padfM, padfMIn, sizeof(double) * nPointsIn);
1115 : }
1116 1900760 : return true;
1117 : }
1118 :
1119 : /************************************************************************/
1120 : /* setPoints() */
1121 : /************************************************************************/
1122 :
1123 : /**
1124 : * \brief Assign all points in a line string.
1125 : *
1126 : * This method clears any existing points assigned to this line string,
1127 : * and assigns a whole new set. It is the most efficient way of assigning
1128 : * the value of a line string.
1129 : *
1130 : * There is no SFCOM analog to this method.
1131 : *
1132 : * @param nPointsIn number of points being passed in paoPointsIn
1133 : * @param paoPointsIn list of points being assigned.
1134 : * @param padfZIn the Z values that go with the points (optional, may be NULL).
1135 : * @return (since 3.10) true in case of success, false in case of memory allocation error
1136 : */
1137 :
1138 20900 : bool OGRSimpleCurve::setPoints(int nPointsIn, const OGRRawPoint *paoPointsIn,
1139 : const double *padfZIn)
1140 :
1141 : {
1142 20900 : if (!setNumPoints(nPointsIn, FALSE)
1143 : #ifdef DEBUG
1144 20900 : || paoPoints == nullptr
1145 : #endif
1146 : )
1147 7 : return false;
1148 :
1149 20893 : if (nPointsIn)
1150 20893 : memcpy(paoPoints, paoPointsIn, sizeof(OGRRawPoint) * nPointsIn);
1151 :
1152 : /* -------------------------------------------------------------------- */
1153 : /* Check 2D/3D. */
1154 : /* -------------------------------------------------------------------- */
1155 20893 : if (padfZIn == nullptr && getCoordinateDimension() > 2)
1156 : {
1157 0 : Make2D();
1158 : }
1159 20893 : else if (padfZIn)
1160 : {
1161 1405 : if (!Make3D())
1162 0 : return false;
1163 1405 : if (padfZ && nPointsIn)
1164 1405 : memcpy(padfZ, padfZIn, sizeof(double) * nPointsIn);
1165 : }
1166 20893 : return true;
1167 : }
1168 :
1169 : /************************************************************************/
1170 : /* setPoints() */
1171 : /************************************************************************/
1172 :
1173 : /**
1174 : * \brief Assign all points in a line string.
1175 : *
1176 : * This method clear any existing points assigned to this line string,
1177 : * and assigns a whole new set.
1178 : *
1179 : * There is no SFCOM analog to this method.
1180 : *
1181 : * @param nPointsIn number of points being passed in padfX and padfY.
1182 : * @param padfX list of X coordinates of points being assigned.
1183 : * @param padfY list of Y coordinates of points being assigned.
1184 : * @param padfZIn list of Z coordinates of points being assigned (defaults to
1185 : * NULL for 2D objects).
1186 : * @return (since 3.10) true in case of success, false in case of memory allocation error
1187 : */
1188 :
1189 39410 : bool OGRSimpleCurve::setPoints(int nPointsIn, const double *padfX,
1190 : const double *padfY, const double *padfZIn)
1191 :
1192 : {
1193 : /* -------------------------------------------------------------------- */
1194 : /* Check 2D/3D. */
1195 : /* -------------------------------------------------------------------- */
1196 39410 : if (padfZIn == nullptr)
1197 36299 : Make2D();
1198 : else
1199 : {
1200 3111 : if (!Make3D())
1201 0 : return false;
1202 : }
1203 :
1204 : /* -------------------------------------------------------------------- */
1205 : /* Assign values. */
1206 : /* -------------------------------------------------------------------- */
1207 39410 : if (!setNumPoints(nPointsIn, FALSE))
1208 0 : return false;
1209 :
1210 452137 : for (int i = 0; i < nPointsIn; i++)
1211 : {
1212 412727 : paoPoints[i].x = padfX[i];
1213 412727 : paoPoints[i].y = padfY[i];
1214 : }
1215 :
1216 39410 : if (padfZ && padfZIn && nPointsIn)
1217 : {
1218 3111 : memcpy(padfZ, padfZIn, sizeof(double) * nPointsIn);
1219 : }
1220 39410 : return true;
1221 : }
1222 :
1223 : /************************************************************************/
1224 : /* setPointsM() */
1225 : /************************************************************************/
1226 :
1227 : /**
1228 : * \brief Assign all points in a line string.
1229 : *
1230 : * This method clear any existing points assigned to this line string,
1231 : * and assigns a whole new set.
1232 : *
1233 : * There is no SFCOM analog to this method.
1234 : *
1235 : * @param nPointsIn number of points being passed in padfX and padfY.
1236 : * @param padfX list of X coordinates of points being assigned.
1237 : * @param padfY list of Y coordinates of points being assigned.
1238 : * @param padfMIn list of M coordinates of points being assigned.
1239 : * @return (since 3.10) true in case of success, false in case of memory allocation error
1240 : */
1241 :
1242 22 : bool OGRSimpleCurve::setPointsM(int nPointsIn, const double *padfX,
1243 : const double *padfY, const double *padfMIn)
1244 :
1245 : {
1246 : /* -------------------------------------------------------------------- */
1247 : /* Check 2D/3D. */
1248 : /* -------------------------------------------------------------------- */
1249 22 : if (padfMIn == nullptr)
1250 4 : RemoveM();
1251 : else
1252 : {
1253 18 : if (!AddM())
1254 0 : return false;
1255 : }
1256 :
1257 : /* -------------------------------------------------------------------- */
1258 : /* Assign values. */
1259 : /* -------------------------------------------------------------------- */
1260 22 : if (!setNumPoints(nPointsIn, FALSE))
1261 0 : return false;
1262 :
1263 90 : for (int i = 0; i < nPointsIn; i++)
1264 : {
1265 68 : paoPoints[i].x = padfX[i];
1266 68 : paoPoints[i].y = padfY[i];
1267 : }
1268 :
1269 22 : if (padfMIn && padfM && nPointsIn)
1270 : {
1271 18 : memcpy(padfM, padfMIn, sizeof(double) * nPointsIn);
1272 : }
1273 22 : return true;
1274 : }
1275 :
1276 : /************************************************************************/
1277 : /* setPoints() */
1278 : /************************************************************************/
1279 :
1280 : /**
1281 : * \brief Assign all points in a line string.
1282 : *
1283 : * This method clear any existing points assigned to this line string,
1284 : * and assigns a whole new set.
1285 : *
1286 : * There is no SFCOM analog to this method.
1287 : *
1288 : * @param nPointsIn number of points being passed in padfX and padfY.
1289 : * @param padfX list of X coordinates of points being assigned.
1290 : * @param padfY list of Y coordinates of points being assigned.
1291 : * @param padfZIn list of Z coordinates of points being assigned.
1292 : * @param padfMIn list of M coordinates of points being assigned.
1293 : * @return (since 3.10) true in case of success, false in case of memory allocation error
1294 : */
1295 :
1296 3504 : bool OGRSimpleCurve::setPoints(int nPointsIn, const double *padfX,
1297 : const double *padfY, const double *padfZIn,
1298 : const double *padfMIn)
1299 :
1300 : {
1301 : /* -------------------------------------------------------------------- */
1302 : /* Check 2D/3D. */
1303 : /* -------------------------------------------------------------------- */
1304 3504 : if (padfZIn == nullptr)
1305 66 : Make2D();
1306 : else
1307 : {
1308 3438 : if (!Make3D())
1309 0 : return false;
1310 : }
1311 :
1312 : /* -------------------------------------------------------------------- */
1313 : /* Check measures. */
1314 : /* -------------------------------------------------------------------- */
1315 3504 : if (padfMIn == nullptr)
1316 3476 : RemoveM();
1317 : else
1318 : {
1319 28 : if (!AddM())
1320 0 : return false;
1321 : }
1322 :
1323 : /* -------------------------------------------------------------------- */
1324 : /* Assign values. */
1325 : /* -------------------------------------------------------------------- */
1326 3504 : if (!setNumPoints(nPointsIn, FALSE))
1327 0 : return false;
1328 :
1329 66284 : for (int i = 0; i < nPointsIn; i++)
1330 : {
1331 62780 : paoPoints[i].x = padfX[i];
1332 62780 : paoPoints[i].y = padfY[i];
1333 : }
1334 :
1335 3504 : if (padfZ != nullptr && padfZIn && nPointsIn)
1336 3438 : memcpy(padfZ, padfZIn, sizeof(double) * nPointsIn);
1337 3504 : if (padfM != nullptr && padfMIn && nPointsIn)
1338 28 : memcpy(padfM, padfMIn, sizeof(double) * nPointsIn);
1339 3504 : return true;
1340 : }
1341 :
1342 : /************************************************************************/
1343 : /* getPoints() */
1344 : /************************************************************************/
1345 :
1346 : /**
1347 : * \brief Returns all points of line string.
1348 : *
1349 : * This method copies all points into user list. This list must be at
1350 : * least sizeof(OGRRawPoint) * OGRGeometry::getNumPoints() byte in size.
1351 : * It also copies all Z coordinates.
1352 : *
1353 : * There is no SFCOM analog to this method.
1354 : *
1355 : * @param paoPointsOut a buffer into which the points is written.
1356 : * @param padfZOut the Z values that go with the points (optional, may be NULL).
1357 : */
1358 :
1359 730 : void OGRSimpleCurve::getPoints(OGRRawPoint *paoPointsOut,
1360 : double *padfZOut) const
1361 : {
1362 730 : if (!paoPointsOut || nPointCount == 0)
1363 3 : return;
1364 :
1365 727 : memcpy(paoPointsOut, paoPoints, sizeof(OGRRawPoint) * nPointCount);
1366 :
1367 : /* -------------------------------------------------------------------- */
1368 : /* Check 2D/3D. */
1369 : /* -------------------------------------------------------------------- */
1370 727 : if (padfZOut)
1371 : {
1372 198 : if (padfZ)
1373 196 : memcpy(padfZOut, padfZ, sizeof(double) * nPointCount);
1374 : else
1375 2 : memset(padfZOut, 0, sizeof(double) * nPointCount);
1376 : }
1377 : }
1378 :
1379 : /**
1380 : * \brief Returns all points of line string.
1381 : *
1382 : * This method copies all points into user arrays. The user provides the
1383 : * stride between 2 consecutive elements of the array.
1384 : *
1385 : * On some CPU architectures, care must be taken so that the arrays are properly
1386 : * aligned.
1387 : *
1388 : * There is no SFCOM analog to this method.
1389 : *
1390 : * @param pabyX a buffer of at least (nXStride * nPointCount) bytes, may be
1391 : * NULL.
1392 : * @param nXStride the number of bytes between 2 elements of pabyX.
1393 : * @param pabyY a buffer of at least (nYStride * nPointCount) bytes, may be
1394 : * NULL.
1395 : * @param nYStride the number of bytes between 2 elements of pabyY.
1396 : * @param pabyZ a buffer of at last size (nZStride * nPointCount) bytes, may be
1397 : * NULL.
1398 : * @param nZStride the number of bytes between 2 elements of pabyZ.
1399 : * @param pabyM a buffer of at last size (nMStride * nPointCount) bytes, may be
1400 : * NULL.
1401 : * @param nMStride the number of bytes between 2 elements of pabyM.
1402 : *
1403 : * @since OGR 2.1.0
1404 : */
1405 :
1406 186 : void OGRSimpleCurve::getPoints(void *pabyX, int nXStride, void *pabyY,
1407 : int nYStride, void *pabyZ, int nZStride,
1408 : void *pabyM, int nMStride) const
1409 : {
1410 186 : if (pabyX != nullptr && nXStride == 0)
1411 0 : return;
1412 186 : if (pabyY != nullptr && nYStride == 0)
1413 0 : return;
1414 186 : if (pabyZ != nullptr && nZStride == 0)
1415 0 : return;
1416 186 : if (pabyM != nullptr && nMStride == 0)
1417 0 : return;
1418 186 : if (nXStride == sizeof(OGRRawPoint) && nYStride == sizeof(OGRRawPoint) &&
1419 : static_cast<char *>(pabyY) ==
1420 186 : static_cast<char *>(pabyX) + sizeof(double) &&
1421 77 : (pabyZ == nullptr || nZStride == sizeof(double)))
1422 : {
1423 186 : getPoints(static_cast<OGRRawPoint *>(pabyX),
1424 : static_cast<double *>(pabyZ));
1425 : }
1426 : else
1427 : {
1428 0 : for (int i = 0; i < nPointCount; i++)
1429 : {
1430 0 : if (pabyX)
1431 0 : *reinterpret_cast<double *>(static_cast<char *>(pabyX) +
1432 0 : i * nXStride) = paoPoints[i].x;
1433 0 : if (pabyY)
1434 0 : *reinterpret_cast<double *>(static_cast<char *>(pabyY) +
1435 0 : i * nYStride) = paoPoints[i].y;
1436 : }
1437 :
1438 0 : if (pabyZ)
1439 : {
1440 0 : if (nZStride == sizeof(double))
1441 : {
1442 0 : if (padfZ)
1443 0 : memcpy(pabyZ, padfZ, sizeof(double) * nPointCount);
1444 : else
1445 0 : memset(pabyZ, 0, sizeof(double) * nPointCount);
1446 : }
1447 : else
1448 : {
1449 0 : for (int i = 0; i < nPointCount; i++)
1450 : {
1451 0 : *reinterpret_cast<double *>(static_cast<char *>(pabyZ) +
1452 0 : i * nZStride) =
1453 0 : (padfZ) ? padfZ[i] : 0.0;
1454 : }
1455 : }
1456 : }
1457 : }
1458 186 : if (pabyM)
1459 : {
1460 58 : if (nMStride == sizeof(double))
1461 : {
1462 58 : if (padfM)
1463 58 : memcpy(pabyM, padfM, sizeof(double) * nPointCount);
1464 : else
1465 0 : memset(pabyM, 0, sizeof(double) * nPointCount);
1466 : }
1467 : else
1468 : {
1469 0 : for (int i = 0; i < nPointCount; i++)
1470 : {
1471 0 : *reinterpret_cast<double *>(static_cast<char *>(pabyM) +
1472 0 : i * nMStride) =
1473 0 : (padfM) ? padfM[i] : 0.0;
1474 : }
1475 : }
1476 : }
1477 : }
1478 :
1479 : /************************************************************************/
1480 : /* reversePoints() */
1481 : /************************************************************************/
1482 :
1483 : /**
1484 : * \brief Reverse point order.
1485 : *
1486 : * This method updates the points in this line string in place
1487 : * reversing the point ordering (first for last, etc).
1488 : */
1489 :
1490 2440 : void OGRSimpleCurve::reversePoints()
1491 :
1492 : {
1493 60029 : for (int i = 0; i < nPointCount / 2; i++)
1494 : {
1495 57589 : std::swap(paoPoints[i], paoPoints[nPointCount - i - 1]);
1496 57589 : if (padfZ)
1497 : {
1498 2401 : std::swap(padfZ[i], padfZ[nPointCount - i - 1]);
1499 : }
1500 :
1501 57589 : if (padfM)
1502 : {
1503 6 : std::swap(padfM[i], padfM[nPointCount - i - 1]);
1504 : }
1505 : }
1506 2440 : }
1507 :
1508 : /************************************************************************/
1509 : /* addSubLineString() */
1510 : /************************************************************************/
1511 :
1512 : /**
1513 : * \brief Add a segment of another linestring to this one.
1514 : *
1515 : * Adds the request range of vertices to the end of this line string
1516 : * in an efficient manner. If the nStartVertex is larger than the
1517 : * nEndVertex then the vertices will be reversed as they are copied.
1518 : *
1519 : * @param poOtherLine the other OGRLineString.
1520 : * @param nStartVertex the first vertex to copy, defaults to 0 to start
1521 : * with the first vertex in the other linestring.
1522 : * @param nEndVertex the last vertex to copy, defaults to -1 indicating
1523 : * the last vertex of the other line string.
1524 : */
1525 :
1526 9256 : void OGRSimpleCurve::addSubLineString(const OGRLineString *poOtherLine,
1527 : int nStartVertex, int nEndVertex)
1528 :
1529 : {
1530 9256 : int nOtherLineNumPoints = poOtherLine->getNumPoints();
1531 9256 : if (nOtherLineNumPoints == 0)
1532 0 : return;
1533 :
1534 : /* -------------------------------------------------------------------- */
1535 : /* Do a bit of argument defaulting and validation. */
1536 : /* -------------------------------------------------------------------- */
1537 9256 : if (nEndVertex == -1)
1538 6476 : nEndVertex = nOtherLineNumPoints - 1;
1539 :
1540 9256 : if (nStartVertex < 0 || nEndVertex < 0 ||
1541 9256 : nStartVertex >= nOtherLineNumPoints ||
1542 : nEndVertex >= nOtherLineNumPoints)
1543 : {
1544 0 : CPLAssert(false);
1545 : return;
1546 : }
1547 :
1548 : /* -------------------------------------------------------------------- */
1549 : /* Grow this linestring to hold the additional points. */
1550 : /* -------------------------------------------------------------------- */
1551 9256 : int nOldPoints = nPointCount;
1552 9256 : int nPointsToAdd = std::abs(nEndVertex - nStartVertex) + 1;
1553 :
1554 9256 : if (!setNumPoints(nPointsToAdd + nOldPoints, FALSE)
1555 : #ifdef DEBUG
1556 9256 : || paoPoints == nullptr
1557 : #endif
1558 : )
1559 0 : return;
1560 :
1561 : /* -------------------------------------------------------------------- */
1562 : /* Copy the x/y points - forward copies use memcpy. */
1563 : /* -------------------------------------------------------------------- */
1564 9256 : if (nEndVertex >= nStartVertex)
1565 : {
1566 9065 : memcpy(paoPoints + nOldPoints, poOtherLine->paoPoints + nStartVertex,
1567 9065 : sizeof(OGRRawPoint) * nPointsToAdd);
1568 9065 : if (poOtherLine->padfZ != nullptr)
1569 : {
1570 1171 : Make3D();
1571 1171 : if (padfZ != nullptr)
1572 : {
1573 1171 : memcpy(padfZ + nOldPoints, poOtherLine->padfZ + nStartVertex,
1574 1171 : sizeof(double) * nPointsToAdd);
1575 : }
1576 : }
1577 9065 : if (poOtherLine->padfM != nullptr)
1578 : {
1579 92 : AddM();
1580 92 : if (padfM != nullptr)
1581 : {
1582 92 : memcpy(padfM + nOldPoints, poOtherLine->padfM + nStartVertex,
1583 92 : sizeof(double) * nPointsToAdd);
1584 : }
1585 : }
1586 : }
1587 :
1588 : /* -------------------------------------------------------------------- */
1589 : /* Copy the x/y points - reverse copies done double by double. */
1590 : /* -------------------------------------------------------------------- */
1591 : else
1592 : {
1593 1480 : for (int i = 0; i < nPointsToAdd; i++)
1594 : {
1595 1289 : paoPoints[i + nOldPoints].x =
1596 1289 : poOtherLine->paoPoints[nStartVertex - i].x;
1597 1289 : paoPoints[i + nOldPoints].y =
1598 1289 : poOtherLine->paoPoints[nStartVertex - i].y;
1599 : }
1600 :
1601 191 : if (poOtherLine->padfZ != nullptr)
1602 : {
1603 0 : Make3D();
1604 0 : if (padfZ != nullptr)
1605 : {
1606 0 : for (int i = 0; i < nPointsToAdd; i++)
1607 : {
1608 0 : padfZ[i + nOldPoints] =
1609 0 : poOtherLine->padfZ[nStartVertex - i];
1610 : }
1611 : }
1612 : }
1613 191 : if (poOtherLine->padfM != nullptr)
1614 : {
1615 0 : AddM();
1616 0 : if (padfM != nullptr)
1617 : {
1618 0 : for (int i = 0; i < nPointsToAdd; i++)
1619 : {
1620 0 : padfM[i + nOldPoints] =
1621 0 : poOtherLine->padfM[nStartVertex - i];
1622 : }
1623 : }
1624 : }
1625 : }
1626 : }
1627 :
1628 : /************************************************************************/
1629 : /* importFromWkb() */
1630 : /* */
1631 : /* Initialize from serialized stream in well known binary */
1632 : /* format. */
1633 : /************************************************************************/
1634 :
1635 5308 : OGRErr OGRSimpleCurve::importFromWkb(const unsigned char *pabyData,
1636 : size_t nSize, OGRwkbVariant eWkbVariant,
1637 : size_t &nBytesConsumedOut)
1638 :
1639 : {
1640 : OGRwkbByteOrder eByteOrder;
1641 5308 : size_t nDataOffset = 0;
1642 5308 : int nNewNumPoints = 0;
1643 :
1644 5308 : nBytesConsumedOut = 0;
1645 5308 : OGRErr eErr = importPreambleOfCollectionFromWkb(pabyData, nSize,
1646 : nDataOffset, eByteOrder, 16,
1647 : nNewNumPoints, eWkbVariant);
1648 5308 : if (eErr != OGRERR_NONE)
1649 203 : return eErr;
1650 :
1651 : // Check if the wkb stream buffer is big enough to store
1652 : // fetched number of points.
1653 5105 : const int dim = CoordinateDimension();
1654 5105 : const size_t nPointSize = dim * sizeof(double);
1655 10210 : if (nNewNumPoints < 0 ||
1656 5105 : static_cast<size_t>(nNewNumPoints) >
1657 5105 : std::numeric_limits<size_t>::max() / nPointSize)
1658 : {
1659 0 : return OGRERR_CORRUPT_DATA;
1660 : }
1661 5105 : const size_t nBufferMinSize = nPointSize * nNewNumPoints;
1662 :
1663 5105 : if (nSize != static_cast<size_t>(-1) && nBufferMinSize > nSize)
1664 : {
1665 67 : CPLError(CE_Failure, CPLE_AppDefined,
1666 : "Length of input WKB is too small");
1667 67 : return OGRERR_NOT_ENOUGH_DATA;
1668 : }
1669 :
1670 5038 : if (!setNumPoints(nNewNumPoints, FALSE))
1671 0 : return OGRERR_NOT_ENOUGH_MEMORY;
1672 :
1673 5038 : nBytesConsumedOut = 9 + 8 * static_cast<size_t>(nPointCount) *
1674 5038 : (2 + ((flags & OGR_G_3D) ? 1 : 0) +
1675 5038 : ((flags & OGR_G_MEASURED) ? 1 : 0));
1676 :
1677 : /* -------------------------------------------------------------------- */
1678 : /* Get the vertex. */
1679 : /* -------------------------------------------------------------------- */
1680 5038 : if ((flags & OGR_G_3D) && (flags & OGR_G_MEASURED))
1681 : {
1682 8832 : for (size_t i = 0; i < static_cast<size_t>(nPointCount); i++)
1683 : {
1684 5328 : memcpy(paoPoints + i, pabyData + 9 + i * 32, 16);
1685 5328 : memcpy(padfZ + i, pabyData + 9 + 16 + i * 32, 8);
1686 5328 : memcpy(padfM + i, pabyData + 9 + 24 + i * 32, 8);
1687 3504 : }
1688 : }
1689 1534 : else if (flags & OGR_G_MEASURED)
1690 : {
1691 252 : for (size_t i = 0; i < static_cast<size_t>(nPointCount); i++)
1692 : {
1693 185 : memcpy(paoPoints + i, pabyData + 9 + i * 24, 16);
1694 185 : memcpy(padfM + i, pabyData + 9 + 16 + i * 24, 8);
1695 : }
1696 : }
1697 1467 : else if (flags & OGR_G_3D)
1698 : {
1699 578 : for (size_t i = 0; i < static_cast<size_t>(nPointCount); i++)
1700 : {
1701 413 : memcpy(paoPoints + i, pabyData + 9 + i * 24, 16);
1702 413 : memcpy(padfZ + i, pabyData + 9 + 16 + i * 24, 8);
1703 : }
1704 : }
1705 1302 : else if (nPointCount)
1706 : {
1707 1272 : memcpy(paoPoints, pabyData + 9, 16 * static_cast<size_t>(nPointCount));
1708 : }
1709 :
1710 : /* -------------------------------------------------------------------- */
1711 : /* Byte swap if needed. */
1712 : /* -------------------------------------------------------------------- */
1713 5038 : if (OGR_SWAP(eByteOrder))
1714 : {
1715 440 : for (size_t i = 0; i < static_cast<size_t>(nPointCount); i++)
1716 : {
1717 310 : CPL_SWAPDOUBLE(&(paoPoints[i].x));
1718 310 : CPL_SWAPDOUBLE(&(paoPoints[i].y));
1719 : }
1720 :
1721 130 : if (flags & OGR_G_3D)
1722 : {
1723 14 : for (size_t i = 0; i < static_cast<size_t>(nPointCount); i++)
1724 : {
1725 10 : CPL_SWAPDOUBLE(padfZ + i);
1726 : }
1727 : }
1728 :
1729 130 : if (flags & OGR_G_MEASURED)
1730 : {
1731 6 : for (size_t i = 0; i < static_cast<size_t>(nPointCount); i++)
1732 : {
1733 4 : CPL_SWAPDOUBLE(padfM + i);
1734 : }
1735 : }
1736 : }
1737 :
1738 5038 : return OGRERR_NONE;
1739 : }
1740 :
1741 : /************************************************************************/
1742 : /* exportToWkb() */
1743 : /* */
1744 : /* Build a well known binary representation of this object. */
1745 : /************************************************************************/
1746 :
1747 9991 : OGRErr OGRSimpleCurve::exportToWkb(unsigned char *pabyData,
1748 : const OGRwkbExportOptions *psOptions) const
1749 :
1750 : {
1751 9991 : if (psOptions == nullptr)
1752 : {
1753 : static const OGRwkbExportOptions defaultOptions;
1754 0 : psOptions = &defaultOptions;
1755 : }
1756 :
1757 : /* -------------------------------------------------------------------- */
1758 : /* Set the byte order. */
1759 : /* -------------------------------------------------------------------- */
1760 9991 : pabyData[0] = DB2_V72_UNFIX_BYTE_ORDER(
1761 : static_cast<unsigned char>(psOptions->eByteOrder));
1762 :
1763 : /* -------------------------------------------------------------------- */
1764 : /* Set the geometry feature type. */
1765 : /* -------------------------------------------------------------------- */
1766 9991 : GUInt32 nGType = getGeometryType();
1767 :
1768 9991 : if (psOptions->eWkbVariant == wkbVariantPostGIS1)
1769 : {
1770 6 : nGType = wkbFlatten(nGType);
1771 6 : if (Is3D())
1772 : // Explicitly set wkb25DBit.
1773 1 : nGType =
1774 1 : static_cast<OGRwkbGeometryType>(nGType | wkb25DBitInternalUse);
1775 6 : if (IsMeasured())
1776 0 : nGType = static_cast<OGRwkbGeometryType>(nGType | 0x40000000);
1777 : }
1778 9985 : else if (psOptions->eWkbVariant == wkbVariantIso)
1779 3741 : nGType = getIsoGeometryType();
1780 :
1781 9991 : if (psOptions->eByteOrder == wkbNDR)
1782 : {
1783 9978 : CPL_LSBPTR32(&nGType);
1784 : }
1785 : else
1786 : {
1787 13 : CPL_MSBPTR32(&nGType);
1788 : }
1789 :
1790 9991 : memcpy(pabyData + 1, &nGType, 4);
1791 :
1792 : /* -------------------------------------------------------------------- */
1793 : /* Copy in the data count. */
1794 : /* -------------------------------------------------------------------- */
1795 9991 : memcpy(pabyData + 5, &nPointCount, 4);
1796 :
1797 : /* -------------------------------------------------------------------- */
1798 : /* Copy in the raw data. */
1799 : /* -------------------------------------------------------------------- */
1800 9991 : if ((flags & OGR_G_3D) && (flags & OGR_G_MEASURED))
1801 : {
1802 4389 : for (size_t i = 0; i < static_cast<size_t>(nPointCount); i++)
1803 : {
1804 2640 : memcpy(pabyData + 9 + 32 * i, paoPoints + i, 16);
1805 2640 : memcpy(pabyData + 9 + 16 + 32 * i, padfZ + i, 8);
1806 2640 : memcpy(pabyData + 9 + 24 + 32 * i, padfM + i, 8);
1807 : }
1808 1749 : OGRRoundCoordinatesIEEE754XYValues<32>(
1809 1749 : psOptions->sPrecision.nXYBitPrecision, pabyData + 9, nPointCount);
1810 1749 : OGRRoundCoordinatesIEEE754<32>(psOptions->sPrecision.nZBitPrecision,
1811 : pabyData + 9 + 2 * sizeof(uint64_t),
1812 1749 : nPointCount);
1813 1749 : OGRRoundCoordinatesIEEE754<32>(psOptions->sPrecision.nMBitPrecision,
1814 : pabyData + 9 + 3 * sizeof(uint64_t),
1815 1749 : nPointCount);
1816 : }
1817 8242 : else if (flags & OGR_G_MEASURED)
1818 : {
1819 107 : for (size_t i = 0; i < static_cast<size_t>(nPointCount); i++)
1820 : {
1821 74 : memcpy(pabyData + 9 + 24 * i, paoPoints + i, 16);
1822 74 : memcpy(pabyData + 9 + 16 + 24 * i, padfM + i, 8);
1823 : }
1824 33 : OGRRoundCoordinatesIEEE754XYValues<24>(
1825 33 : psOptions->sPrecision.nXYBitPrecision, pabyData + 9, nPointCount);
1826 33 : OGRRoundCoordinatesIEEE754<24>(psOptions->sPrecision.nMBitPrecision,
1827 : pabyData + 9 + 2 * sizeof(uint64_t),
1828 33 : nPointCount);
1829 : }
1830 8209 : else if (flags & OGR_G_3D)
1831 : {
1832 63125 : for (size_t i = 0; i < static_cast<size_t>(nPointCount); i++)
1833 : {
1834 59502 : memcpy(pabyData + 9 + 24 * i, paoPoints + i, 16);
1835 59502 : memcpy(pabyData + 9 + 16 + 24 * i, padfZ + i, 8);
1836 : }
1837 3623 : OGRRoundCoordinatesIEEE754XYValues<24>(
1838 3623 : psOptions->sPrecision.nXYBitPrecision, pabyData + 9, nPointCount);
1839 3623 : OGRRoundCoordinatesIEEE754<24>(psOptions->sPrecision.nZBitPrecision,
1840 : pabyData + 9 + 2 * sizeof(uint64_t),
1841 3623 : nPointCount);
1842 : }
1843 4586 : else if (nPointCount)
1844 : {
1845 4515 : memcpy(pabyData + 9, paoPoints, 16 * static_cast<size_t>(nPointCount));
1846 4515 : OGRRoundCoordinatesIEEE754XYValues<16>(
1847 4515 : psOptions->sPrecision.nXYBitPrecision, pabyData + 9, nPointCount);
1848 : }
1849 :
1850 : /* -------------------------------------------------------------------- */
1851 : /* Swap if needed. */
1852 : /* -------------------------------------------------------------------- */
1853 9991 : if (OGR_SWAP(psOptions->eByteOrder))
1854 : {
1855 13 : const int nCount = CPL_SWAP32(nPointCount);
1856 13 : memcpy(pabyData + 5, &nCount, 4);
1857 :
1858 : const size_t nCoords =
1859 13 : CoordinateDimension() * static_cast<size_t>(nPointCount);
1860 175 : for (size_t i = 0; i < nCoords; i++)
1861 : {
1862 162 : CPL_SWAP64PTR(pabyData + 9 + 8 * i);
1863 : }
1864 : }
1865 :
1866 9991 : return OGRERR_NONE;
1867 : }
1868 :
1869 : /************************************************************************/
1870 : /* importFromWkt() */
1871 : /* */
1872 : /* Instantiate from well known text format. Currently this is */
1873 : /* `LINESTRING ( x y, x y, ...)', */
1874 : /************************************************************************/
1875 :
1876 5206 : OGRErr OGRSimpleCurve::importFromWkt(const char **ppszInput)
1877 :
1878 : {
1879 5206 : int bHasZ = FALSE;
1880 5206 : int bHasM = FALSE;
1881 5206 : bool bIsEmpty = false;
1882 : const OGRErr eErr =
1883 5206 : importPreambleFromWkt(ppszInput, &bHasZ, &bHasM, &bIsEmpty);
1884 5206 : flags = 0;
1885 5206 : if (eErr != OGRERR_NONE)
1886 5 : return eErr;
1887 5201 : if (bHasZ)
1888 302 : flags |= OGR_G_3D;
1889 5201 : if (bHasM)
1890 135 : flags |= OGR_G_MEASURED;
1891 5201 : if (bIsEmpty)
1892 : {
1893 85 : return OGRERR_NONE;
1894 : }
1895 :
1896 5116 : const char *pszInput = *ppszInput;
1897 :
1898 : /* -------------------------------------------------------------------- */
1899 : /* Read the point list. */
1900 : /* -------------------------------------------------------------------- */
1901 5116 : int flagsFromInput = flags;
1902 5116 : nPointCount = 0;
1903 :
1904 : pszInput =
1905 5116 : OGRWktReadPointsM(pszInput, &paoPoints, &padfZ, &padfM, &flagsFromInput,
1906 : &m_nPointCapacity, &nPointCount);
1907 5116 : if (pszInput == nullptr)
1908 15 : return OGRERR_CORRUPT_DATA;
1909 :
1910 5101 : if ((flagsFromInput & OGR_G_3D) && !(flags & OGR_G_3D))
1911 : {
1912 272 : if (!set3D(TRUE))
1913 0 : return OGRERR_NOT_ENOUGH_MEMORY;
1914 : }
1915 5101 : if ((flagsFromInput & OGR_G_MEASURED) && !(flags & OGR_G_MEASURED))
1916 : {
1917 0 : if (!setMeasured(TRUE))
1918 0 : return OGRERR_NOT_ENOUGH_MEMORY;
1919 : }
1920 :
1921 5101 : *ppszInput = pszInput;
1922 :
1923 5101 : return OGRERR_NONE;
1924 : }
1925 :
1926 : //! @cond Doxygen_Suppress
1927 : /************************************************************************/
1928 : /* importFromWKTListOnly() */
1929 : /* */
1930 : /* Instantiate from "(x y, x y, ...)" */
1931 : /************************************************************************/
1932 :
1933 1413 : OGRErr OGRSimpleCurve::importFromWKTListOnly(const char **ppszInput, int bHasZ,
1934 : int bHasM,
1935 : OGRRawPoint *&paoPointsIn,
1936 : int &nMaxPointsIn,
1937 : double *&padfZIn)
1938 :
1939 : {
1940 1413 : const char *pszInput = *ppszInput;
1941 :
1942 : /* -------------------------------------------------------------------- */
1943 : /* Read the point list. */
1944 : /* -------------------------------------------------------------------- */
1945 1413 : int flagsFromInput = flags;
1946 1413 : int nPointCountRead = 0;
1947 1413 : double *padfMIn = nullptr;
1948 1413 : if (flagsFromInput == 0) // Flags was not set, this is not called by us.
1949 : {
1950 1413 : if (bHasM)
1951 124 : flagsFromInput |= OGR_G_MEASURED;
1952 1413 : if (bHasZ)
1953 197 : flagsFromInput |= OGR_G_3D;
1954 : }
1955 :
1956 : pszInput =
1957 1413 : OGRWktReadPointsM(pszInput, &paoPointsIn, &padfZIn, &padfMIn,
1958 : &flagsFromInput, &nMaxPointsIn, &nPointCountRead);
1959 :
1960 1413 : if (pszInput == nullptr)
1961 : {
1962 6 : CPLFree(padfMIn);
1963 6 : return OGRERR_CORRUPT_DATA;
1964 : }
1965 1407 : if ((flagsFromInput & OGR_G_3D) && !(flags & OGR_G_3D))
1966 : {
1967 364 : flags |= OGR_G_3D;
1968 364 : bHasZ = TRUE;
1969 : }
1970 1407 : if ((flagsFromInput & OGR_G_MEASURED) && !(flags & OGR_G_MEASURED))
1971 : {
1972 124 : flags |= OGR_G_MEASURED;
1973 124 : bHasM = TRUE;
1974 : }
1975 :
1976 1407 : *ppszInput = pszInput;
1977 :
1978 1407 : if (bHasM && bHasZ)
1979 72 : setPoints(nPointCountRead, paoPointsIn, padfZIn, padfMIn);
1980 1335 : else if (bHasM && !bHasZ)
1981 52 : setPointsM(nPointCountRead, paoPointsIn, padfMIn);
1982 : else
1983 1283 : setPoints(nPointCountRead, paoPointsIn, padfZIn);
1984 :
1985 1407 : CPLFree(padfMIn);
1986 :
1987 1407 : return OGRERR_NONE;
1988 : }
1989 :
1990 : //! @endcond
1991 :
1992 : /************************************************************************/
1993 : /* exportToWkt() */
1994 : /* */
1995 : /* Translate this structure into its well known text format */
1996 : /* equivalent. This could be made a lot more CPU efficient. */
1997 : /************************************************************************/
1998 :
1999 3605 : std::string OGRSimpleCurve::exportToWkt(const OGRWktOptions &opts,
2000 : OGRErr *err) const
2001 : {
2002 : // LINEARRING or LINESTRING or CIRCULARSTRING
2003 7210 : std::string wkt = getGeometryName();
2004 3605 : wkt += wktTypeString(opts.variant);
2005 3605 : if (IsEmpty())
2006 : {
2007 89 : wkt += "EMPTY";
2008 : }
2009 : else
2010 : {
2011 3516 : wkt += '(';
2012 :
2013 3516 : OGRBoolean hasZ = Is3D();
2014 : OGRBoolean hasM =
2015 3516 : (opts.variant != wkbVariantIso ? FALSE : IsMeasured());
2016 :
2017 : try
2018 : {
2019 3516 : const int nOrdinatesPerVertex =
2020 3516 : 2 + ((hasZ) ? 1 : 0) + ((hasM) ? 1 : 0);
2021 : // At least 2 bytes per ordinate: one for the value,
2022 : // and one for the separator...
2023 3516 : wkt.reserve(wkt.size() + 2 * static_cast<size_t>(nPointCount) *
2024 3516 : nOrdinatesPerVertex);
2025 :
2026 34760 : for (int i = 0; i < nPointCount; i++)
2027 : {
2028 31244 : if (i > 0)
2029 27728 : wkt += ',';
2030 :
2031 88736 : wkt += OGRMakeWktCoordinateM(
2032 59070 : paoPoints[i].x, paoPoints[i].y, padfZ ? padfZ[i] : 0.0,
2033 62488 : padfM ? padfM[i] : 0.0, hasZ, hasM, opts);
2034 : }
2035 3516 : wkt += ')';
2036 : }
2037 0 : catch (const std::bad_alloc &e)
2038 : {
2039 0 : CPLError(CE_Failure, CPLE_OutOfMemory, "%s", e.what());
2040 0 : if (err)
2041 0 : *err = OGRERR_FAILURE;
2042 0 : return std::string();
2043 : }
2044 : }
2045 3605 : if (err)
2046 3596 : *err = OGRERR_NONE;
2047 3605 : return wkt;
2048 : }
2049 :
2050 : /************************************************************************/
2051 : /* get_Length() */
2052 : /* */
2053 : /* For now we return a simple euclidean 2D distance. */
2054 : /************************************************************************/
2055 :
2056 1137 : double OGRSimpleCurve::get_Length() const
2057 :
2058 : {
2059 1137 : double dfLength = 0.0;
2060 :
2061 3263 : for (int i = 0; i < nPointCount - 1; i++)
2062 : {
2063 :
2064 2126 : const double dfDeltaX = paoPoints[i + 1].x - paoPoints[i].x;
2065 2126 : const double dfDeltaY = paoPoints[i + 1].y - paoPoints[i].y;
2066 2126 : dfLength += sqrt(dfDeltaX * dfDeltaX + dfDeltaY * dfDeltaY);
2067 : }
2068 :
2069 1137 : return dfLength;
2070 : }
2071 :
2072 : /************************************************************************/
2073 : /* StartPoint() */
2074 : /************************************************************************/
2075 :
2076 283782 : void OGRSimpleCurve::StartPoint(OGRPoint *poPoint) const
2077 :
2078 : {
2079 283782 : getPoint(0, poPoint);
2080 283782 : }
2081 :
2082 : /************************************************************************/
2083 : /* EndPoint() */
2084 : /************************************************************************/
2085 :
2086 223607 : void OGRSimpleCurve::EndPoint(OGRPoint *poPoint) const
2087 :
2088 : {
2089 223607 : getPoint(nPointCount - 1, poPoint);
2090 223607 : }
2091 :
2092 : /************************************************************************/
2093 : /* Value() */
2094 : /* */
2095 : /* Get an interpolated point at some distance along the curve. */
2096 : /************************************************************************/
2097 :
2098 52 : void OGRSimpleCurve::Value(double dfDistance, OGRPoint *poPoint) const
2099 :
2100 : {
2101 52 : if (dfDistance < 0)
2102 : {
2103 1 : StartPoint(poPoint);
2104 1 : return;
2105 : }
2106 :
2107 51 : double dfLength = 0.0;
2108 :
2109 192 : for (int i = 0; i < nPointCount - 1; i++)
2110 : {
2111 191 : const double dfDeltaX = paoPoints[i + 1].x - paoPoints[i].x;
2112 191 : const double dfDeltaY = paoPoints[i + 1].y - paoPoints[i].y;
2113 : const double dfSegLength =
2114 191 : sqrt(dfDeltaX * dfDeltaX + dfDeltaY * dfDeltaY);
2115 :
2116 191 : if (dfSegLength > 0)
2117 : {
2118 171 : if ((dfLength <= dfDistance) &&
2119 171 : ((dfLength + dfSegLength) >= dfDistance))
2120 : {
2121 50 : double dfRatio = (dfDistance - dfLength) / dfSegLength;
2122 :
2123 50 : poPoint->setX(paoPoints[i].x * (1 - dfRatio) +
2124 50 : paoPoints[i + 1].x * dfRatio);
2125 50 : poPoint->setY(paoPoints[i].y * (1 - dfRatio) +
2126 50 : paoPoints[i + 1].y * dfRatio);
2127 :
2128 50 : if (getCoordinateDimension() == 3)
2129 1 : poPoint->setZ(padfZ[i] * (1 - dfRatio) +
2130 1 : padfZ[i + 1] * dfRatio);
2131 :
2132 50 : return;
2133 : }
2134 :
2135 121 : dfLength += dfSegLength;
2136 : }
2137 : }
2138 :
2139 1 : EndPoint(poPoint);
2140 : }
2141 :
2142 : /************************************************************************/
2143 : /* Project() */
2144 : /* */
2145 : /* Return distance of point projected on line from origin of this line. */
2146 : /************************************************************************/
2147 :
2148 : /**
2149 : * \brief Project point on linestring.
2150 : *
2151 : * The input point projected on linestring. This is the shortest distance
2152 : * from point to the linestring. The distance from begin of linestring to
2153 : * the point projection returned.
2154 : *
2155 : * This method is built on the GEOS library. Check it for the
2156 : * definition of the geometry operation.
2157 : * If OGR is built without the GEOS library, this method will always return -1,
2158 : * issuing a CPLE_NotSupported error.
2159 : *
2160 : * @return a distance from the begin of the linestring to the projected point.
2161 : */
2162 :
2163 62 : double OGRSimpleCurve::Project(const OGRPoint *poPoint) const
2164 :
2165 : {
2166 62 : double dfResult = -1;
2167 : #ifndef HAVE_GEOS
2168 : CPL_IGNORE_RET_VAL(poPoint);
2169 : CPLError(CE_Failure, CPLE_NotSupported, "GEOS support not enabled.");
2170 : return dfResult;
2171 : #else
2172 62 : GEOSGeom hThisGeosGeom = nullptr;
2173 62 : GEOSGeom hPointGeosGeom = nullptr;
2174 :
2175 62 : GEOSContextHandle_t hGEOSCtxt = createGEOSContext();
2176 62 : hThisGeosGeom = exportToGEOS(hGEOSCtxt);
2177 62 : hPointGeosGeom = poPoint->exportToGEOS(hGEOSCtxt);
2178 62 : if (hThisGeosGeom != nullptr && hPointGeosGeom != nullptr)
2179 : {
2180 62 : dfResult = GEOSProject_r(hGEOSCtxt, hThisGeosGeom, hPointGeosGeom);
2181 : }
2182 62 : GEOSGeom_destroy_r(hGEOSCtxt, hThisGeosGeom);
2183 62 : GEOSGeom_destroy_r(hGEOSCtxt, hPointGeosGeom);
2184 62 : freeGEOSContext(hGEOSCtxt);
2185 :
2186 62 : return dfResult;
2187 :
2188 : #endif // HAVE_GEOS
2189 : }
2190 :
2191 : /************************************************************************/
2192 : /* getSubLine() */
2193 : /* */
2194 : /* Extracts a portion of this OGRLineString into a new OGRLineString. */
2195 : /************************************************************************/
2196 :
2197 : /**
2198 : * \brief Get the portion of linestring.
2199 : *
2200 : * The portion of the linestring extracted to new one. The input distances
2201 : * (maybe present as ratio of length of linestring) set begin and end of
2202 : * extracted portion.
2203 : *
2204 : * @param dfDistanceFrom The distance from the origin of linestring, where the
2205 : * subline should begins
2206 : * @param dfDistanceTo The distance from the origin of linestring, where the
2207 : * subline should ends
2208 : * @param bAsRatio The flag indicating that distances are the ratio of the
2209 : * linestring length.
2210 : *
2211 : * @return a newly allocated linestring now owned by the caller, or NULL on
2212 : * failure.
2213 : *
2214 : * @since OGR 1.11.0
2215 : */
2216 :
2217 53 : OGRLineString *OGRSimpleCurve::getSubLine(double dfDistanceFrom,
2218 : double dfDistanceTo,
2219 : int bAsRatio) const
2220 :
2221 : {
2222 106 : auto poNewLineString = std::make_unique<OGRLineString>();
2223 :
2224 53 : poNewLineString->assignSpatialReference(getSpatialReference());
2225 53 : poNewLineString->setCoordinateDimension(getCoordinateDimension());
2226 :
2227 53 : const double dfLen = get_Length();
2228 53 : if (bAsRatio == TRUE)
2229 : {
2230 : // Convert to real distance.
2231 0 : dfDistanceFrom *= dfLen;
2232 0 : dfDistanceTo *= dfLen;
2233 : }
2234 :
2235 53 : if (dfDistanceFrom < 0)
2236 0 : dfDistanceFrom = 0;
2237 53 : if (dfDistanceTo > dfLen)
2238 0 : dfDistanceTo = dfLen;
2239 :
2240 53 : if (dfDistanceFrom > dfDistanceTo || dfDistanceFrom >= dfLen)
2241 : {
2242 0 : CPLError(CE_Failure, CPLE_IllegalArg, "Input distances are invalid.");
2243 :
2244 0 : return nullptr;
2245 : }
2246 :
2247 53 : double dfLength = 0.0;
2248 :
2249 : // Get first point.
2250 :
2251 53 : int i = 0; // Used after if blocks.
2252 53 : if (dfDistanceFrom == 0)
2253 : {
2254 : bool bRet;
2255 5 : if (getCoordinateDimension() == 3)
2256 0 : bRet = poNewLineString->addPoint(paoPoints[0].x, paoPoints[0].y,
2257 0 : padfZ[0]);
2258 : else
2259 5 : bRet = poNewLineString->addPoint(paoPoints[0].x, paoPoints[0].y);
2260 5 : if (!bRet)
2261 0 : return nullptr;
2262 : }
2263 : else
2264 : {
2265 367 : for (i = 0; i < nPointCount - 1; i++)
2266 : {
2267 367 : const double dfDeltaX = paoPoints[i + 1].x - paoPoints[i].x;
2268 367 : const double dfDeltaY = paoPoints[i + 1].y - paoPoints[i].y;
2269 : const double dfSegLength =
2270 367 : sqrt(dfDeltaX * dfDeltaX + dfDeltaY * dfDeltaY);
2271 :
2272 367 : if (dfSegLength > 0)
2273 : {
2274 367 : if ((dfLength <= dfDistanceFrom) &&
2275 367 : ((dfLength + dfSegLength) >= dfDistanceFrom))
2276 : {
2277 48 : double dfRatio = (dfDistanceFrom - dfLength) / dfSegLength;
2278 :
2279 48 : double dfX = paoPoints[i].x * (1 - dfRatio) +
2280 48 : paoPoints[i + 1].x * dfRatio;
2281 48 : double dfY = paoPoints[i].y * (1 - dfRatio) +
2282 48 : paoPoints[i + 1].y * dfRatio;
2283 :
2284 : bool bRet;
2285 48 : if (getCoordinateDimension() == 3)
2286 : {
2287 0 : bRet = poNewLineString->addPoint(
2288 : dfX, dfY,
2289 0 : padfZ[i] * (1 - dfRatio) + padfZ[i + 1] * dfRatio);
2290 : }
2291 : else
2292 : {
2293 48 : bRet = poNewLineString->addPoint(dfX, dfY);
2294 : }
2295 48 : if (!bRet)
2296 0 : return nullptr;
2297 :
2298 : // Check if dfDistanceTo is in same segment.
2299 48 : if (dfLength <= dfDistanceTo &&
2300 48 : (dfLength + dfSegLength) >= dfDistanceTo)
2301 : {
2302 27 : dfRatio = (dfDistanceTo - dfLength) / dfSegLength;
2303 :
2304 27 : dfX = paoPoints[i].x * (1 - dfRatio) +
2305 27 : paoPoints[i + 1].x * dfRatio;
2306 27 : dfY = paoPoints[i].y * (1 - dfRatio) +
2307 27 : paoPoints[i + 1].y * dfRatio;
2308 :
2309 27 : if (getCoordinateDimension() == 3)
2310 : {
2311 0 : bRet = poNewLineString->addPoint(
2312 : dfX, dfY,
2313 0 : padfZ[i] * (1 - dfRatio) +
2314 0 : padfZ[i + 1] * dfRatio);
2315 : }
2316 : else
2317 : {
2318 27 : bRet = poNewLineString->addPoint(dfX, dfY);
2319 : }
2320 :
2321 27 : if (!bRet || poNewLineString->getNumPoints() < 2)
2322 : {
2323 0 : return nullptr;
2324 : }
2325 :
2326 27 : return poNewLineString.release();
2327 : }
2328 21 : i++;
2329 21 : dfLength += dfSegLength;
2330 21 : break;
2331 : }
2332 :
2333 319 : dfLength += dfSegLength;
2334 : }
2335 : }
2336 : }
2337 :
2338 : // Add points.
2339 55 : for (; i < nPointCount - 1; i++)
2340 : {
2341 : bool bRet;
2342 55 : if (getCoordinateDimension() == 3)
2343 0 : bRet = poNewLineString->addPoint(paoPoints[i].x, paoPoints[i].y,
2344 0 : padfZ[i]);
2345 : else
2346 55 : bRet = poNewLineString->addPoint(paoPoints[i].x, paoPoints[i].y);
2347 55 : if (!bRet)
2348 0 : return nullptr;
2349 :
2350 55 : const double dfDeltaX = paoPoints[i + 1].x - paoPoints[i].x;
2351 55 : const double dfDeltaY = paoPoints[i + 1].y - paoPoints[i].y;
2352 : const double dfSegLength =
2353 55 : sqrt(dfDeltaX * dfDeltaX + dfDeltaY * dfDeltaY);
2354 :
2355 55 : if (dfSegLength > 0)
2356 : {
2357 55 : if ((dfLength <= dfDistanceTo) &&
2358 55 : ((dfLength + dfSegLength) >= dfDistanceTo))
2359 : {
2360 26 : const double dfRatio = (dfDistanceTo - dfLength) / dfSegLength;
2361 :
2362 26 : const double dfX = paoPoints[i].x * (1 - dfRatio) +
2363 26 : paoPoints[i + 1].x * dfRatio;
2364 26 : const double dfY = paoPoints[i].y * (1 - dfRatio) +
2365 26 : paoPoints[i + 1].y * dfRatio;
2366 :
2367 26 : if (getCoordinateDimension() == 3)
2368 0 : bRet = poNewLineString->addPoint(
2369 : dfX, dfY,
2370 0 : padfZ[i] * (1 - dfRatio) + padfZ[i + 1] * dfRatio);
2371 : else
2372 26 : bRet = poNewLineString->addPoint(dfX, dfY);
2373 26 : if (!bRet)
2374 0 : return nullptr;
2375 :
2376 26 : return poNewLineString.release();
2377 : }
2378 :
2379 29 : dfLength += dfSegLength;
2380 : }
2381 : }
2382 :
2383 : bool bRet;
2384 0 : if (getCoordinateDimension() == 3)
2385 0 : bRet = poNewLineString->addPoint(paoPoints[nPointCount - 1].x,
2386 0 : paoPoints[nPointCount - 1].y,
2387 0 : padfZ[nPointCount - 1]);
2388 : else
2389 0 : bRet = poNewLineString->addPoint(paoPoints[nPointCount - 1].x,
2390 0 : paoPoints[nPointCount - 1].y);
2391 :
2392 0 : if (!bRet || poNewLineString->getNumPoints() < 2)
2393 : {
2394 0 : return nullptr;
2395 : }
2396 :
2397 0 : return poNewLineString.release();
2398 : }
2399 :
2400 : /************************************************************************/
2401 : /* getEnvelope() */
2402 : /************************************************************************/
2403 :
2404 5644480 : void OGRSimpleCurve::getEnvelope(OGREnvelope *psEnvelope) const
2405 :
2406 : {
2407 5644480 : if (IsEmpty())
2408 : {
2409 3 : psEnvelope->MinX = 0.0;
2410 3 : psEnvelope->MaxX = 0.0;
2411 3 : psEnvelope->MinY = 0.0;
2412 3 : psEnvelope->MaxY = 0.0;
2413 3 : return;
2414 : }
2415 :
2416 5644480 : double dfMinX = paoPoints[0].x;
2417 5644480 : double dfMaxX = paoPoints[0].x;
2418 5644480 : double dfMinY = paoPoints[0].y;
2419 5644480 : double dfMaxY = paoPoints[0].y;
2420 :
2421 31917400 : for (int iPoint = 1; iPoint < nPointCount; iPoint++)
2422 : {
2423 26272900 : if (dfMaxX < paoPoints[iPoint].x)
2424 7499950 : dfMaxX = paoPoints[iPoint].x;
2425 26272900 : if (dfMaxY < paoPoints[iPoint].y)
2426 7458730 : dfMaxY = paoPoints[iPoint].y;
2427 26272900 : if (dfMinX > paoPoints[iPoint].x)
2428 4739570 : dfMinX = paoPoints[iPoint].x;
2429 26272900 : if (dfMinY > paoPoints[iPoint].y)
2430 5544390 : dfMinY = paoPoints[iPoint].y;
2431 : }
2432 :
2433 5644480 : psEnvelope->MinX = dfMinX;
2434 5644480 : psEnvelope->MaxX = dfMaxX;
2435 5644480 : psEnvelope->MinY = dfMinY;
2436 5644480 : psEnvelope->MaxY = dfMaxY;
2437 : }
2438 :
2439 : /************************************************************************/
2440 : /* getEnvelope() */
2441 : /************************************************************************/
2442 :
2443 637310 : void OGRSimpleCurve::getEnvelope(OGREnvelope3D *psEnvelope) const
2444 :
2445 : {
2446 637310 : getEnvelope(static_cast<OGREnvelope *>(psEnvelope));
2447 :
2448 637310 : if (IsEmpty() || padfZ == nullptr)
2449 : {
2450 259794 : psEnvelope->MinZ = 0.0;
2451 259794 : psEnvelope->MaxZ = 0.0;
2452 259794 : return;
2453 : }
2454 :
2455 377516 : double dfMinZ = padfZ[0];
2456 377516 : double dfMaxZ = padfZ[0];
2457 :
2458 1549830 : for (int iPoint = 1; iPoint < nPointCount; iPoint++)
2459 : {
2460 1172320 : if (dfMinZ > padfZ[iPoint])
2461 370010 : dfMinZ = padfZ[iPoint];
2462 1172320 : if (dfMaxZ < padfZ[iPoint])
2463 230759 : dfMaxZ = padfZ[iPoint];
2464 : }
2465 :
2466 377516 : psEnvelope->MinZ = dfMinZ;
2467 377516 : psEnvelope->MaxZ = dfMaxZ;
2468 : }
2469 :
2470 : /************************************************************************/
2471 : /* Equals() */
2472 : /************************************************************************/
2473 :
2474 48095 : OGRBoolean OGRSimpleCurve::Equals(const OGRGeometry *poOther) const
2475 :
2476 : {
2477 48095 : if (poOther == this)
2478 1 : return TRUE;
2479 :
2480 48094 : if (poOther->getGeometryType() != getGeometryType())
2481 1 : return FALSE;
2482 :
2483 48093 : if (IsEmpty() && poOther->IsEmpty())
2484 1 : return TRUE;
2485 :
2486 : // TODO(schwehr): Test the SRS.
2487 :
2488 48092 : auto poOLine = poOther->toSimpleCurve();
2489 48092 : if (getNumPoints() != poOLine->getNumPoints())
2490 8 : return FALSE;
2491 :
2492 251687 : for (int iPoint = 0; iPoint < getNumPoints(); iPoint++)
2493 : {
2494 431617 : if (getX(iPoint) != poOLine->getX(iPoint) ||
2495 431617 : getY(iPoint) != poOLine->getY(iPoint) ||
2496 203603 : getZ(iPoint) != poOLine->getZ(iPoint))
2497 14077 : return FALSE;
2498 : }
2499 :
2500 34007 : return TRUE;
2501 : }
2502 :
2503 : /************************************************************************/
2504 : /* transform() */
2505 : /************************************************************************/
2506 :
2507 2350 : OGRErr OGRSimpleCurve::transform(OGRCoordinateTransformation *poCT)
2508 :
2509 : {
2510 : /* -------------------------------------------------------------------- */
2511 : /* Make a copy of the points to operate on, so as to be able to */
2512 : /* keep only valid reprojected points if partial reprojection enabled */
2513 : /* or keeping intact the original geometry if only full reprojection */
2514 : /* allowed. */
2515 : /* -------------------------------------------------------------------- */
2516 : double *xyz = static_cast<double *>(
2517 2350 : VSI_MALLOC_VERBOSE(sizeof(double) * nPointCount * 3));
2518 : int *pabSuccess =
2519 2350 : static_cast<int *>(VSI_CALLOC_VERBOSE(sizeof(int), nPointCount));
2520 2350 : if (xyz == nullptr || pabSuccess == nullptr)
2521 : {
2522 0 : VSIFree(xyz);
2523 0 : VSIFree(pabSuccess);
2524 0 : return OGRERR_NOT_ENOUGH_MEMORY;
2525 : }
2526 :
2527 49657 : for (int i = 0; i < nPointCount; i++)
2528 : {
2529 47307 : xyz[i] = paoPoints[i].x;
2530 47307 : xyz[i + nPointCount] = paoPoints[i].y;
2531 47307 : if (padfZ)
2532 20354 : xyz[i + nPointCount * 2] = padfZ[i];
2533 : else
2534 26953 : xyz[i + nPointCount * 2] = 0.0;
2535 : }
2536 :
2537 : /* -------------------------------------------------------------------- */
2538 : /* Transform and reapply. */
2539 : /* -------------------------------------------------------------------- */
2540 2350 : poCT->Transform(nPointCount, xyz, xyz + nPointCount, xyz + nPointCount * 2,
2541 2350 : nullptr, pabSuccess);
2542 :
2543 2350 : const char *pszEnablePartialReprojection = nullptr;
2544 :
2545 2350 : int j = 0; // Used after for.
2546 49657 : for (int i = 0; i < nPointCount; i++)
2547 : {
2548 47307 : if (pabSuccess[i])
2549 : {
2550 47307 : xyz[j] = xyz[i];
2551 47307 : xyz[j + nPointCount] = xyz[i + nPointCount];
2552 47307 : xyz[j + 2 * nPointCount] = xyz[i + 2 * nPointCount];
2553 47307 : j++;
2554 : }
2555 : else
2556 : {
2557 0 : if (pszEnablePartialReprojection == nullptr)
2558 0 : pszEnablePartialReprojection = CPLGetConfigOption(
2559 : "OGR_ENABLE_PARTIAL_REPROJECTION", nullptr);
2560 0 : if (pszEnablePartialReprojection == nullptr)
2561 : {
2562 : static bool bHasWarned = false;
2563 0 : if (!bHasWarned)
2564 : {
2565 : // Check that there is at least one valid reprojected point
2566 : // and issue an error giving an hint to use
2567 : // OGR_ENABLE_PARTIAL_REPROJECTION.
2568 0 : bool bHasOneValidPoint = j != 0;
2569 0 : for (; i < nPointCount && !bHasOneValidPoint; i++)
2570 : {
2571 0 : if (pabSuccess[i])
2572 0 : bHasOneValidPoint = true;
2573 : }
2574 0 : if (bHasOneValidPoint)
2575 : {
2576 0 : bHasWarned = true;
2577 0 : CPLError(CE_Failure, CPLE_AppDefined,
2578 : "Full reprojection failed, but partial is "
2579 : "possible if you define "
2580 : "OGR_ENABLE_PARTIAL_REPROJECTION "
2581 : "configuration option to TRUE");
2582 : }
2583 : }
2584 :
2585 0 : CPLFree(xyz);
2586 0 : CPLFree(pabSuccess);
2587 0 : return OGRERR_FAILURE;
2588 : }
2589 0 : else if (!CPLTestBool(pszEnablePartialReprojection))
2590 : {
2591 0 : CPLFree(xyz);
2592 0 : CPLFree(pabSuccess);
2593 0 : return OGRERR_FAILURE;
2594 : }
2595 : }
2596 : }
2597 :
2598 2350 : if (j == 0 && nPointCount != 0)
2599 : {
2600 0 : CPLFree(xyz);
2601 0 : CPLFree(pabSuccess);
2602 0 : return OGRERR_FAILURE;
2603 : }
2604 :
2605 2350 : setPoints(j, xyz, xyz + nPointCount,
2606 2350 : (padfZ) ? xyz + nPointCount * 2 : nullptr);
2607 2350 : CPLFree(xyz);
2608 2350 : CPLFree(pabSuccess);
2609 :
2610 2350 : assignSpatialReference(poCT->GetTargetCS());
2611 :
2612 2350 : return OGRERR_NONE;
2613 : }
2614 :
2615 : /************************************************************************/
2616 : /* IsEmpty() */
2617 : /************************************************************************/
2618 :
2619 7771530 : OGRBoolean OGRSimpleCurve::IsEmpty() const
2620 : {
2621 7771530 : return (nPointCount == 0);
2622 : }
2623 :
2624 : /************************************************************************/
2625 : /* OGRSimpleCurve::segmentize() */
2626 : /************************************************************************/
2627 :
2628 63 : bool OGRSimpleCurve::segmentize(double dfMaxLength)
2629 : {
2630 63 : if (dfMaxLength <= 0)
2631 : {
2632 0 : CPLError(CE_Failure, CPLE_AppDefined,
2633 : "dfMaxLength must be strictly positive");
2634 0 : return false;
2635 : }
2636 63 : if (nPointCount < 2)
2637 0 : return true;
2638 :
2639 : // So as to make sure that the same line followed in both directions
2640 : // result in the same segmentized line.
2641 63 : if (paoPoints[0].x < paoPoints[nPointCount - 1].x ||
2642 56 : (paoPoints[0].x == paoPoints[nPointCount - 1].x &&
2643 48 : paoPoints[0].y < paoPoints[nPointCount - 1].y))
2644 : {
2645 13 : reversePoints();
2646 13 : bool bRet = segmentize(dfMaxLength);
2647 13 : reversePoints();
2648 13 : return bRet;
2649 : }
2650 :
2651 50 : int nNewPointCount = 0;
2652 50 : const double dfSquareMaxLength = dfMaxLength * dfMaxLength;
2653 :
2654 : // First pass to compute new number of points
2655 50 : constexpr double REL_EPSILON_LENGTH_SQUARE = 1e-5;
2656 50 : constexpr double REL_EPSILON_ROUND = 1e-2;
2657 707 : for (int i = 0; i < nPointCount; i++)
2658 : {
2659 707 : nNewPointCount++;
2660 :
2661 707 : if (i == nPointCount - 1)
2662 49 : break;
2663 :
2664 : // Must be kept in sync with the second pass loop
2665 658 : const double dfX = paoPoints[i + 1].x - paoPoints[i].x;
2666 658 : const double dfY = paoPoints[i + 1].y - paoPoints[i].y;
2667 658 : const double dfSquareDist = dfX * dfX + dfY * dfY;
2668 658 : if (dfSquareDist - dfSquareMaxLength >
2669 658 : REL_EPSILON_LENGTH_SQUARE * dfSquareMaxLength)
2670 : {
2671 285 : const double dfIntermediatePoints = floor(
2672 285 : sqrt(dfSquareDist / dfSquareMaxLength) - REL_EPSILON_ROUND);
2673 : const int nIntermediatePoints =
2674 285 : DoubleToIntClamp(dfIntermediatePoints);
2675 :
2676 : // TODO(schwehr): Can these be tighter?
2677 : // Limit allocation of paoNewPoints to a few GB of memory.
2678 : // An OGRRawPoint is 2 doubles.
2679 : // kMax is a guess of what a reasonable max might be.
2680 285 : constexpr int kMax = 2 << 26;
2681 285 : if (nNewPointCount > kMax || nIntermediatePoints > kMax)
2682 : {
2683 1 : CPLError(CE_Failure, CPLE_AppDefined,
2684 : "Too many points in a segment: %d or %d",
2685 : nNewPointCount, nIntermediatePoints);
2686 1 : return false;
2687 : }
2688 :
2689 284 : nNewPointCount += nIntermediatePoints;
2690 : }
2691 : }
2692 :
2693 49 : if (nPointCount == nNewPointCount)
2694 8 : return true;
2695 :
2696 : // Allocate new arrays
2697 : OGRRawPoint *paoNewPoints = static_cast<OGRRawPoint *>(
2698 41 : VSI_MALLOC_VERBOSE(sizeof(OGRRawPoint) * nNewPointCount));
2699 41 : if (paoNewPoints == nullptr)
2700 0 : return false;
2701 41 : double *padfNewZ = nullptr;
2702 41 : double *padfNewM = nullptr;
2703 41 : if (padfZ != nullptr)
2704 : {
2705 : padfNewZ = static_cast<double *>(
2706 2 : VSI_MALLOC_VERBOSE(sizeof(double) * nNewPointCount));
2707 2 : if (padfNewZ == nullptr)
2708 : {
2709 0 : VSIFree(paoNewPoints);
2710 0 : return false;
2711 : }
2712 : }
2713 41 : if (padfM != nullptr)
2714 : {
2715 : padfNewM = static_cast<double *>(
2716 2 : VSI_MALLOC_VERBOSE(sizeof(double) * nNewPointCount));
2717 2 : if (padfNewM == nullptr)
2718 : {
2719 0 : VSIFree(paoNewPoints);
2720 0 : VSIFree(padfNewZ);
2721 0 : return false;
2722 : }
2723 : }
2724 :
2725 : // Second pass to fill new arrays
2726 : // Must be kept in sync with the first pass loop
2727 41 : nNewPointCount = 0;
2728 606 : for (int i = 0; i < nPointCount; i++)
2729 : {
2730 606 : paoNewPoints[nNewPointCount] = paoPoints[i];
2731 :
2732 606 : if (padfZ != nullptr)
2733 : {
2734 4 : padfNewZ[nNewPointCount] = padfZ[i];
2735 : }
2736 :
2737 606 : if (padfM != nullptr)
2738 : {
2739 4 : padfNewM[nNewPointCount] = padfM[i];
2740 : }
2741 :
2742 606 : nNewPointCount++;
2743 :
2744 606 : if (i == nPointCount - 1)
2745 41 : break;
2746 :
2747 565 : const double dfX = paoPoints[i + 1].x - paoPoints[i].x;
2748 565 : const double dfY = paoPoints[i + 1].y - paoPoints[i].y;
2749 565 : const double dfSquareDist = dfX * dfX + dfY * dfY;
2750 :
2751 : // Must be kept in sync with the initial pass loop
2752 565 : if (dfSquareDist - dfSquareMaxLength >
2753 565 : REL_EPSILON_LENGTH_SQUARE * dfSquareMaxLength)
2754 : {
2755 282 : const double dfIntermediatePoints = floor(
2756 282 : sqrt(dfSquareDist / dfSquareMaxLength) - REL_EPSILON_ROUND);
2757 : const int nIntermediatePoints =
2758 282 : DoubleToIntClamp(dfIntermediatePoints);
2759 282 : const double dfRatioX =
2760 282 : dfX / (static_cast<double>(nIntermediatePoints) + 1);
2761 282 : const double dfRatioY =
2762 282 : dfY / (static_cast<double>(nIntermediatePoints) + 1);
2763 :
2764 21296 : for (int j = 1; j <= nIntermediatePoints; j++)
2765 : {
2766 : // coverity[overflow_const]
2767 21014 : const int newI = nNewPointCount + j - 1;
2768 21014 : paoNewPoints[newI].x = paoPoints[i].x + j * dfRatioX;
2769 21014 : paoNewPoints[newI].y = paoPoints[i].y + j * dfRatioY;
2770 21014 : if (padfZ != nullptr)
2771 : {
2772 : // No interpolation.
2773 10 : padfNewZ[newI] = padfZ[i];
2774 : }
2775 21014 : if (padfM != nullptr)
2776 : {
2777 : // No interpolation.
2778 2 : padfNewM[newI] = padfM[i];
2779 : }
2780 : }
2781 :
2782 282 : nNewPointCount += nIntermediatePoints;
2783 : }
2784 : }
2785 :
2786 41 : CPLFree(paoPoints);
2787 41 : paoPoints = paoNewPoints;
2788 41 : nPointCount = nNewPointCount;
2789 41 : m_nPointCapacity = nNewPointCount;
2790 :
2791 41 : if (padfZ != nullptr)
2792 : {
2793 2 : CPLFree(padfZ);
2794 2 : padfZ = padfNewZ;
2795 : }
2796 41 : if (padfM != nullptr)
2797 : {
2798 2 : CPLFree(padfM);
2799 2 : padfM = padfNewM;
2800 : }
2801 41 : return true;
2802 : }
2803 :
2804 : /************************************************************************/
2805 : /* swapXY() */
2806 : /************************************************************************/
2807 :
2808 102 : void OGRSimpleCurve::swapXY()
2809 : {
2810 652 : for (int i = 0; i < nPointCount; i++)
2811 : {
2812 550 : std::swap(paoPoints[i].x, paoPoints[i].y);
2813 : }
2814 102 : }
2815 :
2816 : /************************************************************************/
2817 : /* OGRSimpleCurvePointIterator */
2818 : /************************************************************************/
2819 :
2820 : class OGRSimpleCurvePointIterator final : public OGRPointIterator
2821 : {
2822 : CPL_DISALLOW_COPY_ASSIGN(OGRSimpleCurvePointIterator)
2823 :
2824 : const OGRSimpleCurve *poSC = nullptr;
2825 : int iCurPoint = 0;
2826 :
2827 : public:
2828 169 : explicit OGRSimpleCurvePointIterator(const OGRSimpleCurve *poSCIn)
2829 169 : : poSC(poSCIn)
2830 : {
2831 169 : }
2832 :
2833 : OGRBoolean getNextPoint(OGRPoint *p) override;
2834 : };
2835 :
2836 : /************************************************************************/
2837 : /* getNextPoint() */
2838 : /************************************************************************/
2839 :
2840 697 : OGRBoolean OGRSimpleCurvePointIterator::getNextPoint(OGRPoint *p)
2841 : {
2842 697 : if (iCurPoint >= poSC->getNumPoints())
2843 133 : return FALSE;
2844 564 : poSC->getPoint(iCurPoint, p);
2845 564 : iCurPoint++;
2846 564 : return TRUE;
2847 : }
2848 :
2849 : /************************************************************************/
2850 : /* getPointIterator() */
2851 : /************************************************************************/
2852 :
2853 169 : OGRPointIterator *OGRSimpleCurve::getPointIterator() const
2854 : {
2855 169 : return new OGRSimpleCurvePointIterator(this);
2856 : }
2857 :
2858 : /************************************************************************/
2859 : /* OGRLineString( const OGRLineString& ) */
2860 : /************************************************************************/
2861 :
2862 : /**
2863 : * \brief Copy constructor.
2864 : *
2865 : * Note: before GDAL 2.1, only the default implementation of the constructor
2866 : * existed, which could be unsafe to use.
2867 : *
2868 : * @since GDAL 2.1
2869 : */
2870 :
2871 : OGRLineString::OGRLineString(const OGRLineString &) = default;
2872 :
2873 : /************************************************************************/
2874 : /* OGRLineString( OGRLineString&& ) */
2875 : /************************************************************************/
2876 :
2877 : /**
2878 : * \brief Move constructor.
2879 : *
2880 : * @since GDAL 3.11
2881 : */
2882 :
2883 : OGRLineString::OGRLineString(OGRLineString &&) = default;
2884 :
2885 : /************************************************************************/
2886 : /* operator=( const OGRLineString& ) */
2887 : /************************************************************************/
2888 :
2889 : /**
2890 : * \brief Assignment operator.
2891 : *
2892 : * Note: before GDAL 2.1, only the default implementation of the operator
2893 : * existed, which could be unsafe to use.
2894 : *
2895 : * @since GDAL 2.1
2896 : */
2897 :
2898 9 : OGRLineString &OGRLineString::operator=(const OGRLineString &other)
2899 : {
2900 9 : if (this != &other)
2901 : {
2902 8 : OGRSimpleCurve::operator=(other);
2903 : }
2904 9 : return *this;
2905 : }
2906 :
2907 : /************************************************************************/
2908 : /* operator=( OGRLineString&& ) */
2909 : /************************************************************************/
2910 :
2911 : /**
2912 : * \brief Move assignment operator.
2913 : *
2914 : * @since GDAL 3.11
2915 : */
2916 :
2917 4 : OGRLineString &OGRLineString::operator=(OGRLineString &&other)
2918 : {
2919 4 : if (this != &other)
2920 : {
2921 4 : OGRSimpleCurve::operator=(std::move(other));
2922 : }
2923 4 : return *this;
2924 : }
2925 :
2926 : /************************************************************************/
2927 : /* getGeometryType() */
2928 : /************************************************************************/
2929 :
2930 876354 : OGRwkbGeometryType OGRLineString::getGeometryType() const
2931 :
2932 : {
2933 876354 : if ((flags & OGR_G_3D) && (flags & OGR_G_MEASURED))
2934 44081 : return wkbLineStringZM;
2935 832273 : else if (flags & OGR_G_MEASURED)
2936 1188 : return wkbLineStringM;
2937 831085 : else if (flags & OGR_G_3D)
2938 147547 : return wkbLineString25D;
2939 : else
2940 683538 : return wkbLineString;
2941 : }
2942 :
2943 : /************************************************************************/
2944 : /* getGeometryName() */
2945 : /************************************************************************/
2946 :
2947 10581 : const char *OGRLineString::getGeometryName() const
2948 :
2949 : {
2950 10581 : return "LINESTRING";
2951 : }
2952 :
2953 : /************************************************************************/
2954 : /* curveToLine() */
2955 : /************************************************************************/
2956 :
2957 952 : OGRLineString *OGRLineString::CurveToLine(
2958 : CPL_UNUSED double /* dfMaxAngleStepSizeDegrees */,
2959 : CPL_UNUSED const char *const * /* papszOptions */) const
2960 : {
2961 952 : return clone();
2962 : }
2963 :
2964 : /************************************************************************/
2965 : /* get_LinearArea() */
2966 : /************************************************************************/
2967 :
2968 : /**
2969 : * \brief Compute area of ring / closed linestring.
2970 : *
2971 : * The area is computed according to Green's Theorem:
2972 : *
2973 : * Area is "Sum(x(i)*(y(i+1) - y(i-1)))/2" for i = 0 to pointCount-1,
2974 : * assuming the last point is a duplicate of the first.
2975 : *
2976 : * @return computed area.
2977 : */
2978 :
2979 61829 : double OGRSimpleCurve::get_LinearArea() const
2980 :
2981 : {
2982 123656 : if (nPointCount < 2 ||
2983 61827 : (WkbSize() != 0 && /* if not a linearring, check it is closed */
2984 60 : (paoPoints[0].x != paoPoints[nPointCount - 1].x ||
2985 59 : paoPoints[0].y != paoPoints[nPointCount - 1].y)))
2986 : {
2987 2 : return 0;
2988 : }
2989 :
2990 61827 : double dfAreaSum =
2991 61827 : paoPoints[0].x * (paoPoints[1].y - paoPoints[nPointCount - 1].y);
2992 :
2993 3372900 : for (int i = 1; i < nPointCount - 1; i++)
2994 : {
2995 3311080 : dfAreaSum += paoPoints[i].x * (paoPoints[i + 1].y - paoPoints[i - 1].y);
2996 : }
2997 :
2998 61827 : dfAreaSum += paoPoints[nPointCount - 1].x *
2999 61827 : (paoPoints[0].y - paoPoints[nPointCount - 2].y);
3000 :
3001 61827 : return 0.5 * fabs(dfAreaSum);
3002 : }
3003 :
3004 : /************************************************************************/
3005 : /* getCurveGeometry() */
3006 : /************************************************************************/
3007 :
3008 : OGRGeometry *
3009 3201 : OGRLineString::getCurveGeometry(const char *const *papszOptions) const
3010 : {
3011 3201 : return OGRGeometryFactory::curveFromLineString(this, papszOptions);
3012 : }
3013 :
3014 : /************************************************************************/
3015 : /* TransferMembersAndDestroy() */
3016 : /************************************************************************/
3017 : //! @cond Doxygen_Suppress
3018 590 : OGRLineString *OGRLineString::TransferMembersAndDestroy(OGRLineString *poSrc,
3019 : OGRLineString *poDst)
3020 : {
3021 590 : if (poSrc->Is3D())
3022 101 : poDst->flags |= OGR_G_3D;
3023 590 : if (poSrc->IsMeasured())
3024 47 : poDst->flags |= OGR_G_MEASURED;
3025 590 : poDst->assignSpatialReference(poSrc->getSpatialReference());
3026 590 : poDst->nPointCount = poSrc->nPointCount;
3027 590 : poDst->m_nPointCapacity = poSrc->m_nPointCapacity;
3028 590 : poDst->paoPoints = poSrc->paoPoints;
3029 590 : poDst->padfZ = poSrc->padfZ;
3030 590 : poDst->padfM = poSrc->padfM;
3031 590 : poSrc->nPointCount = 0;
3032 590 : poSrc->m_nPointCapacity = 0;
3033 590 : poSrc->paoPoints = nullptr;
3034 590 : poSrc->padfZ = nullptr;
3035 590 : poSrc->padfM = nullptr;
3036 590 : delete poSrc;
3037 590 : return poDst;
3038 : }
3039 :
3040 : //! @endcond
3041 : /************************************************************************/
3042 : /* CastToLinearRing() */
3043 : /************************************************************************/
3044 :
3045 : /**
3046 : * \brief Cast to linear ring.
3047 : *
3048 : * The passed in geometry is consumed and a new one returned (or NULL in case
3049 : * of failure)
3050 : *
3051 : * @param poLS the input geometry - ownership is passed to the method.
3052 : * @return new geometry.
3053 : */
3054 :
3055 526 : OGRLinearRing *OGRLineString::CastToLinearRing(OGRLineString *poLS)
3056 : {
3057 526 : if (poLS->nPointCount < 2 || !poLS->get_IsClosed())
3058 : {
3059 2 : CPLError(CE_Failure, CPLE_AppDefined,
3060 : "Cannot convert non-closed linestring to linearring");
3061 2 : delete poLS;
3062 2 : return nullptr;
3063 : }
3064 524 : OGRLinearRing *poLR = new OGRLinearRing();
3065 524 : TransferMembersAndDestroy(poLS, poLR);
3066 524 : return poLR;
3067 : }
3068 :
3069 : /************************************************************************/
3070 : /* clone() */
3071 : /************************************************************************/
3072 :
3073 300838 : OGRLineString *OGRLineString::clone() const
3074 : {
3075 300838 : auto ret = new (std::nothrow) OGRLineString(*this);
3076 300838 : if (ret)
3077 : {
3078 300838 : if (ret->getNumPoints() != getNumPoints())
3079 : {
3080 0 : delete ret;
3081 0 : ret = nullptr;
3082 : }
3083 : }
3084 300838 : return ret;
3085 : }
3086 :
3087 : //! @cond Doxygen_Suppress
3088 :
3089 : /************************************************************************/
3090 : /* GetCasterToLineString() */
3091 : /************************************************************************/
3092 :
3093 156 : static OGRLineString *CasterToLineString(OGRCurve *poCurve)
3094 : {
3095 156 : return poCurve->toLineString();
3096 : }
3097 :
3098 156 : OGRCurveCasterToLineString OGRLineString::GetCasterToLineString() const
3099 : {
3100 156 : return ::CasterToLineString;
3101 : }
3102 :
3103 : /************************************************************************/
3104 : /* GetCasterToLinearRing() */
3105 : /************************************************************************/
3106 :
3107 526 : OGRLinearRing *OGRLineString::CasterToLinearRing(OGRCurve *poCurve)
3108 : {
3109 526 : return OGRLineString::CastToLinearRing(poCurve->toLineString());
3110 : }
3111 :
3112 526 : OGRCurveCasterToLinearRing OGRLineString::GetCasterToLinearRing() const
3113 : {
3114 526 : return OGRLineString::CasterToLinearRing;
3115 : }
3116 :
3117 : /************************************************************************/
3118 : /* get_Area() */
3119 : /************************************************************************/
3120 :
3121 61828 : double OGRLineString::get_Area() const
3122 : {
3123 61828 : return get_LinearArea();
3124 : }
3125 :
3126 : /************************************************************************/
3127 : /* GetGeodesicAreaOrLength() */
3128 : /************************************************************************/
3129 :
3130 44 : static bool GetGeodesicAreaOrLength(const OGRLineString *poLS,
3131 : const OGRSpatialReference *poSRSOverride,
3132 : double *pdfArea, double *pdfLength)
3133 : {
3134 44 : if (!poSRSOverride)
3135 11 : poSRSOverride = poLS->getSpatialReference();
3136 :
3137 44 : if (!poSRSOverride)
3138 : {
3139 2 : CPLError(CE_Failure, CPLE_AppDefined,
3140 : "Cannot compute %s on ellipsoid due to missing SRS",
3141 : pdfArea ? "area" : "length");
3142 2 : return false;
3143 : }
3144 :
3145 42 : OGRErr eErr = OGRERR_NONE;
3146 42 : double dfSemiMajor = poSRSOverride->GetSemiMajor(&eErr);
3147 42 : if (eErr != OGRERR_NONE)
3148 2 : return false;
3149 40 : const double dfInvFlattening = poSRSOverride->GetInvFlattening(&eErr);
3150 40 : if (eErr != OGRERR_NONE)
3151 0 : return false;
3152 :
3153 : geod_geodesic g;
3154 40 : geod_init(&g, dfSemiMajor,
3155 : dfInvFlattening != 0 ? 1.0 / dfInvFlattening : 0.0);
3156 :
3157 80 : std::vector<double> adfLat;
3158 80 : std::vector<double> adfLon;
3159 40 : const int nPointCount = poLS->getNumPoints();
3160 40 : adfLat.reserve(nPointCount);
3161 40 : adfLon.reserve(nPointCount);
3162 :
3163 80 : OGRSpatialReference oGeogCRS;
3164 40 : if (oGeogCRS.CopyGeogCSFrom(poSRSOverride) != OGRERR_NONE)
3165 : {
3166 0 : CPLError(CE_Failure, CPLE_AppDefined,
3167 : "Cannot reproject geometry to geographic CRS");
3168 0 : return false;
3169 : }
3170 40 : oGeogCRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
3171 : auto poCT = std::unique_ptr<OGRCoordinateTransformation>(
3172 80 : OGRCreateCoordinateTransformation(poSRSOverride, &oGeogCRS));
3173 40 : if (!poCT)
3174 : {
3175 0 : CPLError(CE_Failure, CPLE_AppDefined,
3176 : "Cannot reproject geometry to geographic CRS");
3177 0 : return false;
3178 : }
3179 896 : for (int i = 0; i < nPointCount; ++i)
3180 : {
3181 856 : adfLon.push_back(poLS->getX(i));
3182 856 : adfLat.push_back(poLS->getY(i));
3183 : }
3184 : #ifdef __GNUC__
3185 : #pragma GCC diagnostic push
3186 : #pragma GCC diagnostic ignored "-Wnull-dereference"
3187 : #endif
3188 80 : std::vector<int> anSuccess;
3189 40 : anSuccess.resize(adfLon.size());
3190 : #ifdef __GNUC__
3191 : #pragma GCC diagnostic pop
3192 : #endif
3193 40 : poCT->Transform(adfLon.size(), adfLon.data(), adfLat.data(), nullptr,
3194 : anSuccess.data());
3195 : double dfToDegrees =
3196 40 : oGeogCRS.GetAngularUnits(nullptr) / CPLAtof(SRS_UA_DEGREE_CONV);
3197 40 : if (std::fabs(dfToDegrees - 1) <= 1e-10)
3198 40 : dfToDegrees = 1.0;
3199 896 : for (int i = 0; i < nPointCount; ++i)
3200 : {
3201 856 : if (!anSuccess[i])
3202 : {
3203 0 : CPLError(CE_Failure, CPLE_AppDefined,
3204 : "Cannot reproject geometry to geographic CRS");
3205 0 : return false;
3206 : }
3207 856 : adfLon[i] *= dfToDegrees;
3208 856 : adfLat[i] *= dfToDegrees;
3209 : }
3210 :
3211 40 : geod_polygonarea(&g, adfLat.data(), adfLon.data(),
3212 40 : static_cast<int>(adfLat.size()), pdfArea, pdfLength);
3213 40 : return true;
3214 : }
3215 :
3216 : /************************************************************************/
3217 : /* get_GeodesicArea() */
3218 : /************************************************************************/
3219 :
3220 : double
3221 22 : OGRLineString::get_GeodesicArea(const OGRSpatialReference *poSRSOverride) const
3222 : {
3223 22 : double dfArea = 0;
3224 22 : if (!GetGeodesicAreaOrLength(this, poSRSOverride, &dfArea, nullptr))
3225 2 : return -1.0;
3226 20 : return std::fabs(dfArea);
3227 : }
3228 :
3229 : /************************************************************************/
3230 : /* get_GeodesicLength() */
3231 : /************************************************************************/
3232 :
3233 22 : double OGRLineString::get_GeodesicLength(
3234 : const OGRSpatialReference *poSRSOverride) const
3235 : {
3236 22 : double dfLength = 0;
3237 22 : if (!GetGeodesicAreaOrLength(this, poSRSOverride, nullptr, &dfLength))
3238 2 : return -1;
3239 20 : return dfLength;
3240 : }
3241 :
3242 : /************************************************************************/
3243 : /* get_AreaOfCurveSegments() */
3244 : /************************************************************************/
3245 :
3246 31 : double OGRLineString::get_AreaOfCurveSegments() const
3247 : {
3248 31 : return 0;
3249 : }
3250 :
3251 : /************************************************************************/
3252 : /* isClockwise() */
3253 : /************************************************************************/
3254 :
3255 : /**
3256 : * \brief Returns TRUE if the ring has clockwise winding (or less than 2 points)
3257 : *
3258 : * Assumes that the line is closed.
3259 : *
3260 : * @return TRUE if clockwise otherwise FALSE.
3261 : */
3262 :
3263 79880 : int OGRLineString::isClockwise() const
3264 :
3265 : {
3266 : // WARNING: keep in sync OGRLineString::isClockwise(),
3267 : // OGRCurve::isClockwise() and OGRWKBIsClockwiseRing()
3268 :
3269 79880 : if (nPointCount < 2)
3270 1 : return TRUE;
3271 :
3272 79879 : bool bUseFallback = false;
3273 :
3274 : // Find the lowest rightmost vertex.
3275 79879 : int v = 0; // Used after for.
3276 3809150 : for (int i = 1; i < nPointCount - 1; i++)
3277 : {
3278 : // => v < end.
3279 3729270 : if (paoPoints[i].y < paoPoints[v].y ||
3280 3242680 : (paoPoints[i].y == paoPoints[v].y &&
3281 55976 : paoPoints[i].x > paoPoints[v].x))
3282 : {
3283 515811 : v = i;
3284 515811 : bUseFallback = false;
3285 : }
3286 3213460 : else if (paoPoints[i].y == paoPoints[v].y &&
3287 26747 : paoPoints[i].x == paoPoints[v].x)
3288 : {
3289 : // Two vertex with same coordinates are the lowest rightmost
3290 : // vertex. Cannot use that point as the pivot (#5342).
3291 36 : bUseFallback = true;
3292 : }
3293 : }
3294 :
3295 : // Previous.
3296 79879 : int next = v - 1;
3297 79879 : if (next < 0)
3298 : {
3299 28190 : next = nPointCount - 1 - 1;
3300 : }
3301 :
3302 79879 : constexpr double EPSILON = 1.0E-5;
3303 177768 : const auto epsilonEqual = [](double a, double b, double eps)
3304 177768 : { return ::fabs(a - b) < eps; };
3305 :
3306 96197 : if (epsilonEqual(paoPoints[next].x, paoPoints[v].x, EPSILON) &&
3307 16316 : epsilonEqual(paoPoints[next].y, paoPoints[v].y, EPSILON))
3308 : {
3309 : // Don't try to be too clever by retrying with a next point.
3310 : // This can lead to false results as in the case of #3356.
3311 227 : bUseFallback = true;
3312 : }
3313 :
3314 79840 : const double dx0 = paoPoints[next].x - paoPoints[v].x;
3315 79840 : const double dy0 = paoPoints[next].y - paoPoints[v].y;
3316 :
3317 : // Following.
3318 79840 : next = v + 1;
3319 79840 : if (next >= nPointCount - 1)
3320 : {
3321 18004 : next = 0;
3322 : }
3323 :
3324 81595 : if (epsilonEqual(paoPoints[next].x, paoPoints[v].x, EPSILON) &&
3325 1750 : epsilonEqual(paoPoints[next].y, paoPoints[v].y, EPSILON))
3326 : {
3327 : // Don't try to be too clever by retrying with a next point.
3328 : // This can lead to false results as in the case of #3356.
3329 268 : bUseFallback = true;
3330 : }
3331 :
3332 79851 : const double dx1 = paoPoints[next].x - paoPoints[v].x;
3333 79851 : const double dy1 = paoPoints[next].y - paoPoints[v].y;
3334 :
3335 79851 : const double crossproduct = dx1 * dy0 - dx0 * dy1;
3336 :
3337 79851 : if (!bUseFallback)
3338 : {
3339 79480 : if (crossproduct > 0) // CCW
3340 44786 : return FALSE;
3341 34694 : else if (crossproduct < 0) // CW
3342 34659 : return TRUE;
3343 : }
3344 :
3345 : // This is a degenerate case: the extent of the polygon is less than EPSILON
3346 : // or 2 nearly identical points were found.
3347 : // Try with Green Formula as a fallback, but this is not a guarantee
3348 : // as we'll probably be affected by numerical instabilities.
3349 :
3350 406 : double dfSum =
3351 406 : paoPoints[0].x * (paoPoints[1].y - paoPoints[nPointCount - 1].y);
3352 :
3353 46300 : for (int i = 1; i < nPointCount - 1; i++)
3354 : {
3355 45894 : dfSum += paoPoints[i].x * (paoPoints[i + 1].y - paoPoints[i - 1].y);
3356 : }
3357 :
3358 406 : dfSum += paoPoints[nPointCount - 1].x *
3359 406 : (paoPoints[0].y - paoPoints[nPointCount - 2].y);
3360 :
3361 406 : return dfSum < 0;
3362 : }
3363 :
3364 : //! @endcond
|