Line data Source code
1 : /******************************************************************************
2 : *
3 : * Project: OpenGIS Simple Features Reference Implementation
4 : * Purpose: Implements decoder of shapebin geometry for PGeo
5 : * Author: Frank Warmerdam, warmerdam@pobox.com
6 : * Paul Ramsey, pramsey at cleverelephant.ca
7 : *
8 : ******************************************************************************
9 : * Copyright (c) 2005, Frank Warmerdam <warmerdam@pobox.com>
10 : * Copyright (c) 2011, Paul Ramsey <pramsey at cleverelephant.ca>
11 : * Copyright (c) 2011-2014, Even Rouault <even dot rouault at spatialys.com>
12 : *
13 : * SPDX-License-Identifier: MIT
14 : ****************************************************************************/
15 :
16 : // PGeo == ESRI Personal GeoDatabase.
17 :
18 : #include "cpl_port.h"
19 : #include "ogrpgeogeometry.h"
20 :
21 : #include <algorithm>
22 : #include <cassert>
23 : #include <cmath>
24 : #include <cstddef>
25 : #include <cstring>
26 : #include <limits>
27 : #include <map>
28 : #include <memory>
29 : #include <set>
30 : #include <utility>
31 : #include <vector>
32 :
33 : #include "cpl_conv.h"
34 : #include "cpl_error.h"
35 : #include "cpl_string.h"
36 : #include "cpl_vsi.h"
37 : #include "ogr_api.h"
38 : #include "ogr_core.h"
39 : #include "ogr_p.h"
40 :
41 : constexpr int SHPP_TRISTRIP = 0;
42 : constexpr int SHPP_TRIFAN = 1;
43 : constexpr int SHPP_OUTERRING = 2;
44 : constexpr int SHPP_INNERRING = 3;
45 : constexpr int SHPP_FIRSTRING = 4;
46 : constexpr int SHPP_RING = 5;
47 : constexpr int SHPP_TRIANGLES = 6; // Multipatch 9.0 specific.
48 :
49 : enum CurveType
50 : {
51 : CURVE_ARC_INTERIOR_POINT,
52 : CURVE_ARC_CENTER_POINT,
53 : CURVE_BEZIER,
54 : CURVE_ELLIPSE_BY_CENTER
55 : };
56 :
57 : namespace
58 : {
59 : struct CurveSegment
60 : {
61 : int nStartPointIdx;
62 : CurveType eType;
63 :
64 : union
65 : {
66 : // Arc defined by an intermediate point.
67 : struct
68 : {
69 : double dfX;
70 : double dfY;
71 : } ArcByIntermediatePoint;
72 :
73 : // Deprecated way of defining circular arc by its center and
74 : // winding order.
75 : struct
76 : {
77 : double dfX;
78 : double dfY;
79 : bool bIsCCW;
80 : } ArcByCenterPoint;
81 :
82 : struct
83 : {
84 : double dfX1;
85 : double dfY1;
86 : double dfX2;
87 : double dfY2;
88 : } Bezier;
89 :
90 : struct
91 : {
92 : double dfX;
93 : double dfY;
94 : double dfRotationDeg;
95 : double dfSemiMajor;
96 : double dfRatioSemiMinor;
97 : bool bIsMinor;
98 : bool bIsComplete;
99 : } EllipseByCenter;
100 : } u;
101 : };
102 : } /* namespace */
103 :
104 : constexpr int EXT_SHAPE_SEGMENT_ARC = 1;
105 : constexpr int EXT_SHAPE_SEGMENT_BEZIER = 4;
106 : constexpr int EXT_SHAPE_SEGMENT_ELLIPSE = 5;
107 :
108 : constexpr int EXT_SHAPE_ARC_EMPTY = 0x1;
109 : constexpr int EXT_SHAPE_ARC_CCW = 0x8;
110 : #ifdef DEBUG_VERBOSE
111 : constexpr int EXT_SHAPE_ARC_MINOR = 0x10;
112 : #endif
113 : constexpr int EXT_SHAPE_ARC_LINE = 0x20;
114 : constexpr int EXT_SHAPE_ARC_POINT = 0x40;
115 : constexpr int EXT_SHAPE_ARC_IP = 0x80;
116 :
117 : #ifdef DEBUG_VERBOSE
118 : constexpr int EXT_SHAPE_ELLIPSE_EMPTY = 0x1;
119 : constexpr int EXT_SHAPE_ELLIPSE_LINE = 0x40;
120 : constexpr int EXT_SHAPE_ELLIPSE_POINT = 0x80;
121 : constexpr int EXT_SHAPE_ELLIPSE_CIRCULAR = 0x100;
122 : constexpr int EXT_SHAPE_ELLIPSE_CCW = 0x800;
123 : #endif
124 :
125 : constexpr int EXT_SHAPE_ELLIPSE_CENTER_TO = 0x200;
126 : constexpr int EXT_SHAPE_ELLIPSE_CENTER_FROM = 0x400;
127 : constexpr int EXT_SHAPE_ELLIPSE_MINOR = 0x1000;
128 : constexpr int EXT_SHAPE_ELLIPSE_COMPLETE = 0x2000;
129 :
130 : /************************************************************************/
131 : /* OGRCreateFromMultiPatchPart() */
132 : /************************************************************************/
133 :
134 3475 : static void OGRCreateFromMultiPatchPart(OGRGeometryCollection *poGC,
135 : OGRMultiPolygon *&poMP,
136 : OGRPolygon *&poLastPoly, int nPartType,
137 : int nPartPoints, const double *padfX,
138 : const double *padfY,
139 : const double *padfZ)
140 : {
141 3475 : nPartType &= 0xf;
142 :
143 3475 : if (nPartType == SHPP_TRISTRIP)
144 : {
145 783 : if (poMP != nullptr && poLastPoly != nullptr)
146 : {
147 0 : poMP->addGeometryDirectly(poLastPoly);
148 0 : poLastPoly = nullptr;
149 : }
150 :
151 783 : OGRTriangulatedSurface *poTIN = new OGRTriangulatedSurface();
152 2353 : for (int iBaseVert = 0; iBaseVert < nPartPoints - 2; iBaseVert++)
153 : {
154 1570 : const int iSrcVert = iBaseVert;
155 :
156 3140 : OGRPoint oPoint1(padfX[iSrcVert], padfY[iSrcVert], padfZ[iSrcVert]);
157 :
158 1570 : OGRPoint oPoint2(padfX[iSrcVert + 1], padfY[iSrcVert + 1],
159 3140 : padfZ[iSrcVert + 1]);
160 :
161 1570 : OGRPoint oPoint3(padfX[iSrcVert + 2], padfY[iSrcVert + 2],
162 3140 : padfZ[iSrcVert + 2]);
163 :
164 : OGRTriangle *poTriangle =
165 1570 : new OGRTriangle(oPoint1, oPoint2, oPoint3);
166 :
167 1570 : poTIN->addGeometryDirectly(poTriangle);
168 : }
169 783 : poGC->addGeometryDirectly(poTIN);
170 : }
171 2692 : else if (nPartType == SHPP_TRIFAN)
172 : {
173 680 : if (poMP != nullptr && poLastPoly != nullptr)
174 : {
175 0 : poMP->addGeometryDirectly(poLastPoly);
176 0 : poLastPoly = nullptr;
177 : }
178 :
179 680 : OGRTriangulatedSurface *poTIN = new OGRTriangulatedSurface();
180 2035 : for (int iBaseVert = 0; iBaseVert < nPartPoints - 2; iBaseVert++)
181 : {
182 1355 : const int iSrcVert = iBaseVert;
183 :
184 2710 : OGRPoint oPoint1(padfX[0], padfY[0], padfZ[0]);
185 :
186 1355 : OGRPoint oPoint2(padfX[iSrcVert + 1], padfY[iSrcVert + 1],
187 2710 : padfZ[iSrcVert + 1]);
188 :
189 1355 : OGRPoint oPoint3(padfX[iSrcVert + 2], padfY[iSrcVert + 2],
190 2710 : padfZ[iSrcVert + 2]);
191 :
192 : OGRTriangle *poTriangle =
193 1355 : new OGRTriangle(oPoint1, oPoint2, oPoint3);
194 :
195 1355 : poTIN->addGeometryDirectly(poTriangle);
196 : }
197 680 : poGC->addGeometryDirectly(poTIN);
198 : }
199 2012 : else if (nPartType == SHPP_OUTERRING || nPartType == SHPP_INNERRING ||
200 671 : nPartType == SHPP_FIRSTRING || nPartType == SHPP_RING)
201 : {
202 1341 : if (poMP == nullptr)
203 : {
204 670 : poMP = new OGRMultiPolygon();
205 : }
206 :
207 1341 : if (poMP != nullptr && poLastPoly != nullptr &&
208 671 : (nPartType == SHPP_OUTERRING || nPartType == SHPP_FIRSTRING))
209 : {
210 0 : poMP->addGeometryDirectly(poLastPoly);
211 0 : poLastPoly = nullptr;
212 : }
213 :
214 1341 : if (poLastPoly == nullptr)
215 670 : poLastPoly = new OGRPolygon();
216 :
217 1341 : OGRLinearRing *poRing = new OGRLinearRing;
218 :
219 1341 : poRing->setPoints(nPartPoints, const_cast<double *>(padfX),
220 : const_cast<double *>(padfY),
221 : const_cast<double *>(padfZ));
222 :
223 1341 : poRing->closeRings();
224 :
225 1341 : poLastPoly->addRingDirectly(poRing);
226 : }
227 671 : else if (nPartType == SHPP_TRIANGLES)
228 : {
229 671 : if (poMP != nullptr && poLastPoly != nullptr)
230 : {
231 0 : poMP->addGeometryDirectly(poLastPoly);
232 0 : poLastPoly = nullptr;
233 : }
234 :
235 671 : OGRTriangulatedSurface *poTIN = new OGRTriangulatedSurface();
236 1342 : for (int iBaseVert = 0; iBaseVert < nPartPoints - 2; iBaseVert += 3)
237 : {
238 671 : const int iSrcVert = iBaseVert;
239 :
240 1342 : OGRPoint oPoint1(padfX[iSrcVert], padfY[iSrcVert], padfZ[iSrcVert]);
241 :
242 671 : OGRPoint oPoint2(padfX[iSrcVert + 1], padfY[iSrcVert + 1],
243 1342 : padfZ[iSrcVert + 1]);
244 :
245 671 : OGRPoint oPoint3(padfX[iSrcVert + 2], padfY[iSrcVert + 2],
246 1342 : padfZ[iSrcVert + 2]);
247 :
248 : OGRTriangle *poTriangle =
249 671 : new OGRTriangle(oPoint1, oPoint2, oPoint3);
250 :
251 671 : poTIN->addGeometryDirectly(poTriangle);
252 : }
253 671 : poGC->addGeometryDirectly(poTIN);
254 : }
255 : else
256 0 : CPLDebug("OGR", "Unrecognized parttype %d, ignored.", nPartType);
257 3475 : }
258 :
259 : static bool
260 12 : RegisterEdge(const double *padfX, const double *padfY, const double *padfZ,
261 : int nPart,
262 : std::map<std::vector<double>, std::pair<int, int>> &oMapEdges)
263 : {
264 12 : int idx = 0;
265 12 : if (padfX[0] > padfX[1])
266 : {
267 6 : idx = 1;
268 : }
269 6 : else if (padfX[0] == padfX[1])
270 : {
271 2 : if (padfY[0] > padfY[1])
272 : {
273 0 : idx = 1;
274 : }
275 2 : else if (padfY[0] == padfY[1])
276 : {
277 0 : if (padfZ[0] > padfZ[1])
278 : {
279 0 : idx = 1;
280 : }
281 : }
282 : }
283 24 : std::vector<double> oVector;
284 12 : oVector.push_back(padfX[idx]);
285 12 : oVector.push_back(padfY[idx]);
286 12 : oVector.push_back(padfZ[idx]);
287 12 : oVector.push_back(padfX[1 - idx]);
288 12 : oVector.push_back(padfY[1 - idx]);
289 12 : oVector.push_back(padfZ[1 - idx]);
290 12 : const auto oIter = oMapEdges.find(oVector);
291 12 : if (oIter == oMapEdges.end())
292 : {
293 10 : oMapEdges[oVector] = std::pair(nPart, -1);
294 : }
295 : else
296 : {
297 2 : CPLAssert(oIter->second.first >= 0);
298 2 : if (oIter->second.second < 0)
299 2 : oIter->second.second = nPart;
300 : else
301 0 : return false;
302 : }
303 12 : return true;
304 : }
305 :
306 12 : static const std::pair<int, int> &GetEdgeOwners(
307 : const double *padfX, const double *padfY, const double *padfZ,
308 : const std::map<std::vector<double>, std::pair<int, int>> &oMapEdges)
309 : {
310 12 : int idx = 0;
311 12 : if (padfX[0] > padfX[1])
312 : {
313 6 : idx = 1;
314 : }
315 6 : else if (padfX[0] == padfX[1])
316 : {
317 2 : if (padfY[0] > padfY[1])
318 : {
319 0 : idx = 1;
320 : }
321 2 : else if (padfY[0] == padfY[1])
322 : {
323 0 : if (padfZ[0] > padfZ[1])
324 : {
325 0 : idx = 1;
326 : }
327 : }
328 : }
329 12 : std::vector<double> oVector;
330 12 : oVector.push_back(padfX[idx]);
331 12 : oVector.push_back(padfY[idx]);
332 12 : oVector.push_back(padfZ[idx]);
333 12 : oVector.push_back(padfX[1 - idx]);
334 12 : oVector.push_back(padfY[1 - idx]);
335 12 : oVector.push_back(padfZ[1 - idx]);
336 12 : const auto oIter = oMapEdges.find(oVector);
337 12 : CPLAssert(oIter != oMapEdges.end());
338 24 : return oIter->second;
339 : }
340 :
341 : /************************************************************************/
342 : /* OGRCreateFromMultiPatch() */
343 : /* */
344 : /* Translate a multipatch representation to an OGR geometry */
345 : /************************************************************************/
346 :
347 795 : OGRGeometry *OGRCreateFromMultiPatch(int nParts, const GInt32 *panPartStart,
348 : const GInt32 *panPartType, int nPoints,
349 : const double *padfX, const double *padfY,
350 : const double *padfZ)
351 : {
352 : // Deal with particular case of a patch of OuterRing of 4 points
353 : // that form a TIN. And be robust to consecutive duplicated triangles !
354 1590 : std::map<std::vector<double>, std::pair<int, int>> oMapEdges;
355 795 : bool bTINCandidate = nParts >= 2;
356 1590 : std::set<int> oSetDuplicated;
357 800 : for (int iPart = 0; iPart < nParts && panPartStart != nullptr; iPart++)
358 : {
359 798 : int nPartPoints = 0;
360 :
361 : // Figure out details about this part's vertex list.
362 798 : if (iPart == nParts - 1)
363 123 : nPartPoints = nPoints - panPartStart[iPart];
364 : else
365 675 : nPartPoints = panPartStart[iPart + 1] - panPartStart[iPart];
366 798 : const int nPartStart = panPartStart[iPart];
367 :
368 5 : if (panPartType[iPart] == SHPP_OUTERRING && nPartPoints == 4 &&
369 5 : padfX[nPartStart] == padfX[nPartStart + 3] &&
370 5 : padfY[nPartStart] == padfY[nPartStart + 3] &&
371 5 : padfZ[nPartStart] == padfZ[nPartStart + 3] &&
372 5 : !std::isnan(padfX[nPartStart]) &&
373 5 : !std::isnan(padfX[nPartStart + 1]) &&
374 5 : !std::isnan(padfX[nPartStart + 2]) &&
375 5 : !std::isnan(padfY[nPartStart]) &&
376 5 : !std::isnan(padfY[nPartStart + 1]) &&
377 5 : !std::isnan(padfY[nPartStart + 2]) &&
378 5 : !std::isnan(padfZ[nPartStart]) &&
379 808 : !std::isnan(padfZ[nPartStart + 1]) &&
380 5 : !std::isnan(padfZ[nPartStart + 2]))
381 : {
382 5 : bool bDuplicate = false;
383 5 : if (iPart > 0)
384 : {
385 3 : bDuplicate = true;
386 3 : const int nPrevPartStart = panPartStart[iPart - 1];
387 6 : for (int j = 0; j < 3; j++)
388 : {
389 5 : if (padfX[nPartStart + j] == padfX[nPrevPartStart + j] &&
390 3 : padfY[nPartStart + j] == padfY[nPrevPartStart + j] &&
391 3 : padfZ[nPartStart + j] == padfZ[nPrevPartStart + j])
392 : {
393 : }
394 : else
395 : {
396 2 : bDuplicate = false;
397 2 : break;
398 : }
399 : }
400 : }
401 5 : if (bDuplicate)
402 : {
403 1 : oSetDuplicated.insert(iPart);
404 : }
405 12 : else if (RegisterEdge(padfX + nPartStart, padfY + nPartStart,
406 4 : padfZ + nPartStart, iPart, oMapEdges) &&
407 4 : RegisterEdge(padfX + nPartStart + 1,
408 4 : padfY + nPartStart + 1,
409 8 : padfZ + nPartStart + 1, iPart, oMapEdges) &&
410 4 : RegisterEdge(padfX + nPartStart + 2,
411 4 : padfY + nPartStart + 2,
412 4 : padfZ + nPartStart + 2, iPart, oMapEdges))
413 : {
414 : // ok
415 : }
416 : else
417 : {
418 0 : bTINCandidate = false;
419 0 : break;
420 : }
421 : }
422 : else
423 : {
424 793 : bTINCandidate = false;
425 793 : break;
426 : }
427 : }
428 795 : if (bTINCandidate && panPartStart != nullptr)
429 : {
430 2 : std::set<int> oVisitedParts;
431 2 : std::set<int> oToBeVisitedParts;
432 2 : oToBeVisitedParts.insert(0);
433 6 : while (!oToBeVisitedParts.empty())
434 : {
435 4 : const int iPart = *(oToBeVisitedParts.begin());
436 4 : oVisitedParts.insert(iPart);
437 4 : oToBeVisitedParts.erase(iPart);
438 :
439 4 : const int nPartStart = panPartStart[iPart];
440 16 : for (int j = 0; j < 3; j++)
441 : {
442 : const auto &oPair = GetEdgeOwners(
443 12 : padfX + nPartStart + j, padfY + nPartStart + j,
444 12 : padfZ + nPartStart + j, oMapEdges);
445 12 : const int iOtherPart =
446 12 : (oPair.first == iPart) ? oPair.second : oPair.first;
447 16 : if (iOtherPart >= 0 &&
448 16 : oVisitedParts.find(iOtherPart) == oVisitedParts.end())
449 : {
450 2 : oToBeVisitedParts.insert(iOtherPart);
451 : }
452 : }
453 : }
454 2 : if (static_cast<int>(oVisitedParts.size()) ==
455 2 : nParts - static_cast<int>(oSetDuplicated.size()))
456 : {
457 2 : OGRTriangulatedSurface *poTIN = new OGRTriangulatedSurface();
458 7 : for (int iPart = 0; iPart < nParts; iPart++)
459 : {
460 5 : if (oSetDuplicated.find(iPart) != oSetDuplicated.end())
461 1 : continue;
462 :
463 4 : const int nPartStart = panPartStart[iPart];
464 4 : OGRPoint oPoint1(padfX[nPartStart], padfY[nPartStart],
465 8 : padfZ[nPartStart]);
466 :
467 4 : OGRPoint oPoint2(padfX[nPartStart + 1], padfY[nPartStart + 1],
468 8 : padfZ[nPartStart + 1]);
469 :
470 4 : OGRPoint oPoint3(padfX[nPartStart + 2], padfY[nPartStart + 2],
471 8 : padfZ[nPartStart + 2]);
472 :
473 : OGRTriangle *poTriangle =
474 4 : new OGRTriangle(oPoint1, oPoint2, oPoint3);
475 :
476 4 : poTIN->addGeometryDirectly(poTriangle);
477 : }
478 2 : return poTIN;
479 : }
480 : }
481 :
482 793 : OGRGeometryCollection *poGC = new OGRGeometryCollection();
483 793 : OGRMultiPolygon *poMP = nullptr;
484 793 : OGRPolygon *poLastPoly = nullptr;
485 4268 : for (int iPart = 0; iPart < nParts; iPart++)
486 : {
487 3475 : int nPartPoints = 0;
488 3475 : int nPartStart = 0;
489 :
490 : // Figure out details about this part's vertex list.
491 3475 : if (panPartStart == nullptr)
492 : {
493 0 : nPartPoints = nPoints;
494 : }
495 : else
496 : {
497 :
498 3475 : if (iPart == nParts - 1)
499 793 : nPartPoints = nPoints - panPartStart[iPart];
500 : else
501 2682 : nPartPoints = panPartStart[iPart + 1] - panPartStart[iPart];
502 3475 : nPartStart = panPartStart[iPart];
503 : }
504 :
505 3475 : OGRCreateFromMultiPatchPart(poGC, poMP, poLastPoly, panPartType[iPart],
506 3475 : nPartPoints, padfX + nPartStart,
507 3475 : padfY + nPartStart, padfZ + nPartStart);
508 : }
509 :
510 793 : if (poMP != nullptr && poLastPoly != nullptr)
511 : {
512 670 : poMP->addGeometryDirectly(poLastPoly);
513 : // poLastPoly = NULL;
514 : }
515 793 : if (poMP != nullptr)
516 670 : poGC->addGeometryDirectly(poMP);
517 :
518 793 : if (poGC->getNumGeometries() == 1)
519 : {
520 121 : OGRGeometry *poResultGeom = poGC->getGeometryRef(0);
521 121 : poGC->removeGeometry(0, FALSE);
522 121 : delete poGC;
523 121 : return poResultGeom;
524 : }
525 :
526 : else
527 672 : return poGC;
528 : }
529 :
530 : /************************************************************************/
531 : /* OGRWriteToShapeBin() */
532 : /* */
533 : /* Translate OGR geometry to a shapefile binary representation */
534 : /************************************************************************/
535 :
536 1414 : OGRErr OGRWriteToShapeBin(const OGRGeometry *poGeom, GByte **ppabyShape,
537 : int *pnBytes)
538 : {
539 1414 : int nShpSize = 4; // All types start with integer type number.
540 :
541 : /* -------------------------------------------------------------------- */
542 : /* Null or Empty input maps to SHPT_NULL. */
543 : /* -------------------------------------------------------------------- */
544 1414 : if (!poGeom || poGeom->IsEmpty())
545 : {
546 0 : *ppabyShape = static_cast<GByte *>(VSI_MALLOC_VERBOSE(nShpSize));
547 0 : if (*ppabyShape == nullptr)
548 0 : return OGRERR_FAILURE;
549 0 : GUInt32 zero = SHPT_NULL;
550 0 : memcpy(*ppabyShape, &zero, nShpSize);
551 0 : *pnBytes = nShpSize;
552 0 : return OGRERR_NONE;
553 : }
554 :
555 1414 : const OGRwkbGeometryType nOGRType = wkbFlatten(poGeom->getGeometryType());
556 1414 : const bool b3d = wkbHasZ(poGeom->getGeometryType());
557 1414 : const bool bHasM = wkbHasM(poGeom->getGeometryType());
558 1414 : const int nCoordDims = poGeom->CoordinateDimension();
559 :
560 1414 : int nShpZSize = 0; // Z gets tacked onto the end.
561 1414 : GUInt32 nPoints = 0;
562 1414 : GUInt32 nParts = 0;
563 :
564 : /* -------------------------------------------------------------------- */
565 : /* Calculate the shape buffer size */
566 : /* -------------------------------------------------------------------- */
567 1414 : if (nOGRType == wkbPoint)
568 : {
569 1064 : nShpSize += 8 * nCoordDims;
570 : }
571 350 : else if (nOGRType == wkbLineString)
572 : {
573 61 : const OGRLineString *poLine = poGeom->toLineString();
574 61 : nPoints = poLine->getNumPoints();
575 61 : nParts = 1;
576 61 : nShpSize += 16 * nCoordDims; // xy(z)(m) box.
577 61 : nShpSize += 4; // nparts.
578 61 : nShpSize += 4; // npoints.
579 61 : nShpSize += 4; // Parts[1].
580 61 : nShpSize += 8 * nCoordDims * nPoints; // Points.
581 61 : nShpZSize = 16 + 8 * nPoints;
582 : }
583 289 : else if (nOGRType == wkbPolygon)
584 : {
585 61 : std::unique_ptr<OGRPolygon> poPoly(poGeom->toPolygon()->clone());
586 61 : poPoly->closeRings();
587 61 : nParts = poPoly->getNumInteriorRings() + 1;
588 123 : for (const auto poRing : *poPoly)
589 : {
590 62 : nPoints += poRing->getNumPoints();
591 : }
592 61 : nShpSize += 16 * nCoordDims; // xy(z)(m) box.
593 61 : nShpSize += 4; // nparts.
594 61 : nShpSize += 4; // npoints.
595 61 : nShpSize += 4 * nParts; // parts[nparts]
596 61 : nShpSize += 8 * nCoordDims * nPoints; // Points.
597 61 : nShpZSize = 16 + 8 * nPoints;
598 : }
599 228 : else if (nOGRType == wkbMultiPoint)
600 : {
601 60 : const OGRMultiPoint *poMPoint = poGeom->toMultiPoint();
602 180 : for (const auto poPoint : *poMPoint)
603 : {
604 120 : if (poPoint->IsEmpty())
605 0 : continue;
606 120 : nPoints++;
607 : }
608 60 : nShpSize += 16 * nCoordDims; // xy(z)(m) box.
609 60 : nShpSize += 4; // npoints.
610 60 : nShpSize += 8 * nCoordDims * nPoints; // Points.
611 60 : nShpZSize = 16 + 8 * nPoints;
612 : }
613 168 : else if (nOGRType == wkbMultiLineString)
614 : {
615 60 : const OGRMultiLineString *poMLine = poGeom->toMultiLineString();
616 150 : for (const auto poLine : *poMLine)
617 : {
618 : // Skip empties.
619 90 : if (poLine->IsEmpty())
620 0 : continue;
621 90 : nParts++;
622 90 : nPoints += poLine->getNumPoints();
623 : }
624 60 : nShpSize += 16 * nCoordDims; //* xy(z)(m) box.
625 60 : nShpSize += 4; // nparts.
626 60 : nShpSize += 4; // npoints.
627 60 : nShpSize += 4 * nParts; // parts[nparts].
628 60 : nShpSize += 8 * nCoordDims * nPoints; // Points.
629 60 : nShpZSize = 16 + 8 * nPoints;
630 : }
631 108 : else if (nOGRType == wkbMultiPolygon)
632 : {
633 : std::unique_ptr<OGRMultiPolygon> poMPoly(
634 108 : poGeom->toMultiPolygon()->clone());
635 108 : poMPoly->closeRings();
636 246 : for (const auto poPoly : *poMPoly)
637 : {
638 : // Skip empties.
639 138 : if (poPoly->IsEmpty())
640 0 : continue;
641 :
642 138 : const int nRings = poPoly->getNumInteriorRings() + 1;
643 138 : nParts += nRings;
644 306 : for (const auto poRing : *poPoly)
645 : {
646 168 : nPoints += poRing->getNumPoints();
647 : }
648 : }
649 108 : nShpSize += 16 * nCoordDims; // xy(z)(m) box.
650 108 : nShpSize += 4; // nparts.
651 108 : nShpSize += 4; // npoints.
652 108 : nShpSize += 4 * nParts; // parts[nparts].
653 108 : nShpSize += 8 * nCoordDims * nPoints; // Points.
654 108 : nShpZSize = 16 + 8 * nPoints;
655 : }
656 : else
657 : {
658 0 : return OGRERR_UNSUPPORTED_OPERATION;
659 : }
660 :
661 : // #define WRITE_ARC_HACK
662 : #ifdef WRITE_ARC_HACK
663 : int nShpSizeBeforeCurve = nShpSize;
664 : nShpSize += 4 + 4 + 4 + 20;
665 : #endif
666 : // Allocate our shape buffer.
667 1414 : *ppabyShape = static_cast<GByte *>(VSI_MALLOC_VERBOSE(nShpSize));
668 1414 : if (!*ppabyShape)
669 0 : return OGRERR_FAILURE;
670 :
671 : #ifdef WRITE_ARC_HACK
672 : /* To be used with:
673 : id,WKT
674 : 1,"LINESTRING (1 0,0 1)"
675 : 2,"LINESTRING (5 1,6 0)"
676 : 3,"LINESTRING (10 1,11 0)"
677 : 4,"LINESTRING (16 0,15 1)"
678 : 5,"LINESTRING (21 0,20 1)"
679 : 6,"LINESTRING (31 0,30 2)" <-- not constant radius
680 : */
681 :
682 : GUInt32 nTmp = 1;
683 : memcpy((*ppabyShape) + nShpSizeBeforeCurve, &nTmp, 4);
684 : nTmp = 0;
685 : memcpy((*ppabyShape) + nShpSizeBeforeCurve + 4, &nTmp, 4);
686 : nTmp = EXT_SHAPE_SEGMENT_ARC;
687 : memcpy((*ppabyShape) + nShpSizeBeforeCurve + 8, &nTmp, 4);
688 : static int nCounter = 0;
689 : nCounter++;
690 : if (nCounter == 1)
691 : {
692 : double dfVal = 0;
693 : memcpy((*ppabyShape) + nShpSizeBeforeCurve + 8 + 4, &dfVal, 8);
694 : dfVal = 0;
695 : memcpy((*ppabyShape) + nShpSizeBeforeCurve + 8 + 4 + 8, &dfVal, 8);
696 : nTmp = 0;
697 : memcpy((*ppabyShape) + nShpSizeBeforeCurve + 8 + 4 + 16, &nTmp, 4);
698 : }
699 : else if (nCounter == 2)
700 : {
701 : double dfVal = 5;
702 : memcpy((*ppabyShape) + nShpSizeBeforeCurve + 8 + 4, &dfVal, 8);
703 : dfVal = 0;
704 : memcpy((*ppabyShape) + nShpSizeBeforeCurve + 8 + 4 + 8, &dfVal, 8);
705 : nTmp = EXT_SHAPE_ARC_MINOR;
706 : memcpy((*ppabyShape) + nShpSizeBeforeCurve + 8 + 4 + 16, &nTmp, 4);
707 : }
708 : else if (nCounter == 3)
709 : {
710 : double dfVal = 10;
711 : memcpy((*ppabyShape) + nShpSizeBeforeCurve + 8 + 4, &dfVal, 8);
712 : dfVal = 0;
713 : memcpy((*ppabyShape) + nShpSizeBeforeCurve + 8 + 4 + 8, &dfVal, 8);
714 : nTmp = EXT_SHAPE_ARC_CCW;
715 : memcpy((*ppabyShape) + nShpSizeBeforeCurve + 8 + 4 + 16, &nTmp, 4);
716 : }
717 : else if (nCounter == 4)
718 : {
719 : double dfVal = 15;
720 : memcpy((*ppabyShape) + nShpSizeBeforeCurve + 8 + 4, &dfVal, 8);
721 : dfVal = 0;
722 : memcpy((*ppabyShape) + nShpSizeBeforeCurve + 8 + 4 + 8, &dfVal, 8);
723 : nTmp = EXT_SHAPE_ARC_CCW | EXT_SHAPE_ARC_MINOR;
724 : memcpy((*ppabyShape) + nShpSizeBeforeCurve + 8 + 4 + 16, &nTmp, 4);
725 : }
726 : else if (nCounter == 5)
727 : {
728 : double dfVal = 20;
729 : memcpy((*ppabyShape) + nShpSizeBeforeCurve + 8 + 4, &dfVal, 8);
730 : dfVal = 0;
731 : memcpy((*ppabyShape) + nShpSizeBeforeCurve + 8 + 4 + 8, &dfVal, 8);
732 : // Inconsistent with SP and EP. Only the CCW/not CCW is taken into
733 : // account by ArcGIS.
734 : nTmp = EXT_SHAPE_ARC_MINOR;
735 : memcpy((*ppabyShape) + nShpSizeBeforeCurve + 8 + 4 + 16, &nTmp, 4);
736 : }
737 : else if (nCounter == 6)
738 : {
739 : double dfVal = 30; // Radius inconsistent with SP and EP.
740 : memcpy((*ppabyShape) + nShpSizeBeforeCurve + 8 + 4, &dfVal, 8);
741 : dfVal = 0;
742 : memcpy((*ppabyShape) + nShpSizeBeforeCurve + 8 + 4 + 8, &dfVal, 8);
743 : nTmp = EXT_SHAPE_ARC_MINOR;
744 : memcpy((*ppabyShape) + nShpSizeBeforeCurve + 8 + 4 + 16, &nTmp, 4);
745 : }
746 : #endif
747 :
748 : // Fill in the output size.
749 1414 : *pnBytes = nShpSize;
750 :
751 : // Set up write pointers.
752 1414 : unsigned char *pabyPtr = *ppabyShape;
753 1414 : unsigned char *pabyPtrM = bHasM ? pabyPtr + nShpSize - nShpZSize : nullptr;
754 :
755 1414 : unsigned char *pabyPtrZ = nullptr;
756 1414 : if (b3d)
757 : {
758 182 : if (bHasM)
759 2 : pabyPtrZ = pabyPtrM - nShpZSize;
760 : else
761 180 : pabyPtrZ = pabyPtr + nShpSize - nShpZSize;
762 : }
763 :
764 : /* -------------------------------------------------------------------- */
765 : /* Write in the Shape type number now */
766 : /* -------------------------------------------------------------------- */
767 1414 : GUInt32 nGType = SHPT_NULL;
768 :
769 1414 : switch (nOGRType)
770 : {
771 1064 : case wkbPoint:
772 : {
773 1064 : nGType = (b3d && bHasM) ? SHPT_POINTZM
774 : : (b3d) ? SHPT_POINTZ
775 : : (bHasM) ? SHPT_POINTM
776 : : SHPT_POINT;
777 1064 : break;
778 : }
779 60 : case wkbMultiPoint:
780 : {
781 60 : nGType = (b3d && bHasM) ? SHPT_MULTIPOINTZM
782 : : (b3d) ? SHPT_MULTIPOINTZ
783 : : (bHasM) ? SHPT_MULTIPOINTM
784 : : SHPT_MULTIPOINT;
785 60 : break;
786 : }
787 121 : case wkbLineString:
788 : case wkbMultiLineString:
789 : {
790 121 : nGType = (b3d && bHasM) ? SHPT_ARCZM
791 : : (b3d) ? SHPT_ARCZ
792 : : (bHasM) ? SHPT_ARCM
793 : : SHPT_ARC;
794 121 : break;
795 : }
796 169 : case wkbPolygon:
797 : case wkbMultiPolygon:
798 : {
799 169 : nGType = (b3d && bHasM) ? SHPT_POLYGONZM
800 : : (b3d) ? SHPT_POLYGONZ
801 : : (bHasM) ? SHPT_POLYGONM
802 : : SHPT_POLYGON;
803 169 : break;
804 : }
805 0 : default:
806 : {
807 0 : return OGRERR_UNSUPPORTED_OPERATION;
808 : }
809 : }
810 : // Write in the type number and advance the pointer.
811 : #ifdef WRITE_ARC_HACK
812 : nGType = SHPT_GENERALPOLYLINE | 0x20000000;
813 : #endif
814 :
815 1414 : CPL_LSBPTR32(&nGType);
816 1414 : memcpy(pabyPtr, &nGType, 4);
817 1414 : pabyPtr += 4;
818 :
819 : /* -------------------------------------------------------------------- */
820 : /* POINT and POINTZ */
821 : /* -------------------------------------------------------------------- */
822 1414 : if (nOGRType == wkbPoint)
823 : {
824 1064 : auto poPoint = poGeom->toPoint();
825 1064 : const double x = poPoint->getX();
826 1064 : const double y = poPoint->getY();
827 :
828 : // Copy in the raw data.
829 1064 : memcpy(pabyPtr, &x, 8);
830 1064 : memcpy(pabyPtr + 8, &y, 8);
831 1064 : if (b3d)
832 : {
833 31 : double z = poPoint->getZ();
834 31 : memcpy(pabyPtr + 8 + 8, &z, 8);
835 : }
836 1064 : if (bHasM)
837 : {
838 1 : double m = poPoint->getM();
839 1 : memcpy(pabyPtr + 8 + ((b3d) ? 16 : 8), &m, 8);
840 : }
841 :
842 : // Swap if needed. Shape doubles always LSB.
843 : if (OGR_SWAP(wkbNDR))
844 : {
845 : CPL_SWAPDOUBLE(pabyPtr);
846 : CPL_SWAPDOUBLE(pabyPtr + 8);
847 : if (b3d)
848 : CPL_SWAPDOUBLE(pabyPtr + 8 + 8);
849 : if (bHasM)
850 : CPL_SWAPDOUBLE(pabyPtr + 8 + ((b3d) ? 16 : 8));
851 : }
852 :
853 1064 : return OGRERR_NONE;
854 : }
855 :
856 : /* -------------------------------------------------------------------- */
857 : /* All the non-POINT types require an envelope next */
858 : /* -------------------------------------------------------------------- */
859 350 : OGREnvelope3D envelope;
860 350 : poGeom->getEnvelope(&envelope);
861 350 : memcpy(pabyPtr, &(envelope.MinX), 8);
862 350 : memcpy(pabyPtr + 8, &(envelope.MinY), 8);
863 350 : memcpy(pabyPtr + 8 + 8, &(envelope.MaxX), 8);
864 350 : memcpy(pabyPtr + 8 + 8 + 8, &(envelope.MaxY), 8);
865 :
866 : // Swap box if needed. Shape doubles are always LSB.
867 : if (OGR_SWAP(wkbNDR))
868 : {
869 : for (int i = 0; i < 4; i++)
870 : CPL_SWAPDOUBLE(pabyPtr + 8 * i);
871 : }
872 350 : pabyPtr += 32;
873 :
874 : // Write in the Z bounds at the end of the XY buffer.
875 350 : if (b3d)
876 : {
877 151 : memcpy(pabyPtrZ, &(envelope.MinZ), 8);
878 151 : memcpy(pabyPtrZ + 8, &(envelope.MaxZ), 8);
879 :
880 : // Swap Z bounds if necessary.
881 : if (OGR_SWAP(wkbNDR))
882 : {
883 : for (int i = 0; i < 2; i++)
884 : CPL_SWAPDOUBLE(pabyPtrZ + 8 * i);
885 : }
886 151 : pabyPtrZ += 16;
887 : }
888 :
889 : // Reserve space for the M bounds at the end of the XY buffer.
890 350 : GByte *pabyPtrMBounds = nullptr;
891 350 : double dfMinM = std::numeric_limits<double>::max();
892 350 : double dfMaxM = -dfMinM;
893 350 : if (bHasM)
894 : {
895 1 : pabyPtrMBounds = pabyPtrM;
896 1 : pabyPtrM += 16;
897 : }
898 :
899 : /* -------------------------------------------------------------------- */
900 : /* LINESTRING and LINESTRINGZ */
901 : /* -------------------------------------------------------------------- */
902 350 : if (nOGRType == wkbLineString)
903 : {
904 61 : auto poLine = poGeom->toLineString();
905 :
906 : // Write in the nparts (1).
907 61 : const GUInt32 nPartsLsb = CPL_LSBWORD32(nParts);
908 61 : memcpy(pabyPtr, &nPartsLsb, 4);
909 61 : pabyPtr += 4;
910 :
911 : // Write in the npoints.
912 61 : GUInt32 nPointsLsb = CPL_LSBWORD32(nPoints);
913 61 : memcpy(pabyPtr, &nPointsLsb, 4);
914 61 : pabyPtr += 4;
915 :
916 : // Write in the part index (0).
917 61 : GUInt32 nPartIndex = 0;
918 61 : memcpy(pabyPtr, &nPartIndex, 4);
919 61 : pabyPtr += 4;
920 :
921 : // Write in the point data.
922 61 : poLine->getPoints(reinterpret_cast<OGRRawPoint *>(pabyPtr),
923 : reinterpret_cast<double *>(pabyPtrZ));
924 61 : if (bHasM)
925 : {
926 0 : for (GUInt32 k = 0; k < nPoints; k++)
927 : {
928 0 : double dfM = poLine->getM(k);
929 0 : memcpy(pabyPtrM + 8 * k, &dfM, 8);
930 0 : if (dfM < dfMinM)
931 0 : dfMinM = dfM;
932 0 : if (dfM > dfMaxM)
933 0 : dfMaxM = dfM;
934 : }
935 : }
936 :
937 : // Swap if necessary.
938 : if (OGR_SWAP(wkbNDR))
939 : {
940 : for (GUInt32 k = 0; k < nPoints; k++)
941 : {
942 : CPL_SWAPDOUBLE(pabyPtr + 16 * k);
943 : CPL_SWAPDOUBLE(pabyPtr + 16 * k + 8);
944 : if (b3d)
945 : CPL_SWAPDOUBLE(pabyPtrZ + 8 * k);
946 : if (bHasM)
947 : CPL_SWAPDOUBLE(pabyPtrM + 8 * k);
948 : }
949 : }
950 : }
951 : /* -------------------------------------------------------------------- */
952 : /* POLYGON and POLYGONZ */
953 : /* -------------------------------------------------------------------- */
954 289 : else if (nOGRType == wkbPolygon)
955 : {
956 61 : auto poPoly = poGeom->toPolygon();
957 :
958 : // Write in the part count.
959 61 : GUInt32 nPartsLsb = CPL_LSBWORD32(nParts);
960 61 : memcpy(pabyPtr, &nPartsLsb, 4);
961 61 : pabyPtr += 4;
962 :
963 : // Write in the total point count.
964 61 : GUInt32 nPointsLsb = CPL_LSBWORD32(nPoints);
965 61 : memcpy(pabyPtr, &nPointsLsb, 4);
966 61 : pabyPtr += 4;
967 :
968 : /* --------------------------------------------------------------------
969 : */
970 : /* Now we have to visit each ring and write an index number into */
971 : /* the parts list, and the coordinates into the points list. */
972 : /* to do it in one pass, we will use three write pointers. */
973 : /* pabyPtr writes the part indexes */
974 : /* pabyPoints writes the xy coordinates */
975 : /* pabyPtrZ writes the z coordinates */
976 : /* --------------------------------------------------------------------
977 : */
978 :
979 : // Just past the partindex[nparts] array.
980 61 : unsigned char *pabyPoints = pabyPtr + 4 * nParts;
981 :
982 61 : int nPointIndexCount = 0;
983 :
984 123 : for (GUInt32 i = 0; i < nParts; i++)
985 : {
986 : // Check our Ring and condition it.
987 0 : std::unique_ptr<OGRLinearRing> poRing;
988 62 : if (i == 0)
989 : {
990 61 : poRing.reset(poPoly->getExteriorRing()->clone());
991 61 : assert(poRing);
992 : // Outer ring must be clockwise.
993 61 : if (!poRing->isClockwise())
994 0 : poRing->reversePoints();
995 : }
996 : else
997 : {
998 1 : poRing.reset(poPoly->getInteriorRing(i - 1)->clone());
999 1 : assert(poRing);
1000 : // Inner rings should be anti-clockwise.
1001 1 : if (poRing->isClockwise())
1002 1 : poRing->reversePoints();
1003 : }
1004 :
1005 62 : int nRingNumPoints = poRing->getNumPoints();
1006 :
1007 : #ifndef WRITE_ARC_HACK
1008 : // Cannot write un-closed rings to shape.
1009 62 : if (nRingNumPoints <= 2 || !poRing->get_IsClosed())
1010 0 : return OGRERR_FAILURE;
1011 : #endif
1012 :
1013 : // Write in the part index.
1014 62 : GUInt32 nPartIndex = CPL_LSBWORD32(nPointIndexCount);
1015 62 : memcpy(pabyPtr, &nPartIndex, 4);
1016 :
1017 : // Write in the point data.
1018 62 : poRing->getPoints(reinterpret_cast<OGRRawPoint *>(pabyPoints),
1019 : reinterpret_cast<double *>(pabyPtrZ));
1020 62 : if (bHasM)
1021 : {
1022 0 : for (int k = 0; k < nRingNumPoints; k++)
1023 : {
1024 0 : double dfM = poRing->getM(k);
1025 0 : memcpy(pabyPtrM + 8 * k, &dfM, 8);
1026 0 : if (dfM < dfMinM)
1027 0 : dfMinM = dfM;
1028 0 : if (dfM > dfMaxM)
1029 0 : dfMaxM = dfM;
1030 : }
1031 : }
1032 :
1033 : // Swap if necessary.
1034 : if (OGR_SWAP(wkbNDR))
1035 : {
1036 : for (int k = 0; k < nRingNumPoints; k++)
1037 : {
1038 : CPL_SWAPDOUBLE(pabyPoints + 16 * k);
1039 : CPL_SWAPDOUBLE(pabyPoints + 16 * k + 8);
1040 : if (b3d)
1041 : CPL_SWAPDOUBLE(pabyPtrZ + 8 * k);
1042 : if (bHasM)
1043 : CPL_SWAPDOUBLE(pabyPtrM + 8 * k);
1044 : }
1045 : }
1046 :
1047 62 : nPointIndexCount += nRingNumPoints;
1048 : // Advance the write pointers.
1049 62 : pabyPtr += 4;
1050 62 : pabyPoints += 16 * nRingNumPoints;
1051 62 : if (b3d)
1052 30 : pabyPtrZ += 8 * nRingNumPoints;
1053 62 : if (bHasM)
1054 0 : pabyPtrM += 8 * nRingNumPoints;
1055 : }
1056 : }
1057 : /* -------------------------------------------------------------------- */
1058 : /* MULTIPOINT and MULTIPOINTZ */
1059 : /* -------------------------------------------------------------------- */
1060 228 : else if (nOGRType == wkbMultiPoint)
1061 : {
1062 60 : auto poMPoint = poGeom->toMultiPoint();
1063 :
1064 : // Write in the total point count.
1065 60 : GUInt32 nPointsLsb = CPL_LSBWORD32(nPoints);
1066 60 : memcpy(pabyPtr, &nPointsLsb, 4);
1067 60 : pabyPtr += 4;
1068 :
1069 : /* --------------------------------------------------------------------
1070 : */
1071 : /* Now we have to visit each point write it into the points list */
1072 : /* We will use two write pointers. */
1073 : /* pabyPtr writes the xy coordinates */
1074 : /* pabyPtrZ writes the z coordinates */
1075 : /* --------------------------------------------------------------------
1076 : */
1077 :
1078 180 : for (auto &&poPt : poMPoint)
1079 : {
1080 : // Skip empties.
1081 120 : if (poPt->IsEmpty())
1082 0 : continue;
1083 :
1084 : // Write the coordinates.
1085 120 : double x = poPt->getX();
1086 120 : double y = poPt->getY();
1087 120 : memcpy(pabyPtr, &x, 8);
1088 120 : memcpy(pabyPtr + 8, &y, 8);
1089 120 : if (b3d)
1090 : {
1091 60 : double z = poPt->getZ();
1092 60 : memcpy(pabyPtrZ, &z, 8);
1093 : }
1094 120 : if (bHasM)
1095 : {
1096 0 : double dfM = poPt->getM();
1097 0 : memcpy(pabyPtrM, &dfM, 8);
1098 0 : if (dfM < dfMinM)
1099 0 : dfMinM = dfM;
1100 0 : if (dfM > dfMaxM)
1101 0 : dfMaxM = dfM;
1102 : }
1103 :
1104 : // Swap if necessary.
1105 : if (OGR_SWAP(wkbNDR))
1106 : {
1107 : CPL_SWAPDOUBLE(pabyPtr);
1108 : CPL_SWAPDOUBLE(pabyPtr + 8);
1109 : if (b3d)
1110 : CPL_SWAPDOUBLE(pabyPtrZ);
1111 : if (bHasM)
1112 : CPL_SWAPDOUBLE(pabyPtrM);
1113 : }
1114 :
1115 : // Advance the write pointers.
1116 120 : pabyPtr += 16;
1117 120 : if (b3d)
1118 60 : pabyPtrZ += 8;
1119 120 : if (bHasM)
1120 0 : pabyPtrM += 8;
1121 : }
1122 : }
1123 :
1124 : /* -------------------------------------------------------------------- */
1125 : /* MULTILINESTRING and MULTILINESTRINGZ */
1126 : /* -------------------------------------------------------------------- */
1127 168 : else if (nOGRType == wkbMultiLineString)
1128 : {
1129 60 : auto poMLine = poGeom->toMultiLineString();
1130 :
1131 : // Write in the part count.
1132 60 : GUInt32 nPartsLsb = CPL_LSBWORD32(nParts);
1133 60 : memcpy(pabyPtr, &nPartsLsb, 4);
1134 60 : pabyPtr += 4;
1135 :
1136 : // Write in the total point count.
1137 60 : GUInt32 nPointsLsb = CPL_LSBWORD32(nPoints);
1138 60 : memcpy(pabyPtr, &nPointsLsb, 4);
1139 60 : pabyPtr += 4;
1140 :
1141 : // Just past the partindex[nparts] array.
1142 60 : unsigned char *pabyPoints = pabyPtr + 4 * nParts;
1143 :
1144 60 : int nPointIndexCount = 0;
1145 :
1146 150 : for (auto &&poLine : poMLine)
1147 : {
1148 : // Skip empties.
1149 90 : if (poLine->IsEmpty())
1150 0 : continue;
1151 :
1152 90 : int nLineNumPoints = poLine->getNumPoints();
1153 :
1154 : // Write in the part index.
1155 90 : GUInt32 nPartIndex = CPL_LSBWORD32(nPointIndexCount);
1156 90 : memcpy(pabyPtr, &nPartIndex, 4);
1157 :
1158 : // Write in the point data.
1159 90 : poLine->getPoints(reinterpret_cast<OGRRawPoint *>(pabyPoints),
1160 : reinterpret_cast<double *>(pabyPtrZ));
1161 90 : if (bHasM)
1162 : {
1163 0 : for (int k = 0; k < nLineNumPoints; k++)
1164 : {
1165 0 : double dfM = poLine->getM(k);
1166 0 : memcpy(pabyPtrM + 8 * k, &dfM, 8);
1167 0 : if (dfM < dfMinM)
1168 0 : dfMinM = dfM;
1169 0 : if (dfM > dfMaxM)
1170 0 : dfMaxM = dfM;
1171 : }
1172 : }
1173 :
1174 : // Swap if necessary.
1175 : if (OGR_SWAP(wkbNDR))
1176 : {
1177 : for (int k = 0; k < nLineNumPoints; k++)
1178 : {
1179 : CPL_SWAPDOUBLE(pabyPoints + 16 * k);
1180 : CPL_SWAPDOUBLE(pabyPoints + 16 * k + 8);
1181 : if (b3d)
1182 : CPL_SWAPDOUBLE(pabyPtrZ + 8 * k);
1183 : if (bHasM)
1184 : CPL_SWAPDOUBLE(pabyPtrM + 8 * k);
1185 : }
1186 : }
1187 :
1188 90 : nPointIndexCount += nLineNumPoints;
1189 :
1190 : // Advance the write pointers.
1191 90 : pabyPtr += 4;
1192 90 : pabyPoints += 16 * nLineNumPoints;
1193 90 : if (b3d)
1194 30 : pabyPtrZ += 8 * nLineNumPoints;
1195 90 : if (bHasM)
1196 0 : pabyPtrM += 8 * nLineNumPoints;
1197 : }
1198 : }
1199 : /* -------------------------------------------------------------------- */
1200 : /* MULTIPOLYGON and MULTIPOLYGONZ */
1201 : /* -------------------------------------------------------------------- */
1202 : else // if( nOGRType == wkbMultiPolygon )
1203 : {
1204 108 : auto poMPoly = poGeom->toMultiPolygon();
1205 :
1206 : // Write in the part count.
1207 108 : GUInt32 nPartsLsb = CPL_LSBWORD32(nParts);
1208 108 : memcpy(pabyPtr, &nPartsLsb, 4);
1209 108 : pabyPtr += 4;
1210 :
1211 : // Write in the total point count.
1212 108 : GUInt32 nPointsLsb = CPL_LSBWORD32(nPoints);
1213 108 : memcpy(pabyPtr, &nPointsLsb, 4);
1214 108 : pabyPtr += 4;
1215 :
1216 : /* --------------------------------------------------------------------
1217 : */
1218 : /* Now we have to visit each ring and write an index number into */
1219 : /* the parts list, and the coordinates into the points list. */
1220 : /* to do it in one pass, we will use three write pointers. */
1221 : /* pabyPtr writes the part indexes */
1222 : /* pabyPoints writes the xy coordinates */
1223 : /* pabyPtrZ writes the z coordinates */
1224 : /* --------------------------------------------------------------------
1225 : */
1226 :
1227 : // Just past the partindex[nparts] array.
1228 108 : unsigned char *pabyPoints = pabyPtr + 4 * nParts;
1229 :
1230 108 : int nPointIndexCount = 0;
1231 :
1232 246 : for (auto &&poPoly : poMPoly)
1233 : {
1234 : // Skip empties.
1235 138 : if (poPoly->IsEmpty())
1236 0 : continue;
1237 :
1238 138 : int nRings = 1 + poPoly->getNumInteriorRings();
1239 :
1240 306 : for (int j = 0; j < nRings; j++)
1241 : {
1242 : // Check our Ring and condition it.
1243 0 : std::unique_ptr<OGRLinearRing> poRing;
1244 168 : if (j == 0)
1245 : {
1246 138 : poRing.reset(poPoly->getExteriorRing()->clone());
1247 138 : assert(poRing != nullptr);
1248 : // Outer ring must be clockwise.
1249 138 : if (!poRing->isClockwise())
1250 0 : poRing->reversePoints();
1251 : }
1252 : else
1253 : {
1254 30 : poRing.reset(poPoly->getInteriorRing(j - 1)->clone());
1255 30 : assert(poRing != nullptr);
1256 : // Inner rings should be anti-clockwise.
1257 30 : if (poRing->isClockwise())
1258 0 : poRing->reversePoints();
1259 : }
1260 :
1261 168 : int nRingNumPoints = poRing->getNumPoints();
1262 :
1263 : // Cannot write closed rings to shape.
1264 168 : if (nRingNumPoints <= 2 || !poRing->get_IsClosed())
1265 0 : return OGRERR_FAILURE;
1266 :
1267 : // Write in the part index.
1268 168 : GUInt32 nPartIndex = CPL_LSBWORD32(nPointIndexCount);
1269 168 : memcpy(pabyPtr, &nPartIndex, 4);
1270 :
1271 : // Write in the point data.
1272 168 : poRing->getPoints(reinterpret_cast<OGRRawPoint *>(pabyPoints),
1273 : reinterpret_cast<double *>(pabyPtrZ));
1274 168 : if (bHasM)
1275 : {
1276 8 : for (int k = 0; k < nRingNumPoints; k++)
1277 : {
1278 7 : double dfM = poRing->getM(k);
1279 7 : memcpy(pabyPtrM + 8 * k, &dfM, 8);
1280 7 : if (dfM < dfMinM)
1281 1 : dfMinM = dfM;
1282 7 : if (dfM > dfMaxM)
1283 7 : dfMaxM = dfM;
1284 : }
1285 : }
1286 :
1287 : // Swap if necessary.
1288 : if (OGR_SWAP(wkbNDR))
1289 : {
1290 : for (int k = 0; k < nRingNumPoints; k++)
1291 : {
1292 : CPL_SWAPDOUBLE(pabyPoints + 16 * k);
1293 : CPL_SWAPDOUBLE(pabyPoints + 16 * k + 8);
1294 : if (b3d)
1295 : CPL_SWAPDOUBLE(pabyPtrZ + 8 * k);
1296 : if (bHasM)
1297 : CPL_SWAPDOUBLE(pabyPtrM + 8 * k);
1298 : }
1299 : }
1300 :
1301 168 : nPointIndexCount += nRingNumPoints;
1302 : // Advance the write pointers.
1303 168 : pabyPtr += 4;
1304 168 : pabyPoints += 16 * nRingNumPoints;
1305 168 : if (b3d)
1306 31 : pabyPtrZ += 8 * nRingNumPoints;
1307 168 : if (bHasM)
1308 1 : pabyPtrM += 8 * nRingNumPoints;
1309 : }
1310 : }
1311 : }
1312 :
1313 350 : if (bHasM)
1314 : {
1315 1 : if (dfMinM > dfMaxM)
1316 : {
1317 0 : dfMinM = 0.0;
1318 0 : dfMaxM = 0.0;
1319 : }
1320 1 : memcpy(pabyPtrMBounds, &(dfMinM), 8);
1321 1 : memcpy(pabyPtrMBounds + 8, &(dfMaxM), 8);
1322 :
1323 : // Swap M bounds if necessary.
1324 : if (OGR_SWAP(wkbNDR))
1325 : {
1326 : for (int i = 0; i < 2; i++)
1327 : CPL_SWAPDOUBLE(pabyPtrMBounds + 8 * i);
1328 : }
1329 : }
1330 :
1331 350 : return OGRERR_NONE;
1332 : }
1333 :
1334 : /************************************************************************/
1335 : /* OGRCreateMultiPatch() */
1336 : /************************************************************************/
1337 :
1338 76 : OGRErr OGRCreateMultiPatch(const OGRGeometry *poGeomConst,
1339 : int bAllowSHPTTriangle, int &nParts,
1340 : std::vector<int> &anPartStart,
1341 : std::vector<int> &anPartType, int &nPoints,
1342 : std::vector<OGRRawPoint> &aoPoints,
1343 : std::vector<double> &adfZ)
1344 : {
1345 76 : const OGRwkbGeometryType eType = wkbFlatten(poGeomConst->getGeometryType());
1346 76 : if (eType != wkbPolygon && eType != wkbTriangle &&
1347 72 : eType != wkbMultiPolygon && eType != wkbMultiSurface &&
1348 36 : eType != wkbTIN && eType != wkbPolyhedralSurface &&
1349 : eType != wkbGeometryCollection)
1350 : {
1351 0 : return OGRERR_UNSUPPORTED_OPERATION;
1352 : }
1353 :
1354 152 : std::unique_ptr<OGRGeometry> poGeom(poGeomConst->clone());
1355 76 : poGeom->closeRings();
1356 :
1357 76 : OGRMultiPolygon *poMPoly = nullptr;
1358 76 : std::unique_ptr<OGRGeometry> poGeomToDelete;
1359 76 : if (eType == wkbMultiPolygon)
1360 1 : poMPoly = poGeom->toMultiPolygon();
1361 : else
1362 : {
1363 150 : poGeomToDelete = std::unique_ptr<OGRGeometry>(
1364 150 : OGRGeometryFactory::forceToMultiPolygon(poGeom->clone()));
1365 150 : if (poGeomToDelete.get() &&
1366 75 : wkbFlatten(poGeomToDelete->getGeometryType()) == wkbMultiPolygon)
1367 : {
1368 72 : poMPoly = poGeomToDelete->toMultiPolygon();
1369 : }
1370 : }
1371 76 : if (poMPoly == nullptr)
1372 : {
1373 3 : return OGRERR_UNSUPPORTED_OPERATION;
1374 : }
1375 :
1376 73 : nParts = 0;
1377 73 : anPartStart.clear();
1378 73 : anPartType.clear();
1379 73 : nPoints = 0;
1380 73 : aoPoints.clear();
1381 73 : adfZ.clear();
1382 73 : int nBeginLastPart = 0;
1383 333 : for (const auto poPoly : *poMPoly)
1384 : {
1385 : // Skip empties.
1386 260 : if (poPoly->IsEmpty())
1387 0 : continue;
1388 :
1389 260 : const int nRings = poPoly->getNumInteriorRings() + 1;
1390 260 : const OGRLinearRing *poRing = poPoly->getExteriorRing();
1391 260 : if (nRings == 1 && poRing->getNumPoints() == 4)
1392 : {
1393 230 : int nCorrectedPoints = nPoints;
1394 235 : if (nParts > 0 && anPartType[nParts - 1] == SHPP_OUTERRING &&
1395 5 : nPoints - anPartStart[nParts - 1] == 4)
1396 : {
1397 5 : nCorrectedPoints--;
1398 : }
1399 :
1400 617 : if (nParts > 0 &&
1401 157 : ((anPartType[nParts - 1] == SHPP_TRIANGLES &&
1402 91 : nPoints - anPartStart[nParts - 1] == 3) ||
1403 66 : (anPartType[nParts - 1] == SHPP_OUTERRING &&
1404 5 : nPoints - anPartStart[nParts - 1] == 4) ||
1405 61 : anPartType[nParts - 1] == SHPP_TRIFAN) &&
1406 127 : poRing->getX(0) == aoPoints[nBeginLastPart].x &&
1407 94 : poRing->getY(0) == aoPoints[nBeginLastPart].y &&
1408 33 : poRing->getZ(0) == adfZ[nBeginLastPart] &&
1409 33 : poRing->getX(1) == aoPoints[nCorrectedPoints - 1].x &&
1410 419 : poRing->getY(1) == aoPoints[nCorrectedPoints - 1].y &&
1411 32 : poRing->getZ(1) == adfZ[nCorrectedPoints - 1])
1412 : {
1413 32 : nPoints = nCorrectedPoints;
1414 32 : anPartType[nParts - 1] = SHPP_TRIFAN;
1415 :
1416 32 : aoPoints.resize(nPoints + 1);
1417 32 : adfZ.resize(nPoints + 1);
1418 32 : aoPoints[nPoints].x = poRing->getX(2);
1419 32 : aoPoints[nPoints].y = poRing->getY(2);
1420 32 : adfZ[nPoints] = poRing->getZ(2);
1421 32 : nPoints++;
1422 : }
1423 521 : else if (nParts > 0 &&
1424 125 : ((anPartType[nParts - 1] == SHPP_TRIANGLES &&
1425 60 : nPoints - anPartStart[nParts - 1] == 3) ||
1426 65 : (anPartType[nParts - 1] == SHPP_OUTERRING &&
1427 4 : nPoints - anPartStart[nParts - 1] == 4) ||
1428 61 : anPartType[nParts - 1] == SHPP_TRISTRIP) &&
1429 94 : poRing->getX(0) == aoPoints[nCorrectedPoints - 2].x &&
1430 62 : poRing->getY(0) == aoPoints[nCorrectedPoints - 2].y &&
1431 61 : poRing->getZ(0) == adfZ[nCorrectedPoints - 2] &&
1432 61 : poRing->getX(1) == aoPoints[nCorrectedPoints - 1].x &&
1433 384 : poRing->getY(1) == aoPoints[nCorrectedPoints - 1].y &&
1434 61 : poRing->getZ(1) == adfZ[nCorrectedPoints - 1])
1435 : {
1436 61 : nPoints = nCorrectedPoints;
1437 61 : anPartType[nParts - 1] = SHPP_TRISTRIP;
1438 :
1439 61 : aoPoints.resize(nPoints + 1);
1440 61 : adfZ.resize(nPoints + 1);
1441 61 : aoPoints[nPoints].x = poRing->getX(2);
1442 61 : aoPoints[nPoints].y = poRing->getY(2);
1443 61 : adfZ[nPoints] = poRing->getZ(2);
1444 61 : nPoints++;
1445 : }
1446 : else
1447 : {
1448 137 : if (nParts == 0 || anPartType[nParts - 1] != SHPP_TRIANGLES ||
1449 : !bAllowSHPTTriangle)
1450 : {
1451 137 : nBeginLastPart = nPoints;
1452 :
1453 137 : anPartStart.resize(nParts + 1);
1454 137 : anPartType.resize(nParts + 1);
1455 137 : anPartStart[nParts] = nPoints;
1456 137 : anPartType[nParts] =
1457 137 : bAllowSHPTTriangle ? SHPP_TRIANGLES : SHPP_OUTERRING;
1458 137 : nParts++;
1459 : }
1460 :
1461 137 : aoPoints.resize(nPoints + 4);
1462 137 : adfZ.resize(nPoints + 4);
1463 685 : for (int i = 0; i < 4; i++)
1464 : {
1465 548 : aoPoints[nPoints + i].x = poRing->getX(i);
1466 548 : aoPoints[nPoints + i].y = poRing->getY(i);
1467 548 : adfZ[nPoints + i] = poRing->getZ(i);
1468 : }
1469 137 : nPoints += bAllowSHPTTriangle ? 3 : 4;
1470 : }
1471 : }
1472 : else
1473 : {
1474 30 : anPartStart.resize(nParts + nRings);
1475 30 : anPartType.resize(nParts + nRings);
1476 :
1477 90 : for (int i = 0; i < nRings; i++)
1478 : {
1479 60 : anPartStart[nParts + i] = nPoints;
1480 60 : if (i == 0)
1481 : {
1482 30 : poRing = poPoly->getExteriorRing();
1483 30 : anPartType[nParts + i] = SHPP_OUTERRING;
1484 : }
1485 : else
1486 : {
1487 30 : poRing = poPoly->getInteriorRing(i - 1);
1488 30 : anPartType[nParts + i] = SHPP_INNERRING;
1489 : }
1490 60 : aoPoints.resize(nPoints + poRing->getNumPoints());
1491 60 : adfZ.resize(nPoints + poRing->getNumPoints());
1492 360 : for (int k = 0; k < poRing->getNumPoints(); k++)
1493 : {
1494 300 : aoPoints[nPoints + k].x = poRing->getX(k);
1495 300 : aoPoints[nPoints + k].y = poRing->getY(k);
1496 300 : adfZ[nPoints + k] = poRing->getZ(k);
1497 : }
1498 60 : nPoints += poRing->getNumPoints();
1499 : }
1500 :
1501 30 : nParts += nRings;
1502 : }
1503 : }
1504 :
1505 73 : if (nParts == 1 && anPartType[0] == SHPP_OUTERRING && nPoints == 4)
1506 : {
1507 8 : anPartType[0] = SHPP_TRIFAN;
1508 8 : nPoints = 3;
1509 : }
1510 :
1511 73 : return OGRERR_NONE;
1512 : }
1513 :
1514 : /************************************************************************/
1515 : /* OGRWriteMultiPatchToShapeBin() */
1516 : /************************************************************************/
1517 :
1518 60 : OGRErr OGRWriteMultiPatchToShapeBin(const OGRGeometry *poGeom,
1519 : GByte **ppabyShape, int *pnBytes)
1520 : {
1521 60 : int nParts = 0;
1522 120 : std::vector<int> anPartStart;
1523 120 : std::vector<int> anPartType;
1524 60 : int nPoints = 0;
1525 120 : std::vector<OGRRawPoint> aoPoints;
1526 120 : std::vector<double> adfZ;
1527 60 : OGRErr eErr = OGRCreateMultiPatch(poGeom, TRUE, nParts, anPartStart,
1528 : anPartType, nPoints, aoPoints, adfZ);
1529 60 : if (eErr != OGRERR_NONE)
1530 0 : return eErr;
1531 :
1532 : const bool bOmitZ =
1533 60 : !poGeom->Is3D() &&
1534 0 : CPLTestBool(CPLGetConfigOption("OGR_MULTIPATCH_OMIT_Z", "NO"));
1535 :
1536 60 : int nShpSize = 4; // All types start with integer type number.
1537 60 : nShpSize += 16 * 2; // xy bbox.
1538 60 : nShpSize += 4; // nparts.
1539 60 : nShpSize += 4; // npoints.
1540 60 : nShpSize += 4 * nParts; // panPartStart[nparts].
1541 60 : nShpSize += 4 * nParts; // panPartType[nparts].
1542 60 : nShpSize += 8 * 2 * nPoints; // xy points.
1543 60 : if (!bOmitZ)
1544 : {
1545 60 : nShpSize += 16; // z bbox.
1546 60 : nShpSize += 8 * nPoints; // z points.
1547 : }
1548 :
1549 60 : *pnBytes = nShpSize;
1550 60 : *ppabyShape = static_cast<GByte *>(CPLMalloc(nShpSize));
1551 :
1552 60 : GByte *pabyPtr = *ppabyShape;
1553 :
1554 : // Write in the type number and advance the pointer.
1555 60 : GUInt32 nGType = bOmitZ ? CPL_LSBWORD32(SHPT_GENERALMULTIPATCH)
1556 : : CPL_LSBWORD32(SHPT_MULTIPATCH);
1557 60 : memcpy(pabyPtr, &nGType, 4);
1558 60 : pabyPtr += 4;
1559 :
1560 60 : OGREnvelope3D envelope;
1561 60 : poGeom->getEnvelope(&envelope);
1562 60 : memcpy(pabyPtr, &(envelope.MinX), 8);
1563 60 : memcpy(pabyPtr + 8, &(envelope.MinY), 8);
1564 60 : memcpy(pabyPtr + 8 + 8, &(envelope.MaxX), 8);
1565 60 : memcpy(pabyPtr + 8 + 8 + 8, &(envelope.MaxY), 8);
1566 :
1567 : // Swap box if needed. Shape doubles are always LSB.
1568 : if (OGR_SWAP(wkbNDR))
1569 : {
1570 : for (int i = 0; i < 4; i++)
1571 : CPL_SWAPDOUBLE(pabyPtr + 8 * i);
1572 : }
1573 60 : pabyPtr += 32;
1574 :
1575 : // Write in the part count.
1576 60 : GUInt32 nPartsLsb = CPL_LSBWORD32(nParts);
1577 60 : memcpy(pabyPtr, &nPartsLsb, 4);
1578 60 : pabyPtr += 4;
1579 :
1580 : // Write in the total point count.
1581 60 : GUInt32 nPointsLsb = CPL_LSBWORD32(nPoints);
1582 60 : memcpy(pabyPtr, &nPointsLsb, 4);
1583 60 : pabyPtr += 4;
1584 :
1585 240 : for (int i = 0; i < nParts; i++)
1586 : {
1587 180 : int nPartStart = CPL_LSBWORD32(anPartStart[i]);
1588 180 : memcpy(pabyPtr, &nPartStart, 4);
1589 180 : pabyPtr += 4;
1590 : }
1591 240 : for (int i = 0; i < nParts; i++)
1592 : {
1593 180 : int nPartType = CPL_LSBWORD32(anPartType[i]);
1594 180 : memcpy(pabyPtr, &nPartType, 4);
1595 180 : pabyPtr += 4;
1596 : }
1597 :
1598 60 : if (!aoPoints.empty())
1599 60 : memcpy(pabyPtr, aoPoints.data(), 2 * 8 * nPoints);
1600 :
1601 : // Swap box if needed. Shape doubles are always LSB.
1602 : if (OGR_SWAP(wkbNDR))
1603 : {
1604 : for (int i = 0; i < 2 * nPoints; i++)
1605 : CPL_SWAPDOUBLE(pabyPtr + 8 * i);
1606 : }
1607 60 : pabyPtr += 2 * 8 * nPoints;
1608 :
1609 60 : if (!bOmitZ)
1610 : {
1611 60 : memcpy(pabyPtr, &(envelope.MinZ), 8);
1612 60 : memcpy(pabyPtr + 8, &(envelope.MaxZ), 8);
1613 : if (OGR_SWAP(wkbNDR))
1614 : {
1615 : for (int i = 0; i < 2; i++)
1616 : CPL_SWAPDOUBLE(pabyPtr + 8 * i);
1617 : }
1618 60 : pabyPtr += 16;
1619 :
1620 60 : if (!adfZ.empty())
1621 60 : memcpy(pabyPtr, adfZ.data(), 8 * nPoints);
1622 : // Swap box if needed. Shape doubles are always LSB.
1623 : if (OGR_SWAP(wkbNDR))
1624 : {
1625 : for (int i = 0; i < nPoints; i++)
1626 : CPL_SWAPDOUBLE(pabyPtr + 8 * i);
1627 : }
1628 : // pabyPtr += 8 * nPoints;
1629 : }
1630 :
1631 60 : return OGRERR_NONE;
1632 : }
1633 :
1634 : /************************************************************************/
1635 : /* GetAngleOnEllipse() */
1636 : /************************************************************************/
1637 :
1638 : // Return the angle in deg [0, 360] of dfArcX,dfArcY regarding the
1639 : // ellipse semi-major axis.
1640 26 : static double GetAngleOnEllipse(double dfPointOnArcX, double dfPointOnArcY,
1641 : double dfCenterX, double dfCenterY,
1642 : double dfRotationDeg, // Ellipse rotation.
1643 : double dfSemiMajor, double dfSemiMinor)
1644 : {
1645 : // Invert the following equation where cosA, sinA are unknown:
1646 : // dfPointOnArcX-dfCenterX = cosA*M*cosRot + sinA*m*sinRot
1647 : // dfPointOnArcY-dfCenterY = -cosA*M*sinRot + sinA*m*cosRot
1648 :
1649 26 : if (dfSemiMajor == 0.0 || dfSemiMinor == 0.0)
1650 0 : return 0.0;
1651 26 : const double dfRotationRadians = dfRotationDeg * M_PI / 180.0;
1652 26 : const double dfCosRot = cos(dfRotationRadians);
1653 26 : const double dfSinRot = sin(dfRotationRadians);
1654 26 : const double dfDeltaX = dfPointOnArcX - dfCenterX;
1655 26 : const double dfDeltaY = dfPointOnArcY - dfCenterY;
1656 26 : const double dfCosA =
1657 26 : (dfCosRot * dfDeltaX - dfSinRot * dfDeltaY) / dfSemiMajor;
1658 26 : const double dfSinA =
1659 26 : (dfSinRot * dfDeltaX + dfCosRot * dfDeltaY) / dfSemiMinor;
1660 : // We could check that dfCosA^2 + dfSinA^2 ~= 1 to verify that the point
1661 : // is on the ellipse.
1662 26 : const double dfAngle = atan2(dfSinA, dfCosA) / M_PI * 180;
1663 26 : if (dfAngle < -180)
1664 0 : return dfAngle + 360;
1665 26 : return dfAngle;
1666 : }
1667 :
1668 : /************************************************************************/
1669 : /* OGRShapeCreateCompoundCurve() */
1670 : /************************************************************************/
1671 :
1672 122 : static OGRCurve *OGRShapeCreateCompoundCurve(int nPartStartIdx, int nPartPoints,
1673 : const CurveSegment *pasCurves,
1674 : int nCurves, int nFirstCurveIdx,
1675 : /* const */ double *padfX,
1676 : /* const */ double *padfY,
1677 : /* const */ double *padfZ,
1678 : /* const */ double *padfM,
1679 : int *pnLastCurveIdx)
1680 : {
1681 244 : auto poCC = std::make_unique<OGRCompoundCurve>();
1682 122 : int nLastPointIdx = nPartStartIdx;
1683 122 : bool bHasCircularArcs = false;
1684 122 : int i = nFirstCurveIdx; // Used after for.
1685 296 : for (; i < nCurves; i++)
1686 : {
1687 200 : const int nStartPointIdx = pasCurves[i].nStartPointIdx;
1688 :
1689 200 : if (nStartPointIdx < nPartStartIdx)
1690 : {
1691 : // Shouldn't happen normally, but who knows.
1692 0 : continue;
1693 : }
1694 :
1695 : // For a multi-part geometry, stop when the start index of the curve
1696 : // exceeds the last point index of the part
1697 200 : if (nStartPointIdx >= nPartStartIdx + nPartPoints)
1698 : {
1699 26 : if (pnLastCurveIdx)
1700 26 : *pnLastCurveIdx = i;
1701 26 : break;
1702 : }
1703 :
1704 : // Add linear segments between end of last curve portion (or beginning
1705 : // of the part) and start of current curve.
1706 174 : if (nStartPointIdx > nLastPointIdx)
1707 : {
1708 39 : OGRLineString *poLine = new OGRLineString();
1709 43 : poLine->setPoints(
1710 39 : nStartPointIdx - nLastPointIdx + 1, padfX + nLastPointIdx,
1711 39 : padfY + nLastPointIdx,
1712 8 : padfZ != nullptr ? padfZ + nLastPointIdx : nullptr,
1713 4 : padfM != nullptr ? padfM + nLastPointIdx : nullptr);
1714 39 : poCC->addCurveDirectly(poLine);
1715 : }
1716 :
1717 174 : if (pasCurves[i].eType == CURVE_ARC_INTERIOR_POINT &&
1718 113 : nStartPointIdx + 1 < nPartStartIdx + nPartPoints)
1719 : {
1720 113 : OGRPoint p1(padfX[nStartPointIdx], padfY[nStartPointIdx],
1721 26 : padfZ != nullptr ? padfZ[nStartPointIdx] : 0.0,
1722 113 : padfM != nullptr ? padfM[nStartPointIdx] : 0.0);
1723 113 : OGRPoint p2(pasCurves[i].u.ArcByIntermediatePoint.dfX,
1724 113 : pasCurves[i].u.ArcByIntermediatePoint.dfY,
1725 113 : padfZ != nullptr ? padfZ[nStartPointIdx] : 0.0);
1726 113 : OGRPoint p3(padfX[nStartPointIdx + 1], padfY[nStartPointIdx + 1],
1727 26 : padfZ != nullptr ? padfZ[nStartPointIdx + 1] : 0.0,
1728 113 : padfM != nullptr ? padfM[nStartPointIdx + 1] : 0.0);
1729 :
1730 : // Some software (like QGIS, see https://hub.qgis.org/issues/15116)
1731 : // do not like 3-point circles, so use a 5 point variant.
1732 113 : if (p1.getX() == p3.getX() && p1.getY() == p3.getY())
1733 : {
1734 8 : if (p1.getX() != p2.getX() || p1.getY() != p2.getY())
1735 : {
1736 8 : bHasCircularArcs = true;
1737 8 : OGRCircularString *poCS = new OGRCircularString();
1738 8 : const double dfCenterX = (p1.getX() + p2.getX()) / 2;
1739 8 : const double dfCenterY = (p1.getY() + p2.getY()) / 2;
1740 8 : poCS->addPoint(&p1);
1741 16 : OGRPoint pInterm1(dfCenterX - (p2.getY() - dfCenterY),
1742 8 : dfCenterY + (p1.getX() - dfCenterX),
1743 2 : padfZ != nullptr ? padfZ[nStartPointIdx]
1744 24 : : 0.0);
1745 8 : poCS->addPoint(&pInterm1);
1746 8 : poCS->addPoint(&p2);
1747 16 : OGRPoint pInterm2(dfCenterX + (p2.getY() - dfCenterY),
1748 8 : dfCenterY - (p1.getX() - dfCenterX),
1749 2 : padfZ != nullptr ? padfZ[nStartPointIdx]
1750 24 : : 0.0);
1751 8 : poCS->addPoint(&pInterm2);
1752 8 : poCS->addPoint(&p3);
1753 8 : poCS->set3D(padfZ != nullptr);
1754 8 : poCS->setMeasured(padfM != nullptr);
1755 8 : poCC->addCurveDirectly(poCS);
1756 : }
1757 : }
1758 : else
1759 : {
1760 105 : bHasCircularArcs = true;
1761 105 : OGRCircularString *poCS = new OGRCircularString();
1762 105 : poCS->addPoint(&p1);
1763 105 : poCS->addPoint(&p2);
1764 105 : poCS->addPoint(&p3);
1765 105 : poCS->set3D(padfZ != nullptr);
1766 105 : poCS->setMeasured(padfM != nullptr);
1767 105 : if (poCC->addCurveDirectly(poCS) != OGRERR_NONE)
1768 : {
1769 0 : delete poCS;
1770 0 : return nullptr;
1771 : }
1772 113 : }
1773 : }
1774 :
1775 61 : else if (pasCurves[i].eType == CURVE_ARC_CENTER_POINT &&
1776 12 : nStartPointIdx + 1 < nPartStartIdx + nPartPoints)
1777 : {
1778 12 : const double dfCenterX = pasCurves[i].u.ArcByCenterPoint.dfX;
1779 12 : const double dfCenterY = pasCurves[i].u.ArcByCenterPoint.dfY;
1780 12 : double dfDeltaY = padfY[nStartPointIdx] - dfCenterY;
1781 12 : double dfDeltaX = padfX[nStartPointIdx] - dfCenterX;
1782 12 : double dfAngleStart = atan2(dfDeltaY, dfDeltaX);
1783 12 : dfDeltaY = padfY[nStartPointIdx + 1] - dfCenterY;
1784 12 : dfDeltaX = padfX[nStartPointIdx + 1] - dfCenterX;
1785 12 : double dfAngleEnd = atan2(dfDeltaY, dfDeltaX);
1786 : // Note: This definition from center and 2 points may be
1787 : // not a circle.
1788 12 : double dfRadius = sqrt(dfDeltaX * dfDeltaX + dfDeltaY * dfDeltaY);
1789 12 : if (pasCurves[i].u.ArcByCenterPoint.bIsCCW)
1790 : {
1791 4 : if (dfAngleStart >= dfAngleEnd)
1792 2 : dfAngleEnd += 2 * M_PI;
1793 : }
1794 : else
1795 : {
1796 8 : if (dfAngleStart <= dfAngleEnd)
1797 6 : dfAngleEnd -= 2 * M_PI;
1798 : }
1799 12 : const double dfMidAngle = (dfAngleStart + dfAngleEnd) / 2;
1800 12 : OGRPoint p1(padfX[nStartPointIdx], padfY[nStartPointIdx],
1801 0 : padfZ != nullptr ? padfZ[nStartPointIdx] : 0.0,
1802 24 : padfM != nullptr ? padfM[nStartPointIdx] : 0.0);
1803 12 : OGRPoint p2(dfCenterX + dfRadius * cos(dfMidAngle),
1804 12 : dfCenterY + dfRadius * sin(dfMidAngle),
1805 24 : padfZ != nullptr ? padfZ[nStartPointIdx] : 0.0);
1806 12 : OGRPoint p3(padfX[nStartPointIdx + 1], padfY[nStartPointIdx + 1],
1807 0 : padfZ != nullptr ? padfZ[nStartPointIdx + 1] : 0.0,
1808 24 : padfM != nullptr ? padfM[nStartPointIdx + 1] : 0.0);
1809 :
1810 12 : bHasCircularArcs = true;
1811 12 : OGRCircularString *poCS = new OGRCircularString();
1812 12 : poCS->addPoint(&p1);
1813 12 : poCS->addPoint(&p2);
1814 12 : poCS->addPoint(&p3);
1815 12 : poCS->set3D(padfZ != nullptr);
1816 12 : poCS->setMeasured(padfM != nullptr);
1817 24 : poCC->addCurveDirectly(poCS);
1818 : }
1819 :
1820 49 : else if (pasCurves[i].eType == CURVE_BEZIER &&
1821 36 : nStartPointIdx + 1 < nPartStartIdx + nPartPoints)
1822 : {
1823 36 : OGRLineString *poLine = new OGRLineString();
1824 36 : const double dfX0 = padfX[nStartPointIdx];
1825 36 : const double dfY0 = padfY[nStartPointIdx];
1826 36 : const double dfX1 = pasCurves[i].u.Bezier.dfX1;
1827 36 : const double dfY1 = pasCurves[i].u.Bezier.dfY1;
1828 36 : const double dfX2 = pasCurves[i].u.Bezier.dfX2;
1829 36 : const double dfY2 = pasCurves[i].u.Bezier.dfY2;
1830 36 : const double dfX3 = padfX[nStartPointIdx + 1];
1831 36 : const double dfY3 = padfY[nStartPointIdx + 1];
1832 36 : double dfStartAngle = atan2(dfY1 - dfY0, dfX1 - dfX0);
1833 36 : double dfEndAngle = atan2(dfY3 - dfY2, dfX3 - dfX2);
1834 36 : if (dfStartAngle + M_PI < dfEndAngle)
1835 2 : dfStartAngle += 2 * M_PI;
1836 34 : else if (dfEndAngle + M_PI < dfStartAngle)
1837 0 : dfEndAngle += 2 * M_PI;
1838 : const double dfStepSizeRad =
1839 36 : OGRGeometryFactory::GetDefaultArcStepSize() / 180.0 * M_PI;
1840 36 : const double dfLengthTangentStart =
1841 36 : (dfX1 - dfX0) * (dfX1 - dfX0) + (dfY1 - dfY0) * (dfY1 - dfY0);
1842 36 : const double dfLengthTangentEnd =
1843 36 : (dfX3 - dfX2) * (dfX3 - dfX2) + (dfY3 - dfY2) * (dfY3 - dfY2);
1844 36 : const double dfLength =
1845 36 : (dfX3 - dfX0) * (dfX3 - dfX0) + (dfY3 - dfY0) * (dfY3 - dfY0);
1846 : // Heuristics to compute number of steps: we take into account the
1847 : // angular difference between the start and end tangent. And we
1848 : // also take into account the relative length of the tangent vs
1849 : // the length of the straight segment
1850 : const int nSteps =
1851 : (dfLength < 1e-9)
1852 36 : ? 1
1853 36 : : static_cast<int>(std::min(
1854 72 : 1000.0,
1855 144 : ceil(std::max(2.0, fabs(dfEndAngle - dfStartAngle) /
1856 36 : dfStepSizeRad) *
1857 108 : std::max(1.0, 5.0 *
1858 36 : (dfLengthTangentStart +
1859 72 : dfLengthTangentEnd) /
1860 36 : dfLength))));
1861 36 : poLine->setNumPoints(nSteps + 1);
1862 84 : poLine->setPoint(0, dfX0, dfY0,
1863 24 : padfZ != nullptr ? padfZ[nStartPointIdx] : 0.0,
1864 24 : padfM != nullptr ? padfM[nStartPointIdx] : 0.0);
1865 1317 : for (int j = 1; j < nSteps; j++)
1866 : {
1867 1281 : const double t = static_cast<double>(j) / nSteps;
1868 : // Third-order Bezier interpolation.
1869 1281 : poLine->setPoint(
1870 : j,
1871 1281 : (1 - t) * (1 - t) * (1 - t) * dfX0 +
1872 1281 : 3 * (1 - t) * (1 - t) * t * dfX1 +
1873 1281 : 3 * (1 - t) * t * t * dfX2 + t * t * t * dfX3,
1874 1281 : (1 - t) * (1 - t) * (1 - t) * dfY0 +
1875 1281 : 3 * (1 - t) * (1 - t) * t * dfY1 +
1876 1281 : 3 * (1 - t) * t * t * dfY2 + t * t * t * dfY3);
1877 : }
1878 84 : poLine->setPoint(nSteps, dfX3, dfY3,
1879 24 : padfZ != nullptr ? padfZ[nStartPointIdx + 1] : 0.0,
1880 24 : padfM != nullptr ? padfM[nStartPointIdx + 1]
1881 : : 0.0);
1882 36 : poLine->set3D(padfZ != nullptr);
1883 36 : poLine->setMeasured(padfM != nullptr);
1884 36 : if (poCC->addCurveDirectly(poLine) != OGRERR_NONE)
1885 : {
1886 0 : delete poLine;
1887 0 : return nullptr;
1888 36 : }
1889 : }
1890 :
1891 13 : else if (pasCurves[i].eType == CURVE_ELLIPSE_BY_CENTER &&
1892 13 : nStartPointIdx + 1 < nPartStartIdx + nPartPoints)
1893 : {
1894 13 : const double dfSemiMinor =
1895 13 : pasCurves[i].u.EllipseByCenter.dfSemiMajor *
1896 13 : pasCurves[i].u.EllipseByCenter.dfRatioSemiMinor;
1897 : // Different sign conventions between ext shape
1898 : // (trigonometric, CCW) and approximateArcAngles (CW).
1899 13 : const double dfRotationDeg =
1900 13 : -pasCurves[i].u.EllipseByCenter.dfRotationDeg;
1901 13 : const double dfAngleStart = GetAngleOnEllipse(
1902 13 : padfX[nStartPointIdx], padfY[nStartPointIdx],
1903 13 : pasCurves[i].u.EllipseByCenter.dfX,
1904 13 : pasCurves[i].u.EllipseByCenter.dfY, dfRotationDeg,
1905 13 : pasCurves[i].u.EllipseByCenter.dfSemiMajor, dfSemiMinor);
1906 13 : const double dfAngleEnd = GetAngleOnEllipse(
1907 13 : padfX[nStartPointIdx + 1], padfY[nStartPointIdx + 1],
1908 13 : pasCurves[i].u.EllipseByCenter.dfX,
1909 13 : pasCurves[i].u.EllipseByCenter.dfY, dfRotationDeg,
1910 13 : pasCurves[i].u.EllipseByCenter.dfSemiMajor, dfSemiMinor);
1911 : // CPLDebug("OGR", "Start angle=%f, End angle=%f",
1912 : // dfAngleStart, dfAngleEnd);
1913 : // Approximatearcangles() use CW.
1914 13 : double dfAngleStartForApprox = -dfAngleStart;
1915 13 : double dfAngleEndForApprox = -dfAngleEnd;
1916 13 : if (pasCurves[i].u.EllipseByCenter.bIsComplete)
1917 : {
1918 9 : dfAngleEndForApprox = dfAngleStartForApprox + 360;
1919 : }
1920 4 : else if (pasCurves[i].u.EllipseByCenter.bIsMinor)
1921 : {
1922 4 : if (dfAngleEndForApprox > dfAngleStartForApprox + 180)
1923 : {
1924 0 : dfAngleEndForApprox -= 360;
1925 : }
1926 4 : else if (dfAngleEndForApprox < dfAngleStartForApprox - 180)
1927 : {
1928 0 : dfAngleEndForApprox += 360;
1929 : }
1930 : }
1931 : else
1932 : {
1933 0 : if (dfAngleEndForApprox > dfAngleStartForApprox &&
1934 0 : dfAngleEndForApprox < dfAngleStartForApprox + 180)
1935 : {
1936 0 : dfAngleEndForApprox -= 360;
1937 : }
1938 0 : else if (dfAngleEndForApprox < dfAngleStartForApprox &&
1939 0 : dfAngleEndForApprox > dfAngleStartForApprox - 180)
1940 : {
1941 0 : dfAngleEndForApprox += 360;
1942 : }
1943 : }
1944 : OGRLineString *poLine =
1945 : OGRGeometryFactory::approximateArcAngles(
1946 13 : pasCurves[i].u.EllipseByCenter.dfX,
1947 13 : pasCurves[i].u.EllipseByCenter.dfY,
1948 2 : padfZ != nullptr ? padfZ[nStartPointIdx] : 0.0,
1949 13 : pasCurves[i].u.EllipseByCenter.dfSemiMajor, dfSemiMinor,
1950 : dfRotationDeg, dfAngleStartForApprox, dfAngleEndForApprox,
1951 : 0)
1952 13 : ->toLineString();
1953 13 : if (poLine->getNumPoints() >= 2)
1954 : {
1955 15 : poLine->setPoint(
1956 13 : 0, padfX[nStartPointIdx], padfY[nStartPointIdx],
1957 2 : padfZ != nullptr ? padfZ[nStartPointIdx] : 0.0,
1958 2 : padfM != nullptr ? padfM[nStartPointIdx] : 0.0);
1959 15 : poLine->setPoint(
1960 13 : poLine->getNumPoints() - 1, padfX[nStartPointIdx + 1],
1961 13 : padfY[nStartPointIdx + 1],
1962 2 : padfZ != nullptr ? padfZ[nStartPointIdx + 1] : 0.0,
1963 2 : padfM != nullptr ? padfM[nStartPointIdx + 1] : 0.0);
1964 : }
1965 13 : poLine->set3D(padfZ != nullptr);
1966 13 : poLine->setMeasured(padfM != nullptr);
1967 13 : poCC->addCurveDirectly(poLine);
1968 : }
1969 :
1970 : // Should not happen normally.
1971 0 : else if (nStartPointIdx + 1 < nPartStartIdx + nPartPoints)
1972 : {
1973 0 : OGRLineString *poLine = new OGRLineString();
1974 0 : poLine->setPoints(
1975 0 : 2, padfX + nStartPointIdx, padfY + nStartPointIdx,
1976 0 : padfZ != nullptr ? padfZ + nStartPointIdx : nullptr,
1977 0 : padfM != nullptr ? padfM + nStartPointIdx : nullptr);
1978 0 : poCC->addCurveDirectly(poLine);
1979 : }
1980 174 : nLastPointIdx = nStartPointIdx + 1;
1981 : }
1982 122 : if (i == nCurves)
1983 : {
1984 96 : if (pnLastCurveIdx)
1985 17 : *pnLastCurveIdx = i;
1986 : }
1987 :
1988 : // Add terminating linear segments
1989 122 : if (nLastPointIdx < nPartStartIdx + nPartPoints - 1)
1990 : {
1991 49 : OGRLineString *poLine = new OGRLineString();
1992 59 : poLine->setPoints(nPartStartIdx + nPartPoints - 1 - nLastPointIdx + 1,
1993 49 : padfX + nLastPointIdx, padfY + nLastPointIdx,
1994 14 : padfZ != nullptr ? padfZ + nLastPointIdx : nullptr,
1995 10 : padfM != nullptr ? padfM + nLastPointIdx : nullptr);
1996 49 : if (poCC->addCurveDirectly(poLine) != OGRERR_NONE)
1997 : {
1998 0 : delete poLine;
1999 0 : return nullptr;
2000 : }
2001 : }
2002 :
2003 122 : if (!bHasCircularArcs)
2004 : {
2005 29 : OGRwkbGeometryType eLSType = wkbLineString;
2006 29 : if (OGR_GT_HasZ(poCC->getGeometryType()))
2007 7 : eLSType = OGR_GT_SetZ(eLSType);
2008 29 : if (OGR_GT_HasM(poCC->getGeometryType()))
2009 7 : eLSType = OGR_GT_SetM(eLSType);
2010 29 : return reinterpret_cast<OGRCurve *>(OGR_G_ForceTo(
2011 58 : OGRGeometry::ToHandle(poCC.release()), eLSType, nullptr));
2012 : }
2013 : else
2014 93 : return poCC.release();
2015 : }
2016 :
2017 : /************************************************************************/
2018 : /* OGRCreateFromShapeBin() */
2019 : /* */
2020 : /* Translate shapefile binary representation to an OGR */
2021 : /* geometry. */
2022 : /************************************************************************/
2023 :
2024 1914 : OGRErr OGRCreateFromShapeBin(GByte *pabyShape, OGRGeometry **ppoGeom,
2025 : int nBytes)
2026 :
2027 : {
2028 1914 : *ppoGeom = nullptr;
2029 :
2030 1914 : if (nBytes < 4)
2031 : {
2032 0 : CPLError(CE_Failure, CPLE_AppDefined,
2033 : "Shape buffer size (%d) too small", nBytes);
2034 0 : return OGRERR_FAILURE;
2035 : }
2036 :
2037 : /* -------------------------------------------------------------------- */
2038 : /* Detect zlib compressed shapes and uncompress buffer if necessary */
2039 : /* NOTE: this seems to be an undocumented feature, even in the */
2040 : /* extended_shapefile_format.pdf found in the FileGDB API */
2041 : /* documentation. */
2042 : /* -------------------------------------------------------------------- */
2043 1914 : if (nBytes >= 14 && pabyShape[12] == 0x78 &&
2044 0 : pabyShape[13] == 0xDA /* zlib marker */)
2045 : {
2046 0 : GInt32 nUncompressedSize = 0;
2047 0 : GInt32 nCompressedSize = 0;
2048 0 : memcpy(&nUncompressedSize, pabyShape + 4, 4);
2049 0 : memcpy(&nCompressedSize, pabyShape + 8, 4);
2050 0 : CPL_LSBPTR32(&nUncompressedSize);
2051 0 : CPL_LSBPTR32(&nCompressedSize);
2052 0 : if (nCompressedSize + 12 == nBytes && nUncompressedSize > 0)
2053 : {
2054 : GByte *pabyUncompressedBuffer =
2055 0 : static_cast<GByte *>(VSI_MALLOC_VERBOSE(nUncompressedSize));
2056 0 : if (pabyUncompressedBuffer == nullptr)
2057 : {
2058 0 : return OGRERR_FAILURE;
2059 : }
2060 :
2061 0 : size_t nRealUncompressedSize = 0;
2062 0 : if (CPLZLibInflate(pabyShape + 12, nCompressedSize,
2063 : pabyUncompressedBuffer, nUncompressedSize,
2064 0 : &nRealUncompressedSize) == nullptr)
2065 : {
2066 0 : CPLError(CE_Failure, CPLE_AppDefined,
2067 : "CPLZLibInflate() failed");
2068 0 : VSIFree(pabyUncompressedBuffer);
2069 0 : return OGRERR_FAILURE;
2070 : }
2071 :
2072 : const OGRErr eErr =
2073 0 : OGRCreateFromShapeBin(pabyUncompressedBuffer, ppoGeom,
2074 : static_cast<int>(nRealUncompressedSize));
2075 :
2076 0 : VSIFree(pabyUncompressedBuffer);
2077 :
2078 0 : return eErr;
2079 : }
2080 : }
2081 :
2082 1914 : int nSHPType = pabyShape[0];
2083 :
2084 : /* -------------------------------------------------------------------- */
2085 : /* Return a NULL geometry when SHPT_NULL is encountered. */
2086 : /* Watch out, null return does not mean "bad data" it means */
2087 : /* "no geometry here". Watch the OGRErr for the error status */
2088 : /* -------------------------------------------------------------------- */
2089 1914 : if (nSHPType == SHPT_NULL)
2090 : {
2091 0 : *ppoGeom = nullptr;
2092 0 : return OGRERR_NONE;
2093 : }
2094 :
2095 : #if DEBUG_VERBOSE
2096 : CPLDebug("PGeo", "Shape type read from PGeo data is nSHPType = %d",
2097 : nSHPType);
2098 : #endif
2099 :
2100 1914 : const bool bIsExtended =
2101 1914 : nSHPType >= SHPT_GENERALPOLYLINE && nSHPType <= SHPT_GENERALMULTIPATCH;
2102 :
2103 1914 : const bool bHasZ =
2104 1802 : (nSHPType == SHPT_POINTZ || nSHPType == SHPT_POINTZM ||
2105 1690 : nSHPType == SHPT_MULTIPOINTZ || nSHPType == SHPT_MULTIPOINTZM ||
2106 1466 : nSHPType == SHPT_POLYGONZ || nSHPType == SHPT_POLYGONZM ||
2107 1238 : nSHPType == SHPT_ARCZ || nSHPType == SHPT_ARCZM ||
2108 3813 : nSHPType == SHPT_MULTIPATCH || nSHPType == SHPT_MULTIPATCHM ||
2109 97 : (bIsExtended && (pabyShape[3] & 0x80) != 0));
2110 :
2111 1914 : const bool bHasM =
2112 1914 : (nSHPType == SHPT_POINTM || nSHPType == SHPT_POINTZM ||
2113 1914 : nSHPType == SHPT_MULTIPOINTM || nSHPType == SHPT_MULTIPOINTZM ||
2114 1914 : nSHPType == SHPT_POLYGONM || nSHPType == SHPT_POLYGONZM ||
2115 1910 : nSHPType == SHPT_ARCM || nSHPType == SHPT_ARCZM ||
2116 3925 : nSHPType == SHPT_MULTIPATCHM ||
2117 97 : (bIsExtended && (pabyShape[3] & 0x40) != 0));
2118 :
2119 1914 : const bool bHasCurves = (bIsExtended && (pabyShape[3] & 0x20) != 0);
2120 :
2121 1914 : switch (nSHPType)
2122 : {
2123 44 : case SHPT_GENERALPOLYLINE:
2124 44 : nSHPType = SHPT_ARC;
2125 44 : break;
2126 53 : case SHPT_GENERALPOLYGON:
2127 53 : nSHPType = SHPT_POLYGON;
2128 53 : break;
2129 0 : case SHPT_GENERALPOINT:
2130 0 : nSHPType = SHPT_POINT;
2131 0 : break;
2132 0 : case SHPT_GENERALMULTIPOINT:
2133 0 : nSHPType = SHPT_MULTIPOINT;
2134 0 : break;
2135 0 : case SHPT_GENERALMULTIPATCH:
2136 0 : nSHPType = SHPT_MULTIPATCH;
2137 : }
2138 :
2139 : /* ==================================================================== */
2140 : /* Extract vertices for a Polygon or Arc. */
2141 : /* ==================================================================== */
2142 1914 : if (nSHPType == SHPT_ARC || nSHPType == SHPT_ARCZ ||
2143 1419 : nSHPType == SHPT_ARCM || nSHPType == SHPT_ARCZM ||
2144 911 : nSHPType == SHPT_POLYGON || nSHPType == SHPT_POLYGONZ ||
2145 687 : nSHPType == SHPT_POLYGONM || nSHPType == SHPT_POLYGONZM ||
2146 459 : nSHPType == SHPT_MULTIPATCH || nSHPType == SHPT_MULTIPATCHM)
2147 : {
2148 1455 : if (nBytes < 44)
2149 : {
2150 0 : CPLError(CE_Failure, CPLE_AppDefined,
2151 : "Corrupted Shape : nBytes=%d, nSHPType=%d", nBytes,
2152 : nSHPType);
2153 0 : return OGRERR_FAILURE;
2154 : }
2155 :
2156 : /* --------------------------------------------------------------------
2157 : */
2158 : /* Extract part/point count, and build vertex and part arrays */
2159 : /* to proper size. */
2160 : /* --------------------------------------------------------------------
2161 : */
2162 1455 : GInt32 nPoints = 0;
2163 1455 : memcpy(&nPoints, pabyShape + 40, 4);
2164 1455 : GInt32 nParts = 0;
2165 1455 : memcpy(&nParts, pabyShape + 36, 4);
2166 :
2167 1455 : CPL_LSBPTR32(&nPoints);
2168 1455 : CPL_LSBPTR32(&nParts);
2169 :
2170 1455 : if (nPoints < 0 || nParts < 0 || nPoints > 50 * 1000 * 1000 ||
2171 1455 : nParts > 10 * 1000 * 1000)
2172 : {
2173 0 : CPLError(CE_Failure, CPLE_AppDefined,
2174 : "Corrupted Shape : nPoints=%d, nParts=%d.", nPoints,
2175 : nParts);
2176 0 : return OGRERR_FAILURE;
2177 : }
2178 :
2179 1455 : const bool bIsMultiPatch =
2180 1455 : nSHPType == SHPT_MULTIPATCH || nSHPType == SHPT_MULTIPATCHM;
2181 :
2182 : // With the previous checks on nPoints and nParts,
2183 : // we should not overflow here and after
2184 : // since 50 M * (16 + 8 + 8) = 1 600 MB.
2185 1455 : int nRequiredSize = 44 + 4 * nParts + 16 * nPoints;
2186 1455 : if (bHasZ)
2187 : {
2188 699 : nRequiredSize += 16 + 8 * nPoints;
2189 : }
2190 1455 : if (bHasM)
2191 : {
2192 25 : nRequiredSize += 16 + 8 * nPoints;
2193 : }
2194 1455 : if (bIsMultiPatch)
2195 : {
2196 224 : nRequiredSize += 4 * nParts;
2197 : }
2198 1455 : if (nRequiredSize > nBytes)
2199 : {
2200 0 : CPLError(CE_Failure, CPLE_AppDefined,
2201 : "Corrupted Shape : nPoints=%d, nParts=%d, nBytes=%d, "
2202 : "nSHPType=%d, nRequiredSize=%d",
2203 : nPoints, nParts, nBytes, nSHPType, nRequiredSize);
2204 0 : return OGRERR_FAILURE;
2205 : }
2206 :
2207 : GInt32 *panPartStart =
2208 1455 : static_cast<GInt32 *>(VSI_MALLOC2_VERBOSE(nParts, sizeof(GInt32)));
2209 1455 : if (nParts != 0 && panPartStart == nullptr)
2210 : {
2211 0 : return OGRERR_FAILURE;
2212 : }
2213 :
2214 : /* --------------------------------------------------------------------
2215 : */
2216 : /* Copy out the part array from the record. */
2217 : /* --------------------------------------------------------------------
2218 : */
2219 1455 : memcpy(panPartStart, pabyShape + 44, 4 * nParts);
2220 3736 : for (int i = 0; i < nParts; i++)
2221 : {
2222 2281 : CPL_LSBPTR32(panPartStart + i);
2223 :
2224 : // Check that the offset is inside the vertex array.
2225 2281 : if (panPartStart[i] < 0 || panPartStart[i] >= nPoints)
2226 : {
2227 0 : CPLError(
2228 : CE_Failure, CPLE_AppDefined,
2229 : "Corrupted Shape : panPartStart[%d] = %d, nPoints = %d", i,
2230 0 : panPartStart[i], nPoints);
2231 0 : CPLFree(panPartStart);
2232 0 : return OGRERR_FAILURE;
2233 : }
2234 2281 : if (i > 0 && panPartStart[i] <= panPartStart[i - 1])
2235 : {
2236 0 : CPLError(CE_Failure, CPLE_AppDefined,
2237 : "Corrupted Shape : panPartStart[%d] = %d, "
2238 : "panPartStart[%d] = %d",
2239 0 : i, panPartStart[i], i - 1, panPartStart[i - 1]);
2240 0 : CPLFree(panPartStart);
2241 0 : return OGRERR_FAILURE;
2242 : }
2243 : }
2244 :
2245 1455 : int nOffset = 44 + 4 * nParts;
2246 :
2247 : /* --------------------------------------------------------------------
2248 : */
2249 : /* If this is a multipatch, we will also have parts types. */
2250 : /* --------------------------------------------------------------------
2251 : */
2252 1455 : GInt32 *panPartType = nullptr;
2253 :
2254 1455 : if (bIsMultiPatch)
2255 : {
2256 : panPartType = static_cast<GInt32 *>(
2257 224 : VSI_MALLOC2_VERBOSE(nParts, sizeof(GInt32)));
2258 224 : if (panPartType == nullptr)
2259 : {
2260 0 : CPLFree(panPartStart);
2261 0 : return OGRERR_FAILURE;
2262 : }
2263 :
2264 224 : memcpy(panPartType, pabyShape + nOffset, 4 * nParts);
2265 896 : for (int i = 0; i < nParts; i++)
2266 : {
2267 672 : CPL_LSBPTR32(panPartType + i);
2268 : }
2269 224 : nOffset += 4 * nParts;
2270 : }
2271 :
2272 : /* --------------------------------------------------------------------
2273 : */
2274 : /* Copy out the vertices from the record. */
2275 : /* --------------------------------------------------------------------
2276 : */
2277 : double *padfX =
2278 1455 : static_cast<double *>(VSI_MALLOC_VERBOSE(sizeof(double) * nPoints));
2279 : double *padfY =
2280 1455 : static_cast<double *>(VSI_MALLOC_VERBOSE(sizeof(double) * nPoints));
2281 : double *padfZ =
2282 1455 : static_cast<double *>(VSI_CALLOC_VERBOSE(sizeof(double), nPoints));
2283 1455 : double *padfM = static_cast<double *>(
2284 25 : bHasM ? VSI_CALLOC_VERBOSE(sizeof(double), nPoints) : nullptr);
2285 :
2286 1455 : if (nPoints != 0 && (padfX == nullptr || padfY == nullptr ||
2287 1454 : padfZ == nullptr || (bHasM && padfM == nullptr)))
2288 : {
2289 0 : CPLFree(panPartStart);
2290 0 : CPLFree(panPartType);
2291 0 : CPLFree(padfX);
2292 0 : CPLFree(padfY);
2293 0 : CPLFree(padfZ);
2294 0 : CPLFree(padfM);
2295 0 : return OGRERR_FAILURE;
2296 : }
2297 :
2298 16095 : for (int i = 0; i < nPoints; i++)
2299 : {
2300 14640 : memcpy(padfX + i, pabyShape + nOffset + i * 16, 8);
2301 14640 : memcpy(padfY + i, pabyShape + nOffset + i * 16 + 8, 8);
2302 14640 : CPL_LSBPTR64(padfX + i);
2303 14640 : CPL_LSBPTR64(padfY + i);
2304 : }
2305 :
2306 1455 : nOffset += 16 * nPoints;
2307 :
2308 : /* --------------------------------------------------------------------
2309 : */
2310 : /* If we have a Z coordinate, collect that now. */
2311 : /* --------------------------------------------------------------------
2312 : */
2313 1455 : if (bHasZ)
2314 : {
2315 5207 : for (int i = 0; i < nPoints; i++)
2316 : {
2317 4508 : memcpy(padfZ + i, pabyShape + nOffset + 16 + i * 8, 8);
2318 4508 : CPL_LSBPTR64(padfZ + i);
2319 : }
2320 :
2321 699 : nOffset += 16 + 8 * nPoints;
2322 : }
2323 :
2324 : /* --------------------------------------------------------------------
2325 : */
2326 : /* If we have a M coordinate, collect that now. */
2327 : /* --------------------------------------------------------------------
2328 : */
2329 1455 : if (bHasM)
2330 : {
2331 25 : bool bIsAllNAN = nPoints > 0;
2332 155 : for (int i = 0; i < nPoints; i++)
2333 : {
2334 130 : memcpy(padfM + i, pabyShape + nOffset + 16 + i * 8, 8);
2335 130 : CPL_LSBPTR64(padfM + i);
2336 130 : bIsAllNAN &= std::isnan(padfM[i]);
2337 : }
2338 25 : if (bIsAllNAN)
2339 : {
2340 4 : CPLFree(padfM);
2341 4 : padfM = nullptr;
2342 : }
2343 :
2344 25 : nOffset += 16 + 8 * nPoints;
2345 : }
2346 :
2347 : /* --------------------------------------------------------------------
2348 : */
2349 : /* If we have curves, collect them now. */
2350 : /* --------------------------------------------------------------------
2351 : */
2352 1455 : int nCurves = 0;
2353 1455 : CurveSegment *pasCurves = nullptr;
2354 1455 : if (bHasCurves && nOffset + 4 <= nBytes)
2355 : {
2356 97 : memcpy(&nCurves, pabyShape + nOffset, 4);
2357 97 : CPL_LSBPTR32(&nCurves);
2358 97 : nOffset += 4;
2359 : #ifdef DEBUG_VERBOSE
2360 : CPLDebug("OGR", "nCurves = %d", nCurves);
2361 : #endif
2362 97 : if (nCurves < 0 || nCurves > (nBytes - nOffset) / (8 + 20))
2363 : {
2364 0 : CPLDebug("OGR", "Invalid nCurves = %d", nCurves);
2365 0 : nCurves = 0;
2366 : }
2367 : pasCurves = static_cast<CurveSegment *>(
2368 97 : VSI_MALLOC2_VERBOSE(sizeof(CurveSegment), nCurves));
2369 97 : if (pasCurves == nullptr)
2370 : {
2371 0 : nCurves = 0;
2372 : }
2373 97 : int iCurve = 0;
2374 272 : for (int i = 0; i < nCurves; i++)
2375 : {
2376 175 : if (nOffset + 8 > nBytes)
2377 : {
2378 0 : CPLDebug("OGR", "Not enough bytes");
2379 0 : break;
2380 : }
2381 175 : int nStartPointIdx = 0;
2382 175 : memcpy(&nStartPointIdx, pabyShape + nOffset, 4);
2383 175 : CPL_LSBPTR32(&nStartPointIdx);
2384 175 : nOffset += 4;
2385 175 : int nSegmentType = 0;
2386 175 : memcpy(&nSegmentType, pabyShape + nOffset, 4);
2387 175 : CPL_LSBPTR32(&nSegmentType);
2388 175 : nOffset += 4;
2389 : #ifdef DEBUG_VERBOSE
2390 : CPLDebug("OGR", "[%d] nStartPointIdx = %d, segmentType = %d", i,
2391 : nSegmentType, nSegmentType);
2392 : #endif
2393 175 : if (nStartPointIdx < 0 || nStartPointIdx >= nPoints ||
2394 78 : (iCurve > 0 &&
2395 78 : nStartPointIdx <= pasCurves[iCurve - 1].nStartPointIdx))
2396 : {
2397 0 : CPLDebug("OGR", "Invalid nStartPointIdx = %d",
2398 : nStartPointIdx);
2399 0 : break;
2400 : }
2401 175 : pasCurves[iCurve].nStartPointIdx = nStartPointIdx;
2402 175 : if (nSegmentType == EXT_SHAPE_SEGMENT_ARC)
2403 : {
2404 126 : if (nOffset + 20 > nBytes)
2405 : {
2406 0 : CPLDebug("OGR", "Not enough bytes");
2407 0 : break;
2408 : }
2409 126 : double dfVal1 = 0.0;
2410 126 : double dfVal2 = 0.0;
2411 126 : memcpy(&dfVal1, pabyShape + nOffset + 0, 8);
2412 126 : CPL_LSBPTR64(&dfVal1);
2413 126 : memcpy(&dfVal2, pabyShape + nOffset + 8, 8);
2414 126 : CPL_LSBPTR64(&dfVal2);
2415 126 : int nBits = 0;
2416 126 : memcpy(&nBits, pabyShape + nOffset + 16, 4);
2417 126 : CPL_LSBPTR32(&nBits);
2418 :
2419 : #ifdef DEBUG_VERBOSE
2420 : CPLDebug("OGR", "Arc: ");
2421 : CPLDebug("OGR", " dfVal1 = %f, dfVal2 = %f, nBits=%X",
2422 : dfVal1, dfVal2, nBits);
2423 : if (nBits & EXT_SHAPE_ARC_EMPTY)
2424 : CPLDebug("OGR", " IsEmpty");
2425 : if (nBits & EXT_SHAPE_ARC_CCW)
2426 : CPLDebug("OGR", " IsCCW");
2427 : if (nBits & EXT_SHAPE_ARC_MINOR)
2428 : CPLDebug("OGR", " IsMinor");
2429 : if (nBits & EXT_SHAPE_ARC_LINE)
2430 : CPLDebug("OGR", " IsLine");
2431 : if (nBits & EXT_SHAPE_ARC_POINT)
2432 : CPLDebug("OGR", " IsPoint");
2433 : if (nBits & EXT_SHAPE_ARC_IP)
2434 : CPLDebug("OGR", " DefinedIP");
2435 : #endif
2436 126 : if ((nBits & EXT_SHAPE_ARC_IP) != 0 &&
2437 114 : (nBits & EXT_SHAPE_ARC_LINE) == 0)
2438 : {
2439 113 : pasCurves[iCurve].eType = CURVE_ARC_INTERIOR_POINT;
2440 113 : pasCurves[iCurve].u.ArcByIntermediatePoint.dfX = dfVal1;
2441 113 : pasCurves[iCurve].u.ArcByIntermediatePoint.dfY = dfVal2;
2442 113 : iCurve++;
2443 : }
2444 13 : else if ((nBits & EXT_SHAPE_ARC_EMPTY) == 0 &&
2445 13 : (nBits & EXT_SHAPE_ARC_LINE) == 0 &&
2446 12 : (nBits & EXT_SHAPE_ARC_POINT) == 0)
2447 : {
2448 : // This is the old deprecated way
2449 12 : pasCurves[iCurve].eType = CURVE_ARC_CENTER_POINT;
2450 12 : pasCurves[iCurve].u.ArcByCenterPoint.dfX = dfVal1;
2451 12 : pasCurves[iCurve].u.ArcByCenterPoint.dfY = dfVal2;
2452 12 : pasCurves[iCurve].u.ArcByCenterPoint.bIsCCW =
2453 12 : (nBits & EXT_SHAPE_ARC_CCW) != 0;
2454 12 : iCurve++;
2455 : }
2456 126 : nOffset += 16 + 4;
2457 : }
2458 49 : else if (nSegmentType == EXT_SHAPE_SEGMENT_BEZIER)
2459 : {
2460 36 : if (nOffset + 32 > nBytes)
2461 : {
2462 0 : CPLDebug("OGR", "Not enough bytes");
2463 0 : break;
2464 : }
2465 36 : double dfX1 = 0.0;
2466 36 : double dfY1 = 0.0;
2467 36 : memcpy(&dfX1, pabyShape + nOffset + 0, 8);
2468 36 : CPL_LSBPTR64(&dfX1);
2469 36 : memcpy(&dfY1, pabyShape + nOffset + 8, 8);
2470 36 : CPL_LSBPTR64(&dfY1);
2471 36 : double dfX2 = 0.0;
2472 36 : double dfY2 = 0.0;
2473 36 : memcpy(&dfX2, pabyShape + nOffset + 16, 8);
2474 36 : CPL_LSBPTR64(&dfX2);
2475 36 : memcpy(&dfY2, pabyShape + nOffset + 24, 8);
2476 36 : CPL_LSBPTR64(&dfY2);
2477 : #ifdef DEBUG_VERBOSE
2478 : CPLDebug("OGR", "Bezier:");
2479 : CPLDebug("OGR", " dfX1 = %f, dfY1 = %f", dfX1, dfY1);
2480 : CPLDebug("OGR", " dfX2 = %f, dfY2 = %f", dfX2, dfY2);
2481 : #endif
2482 36 : pasCurves[iCurve].eType = CURVE_BEZIER;
2483 36 : pasCurves[iCurve].u.Bezier.dfX1 = dfX1;
2484 36 : pasCurves[iCurve].u.Bezier.dfY1 = dfY1;
2485 36 : pasCurves[iCurve].u.Bezier.dfX2 = dfX2;
2486 36 : pasCurves[iCurve].u.Bezier.dfY2 = dfY2;
2487 36 : iCurve++;
2488 36 : nOffset += 32;
2489 : }
2490 13 : else if (nSegmentType == EXT_SHAPE_SEGMENT_ELLIPSE)
2491 : {
2492 13 : if (nOffset + 44 > nBytes)
2493 : {
2494 0 : CPLDebug("OGR", "Not enough bytes");
2495 0 : break;
2496 : }
2497 13 : double dfVS0 = 0.0;
2498 13 : memcpy(&dfVS0, pabyShape + nOffset, 8);
2499 13 : nOffset += 8;
2500 13 : CPL_LSBPTR64(&dfVS0);
2501 :
2502 13 : double dfVS1 = 0.0;
2503 13 : memcpy(&dfVS1, pabyShape + nOffset, 8);
2504 13 : nOffset += 8;
2505 13 : CPL_LSBPTR64(&dfVS1);
2506 :
2507 13 : double dfRotationOrFromV = 0.0;
2508 13 : memcpy(&dfRotationOrFromV, pabyShape + nOffset, 8);
2509 13 : nOffset += 8;
2510 13 : CPL_LSBPTR64(&dfRotationOrFromV);
2511 :
2512 13 : double dfSemiMajor = 0.0;
2513 13 : memcpy(&dfSemiMajor, pabyShape + nOffset, 8);
2514 13 : nOffset += 8;
2515 13 : CPL_LSBPTR64(&dfSemiMajor);
2516 :
2517 13 : double dfMinorMajorRatioOrDeltaV = 0.0;
2518 13 : memcpy(&dfMinorMajorRatioOrDeltaV, pabyShape + nOffset, 8);
2519 13 : nOffset += 8;
2520 13 : CPL_LSBPTR64(&dfMinorMajorRatioOrDeltaV);
2521 :
2522 13 : int nBits = 0;
2523 13 : memcpy(&nBits, pabyShape + nOffset, 4);
2524 13 : CPL_LSBPTR32(&nBits);
2525 13 : nOffset += 4;
2526 :
2527 : #ifdef DEBUG_VERBOSE
2528 : CPLDebug("OGR", "Ellipse:");
2529 : CPLDebug("OGR", " dfVS0 = %f", dfVS0);
2530 : CPLDebug("OGR", " dfVS1 = %f", dfVS1);
2531 : CPLDebug("OGR", " dfRotationOrFromV = %f",
2532 : dfRotationOrFromV);
2533 : CPLDebug("OGR", " dfSemiMajor = %f", dfSemiMajor);
2534 : CPLDebug("OGR", " dfMinorMajorRatioOrDeltaV = %f",
2535 : dfMinorMajorRatioOrDeltaV);
2536 : CPLDebug("OGR", " nBits=%X", nBits);
2537 :
2538 : if (nBits & EXT_SHAPE_ELLIPSE_EMPTY)
2539 : CPLDebug("OGR", " IsEmpty");
2540 : if (nBits & EXT_SHAPE_ELLIPSE_LINE)
2541 : CPLDebug("OGR", " IsLine");
2542 : if (nBits & EXT_SHAPE_ELLIPSE_POINT)
2543 : CPLDebug("OGR", " IsPoint");
2544 : if (nBits & EXT_SHAPE_ELLIPSE_CIRCULAR)
2545 : CPLDebug("OGR", " IsCircular");
2546 : if (nBits & EXT_SHAPE_ELLIPSE_CENTER_TO)
2547 : CPLDebug("OGR", " CenterTo");
2548 : if (nBits & EXT_SHAPE_ELLIPSE_CENTER_FROM)
2549 : CPLDebug("OGR", " CenterFrom");
2550 : if (nBits & EXT_SHAPE_ELLIPSE_CCW)
2551 : CPLDebug("OGR", " IsCCW");
2552 : if (nBits & EXT_SHAPE_ELLIPSE_MINOR)
2553 : CPLDebug("OGR", " IsMinor");
2554 : if (nBits & EXT_SHAPE_ELLIPSE_COMPLETE)
2555 : CPLDebug("OGR", " IsComplete");
2556 : #endif
2557 :
2558 13 : if ((nBits & EXT_SHAPE_ELLIPSE_CENTER_TO) == 0 &&
2559 13 : (nBits & EXT_SHAPE_ELLIPSE_CENTER_FROM) == 0)
2560 : {
2561 13 : pasCurves[iCurve].eType = CURVE_ELLIPSE_BY_CENTER;
2562 13 : pasCurves[iCurve].u.EllipseByCenter.dfX = dfVS0;
2563 13 : pasCurves[iCurve].u.EllipseByCenter.dfY = dfVS1;
2564 13 : pasCurves[iCurve].u.EllipseByCenter.dfRotationDeg =
2565 13 : dfRotationOrFromV / M_PI * 180;
2566 13 : pasCurves[iCurve].u.EllipseByCenter.dfSemiMajor =
2567 : dfSemiMajor;
2568 13 : pasCurves[iCurve].u.EllipseByCenter.dfRatioSemiMinor =
2569 : dfMinorMajorRatioOrDeltaV;
2570 13 : pasCurves[iCurve].u.EllipseByCenter.bIsMinor =
2571 13 : ((nBits & EXT_SHAPE_ELLIPSE_MINOR) != 0);
2572 13 : pasCurves[iCurve].u.EllipseByCenter.bIsComplete =
2573 13 : ((nBits & EXT_SHAPE_ELLIPSE_COMPLETE) != 0);
2574 13 : iCurve++;
2575 : }
2576 : }
2577 : else
2578 : {
2579 0 : CPLDebug("OGR", "unhandled segmentType = %d", nSegmentType);
2580 : }
2581 : }
2582 :
2583 97 : nCurves = iCurve;
2584 : }
2585 :
2586 : /* --------------------------------------------------------------------
2587 : */
2588 : /* Build corresponding OGR objects. */
2589 : /* --------------------------------------------------------------------
2590 : */
2591 1455 : if (nSHPType == SHPT_ARC || nSHPType == SHPT_ARCZ ||
2592 960 : nSHPType == SHPT_ARCM || nSHPType == SHPT_ARCZM)
2593 : {
2594 : /* --------------------------------------------------------------------
2595 : */
2596 : /* Arc - As LineString */
2597 : /* --------------------------------------------------------------------
2598 : */
2599 495 : if (nParts == 1)
2600 : {
2601 378 : if (nCurves > 0)
2602 : {
2603 38 : *ppoGeom = OGRShapeCreateCompoundCurve(
2604 : 0, nPoints, pasCurves, nCurves, 0, padfX, padfY,
2605 : bHasZ ? padfZ : nullptr, padfM, nullptr);
2606 : }
2607 : else
2608 : {
2609 340 : OGRLineString *poLine = new OGRLineString();
2610 340 : *ppoGeom = poLine;
2611 :
2612 340 : poLine->setPoints(nPoints, padfX, padfY, padfZ, padfM);
2613 : }
2614 : }
2615 :
2616 : /* --------------------------------------------------------------------
2617 : */
2618 : /* Arc - As MultiLineString */
2619 : /* --------------------------------------------------------------------
2620 : */
2621 : else
2622 : {
2623 117 : if (nCurves > 0)
2624 : {
2625 5 : OGRMultiCurve *poMulti = new OGRMultiCurve;
2626 5 : *ppoGeom = poMulti;
2627 :
2628 5 : int iCurveIdx = 0;
2629 18 : for (int i = 0; i < nParts; i++)
2630 : {
2631 13 : const int nVerticesInThisPart =
2632 13 : i == nParts - 1
2633 13 : ? nPoints - panPartStart[i]
2634 8 : : panPartStart[i + 1] - panPartStart[i];
2635 :
2636 13 : OGRCurve *poCurve = OGRShapeCreateCompoundCurve(
2637 13 : panPartStart[i], nVerticesInThisPart, pasCurves,
2638 : nCurves, iCurveIdx, padfX, padfY,
2639 : bHasZ ? padfZ : nullptr, padfM, &iCurveIdx);
2640 26 : if (poCurve == nullptr || poMulti->addGeometryDirectly(
2641 13 : poCurve) != OGRERR_NONE)
2642 : {
2643 0 : delete poCurve;
2644 0 : delete poMulti;
2645 0 : *ppoGeom = nullptr;
2646 0 : break;
2647 : }
2648 : }
2649 : }
2650 : else
2651 : {
2652 112 : OGRMultiLineString *poMulti = new OGRMultiLineString;
2653 112 : *ppoGeom = poMulti;
2654 :
2655 336 : for (int i = 0; i < nParts; i++)
2656 : {
2657 224 : OGRLineString *poLine = new OGRLineString;
2658 224 : const int nVerticesInThisPart =
2659 224 : i == nParts - 1
2660 224 : ? nPoints - panPartStart[i]
2661 112 : : panPartStart[i + 1] - panPartStart[i];
2662 :
2663 224 : poLine->setPoints(
2664 224 : nVerticesInThisPart, padfX + panPartStart[i],
2665 224 : padfY + panPartStart[i], padfZ + panPartStart[i],
2666 0 : padfM != nullptr ? padfM + panPartStart[i]
2667 : : nullptr);
2668 :
2669 224 : poMulti->addGeometryDirectly(poLine);
2670 : }
2671 : }
2672 495 : }
2673 : } // ARC.
2674 :
2675 : /* --------------------------------------------------------------------
2676 : */
2677 : /* Polygon */
2678 : /* --------------------------------------------------------------------
2679 : */
2680 960 : else if (nSHPType == SHPT_POLYGON || nSHPType == SHPT_POLYGONZ ||
2681 228 : nSHPType == SHPT_POLYGONM || nSHPType == SHPT_POLYGONZM)
2682 : {
2683 736 : if (nCurves > 0 && nParts != 0)
2684 : {
2685 53 : if (nParts == 1)
2686 : {
2687 41 : OGRCurvePolygon *poOGRPoly = new OGRCurvePolygon;
2688 41 : *ppoGeom = poOGRPoly;
2689 41 : const int nVerticesInThisPart = nPoints - panPartStart[0];
2690 :
2691 41 : OGRCurve *poRing = OGRShapeCreateCompoundCurve(
2692 : panPartStart[0], nVerticesInThisPart, pasCurves,
2693 : nCurves, 0, padfX, padfY, bHasZ ? padfZ : nullptr,
2694 : padfM, nullptr);
2695 82 : if (poRing == nullptr ||
2696 41 : poOGRPoly->addRingDirectly(poRing) != OGRERR_NONE)
2697 : {
2698 0 : delete poRing;
2699 0 : delete poOGRPoly;
2700 0 : *ppoGeom = nullptr;
2701 : }
2702 : }
2703 : else
2704 : {
2705 12 : OGRGeometry *poOGR = nullptr;
2706 : OGRCurvePolygon **tabPolygons =
2707 12 : new OGRCurvePolygon *[nParts];
2708 :
2709 12 : int iCurveIdx = 0;
2710 42 : for (int i = 0; i < nParts; i++)
2711 : {
2712 30 : tabPolygons[i] = new OGRCurvePolygon();
2713 30 : const int nVerticesInThisPart =
2714 30 : i == nParts - 1
2715 30 : ? nPoints - panPartStart[i]
2716 18 : : panPartStart[i + 1] - panPartStart[i];
2717 :
2718 30 : OGRCurve *poRing = OGRShapeCreateCompoundCurve(
2719 30 : panPartStart[i], nVerticesInThisPart, pasCurves,
2720 : nCurves, iCurveIdx, padfX, padfY,
2721 : bHasZ ? padfZ : nullptr, padfM, &iCurveIdx);
2722 60 : if (poRing == nullptr ||
2723 30 : tabPolygons[i]->addRingDirectly(poRing) !=
2724 : OGRERR_NONE)
2725 : {
2726 0 : delete poRing;
2727 0 : for (; i >= 0; --i)
2728 0 : delete tabPolygons[i];
2729 0 : delete[] tabPolygons;
2730 0 : tabPolygons = nullptr;
2731 0 : *ppoGeom = nullptr;
2732 0 : break;
2733 : }
2734 : }
2735 :
2736 12 : if (tabPolygons != nullptr)
2737 : {
2738 12 : int isValidGeometry = FALSE;
2739 12 : const char *papszOptions[] = {"METHOD=ONLY_CCW",
2740 : nullptr};
2741 12 : poOGR = OGRGeometryFactory::organizePolygons(
2742 : reinterpret_cast<OGRGeometry **>(tabPolygons),
2743 : nParts, &isValidGeometry, papszOptions);
2744 :
2745 12 : if (!isValidGeometry)
2746 : {
2747 0 : CPLError(CE_Warning, CPLE_AppDefined,
2748 : "Geometry of polygon cannot be translated "
2749 : "to Simple Geometry. All polygons will "
2750 : "be contained in a multipolygon.");
2751 : }
2752 :
2753 12 : *ppoGeom = poOGR;
2754 12 : delete[] tabPolygons;
2755 : }
2756 53 : }
2757 : }
2758 683 : else if (nParts != 0)
2759 : {
2760 682 : if (nParts == 1)
2761 : {
2762 568 : OGRPolygon *poOGRPoly = new OGRPolygon;
2763 568 : *ppoGeom = poOGRPoly;
2764 568 : OGRLinearRing *poRing = new OGRLinearRing;
2765 568 : int nVerticesInThisPart = nPoints - panPartStart[0];
2766 :
2767 568 : poRing->setPoints(
2768 568 : nVerticesInThisPart, padfX + panPartStart[0],
2769 568 : padfY + panPartStart[0], padfZ + panPartStart[0],
2770 2 : padfM != nullptr ? padfM + panPartStart[0] : nullptr);
2771 :
2772 568 : if (poOGRPoly->addRingDirectly(poRing) != OGRERR_NONE)
2773 : {
2774 0 : delete poRing;
2775 0 : delete poOGRPoly;
2776 0 : *ppoGeom = nullptr;
2777 : }
2778 : }
2779 : else
2780 : {
2781 114 : OGRGeometry *poOGR = nullptr;
2782 114 : OGRPolygon **tabPolygons = new OGRPolygon *[nParts];
2783 :
2784 469 : for (int i = 0; i < nParts; i++)
2785 : {
2786 355 : tabPolygons[i] = new OGRPolygon();
2787 355 : OGRLinearRing *poRing = new OGRLinearRing;
2788 355 : const int nVerticesInThisPart =
2789 355 : i == nParts - 1
2790 355 : ? nPoints - panPartStart[i]
2791 241 : : panPartStart[i + 1] - panPartStart[i];
2792 :
2793 355 : poRing->setPoints(
2794 355 : nVerticesInThisPart, padfX + panPartStart[i],
2795 355 : padfY + panPartStart[i], padfZ + panPartStart[i],
2796 0 : padfM != nullptr ? padfM + panPartStart[i]
2797 : : nullptr);
2798 355 : if (tabPolygons[i]->addRingDirectly(poRing) !=
2799 : OGRERR_NONE)
2800 : {
2801 0 : delete poRing;
2802 0 : for (; i >= 0; --i)
2803 0 : delete tabPolygons[i];
2804 0 : delete[] tabPolygons;
2805 0 : tabPolygons = nullptr;
2806 0 : *ppoGeom = nullptr;
2807 0 : break;
2808 : }
2809 : }
2810 :
2811 114 : if (tabPolygons != nullptr)
2812 : {
2813 114 : int isValidGeometry = FALSE;
2814 : // The outer ring is supposed to be clockwise oriented
2815 : // If it is not, then use the default/slow method.
2816 114 : const char *papszOptions[] = {
2817 114 : !(tabPolygons[0]->getExteriorRing()->isClockwise())
2818 114 : ? "METHOD=DEFAULT"
2819 : : "METHOD=ONLY_CCW",
2820 114 : nullptr};
2821 114 : poOGR = OGRGeometryFactory::organizePolygons(
2822 : reinterpret_cast<OGRGeometry **>(tabPolygons),
2823 : nParts, &isValidGeometry, papszOptions);
2824 :
2825 114 : if (!isValidGeometry)
2826 : {
2827 0 : CPLError(CE_Warning, CPLE_AppDefined,
2828 : "Geometry of polygon cannot be translated "
2829 : "to Simple Geometry. All polygons will be "
2830 : "contained in a multipolygon.");
2831 : }
2832 :
2833 114 : *ppoGeom = poOGR;
2834 114 : delete[] tabPolygons;
2835 : }
2836 : }
2837 : }
2838 : else
2839 : {
2840 1 : *ppoGeom = new OGRPolygon();
2841 736 : }
2842 : } // Polygon.
2843 :
2844 : /* --------------------------------------------------------------------
2845 : */
2846 : /* Multipatch */
2847 : /* --------------------------------------------------------------------
2848 : */
2849 224 : else if (bIsMultiPatch)
2850 : {
2851 224 : *ppoGeom =
2852 224 : OGRCreateFromMultiPatch(nParts, panPartStart, panPartType,
2853 : nPoints, padfX, padfY, padfZ);
2854 : }
2855 :
2856 1455 : CPLFree(panPartStart);
2857 1455 : CPLFree(panPartType);
2858 1455 : CPLFree(padfX);
2859 1455 : CPLFree(padfY);
2860 1455 : CPLFree(padfZ);
2861 1455 : CPLFree(padfM);
2862 1455 : CPLFree(pasCurves);
2863 :
2864 1455 : if (*ppoGeom != nullptr)
2865 : {
2866 1455 : if (!bHasZ)
2867 756 : (*ppoGeom)->set3D(FALSE);
2868 : }
2869 :
2870 1455 : return *ppoGeom != nullptr ? OGRERR_NONE : OGRERR_FAILURE;
2871 : }
2872 :
2873 : /* ==================================================================== */
2874 : /* Extract vertices for a MultiPoint. */
2875 : /* ==================================================================== */
2876 459 : else if (nSHPType == SHPT_MULTIPOINT || nSHPType == SHPT_MULTIPOINTM ||
2877 235 : nSHPType == SHPT_MULTIPOINTZ || nSHPType == SHPT_MULTIPOINTZM)
2878 : {
2879 224 : GInt32 nPoints = 0;
2880 224 : memcpy(&nPoints, pabyShape + 36, 4);
2881 224 : CPL_LSBPTR32(&nPoints);
2882 :
2883 224 : if (nPoints < 0 || nPoints > 50 * 1000 * 1000)
2884 : {
2885 0 : CPLError(CE_Failure, CPLE_AppDefined,
2886 : "Corrupted Shape : nPoints=%d.", nPoints);
2887 0 : return OGRERR_FAILURE;
2888 : }
2889 :
2890 224 : const GInt32 nOffsetZ = 40 + 2 * 8 * nPoints + 2 * 8;
2891 224 : GInt32 nOffsetM = 0;
2892 224 : if (bHasM)
2893 0 : nOffsetM = bHasZ ? nOffsetZ + 2 * 8 * 8 * nPoints : nOffsetZ;
2894 :
2895 224 : OGRMultiPoint *poMultiPt = new OGRMultiPoint;
2896 224 : *ppoGeom = poMultiPt;
2897 :
2898 672 : for (int i = 0; i < nPoints; i++)
2899 : {
2900 448 : OGRPoint *poPt = new OGRPoint;
2901 :
2902 : // Copy X.
2903 448 : double x = 0.0;
2904 448 : memcpy(&x, pabyShape + 40 + i * 16, 8);
2905 448 : CPL_LSBPTR64(&x);
2906 448 : poPt->setX(x);
2907 :
2908 : // Copy Y.
2909 448 : double y = 0.0;
2910 448 : memcpy(&y, pabyShape + 40 + i * 16 + 8, 8);
2911 448 : CPL_LSBPTR64(&y);
2912 448 : poPt->setY(y);
2913 :
2914 : // Copy Z.
2915 448 : if (bHasZ)
2916 : {
2917 224 : double z = 0.0;
2918 224 : memcpy(&z, pabyShape + nOffsetZ + i * 8, 8);
2919 224 : CPL_LSBPTR64(&z);
2920 224 : poPt->setZ(z);
2921 : }
2922 :
2923 : // Copy M.
2924 448 : if (bHasM)
2925 : {
2926 0 : double m = 0.0;
2927 0 : memcpy(&m, pabyShape + nOffsetM + i * 8, 8);
2928 0 : CPL_LSBPTR64(&m);
2929 0 : poPt->setM(m);
2930 : }
2931 :
2932 448 : poMultiPt->addGeometryDirectly(poPt);
2933 : }
2934 :
2935 224 : poMultiPt->set3D(bHasZ);
2936 224 : poMultiPt->setMeasured(bHasM);
2937 :
2938 224 : return OGRERR_NONE;
2939 : }
2940 :
2941 : /* ==================================================================== */
2942 : /* Extract vertices for a point. */
2943 : /* ==================================================================== */
2944 235 : else if (nSHPType == SHPT_POINT || nSHPType == SHPT_POINTM ||
2945 0 : nSHPType == SHPT_POINTZ || nSHPType == SHPT_POINTZM)
2946 : {
2947 235 : if (nBytes < 4 + 8 + 8 + ((bHasZ) ? 8 : 0) + ((bHasM) ? 8 : 0))
2948 : {
2949 0 : CPLError(CE_Failure, CPLE_AppDefined,
2950 : "Corrupted Shape : nBytes=%d, nSHPType=%d", nBytes,
2951 : nSHPType);
2952 0 : return OGRERR_FAILURE;
2953 : }
2954 :
2955 235 : double dfX = 0.0;
2956 235 : double dfY = 0.0;
2957 :
2958 235 : memcpy(&dfX, pabyShape + 4, 8);
2959 235 : memcpy(&dfY, pabyShape + 4 + 8, 8);
2960 :
2961 235 : CPL_LSBPTR64(&dfX);
2962 235 : CPL_LSBPTR64(&dfY);
2963 : // int nOffset = 20 + 8;
2964 :
2965 235 : double dfZ = 0.0;
2966 235 : if (bHasZ)
2967 : {
2968 112 : memcpy(&dfZ, pabyShape + 4 + 16, 8);
2969 112 : CPL_LSBPTR64(&dfZ);
2970 : }
2971 :
2972 235 : double dfM = 0.0;
2973 235 : if (bHasM)
2974 : {
2975 0 : memcpy(&dfM, pabyShape + 4 + 16 + ((bHasZ) ? 8 : 0), 8);
2976 0 : CPL_LSBPTR64(&dfM);
2977 : }
2978 :
2979 235 : if (bHasZ && bHasM)
2980 : {
2981 0 : *ppoGeom = new OGRPoint(dfX, dfY, dfZ, dfM);
2982 : }
2983 235 : else if (bHasZ)
2984 : {
2985 112 : *ppoGeom = new OGRPoint(dfX, dfY, dfZ);
2986 : }
2987 123 : else if (bHasM)
2988 : {
2989 0 : OGRPoint *poPoint = new OGRPoint(dfX, dfY);
2990 0 : poPoint->setM(dfM);
2991 0 : *ppoGeom = poPoint;
2992 : }
2993 : else
2994 : {
2995 123 : *ppoGeom = new OGRPoint(dfX, dfY);
2996 : }
2997 :
2998 235 : return OGRERR_NONE;
2999 : }
3000 :
3001 0 : CPLError(CE_Failure, CPLE_AppDefined, "Unsupported geometry type: %d",
3002 : nSHPType);
3003 :
3004 0 : return OGRERR_FAILURE;
3005 : }
|