/src/gdal/ogr/ogrsf_frmts/shape/ogrshapelayer.cpp
Line  | Count  | Source (jump to first uncovered line)  | 
1  |  | /******************************************************************************  | 
2  |  |  *  | 
3  |  |  * Project:  OpenGIS Simple Features Reference Implementation  | 
4  |  |  * Purpose:  Implements OGRShapeLayer class.  | 
5  |  |  * Author:   Frank Warmerdam, warmerdam@pobox.com  | 
6  |  |  *  | 
7  |  |  ******************************************************************************  | 
8  |  |  * Copyright (c) 1999,  Les Technologies SoftMap Inc.  | 
9  |  |  * Copyright (c) 2007-2013, Even Rouault <even dot rouault at spatialys.com>  | 
10  |  |  *  | 
11  |  |  * SPDX-License-Identifier: MIT  | 
12  |  |  ****************************************************************************/  | 
13  |  |  | 
14  |  | #include "ogrshape.h"  | 
15  |  |  | 
16  |  | #include <cerrno>  | 
17  |  | #include <limits>  | 
18  |  | #include <cmath>  | 
19  |  | #include <cstddef>  | 
20  |  | #include <cstdio>  | 
21  |  | #include <cstdlib>  | 
22  |  | #include <cstring>  | 
23  |  | #include <ctime>  | 
24  |  | #include <algorithm>  | 
25  |  | #include <string>  | 
26  |  |  | 
27  |  | #include "cpl_conv.h"  | 
28  |  | #include "cpl_error.h"  | 
29  |  | #include "cpl_multiproc.h"  | 
30  |  | #include "cpl_port.h"  | 
31  |  | #include "cpl_string.h"  | 
32  |  | #include "cpl_time.h"  | 
33  |  | #include "cpl_vsi.h"  | 
34  |  | #include "ogr_core.h"  | 
35  |  | #include "ogr_feature.h"  | 
36  |  | #include "ogr_geometry.h"  | 
37  |  | #include "ogr_p.h"  | 
38  |  | #include "ogr_spatialref.h"  | 
39  |  | #include "ogr_srs_api.h"  | 
40  |  | #include "ogrlayerpool.h"  | 
41  |  | #include "ograrrowarrayhelper.h"  | 
42  |  | #include "ogrsf_frmts.h"  | 
43  |  | #include "shapefil.h"  | 
44  |  | #include "shp_vsi.h"  | 
45  |  |  | 
46  |  | /************************************************************************/  | 
47  |  | /*                           OGRShapeLayer()                            */  | 
48  |  | /************************************************************************/  | 
49  |  |  | 
50  |  | OGRShapeLayer::OGRShapeLayer(OGRShapeDataSource *poDSIn,  | 
51  |  |                              const char *pszFullNameIn, SHPHandle hSHPIn,  | 
52  |  |                              DBFHandle hDBFIn,  | 
53  |  |                              const OGRSpatialReference *poSRSIn, bool bSRSSetIn,  | 
54  |  |                              const std::string &osPrjFilename, bool bUpdate,  | 
55  |  |                              OGRwkbGeometryType eReqType,  | 
56  |  |                              CSLConstList papszCreateOptions)  | 
57  | 0  |     : OGRAbstractProxiedLayer(poDSIn->GetPool()), m_poDS(poDSIn),  | 
58  | 0  |       m_osFullName(pszFullNameIn), m_hSHP(hSHPIn), m_hDBF(hDBFIn),  | 
59  | 0  |       m_bUpdateAccess(bUpdate), m_eRequestedGeomType(eReqType),  | 
60  | 0  |       m_bHSHPWasNonNULL(hSHPIn != nullptr), m_bHDBFWasNonNULL(hDBFIn != nullptr)  | 
61  | 0  | { | 
62  | 0  |     if (m_hSHP != nullptr)  | 
63  | 0  |     { | 
64  | 0  |         m_nTotalShapeCount = m_hSHP->nRecords;  | 
65  | 0  |         if (m_hDBF != nullptr && m_hDBF->nRecords != m_nTotalShapeCount)  | 
66  | 0  |         { | 
67  | 0  |             CPLDebug("Shape", | 
68  | 0  |                      "Inconsistent record number in .shp (%d) and in .dbf (%d)",  | 
69  | 0  |                      m_hSHP->nRecords, m_hDBF->nRecords);  | 
70  | 0  |         }  | 
71  | 0  |     }  | 
72  | 0  |     else if (m_hDBF != nullptr)  | 
73  | 0  |     { | 
74  | 0  |         m_nTotalShapeCount = m_hDBF->nRecords;  | 
75  | 0  |     }  | 
76  | 0  | #ifdef DEBUG  | 
77  | 0  |     else  | 
78  | 0  |     { | 
79  | 0  |         CPLError(CE_Fatal, CPLE_AssertionFailed,  | 
80  | 0  |                  "Should not happen: Both m_hSHP and m_hDBF are nullptrs");  | 
81  | 0  |     }  | 
82  | 0  | #endif  | 
83  |  | 
  | 
84  | 0  |     if (!TouchLayer())  | 
85  | 0  |     { | 
86  | 0  |         CPLDebug("Shape", "TouchLayer in shape ctor failed. "); | 
87  | 0  |     }  | 
88  |  | 
  | 
89  | 0  |     if (m_hDBF != nullptr && m_hDBF->pszCodePage != nullptr)  | 
90  | 0  |     { | 
91  | 0  |         CPLDebug("Shape", "DBF Codepage = %s for %s", m_hDBF->pszCodePage, | 
92  | 0  |                  m_osFullName.c_str());  | 
93  |  |  | 
94  |  |         // Not too sure about this, but it seems like better than nothing.  | 
95  | 0  |         m_osEncoding = ConvertCodePage(m_hDBF->pszCodePage);  | 
96  | 0  |     }  | 
97  |  | 
  | 
98  | 0  |     if (m_hDBF != nullptr)  | 
99  | 0  |     { | 
100  | 0  |         if (!(m_hDBF->nUpdateYearSince1900 == 95 && m_hDBF->nUpdateMonth == 7 &&  | 
101  | 0  |               m_hDBF->nUpdateDay == 26))  | 
102  | 0  |         { | 
103  | 0  |             SetMetadataItem("DBF_DATE_LAST_UPDATE", | 
104  | 0  |                             CPLSPrintf("%04d-%02d-%02d", | 
105  | 0  |                                        m_hDBF->nUpdateYearSince1900 + 1900,  | 
106  | 0  |                                        m_hDBF->nUpdateMonth,  | 
107  | 0  |                                        m_hDBF->nUpdateDay));  | 
108  | 0  |         }  | 
109  | 0  |         struct tm tm;  | 
110  | 0  |         CPLUnixTimeToYMDHMS(time(nullptr), &tm);  | 
111  | 0  |         DBFSetLastModifiedDate(m_hDBF, tm.tm_year, tm.tm_mon + 1, tm.tm_mday);  | 
112  | 0  |     }  | 
113  |  | 
  | 
114  | 0  |     const char *pszShapeEncoding =  | 
115  | 0  |         CSLFetchNameValue(m_poDS->GetOpenOptions(), "ENCODING");  | 
116  | 0  |     if (pszShapeEncoding == nullptr && m_osEncoding == "")  | 
117  | 0  |         pszShapeEncoding = CSLFetchNameValue(papszCreateOptions, "ENCODING");  | 
118  | 0  |     if (pszShapeEncoding == nullptr)  | 
119  | 0  |         pszShapeEncoding = CPLGetConfigOption("SHAPE_ENCODING", nullptr); | 
120  | 0  |     if (pszShapeEncoding != nullptr)  | 
121  | 0  |         m_osEncoding = pszShapeEncoding;  | 
122  |  | 
  | 
123  | 0  |     if (m_osEncoding != "")  | 
124  | 0  |     { | 
125  | 0  |         CPLDebug("Shape", "Treating as encoding '%s'.", m_osEncoding.c_str()); | 
126  |  | 
  | 
127  | 0  |         if (!OGRShapeLayer::TestCapability(OLCStringsAsUTF8))  | 
128  | 0  |         { | 
129  | 0  |             CPLDebug("Shape", "Cannot recode from '%s'. Disabling recoding", | 
130  | 0  |                      m_osEncoding.c_str());  | 
131  | 0  |             m_osEncoding = "";  | 
132  | 0  |         }  | 
133  | 0  |     }  | 
134  | 0  |     SetMetadataItem("SOURCE_ENCODING", m_osEncoding, "SHAPEFILE"); | 
135  |  | 
  | 
136  | 0  |     m_poFeatureDefn = SHPReadOGRFeatureDefn(  | 
137  | 0  |         CPLGetBasenameSafe(m_osFullName.c_str()).c_str(), m_hSHP, m_hDBF,  | 
138  | 0  |         m_osEncoding,  | 
139  | 0  |         CPLFetchBool(m_poDS->GetOpenOptions(), "ADJUST_TYPE", false));  | 
140  |  |  | 
141  |  |     // To make sure that  | 
142  |  |     //  GetLayerDefn()->GetGeomFieldDefn(0)->GetSpatialRef() == GetSpatialRef()  | 
143  | 0  |     OGRwkbGeometryType eGeomType = m_poFeatureDefn->GetGeomType();  | 
144  | 0  |     if (eGeomType != wkbNone)  | 
145  | 0  |     { | 
146  | 0  |         OGRwkbGeometryType eType = wkbUnknown;  | 
147  |  | 
  | 
148  | 0  |         if (m_eRequestedGeomType == wkbNone)  | 
149  | 0  |         { | 
150  | 0  |             eType = eGeomType;  | 
151  |  | 
  | 
152  | 0  |             const char *pszAdjustGeomType = CSLFetchNameValueDef(  | 
153  | 0  |                 m_poDS->GetOpenOptions(), "ADJUST_GEOM_TYPE", "FIRST_SHAPE");  | 
154  | 0  |             const bool bFirstShape = EQUAL(pszAdjustGeomType, "FIRST_SHAPE");  | 
155  | 0  |             const bool bAllShapes = EQUAL(pszAdjustGeomType, "ALL_SHAPES");  | 
156  | 0  |             if ((m_hSHP != nullptr) && (m_hSHP->nRecords > 0) &&  | 
157  | 0  |                 wkbHasM(eType) && (bFirstShape || bAllShapes))  | 
158  | 0  |             { | 
159  | 0  |                 bool bMIsUsed = false;  | 
160  | 0  |                 for (int iShape = 0; iShape < m_hSHP->nRecords; iShape++)  | 
161  | 0  |                 { | 
162  | 0  |                     SHPObject *psShape = SHPReadObject(m_hSHP, iShape);  | 
163  | 0  |                     if (psShape)  | 
164  | 0  |                     { | 
165  | 0  |                         if (psShape->bMeasureIsUsed && psShape->nVertices > 0 &&  | 
166  | 0  |                             psShape->padfM != nullptr)  | 
167  | 0  |                         { | 
168  | 0  |                             for (int i = 0; i < psShape->nVertices; i++)  | 
169  | 0  |                             { | 
170  |  |                                 // Per the spec, if the M value is smaller than  | 
171  |  |                                 // -1e38, it is a nodata value.  | 
172  | 0  |                                 if (psShape->padfM[i] > -1e38)  | 
173  | 0  |                                 { | 
174  | 0  |                                     bMIsUsed = true;  | 
175  | 0  |                                     break;  | 
176  | 0  |                                 }  | 
177  | 0  |                             }  | 
178  | 0  |                         }  | 
179  |  | 
  | 
180  | 0  |                         SHPDestroyObject(psShape);  | 
181  | 0  |                     }  | 
182  | 0  |                     if (bFirstShape || bMIsUsed)  | 
183  | 0  |                         break;  | 
184  | 0  |                 }  | 
185  | 0  |                 if (!bMIsUsed)  | 
186  | 0  |                     eType = OGR_GT_SetModifier(eType, wkbHasZ(eType), FALSE);  | 
187  | 0  |             }  | 
188  | 0  |         }  | 
189  | 0  |         else  | 
190  | 0  |         { | 
191  | 0  |             eType = m_eRequestedGeomType;  | 
192  | 0  |         }  | 
193  |  | 
  | 
194  | 0  |         OGRSpatialReference *poSRSClone = poSRSIn ? poSRSIn->Clone() : nullptr;  | 
195  | 0  |         if (poSRSClone)  | 
196  | 0  |         { | 
197  | 0  |             poSRSClone->SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);  | 
198  | 0  |         }  | 
199  | 0  |         auto poGeomFieldDefn = std::make_unique<OGRShapeGeomFieldDefn>(  | 
200  | 0  |             m_osFullName.c_str(), eType, bSRSSetIn, poSRSClone);  | 
201  | 0  |         if (!osPrjFilename.empty())  | 
202  | 0  |             poGeomFieldDefn->SetPrjFilename(osPrjFilename);  | 
203  | 0  |         if (poSRSClone)  | 
204  | 0  |             poSRSClone->Release();  | 
205  | 0  |         m_poFeatureDefn->SetGeomType(wkbNone);  | 
206  | 0  |         m_poFeatureDefn->AddGeomFieldDefn(std::move(poGeomFieldDefn));  | 
207  | 0  |     }  | 
208  |  | 
  | 
209  | 0  |     SetDescription(m_poFeatureDefn->GetName());  | 
210  | 0  |     m_bRewindOnWrite = CPLTestBool(CPLGetConfigOption(  | 
211  | 0  |         "SHAPE_REWIND_ON_WRITE",  | 
212  | 0  |         m_hSHP != nullptr && m_hSHP->nShapeType != SHPT_MULTIPATCH ? "NO"  | 
213  | 0  |                                                                    : "YES"));  | 
214  |  | 
  | 
215  | 0  |     m_poFeatureDefn->Seal(/* bSealFields = */ true);  | 
216  | 0  | }  | 
217  |  |  | 
218  |  | /************************************************************************/  | 
219  |  | /*                           ~OGRShapeLayer()                           */  | 
220  |  | /************************************************************************/  | 
221  |  |  | 
222  |  | OGRShapeLayer::~OGRShapeLayer()  | 
223  |  |  | 
224  | 0  | { | 
225  | 0  |     if (m_eNeedRepack == YES && m_bAutoRepack)  | 
226  | 0  |         Repack();  | 
227  |  | 
  | 
228  | 0  |     if (m_bResizeAtClose && m_hDBF != nullptr)  | 
229  | 0  |     { | 
230  | 0  |         ResizeDBF();  | 
231  | 0  |     }  | 
232  | 0  |     if (m_bCreateSpatialIndexAtClose && m_hSHP != nullptr)  | 
233  | 0  |     { | 
234  | 0  |         CreateSpatialIndex(0);  | 
235  | 0  |     }  | 
236  |  | 
  | 
237  | 0  |     if (m_nFeaturesRead > 0 && m_poFeatureDefn != nullptr)  | 
238  | 0  |     { | 
239  | 0  |         CPLDebug("Shape", "%d features read on layer '%s'.", | 
240  | 0  |                  static_cast<int>(m_nFeaturesRead), m_poFeatureDefn->GetName());  | 
241  | 0  |     }  | 
242  |  | 
  | 
243  | 0  |     ClearMatchingFIDs();  | 
244  | 0  |     ClearSpatialFIDs();  | 
245  |  | 
  | 
246  | 0  |     if (m_poFeatureDefn != nullptr)  | 
247  | 0  |         m_poFeatureDefn->Release();  | 
248  |  | 
  | 
249  | 0  |     if (m_hDBF != nullptr)  | 
250  | 0  |         DBFClose(m_hDBF);  | 
251  |  | 
  | 
252  | 0  |     if (m_hSHP != nullptr)  | 
253  | 0  |         SHPClose(m_hSHP);  | 
254  |  | 
  | 
255  | 0  |     if (m_hQIX != nullptr)  | 
256  | 0  |         SHPCloseDiskTree(m_hQIX);  | 
257  |  | 
  | 
258  | 0  |     if (m_hSBN != nullptr)  | 
259  | 0  |         SBNCloseDiskTree(m_hSBN);  | 
260  | 0  | }  | 
261  |  |  | 
262  |  | /************************************************************************/  | 
263  |  | /*                       SetModificationDate()                          */  | 
264  |  | /************************************************************************/  | 
265  |  |  | 
266  |  | void OGRShapeLayer::SetModificationDate(const char *pszStr)  | 
267  | 0  | { | 
268  | 0  |     if (m_hDBF && pszStr)  | 
269  | 0  |     { | 
270  | 0  |         int year = 0;  | 
271  | 0  |         int month = 0;  | 
272  | 0  |         int day = 0;  | 
273  | 0  |         if ((sscanf(pszStr, "%04d-%02d-%02d", &year, &month, &day) == 3 ||  | 
274  | 0  |              sscanf(pszStr, "%04d/%02d/%02d", &year, &month, &day) == 3) &&  | 
275  | 0  |             (year >= 1900 && year <= 1900 + 255 && month >= 1 && month <= 12 &&  | 
276  | 0  |              day >= 1 && day <= 31))  | 
277  | 0  |         { | 
278  | 0  |             DBFSetLastModifiedDate(m_hDBF, year - 1900, month, day);  | 
279  | 0  |         }  | 
280  | 0  |     }  | 
281  | 0  | }  | 
282  |  |  | 
283  |  | /************************************************************************/  | 
284  |  | /*                       SetWriteDBFEOFChar()                           */  | 
285  |  | /************************************************************************/  | 
286  |  |  | 
287  |  | void OGRShapeLayer::SetWriteDBFEOFChar(bool b)  | 
288  | 0  | { | 
289  | 0  |     if (m_hDBF)  | 
290  | 0  |     { | 
291  | 0  |         DBFSetWriteEndOfFileChar(m_hDBF, b);  | 
292  | 0  |     }  | 
293  | 0  | }  | 
294  |  |  | 
295  |  | /************************************************************************/  | 
296  |  | /*                          ConvertCodePage()                           */  | 
297  |  | /************************************************************************/  | 
298  |  |  | 
299  |  | static CPLString GetEncodingFromLDIDNumber(int nLDID)  | 
300  | 0  | { | 
301  | 0  |     int nCP = -1;  // Windows code page.  | 
302  |  |  | 
303  |  |     // http://www.autopark.ru/ASBProgrammerGuide/DBFSTRUC.HTM  | 
304  | 0  |     switch (nLDID)  | 
305  | 0  |     { | 
306  | 0  |         case 1:  | 
307  | 0  |             nCP = 437;  | 
308  | 0  |             break;  | 
309  | 0  |         case 2:  | 
310  | 0  |             nCP = 850;  | 
311  | 0  |             break;  | 
312  | 0  |         case 3:  | 
313  | 0  |             nCP = 1252;  | 
314  | 0  |             break;  | 
315  | 0  |         case 4:  | 
316  | 0  |             nCP = 10000;  | 
317  | 0  |             break;  | 
318  | 0  |         case 8:  | 
319  | 0  |             nCP = 865;  | 
320  | 0  |             break;  | 
321  | 0  |         case 10:  | 
322  | 0  |             nCP = 850;  | 
323  | 0  |             break;  | 
324  | 0  |         case 11:  | 
325  | 0  |             nCP = 437;  | 
326  | 0  |             break;  | 
327  | 0  |         case 13:  | 
328  | 0  |             nCP = 437;  | 
329  | 0  |             break;  | 
330  | 0  |         case 14:  | 
331  | 0  |             nCP = 850;  | 
332  | 0  |             break;  | 
333  | 0  |         case 15:  | 
334  | 0  |             nCP = 437;  | 
335  | 0  |             break;  | 
336  | 0  |         case 16:  | 
337  | 0  |             nCP = 850;  | 
338  | 0  |             break;  | 
339  | 0  |         case 17:  | 
340  | 0  |             nCP = 437;  | 
341  | 0  |             break;  | 
342  | 0  |         case 18:  | 
343  | 0  |             nCP = 850;  | 
344  | 0  |             break;  | 
345  | 0  |         case 19:  | 
346  | 0  |             nCP = 932;  | 
347  | 0  |             break;  | 
348  | 0  |         case 20:  | 
349  | 0  |             nCP = 850;  | 
350  | 0  |             break;  | 
351  | 0  |         case 21:  | 
352  | 0  |             nCP = 437;  | 
353  | 0  |             break;  | 
354  | 0  |         case 22:  | 
355  | 0  |             nCP = 850;  | 
356  | 0  |             break;  | 
357  | 0  |         case 23:  | 
358  | 0  |             nCP = 865;  | 
359  | 0  |             break;  | 
360  | 0  |         case 24:  | 
361  | 0  |             nCP = 437;  | 
362  | 0  |             break;  | 
363  | 0  |         case 25:  | 
364  | 0  |             nCP = 437;  | 
365  | 0  |             break;  | 
366  | 0  |         case 26:  | 
367  | 0  |             nCP = 850;  | 
368  | 0  |             break;  | 
369  | 0  |         case 27:  | 
370  | 0  |             nCP = 437;  | 
371  | 0  |             break;  | 
372  | 0  |         case 28:  | 
373  | 0  |             nCP = 863;  | 
374  | 0  |             break;  | 
375  | 0  |         case 29:  | 
376  | 0  |             nCP = 850;  | 
377  | 0  |             break;  | 
378  | 0  |         case 31:  | 
379  | 0  |             nCP = 852;  | 
380  | 0  |             break;  | 
381  | 0  |         case 34:  | 
382  | 0  |             nCP = 852;  | 
383  | 0  |             break;  | 
384  | 0  |         case 35:  | 
385  | 0  |             nCP = 852;  | 
386  | 0  |             break;  | 
387  | 0  |         case 36:  | 
388  | 0  |             nCP = 860;  | 
389  | 0  |             break;  | 
390  | 0  |         case 37:  | 
391  | 0  |             nCP = 850;  | 
392  | 0  |             break;  | 
393  | 0  |         case 38:  | 
394  | 0  |             nCP = 866;  | 
395  | 0  |             break;  | 
396  | 0  |         case 55:  | 
397  | 0  |             nCP = 850;  | 
398  | 0  |             break;  | 
399  | 0  |         case 64:  | 
400  | 0  |             nCP = 852;  | 
401  | 0  |             break;  | 
402  | 0  |         case 77:  | 
403  | 0  |             nCP = 936;  | 
404  | 0  |             break;  | 
405  | 0  |         case 78:  | 
406  | 0  |             nCP = 949;  | 
407  | 0  |             break;  | 
408  | 0  |         case 79:  | 
409  | 0  |             nCP = 950;  | 
410  | 0  |             break;  | 
411  | 0  |         case 80:  | 
412  | 0  |             nCP = 874;  | 
413  | 0  |             break;  | 
414  | 0  |         case 87:  | 
415  | 0  |             return CPL_ENC_ISO8859_1;  | 
416  | 0  |         case 88:  | 
417  | 0  |             nCP = 1252;  | 
418  | 0  |             break;  | 
419  | 0  |         case 89:  | 
420  | 0  |             nCP = 1252;  | 
421  | 0  |             break;  | 
422  | 0  |         case 100:  | 
423  | 0  |             nCP = 852;  | 
424  | 0  |             break;  | 
425  | 0  |         case 101:  | 
426  | 0  |             nCP = 866;  | 
427  | 0  |             break;  | 
428  | 0  |         case 102:  | 
429  | 0  |             nCP = 865;  | 
430  | 0  |             break;  | 
431  | 0  |         case 103:  | 
432  | 0  |             nCP = 861;  | 
433  | 0  |             break;  | 
434  | 0  |         case 104:  | 
435  | 0  |             nCP = 895;  | 
436  | 0  |             break;  | 
437  | 0  |         case 105:  | 
438  | 0  |             nCP = 620;  | 
439  | 0  |             break;  | 
440  | 0  |         case 106:  | 
441  | 0  |             nCP = 737;  | 
442  | 0  |             break;  | 
443  | 0  |         case 107:  | 
444  | 0  |             nCP = 857;  | 
445  | 0  |             break;  | 
446  | 0  |         case 108:  | 
447  | 0  |             nCP = 863;  | 
448  | 0  |             break;  | 
449  | 0  |         case 120:  | 
450  | 0  |             nCP = 950;  | 
451  | 0  |             break;  | 
452  | 0  |         case 121:  | 
453  | 0  |             nCP = 949;  | 
454  | 0  |             break;  | 
455  | 0  |         case 122:  | 
456  | 0  |             nCP = 936;  | 
457  | 0  |             break;  | 
458  | 0  |         case 123:  | 
459  | 0  |             nCP = 932;  | 
460  | 0  |             break;  | 
461  | 0  |         case 124:  | 
462  | 0  |             nCP = 874;  | 
463  | 0  |             break;  | 
464  | 0  |         case 134:  | 
465  | 0  |             nCP = 737;  | 
466  | 0  |             break;  | 
467  | 0  |         case 135:  | 
468  | 0  |             nCP = 852;  | 
469  | 0  |             break;  | 
470  | 0  |         case 136:  | 
471  | 0  |             nCP = 857;  | 
472  | 0  |             break;  | 
473  | 0  |         case 150:  | 
474  | 0  |             nCP = 10007;  | 
475  | 0  |             break;  | 
476  | 0  |         case 151:  | 
477  | 0  |             nCP = 10029;  | 
478  | 0  |             break;  | 
479  | 0  |         case 200:  | 
480  | 0  |             nCP = 1250;  | 
481  | 0  |             break;  | 
482  | 0  |         case 201:  | 
483  | 0  |             nCP = 1251;  | 
484  | 0  |             break;  | 
485  | 0  |         case 202:  | 
486  | 0  |             nCP = 1254;  | 
487  | 0  |             break;  | 
488  | 0  |         case 203:  | 
489  | 0  |             nCP = 1253;  | 
490  | 0  |             break;  | 
491  | 0  |         case 204:  | 
492  | 0  |             nCP = 1257;  | 
493  | 0  |             break;  | 
494  | 0  |         default:  | 
495  | 0  |             break;  | 
496  | 0  |     }  | 
497  |  |  | 
498  | 0  |     if (nCP < 0)  | 
499  | 0  |         return CPLString();  | 
500  | 0  |     return CPLString().Printf("CP%d", nCP); | 
501  | 0  | }  | 
502  |  |  | 
503  |  | static CPLString GetEncodingFromCPG(const char *pszCPG)  | 
504  | 0  | { | 
505  |  |     // see https://support.esri.com/en/technical-article/000013192  | 
506  | 0  |     CPLString m_osEncodingFromCPG;  | 
507  | 0  |     const int nCPG = atoi(pszCPG);  | 
508  | 0  |     if ((nCPG >= 437 && nCPG <= 950) || (nCPG >= 1250 && nCPG <= 1258))  | 
509  | 0  |     { | 
510  | 0  |         m_osEncodingFromCPG.Printf("CP%d", nCPG); | 
511  | 0  |     }  | 
512  | 0  |     else if (STARTS_WITH_CI(pszCPG, "8859"))  | 
513  | 0  |     { | 
514  | 0  |         if (pszCPG[4] == '-')  | 
515  | 0  |             m_osEncodingFromCPG.Printf("ISO-8859-%s", pszCPG + 5); | 
516  | 0  |         else  | 
517  | 0  |             m_osEncodingFromCPG.Printf("ISO-8859-%s", pszCPG + 4); | 
518  | 0  |     }  | 
519  | 0  |     else if (STARTS_WITH_CI(pszCPG, "UTF-8") || STARTS_WITH_CI(pszCPG, "UTF8"))  | 
520  | 0  |         m_osEncodingFromCPG = CPL_ENC_UTF8;  | 
521  | 0  |     else if (STARTS_WITH_CI(pszCPG, "ANSI 1251"))  | 
522  | 0  |         m_osEncodingFromCPG = "CP1251";  | 
523  | 0  |     else  | 
524  | 0  |     { | 
525  |  |         // Try just using the CPG value directly.  Works for stuff like Big5.  | 
526  | 0  |         m_osEncodingFromCPG = pszCPG;  | 
527  | 0  |     }  | 
528  | 0  |     return m_osEncodingFromCPG;  | 
529  | 0  | }  | 
530  |  |  | 
531  |  | CPLString OGRShapeLayer::ConvertCodePage(const char *pszCodePage)  | 
532  |  |  | 
533  | 0  | { | 
534  | 0  |     CPLString l_m_osEncoding;  | 
535  |  | 
  | 
536  | 0  |     if (pszCodePage == nullptr)  | 
537  | 0  |         return l_m_osEncoding;  | 
538  |  |  | 
539  | 0  |     std::string m_osEncodingFromLDID;  | 
540  | 0  |     if (m_hDBF->iLanguageDriver != 0)  | 
541  | 0  |     { | 
542  | 0  |         SetMetadataItem("LDID_VALUE", CPLSPrintf("%d", m_hDBF->iLanguageDriver), | 
543  | 0  |                         "SHAPEFILE");  | 
544  |  | 
  | 
545  | 0  |         m_osEncodingFromLDID =  | 
546  | 0  |             GetEncodingFromLDIDNumber(m_hDBF->iLanguageDriver);  | 
547  | 0  |     }  | 
548  | 0  |     if (!m_osEncodingFromLDID.empty())  | 
549  | 0  |     { | 
550  | 0  |         SetMetadataItem("ENCODING_FROM_LDID", m_osEncodingFromLDID.c_str(), | 
551  | 0  |                         "SHAPEFILE");  | 
552  | 0  |     }  | 
553  |  | 
  | 
554  | 0  |     std::string m_osEncodingFromCPG;  | 
555  | 0  |     if (!STARTS_WITH_CI(pszCodePage, "LDID/"))  | 
556  | 0  |     { | 
557  | 0  |         SetMetadataItem("CPG_VALUE", pszCodePage, "SHAPEFILE"); | 
558  |  | 
  | 
559  | 0  |         m_osEncodingFromCPG = GetEncodingFromCPG(pszCodePage);  | 
560  |  | 
  | 
561  | 0  |         if (!m_osEncodingFromCPG.empty())  | 
562  | 0  |             SetMetadataItem("ENCODING_FROM_CPG", m_osEncodingFromCPG.c_str(), | 
563  | 0  |                             "SHAPEFILE");  | 
564  |  | 
  | 
565  | 0  |         l_m_osEncoding = std::move(m_osEncodingFromCPG);  | 
566  | 0  |     }  | 
567  | 0  |     else if (!m_osEncodingFromLDID.empty())  | 
568  | 0  |     { | 
569  | 0  |         l_m_osEncoding = std::move(m_osEncodingFromLDID);  | 
570  | 0  |     }  | 
571  |  | 
  | 
572  | 0  |     return l_m_osEncoding;  | 
573  | 0  | }  | 
574  |  |  | 
575  |  | /************************************************************************/  | 
576  |  | /*                            CheckForQIX()                             */  | 
577  |  | /************************************************************************/  | 
578  |  |  | 
579  |  | bool OGRShapeLayer::CheckForQIX()  | 
580  |  |  | 
581  | 0  | { | 
582  | 0  |     if (m_bCheckedForQIX)  | 
583  | 0  |         return m_hQIX != nullptr;  | 
584  |  |  | 
585  | 0  |     const std::string osQIXFilename =  | 
586  | 0  |         CPLResetExtensionSafe(m_osFullName.c_str(), "qix");  | 
587  |  | 
  | 
588  | 0  |     m_hQIX = SHPOpenDiskTree(osQIXFilename.c_str(), nullptr);  | 
589  |  | 
  | 
590  | 0  |     m_bCheckedForQIX = true;  | 
591  |  | 
  | 
592  | 0  |     return m_hQIX != nullptr;  | 
593  | 0  | }  | 
594  |  |  | 
595  |  | /************************************************************************/  | 
596  |  | /*                            CheckForSBN()                             */  | 
597  |  | /************************************************************************/  | 
598  |  |  | 
599  |  | bool OGRShapeLayer::CheckForSBN()  | 
600  |  |  | 
601  | 0  | { | 
602  | 0  |     if (m_bCheckedForSBN)  | 
603  | 0  |         return m_hSBN != nullptr;  | 
604  |  |  | 
605  | 0  |     const std::string osSBNFilename =  | 
606  | 0  |         CPLResetExtensionSafe(m_osFullName.c_str(), "sbn");  | 
607  |  | 
  | 
608  | 0  |     m_hSBN = SBNOpenDiskTree(osSBNFilename.c_str(), nullptr);  | 
609  |  | 
  | 
610  | 0  |     m_bCheckedForSBN = true;  | 
611  |  | 
  | 
612  | 0  |     return m_hSBN != nullptr;  | 
613  | 0  | }  | 
614  |  |  | 
615  |  | /************************************************************************/  | 
616  |  | /*                            ScanIndices()                             */  | 
617  |  | /*                                                                      */  | 
618  |  | /*      Utilize optional spatial and attribute indices if they are      */  | 
619  |  | /*      available.                                                      */  | 
620  |  | /************************************************************************/  | 
621  |  |  | 
622  |  | bool OGRShapeLayer::ScanIndices()  | 
623  |  |  | 
624  | 0  | { | 
625  | 0  |     m_iMatchingFID = 0;  | 
626  |  |  | 
627  |  |     /* -------------------------------------------------------------------- */  | 
628  |  |     /*      Utilize attribute index if appropriate.                         */  | 
629  |  |     /* -------------------------------------------------------------------- */  | 
630  | 0  |     if (m_poAttrQuery != nullptr)  | 
631  | 0  |     { | 
632  | 0  |         CPLAssert(m_panMatchingFIDs == nullptr);  | 
633  |  |  | 
634  | 0  |         InitializeIndexSupport(m_osFullName.c_str());  | 
635  |  | 
  | 
636  | 0  |         m_panMatchingFIDs =  | 
637  | 0  |             m_poAttrQuery->EvaluateAgainstIndices(this, nullptr);  | 
638  | 0  |     }  | 
639  |  |  | 
640  |  |     /* -------------------------------------------------------------------- */  | 
641  |  |     /*      Check for spatial index if we have a spatial query.             */  | 
642  |  |     /* -------------------------------------------------------------------- */  | 
643  |  |  | 
644  | 0  |     if (m_poFilterGeom == nullptr || m_hSHP == nullptr)  | 
645  | 0  |         return true;  | 
646  |  |  | 
647  | 0  |     OGREnvelope oSpatialFilterEnvelope;  | 
648  | 0  |     bool bTryQIXorSBN = true;  | 
649  |  | 
  | 
650  | 0  |     m_poFilterGeom->getEnvelope(&oSpatialFilterEnvelope);  | 
651  |  | 
  | 
652  | 0  |     OGREnvelope oLayerExtent;  | 
653  | 0  |     if (GetExtent(&oLayerExtent, TRUE) == OGRERR_NONE)  | 
654  | 0  |     { | 
655  | 0  |         if (oSpatialFilterEnvelope.Contains(oLayerExtent))  | 
656  | 0  |         { | 
657  |  |             // The spatial filter is larger than the layer extent. No use of  | 
658  |  |             // .qix file for now.  | 
659  | 0  |             return true;  | 
660  | 0  |         }  | 
661  | 0  |         else if (!oSpatialFilterEnvelope.Intersects(oLayerExtent))  | 
662  | 0  |         { | 
663  |  |             // No intersection : no need to check for .qix or .sbn.  | 
664  | 0  |             bTryQIXorSBN = false;  | 
665  |  |  | 
666  |  |             // Set an empty result for spatial FIDs.  | 
667  | 0  |             free(m_panSpatialFIDs);  | 
668  | 0  |             m_panSpatialFIDs = static_cast<int *>(calloc(1, sizeof(int)));  | 
669  | 0  |             m_nSpatialFIDCount = 0;  | 
670  |  | 
  | 
671  | 0  |             delete m_poFilterGeomLastValid;  | 
672  | 0  |             m_poFilterGeomLastValid = m_poFilterGeom->clone();  | 
673  | 0  |         }  | 
674  | 0  |     }  | 
675  |  |  | 
676  | 0  |     if (bTryQIXorSBN)  | 
677  | 0  |     { | 
678  | 0  |         if (!m_bCheckedForQIX)  | 
679  | 0  |             CPL_IGNORE_RET_VAL(CheckForQIX());  | 
680  | 0  |         if (m_hQIX == nullptr && !m_bCheckedForSBN)  | 
681  | 0  |             CPL_IGNORE_RET_VAL(CheckForSBN());  | 
682  | 0  |     }  | 
683  |  |  | 
684  |  |     /* -------------------------------------------------------------------- */  | 
685  |  |     /*      Compute spatial index if appropriate.                           */  | 
686  |  |     /* -------------------------------------------------------------------- */  | 
687  | 0  |     if (bTryQIXorSBN && (m_hQIX != nullptr || m_hSBN != nullptr) &&  | 
688  | 0  |         m_panSpatialFIDs == nullptr)  | 
689  | 0  |     { | 
690  | 0  |         double adfBoundsMin[4] = {oSpatialFilterEnvelope.MinX, | 
691  | 0  |                                   oSpatialFilterEnvelope.MinY, 0.0, 0.0};  | 
692  | 0  |         double adfBoundsMax[4] = {oSpatialFilterEnvelope.MaxX, | 
693  | 0  |                                   oSpatialFilterEnvelope.MaxY, 0.0, 0.0};  | 
694  |  | 
  | 
695  | 0  |         if (m_hQIX != nullptr)  | 
696  | 0  |             m_panSpatialFIDs = SHPSearchDiskTreeEx(  | 
697  | 0  |                 m_hQIX, adfBoundsMin, adfBoundsMax, &m_nSpatialFIDCount);  | 
698  | 0  |         else  | 
699  | 0  |             m_panSpatialFIDs = SBNSearchDiskTree(  | 
700  | 0  |                 m_hSBN, adfBoundsMin, adfBoundsMax, &m_nSpatialFIDCount);  | 
701  |  | 
  | 
702  | 0  |         CPLDebug("SHAPE", "Used spatial index, got %d matches.", | 
703  | 0  |                  m_nSpatialFIDCount);  | 
704  |  | 
  | 
705  | 0  |         delete m_poFilterGeomLastValid;  | 
706  | 0  |         m_poFilterGeomLastValid = m_poFilterGeom->clone();  | 
707  | 0  |     }  | 
708  |  |  | 
709  |  |     /* -------------------------------------------------------------------- */  | 
710  |  |     /*      Use spatial index if appropriate.                               */  | 
711  |  |     /* -------------------------------------------------------------------- */  | 
712  | 0  |     if (m_panSpatialFIDs != nullptr)  | 
713  | 0  |     { | 
714  |  |         // Use resulting list as matching FID list (but reallocate and  | 
715  |  |         // terminate with OGRNullFID).  | 
716  | 0  |         if (m_panMatchingFIDs == nullptr)  | 
717  | 0  |         { | 
718  | 0  |             m_panMatchingFIDs = static_cast<GIntBig *>(  | 
719  | 0  |                 CPLMalloc(sizeof(GIntBig) * (m_nSpatialFIDCount + 1)));  | 
720  | 0  |             for (int i = 0; i < m_nSpatialFIDCount; i++)  | 
721  | 0  |                 m_panMatchingFIDs[i] =  | 
722  | 0  |                     static_cast<GIntBig>(m_panSpatialFIDs[i]);  | 
723  | 0  |             m_panMatchingFIDs[m_nSpatialFIDCount] = OGRNullFID;  | 
724  | 0  |         }  | 
725  |  |         // Cull attribute index matches based on those in the spatial index  | 
726  |  |         // result set.  We assume that the attribute results are in sorted  | 
727  |  |         // order.  | 
728  | 0  |         else  | 
729  | 0  |         { | 
730  | 0  |             int iWrite = 0;  | 
731  | 0  |             int iSpatial = 0;  | 
732  |  | 
  | 
733  | 0  |             for (int iRead = 0; m_panMatchingFIDs[iRead] != OGRNullFID; iRead++)  | 
734  | 0  |             { | 
735  | 0  |                 while (iSpatial < m_nSpatialFIDCount &&  | 
736  | 0  |                        m_panSpatialFIDs[iSpatial] < m_panMatchingFIDs[iRead])  | 
737  | 0  |                     iSpatial++;  | 
738  |  | 
  | 
739  | 0  |                 if (iSpatial == m_nSpatialFIDCount)  | 
740  | 0  |                     continue;  | 
741  |  |  | 
742  | 0  |                 if (m_panSpatialFIDs[iSpatial] == m_panMatchingFIDs[iRead])  | 
743  | 0  |                     m_panMatchingFIDs[iWrite++] = m_panMatchingFIDs[iRead];  | 
744  | 0  |             }  | 
745  | 0  |             m_panMatchingFIDs[iWrite] = OGRNullFID;  | 
746  | 0  |         }  | 
747  |  | 
  | 
748  | 0  |         if (m_nSpatialFIDCount > 100000)  | 
749  | 0  |         { | 
750  | 0  |             ClearSpatialFIDs();  | 
751  | 0  |         }  | 
752  | 0  |     }  | 
753  |  | 
  | 
754  | 0  |     return true;  | 
755  | 0  | }  | 
756  |  |  | 
757  |  | /************************************************************************/  | 
758  |  | /*                            ResetReading()                            */  | 
759  |  | /************************************************************************/  | 
760  |  |  | 
761  |  | void OGRShapeLayer::ResetReading()  | 
762  |  |  | 
763  | 0  | { | 
764  | 0  |     if (!TouchLayer())  | 
765  | 0  |         return;  | 
766  |  |  | 
767  | 0  |     m_iMatchingFID = 0;  | 
768  |  | 
  | 
769  | 0  |     m_iNextShapeId = 0;  | 
770  |  | 
  | 
771  | 0  |     if (m_bHeaderDirty && m_bUpdateAccess)  | 
772  | 0  |         SyncToDisk();  | 
773  |  | 
  | 
774  | 0  |     if (m_hDBF)  | 
775  | 0  |         VSIFClearErrL(VSI_SHP_GetVSIL(m_hDBF->fp));  | 
776  | 0  | }  | 
777  |  |  | 
778  |  | /************************************************************************/  | 
779  |  | /*                        ClearMatchingFIDs()                           */  | 
780  |  | /************************************************************************/  | 
781  |  |  | 
782  |  | void OGRShapeLayer::ClearMatchingFIDs()  | 
783  | 0  | { | 
784  |  |     /* -------------------------------------------------------------------- */  | 
785  |  |     /*      Clear previous index search result, if any.                     */  | 
786  |  |     /* -------------------------------------------------------------------- */  | 
787  | 0  |     CPLFree(m_panMatchingFIDs);  | 
788  | 0  |     m_panMatchingFIDs = nullptr;  | 
789  | 0  | }  | 
790  |  |  | 
791  |  | /************************************************************************/  | 
792  |  | /*                        ClearSpatialFIDs()                           */  | 
793  |  | /************************************************************************/  | 
794  |  |  | 
795  |  | void OGRShapeLayer::ClearSpatialFIDs()  | 
796  | 0  | { | 
797  | 0  |     if (m_panSpatialFIDs != nullptr)  | 
798  | 0  |     { | 
799  | 0  |         CPLDebug("SHAPE", "Clear m_panSpatialFIDs"); | 
800  | 0  |         free(m_panSpatialFIDs);  | 
801  | 0  |     }  | 
802  | 0  |     m_panSpatialFIDs = nullptr;  | 
803  | 0  |     m_nSpatialFIDCount = 0;  | 
804  |  | 
  | 
805  | 0  |     delete m_poFilterGeomLastValid;  | 
806  | 0  |     m_poFilterGeomLastValid = nullptr;  | 
807  | 0  | }  | 
808  |  |  | 
809  |  | /************************************************************************/  | 
810  |  | /*                         ISetSpatialFilter()                          */  | 
811  |  | /************************************************************************/  | 
812  |  |  | 
813  |  | OGRErr OGRShapeLayer::ISetSpatialFilter(int iGeomField,  | 
814  |  |                                         const OGRGeometry *poGeomIn)  | 
815  | 0  | { | 
816  | 0  |     ClearMatchingFIDs();  | 
817  |  | 
  | 
818  | 0  |     if (poGeomIn == nullptr)  | 
819  | 0  |     { | 
820  |  |         // Do nothing.  | 
821  | 0  |     }  | 
822  | 0  |     else if (m_poFilterGeomLastValid != nullptr &&  | 
823  | 0  |              m_poFilterGeomLastValid->Equals(poGeomIn))  | 
824  | 0  |     { | 
825  |  |         // Do nothing.  | 
826  | 0  |     }  | 
827  | 0  |     else if (m_panSpatialFIDs != nullptr)  | 
828  | 0  |     { | 
829  |  |         // We clear the spatialFIDs only if we have a new non-NULL spatial  | 
830  |  |         // filter, otherwise we keep the previous result cached. This can be  | 
831  |  |         // useful when several SQL layers rely on the same table layer, and use  | 
832  |  |         // the same spatial filters. But as there is in the destructor of  | 
833  |  |         // OGRGenSQLResultsLayer a clearing of the spatial filter of the table  | 
834  |  |         // layer, we need this trick.  | 
835  | 0  |         ClearSpatialFIDs();  | 
836  | 0  |     }  | 
837  |  | 
  | 
838  | 0  |     return OGRLayer::ISetSpatialFilter(iGeomField, poGeomIn);  | 
839  | 0  | }  | 
840  |  |  | 
841  |  | /************************************************************************/  | 
842  |  | /*                         SetAttributeFilter()                         */  | 
843  |  | /************************************************************************/  | 
844  |  |  | 
845  |  | OGRErr OGRShapeLayer::SetAttributeFilter(const char *pszAttributeFilter)  | 
846  | 0  | { | 
847  | 0  |     ClearMatchingFIDs();  | 
848  |  | 
  | 
849  | 0  |     return OGRLayer::SetAttributeFilter(pszAttributeFilter);  | 
850  | 0  | }  | 
851  |  |  | 
852  |  | /************************************************************************/  | 
853  |  | /*                           SetNextByIndex()                           */  | 
854  |  | /*                                                                      */  | 
855  |  | /*      If we already have an FID list, we can easily reposition        */  | 
856  |  | /*      ourselves in it.                                                */  | 
857  |  | /************************************************************************/  | 
858  |  |  | 
859  |  | OGRErr OGRShapeLayer::SetNextByIndex(GIntBig nIndex)  | 
860  |  |  | 
861  | 0  | { | 
862  | 0  |     if (!TouchLayer())  | 
863  | 0  |         return OGRERR_FAILURE;  | 
864  |  |  | 
865  | 0  |     if (nIndex < 0 || nIndex > INT_MAX)  | 
866  | 0  |         return OGRERR_FAILURE;  | 
867  |  |  | 
868  |  |     // Eventually we should try to use m_panMatchingFIDs list  | 
869  |  |     // if available and appropriate.  | 
870  | 0  |     if (m_poFilterGeom != nullptr || m_poAttrQuery != nullptr)  | 
871  | 0  |         return OGRLayer::SetNextByIndex(nIndex);  | 
872  |  |  | 
873  | 0  |     m_iNextShapeId = static_cast<int>(nIndex);  | 
874  |  | 
  | 
875  | 0  |     return OGRERR_NONE;  | 
876  | 0  | }  | 
877  |  |  | 
878  |  | /************************************************************************/  | 
879  |  | /*                             FetchShape()                             */  | 
880  |  | /*                                                                      */  | 
881  |  | /*      Take a shape id, a geometry, and a feature, and set the feature */  | 
882  |  | /*      if the shapeid bbox intersects the geometry.                    */  | 
883  |  | /************************************************************************/  | 
884  |  |  | 
885  |  | OGRFeature *OGRShapeLayer::FetchShape(int iShapeId)  | 
886  |  |  | 
887  | 0  | { | 
888  | 0  |     OGRFeature *poFeature = nullptr;  | 
889  |  | 
  | 
890  | 0  |     if (m_poFilterGeom != nullptr && m_hSHP != nullptr)  | 
891  | 0  |     { | 
892  | 0  |         SHPObject *psShape = SHPReadObject(m_hSHP, iShapeId);  | 
893  |  |  | 
894  |  |         // do not trust degenerate bounds on non-point geometries  | 
895  |  |         // or bounds on null shapes.  | 
896  | 0  |         if (psShape == nullptr ||  | 
897  | 0  |             (psShape->nSHPType != SHPT_POINT &&  | 
898  | 0  |              psShape->nSHPType != SHPT_POINTZ &&  | 
899  | 0  |              psShape->nSHPType != SHPT_POINTM &&  | 
900  | 0  |              (psShape->dfXMin == psShape->dfXMax ||  | 
901  | 0  |               psShape->dfYMin == psShape->dfYMax)) ||  | 
902  | 0  |             psShape->nSHPType == SHPT_NULL)  | 
903  | 0  |         { | 
904  | 0  |             poFeature = SHPReadOGRFeature(m_hSHP, m_hDBF, m_poFeatureDefn,  | 
905  | 0  |                                           iShapeId, psShape, m_osEncoding,  | 
906  | 0  |                                           m_bHasWarnedWrongWindingOrder);  | 
907  | 0  |         }  | 
908  | 0  |         else if (m_sFilterEnvelope.MaxX < psShape->dfXMin ||  | 
909  | 0  |                  m_sFilterEnvelope.MaxY < psShape->dfYMin ||  | 
910  | 0  |                  psShape->dfXMax < m_sFilterEnvelope.MinX ||  | 
911  | 0  |                  psShape->dfYMax < m_sFilterEnvelope.MinY)  | 
912  | 0  |         { | 
913  | 0  |             SHPDestroyObject(psShape);  | 
914  | 0  |             poFeature = nullptr;  | 
915  | 0  |         }  | 
916  | 0  |         else  | 
917  | 0  |         { | 
918  | 0  |             poFeature = SHPReadOGRFeature(m_hSHP, m_hDBF, m_poFeatureDefn,  | 
919  | 0  |                                           iShapeId, psShape, m_osEncoding,  | 
920  | 0  |                                           m_bHasWarnedWrongWindingOrder);  | 
921  | 0  |         }  | 
922  | 0  |     }  | 
923  | 0  |     else  | 
924  | 0  |     { | 
925  | 0  |         poFeature = SHPReadOGRFeature(m_hSHP, m_hDBF, m_poFeatureDefn, iShapeId,  | 
926  | 0  |                                       nullptr, m_osEncoding,  | 
927  | 0  |                                       m_bHasWarnedWrongWindingOrder);  | 
928  | 0  |     }  | 
929  |  | 
  | 
930  | 0  |     return poFeature;  | 
931  | 0  | }  | 
932  |  |  | 
933  |  | /************************************************************************/  | 
934  |  | /*                           GetNextFeature()                           */  | 
935  |  | /************************************************************************/  | 
936  |  |  | 
937  |  | OGRFeature *OGRShapeLayer::GetNextFeature()  | 
938  |  |  | 
939  | 0  | { | 
940  | 0  |     if (!TouchLayer())  | 
941  | 0  |         return nullptr;  | 
942  |  |  | 
943  |  |     /* -------------------------------------------------------------------- */  | 
944  |  |     /*      Collect a matching list if we have attribute or spatial         */  | 
945  |  |     /*      indices.  Only do this on the first request for a given pass    */  | 
946  |  |     /*      of course.                                                      */  | 
947  |  |     /* -------------------------------------------------------------------- */  | 
948  | 0  |     if ((m_poAttrQuery != nullptr || m_poFilterGeom != nullptr) &&  | 
949  | 0  |         m_iNextShapeId == 0 && m_panMatchingFIDs == nullptr)  | 
950  | 0  |     { | 
951  | 0  |         ScanIndices();  | 
952  | 0  |     }  | 
953  |  |  | 
954  |  |     /* -------------------------------------------------------------------- */  | 
955  |  |     /*      Loop till we find a feature matching our criteria.              */  | 
956  |  |     /* -------------------------------------------------------------------- */  | 
957  | 0  |     OGRFeature *poFeature = nullptr;  | 
958  |  | 
  | 
959  | 0  |     while (true)  | 
960  | 0  |     { | 
961  | 0  |         if (m_panMatchingFIDs != nullptr)  | 
962  | 0  |         { | 
963  | 0  |             if (m_panMatchingFIDs[m_iMatchingFID] == OGRNullFID)  | 
964  | 0  |             { | 
965  | 0  |                 return nullptr;  | 
966  | 0  |             }  | 
967  |  |  | 
968  |  |             // Check the shape object's geometry, and if it matches  | 
969  |  |             // any spatial filter, return it.  | 
970  | 0  |             poFeature =  | 
971  | 0  |                 FetchShape(static_cast<int>(m_panMatchingFIDs[m_iMatchingFID]));  | 
972  |  | 
  | 
973  | 0  |             m_iMatchingFID++;  | 
974  | 0  |         }  | 
975  | 0  |         else  | 
976  | 0  |         { | 
977  | 0  |             if (m_iNextShapeId >= m_nTotalShapeCount)  | 
978  | 0  |             { | 
979  | 0  |                 return nullptr;  | 
980  | 0  |             }  | 
981  |  |  | 
982  | 0  |             if (m_hDBF)  | 
983  | 0  |             { | 
984  | 0  |                 if (DBFIsRecordDeleted(m_hDBF, m_iNextShapeId))  | 
985  | 0  |                     poFeature = nullptr;  | 
986  | 0  |                 else if (VSIFEofL(VSI_SHP_GetVSIL(m_hDBF->fp)) ||  | 
987  | 0  |                          VSIFErrorL(VSI_SHP_GetVSIL(m_hDBF->fp)))  | 
988  | 0  |                     return nullptr;  //* I/O error.  | 
989  | 0  |                 else  | 
990  | 0  |                     poFeature = FetchShape(m_iNextShapeId);  | 
991  | 0  |             }  | 
992  | 0  |             else  | 
993  | 0  |                 poFeature = FetchShape(m_iNextShapeId);  | 
994  |  |  | 
995  | 0  |             m_iNextShapeId++;  | 
996  | 0  |         }  | 
997  |  |  | 
998  | 0  |         if (poFeature != nullptr)  | 
999  | 0  |         { | 
1000  | 0  |             OGRGeometry *poGeom = poFeature->GetGeometryRef();  | 
1001  | 0  |             if (poGeom != nullptr)  | 
1002  | 0  |             { | 
1003  | 0  |                 poGeom->assignSpatialReference(GetSpatialRef());  | 
1004  | 0  |             }  | 
1005  |  | 
  | 
1006  | 0  |             m_nFeaturesRead++;  | 
1007  |  | 
  | 
1008  | 0  |             if ((m_poFilterGeom == nullptr || FilterGeometry(poGeom)) &&  | 
1009  | 0  |                 (m_poAttrQuery == nullptr ||  | 
1010  | 0  |                  m_poAttrQuery->Evaluate(poFeature)))  | 
1011  | 0  |             { | 
1012  | 0  |                 return poFeature;  | 
1013  | 0  |             }  | 
1014  |  |  | 
1015  | 0  |             delete poFeature;  | 
1016  | 0  |         }  | 
1017  | 0  |     }  | 
1018  | 0  | }  | 
1019  |  |  | 
1020  |  | /************************************************************************/  | 
1021  |  | /*                             GetFeature()                             */  | 
1022  |  | /************************************************************************/  | 
1023  |  |  | 
1024  |  | OGRFeature *OGRShapeLayer::GetFeature(GIntBig nFeatureId)  | 
1025  |  |  | 
1026  | 0  | { | 
1027  | 0  |     if (!TouchLayer() || nFeatureId > INT_MAX)  | 
1028  | 0  |         return nullptr;  | 
1029  |  |  | 
1030  | 0  |     OGRFeature *poFeature = SHPReadOGRFeature(  | 
1031  | 0  |         m_hSHP, m_hDBF, m_poFeatureDefn, static_cast<int>(nFeatureId), nullptr,  | 
1032  | 0  |         m_osEncoding, m_bHasWarnedWrongWindingOrder);  | 
1033  |  | 
  | 
1034  | 0  |     if (poFeature == nullptr)  | 
1035  | 0  |     { | 
1036  |  |         // Reading shape feature failed.  | 
1037  | 0  |         return nullptr;  | 
1038  | 0  |     }  | 
1039  |  |  | 
1040  | 0  |     if (poFeature->GetGeometryRef() != nullptr)  | 
1041  | 0  |     { | 
1042  | 0  |         poFeature->GetGeometryRef()->assignSpatialReference(GetSpatialRef());  | 
1043  | 0  |     }  | 
1044  |  | 
  | 
1045  | 0  |     m_nFeaturesRead++;  | 
1046  |  | 
  | 
1047  | 0  |     return poFeature;  | 
1048  | 0  | }  | 
1049  |  |  | 
1050  |  | /************************************************************************/  | 
1051  |  | /*                             StartUpdate()                            */  | 
1052  |  | /************************************************************************/  | 
1053  |  |  | 
1054  |  | bool OGRShapeLayer::StartUpdate(const char *pszOperation)  | 
1055  | 0  | { | 
1056  | 0  |     if (!m_poDS->UncompressIfNeeded())  | 
1057  | 0  |         return false;  | 
1058  |  |  | 
1059  | 0  |     if (!TouchLayer())  | 
1060  | 0  |         return false;  | 
1061  |  |  | 
1062  | 0  |     if (!m_bUpdateAccess)  | 
1063  | 0  |     { | 
1064  | 0  |         CPLError(CE_Failure, CPLE_NotSupported,  | 
1065  | 0  |                  "%s : unsupported operation on a read-only datasource.",  | 
1066  | 0  |                  pszOperation);  | 
1067  | 0  |         return false;  | 
1068  | 0  |     }  | 
1069  |  |  | 
1070  | 0  |     return true;  | 
1071  | 0  | }  | 
1072  |  |  | 
1073  |  | /************************************************************************/  | 
1074  |  | /*                             ISetFeature()                             */  | 
1075  |  | /************************************************************************/  | 
1076  |  |  | 
1077  |  | OGRErr OGRShapeLayer::ISetFeature(OGRFeature *poFeature)  | 
1078  |  |  | 
1079  | 0  | { | 
1080  | 0  |     if (!StartUpdate("SetFeature")) | 
1081  | 0  |         return OGRERR_FAILURE;  | 
1082  |  |  | 
1083  | 0  |     GIntBig nFID = poFeature->GetFID();  | 
1084  | 0  |     if (nFID < 0 || (m_hSHP != nullptr && nFID >= m_hSHP->nRecords) ||  | 
1085  | 0  |         (m_hDBF != nullptr && nFID >= m_hDBF->nRecords))  | 
1086  | 0  |     { | 
1087  | 0  |         return OGRERR_NON_EXISTING_FEATURE;  | 
1088  | 0  |     }  | 
1089  |  |  | 
1090  | 0  |     m_bHeaderDirty = true;  | 
1091  | 0  |     if (CheckForQIX() || CheckForSBN())  | 
1092  | 0  |         DropSpatialIndex();  | 
1093  |  | 
  | 
1094  | 0  |     unsigned int nOffset = 0;  | 
1095  | 0  |     unsigned int nSize = 0;  | 
1096  | 0  |     bool bIsLastRecord = false;  | 
1097  | 0  |     if (m_hSHP != nullptr)  | 
1098  | 0  |     { | 
1099  | 0  |         nOffset = m_hSHP->panRecOffset[nFID];  | 
1100  | 0  |         nSize = m_hSHP->panRecSize[nFID];  | 
1101  | 0  |         bIsLastRecord = (nOffset + nSize + 8 == m_hSHP->nFileSize);  | 
1102  | 0  |     }  | 
1103  |  | 
  | 
1104  | 0  |     OGRErr eErr = SHPWriteOGRFeature(m_hSHP, m_hDBF, m_poFeatureDefn, poFeature,  | 
1105  | 0  |                                      m_osEncoding, &m_bTruncationWarningEmitted,  | 
1106  | 0  |                                      m_bRewindOnWrite);  | 
1107  |  | 
  | 
1108  | 0  |     if (m_hSHP != nullptr)  | 
1109  | 0  |     { | 
1110  | 0  |         if (bIsLastRecord)  | 
1111  | 0  |         { | 
1112  |  |             // Optimization: we don't need repacking if this is the last  | 
1113  |  |             // record of the file. Just potential truncation  | 
1114  | 0  |             CPLAssert(nOffset == m_hSHP->panRecOffset[nFID]);  | 
1115  | 0  |             CPLAssert(m_hSHP->panRecOffset[nFID] + m_hSHP->panRecSize[nFID] +  | 
1116  | 0  |                           8 ==  | 
1117  | 0  |                       m_hSHP->nFileSize);  | 
1118  | 0  |             if (m_hSHP->panRecSize[nFID] < nSize)  | 
1119  | 0  |             { | 
1120  | 0  |                 VSIFTruncateL(VSI_SHP_GetVSIL(m_hSHP->fpSHP),  | 
1121  | 0  |                               m_hSHP->nFileSize);  | 
1122  | 0  |             }  | 
1123  | 0  |         }  | 
1124  | 0  |         else if (nOffset != m_hSHP->panRecOffset[nFID] ||  | 
1125  | 0  |                  nSize != m_hSHP->panRecSize[nFID])  | 
1126  | 0  |         { | 
1127  | 0  |             m_bSHPNeedsRepack = true;  | 
1128  | 0  |             m_eNeedRepack = YES;  | 
1129  | 0  |         }  | 
1130  | 0  |     }  | 
1131  |  |  | 
1132  | 0  |     return eErr;  | 
1133  | 0  | }  | 
1134  |  |  | 
1135  |  | /************************************************************************/  | 
1136  |  | /*                           DeleteFeature()                            */  | 
1137  |  | /************************************************************************/  | 
1138  |  |  | 
1139  |  | OGRErr OGRShapeLayer::DeleteFeature(GIntBig nFID)  | 
1140  |  |  | 
1141  | 0  | { | 
1142  | 0  |     if (!StartUpdate("DeleteFeature")) | 
1143  | 0  |         return OGRERR_FAILURE;  | 
1144  |  |  | 
1145  | 0  |     if (nFID < 0 || (m_hSHP != nullptr && nFID >= m_hSHP->nRecords) ||  | 
1146  | 0  |         (m_hDBF != nullptr && nFID >= m_hDBF->nRecords))  | 
1147  | 0  |     { | 
1148  | 0  |         return OGRERR_NON_EXISTING_FEATURE;  | 
1149  | 0  |     }  | 
1150  |  |  | 
1151  | 0  |     if (!m_hDBF)  | 
1152  | 0  |     { | 
1153  | 0  |         CPLError(CE_Failure, CPLE_AppDefined,  | 
1154  | 0  |                  "Attempt to delete shape in shapefile with no .dbf file.  "  | 
1155  | 0  |                  "Deletion is done by marking record deleted in dbf "  | 
1156  | 0  |                  "and is not supported without a .dbf file.");  | 
1157  | 0  |         return OGRERR_FAILURE;  | 
1158  | 0  |     }  | 
1159  |  |  | 
1160  | 0  |     if (DBFIsRecordDeleted(m_hDBF, static_cast<int>(nFID)))  | 
1161  | 0  |     { | 
1162  | 0  |         return OGRERR_NON_EXISTING_FEATURE;  | 
1163  | 0  |     }  | 
1164  |  |  | 
1165  | 0  |     if (!DBFMarkRecordDeleted(m_hDBF, static_cast<int>(nFID), TRUE))  | 
1166  | 0  |         return OGRERR_FAILURE;  | 
1167  |  |  | 
1168  | 0  |     m_bHeaderDirty = true;  | 
1169  | 0  |     if (CheckForQIX() || CheckForSBN())  | 
1170  | 0  |         DropSpatialIndex();  | 
1171  | 0  |     m_eNeedRepack = YES;  | 
1172  |  | 
  | 
1173  | 0  |     return OGRERR_NONE;  | 
1174  | 0  | }  | 
1175  |  |  | 
1176  |  | /************************************************************************/  | 
1177  |  | /*                           ICreateFeature()                            */  | 
1178  |  | /************************************************************************/  | 
1179  |  |  | 
1180  |  | OGRErr OGRShapeLayer::ICreateFeature(OGRFeature *poFeature)  | 
1181  |  |  | 
1182  | 0  | { | 
1183  | 0  |     if (!StartUpdate("CreateFeature")) | 
1184  | 0  |         return OGRERR_FAILURE;  | 
1185  |  |  | 
1186  | 0  |     if (m_hDBF != nullptr &&  | 
1187  | 0  |         !VSI_SHP_WriteMoreDataOK(m_hDBF->fp, m_hDBF->nRecordLength))  | 
1188  | 0  |     { | 
1189  | 0  |         return OGRERR_FAILURE;  | 
1190  | 0  |     }  | 
1191  |  |  | 
1192  | 0  |     m_bHeaderDirty = true;  | 
1193  | 0  |     if (CheckForQIX() || CheckForSBN())  | 
1194  | 0  |         DropSpatialIndex();  | 
1195  |  | 
  | 
1196  | 0  |     poFeature->SetFID(OGRNullFID);  | 
1197  |  | 
  | 
1198  | 0  |     if (m_nTotalShapeCount == 0 &&  | 
1199  | 0  |         wkbFlatten(m_eRequestedGeomType) == wkbUnknown && m_hSHP != nullptr &&  | 
1200  | 0  |         m_hSHP->nShapeType != SHPT_MULTIPATCH &&  | 
1201  | 0  |         poFeature->GetGeometryRef() != nullptr)  | 
1202  | 0  |     { | 
1203  | 0  |         OGRGeometry *poGeom = poFeature->GetGeometryRef();  | 
1204  | 0  |         int nShapeType = -1;  | 
1205  |  | 
  | 
1206  | 0  |         switch (poGeom->getGeometryType())  | 
1207  | 0  |         { | 
1208  | 0  |             case wkbPoint:  | 
1209  | 0  |                 nShapeType = SHPT_POINT;  | 
1210  | 0  |                 m_eRequestedGeomType = wkbPoint;  | 
1211  | 0  |                 break;  | 
1212  |  |  | 
1213  | 0  |             case wkbPoint25D:  | 
1214  | 0  |                 nShapeType = SHPT_POINTZ;  | 
1215  | 0  |                 m_eRequestedGeomType = wkbPoint25D;  | 
1216  | 0  |                 break;  | 
1217  |  |  | 
1218  | 0  |             case wkbPointM:  | 
1219  | 0  |                 nShapeType = SHPT_POINTM;  | 
1220  | 0  |                 m_eRequestedGeomType = wkbPointM;  | 
1221  | 0  |                 break;  | 
1222  |  |  | 
1223  | 0  |             case wkbPointZM:  | 
1224  | 0  |                 nShapeType = SHPT_POINTZ;  | 
1225  | 0  |                 m_eRequestedGeomType = wkbPointZM;  | 
1226  | 0  |                 break;  | 
1227  |  |  | 
1228  | 0  |             case wkbMultiPoint:  | 
1229  | 0  |                 nShapeType = SHPT_MULTIPOINT;  | 
1230  | 0  |                 m_eRequestedGeomType = wkbMultiPoint;  | 
1231  | 0  |                 break;  | 
1232  |  |  | 
1233  | 0  |             case wkbMultiPoint25D:  | 
1234  | 0  |                 nShapeType = SHPT_MULTIPOINTZ;  | 
1235  | 0  |                 m_eRequestedGeomType = wkbMultiPoint25D;  | 
1236  | 0  |                 break;  | 
1237  |  |  | 
1238  | 0  |             case wkbMultiPointM:  | 
1239  | 0  |                 nShapeType = SHPT_MULTIPOINTM;  | 
1240  | 0  |                 m_eRequestedGeomType = wkbMultiPointM;  | 
1241  | 0  |                 break;  | 
1242  |  |  | 
1243  | 0  |             case wkbMultiPointZM:  | 
1244  | 0  |                 nShapeType = SHPT_MULTIPOINTZ;  | 
1245  | 0  |                 m_eRequestedGeomType = wkbMultiPointM;  | 
1246  | 0  |                 break;  | 
1247  |  |  | 
1248  | 0  |             case wkbLineString:  | 
1249  | 0  |             case wkbMultiLineString:  | 
1250  | 0  |                 nShapeType = SHPT_ARC;  | 
1251  | 0  |                 m_eRequestedGeomType = wkbLineString;  | 
1252  | 0  |                 break;  | 
1253  |  |  | 
1254  | 0  |             case wkbLineString25D:  | 
1255  | 0  |             case wkbMultiLineString25D:  | 
1256  | 0  |                 nShapeType = SHPT_ARCZ;  | 
1257  | 0  |                 m_eRequestedGeomType = wkbLineString25D;  | 
1258  | 0  |                 break;  | 
1259  |  |  | 
1260  | 0  |             case wkbLineStringM:  | 
1261  | 0  |             case wkbMultiLineStringM:  | 
1262  | 0  |                 nShapeType = SHPT_ARCM;  | 
1263  | 0  |                 m_eRequestedGeomType = wkbLineStringM;  | 
1264  | 0  |                 break;  | 
1265  |  |  | 
1266  | 0  |             case wkbLineStringZM:  | 
1267  | 0  |             case wkbMultiLineStringZM:  | 
1268  | 0  |                 nShapeType = SHPT_ARCZ;  | 
1269  | 0  |                 m_eRequestedGeomType = wkbLineStringZM;  | 
1270  | 0  |                 break;  | 
1271  |  |  | 
1272  | 0  |             case wkbPolygon:  | 
1273  | 0  |             case wkbMultiPolygon:  | 
1274  | 0  |             case wkbTriangle:  | 
1275  | 0  |                 nShapeType = SHPT_POLYGON;  | 
1276  | 0  |                 m_eRequestedGeomType = wkbPolygon;  | 
1277  | 0  |                 break;  | 
1278  |  |  | 
1279  | 0  |             case wkbPolygon25D:  | 
1280  | 0  |             case wkbMultiPolygon25D:  | 
1281  | 0  |             case wkbTriangleZ:  | 
1282  | 0  |                 nShapeType = SHPT_POLYGONZ;  | 
1283  | 0  |                 m_eRequestedGeomType = wkbPolygon25D;  | 
1284  | 0  |                 break;  | 
1285  |  |  | 
1286  | 0  |             case wkbPolygonM:  | 
1287  | 0  |             case wkbMultiPolygonM:  | 
1288  | 0  |             case wkbTriangleM:  | 
1289  | 0  |                 nShapeType = SHPT_POLYGONM;  | 
1290  | 0  |                 m_eRequestedGeomType = wkbPolygonM;  | 
1291  | 0  |                 break;  | 
1292  |  |  | 
1293  | 0  |             case wkbPolygonZM:  | 
1294  | 0  |             case wkbMultiPolygonZM:  | 
1295  | 0  |             case wkbTriangleZM:  | 
1296  | 0  |                 nShapeType = SHPT_POLYGONZ;  | 
1297  | 0  |                 m_eRequestedGeomType = wkbPolygonZM;  | 
1298  | 0  |                 break;  | 
1299  |  |  | 
1300  | 0  |             default:  | 
1301  | 0  |                 nShapeType = -1;  | 
1302  | 0  |                 break;  | 
1303  | 0  |         }  | 
1304  |  |  | 
1305  | 0  |         if (wkbFlatten(poGeom->getGeometryType()) == wkbTIN ||  | 
1306  | 0  |             wkbFlatten(poGeom->getGeometryType()) == wkbPolyhedralSurface)  | 
1307  | 0  |         { | 
1308  | 0  |             nShapeType = SHPT_MULTIPATCH;  | 
1309  | 0  |             m_eRequestedGeomType = wkbUnknown;  | 
1310  | 0  |         }  | 
1311  |  | 
  | 
1312  | 0  |         if (wkbFlatten(poGeom->getGeometryType()) == wkbGeometryCollection)  | 
1313  | 0  |         { | 
1314  | 0  |             const OGRGeometryCollection *poGC = poGeom->toGeometryCollection();  | 
1315  | 0  |             bool bIsMultiPatchCompatible = false;  | 
1316  | 0  |             for (int iGeom = 0; iGeom < poGC->getNumGeometries(); iGeom++)  | 
1317  | 0  |             { | 
1318  | 0  |                 OGRwkbGeometryType eSubGeomType =  | 
1319  | 0  |                     wkbFlatten(poGC->getGeometryRef(iGeom)->getGeometryType());  | 
1320  | 0  |                 if (eSubGeomType == wkbTIN ||  | 
1321  | 0  |                     eSubGeomType == wkbPolyhedralSurface)  | 
1322  | 0  |                 { | 
1323  | 0  |                     bIsMultiPatchCompatible = true;  | 
1324  | 0  |                 }  | 
1325  | 0  |                 else if (eSubGeomType != wkbMultiPolygon)  | 
1326  | 0  |                 { | 
1327  | 0  |                     bIsMultiPatchCompatible = false;  | 
1328  | 0  |                     break;  | 
1329  | 0  |                 }  | 
1330  | 0  |             }  | 
1331  | 0  |             if (bIsMultiPatchCompatible)  | 
1332  | 0  |             { | 
1333  | 0  |                 nShapeType = SHPT_MULTIPATCH;  | 
1334  | 0  |                 m_eRequestedGeomType = wkbUnknown;  | 
1335  | 0  |             }  | 
1336  | 0  |         }  | 
1337  |  | 
  | 
1338  | 0  |         if (nShapeType != -1)  | 
1339  | 0  |         { | 
1340  | 0  |             whileUnsealing(m_poFeatureDefn)->SetGeomType(m_eRequestedGeomType);  | 
1341  | 0  |             ResetGeomType(nShapeType);  | 
1342  | 0  |         }  | 
1343  | 0  |     }  | 
1344  |  |  | 
1345  | 0  |     const OGRErr eErr = SHPWriteOGRFeature(  | 
1346  | 0  |         m_hSHP, m_hDBF, m_poFeatureDefn, poFeature, m_osEncoding,  | 
1347  | 0  |         &m_bTruncationWarningEmitted, m_bRewindOnWrite);  | 
1348  |  | 
  | 
1349  | 0  |     if (m_hSHP != nullptr)  | 
1350  | 0  |         m_nTotalShapeCount = m_hSHP->nRecords;  | 
1351  | 0  |     else if (m_hDBF != nullptr)  | 
1352  | 0  |         m_nTotalShapeCount = m_hDBF->nRecords;  | 
1353  | 0  | #ifdef DEBUG  | 
1354  | 0  |     else  // Silence coverity.  | 
1355  | 0  |         CPLError(CE_Fatal, CPLE_AssertionFailed,  | 
1356  | 0  |                  "Should not happen: Both m_hSHP and m_hDBF are nullptrs");  | 
1357  | 0  | #endif  | 
1358  |  | 
  | 
1359  | 0  |     return eErr;  | 
1360  | 0  | }  | 
1361  |  |  | 
1362  |  | /************************************************************************/  | 
1363  |  | /*               GetFeatureCountWithSpatialFilterOnly()                 */  | 
1364  |  | /*                                                                      */  | 
1365  |  | /* Specialized implementation of GetFeatureCount() when there is *only* */  | 
1366  |  | /* a spatial filter and no attribute filter.                            */  | 
1367  |  | /************************************************************************/  | 
1368  |  |  | 
1369  |  | int OGRShapeLayer::GetFeatureCountWithSpatialFilterOnly()  | 
1370  |  |  | 
1371  | 0  | { | 
1372  |  |     /* -------------------------------------------------------------------- */  | 
1373  |  |     /*      Collect a matching list if we have attribute or spatial         */  | 
1374  |  |     /*      indices.  Only do this on the first request for a given pass    */  | 
1375  |  |     /*      of course.                                                      */  | 
1376  |  |     /* -------------------------------------------------------------------- */  | 
1377  | 0  |     if (m_panMatchingFIDs == nullptr)  | 
1378  | 0  |     { | 
1379  | 0  |         ScanIndices();  | 
1380  | 0  |     }  | 
1381  |  | 
  | 
1382  | 0  |     int nFeatureCount = 0;  | 
1383  | 0  |     int iLocalMatchingFID = 0;  | 
1384  | 0  |     int iLocalNextShapeId = 0;  | 
1385  | 0  |     bool bExpectPoints = false;  | 
1386  |  | 
  | 
1387  | 0  |     if (wkbFlatten(m_poFeatureDefn->GetGeomType()) == wkbPoint)  | 
1388  | 0  |         bExpectPoints = true;  | 
1389  |  |  | 
1390  |  |     /* -------------------------------------------------------------------- */  | 
1391  |  |     /*      Loop till we find a feature matching our criteria.              */  | 
1392  |  |     /* -------------------------------------------------------------------- */  | 
1393  |  | 
  | 
1394  | 0  |     SHPObject sShape;  | 
1395  | 0  |     memset(&sShape, 0, sizeof(sShape));  | 
1396  |  | 
  | 
1397  | 0  |     while (true)  | 
1398  | 0  |     { | 
1399  | 0  |         int iShape = -1;  | 
1400  |  | 
  | 
1401  | 0  |         if (m_panMatchingFIDs != nullptr)  | 
1402  | 0  |         { | 
1403  | 0  |             iShape = static_cast<int>(m_panMatchingFIDs[iLocalMatchingFID]);  | 
1404  | 0  |             if (iShape == OGRNullFID)  | 
1405  | 0  |                 break;  | 
1406  | 0  |             iLocalMatchingFID++;  | 
1407  | 0  |         }  | 
1408  | 0  |         else  | 
1409  | 0  |         { | 
1410  | 0  |             if (iLocalNextShapeId >= m_nTotalShapeCount)  | 
1411  | 0  |                 break;  | 
1412  | 0  |             iShape = iLocalNextShapeId++;  | 
1413  |  | 
  | 
1414  | 0  |             if (m_hDBF)  | 
1415  | 0  |             { | 
1416  | 0  |                 if (DBFIsRecordDeleted(m_hDBF, iShape))  | 
1417  | 0  |                     continue;  | 
1418  |  |  | 
1419  | 0  |                 if (VSIFEofL(VSI_SHP_GetVSIL(m_hDBF->fp)) ||  | 
1420  | 0  |                     VSIFErrorL(VSI_SHP_GetVSIL(m_hDBF->fp)))  | 
1421  | 0  |                     break;  | 
1422  | 0  |             }  | 
1423  | 0  |         }  | 
1424  |  |  | 
1425  |  |         // Read full shape for point layers.  | 
1426  | 0  |         SHPObject *psShape = nullptr;  | 
1427  | 0  |         if (bExpectPoints ||  | 
1428  | 0  |             m_hSHP->panRecOffset[iShape] == 0 /* lazy shx loading case */)  | 
1429  | 0  |             psShape = SHPReadObject(m_hSHP, iShape);  | 
1430  |  |  | 
1431  |  |         /* --------------------------------------------------------------------  | 
1432  |  |          */  | 
1433  |  |         /*      Only read feature type and bounding box for now. In case of */  | 
1434  |  |         /*      inconclusive tests on bounding box only, we will read the full  | 
1435  |  |          */  | 
1436  |  |         /*      shape later. */  | 
1437  |  |         /* --------------------------------------------------------------------  | 
1438  |  |          */  | 
1439  | 0  |         else if (iShape >= 0 && iShape < m_hSHP->nRecords &&  | 
1440  | 0  |                  m_hSHP->panRecSize[iShape] > 4 + 8 * 4)  | 
1441  | 0  |         { | 
1442  | 0  |             GByte abyBuf[4 + 8 * 4] = {}; | 
1443  | 0  |             if (m_hSHP->sHooks.FSeek(  | 
1444  | 0  |                     m_hSHP->fpSHP, m_hSHP->panRecOffset[iShape] + 8, 0) == 0 &&  | 
1445  | 0  |                 m_hSHP->sHooks.FRead(abyBuf, sizeof(abyBuf), 1,  | 
1446  | 0  |                                      m_hSHP->fpSHP) == 1)  | 
1447  | 0  |             { | 
1448  | 0  |                 memcpy(&(sShape.nSHPType), abyBuf, 4);  | 
1449  | 0  |                 CPL_LSBPTR32(&(sShape.nSHPType));  | 
1450  | 0  |                 if (sShape.nSHPType != SHPT_NULL &&  | 
1451  | 0  |                     sShape.nSHPType != SHPT_POINT &&  | 
1452  | 0  |                     sShape.nSHPType != SHPT_POINTM &&  | 
1453  | 0  |                     sShape.nSHPType != SHPT_POINTZ)  | 
1454  | 0  |                 { | 
1455  | 0  |                     psShape = &sShape;  | 
1456  | 0  |                     memcpy(&(sShape.dfXMin), abyBuf + 4, 8);  | 
1457  | 0  |                     memcpy(&(sShape.dfYMin), abyBuf + 12, 8);  | 
1458  | 0  |                     memcpy(&(sShape.dfXMax), abyBuf + 20, 8);  | 
1459  | 0  |                     memcpy(&(sShape.dfYMax), abyBuf + 28, 8);  | 
1460  | 0  |                     CPL_LSBPTR64(&(sShape.dfXMin));  | 
1461  | 0  |                     CPL_LSBPTR64(&(sShape.dfYMin));  | 
1462  | 0  |                     CPL_LSBPTR64(&(sShape.dfXMax));  | 
1463  | 0  |                     CPL_LSBPTR64(&(sShape.dfYMax));  | 
1464  | 0  |                 }  | 
1465  | 0  |             }  | 
1466  | 0  |             else  | 
1467  | 0  |             { | 
1468  | 0  |                 break;  | 
1469  | 0  |             }  | 
1470  | 0  |         }  | 
1471  |  |  | 
1472  | 0  |         if (psShape != nullptr && psShape->nSHPType != SHPT_NULL)  | 
1473  | 0  |         { | 
1474  | 0  |             OGRGeometry *poGeometry = nullptr;  | 
1475  | 0  |             OGREnvelope sGeomEnv;  | 
1476  |  |             // Test if we have a degenerated bounding box.  | 
1477  | 0  |             if (psShape->nSHPType != SHPT_POINT &&  | 
1478  | 0  |                 psShape->nSHPType != SHPT_POINTZ &&  | 
1479  | 0  |                 psShape->nSHPType != SHPT_POINTM &&  | 
1480  | 0  |                 (psShape->dfXMin == psShape->dfXMax ||  | 
1481  | 0  |                  psShape->dfYMin == psShape->dfYMax))  | 
1482  | 0  |             { | 
1483  |  |                 // Need to read the full geometry to compute the envelope.  | 
1484  | 0  |                 if (psShape == &sShape)  | 
1485  | 0  |                     psShape = SHPReadObject(m_hSHP, iShape);  | 
1486  |  | 
  | 
1487  | 0  |                 if (psShape)  | 
1488  | 0  |                 { | 
1489  | 0  |                     poGeometry = SHPReadOGRObject(  | 
1490  | 0  |                         m_hSHP, iShape, psShape, m_bHasWarnedWrongWindingOrder);  | 
1491  | 0  |                     if (poGeometry)  | 
1492  | 0  |                         poGeometry->getEnvelope(&sGeomEnv);  | 
1493  | 0  |                     psShape = nullptr;  | 
1494  | 0  |                 }  | 
1495  | 0  |             }  | 
1496  | 0  |             else  | 
1497  | 0  |             { | 
1498  |  |                 // Trust the shape bounding box as the shape envelope.  | 
1499  | 0  |                 sGeomEnv.MinX = psShape->dfXMin;  | 
1500  | 0  |                 sGeomEnv.MinY = psShape->dfYMin;  | 
1501  | 0  |                 sGeomEnv.MaxX = psShape->dfXMax;  | 
1502  | 0  |                 sGeomEnv.MaxY = psShape->dfYMax;  | 
1503  | 0  |             }  | 
1504  |  |  | 
1505  |  |             /* --------------------------------------------------------------------  | 
1506  |  |              */  | 
1507  |  |             /*      If there is no */  | 
1508  |  |             /*      intersection between the envelopes we are sure not to have  | 
1509  |  |              */  | 
1510  |  |             /*      any intersection. */  | 
1511  |  |             /* --------------------------------------------------------------------  | 
1512  |  |              */  | 
1513  | 0  |             if (sGeomEnv.MaxX < m_sFilterEnvelope.MinX ||  | 
1514  | 0  |                 sGeomEnv.MaxY < m_sFilterEnvelope.MinY ||  | 
1515  | 0  |                 m_sFilterEnvelope.MaxX < sGeomEnv.MinX ||  | 
1516  | 0  |                 m_sFilterEnvelope.MaxY < sGeomEnv.MinY)  | 
1517  | 0  |             { | 
1518  | 0  |             }  | 
1519  |  |             /* --------------------------------------------------------------------  | 
1520  |  |              */  | 
1521  |  |             /*      If the filter geometry is its own envelope and if the */  | 
1522  |  |             /*      envelope of the geometry is inside the filter geometry, */  | 
1523  |  |             /*      the geometry itself is inside the filter geometry */  | 
1524  |  |             /* --------------------------------------------------------------------  | 
1525  |  |              */  | 
1526  | 0  |             else if (m_bFilterIsEnvelope &&  | 
1527  | 0  |                      sGeomEnv.MinX >= m_sFilterEnvelope.MinX &&  | 
1528  | 0  |                      sGeomEnv.MinY >= m_sFilterEnvelope.MinY &&  | 
1529  | 0  |                      sGeomEnv.MaxX <= m_sFilterEnvelope.MaxX &&  | 
1530  | 0  |                      sGeomEnv.MaxY <= m_sFilterEnvelope.MaxY)  | 
1531  | 0  |             { | 
1532  | 0  |                 nFeatureCount++;  | 
1533  | 0  |             }  | 
1534  | 0  |             else  | 
1535  | 0  |             { | 
1536  |  |                 /* --------------------------------------------------------------------  | 
1537  |  |                  */  | 
1538  |  |                 /*      Fallback to full intersect test (using GEOS) if we still  | 
1539  |  |                  */  | 
1540  |  |                 /*      don't know for sure. */  | 
1541  |  |                 /* --------------------------------------------------------------------  | 
1542  |  |                  */  | 
1543  | 0  |                 if (OGRGeometryFactory::haveGEOS())  | 
1544  | 0  |                 { | 
1545  |  |                     // Read the full geometry.  | 
1546  | 0  |                     if (poGeometry == nullptr)  | 
1547  | 0  |                     { | 
1548  | 0  |                         if (psShape == &sShape)  | 
1549  | 0  |                             psShape = SHPReadObject(m_hSHP, iShape);  | 
1550  | 0  |                         if (psShape)  | 
1551  | 0  |                         { | 
1552  | 0  |                             poGeometry =  | 
1553  | 0  |                                 SHPReadOGRObject(m_hSHP, iShape, psShape,  | 
1554  | 0  |                                                  m_bHasWarnedWrongWindingOrder);  | 
1555  | 0  |                             psShape = nullptr;  | 
1556  | 0  |                         }  | 
1557  | 0  |                     }  | 
1558  | 0  |                     if (poGeometry == nullptr)  | 
1559  | 0  |                     { | 
1560  | 0  |                         nFeatureCount++;  | 
1561  | 0  |                     }  | 
1562  | 0  |                     else if (m_pPreparedFilterGeom != nullptr)  | 
1563  | 0  |                     { | 
1564  | 0  |                         if (OGRPreparedGeometryIntersects(  | 
1565  | 0  |                                 m_pPreparedFilterGeom,  | 
1566  | 0  |                                 OGRGeometry::ToHandle(poGeometry)))  | 
1567  | 0  |                         { | 
1568  | 0  |                             nFeatureCount++;  | 
1569  | 0  |                         }  | 
1570  | 0  |                     }  | 
1571  | 0  |                     else if (m_poFilterGeom->Intersects(poGeometry))  | 
1572  | 0  |                         nFeatureCount++;  | 
1573  | 0  |                 }  | 
1574  | 0  |                 else  | 
1575  | 0  |                 { | 
1576  | 0  |                     nFeatureCount++;  | 
1577  | 0  |                 }  | 
1578  | 0  |             }  | 
1579  |  | 
  | 
1580  | 0  |             delete poGeometry;  | 
1581  | 0  |         }  | 
1582  | 0  |         else  | 
1583  | 0  |         { | 
1584  | 0  |             nFeatureCount++;  | 
1585  | 0  |         }  | 
1586  |  | 
  | 
1587  | 0  |         if (psShape && psShape != &sShape)  | 
1588  | 0  |             SHPDestroyObject(psShape);  | 
1589  | 0  |     }  | 
1590  |  | 
  | 
1591  | 0  |     return nFeatureCount;  | 
1592  | 0  | }  | 
1593  |  |  | 
1594  |  | /************************************************************************/  | 
1595  |  | /*                          GetFeatureCount()                           */  | 
1596  |  | /************************************************************************/  | 
1597  |  |  | 
1598  |  | GIntBig OGRShapeLayer::GetFeatureCount(int bForce)  | 
1599  |  |  | 
1600  | 0  | { | 
1601  |  |     // Check if the spatial filter is non-trivial.  | 
1602  | 0  |     bool bHasTrivialSpatialFilter = false;  | 
1603  | 0  |     if (m_poFilterGeom != nullptr)  | 
1604  | 0  |     { | 
1605  | 0  |         OGREnvelope oSpatialFilterEnvelope;  | 
1606  | 0  |         m_poFilterGeom->getEnvelope(&oSpatialFilterEnvelope);  | 
1607  |  | 
  | 
1608  | 0  |         OGREnvelope oLayerExtent;  | 
1609  | 0  |         if (GetExtent(&oLayerExtent, TRUE) == OGRERR_NONE)  | 
1610  | 0  |         { | 
1611  | 0  |             if (oSpatialFilterEnvelope.Contains(oLayerExtent))  | 
1612  | 0  |             { | 
1613  | 0  |                 bHasTrivialSpatialFilter = true;  | 
1614  | 0  |             }  | 
1615  | 0  |             else  | 
1616  | 0  |             { | 
1617  | 0  |                 bHasTrivialSpatialFilter = false;  | 
1618  | 0  |             }  | 
1619  | 0  |         }  | 
1620  | 0  |         else  | 
1621  | 0  |         { | 
1622  | 0  |             bHasTrivialSpatialFilter = false;  | 
1623  | 0  |         }  | 
1624  | 0  |     }  | 
1625  | 0  |     else  | 
1626  | 0  |     { | 
1627  | 0  |         bHasTrivialSpatialFilter = true;  | 
1628  | 0  |     }  | 
1629  |  | 
  | 
1630  | 0  |     if (bHasTrivialSpatialFilter && m_poAttrQuery == nullptr)  | 
1631  | 0  |         return m_nTotalShapeCount;  | 
1632  |  |  | 
1633  | 0  |     if (!TouchLayer())  | 
1634  | 0  |         return 0;  | 
1635  |  |  | 
1636  |  |     // Spatial filter only.  | 
1637  | 0  |     if (m_poAttrQuery == nullptr && m_hSHP != nullptr)  | 
1638  | 0  |     { | 
1639  | 0  |         return GetFeatureCountWithSpatialFilterOnly();  | 
1640  | 0  |     }  | 
1641  |  |  | 
1642  |  |     // Attribute filter only.  | 
1643  | 0  |     if (m_poAttrQuery != nullptr && m_poFilterGeom == nullptr)  | 
1644  | 0  |     { | 
1645  |  |         // See if we can ignore reading geometries.  | 
1646  | 0  |         const bool bSaveGeometryIgnored =  | 
1647  | 0  |             CPL_TO_BOOL(m_poFeatureDefn->IsGeometryIgnored());  | 
1648  | 0  |         if (!AttributeFilterEvaluationNeedsGeometry())  | 
1649  | 0  |             m_poFeatureDefn->SetGeometryIgnored(TRUE);  | 
1650  |  | 
  | 
1651  | 0  |         GIntBig nRet = OGRLayer::GetFeatureCount(bForce);  | 
1652  |  | 
  | 
1653  | 0  |         m_poFeatureDefn->SetGeometryIgnored(bSaveGeometryIgnored);  | 
1654  | 0  |         return nRet;  | 
1655  | 0  |     }  | 
1656  |  |  | 
1657  | 0  |     return OGRLayer::GetFeatureCount(bForce);  | 
1658  | 0  | }  | 
1659  |  |  | 
1660  |  | /************************************************************************/  | 
1661  |  | /*                            IGetExtent()                              */  | 
1662  |  | /*                                                                      */  | 
1663  |  | /*      Fetch extent of the data currently stored in the dataset.       */  | 
1664  |  | /*      The bForce flag has no effect on SHP files since that value     */  | 
1665  |  | /*      is always in the header.                                        */  | 
1666  |  | /*                                                                      */  | 
1667  |  | /*      Returns OGRERR_NONE/OGRRERR_FAILURE.                            */  | 
1668  |  | /************************************************************************/  | 
1669  |  |  | 
1670  |  | OGRErr OGRShapeLayer::IGetExtent(int iGeomField, OGREnvelope *psExtent,  | 
1671  |  |                                  bool bForce)  | 
1672  |  |  | 
1673  | 0  | { | 
1674  | 0  |     if (!TouchLayer())  | 
1675  | 0  |         return OGRERR_FAILURE;  | 
1676  |  |  | 
1677  | 0  |     if (m_hSHP == nullptr)  | 
1678  | 0  |         return OGRERR_FAILURE;  | 
1679  |  |  | 
1680  | 0  |     double adMin[4] = {0.0, 0.0, 0.0, 0.0}; | 
1681  | 0  |     double adMax[4] = {0.0, 0.0, 0.0, 0.0}; | 
1682  |  | 
  | 
1683  | 0  |     SHPGetInfo(m_hSHP, nullptr, nullptr, adMin, adMax);  | 
1684  |  | 
  | 
1685  | 0  |     psExtent->MinX = adMin[0];  | 
1686  | 0  |     psExtent->MinY = adMin[1];  | 
1687  | 0  |     psExtent->MaxX = adMax[0];  | 
1688  | 0  |     psExtent->MaxY = adMax[1];  | 
1689  |  | 
  | 
1690  | 0  |     if (std::isnan(adMin[0]) || std::isnan(adMin[1]) || std::isnan(adMax[0]) ||  | 
1691  | 0  |         std::isnan(adMax[1]))  | 
1692  | 0  |     { | 
1693  | 0  |         CPLDebug("SHAPE", "Invalid extent in shape header"); | 
1694  |  |  | 
1695  |  |         // Disable filters to avoid infinite recursion in GetNextFeature()  | 
1696  |  |         // that calls ScanIndices() that call GetExtent.  | 
1697  | 0  |         OGRFeatureQuery *poAttrQuery = m_poAttrQuery;  | 
1698  | 0  |         m_poAttrQuery = nullptr;  | 
1699  | 0  |         OGRGeometry *poFilterGeom = m_poFilterGeom;  | 
1700  | 0  |         m_poFilterGeom = nullptr;  | 
1701  |  | 
  | 
1702  | 0  |         psExtent->MinX = 0;  | 
1703  | 0  |         psExtent->MinY = 0;  | 
1704  | 0  |         psExtent->MaxX = 0;  | 
1705  | 0  |         psExtent->MaxY = 0;  | 
1706  |  | 
  | 
1707  | 0  |         const OGRErr eErr = OGRLayer::IGetExtent(iGeomField, psExtent, bForce);  | 
1708  |  | 
  | 
1709  | 0  |         m_poAttrQuery = poAttrQuery;  | 
1710  | 0  |         m_poFilterGeom = poFilterGeom;  | 
1711  | 0  |         return eErr;  | 
1712  | 0  |     }  | 
1713  |  |  | 
1714  | 0  |     return OGRERR_NONE;  | 
1715  | 0  | }  | 
1716  |  |  | 
1717  |  | OGRErr OGRShapeLayer::IGetExtent3D(int iGeomField, OGREnvelope3D *psExtent3D,  | 
1718  |  |                                    bool bForce)  | 
1719  | 0  | { | 
1720  | 0  |     if (m_poFilterGeom || m_poAttrQuery)  | 
1721  | 0  |         return OGRLayer::IGetExtent3D(iGeomField, psExtent3D, bForce);  | 
1722  |  |  | 
1723  | 0  |     if (!TouchLayer())  | 
1724  | 0  |         return OGRERR_FAILURE;  | 
1725  |  |  | 
1726  | 0  |     if (m_hSHP == nullptr)  | 
1727  | 0  |         return OGRERR_FAILURE;  | 
1728  |  |  | 
1729  | 0  |     double adMin[4] = {0.0, 0.0, 0.0, 0.0}; | 
1730  | 0  |     double adMax[4] = {0.0, 0.0, 0.0, 0.0}; | 
1731  |  | 
  | 
1732  | 0  |     SHPGetInfo(m_hSHP, nullptr, nullptr, adMin, adMax);  | 
1733  |  | 
  | 
1734  | 0  |     psExtent3D->MinX = adMin[0];  | 
1735  | 0  |     psExtent3D->MinY = adMin[1];  | 
1736  | 0  |     psExtent3D->MaxX = adMax[0];  | 
1737  | 0  |     psExtent3D->MaxY = adMax[1];  | 
1738  |  | 
  | 
1739  | 0  |     if (OGR_GT_HasZ(m_poFeatureDefn->GetGeomType()))  | 
1740  | 0  |     { | 
1741  | 0  |         psExtent3D->MinZ = adMin[2];  | 
1742  | 0  |         psExtent3D->MaxZ = adMax[2];  | 
1743  | 0  |     }  | 
1744  | 0  |     else  | 
1745  | 0  |     { | 
1746  | 0  |         psExtent3D->MinZ = std::numeric_limits<double>::infinity();  | 
1747  | 0  |         psExtent3D->MaxZ = -std::numeric_limits<double>::infinity();  | 
1748  | 0  |     }  | 
1749  |  | 
  | 
1750  | 0  |     if (std::isnan(adMin[0]) || std::isnan(adMin[1]) || std::isnan(adMax[0]) ||  | 
1751  | 0  |         std::isnan(adMax[1]))  | 
1752  | 0  |     { | 
1753  | 0  |         CPLDebug("SHAPE", "Invalid extent in shape header"); | 
1754  |  |  | 
1755  |  |         // Disable filters to avoid infinite recursion in GetNextFeature()  | 
1756  |  |         // that calls ScanIndices() that call GetExtent.  | 
1757  | 0  |         OGRFeatureQuery *poAttrQuery = m_poAttrQuery;  | 
1758  | 0  |         m_poAttrQuery = nullptr;  | 
1759  | 0  |         OGRGeometry *poFilterGeom = m_poFilterGeom;  | 
1760  | 0  |         m_poFilterGeom = nullptr;  | 
1761  |  | 
  | 
1762  | 0  |         const OGRErr eErr =  | 
1763  | 0  |             OGRLayer::IGetExtent3D(iGeomField, psExtent3D, bForce);  | 
1764  |  | 
  | 
1765  | 0  |         m_poAttrQuery = poAttrQuery;  | 
1766  | 0  |         m_poFilterGeom = poFilterGeom;  | 
1767  | 0  |         return eErr;  | 
1768  | 0  |     }  | 
1769  |  |  | 
1770  | 0  |     return OGRERR_NONE;  | 
1771  | 0  | }  | 
1772  |  |  | 
1773  |  | /************************************************************************/  | 
1774  |  | /*                           TestCapability()                           */  | 
1775  |  | /************************************************************************/  | 
1776  |  |  | 
1777  |  | int OGRShapeLayer::TestCapability(const char *pszCap)  | 
1778  |  |  | 
1779  | 0  | { | 
1780  | 0  |     if (!TouchLayer())  | 
1781  | 0  |         return FALSE;  | 
1782  |  |  | 
1783  | 0  |     if (EQUAL(pszCap, OLCRandomRead))  | 
1784  | 0  |         return TRUE;  | 
1785  |  |  | 
1786  | 0  |     if (EQUAL(pszCap, OLCSequentialWrite) || EQUAL(pszCap, OLCRandomWrite))  | 
1787  | 0  |         return m_bUpdateAccess;  | 
1788  |  |  | 
1789  | 0  |     if (EQUAL(pszCap, OLCFastFeatureCount))  | 
1790  | 0  |     { | 
1791  | 0  |         if (!(m_poFilterGeom == nullptr || CheckForQIX() || CheckForSBN()))  | 
1792  | 0  |             return FALSE;  | 
1793  |  |  | 
1794  | 0  |         if (m_poAttrQuery != nullptr)  | 
1795  | 0  |         { | 
1796  | 0  |             InitializeIndexSupport(m_osFullName.c_str());  | 
1797  | 0  |             return m_poAttrQuery->CanUseIndex(this);  | 
1798  | 0  |         }  | 
1799  | 0  |         return TRUE;  | 
1800  | 0  |     }  | 
1801  |  |  | 
1802  | 0  |     if (EQUAL(pszCap, OLCDeleteFeature))  | 
1803  | 0  |         return m_bUpdateAccess;  | 
1804  |  |  | 
1805  | 0  |     if (EQUAL(pszCap, OLCFastSpatialFilter))  | 
1806  | 0  |         return CheckForQIX() || CheckForSBN();  | 
1807  |  |  | 
1808  | 0  |     if (EQUAL(pszCap, OLCFastGetExtent))  | 
1809  | 0  |         return TRUE;  | 
1810  |  |  | 
1811  | 0  |     if (EQUAL(pszCap, OLCFastGetExtent3D))  | 
1812  | 0  |         return m_poFilterGeom == nullptr && m_poAttrQuery == nullptr;  | 
1813  |  |  | 
1814  | 0  |     if (EQUAL(pszCap, OLCFastSetNextByIndex))  | 
1815  | 0  |         return m_poFilterGeom == nullptr && m_poAttrQuery == nullptr;  | 
1816  |  |  | 
1817  | 0  |     if (EQUAL(pszCap, OLCCreateField))  | 
1818  | 0  |         return m_bUpdateAccess;  | 
1819  |  |  | 
1820  | 0  |     if (EQUAL(pszCap, OLCDeleteField))  | 
1821  | 0  |         return m_bUpdateAccess;  | 
1822  |  |  | 
1823  | 0  |     if (EQUAL(pszCap, OLCReorderFields))  | 
1824  | 0  |         return m_bUpdateAccess;  | 
1825  |  |  | 
1826  | 0  |     if (EQUAL(pszCap, OLCAlterFieldDefn) ||  | 
1827  | 0  |         EQUAL(pszCap, OLCAlterGeomFieldDefn))  | 
1828  | 0  |         return m_bUpdateAccess;  | 
1829  |  |  | 
1830  | 0  |     if (EQUAL(pszCap, OLCRename))  | 
1831  | 0  |         return m_bUpdateAccess;  | 
1832  |  |  | 
1833  | 0  |     if (EQUAL(pszCap, OLCIgnoreFields))  | 
1834  | 0  |         return TRUE;  | 
1835  |  |  | 
1836  | 0  |     if (EQUAL(pszCap, OLCStringsAsUTF8))  | 
1837  | 0  |     { | 
1838  |  |         // No encoding defined: we don't know.  | 
1839  | 0  |         if (m_osEncoding.empty())  | 
1840  | 0  |             return FALSE;  | 
1841  |  |  | 
1842  | 0  |         if (m_hDBF == nullptr || DBFGetFieldCount(m_hDBF) == 0)  | 
1843  | 0  |             return TRUE;  | 
1844  |  |  | 
1845  |  |         // Otherwise test that we can re-encode field names to UTF-8.  | 
1846  | 0  |         const int nFieldCount = DBFGetFieldCount(m_hDBF);  | 
1847  | 0  |         for (int i = 0; i < nFieldCount; i++)  | 
1848  | 0  |         { | 
1849  | 0  |             char szFieldName[XBASE_FLDNAME_LEN_READ + 1] = {}; | 
1850  | 0  |             int nWidth = 0;  | 
1851  | 0  |             int nPrecision = 0;  | 
1852  |  | 
  | 
1853  | 0  |             DBFGetFieldInfo(m_hDBF, i, szFieldName, &nWidth, &nPrecision);  | 
1854  |  | 
  | 
1855  | 0  |             if (!CPLCanRecode(szFieldName, m_osEncoding, CPL_ENC_UTF8))  | 
1856  | 0  |             { | 
1857  | 0  |                 return FALSE;  | 
1858  | 0  |             }  | 
1859  | 0  |         }  | 
1860  |  |  | 
1861  | 0  |         return TRUE;  | 
1862  | 0  |     }  | 
1863  |  |  | 
1864  | 0  |     if (EQUAL(pszCap, OLCMeasuredGeometries))  | 
1865  | 0  |         return TRUE;  | 
1866  |  |  | 
1867  | 0  |     if (EQUAL(pszCap, OLCZGeometries))  | 
1868  | 0  |         return TRUE;  | 
1869  |  |  | 
1870  | 0  |     return FALSE;  | 
1871  | 0  | }  | 
1872  |  |  | 
1873  |  | /************************************************************************/  | 
1874  |  | /*                            CreateField()                             */  | 
1875  |  | /************************************************************************/  | 
1876  |  |  | 
1877  |  | OGRErr OGRShapeLayer::CreateField(const OGRFieldDefn *poFieldDefn,  | 
1878  |  |                                   int bApproxOK)  | 
1879  |  |  | 
1880  | 0  | { | 
1881  | 0  |     if (!StartUpdate("CreateField")) | 
1882  | 0  |         return OGRERR_FAILURE;  | 
1883  |  |  | 
1884  | 0  |     CPLAssert(nullptr != poFieldDefn);  | 
1885  |  |  | 
1886  | 0  |     bool bDBFJustCreated = false;  | 
1887  | 0  |     if (m_hDBF == nullptr)  | 
1888  | 0  |     { | 
1889  | 0  |         const CPLString osFilename =  | 
1890  | 0  |             CPLResetExtensionSafe(m_osFullName.c_str(), "dbf");  | 
1891  | 0  |         m_hDBF = DBFCreate(osFilename);  | 
1892  |  | 
  | 
1893  | 0  |         if (m_hDBF == nullptr)  | 
1894  | 0  |         { | 
1895  | 0  |             CPLError(CE_Failure, CPLE_OpenFailed,  | 
1896  | 0  |                      "Failed to create DBF file `%s'.", osFilename.c_str());  | 
1897  | 0  |             return OGRERR_FAILURE;  | 
1898  | 0  |         }  | 
1899  |  |  | 
1900  | 0  |         bDBFJustCreated = true;  | 
1901  | 0  |     }  | 
1902  |  |  | 
1903  | 0  |     if (m_hDBF->nHeaderLength + XBASE_FLDHDR_SZ > 65535)  | 
1904  | 0  |     { | 
1905  | 0  |         CPLError(CE_Failure, CPLE_NotSupported,  | 
1906  | 0  |                  "Cannot add field %s. Header length limit reached "  | 
1907  | 0  |                  "(max 65535 bytes, 2046 fields).",  | 
1908  | 0  |                  poFieldDefn->GetNameRef());  | 
1909  | 0  |         return OGRERR_FAILURE;  | 
1910  | 0  |     }  | 
1911  |  |  | 
1912  | 0  |     CPLErrorReset();  | 
1913  |  | 
  | 
1914  | 0  |     if (m_poFeatureDefn->GetFieldCount() == 255)  | 
1915  | 0  |     { | 
1916  | 0  |         CPLError(CE_Warning, CPLE_AppDefined,  | 
1917  | 0  |                  "Creating a 256th field, "  | 
1918  | 0  |                  "but some DBF readers might only support 255 fields");  | 
1919  | 0  |     }  | 
1920  |  |  | 
1921  |  |     /* -------------------------------------------------------------------- */  | 
1922  |  |     /*      Normalize field name                                            */  | 
1923  |  |     /* -------------------------------------------------------------------- */  | 
1924  | 0  |     CPLString osFieldName;  | 
1925  | 0  |     if (!m_osEncoding.empty())  | 
1926  | 0  |     { | 
1927  | 0  |         CPLClearRecodeWarningFlags();  | 
1928  | 0  |         CPLPushErrorHandler(CPLQuietErrorHandler);  | 
1929  | 0  |         CPLErr eLastErr = CPLGetLastErrorType();  | 
1930  | 0  |         char *const pszRecoded =  | 
1931  | 0  |             CPLRecode(poFieldDefn->GetNameRef(), CPL_ENC_UTF8, m_osEncoding);  | 
1932  | 0  |         CPLPopErrorHandler();  | 
1933  | 0  |         osFieldName = pszRecoded;  | 
1934  | 0  |         CPLFree(pszRecoded);  | 
1935  | 0  |         if (CPLGetLastErrorType() != eLastErr)  | 
1936  | 0  |         { | 
1937  | 0  |             CPLError(CE_Failure, CPLE_AppDefined,  | 
1938  | 0  |                      "Failed to create field name '%s': cannot convert to %s",  | 
1939  | 0  |                      poFieldDefn->GetNameRef(), m_osEncoding.c_str());  | 
1940  | 0  |             return OGRERR_FAILURE;  | 
1941  | 0  |         }  | 
1942  | 0  |     }  | 
1943  | 0  |     else  | 
1944  | 0  |     { | 
1945  | 0  |         osFieldName = poFieldDefn->GetNameRef();  | 
1946  | 0  |     }  | 
1947  |  |  | 
1948  | 0  |     const int nNameSize = static_cast<int>(osFieldName.size());  | 
1949  | 0  |     char szNewFieldName[XBASE_FLDNAME_LEN_WRITE + 1];  | 
1950  | 0  |     CPLString osRadixFieldName;  | 
1951  | 0  |     CPLString osRadixFieldNameUC;  | 
1952  | 0  |     { | 
1953  | 0  |         char *pszTmp = CPLScanString(  | 
1954  | 0  |             osFieldName, std::min(nNameSize, XBASE_FLDNAME_LEN_WRITE), TRUE,  | 
1955  | 0  |             TRUE);  | 
1956  | 0  |         strncpy(szNewFieldName, pszTmp, sizeof(szNewFieldName) - 1);  | 
1957  | 0  |         szNewFieldName[sizeof(szNewFieldName) - 1] = '\0';  | 
1958  | 0  |         osRadixFieldName = pszTmp;  | 
1959  | 0  |         osRadixFieldNameUC = CPLString(osRadixFieldName).toupper();  | 
1960  | 0  |         CPLFree(pszTmp);  | 
1961  | 0  |     }  | 
1962  |  | 
  | 
1963  | 0  |     CPLString osNewFieldNameUC(szNewFieldName);  | 
1964  | 0  |     osNewFieldNameUC.toupper();  | 
1965  |  | 
  | 
1966  | 0  |     if (m_oSetUCFieldName.empty())  | 
1967  | 0  |     { | 
1968  | 0  |         for (int i = 0; i < m_poFeatureDefn->GetFieldCount(); i++)  | 
1969  | 0  |         { | 
1970  | 0  |             m_oSetUCFieldName.insert(  | 
1971  | 0  |                 CPLString(m_poFeatureDefn->GetFieldDefn(i)->GetNameRef())  | 
1972  | 0  |                     .toupper());  | 
1973  | 0  |         }  | 
1974  | 0  |     }  | 
1975  |  | 
  | 
1976  | 0  |     bool bFoundFieldName =  | 
1977  | 0  |         m_oSetUCFieldName.find(osNewFieldNameUC) != m_oSetUCFieldName.end();  | 
1978  |  | 
  | 
1979  | 0  |     if (!bApproxOK && (bFoundFieldName || !EQUAL(osFieldName, szNewFieldName)))  | 
1980  | 0  |     { | 
1981  | 0  |         CPLError(CE_Failure, CPLE_NotSupported,  | 
1982  | 0  |                  "Failed to add field named '%s'", poFieldDefn->GetNameRef());  | 
1983  |  | 
  | 
1984  | 0  |         return OGRERR_FAILURE;  | 
1985  | 0  |     }  | 
1986  |  |  | 
1987  | 0  |     if (bFoundFieldName)  | 
1988  | 0  |     { | 
1989  | 0  |         int nRenameNum = 1;  | 
1990  | 0  |         while (bFoundFieldName && nRenameNum < 10)  | 
1991  | 0  |         { | 
1992  | 0  |             CPLsnprintf(szNewFieldName, sizeof(szNewFieldName), "%.8s_%.1d",  | 
1993  | 0  |                         osRadixFieldName.c_str(), nRenameNum);  | 
1994  | 0  |             osNewFieldNameUC.Printf("%.8s_%.1d", osRadixFieldNameUC.c_str(), | 
1995  | 0  |                                     nRenameNum);  | 
1996  | 0  |             bFoundFieldName = m_oSetUCFieldName.find(osNewFieldNameUC) !=  | 
1997  | 0  |                               m_oSetUCFieldName.end();  | 
1998  | 0  |             nRenameNum++;  | 
1999  | 0  |         }  | 
2000  |  | 
  | 
2001  | 0  |         while (bFoundFieldName && nRenameNum < 100)  | 
2002  | 0  |         { | 
2003  | 0  |             CPLsnprintf(szNewFieldName, sizeof(szNewFieldName), "%.8s%.2d",  | 
2004  | 0  |                         osRadixFieldName.c_str(), nRenameNum);  | 
2005  | 0  |             osNewFieldNameUC.Printf("%.8s%.2d", osRadixFieldNameUC.c_str(), | 
2006  | 0  |                                     nRenameNum);  | 
2007  | 0  |             bFoundFieldName = m_oSetUCFieldName.find(osNewFieldNameUC) !=  | 
2008  | 0  |                               m_oSetUCFieldName.end();  | 
2009  | 0  |             nRenameNum++;  | 
2010  | 0  |         }  | 
2011  |  | 
  | 
2012  | 0  |         if (bFoundFieldName)  | 
2013  | 0  |         { | 
2014  |  |             // One hundred similar field names!!?  | 
2015  | 0  |             CPLError(  | 
2016  | 0  |                 CE_Failure, CPLE_NotSupported,  | 
2017  | 0  |                 "Too many field names like '%s' when truncated to %d letters "  | 
2018  | 0  |                 "for Shapefile format.",  | 
2019  | 0  |                 poFieldDefn->GetNameRef(), XBASE_FLDNAME_LEN_WRITE);  | 
2020  | 0  |             return OGRERR_FAILURE;  | 
2021  | 0  |         }  | 
2022  | 0  |     }  | 
2023  |  |  | 
2024  | 0  |     OGRFieldDefn oModFieldDefn(poFieldDefn);  | 
2025  |  | 
  | 
2026  | 0  |     if (!EQUAL(osFieldName, szNewFieldName))  | 
2027  | 0  |     { | 
2028  | 0  |         CPLError(CE_Warning, CPLE_NotSupported,  | 
2029  | 0  |                  "Normalized/laundered field name: '%s' to '%s'",  | 
2030  | 0  |                  poFieldDefn->GetNameRef(), szNewFieldName);  | 
2031  |  |  | 
2032  |  |         // Set field name with normalized value.  | 
2033  | 0  |         oModFieldDefn.SetName(szNewFieldName);  | 
2034  | 0  |     }  | 
2035  |  |  | 
2036  |  |     /* -------------------------------------------------------------------- */  | 
2037  |  |     /*      Add field to layer                                              */  | 
2038  |  |     /* -------------------------------------------------------------------- */  | 
2039  | 0  |     char chType = 'C';  | 
2040  | 0  |     int nWidth = 0;  | 
2041  | 0  |     int nDecimals = 0;  | 
2042  |  | 
  | 
2043  | 0  |     switch (oModFieldDefn.GetType())  | 
2044  | 0  |     { | 
2045  | 0  |         case OFTInteger:  | 
2046  | 0  |             if (oModFieldDefn.GetSubType() == OFSTBoolean)  | 
2047  | 0  |             { | 
2048  | 0  |                 chType = 'L';  | 
2049  | 0  |                 nWidth = 1;  | 
2050  | 0  |             }  | 
2051  | 0  |             else  | 
2052  | 0  |             { | 
2053  | 0  |                 chType = 'N';  | 
2054  | 0  |                 nWidth = oModFieldDefn.GetWidth();  | 
2055  | 0  |                 if (nWidth == 0)  | 
2056  | 0  |                     nWidth = 9;  | 
2057  | 0  |             }  | 
2058  | 0  |             break;  | 
2059  |  |  | 
2060  | 0  |         case OFTInteger64:  | 
2061  | 0  |             chType = 'N';  | 
2062  | 0  |             nWidth = oModFieldDefn.GetWidth();  | 
2063  | 0  |             if (nWidth == 0)  | 
2064  | 0  |                 nWidth = 18;  | 
2065  | 0  |             break;  | 
2066  |  |  | 
2067  | 0  |         case OFTReal:  | 
2068  | 0  |             chType = 'N';  | 
2069  | 0  |             nWidth = oModFieldDefn.GetWidth();  | 
2070  | 0  |             nDecimals = oModFieldDefn.GetPrecision();  | 
2071  | 0  |             if (nWidth == 0)  | 
2072  | 0  |             { | 
2073  | 0  |                 nWidth = 24;  | 
2074  | 0  |                 nDecimals = 15;  | 
2075  | 0  |             }  | 
2076  | 0  |             break;  | 
2077  |  |  | 
2078  | 0  |         case OFTString:  | 
2079  | 0  |             chType = 'C';  | 
2080  | 0  |             nWidth = oModFieldDefn.GetWidth();  | 
2081  | 0  |             if (nWidth == 0)  | 
2082  | 0  |                 nWidth = 80;  | 
2083  | 0  |             else if (nWidth > OGR_DBF_MAX_FIELD_WIDTH)  | 
2084  | 0  |             { | 
2085  | 0  |                 CPLError(CE_Warning, CPLE_AppDefined,  | 
2086  | 0  |                          "Field %s of width %d truncated to %d.",  | 
2087  | 0  |                          szNewFieldName, nWidth, OGR_DBF_MAX_FIELD_WIDTH);  | 
2088  | 0  |                 nWidth = OGR_DBF_MAX_FIELD_WIDTH;  | 
2089  | 0  |             }  | 
2090  | 0  |             break;  | 
2091  |  |  | 
2092  | 0  |         case OFTDate:  | 
2093  | 0  |             chType = 'D';  | 
2094  | 0  |             nWidth = 8;  | 
2095  | 0  |             break;  | 
2096  |  |  | 
2097  | 0  |         case OFTDateTime:  | 
2098  | 0  |             CPLError(  | 
2099  | 0  |                 CE_Warning, CPLE_NotSupported,  | 
2100  | 0  |                 "Field %s created as String field, though DateTime requested.",  | 
2101  | 0  |                 szNewFieldName);  | 
2102  | 0  |             chType = 'C';  | 
2103  | 0  |             nWidth = static_cast<int>(strlen("YYYY-MM-DDTHH:MM:SS.sss+HH:MM")); | 
2104  | 0  |             oModFieldDefn.SetType(OFTString);  | 
2105  | 0  |             break;  | 
2106  |  |  | 
2107  | 0  |         default:  | 
2108  | 0  |             CPLError(CE_Failure, CPLE_NotSupported,  | 
2109  | 0  |                      "Can't create fields of type %s on shapefile layers.",  | 
2110  | 0  |                      OGRFieldDefn::GetFieldTypeName(oModFieldDefn.GetType()));  | 
2111  |  | 
  | 
2112  | 0  |             return OGRERR_FAILURE;  | 
2113  | 0  |             break;  | 
2114  | 0  |     }  | 
2115  |  |  | 
2116  | 0  |     oModFieldDefn.SetWidth(nWidth);  | 
2117  | 0  |     oModFieldDefn.SetPrecision(nDecimals);  | 
2118  |  |  | 
2119  |  |     // Suppress the dummy FID field if we have created it just before.  | 
2120  | 0  |     if (DBFGetFieldCount(m_hDBF) == 1 && m_poFeatureDefn->GetFieldCount() == 0)  | 
2121  | 0  |     { | 
2122  | 0  |         DBFDeleteField(m_hDBF, 0);  | 
2123  | 0  |     }  | 
2124  |  | 
  | 
2125  | 0  |     const int iNewField = DBFAddNativeFieldType(m_hDBF, szNewFieldName, chType,  | 
2126  | 0  |                                                 nWidth, nDecimals);  | 
2127  |  | 
  | 
2128  | 0  |     if (iNewField != -1)  | 
2129  | 0  |     { | 
2130  | 0  |         m_oSetUCFieldName.insert(std::move(osNewFieldNameUC));  | 
2131  |  | 
  | 
2132  | 0  |         whileUnsealing(m_poFeatureDefn)->AddFieldDefn(&oModFieldDefn);  | 
2133  |  | 
  | 
2134  | 0  |         if (bDBFJustCreated)  | 
2135  | 0  |         { | 
2136  | 0  |             for (int i = 0; i < m_nTotalShapeCount; i++)  | 
2137  | 0  |             { | 
2138  | 0  |                 DBFWriteNULLAttribute(m_hDBF, i, 0);  | 
2139  | 0  |             }  | 
2140  | 0  |         }  | 
2141  |  | 
  | 
2142  | 0  |         return OGRERR_NONE;  | 
2143  | 0  |     }  | 
2144  |  |  | 
2145  | 0  |     CPLError(CE_Failure, CPLE_AppDefined,  | 
2146  | 0  |              "Can't create field %s in Shape DBF file, reason unknown.",  | 
2147  | 0  |              szNewFieldName);  | 
2148  |  | 
  | 
2149  | 0  |     return OGRERR_FAILURE;  | 
2150  | 0  | }  | 
2151  |  |  | 
2152  |  | /************************************************************************/  | 
2153  |  | /*                            DeleteField()                             */  | 
2154  |  | /************************************************************************/  | 
2155  |  |  | 
2156  |  | OGRErr OGRShapeLayer::DeleteField(int iField)  | 
2157  | 0  | { | 
2158  | 0  |     if (!StartUpdate("DeleteField")) | 
2159  | 0  |         return OGRERR_FAILURE;  | 
2160  |  |  | 
2161  | 0  |     if (iField < 0 || iField >= m_poFeatureDefn->GetFieldCount())  | 
2162  | 0  |     { | 
2163  | 0  |         CPLError(CE_Failure, CPLE_NotSupported, "Invalid field index");  | 
2164  | 0  |         return OGRERR_FAILURE;  | 
2165  | 0  |     }  | 
2166  |  |  | 
2167  | 0  |     m_oSetUCFieldName.clear();  | 
2168  |  | 
  | 
2169  | 0  |     if (DBFDeleteField(m_hDBF, iField))  | 
2170  | 0  |     { | 
2171  | 0  |         TruncateDBF();  | 
2172  |  | 
  | 
2173  | 0  |         return whileUnsealing(m_poFeatureDefn)->DeleteFieldDefn(iField);  | 
2174  | 0  |     }  | 
2175  |  |  | 
2176  | 0  |     return OGRERR_FAILURE;  | 
2177  | 0  | }  | 
2178  |  |  | 
2179  |  | /************************************************************************/  | 
2180  |  | /*                           ReorderFields()                            */  | 
2181  |  | /************************************************************************/  | 
2182  |  |  | 
2183  |  | OGRErr OGRShapeLayer::ReorderFields(int *panMap)  | 
2184  | 0  | { | 
2185  | 0  |     if (!StartUpdate("ReorderFields")) | 
2186  | 0  |         return OGRERR_FAILURE;  | 
2187  |  |  | 
2188  | 0  |     if (m_poFeatureDefn->GetFieldCount() == 0)  | 
2189  | 0  |         return OGRERR_NONE;  | 
2190  |  |  | 
2191  | 0  |     OGRErr eErr = OGRCheckPermutation(panMap, m_poFeatureDefn->GetFieldCount());  | 
2192  | 0  |     if (eErr != OGRERR_NONE)  | 
2193  | 0  |         return eErr;  | 
2194  |  |  | 
2195  | 0  |     if (DBFReorderFields(m_hDBF, panMap))  | 
2196  | 0  |     { | 
2197  | 0  |         return whileUnsealing(m_poFeatureDefn)->ReorderFieldDefns(panMap);  | 
2198  | 0  |     }  | 
2199  |  |  | 
2200  | 0  |     return OGRERR_FAILURE;  | 
2201  | 0  | }  | 
2202  |  |  | 
2203  |  | /************************************************************************/  | 
2204  |  | /*                           AlterFieldDefn()                           */  | 
2205  |  | /************************************************************************/  | 
2206  |  |  | 
2207  |  | OGRErr OGRShapeLayer::AlterFieldDefn(int iField, OGRFieldDefn *poNewFieldDefn,  | 
2208  |  |                                      int nFlagsIn)  | 
2209  | 0  | { | 
2210  | 0  |     if (!StartUpdate("AlterFieldDefn")) | 
2211  | 0  |         return OGRERR_FAILURE;  | 
2212  |  |  | 
2213  | 0  |     if (iField < 0 || iField >= m_poFeatureDefn->GetFieldCount())  | 
2214  | 0  |     { | 
2215  | 0  |         CPLError(CE_Failure, CPLE_NotSupported, "Invalid field index");  | 
2216  | 0  |         return OGRERR_FAILURE;  | 
2217  | 0  |     }  | 
2218  |  |  | 
2219  | 0  |     m_oSetUCFieldName.clear();  | 
2220  |  | 
  | 
2221  | 0  |     OGRFieldDefn *poFieldDefn = m_poFeatureDefn->GetFieldDefn(iField);  | 
2222  | 0  |     OGRFieldType eType = poFieldDefn->GetType();  | 
2223  |  | 
  | 
2224  | 0  |     auto oTemporaryUnsealer(poFieldDefn->GetTemporaryUnsealer());  | 
2225  |  |  | 
2226  |  |     // On reading we support up to 11 characters  | 
2227  | 0  |     char szFieldName[XBASE_FLDNAME_LEN_READ + 1] = {}; | 
2228  | 0  |     int nWidth = 0;  | 
2229  | 0  |     int nPrecision = 0;  | 
2230  | 0  |     DBFGetFieldInfo(m_hDBF, iField, szFieldName, &nWidth, &nPrecision);  | 
2231  | 0  |     char chNativeType = DBFGetNativeFieldType(m_hDBF, iField);  | 
2232  |  | 
  | 
2233  | 0  |     if ((nFlagsIn & ALTER_TYPE_FLAG) &&  | 
2234  | 0  |         poNewFieldDefn->GetType() != poFieldDefn->GetType())  | 
2235  | 0  |     { | 
2236  | 0  |         if (poNewFieldDefn->GetType() == OFTInteger64 &&  | 
2237  | 0  |             poFieldDefn->GetType() == OFTInteger)  | 
2238  | 0  |         { | 
2239  | 0  |             eType = poNewFieldDefn->GetType();  | 
2240  | 0  |         }  | 
2241  | 0  |         else if (poNewFieldDefn->GetType() != OFTString)  | 
2242  | 0  |         { | 
2243  | 0  |             CPLError(CE_Failure, CPLE_NotSupported,  | 
2244  | 0  |                      "Can only convert to OFTString");  | 
2245  | 0  |             return OGRERR_FAILURE;  | 
2246  | 0  |         }  | 
2247  | 0  |         else  | 
2248  | 0  |         { | 
2249  | 0  |             chNativeType = 'C';  | 
2250  | 0  |             eType = poNewFieldDefn->GetType();  | 
2251  | 0  |         }  | 
2252  | 0  |     }  | 
2253  |  |  | 
2254  | 0  |     if (nFlagsIn & ALTER_NAME_FLAG)  | 
2255  | 0  |     { | 
2256  | 0  |         CPLString osFieldName;  | 
2257  | 0  |         if (!m_osEncoding.empty())  | 
2258  | 0  |         { | 
2259  | 0  |             CPLClearRecodeWarningFlags();  | 
2260  | 0  |             CPLErrorReset();  | 
2261  | 0  |             CPLPushErrorHandler(CPLQuietErrorHandler);  | 
2262  | 0  |             char *pszRecoded = CPLRecode(poNewFieldDefn->GetNameRef(),  | 
2263  | 0  |                                          CPL_ENC_UTF8, m_osEncoding);  | 
2264  | 0  |             CPLPopErrorHandler();  | 
2265  | 0  |             osFieldName = pszRecoded;  | 
2266  | 0  |             CPLFree(pszRecoded);  | 
2267  | 0  |             if (CPLGetLastErrorType() != 0)  | 
2268  | 0  |             { | 
2269  | 0  |                 CPLError(CE_Failure, CPLE_AppDefined,  | 
2270  | 0  |                          "Failed to rename field name to '%s': "  | 
2271  | 0  |                          "cannot convert to %s",  | 
2272  | 0  |                          poNewFieldDefn->GetNameRef(), m_osEncoding.c_str());  | 
2273  | 0  |                 return OGRERR_FAILURE;  | 
2274  | 0  |             }  | 
2275  | 0  |         }  | 
2276  | 0  |         else  | 
2277  | 0  |         { | 
2278  | 0  |             osFieldName = poNewFieldDefn->GetNameRef();  | 
2279  | 0  |         }  | 
2280  |  |  | 
2281  | 0  |         strncpy(szFieldName, osFieldName, sizeof(szFieldName) - 1);  | 
2282  | 0  |         szFieldName[sizeof(szFieldName) - 1] = '\0';  | 
2283  | 0  |     }  | 
2284  | 0  |     if (nFlagsIn & ALTER_WIDTH_PRECISION_FLAG)  | 
2285  | 0  |     { | 
2286  | 0  |         nWidth = poNewFieldDefn->GetWidth();  | 
2287  | 0  |         nPrecision = poNewFieldDefn->GetPrecision();  | 
2288  | 0  |     }  | 
2289  |  | 
  | 
2290  | 0  |     if (DBFAlterFieldDefn(m_hDBF, iField, szFieldName, chNativeType, nWidth,  | 
2291  | 0  |                           nPrecision))  | 
2292  | 0  |     { | 
2293  | 0  |         if (nFlagsIn & ALTER_TYPE_FLAG)  | 
2294  | 0  |             poFieldDefn->SetType(eType);  | 
2295  | 0  |         if (nFlagsIn & ALTER_NAME_FLAG)  | 
2296  | 0  |             poFieldDefn->SetName(poNewFieldDefn->GetNameRef());  | 
2297  | 0  |         if (nFlagsIn & ALTER_WIDTH_PRECISION_FLAG)  | 
2298  | 0  |         { | 
2299  | 0  |             poFieldDefn->SetWidth(nWidth);  | 
2300  | 0  |             poFieldDefn->SetPrecision(nPrecision);  | 
2301  |  | 
  | 
2302  | 0  |             TruncateDBF();  | 
2303  | 0  |         }  | 
2304  | 0  |         return OGRERR_NONE;  | 
2305  | 0  |     }  | 
2306  |  |  | 
2307  | 0  |     return OGRERR_FAILURE;  | 
2308  | 0  | }  | 
2309  |  |  | 
2310  |  | /************************************************************************/  | 
2311  |  | /*                         AlterGeomFieldDefn()                         */  | 
2312  |  | /************************************************************************/  | 
2313  |  |  | 
2314  |  | OGRErr OGRShapeLayer::AlterGeomFieldDefn(  | 
2315  |  |     int iGeomField, const OGRGeomFieldDefn *poNewGeomFieldDefn, int nFlagsIn)  | 
2316  | 0  | { | 
2317  | 0  |     if (!StartUpdate("AlterGeomFieldDefn")) | 
2318  | 0  |         return OGRERR_FAILURE;  | 
2319  |  |  | 
2320  | 0  |     if (iGeomField < 0 || iGeomField >= m_poFeatureDefn->GetGeomFieldCount())  | 
2321  | 0  |     { | 
2322  | 0  |         CPLError(CE_Failure, CPLE_NotSupported, "Invalid field index");  | 
2323  | 0  |         return OGRERR_FAILURE;  | 
2324  | 0  |     }  | 
2325  |  |  | 
2326  | 0  |     auto poFieldDefn = cpl::down_cast<OGRShapeGeomFieldDefn *>(  | 
2327  | 0  |         m_poFeatureDefn->GetGeomFieldDefn(iGeomField));  | 
2328  | 0  |     auto oTemporaryUnsealer(poFieldDefn->GetTemporaryUnsealer());  | 
2329  |  | 
  | 
2330  | 0  |     if (nFlagsIn & ALTER_GEOM_FIELD_DEFN_NAME_FLAG)  | 
2331  | 0  |     { | 
2332  | 0  |         if (strcmp(poNewGeomFieldDefn->GetNameRef(),  | 
2333  | 0  |                    poFieldDefn->GetNameRef()) != 0)  | 
2334  | 0  |         { | 
2335  | 0  |             CPLError(CE_Failure, CPLE_NotSupported,  | 
2336  | 0  |                      "Altering the geometry field name is not supported for "  | 
2337  | 0  |                      "shapefiles");  | 
2338  | 0  |             return OGRERR_FAILURE;  | 
2339  | 0  |         }  | 
2340  | 0  |     }  | 
2341  |  |  | 
2342  | 0  |     if (nFlagsIn & ALTER_GEOM_FIELD_DEFN_TYPE_FLAG)  | 
2343  | 0  |     { | 
2344  | 0  |         if (poFieldDefn->GetType() != poNewGeomFieldDefn->GetType())  | 
2345  | 0  |         { | 
2346  | 0  |             CPLError(CE_Failure, CPLE_NotSupported,  | 
2347  | 0  |                      "Altering the geometry field type is not supported for "  | 
2348  | 0  |                      "shapefiles");  | 
2349  | 0  |             return OGRERR_FAILURE;  | 
2350  | 0  |         }  | 
2351  | 0  |     }  | 
2352  |  |  | 
2353  | 0  |     if (nFlagsIn & ALTER_GEOM_FIELD_DEFN_SRS_COORD_EPOCH_FLAG)  | 
2354  | 0  |     { | 
2355  | 0  |         const auto poNewSRSRef = poNewGeomFieldDefn->GetSpatialRef();  | 
2356  | 0  |         if (poNewSRSRef && poNewSRSRef->GetCoordinateEpoch() > 0)  | 
2357  | 0  |         { | 
2358  | 0  |             CPLError(CE_Failure, CPLE_NotSupported,  | 
2359  | 0  |                      "Setting a coordinate epoch is not supported for "  | 
2360  | 0  |                      "shapefiles");  | 
2361  | 0  |             return OGRERR_FAILURE;  | 
2362  | 0  |         }  | 
2363  | 0  |     }  | 
2364  |  |  | 
2365  | 0  |     if (nFlagsIn & ALTER_GEOM_FIELD_DEFN_SRS_FLAG)  | 
2366  | 0  |     { | 
2367  | 0  |         if (poFieldDefn->GetPrjFilename().empty())  | 
2368  | 0  |         { | 
2369  | 0  |             poFieldDefn->SetPrjFilename(  | 
2370  | 0  |                 CPLResetExtensionSafe(m_osFullName.c_str(), "prj").c_str());  | 
2371  | 0  |         }  | 
2372  |  | 
  | 
2373  | 0  |         const auto poNewSRSRef = poNewGeomFieldDefn->GetSpatialRef();  | 
2374  | 0  |         if (poNewSRSRef)  | 
2375  | 0  |         { | 
2376  | 0  |             char *pszWKT = nullptr;  | 
2377  | 0  |             VSILFILE *fp = nullptr;  | 
2378  | 0  |             const char *const apszOptions[] = {"FORMAT=WKT1_ESRI", nullptr}; | 
2379  | 0  |             if (poNewSRSRef->exportToWkt(&pszWKT, apszOptions) == OGRERR_NONE &&  | 
2380  | 0  |                 (fp = VSIFOpenL(poFieldDefn->GetPrjFilename().c_str(), "wt")) !=  | 
2381  | 0  |                     nullptr)  | 
2382  | 0  |             { | 
2383  | 0  |                 VSIFWriteL(pszWKT, strlen(pszWKT), 1, fp);  | 
2384  | 0  |                 VSIFCloseL(fp);  | 
2385  | 0  |             }  | 
2386  | 0  |             else  | 
2387  | 0  |             { | 
2388  | 0  |                 CPLError(CE_Failure, CPLE_FileIO, "Cannot write %s",  | 
2389  | 0  |                          poFieldDefn->GetPrjFilename().c_str());  | 
2390  | 0  |                 CPLFree(pszWKT);  | 
2391  | 0  |                 return OGRERR_FAILURE;  | 
2392  | 0  |             }  | 
2393  |  |  | 
2394  | 0  |             CPLFree(pszWKT);  | 
2395  |  | 
  | 
2396  | 0  |             auto poNewSRS = poNewSRSRef->Clone();  | 
2397  | 0  |             poFieldDefn->SetSpatialRef(poNewSRS);  | 
2398  | 0  |             poNewSRS->Release();  | 
2399  | 0  |         }  | 
2400  | 0  |         else  | 
2401  | 0  |         { | 
2402  | 0  |             poFieldDefn->SetSpatialRef(nullptr);  | 
2403  | 0  |             VSIStatBufL sStat;  | 
2404  | 0  |             if (VSIStatL(poFieldDefn->GetPrjFilename().c_str(), &sStat) == 0 &&  | 
2405  | 0  |                 VSIUnlink(poFieldDefn->GetPrjFilename().c_str()) != 0)  | 
2406  | 0  |             { | 
2407  | 0  |                 CPLError(CE_Failure, CPLE_FileIO, "Cannot delete %s",  | 
2408  | 0  |                          poFieldDefn->GetPrjFilename().c_str());  | 
2409  | 0  |                 return OGRERR_FAILURE;  | 
2410  | 0  |             }  | 
2411  | 0  |         }  | 
2412  | 0  |         poFieldDefn->SetSRSSet();  | 
2413  | 0  |     }  | 
2414  |  |  | 
2415  | 0  |     if (nFlagsIn & ALTER_GEOM_FIELD_DEFN_NAME_FLAG)  | 
2416  | 0  |         poFieldDefn->SetName(poNewGeomFieldDefn->GetNameRef());  | 
2417  | 0  |     if (nFlagsIn & ALTER_GEOM_FIELD_DEFN_NULLABLE_FLAG)  | 
2418  | 0  |         poFieldDefn->SetNullable(poNewGeomFieldDefn->IsNullable());  | 
2419  |  | 
  | 
2420  | 0  |     return OGRERR_NONE;  | 
2421  | 0  | }  | 
2422  |  |  | 
2423  |  | /************************************************************************/  | 
2424  |  | /*                           GetSpatialRef()                            */  | 
2425  |  | /************************************************************************/  | 
2426  |  |  | 
2427  |  | const OGRSpatialReference *OGRShapeGeomFieldDefn::GetSpatialRef() const  | 
2428  |  |  | 
2429  | 0  | { | 
2430  | 0  |     if (m_bSRSSet)  | 
2431  | 0  |         return poSRS;  | 
2432  |  |  | 
2433  | 0  |     m_bSRSSet = true;  | 
2434  |  |  | 
2435  |  |     /* -------------------------------------------------------------------- */  | 
2436  |  |     /*      Is there an associated .prj file we can read?                   */  | 
2437  |  |     /* -------------------------------------------------------------------- */  | 
2438  | 0  |     std::string l_osPrjFile =  | 
2439  | 0  |         CPLResetExtensionSafe(m_osFullName.c_str(), "prj");  | 
2440  |  | 
  | 
2441  | 0  |     char *apszOptions[] = { | 
2442  | 0  |         const_cast<char *>("EMIT_ERROR_IF_CANNOT_OPEN_FILE=FALSE"), nullptr}; | 
2443  | 0  |     char **papszLines = CSLLoad2(l_osPrjFile.c_str(), -1, -1, apszOptions);  | 
2444  | 0  |     if (papszLines == nullptr)  | 
2445  | 0  |     { | 
2446  | 0  |         l_osPrjFile = CPLResetExtensionSafe(m_osFullName.c_str(), "PRJ");  | 
2447  | 0  |         papszLines = CSLLoad2(l_osPrjFile.c_str(), -1, -1, apszOptions);  | 
2448  | 0  |     }  | 
2449  |  | 
  | 
2450  | 0  |     if (papszLines != nullptr)  | 
2451  | 0  |     { | 
2452  | 0  |         m_osPrjFile = std::move(l_osPrjFile);  | 
2453  |  | 
  | 
2454  | 0  |         auto poSRSNonConst = new OGRSpatialReference();  | 
2455  | 0  |         poSRSNonConst->SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);  | 
2456  |  |         // Remove UTF-8 BOM if found  | 
2457  |  |         // http://lists.osgeo.org/pipermail/gdal-dev/2014-July/039527.html  | 
2458  | 0  |         if (static_cast<unsigned char>(papszLines[0][0]) == 0xEF &&  | 
2459  | 0  |             static_cast<unsigned char>(papszLines[0][1]) == 0xBB &&  | 
2460  | 0  |             static_cast<unsigned char>(papszLines[0][2]) == 0xBF)  | 
2461  | 0  |         { | 
2462  | 0  |             memmove(papszLines[0], papszLines[0] + 3,  | 
2463  | 0  |                     strlen(papszLines[0] + 3) + 1);  | 
2464  | 0  |         }  | 
2465  | 0  |         if (STARTS_WITH_CI(papszLines[0], "GEOGCS["))  | 
2466  | 0  |         { | 
2467  |  |             // Strip AXIS[] in GEOGCS to address use case of  | 
2468  |  |             // https://github.com/OSGeo/gdal/issues/8452  | 
2469  | 0  |             std::string osVal;  | 
2470  | 0  |             for (CSLConstList papszIter = papszLines; *papszIter; ++papszIter)  | 
2471  | 0  |                 osVal += *papszIter;  | 
2472  | 0  |             OGR_SRSNode oSRSNode;  | 
2473  | 0  |             const char *pszVal = osVal.c_str();  | 
2474  | 0  |             if (oSRSNode.importFromWkt(&pszVal) == OGRERR_NONE)  | 
2475  | 0  |             { | 
2476  | 0  |                 oSRSNode.StripNodes("AXIS"); | 
2477  | 0  |                 char *pszWKT = nullptr;  | 
2478  | 0  |                 oSRSNode.exportToWkt(&pszWKT);  | 
2479  | 0  |                 if (pszWKT)  | 
2480  | 0  |                 { | 
2481  | 0  |                     CSLDestroy(papszLines);  | 
2482  | 0  |                     papszLines =  | 
2483  | 0  |                         static_cast<char **>(CPLCalloc(2, sizeof(char *)));  | 
2484  | 0  |                     papszLines[0] = pszWKT;  | 
2485  | 0  |                 }  | 
2486  | 0  |             }  | 
2487  | 0  |         }  | 
2488  | 0  |         if (poSRSNonConst->importFromESRI(papszLines) != OGRERR_NONE)  | 
2489  | 0  |         { | 
2490  | 0  |             delete poSRSNonConst;  | 
2491  | 0  |             poSRSNonConst = nullptr;  | 
2492  | 0  |         }  | 
2493  | 0  |         CSLDestroy(papszLines);  | 
2494  |  | 
  | 
2495  | 0  |         if (poSRSNonConst)  | 
2496  | 0  |         { | 
2497  | 0  |             if (CPLTestBool(CPLGetConfigOption("USE_OSR_FIND_MATCHES", "YES"))) | 
2498  | 0  |             { | 
2499  | 0  |                 auto poSRSMatch = poSRSNonConst->FindBestMatch();  | 
2500  | 0  |                 if (poSRSMatch)  | 
2501  | 0  |                 { | 
2502  | 0  |                     poSRSNonConst->Release();  | 
2503  | 0  |                     poSRSNonConst = poSRSMatch;  | 
2504  | 0  |                     poSRSNonConst->SetAxisMappingStrategy(  | 
2505  | 0  |                         OAMS_TRADITIONAL_GIS_ORDER);  | 
2506  | 0  |                 }  | 
2507  | 0  |             }  | 
2508  | 0  |             else  | 
2509  | 0  |             { | 
2510  | 0  |                 poSRSNonConst->AutoIdentifyEPSG();  | 
2511  | 0  |             }  | 
2512  | 0  |             poSRS = poSRSNonConst;  | 
2513  | 0  |         }  | 
2514  | 0  |     }  | 
2515  |  | 
  | 
2516  | 0  |     return poSRS;  | 
2517  | 0  | }  | 
2518  |  |  | 
2519  |  | /************************************************************************/  | 
2520  |  | /*                           ResetGeomType()                            */  | 
2521  |  | /*                                                                      */  | 
2522  |  | /*      Modify the geometry type for this file.  Used to convert to     */  | 
2523  |  | /*      a different geometry type when a layer was created with a       */  | 
2524  |  | /*      type of unknown, and we get to the first feature to             */  | 
2525  |  | /*      establish the type.                                             */  | 
2526  |  | /************************************************************************/  | 
2527  |  |  | 
2528  |  | int OGRShapeLayer::ResetGeomType(int nNewGeomType)  | 
2529  |  |  | 
2530  | 0  | { | 
2531  | 0  |     if (m_nTotalShapeCount > 0)  | 
2532  | 0  |         return FALSE;  | 
2533  |  |  | 
2534  | 0  |     if (m_hSHP->fpSHX == nullptr)  | 
2535  | 0  |     { | 
2536  | 0  |         CPLError(CE_Failure, CPLE_NotSupported,  | 
2537  | 0  |                  "OGRShapeLayer::ResetGeomType failed: SHX file is closed");  | 
2538  | 0  |         return FALSE;  | 
2539  | 0  |     }  | 
2540  |  |  | 
2541  |  |     /* -------------------------------------------------------------------- */  | 
2542  |  |     /*      Update .shp header.                                             */  | 
2543  |  |     /* -------------------------------------------------------------------- */  | 
2544  | 0  |     int nStartPos = static_cast<int>(m_hSHP->sHooks.FTell(m_hSHP->fpSHP));  | 
2545  |  | 
  | 
2546  | 0  |     char abyHeader[100] = {}; | 
2547  | 0  |     if (m_hSHP->sHooks.FSeek(m_hSHP->fpSHP, 0, SEEK_SET) != 0 ||  | 
2548  | 0  |         m_hSHP->sHooks.FRead(abyHeader, 100, 1, m_hSHP->fpSHP) != 1)  | 
2549  | 0  |         return FALSE;  | 
2550  |  |  | 
2551  | 0  |     *(reinterpret_cast<GInt32 *>(abyHeader + 32)) = CPL_LSBWORD32(nNewGeomType);  | 
2552  |  | 
  | 
2553  | 0  |     if (m_hSHP->sHooks.FSeek(m_hSHP->fpSHP, 0, SEEK_SET) != 0 ||  | 
2554  | 0  |         m_hSHP->sHooks.FWrite(abyHeader, 100, 1, m_hSHP->fpSHP) != 1)  | 
2555  | 0  |         return FALSE;  | 
2556  |  |  | 
2557  | 0  |     if (m_hSHP->sHooks.FSeek(m_hSHP->fpSHP, nStartPos, SEEK_SET) != 0)  | 
2558  | 0  |         return FALSE;  | 
2559  |  |  | 
2560  |  |     /* -------------------------------------------------------------------- */  | 
2561  |  |     /*      Update .shx header.                                             */  | 
2562  |  |     /* -------------------------------------------------------------------- */  | 
2563  | 0  |     nStartPos = static_cast<int>(m_hSHP->sHooks.FTell(m_hSHP->fpSHX));  | 
2564  |  | 
  | 
2565  | 0  |     if (m_hSHP->sHooks.FSeek(m_hSHP->fpSHX, 0, SEEK_SET) != 0 ||  | 
2566  | 0  |         m_hSHP->sHooks.FRead(abyHeader, 100, 1, m_hSHP->fpSHX) != 1)  | 
2567  | 0  |         return FALSE;  | 
2568  |  |  | 
2569  | 0  |     *(reinterpret_cast<GInt32 *>(abyHeader + 32)) = CPL_LSBWORD32(nNewGeomType);  | 
2570  |  | 
  | 
2571  | 0  |     if (m_hSHP->sHooks.FSeek(m_hSHP->fpSHX, 0, SEEK_SET) != 0 ||  | 
2572  | 0  |         m_hSHP->sHooks.FWrite(abyHeader, 100, 1, m_hSHP->fpSHX) != 1)  | 
2573  | 0  |         return FALSE;  | 
2574  |  |  | 
2575  | 0  |     if (m_hSHP->sHooks.FSeek(m_hSHP->fpSHX, nStartPos, SEEK_SET) != 0)  | 
2576  | 0  |         return FALSE;  | 
2577  |  |  | 
2578  |  |     /* -------------------------------------------------------------------- */  | 
2579  |  |     /*      Update other information.                                       */  | 
2580  |  |     /* -------------------------------------------------------------------- */  | 
2581  | 0  |     m_hSHP->nShapeType = nNewGeomType;  | 
2582  |  | 
  | 
2583  | 0  |     return TRUE;  | 
2584  | 0  | }  | 
2585  |  |  | 
2586  |  | /************************************************************************/  | 
2587  |  | /*                             SyncToDisk()                             */  | 
2588  |  | /************************************************************************/  | 
2589  |  |  | 
2590  |  | OGRErr OGRShapeLayer::SyncToDisk()  | 
2591  |  |  | 
2592  | 0  | { | 
2593  | 0  |     if (!TouchLayer())  | 
2594  | 0  |         return OGRERR_FAILURE;  | 
2595  |  |  | 
2596  | 0  |     if (m_bHeaderDirty)  | 
2597  | 0  |     { | 
2598  | 0  |         if (m_hSHP != nullptr)  | 
2599  | 0  |             SHPWriteHeader(m_hSHP);  | 
2600  |  | 
  | 
2601  | 0  |         if (m_hDBF != nullptr)  | 
2602  | 0  |             DBFUpdateHeader(m_hDBF);  | 
2603  |  | 
  | 
2604  | 0  |         m_bHeaderDirty = false;  | 
2605  | 0  |     }  | 
2606  |  | 
  | 
2607  | 0  |     if (m_hSHP != nullptr)  | 
2608  | 0  |     { | 
2609  | 0  |         m_hSHP->sHooks.FFlush(m_hSHP->fpSHP);  | 
2610  | 0  |         if (m_hSHP->fpSHX != nullptr)  | 
2611  | 0  |             m_hSHP->sHooks.FFlush(m_hSHP->fpSHX);  | 
2612  | 0  |     }  | 
2613  |  | 
  | 
2614  | 0  |     if (m_hDBF != nullptr)  | 
2615  | 0  |     { | 
2616  | 0  |         m_hDBF->sHooks.FFlush(m_hDBF->fp);  | 
2617  | 0  |     }  | 
2618  |  | 
  | 
2619  | 0  |     if (m_eNeedRepack == YES && m_bAutoRepack)  | 
2620  | 0  |         Repack();  | 
2621  |  | 
  | 
2622  | 0  |     return OGRERR_NONE;  | 
2623  | 0  | }  | 
2624  |  |  | 
2625  |  | /************************************************************************/  | 
2626  |  | /*                          DropSpatialIndex()                          */  | 
2627  |  | /************************************************************************/  | 
2628  |  |  | 
2629  |  | OGRErr OGRShapeLayer::DropSpatialIndex()  | 
2630  |  |  | 
2631  | 0  | { | 
2632  | 0  |     if (!StartUpdate("DropSpatialIndex")) | 
2633  | 0  |         return OGRERR_FAILURE;  | 
2634  |  |  | 
2635  | 0  |     if (!CheckForQIX() && !CheckForSBN())  | 
2636  | 0  |     { | 
2637  | 0  |         CPLError(CE_Warning, CPLE_AppDefined,  | 
2638  | 0  |                  "Layer %s has no spatial index, DROP SPATIAL INDEX failed.",  | 
2639  | 0  |                  m_poFeatureDefn->GetName());  | 
2640  | 0  |         return OGRERR_FAILURE;  | 
2641  | 0  |     }  | 
2642  |  |  | 
2643  | 0  |     const bool bHadQIX = m_hQIX != nullptr;  | 
2644  |  | 
  | 
2645  | 0  |     SHPCloseDiskTree(m_hQIX);  | 
2646  | 0  |     m_hQIX = nullptr;  | 
2647  | 0  |     m_bCheckedForQIX = false;  | 
2648  |  | 
  | 
2649  | 0  |     SBNCloseDiskTree(m_hSBN);  | 
2650  | 0  |     m_hSBN = nullptr;  | 
2651  | 0  |     m_bCheckedForSBN = false;  | 
2652  |  | 
  | 
2653  | 0  |     if (bHadQIX)  | 
2654  | 0  |     { | 
2655  | 0  |         const std::string osQIXFilename =  | 
2656  | 0  |             CPLResetExtensionSafe(m_osFullName.c_str(), "qix");  | 
2657  | 0  |         CPLDebug("SHAPE", "Unlinking index file %s", osQIXFilename.c_str()); | 
2658  |  | 
  | 
2659  | 0  |         if (VSIUnlink(osQIXFilename.c_str()) != 0)  | 
2660  | 0  |         { | 
2661  | 0  |             CPLError(CE_Failure, CPLE_AppDefined,  | 
2662  | 0  |                      "Failed to delete file %s.\n%s", osQIXFilename.c_str(),  | 
2663  | 0  |                      VSIStrerror(errno));  | 
2664  | 0  |             return OGRERR_FAILURE;  | 
2665  | 0  |         }  | 
2666  | 0  |     }  | 
2667  |  |  | 
2668  | 0  |     if (!m_bSbnSbxDeleted)  | 
2669  | 0  |     { | 
2670  | 0  |         const char papszExt[2][4] = {"sbn", "sbx"}; | 
2671  | 0  |         for (int i = 0; i < 2; i++)  | 
2672  | 0  |         { | 
2673  | 0  |             const std::string osIndexFilename =  | 
2674  | 0  |                 CPLResetExtensionSafe(m_osFullName.c_str(), papszExt[i]);  | 
2675  | 0  |             CPLDebug("SHAPE", "Trying to unlink index file %s", | 
2676  | 0  |                      osIndexFilename.c_str());  | 
2677  |  | 
  | 
2678  | 0  |             if (VSIUnlink(osIndexFilename.c_str()) != 0)  | 
2679  | 0  |             { | 
2680  | 0  |                 CPLDebug("SHAPE", "Failed to delete file %s.\n%s", | 
2681  | 0  |                          osIndexFilename.c_str(), VSIStrerror(errno));  | 
2682  | 0  |             }  | 
2683  | 0  |         }  | 
2684  | 0  |     }  | 
2685  | 0  |     m_bSbnSbxDeleted = true;  | 
2686  |  | 
  | 
2687  | 0  |     ClearSpatialFIDs();  | 
2688  |  | 
  | 
2689  | 0  |     return OGRERR_NONE;  | 
2690  | 0  | }  | 
2691  |  |  | 
2692  |  | /************************************************************************/  | 
2693  |  | /*                         CreateSpatialIndex()                         */  | 
2694  |  | /************************************************************************/  | 
2695  |  |  | 
2696  |  | OGRErr OGRShapeLayer::CreateSpatialIndex(int nMaxDepth)  | 
2697  |  |  | 
2698  | 0  | { | 
2699  | 0  |     if (!StartUpdate("CreateSpatialIndex")) | 
2700  | 0  |         return OGRERR_FAILURE;  | 
2701  |  |  | 
2702  |  |     /* -------------------------------------------------------------------- */  | 
2703  |  |     /*      If we have an existing spatial index, blow it away first.       */  | 
2704  |  |     /* -------------------------------------------------------------------- */  | 
2705  | 0  |     if (CheckForQIX())  | 
2706  | 0  |         DropSpatialIndex();  | 
2707  |  | 
  | 
2708  | 0  |     m_bCheckedForQIX = false;  | 
2709  |  |  | 
2710  |  |     /* -------------------------------------------------------------------- */  | 
2711  |  |     /*      Build a quadtree structure for this file.                       */  | 
2712  |  |     /* -------------------------------------------------------------------- */  | 
2713  | 0  |     OGRShapeLayer::SyncToDisk();  | 
2714  | 0  |     SHPTree *psTree = SHPCreateTree(m_hSHP, 2, nMaxDepth, nullptr, nullptr);  | 
2715  |  | 
  | 
2716  | 0  |     if (nullptr == psTree)  | 
2717  | 0  |     { | 
2718  |  |         // TODO(mloskot): Is it better to return OGRERR_NOT_ENOUGH_MEMORY?  | 
2719  | 0  |         CPLDebug("SHAPE", | 
2720  | 0  |                  "Index creation failure. Likely, memory allocation error.");  | 
2721  |  | 
  | 
2722  | 0  |         return OGRERR_FAILURE;  | 
2723  | 0  |     }  | 
2724  |  |  | 
2725  |  |     /* -------------------------------------------------------------------- */  | 
2726  |  |     /*      Trim unused nodes from the tree.                                */  | 
2727  |  |     /* -------------------------------------------------------------------- */  | 
2728  | 0  |     SHPTreeTrimExtraNodes(psTree);  | 
2729  |  |  | 
2730  |  |     /* -------------------------------------------------------------------- */  | 
2731  |  |     /*      Dump tree to .qix file.                                         */  | 
2732  |  |     /* -------------------------------------------------------------------- */  | 
2733  | 0  |     char *pszQIXFilename =  | 
2734  | 0  |         CPLStrdup(CPLResetExtensionSafe(m_osFullName.c_str(), "qix").c_str());  | 
2735  |  | 
  | 
2736  | 0  |     CPLDebug("SHAPE", "Creating index file %s", pszQIXFilename); | 
2737  |  | 
  | 
2738  | 0  |     SHPWriteTree(psTree, pszQIXFilename);  | 
2739  | 0  |     CPLFree(pszQIXFilename);  | 
2740  |  |  | 
2741  |  |     /* -------------------------------------------------------------------- */  | 
2742  |  |     /*      cleanup                                                         */  | 
2743  |  |     /* -------------------------------------------------------------------- */  | 
2744  | 0  |     SHPDestroyTree(psTree);  | 
2745  |  | 
  | 
2746  | 0  |     CPL_IGNORE_RET_VAL(CheckForQIX());  | 
2747  |  | 
  | 
2748  | 0  |     return OGRERR_NONE;  | 
2749  | 0  | }  | 
2750  |  |  | 
2751  |  | /************************************************************************/  | 
2752  |  | /*                       CheckFileDeletion()                            */  | 
2753  |  | /************************************************************************/  | 
2754  |  |  | 
2755  |  | static void CheckFileDeletion(const CPLString &osFilename)  | 
2756  | 0  | { | 
2757  |  |     // On Windows, sometimes the file is still triansiently reported  | 
2758  |  |     // as existing although being deleted, which makes QGIS things that  | 
2759  |  |     // an issue arose. The following helps to reduce that risk.  | 
2760  | 0  |     VSIStatBufL sStat;  | 
2761  | 0  |     if (VSIStatL(osFilename, &sStat) == 0 && VSIStatL(osFilename, &sStat) == 0)  | 
2762  | 0  |     { | 
2763  | 0  |         CPLDebug("Shape", | 
2764  | 0  |                  "File %s is still reported as existing whereas "  | 
2765  | 0  |                  "it should have been deleted",  | 
2766  | 0  |                  osFilename.c_str());  | 
2767  | 0  |     }  | 
2768  | 0  | }  | 
2769  |  |  | 
2770  |  | /************************************************************************/  | 
2771  |  | /*                         ForceDeleteFile()                            */  | 
2772  |  | /************************************************************************/  | 
2773  |  |  | 
2774  |  | static void ForceDeleteFile(const CPLString &osFilename)  | 
2775  | 0  | { | 
2776  | 0  |     if (VSIUnlink(osFilename) != 0)  | 
2777  | 0  |     { | 
2778  |  |         // In case of failure retry with a small delay (Windows specific)  | 
2779  | 0  |         CPLSleep(0.1);  | 
2780  | 0  |         if (VSIUnlink(osFilename) != 0)  | 
2781  | 0  |         { | 
2782  | 0  |             CPLDebug("Shape", "Cannot delete %s : %s", osFilename.c_str(), | 
2783  | 0  |                      VSIStrerror(errno));  | 
2784  | 0  |         }  | 
2785  | 0  |     }  | 
2786  | 0  |     CheckFileDeletion(osFilename);  | 
2787  | 0  | }  | 
2788  |  |  | 
2789  |  | /************************************************************************/  | 
2790  |  | /*                               Repack()                               */  | 
2791  |  | /*                                                                      */  | 
2792  |  | /*      Repack the shape and dbf file, dropping deleted records.        */  | 
2793  |  | /*      FIDs may change.                                                */  | 
2794  |  | /************************************************************************/  | 
2795  |  |  | 
2796  |  | OGRErr OGRShapeLayer::Repack()  | 
2797  |  |  | 
2798  | 0  | { | 
2799  | 0  |     if (m_eNeedRepack == NO)  | 
2800  | 0  |     { | 
2801  | 0  |         CPLDebug("Shape", "REPACK: nothing to do. Was done previously"); | 
2802  | 0  |         return OGRERR_NONE;  | 
2803  | 0  |     }  | 
2804  |  |  | 
2805  | 0  |     if (!StartUpdate("Repack")) | 
2806  | 0  |         return OGRERR_FAILURE;  | 
2807  |  |  | 
2808  |  |     /* -------------------------------------------------------------------- */  | 
2809  |  |     /*      Build a list of records to be dropped.                          */  | 
2810  |  |     /* -------------------------------------------------------------------- */  | 
2811  | 0  |     std::vector<int> anRecordsToDelete;  | 
2812  | 0  |     OGRErr eErr = OGRERR_NONE;  | 
2813  |  | 
  | 
2814  | 0  |     CPLDebug("Shape", "REPACK: Checking if features have been deleted"); | 
2815  |  | 
  | 
2816  | 0  |     if (m_hDBF != nullptr)  | 
2817  | 0  |     { | 
2818  | 0  |         try  | 
2819  | 0  |         { | 
2820  | 0  |             for (int iShape = 0; iShape < m_nTotalShapeCount; iShape++)  | 
2821  | 0  |             { | 
2822  | 0  |                 if (DBFIsRecordDeleted(m_hDBF, iShape))  | 
2823  | 0  |                 { | 
2824  | 0  |                     anRecordsToDelete.push_back(iShape);  | 
2825  | 0  |                 }  | 
2826  | 0  |                 if (VSIFEofL(VSI_SHP_GetVSIL(m_hDBF->fp)) ||  | 
2827  | 0  |                     VSIFErrorL(VSI_SHP_GetVSIL(m_hDBF->fp)))  | 
2828  | 0  |                 { | 
2829  | 0  |                     return OGRERR_FAILURE;  // I/O error.  | 
2830  | 0  |                 }  | 
2831  | 0  |             }  | 
2832  | 0  |         }  | 
2833  | 0  |         catch (const std::bad_alloc &)  | 
2834  | 0  |         { | 
2835  | 0  |             CPLError(CE_Failure, CPLE_OutOfMemory, "Out of memory in Repack()");  | 
2836  | 0  |             return OGRERR_FAILURE;  | 
2837  | 0  |         }  | 
2838  | 0  |     }  | 
2839  |  |  | 
2840  |  |     /* -------------------------------------------------------------------- */  | 
2841  |  |     /*      If there are no records marked for deletion, we take no         */  | 
2842  |  |     /*      action.                                                         */  | 
2843  |  |     /* -------------------------------------------------------------------- */  | 
2844  | 0  |     if (anRecordsToDelete.empty() && !m_bSHPNeedsRepack)  | 
2845  | 0  |     { | 
2846  | 0  |         CPLDebug("Shape", "REPACK: nothing to do"); | 
2847  | 0  |         return OGRERR_NONE;  | 
2848  | 0  |     }  | 
2849  |  |  | 
2850  |  |     /* -------------------------------------------------------------------- */  | 
2851  |  |     /*      Find existing filenames with exact case (see #3293).            */  | 
2852  |  |     /* -------------------------------------------------------------------- */  | 
2853  | 0  |     const CPLString osDirname(CPLGetPathSafe(m_osFullName.c_str()));  | 
2854  | 0  |     const CPLString osBasename(CPLGetBasenameSafe(m_osFullName.c_str()));  | 
2855  |  | 
  | 
2856  | 0  |     CPLString osDBFName;  | 
2857  | 0  |     CPLString osSHPName;  | 
2858  | 0  |     CPLString osSHXName;  | 
2859  | 0  |     CPLString osCPGName;  | 
2860  | 0  |     char **papszCandidates = VSIReadDir(osDirname);  | 
2861  | 0  |     int i = 0;  | 
2862  | 0  |     while (papszCandidates != nullptr && papszCandidates[i] != nullptr)  | 
2863  | 0  |     { | 
2864  | 0  |         const CPLString osCandidateBasename =  | 
2865  | 0  |             CPLGetBasenameSafe(papszCandidates[i]);  | 
2866  | 0  |         const CPLString osCandidateExtension =  | 
2867  | 0  |             CPLGetExtensionSafe(papszCandidates[i]);  | 
2868  |  | #ifdef _WIN32  | 
2869  |  |         // On Windows, as filenames are case insensitive, a shapefile layer can  | 
2870  |  |         // be made of foo.shp and FOO.DBF, so use case insensitive comparison.  | 
2871  |  |         if (EQUAL(osCandidateBasename, osBasename))  | 
2872  |  | #else  | 
2873  | 0  |         if (osCandidateBasename.compare(osBasename) == 0)  | 
2874  | 0  | #endif  | 
2875  | 0  |         { | 
2876  | 0  |             if (EQUAL(osCandidateExtension, "dbf"))  | 
2877  | 0  |                 osDBFName =  | 
2878  | 0  |                     CPLFormFilenameSafe(osDirname, papszCandidates[i], nullptr);  | 
2879  | 0  |             else if (EQUAL(osCandidateExtension, "shp"))  | 
2880  | 0  |                 osSHPName =  | 
2881  | 0  |                     CPLFormFilenameSafe(osDirname, papszCandidates[i], nullptr);  | 
2882  | 0  |             else if (EQUAL(osCandidateExtension, "shx"))  | 
2883  | 0  |                 osSHXName =  | 
2884  | 0  |                     CPLFormFilenameSafe(osDirname, papszCandidates[i], nullptr);  | 
2885  | 0  |             else if (EQUAL(osCandidateExtension, "cpg"))  | 
2886  | 0  |                 osCPGName =  | 
2887  | 0  |                     CPLFormFilenameSafe(osDirname, papszCandidates[i], nullptr);  | 
2888  | 0  |         }  | 
2889  |  | 
  | 
2890  | 0  |         i++;  | 
2891  | 0  |     }  | 
2892  | 0  |     CSLDestroy(papszCandidates);  | 
2893  | 0  |     papszCandidates = nullptr;  | 
2894  |  | 
  | 
2895  | 0  |     if (m_hDBF != nullptr && osDBFName.empty())  | 
2896  | 0  |     { | 
2897  | 0  |         CPLError(CE_Failure, CPLE_AppDefined,  | 
2898  | 0  |                  "Cannot find the filename of the DBF file, but we managed to "  | 
2899  | 0  |                  "open it before !");  | 
2900  |  |         // Should not happen, really.  | 
2901  | 0  |         return OGRERR_FAILURE;  | 
2902  | 0  |     }  | 
2903  |  |  | 
2904  | 0  |     if (m_hSHP != nullptr && osSHPName.empty())  | 
2905  | 0  |     { | 
2906  | 0  |         CPLError(CE_Failure, CPLE_AppDefined,  | 
2907  | 0  |                  "Cannot find the filename of the SHP file, but we managed to "  | 
2908  | 0  |                  "open it before !");  | 
2909  |  |         // Should not happen, really.  | 
2910  | 0  |         return OGRERR_FAILURE;  | 
2911  | 0  |     }  | 
2912  |  |  | 
2913  | 0  |     if (m_hSHP != nullptr && osSHXName.empty())  | 
2914  | 0  |     { | 
2915  | 0  |         CPLError(CE_Failure, CPLE_AppDefined,  | 
2916  | 0  |                  "Cannot find the filename of the SHX file, but we managed to "  | 
2917  | 0  |                  "open it before !");  | 
2918  |  |         // Should not happen, really.  | 
2919  | 0  |         return OGRERR_FAILURE;  | 
2920  | 0  |     }  | 
2921  |  |  | 
2922  |  |     /* -------------------------------------------------------------------- */  | 
2923  |  |     /*      Cleanup any existing spatial index.  It will become             */  | 
2924  |  |     /*      meaningless when the fids change.                               */  | 
2925  |  |     /* -------------------------------------------------------------------- */  | 
2926  | 0  |     if (CheckForQIX() || CheckForSBN())  | 
2927  | 0  |         DropSpatialIndex();  | 
2928  |  |  | 
2929  |  |     /* -------------------------------------------------------------------- */  | 
2930  |  |     /*      Create a new dbf file, matching the old.                        */  | 
2931  |  |     /* -------------------------------------------------------------------- */  | 
2932  | 0  |     bool bMustReopenDBF = false;  | 
2933  | 0  |     CPLString oTempFileDBF;  | 
2934  | 0  |     const int nNewRecords =  | 
2935  | 0  |         m_nTotalShapeCount - static_cast<int>(anRecordsToDelete.size());  | 
2936  |  | 
  | 
2937  | 0  |     if (m_hDBF != nullptr && !anRecordsToDelete.empty())  | 
2938  | 0  |     { | 
2939  | 0  |         CPLDebug("Shape", "REPACK: repacking .dbf"); | 
2940  | 0  |         bMustReopenDBF = true;  | 
2941  |  | 
  | 
2942  | 0  |         oTempFileDBF = CPLFormFilenameSafe(osDirname, osBasename, nullptr);  | 
2943  | 0  |         oTempFileDBF += "_packed.dbf";  | 
2944  |  | 
  | 
2945  | 0  |         DBFHandle hNewDBF = DBFCloneEmpty(m_hDBF, oTempFileDBF);  | 
2946  | 0  |         if (hNewDBF == nullptr)  | 
2947  | 0  |         { | 
2948  | 0  |             CPLError(CE_Failure, CPLE_OpenFailed,  | 
2949  | 0  |                      "Failed to create temp file %s.", oTempFileDBF.c_str());  | 
2950  | 0  |             return OGRERR_FAILURE;  | 
2951  | 0  |         }  | 
2952  |  |  | 
2953  |  |         // Delete temporary .cpg file if existing.  | 
2954  | 0  |         if (!osCPGName.empty())  | 
2955  | 0  |         { | 
2956  | 0  |             CPLString oCPGTempFile =  | 
2957  | 0  |                 CPLFormFilenameSafe(osDirname, osBasename, nullptr);  | 
2958  | 0  |             oCPGTempFile += "_packed.cpg";  | 
2959  | 0  |             ForceDeleteFile(oCPGTempFile);  | 
2960  | 0  |         }  | 
2961  |  |  | 
2962  |  |         /* --------------------------------------------------------------------  | 
2963  |  |          */  | 
2964  |  |         /*      Copy over all records that are not deleted. */  | 
2965  |  |         /* --------------------------------------------------------------------  | 
2966  |  |          */  | 
2967  | 0  |         int iDestShape = 0;  | 
2968  | 0  |         size_t iNextDeletedShape = 0;  | 
2969  |  | 
  | 
2970  | 0  |         for (int iShape = 0; iShape < m_nTotalShapeCount && eErr == OGRERR_NONE;  | 
2971  | 0  |              iShape++)  | 
2972  | 0  |         { | 
2973  | 0  |             if (iNextDeletedShape < anRecordsToDelete.size() &&  | 
2974  | 0  |                 anRecordsToDelete[iNextDeletedShape] == iShape)  | 
2975  | 0  |             { | 
2976  | 0  |                 iNextDeletedShape++;  | 
2977  | 0  |             }  | 
2978  | 0  |             else  | 
2979  | 0  |             { | 
2980  | 0  |                 void *pTuple = const_cast<char *>(DBFReadTuple(m_hDBF, iShape));  | 
2981  | 0  |                 if (pTuple == nullptr ||  | 
2982  | 0  |                     !DBFWriteTuple(hNewDBF, iDestShape++, pTuple))  | 
2983  | 0  |                 { | 
2984  | 0  |                     CPLError(CE_Failure, CPLE_AppDefined,  | 
2985  | 0  |                              "Error writing record %d in .dbf", iShape);  | 
2986  | 0  |                     eErr = OGRERR_FAILURE;  | 
2987  | 0  |                 }  | 
2988  | 0  |             }  | 
2989  | 0  |         }  | 
2990  |  | 
  | 
2991  | 0  |         DBFClose(hNewDBF);  | 
2992  |  | 
  | 
2993  | 0  |         if (eErr != OGRERR_NONE)  | 
2994  | 0  |         { | 
2995  | 0  |             VSIUnlink(oTempFileDBF);  | 
2996  | 0  |             return eErr;  | 
2997  | 0  |         }  | 
2998  | 0  |     }  | 
2999  |  |  | 
3000  |  |     /* -------------------------------------------------------------------- */  | 
3001  |  |     /*      Now create a shapefile matching the old one.                    */  | 
3002  |  |     /* -------------------------------------------------------------------- */  | 
3003  | 0  |     bool bMustReopenSHP = m_hSHP != nullptr;  | 
3004  | 0  |     CPLString oTempFileSHP;  | 
3005  | 0  |     CPLString oTempFileSHX;  | 
3006  |  | 
  | 
3007  | 0  |     SHPInfo sSHPInfo;  | 
3008  | 0  |     memset(&sSHPInfo, 0, sizeof(sSHPInfo));  | 
3009  | 0  |     unsigned int *panRecOffsetNew = nullptr;  | 
3010  | 0  |     unsigned int *panRecSizeNew = nullptr;  | 
3011  |  |  | 
3012  |  |     // On Windows, use the pack-in-place approach, ie copy the content of  | 
3013  |  |     // the _packed files on top of the existing opened files. This avoids  | 
3014  |  |     // many issues with files being locked, at the expense of more I/O  | 
3015  | 0  |     const bool bPackInPlace =  | 
3016  | 0  |         CPLTestBool(CPLGetConfigOption("OGR_SHAPE_PACK_IN_PLACE", | 
3017  |  | #ifdef _WIN32  | 
3018  |  |                                        "YES"  | 
3019  |  | #else  | 
3020  | 0  |                                        "NO"  | 
3021  | 0  | #endif  | 
3022  | 0  |                                        ));  | 
3023  |  | 
  | 
3024  | 0  |     if (m_hSHP != nullptr)  | 
3025  | 0  |     { | 
3026  | 0  |         CPLDebug("Shape", "REPACK: repacking .shp + .shx"); | 
3027  |  | 
  | 
3028  | 0  |         oTempFileSHP = CPLFormFilenameSafe(osDirname, osBasename, nullptr);  | 
3029  | 0  |         oTempFileSHP += "_packed.shp";  | 
3030  | 0  |         oTempFileSHX = CPLFormFilenameSafe(osDirname, osBasename, nullptr);  | 
3031  | 0  |         oTempFileSHX += "_packed.shx";  | 
3032  |  | 
  | 
3033  | 0  |         SHPHandle hNewSHP = SHPCreate(oTempFileSHP, m_hSHP->nShapeType);  | 
3034  | 0  |         if (hNewSHP == nullptr)  | 
3035  | 0  |         { | 
3036  | 0  |             if (!oTempFileDBF.empty())  | 
3037  | 0  |                 VSIUnlink(oTempFileDBF);  | 
3038  | 0  |             return OGRERR_FAILURE;  | 
3039  | 0  |         }  | 
3040  |  |  | 
3041  |  |         /* --------------------------------------------------------------------  | 
3042  |  |          */  | 
3043  |  |         /*      Copy over all records that are not deleted. */  | 
3044  |  |         /* --------------------------------------------------------------------  | 
3045  |  |          */  | 
3046  | 0  |         size_t iNextDeletedShape = 0;  | 
3047  |  | 
  | 
3048  | 0  |         for (int iShape = 0; iShape < m_nTotalShapeCount && eErr == OGRERR_NONE;  | 
3049  | 0  |              iShape++)  | 
3050  | 0  |         { | 
3051  | 0  |             if (iNextDeletedShape < anRecordsToDelete.size() &&  | 
3052  | 0  |                 anRecordsToDelete[iNextDeletedShape] == iShape)  | 
3053  | 0  |             { | 
3054  | 0  |                 iNextDeletedShape++;  | 
3055  | 0  |             }  | 
3056  | 0  |             else  | 
3057  | 0  |             { | 
3058  | 0  |                 SHPObject *hObject = SHPReadObject(m_hSHP, iShape);  | 
3059  | 0  |                 if (hObject == nullptr ||  | 
3060  | 0  |                     SHPWriteObject(hNewSHP, -1, hObject) == -1)  | 
3061  | 0  |                 { | 
3062  | 0  |                     CPLError(CE_Failure, CPLE_AppDefined,  | 
3063  | 0  |                              "Error writing record %d in .shp", iShape);  | 
3064  | 0  |                     eErr = OGRERR_FAILURE;  | 
3065  | 0  |                 }  | 
3066  |  | 
  | 
3067  | 0  |                 if (hObject)  | 
3068  | 0  |                     SHPDestroyObject(hObject);  | 
3069  | 0  |             }  | 
3070  | 0  |         }  | 
3071  |  | 
  | 
3072  | 0  |         if (bPackInPlace)  | 
3073  | 0  |         { | 
3074  |  |             // Backup information of the updated shape context so as to  | 
3075  |  |             // restore it later in the current shape context  | 
3076  | 0  |             memcpy(&sSHPInfo, hNewSHP, sizeof(sSHPInfo));  | 
3077  |  |  | 
3078  |  |             // Use malloc like shapelib does  | 
3079  | 0  |             panRecOffsetNew = reinterpret_cast<unsigned int *>(  | 
3080  | 0  |                 malloc(sizeof(unsigned int) * hNewSHP->nMaxRecords));  | 
3081  | 0  |             panRecSizeNew = reinterpret_cast<unsigned int *>(  | 
3082  | 0  |                 malloc(sizeof(unsigned int) * hNewSHP->nMaxRecords));  | 
3083  | 0  |             if (panRecOffsetNew == nullptr || panRecSizeNew == nullptr)  | 
3084  | 0  |             { | 
3085  | 0  |                 CPLError(CE_Failure, CPLE_OutOfMemory,  | 
3086  | 0  |                          "Cannot allocate panRecOffsetNew/panRecSizeNew");  | 
3087  | 0  |                 eErr = OGRERR_FAILURE;  | 
3088  | 0  |             }  | 
3089  | 0  |             else  | 
3090  | 0  |             { | 
3091  | 0  |                 memcpy(panRecOffsetNew, hNewSHP->panRecOffset,  | 
3092  | 0  |                        sizeof(unsigned int) * hNewSHP->nRecords);  | 
3093  | 0  |                 memcpy(panRecSizeNew, hNewSHP->panRecSize,  | 
3094  | 0  |                        sizeof(unsigned int) * hNewSHP->nRecords);  | 
3095  | 0  |             }  | 
3096  | 0  |         }  | 
3097  |  | 
  | 
3098  | 0  |         SHPClose(hNewSHP);  | 
3099  |  | 
  | 
3100  | 0  |         if (eErr != OGRERR_NONE)  | 
3101  | 0  |         { | 
3102  | 0  |             VSIUnlink(oTempFileSHP);  | 
3103  | 0  |             VSIUnlink(oTempFileSHX);  | 
3104  | 0  |             if (!oTempFileDBF.empty())  | 
3105  | 0  |                 VSIUnlink(oTempFileDBF);  | 
3106  | 0  |             free(panRecOffsetNew);  | 
3107  | 0  |             free(panRecSizeNew);  | 
3108  | 0  |             return eErr;  | 
3109  | 0  |         }  | 
3110  | 0  |     }  | 
3111  |  |  | 
3112  |  |     // We could also use pack in place for Unix but this involves extra I/O  | 
3113  |  |     // w.r.t to the delete and rename approach  | 
3114  |  |  | 
3115  | 0  |     if (bPackInPlace)  | 
3116  | 0  |     { | 
3117  | 0  |         if (m_hDBF != nullptr && !oTempFileDBF.empty())  | 
3118  | 0  |         { | 
3119  | 0  |             if (!OGRShapeDataSource::CopyInPlace(VSI_SHP_GetVSIL(m_hDBF->fp),  | 
3120  | 0  |                                                  oTempFileDBF))  | 
3121  | 0  |             { | 
3122  | 0  |                 CPLError(  | 
3123  | 0  |                     CE_Failure, CPLE_FileIO,  | 
3124  | 0  |                     "An error occurred while copying the content of %s on top "  | 
3125  | 0  |                     "of %s. "  | 
3126  | 0  |                     "The non corrupted version is in the _packed.dbf, "  | 
3127  | 0  |                     "_packed.shp and _packed.shx files that you should rename "  | 
3128  | 0  |                     "on top of the main ones.",  | 
3129  | 0  |                     oTempFileDBF.c_str(), VSI_SHP_GetFilename(m_hDBF->fp));  | 
3130  | 0  |                 free(panRecOffsetNew);  | 
3131  | 0  |                 free(panRecSizeNew);  | 
3132  |  | 
  | 
3133  | 0  |                 DBFClose(m_hDBF);  | 
3134  | 0  |                 m_hDBF = nullptr;  | 
3135  | 0  |                 if (m_hSHP != nullptr)  | 
3136  | 0  |                 { | 
3137  | 0  |                     SHPClose(m_hSHP);  | 
3138  | 0  |                     m_hSHP = nullptr;  | 
3139  | 0  |                 }  | 
3140  |  | 
  | 
3141  | 0  |                 return OGRERR_FAILURE;  | 
3142  | 0  |             }  | 
3143  |  |  | 
3144  |  |             // Refresh current handle  | 
3145  | 0  |             m_hDBF->nRecords = nNewRecords;  | 
3146  | 0  |         }  | 
3147  |  |  | 
3148  | 0  |         if (m_hSHP != nullptr && !oTempFileSHP.empty())  | 
3149  | 0  |         { | 
3150  | 0  |             if (!OGRShapeDataSource::CopyInPlace(VSI_SHP_GetVSIL(m_hSHP->fpSHP),  | 
3151  | 0  |                                                  oTempFileSHP))  | 
3152  | 0  |             { | 
3153  | 0  |                 CPLError(  | 
3154  | 0  |                     CE_Failure, CPLE_FileIO,  | 
3155  | 0  |                     "An error occurred while copying the content of %s on top "  | 
3156  | 0  |                     "of %s. "  | 
3157  | 0  |                     "The non corrupted version is in the _packed.dbf, "  | 
3158  | 0  |                     "_packed.shp and _packed.shx files that you should rename "  | 
3159  | 0  |                     "on top of the main ones.",  | 
3160  | 0  |                     oTempFileSHP.c_str(), VSI_SHP_GetFilename(m_hSHP->fpSHP));  | 
3161  | 0  |                 free(panRecOffsetNew);  | 
3162  | 0  |                 free(panRecSizeNew);  | 
3163  |  | 
  | 
3164  | 0  |                 if (m_hDBF != nullptr)  | 
3165  | 0  |                 { | 
3166  | 0  |                     DBFClose(m_hDBF);  | 
3167  | 0  |                     m_hDBF = nullptr;  | 
3168  | 0  |                 }  | 
3169  | 0  |                 SHPClose(m_hSHP);  | 
3170  | 0  |                 m_hSHP = nullptr;  | 
3171  |  | 
  | 
3172  | 0  |                 return OGRERR_FAILURE;  | 
3173  | 0  |             }  | 
3174  | 0  |             if (!OGRShapeDataSource::CopyInPlace(VSI_SHP_GetVSIL(m_hSHP->fpSHX),  | 
3175  | 0  |                                                  oTempFileSHX))  | 
3176  | 0  |             { | 
3177  | 0  |                 CPLError(  | 
3178  | 0  |                     CE_Failure, CPLE_FileIO,  | 
3179  | 0  |                     "An error occurred while copying the content of %s on top "  | 
3180  | 0  |                     "of %s. "  | 
3181  | 0  |                     "The non corrupted version is in the _packed.dbf, "  | 
3182  | 0  |                     "_packed.shp and _packed.shx files that you should rename "  | 
3183  | 0  |                     "on top of the main ones.",  | 
3184  | 0  |                     oTempFileSHX.c_str(), VSI_SHP_GetFilename(m_hSHP->fpSHX));  | 
3185  | 0  |                 free(panRecOffsetNew);  | 
3186  | 0  |                 free(panRecSizeNew);  | 
3187  |  | 
  | 
3188  | 0  |                 if (m_hDBF != nullptr)  | 
3189  | 0  |                 { | 
3190  | 0  |                     DBFClose(m_hDBF);  | 
3191  | 0  |                     m_hDBF = nullptr;  | 
3192  | 0  |                 }  | 
3193  | 0  |                 SHPClose(m_hSHP);  | 
3194  | 0  |                 m_hSHP = nullptr;  | 
3195  |  | 
  | 
3196  | 0  |                 return OGRERR_FAILURE;  | 
3197  | 0  |             }  | 
3198  |  |  | 
3199  |  |             // Refresh current handle  | 
3200  | 0  |             m_hSHP->nRecords = sSHPInfo.nRecords;  | 
3201  | 0  |             m_hSHP->nMaxRecords = sSHPInfo.nMaxRecords;  | 
3202  | 0  |             m_hSHP->nFileSize = sSHPInfo.nFileSize;  | 
3203  | 0  |             CPLAssert(sizeof(sSHPInfo.adBoundsMin) == 4 * sizeof(double));  | 
3204  | 0  |             memcpy(m_hSHP->adBoundsMin, sSHPInfo.adBoundsMin,  | 
3205  | 0  |                    sizeof(sSHPInfo.adBoundsMin));  | 
3206  | 0  |             memcpy(m_hSHP->adBoundsMax, sSHPInfo.adBoundsMax,  | 
3207  | 0  |                    sizeof(sSHPInfo.adBoundsMax));  | 
3208  | 0  |             free(m_hSHP->panRecOffset);  | 
3209  | 0  |             free(m_hSHP->panRecSize);  | 
3210  | 0  |             m_hSHP->panRecOffset = panRecOffsetNew;  | 
3211  | 0  |             m_hSHP->panRecSize = panRecSizeNew;  | 
3212  | 0  |         }  | 
3213  | 0  |         else  | 
3214  | 0  |         { | 
3215  |  |             // The free() are not really necessary but CSA doesn't realize it  | 
3216  | 0  |             free(panRecOffsetNew);  | 
3217  | 0  |             free(panRecSizeNew);  | 
3218  | 0  |         }  | 
3219  |  |  | 
3220  |  |         // Now that everything is successful, we can delete the temp files  | 
3221  | 0  |         if (!oTempFileDBF.empty())  | 
3222  | 0  |         { | 
3223  | 0  |             ForceDeleteFile(oTempFileDBF);  | 
3224  | 0  |         }  | 
3225  | 0  |         if (!oTempFileSHP.empty())  | 
3226  | 0  |         { | 
3227  | 0  |             ForceDeleteFile(oTempFileSHP);  | 
3228  | 0  |             ForceDeleteFile(oTempFileSHX);  | 
3229  | 0  |         }  | 
3230  | 0  |     }  | 
3231  | 0  |     else  | 
3232  | 0  |     { | 
3233  |  |         /* --------------------------------------------------------------------  | 
3234  |  |          */  | 
3235  |  |         /*      Cleanup the old .dbf, .shp, .shx and rename the new ones. */  | 
3236  |  |         /* --------------------------------------------------------------------  | 
3237  |  |          */  | 
3238  | 0  |         if (!oTempFileDBF.empty())  | 
3239  | 0  |         { | 
3240  | 0  |             DBFClose(m_hDBF);  | 
3241  | 0  |             m_hDBF = nullptr;  | 
3242  |  | 
  | 
3243  | 0  |             if (VSIUnlink(osDBFName) != 0)  | 
3244  | 0  |             { | 
3245  | 0  |                 CPLError(CE_Failure, CPLE_FileIO,  | 
3246  | 0  |                          "Failed to delete old DBF file: %s",  | 
3247  | 0  |                          VSIStrerror(errno));  | 
3248  |  | 
  | 
3249  | 0  |                 m_hDBF =  | 
3250  | 0  |                     m_poDS->DS_DBFOpen(osDBFName, m_bUpdateAccess ? "r+" : "r");  | 
3251  |  | 
  | 
3252  | 0  |                 VSIUnlink(oTempFileDBF);  | 
3253  |  | 
  | 
3254  | 0  |                 return OGRERR_FAILURE;  | 
3255  | 0  |             }  | 
3256  |  |  | 
3257  | 0  |             if (VSIRename(oTempFileDBF, osDBFName) != 0)  | 
3258  | 0  |             { | 
3259  | 0  |                 CPLError(CE_Failure, CPLE_FileIO,  | 
3260  | 0  |                          "Can not rename new DBF file: %s", VSIStrerror(errno));  | 
3261  | 0  |                 return OGRERR_FAILURE;  | 
3262  | 0  |             }  | 
3263  |  |  | 
3264  | 0  |             CheckFileDeletion(oTempFileDBF);  | 
3265  | 0  |         }  | 
3266  |  |  | 
3267  | 0  |         if (!oTempFileSHP.empty())  | 
3268  | 0  |         { | 
3269  | 0  |             SHPClose(m_hSHP);  | 
3270  | 0  |             m_hSHP = nullptr;  | 
3271  |  | 
  | 
3272  | 0  |             if (VSIUnlink(osSHPName) != 0)  | 
3273  | 0  |             { | 
3274  | 0  |                 CPLError(CE_Failure, CPLE_FileIO,  | 
3275  | 0  |                          "Can not delete old SHP file: %s", VSIStrerror(errno));  | 
3276  | 0  |                 return OGRERR_FAILURE;  | 
3277  | 0  |             }  | 
3278  |  |  | 
3279  | 0  |             if (VSIUnlink(osSHXName) != 0)  | 
3280  | 0  |             { | 
3281  | 0  |                 CPLError(CE_Failure, CPLE_FileIO,  | 
3282  | 0  |                          "Can not delete old SHX file: %s", VSIStrerror(errno));  | 
3283  | 0  |                 return OGRERR_FAILURE;  | 
3284  | 0  |             }  | 
3285  |  |  | 
3286  | 0  |             if (VSIRename(oTempFileSHP, osSHPName) != 0)  | 
3287  | 0  |             { | 
3288  | 0  |                 CPLError(CE_Failure, CPLE_FileIO,  | 
3289  | 0  |                          "Can not rename new SHP file: %s", VSIStrerror(errno));  | 
3290  | 0  |                 return OGRERR_FAILURE;  | 
3291  | 0  |             }  | 
3292  |  |  | 
3293  | 0  |             if (VSIRename(oTempFileSHX, osSHXName) != 0)  | 
3294  | 0  |             { | 
3295  | 0  |                 CPLError(CE_Failure, CPLE_FileIO,  | 
3296  | 0  |                          "Can not rename new SHX file: %s", VSIStrerror(errno));  | 
3297  | 0  |                 return OGRERR_FAILURE;  | 
3298  | 0  |             }  | 
3299  |  |  | 
3300  | 0  |             CheckFileDeletion(oTempFileSHP);  | 
3301  | 0  |             CheckFileDeletion(oTempFileSHX);  | 
3302  | 0  |         }  | 
3303  |  |  | 
3304  |  |         /* --------------------------------------------------------------------  | 
3305  |  |          */  | 
3306  |  |         /*      Reopen the shapefile */  | 
3307  |  |         /*                                                                      */  | 
3308  |  |         /* We do not need to reimplement OGRShapeDataSource::OpenFile() here */  | 
3309  |  |         /* with the fully featured error checking. */  | 
3310  |  |         /* If all operations above succeeded, then all necessary files are */  | 
3311  |  |         /* in the right place and accessible. */  | 
3312  |  |         /* --------------------------------------------------------------------  | 
3313  |  |          */  | 
3314  |  |  | 
3315  | 0  |         const char *const pszAccess = m_bUpdateAccess ? "r+" : "r";  | 
3316  |  | 
  | 
3317  | 0  |         if (bMustReopenSHP)  | 
3318  | 0  |             m_hSHP = m_poDS->DS_SHPOpen(osSHPName, pszAccess);  | 
3319  | 0  |         if (bMustReopenDBF)  | 
3320  | 0  |             m_hDBF = m_poDS->DS_DBFOpen(osDBFName, pszAccess);  | 
3321  |  | 
  | 
3322  | 0  |         if ((bMustReopenSHP && nullptr == m_hSHP) ||  | 
3323  | 0  |             (bMustReopenDBF && nullptr == m_hDBF))  | 
3324  | 0  |             return OGRERR_FAILURE;  | 
3325  | 0  |     }  | 
3326  |  |  | 
3327  |  |     /* -------------------------------------------------------------------- */  | 
3328  |  |     /*      Update total shape count.                                       */  | 
3329  |  |     /* -------------------------------------------------------------------- */  | 
3330  | 0  |     if (m_hDBF != nullptr)  | 
3331  | 0  |         m_nTotalShapeCount = m_hDBF->nRecords;  | 
3332  | 0  |     m_bSHPNeedsRepack = false;  | 
3333  | 0  |     m_eNeedRepack = NO;  | 
3334  |  | 
  | 
3335  | 0  |     return OGRERR_NONE;  | 
3336  | 0  | }  | 
3337  |  |  | 
3338  |  | /************************************************************************/  | 
3339  |  | /*                               ResizeDBF()                            */  | 
3340  |  | /*                                                                      */  | 
3341  |  | /*      Autoshrink columns of the DBF file to their minimum             */  | 
3342  |  | /*      size, according to the existing data.                           */  | 
3343  |  | /************************************************************************/  | 
3344  |  |  | 
3345  |  | OGRErr OGRShapeLayer::ResizeDBF()  | 
3346  |  |  | 
3347  | 0  | { | 
3348  | 0  |     if (!StartUpdate("ResizeDBF")) | 
3349  | 0  |         return OGRERR_FAILURE;  | 
3350  |  |  | 
3351  | 0  |     if (m_hDBF == nullptr)  | 
3352  | 0  |     { | 
3353  | 0  |         CPLError(  | 
3354  | 0  |             CE_Failure, CPLE_NotSupported,  | 
3355  | 0  |             "Attempt to RESIZE a shapefile with no .dbf file not supported.");  | 
3356  | 0  |         return OGRERR_FAILURE;  | 
3357  | 0  |     }  | 
3358  |  |  | 
3359  |  |     /* Look which columns must be examined */  | 
3360  | 0  |     int *panColMap = static_cast<int *>(  | 
3361  | 0  |         CPLMalloc(m_poFeatureDefn->GetFieldCount() * sizeof(int)));  | 
3362  | 0  |     int *panBestWidth = static_cast<int *>(  | 
3363  | 0  |         CPLMalloc(m_poFeatureDefn->GetFieldCount() * sizeof(int)));  | 
3364  | 0  |     int nStringCols = 0;  | 
3365  | 0  |     for (int i = 0; i < m_poFeatureDefn->GetFieldCount(); i++)  | 
3366  | 0  |     { | 
3367  | 0  |         if (m_poFeatureDefn->GetFieldDefn(i)->GetType() == OFTString ||  | 
3368  | 0  |             m_poFeatureDefn->GetFieldDefn(i)->GetType() == OFTInteger ||  | 
3369  | 0  |             m_poFeatureDefn->GetFieldDefn(i)->GetType() == OFTInteger64)  | 
3370  | 0  |         { | 
3371  | 0  |             panColMap[nStringCols] = i;  | 
3372  | 0  |             panBestWidth[nStringCols] = 1;  | 
3373  | 0  |             nStringCols++;  | 
3374  | 0  |         }  | 
3375  | 0  |     }  | 
3376  |  | 
  | 
3377  | 0  |     if (nStringCols == 0)  | 
3378  | 0  |     { | 
3379  |  |         // Nothing to do.  | 
3380  | 0  |         CPLFree(panColMap);  | 
3381  | 0  |         CPLFree(panBestWidth);  | 
3382  | 0  |         return OGRERR_NONE;  | 
3383  | 0  |     }  | 
3384  |  |  | 
3385  | 0  |     CPLDebug("SHAPE", "Computing optimal column size..."); | 
3386  |  | 
  | 
3387  | 0  |     bool bAlreadyWarned = false;  | 
3388  | 0  |     for (int i = 0; i < m_hDBF->nRecords; i++)  | 
3389  | 0  |     { | 
3390  | 0  |         if (!DBFIsRecordDeleted(m_hDBF, i))  | 
3391  | 0  |         { | 
3392  | 0  |             for (int j = 0; j < nStringCols; j++)  | 
3393  | 0  |             { | 
3394  | 0  |                 if (DBFIsAttributeNULL(m_hDBF, i, panColMap[j]))  | 
3395  | 0  |                     continue;  | 
3396  |  |  | 
3397  | 0  |                 const char *pszVal =  | 
3398  | 0  |                     DBFReadStringAttribute(m_hDBF, i, panColMap[j]);  | 
3399  | 0  |                 const int nLen = static_cast<int>(strlen(pszVal));  | 
3400  | 0  |                 if (nLen > panBestWidth[j])  | 
3401  | 0  |                     panBestWidth[j] = nLen;  | 
3402  | 0  |             }  | 
3403  | 0  |         }  | 
3404  | 0  |         else if (!bAlreadyWarned)  | 
3405  | 0  |         { | 
3406  | 0  |             bAlreadyWarned = true;  | 
3407  | 0  |             CPLDebug(  | 
3408  | 0  |                 "SHAPE",  | 
3409  | 0  |                 "DBF file would also need a REPACK due to deleted records");  | 
3410  | 0  |         }  | 
3411  | 0  |     }  | 
3412  |  | 
  | 
3413  | 0  |     for (int j = 0; j < nStringCols; j++)  | 
3414  | 0  |     { | 
3415  | 0  |         const int iField = panColMap[j];  | 
3416  | 0  |         OGRFieldDefn *const poFieldDefn = m_poFeatureDefn->GetFieldDefn(iField);  | 
3417  |  | 
  | 
3418  | 0  |         const char chNativeType = DBFGetNativeFieldType(m_hDBF, iField);  | 
3419  | 0  |         char szFieldName[XBASE_FLDNAME_LEN_READ + 1] = {}; | 
3420  | 0  |         int nOriWidth = 0;  | 
3421  | 0  |         int nPrecision = 0;  | 
3422  | 0  |         DBFGetFieldInfo(m_hDBF, iField, szFieldName, &nOriWidth, &nPrecision);  | 
3423  |  | 
  | 
3424  | 0  |         if (panBestWidth[j] < nOriWidth)  | 
3425  | 0  |         { | 
3426  | 0  |             CPLDebug("SHAPE", | 
3427  | 0  |                      "Shrinking field %d (%s) from %d to %d characters", iField,  | 
3428  | 0  |                      poFieldDefn->GetNameRef(), nOriWidth, panBestWidth[j]);  | 
3429  |  | 
  | 
3430  | 0  |             if (!DBFAlterFieldDefn(m_hDBF, iField, szFieldName, chNativeType,  | 
3431  | 0  |                                    panBestWidth[j], nPrecision))  | 
3432  | 0  |             { | 
3433  | 0  |                 CPLError(  | 
3434  | 0  |                     CE_Failure, CPLE_AppDefined,  | 
3435  | 0  |                     "Shrinking field %d (%s) from %d to %d characters failed",  | 
3436  | 0  |                     iField, poFieldDefn->GetNameRef(), nOriWidth,  | 
3437  | 0  |                     panBestWidth[j]);  | 
3438  |  | 
  | 
3439  | 0  |                 CPLFree(panColMap);  | 
3440  | 0  |                 CPLFree(panBestWidth);  | 
3441  |  | 
  | 
3442  | 0  |                 return OGRERR_FAILURE;  | 
3443  | 0  |             }  | 
3444  | 0  |             else  | 
3445  | 0  |             { | 
3446  | 0  |                 whileUnsealing(poFieldDefn)->SetWidth(panBestWidth[j]);  | 
3447  | 0  |             }  | 
3448  | 0  |         }  | 
3449  | 0  |     }  | 
3450  |  |  | 
3451  | 0  |     TruncateDBF();  | 
3452  |  | 
  | 
3453  | 0  |     CPLFree(panColMap);  | 
3454  | 0  |     CPLFree(panBestWidth);  | 
3455  |  | 
  | 
3456  | 0  |     return OGRERR_NONE;  | 
3457  | 0  | }  | 
3458  |  |  | 
3459  |  | /************************************************************************/  | 
3460  |  | /*                          TruncateDBF()                               */  | 
3461  |  | /************************************************************************/  | 
3462  |  |  | 
3463  |  | void OGRShapeLayer::TruncateDBF()  | 
3464  | 0  | { | 
3465  | 0  |     if (m_hDBF == nullptr)  | 
3466  | 0  |         return;  | 
3467  |  |  | 
3468  | 0  |     m_hDBF->sHooks.FSeek(m_hDBF->fp, 0, SEEK_END);  | 
3469  | 0  |     vsi_l_offset nOldSize = m_hDBF->sHooks.FTell(m_hDBF->fp);  | 
3470  | 0  |     vsi_l_offset nNewSize =  | 
3471  | 0  |         m_hDBF->nRecordLength * static_cast<SAOffset>(m_hDBF->nRecords) +  | 
3472  | 0  |         m_hDBF->nHeaderLength;  | 
3473  | 0  |     if (m_hDBF->bWriteEndOfFileChar)  | 
3474  | 0  |         nNewSize++;  | 
3475  | 0  |     if (nNewSize < nOldSize)  | 
3476  | 0  |     { | 
3477  | 0  |         CPLDebug("SHAPE", | 
3478  | 0  |                  "Truncating DBF file from " CPL_FRMT_GUIB " to " CPL_FRMT_GUIB  | 
3479  | 0  |                  " bytes",  | 
3480  | 0  |                  nOldSize, nNewSize);  | 
3481  | 0  |         VSIFTruncateL(VSI_SHP_GetVSIL(m_hDBF->fp), nNewSize);  | 
3482  | 0  |     }  | 
3483  | 0  |     m_hDBF->sHooks.FSeek(m_hDBF->fp, 0, SEEK_SET);  | 
3484  | 0  | }  | 
3485  |  |  | 
3486  |  | /************************************************************************/  | 
3487  |  | /*                        RecomputeExtent()                             */  | 
3488  |  | /*                                                                      */  | 
3489  |  | /*      Force recomputation of the extent of the .SHP file              */  | 
3490  |  | /************************************************************************/  | 
3491  |  |  | 
3492  |  | OGRErr OGRShapeLayer::RecomputeExtent()  | 
3493  | 0  | { | 
3494  | 0  |     if (!StartUpdate("RecomputeExtent")) | 
3495  | 0  |         return OGRERR_FAILURE;  | 
3496  |  |  | 
3497  | 0  |     if (m_hSHP == nullptr)  | 
3498  | 0  |     { | 
3499  | 0  |         CPLError(CE_Failure, CPLE_AppDefined,  | 
3500  | 0  |                  "The RECOMPUTE EXTENT operation is not permitted on a layer "  | 
3501  | 0  |                  "without .SHP file.");  | 
3502  | 0  |         return OGRERR_FAILURE;  | 
3503  | 0  |     }  | 
3504  |  |  | 
3505  | 0  |     double adBoundsMin[4] = {0.0, 0.0, 0.0, 0.0}; | 
3506  | 0  |     double adBoundsMax[4] = {0.0, 0.0, 0.0, 0.0}; | 
3507  |  | 
  | 
3508  | 0  |     bool bHasBeenInit = false;  | 
3509  |  | 
  | 
3510  | 0  |     for (int iShape = 0; iShape < m_nTotalShapeCount; iShape++)  | 
3511  | 0  |     { | 
3512  | 0  |         if (m_hDBF == nullptr || !DBFIsRecordDeleted(m_hDBF, iShape))  | 
3513  | 0  |         { | 
3514  | 0  |             SHPObject *psObject = SHPReadObject(m_hSHP, iShape);  | 
3515  | 0  |             if (psObject != nullptr && psObject->nSHPType != SHPT_NULL &&  | 
3516  | 0  |                 psObject->nVertices != 0)  | 
3517  | 0  |             { | 
3518  | 0  |                 if (!bHasBeenInit)  | 
3519  | 0  |                 { | 
3520  | 0  |                     bHasBeenInit = true;  | 
3521  | 0  |                     adBoundsMin[0] = psObject->padfX[0];  | 
3522  | 0  |                     adBoundsMax[0] = psObject->padfX[0];  | 
3523  | 0  |                     adBoundsMin[1] = psObject->padfY[0];  | 
3524  | 0  |                     adBoundsMax[1] = psObject->padfY[0];  | 
3525  | 0  |                     if (psObject->padfZ)  | 
3526  | 0  |                     { | 
3527  | 0  |                         adBoundsMin[2] = psObject->padfZ[0];  | 
3528  | 0  |                         adBoundsMax[2] = psObject->padfZ[0];  | 
3529  | 0  |                     }  | 
3530  | 0  |                     if (psObject->padfM)  | 
3531  | 0  |                     { | 
3532  | 0  |                         adBoundsMin[3] = psObject->padfM[0];  | 
3533  | 0  |                         adBoundsMax[3] = psObject->padfM[0];  | 
3534  | 0  |                     }  | 
3535  | 0  |                 }  | 
3536  |  | 
  | 
3537  | 0  |                 for (int i = 0; i < psObject->nVertices; i++)  | 
3538  | 0  |                 { | 
3539  | 0  |                     adBoundsMin[0] =  | 
3540  | 0  |                         std::min(adBoundsMin[0], psObject->padfX[i]);  | 
3541  | 0  |                     adBoundsMin[1] =  | 
3542  | 0  |                         std::min(adBoundsMin[1], psObject->padfY[i]);  | 
3543  | 0  |                     adBoundsMax[0] =  | 
3544  | 0  |                         std::max(adBoundsMax[0], psObject->padfX[i]);  | 
3545  | 0  |                     adBoundsMax[1] =  | 
3546  | 0  |                         std::max(adBoundsMax[1], psObject->padfY[i]);  | 
3547  | 0  |                     if (psObject->padfZ)  | 
3548  | 0  |                     { | 
3549  | 0  |                         adBoundsMin[2] =  | 
3550  | 0  |                             std::min(adBoundsMin[2], psObject->padfZ[i]);  | 
3551  | 0  |                         adBoundsMax[2] =  | 
3552  | 0  |                             std::max(adBoundsMax[2], psObject->padfZ[i]);  | 
3553  | 0  |                     }  | 
3554  | 0  |                     if (psObject->padfM)  | 
3555  | 0  |                     { | 
3556  | 0  |                         adBoundsMax[3] =  | 
3557  | 0  |                             std::max(adBoundsMax[3], psObject->padfM[i]);  | 
3558  | 0  |                         adBoundsMin[3] =  | 
3559  | 0  |                             std::min(adBoundsMin[3], psObject->padfM[i]);  | 
3560  | 0  |                     }  | 
3561  | 0  |                 }  | 
3562  | 0  |             }  | 
3563  | 0  |             SHPDestroyObject(psObject);  | 
3564  | 0  |         }  | 
3565  | 0  |     }  | 
3566  |  | 
  | 
3567  | 0  |     if (memcmp(m_hSHP->adBoundsMin, adBoundsMin, 4 * sizeof(double)) != 0 ||  | 
3568  | 0  |         memcmp(m_hSHP->adBoundsMax, adBoundsMax, 4 * sizeof(double)) != 0)  | 
3569  | 0  |     { | 
3570  | 0  |         m_bHeaderDirty = true;  | 
3571  | 0  |         m_hSHP->bUpdated = TRUE;  | 
3572  | 0  |         memcpy(m_hSHP->adBoundsMin, adBoundsMin, 4 * sizeof(double));  | 
3573  | 0  |         memcpy(m_hSHP->adBoundsMax, adBoundsMax, 4 * sizeof(double));  | 
3574  | 0  |     }  | 
3575  |  | 
  | 
3576  | 0  |     return OGRERR_NONE;  | 
3577  | 0  | }  | 
3578  |  |  | 
3579  |  | /************************************************************************/  | 
3580  |  | /*                              TouchLayer()                            */  | 
3581  |  | /************************************************************************/  | 
3582  |  |  | 
3583  |  | bool OGRShapeLayer::TouchLayer()  | 
3584  | 0  | { | 
3585  | 0  |     m_poDS->SetLastUsedLayer(this);  | 
3586  |  | 
  | 
3587  | 0  |     if (m_eFileDescriptorsState == FD_OPENED)  | 
3588  | 0  |         return true;  | 
3589  | 0  |     if (m_eFileDescriptorsState == FD_CANNOT_REOPEN)  | 
3590  | 0  |         return false;  | 
3591  |  |  | 
3592  | 0  |     return ReopenFileDescriptors();  | 
3593  | 0  | }  | 
3594  |  |  | 
3595  |  | /************************************************************************/  | 
3596  |  | /*                        ReopenFileDescriptors()                       */  | 
3597  |  | /************************************************************************/  | 
3598  |  |  | 
3599  |  | bool OGRShapeLayer::ReopenFileDescriptors()  | 
3600  | 0  | { | 
3601  | 0  |     CPLDebug("SHAPE", "ReopenFileDescriptors(%s)", m_osFullName.c_str()); | 
3602  |  | 
  | 
3603  | 0  |     const bool bRealUpdateAccess =  | 
3604  | 0  |         m_bUpdateAccess &&  | 
3605  | 0  |         (!m_poDS->IsZip() || !m_poDS->GetTemporaryUnzipDir().empty());  | 
3606  |  | 
  | 
3607  | 0  |     if (m_bHSHPWasNonNULL)  | 
3608  | 0  |     { | 
3609  | 0  |         m_hSHP = m_poDS->DS_SHPOpen(m_osFullName.c_str(),  | 
3610  | 0  |                                     bRealUpdateAccess ? "r+" : "r");  | 
3611  |  | 
  | 
3612  | 0  |         if (m_hSHP == nullptr)  | 
3613  | 0  |         { | 
3614  | 0  |             m_eFileDescriptorsState = FD_CANNOT_REOPEN;  | 
3615  | 0  |             return false;  | 
3616  | 0  |         }  | 
3617  | 0  |     }  | 
3618  |  |  | 
3619  | 0  |     if (m_bHDBFWasNonNULL)  | 
3620  | 0  |     { | 
3621  | 0  |         m_hDBF = m_poDS->DS_DBFOpen(m_osFullName.c_str(),  | 
3622  | 0  |                                     bRealUpdateAccess ? "r+" : "r");  | 
3623  |  | 
  | 
3624  | 0  |         if (m_hDBF == nullptr)  | 
3625  | 0  |         { | 
3626  | 0  |             CPLError(  | 
3627  | 0  |                 CE_Failure, CPLE_OpenFailed, "Cannot reopen %s",  | 
3628  | 0  |                 CPLResetExtensionSafe(m_osFullName.c_str(), "dbf").c_str());  | 
3629  | 0  |             m_eFileDescriptorsState = FD_CANNOT_REOPEN;  | 
3630  | 0  |             return false;  | 
3631  | 0  |         }  | 
3632  | 0  |     }  | 
3633  |  |  | 
3634  | 0  |     m_eFileDescriptorsState = FD_OPENED;  | 
3635  |  | 
  | 
3636  | 0  |     return true;  | 
3637  | 0  | }  | 
3638  |  |  | 
3639  |  | /************************************************************************/  | 
3640  |  | /*                        CloseUnderlyingLayer()                        */  | 
3641  |  | /************************************************************************/  | 
3642  |  |  | 
3643  |  | void OGRShapeLayer::CloseUnderlyingLayer()  | 
3644  | 0  | { | 
3645  | 0  |     CPLDebug("SHAPE", "CloseUnderlyingLayer(%s)", m_osFullName.c_str()); | 
3646  |  | 
  | 
3647  | 0  |     if (m_hDBF != nullptr)  | 
3648  | 0  |         DBFClose(m_hDBF);  | 
3649  | 0  |     m_hDBF = nullptr;  | 
3650  |  | 
  | 
3651  | 0  |     if (m_hSHP != nullptr)  | 
3652  | 0  |         SHPClose(m_hSHP);  | 
3653  | 0  |     m_hSHP = nullptr;  | 
3654  |  |  | 
3655  |  |     // We close QIX and reset the check flag, so that CheckForQIX()  | 
3656  |  |     // will retry opening it if necessary when the layer is active again.  | 
3657  | 0  |     if (m_hQIX != nullptr)  | 
3658  | 0  |         SHPCloseDiskTree(m_hQIX);  | 
3659  | 0  |     m_hQIX = nullptr;  | 
3660  | 0  |     m_bCheckedForQIX = false;  | 
3661  |  | 
  | 
3662  | 0  |     if (m_hSBN != nullptr)  | 
3663  | 0  |         SBNCloseDiskTree(m_hSBN);  | 
3664  | 0  |     m_hSBN = nullptr;  | 
3665  | 0  |     m_bCheckedForSBN = false;  | 
3666  |  | 
  | 
3667  | 0  |     m_eFileDescriptorsState = FD_CLOSED;  | 
3668  | 0  | }  | 
3669  |  |  | 
3670  |  | /************************************************************************/  | 
3671  |  | /*                            AddToFileList()                           */  | 
3672  |  | /************************************************************************/  | 
3673  |  |  | 
3674  |  | void OGRShapeLayer::AddToFileList(CPLStringList &oFileList)  | 
3675  | 0  | { | 
3676  | 0  |     if (!TouchLayer())  | 
3677  | 0  |         return;  | 
3678  |  |  | 
3679  | 0  |     if (m_hSHP)  | 
3680  | 0  |     { | 
3681  | 0  |         const char *pszSHPFilename = VSI_SHP_GetFilename(m_hSHP->fpSHP);  | 
3682  | 0  |         oFileList.AddStringDirectly(VSIGetCanonicalFilename(pszSHPFilename));  | 
3683  | 0  |         const std::string osSHPExt = CPLGetExtensionSafe(pszSHPFilename);  | 
3684  | 0  |         const std::string osSHXFilename = CPLResetExtensionSafe(  | 
3685  | 0  |             pszSHPFilename, (osSHPExt[0] == 's') ? "shx" : "SHX");  | 
3686  | 0  |         oFileList.AddStringDirectly(  | 
3687  | 0  |             VSIGetCanonicalFilename(osSHXFilename.c_str()));  | 
3688  | 0  |     }  | 
3689  |  | 
  | 
3690  | 0  |     if (m_hDBF)  | 
3691  | 0  |     { | 
3692  | 0  |         const char *pszDBFFilename = VSI_SHP_GetFilename(m_hDBF->fp);  | 
3693  | 0  |         oFileList.AddStringDirectly(VSIGetCanonicalFilename(pszDBFFilename));  | 
3694  | 0  |         if (m_hDBF->pszCodePage != nullptr && m_hDBF->iLanguageDriver == 0)  | 
3695  | 0  |         { | 
3696  | 0  |             const std::string osDBFExt = CPLGetExtensionSafe(pszDBFFilename);  | 
3697  | 0  |             const std::string osCPGFilename = CPLResetExtensionSafe(  | 
3698  | 0  |                 pszDBFFilename, (osDBFExt[0] == 'd') ? "cpg" : "CPG");  | 
3699  | 0  |             oFileList.AddStringDirectly(  | 
3700  | 0  |                 VSIGetCanonicalFilename(osCPGFilename.c_str()));  | 
3701  | 0  |         }  | 
3702  | 0  |     }  | 
3703  |  | 
  | 
3704  | 0  |     if (m_hSHP)  | 
3705  | 0  |     { | 
3706  | 0  |         if (GetSpatialRef() != nullptr)  | 
3707  | 0  |         { | 
3708  | 0  |             OGRShapeGeomFieldDefn *poGeomFieldDefn =  | 
3709  | 0  |                 cpl::down_cast<OGRShapeGeomFieldDefn *>(  | 
3710  | 0  |                     GetLayerDefn()->GetGeomFieldDefn(0));  | 
3711  | 0  |             oFileList.AddStringDirectly(  | 
3712  | 0  |                 VSIGetCanonicalFilename(poGeomFieldDefn->GetPrjFilename()));  | 
3713  | 0  |         }  | 
3714  | 0  |         if (CheckForQIX())  | 
3715  | 0  |         { | 
3716  | 0  |             const std::string osQIXFilename =  | 
3717  | 0  |                 CPLResetExtensionSafe(m_osFullName.c_str(), "qix");  | 
3718  | 0  |             oFileList.AddStringDirectly(  | 
3719  | 0  |                 VSIGetCanonicalFilename(osQIXFilename.c_str()));  | 
3720  | 0  |         }  | 
3721  | 0  |         else if (CheckForSBN())  | 
3722  | 0  |         { | 
3723  | 0  |             const std::string osSBNFilename =  | 
3724  | 0  |                 CPLResetExtensionSafe(m_osFullName.c_str(), "sbn");  | 
3725  | 0  |             oFileList.AddStringDirectly(  | 
3726  | 0  |                 VSIGetCanonicalFilename(osSBNFilename.c_str()));  | 
3727  | 0  |             const std::string osSBXFilename =  | 
3728  | 0  |                 CPLResetExtensionSafe(m_osFullName.c_str(), "sbx");  | 
3729  | 0  |             oFileList.AddStringDirectly(  | 
3730  | 0  |                 VSIGetCanonicalFilename(osSBXFilename.c_str()));  | 
3731  | 0  |         }  | 
3732  | 0  |     }  | 
3733  | 0  | }  | 
3734  |  |  | 
3735  |  | /************************************************************************/  | 
3736  |  | /*                   UpdateFollowingDeOrRecompression()                 */  | 
3737  |  | /************************************************************************/  | 
3738  |  |  | 
3739  |  | void OGRShapeLayer::UpdateFollowingDeOrRecompression()  | 
3740  | 0  | { | 
3741  | 0  |     CPLAssert(m_poDS->IsZip());  | 
3742  | 0  |     CPLString osDSDir = m_poDS->GetTemporaryUnzipDir();  | 
3743  | 0  |     if (osDSDir.empty())  | 
3744  | 0  |         osDSDir = m_poDS->GetVSIZipPrefixeDir();  | 
3745  |  | 
  | 
3746  | 0  |     if (GetSpatialRef() != nullptr)  | 
3747  | 0  |     { | 
3748  | 0  |         OGRShapeGeomFieldDefn *poGeomFieldDefn =  | 
3749  | 0  |             cpl::down_cast<OGRShapeGeomFieldDefn *>(  | 
3750  | 0  |                 GetLayerDefn()->GetGeomFieldDefn(0));  | 
3751  | 0  |         poGeomFieldDefn->SetPrjFilename(  | 
3752  | 0  |             CPLFormFilenameSafe(  | 
3753  | 0  |                 osDSDir.c_str(),  | 
3754  | 0  |                 CPLGetFilename(poGeomFieldDefn->GetPrjFilename().c_str()),  | 
3755  | 0  |                 nullptr)  | 
3756  | 0  |                 .c_str());  | 
3757  | 0  |     }  | 
3758  |  | 
  | 
3759  | 0  |     m_osFullName = CPLFormFilenameSafe(  | 
3760  | 0  |         osDSDir, CPLGetFilename(m_osFullName.c_str()), nullptr);  | 
3761  | 0  |     CloseUnderlyingLayer();  | 
3762  | 0  | }  | 
3763  |  |  | 
3764  |  | /************************************************************************/  | 
3765  |  | /*                           Rename()                                   */  | 
3766  |  | /************************************************************************/  | 
3767  |  |  | 
3768  |  | OGRErr OGRShapeLayer::Rename(const char *pszNewName)  | 
3769  | 0  | { | 
3770  | 0  |     if (!TestCapability(OLCRename))  | 
3771  | 0  |         return OGRERR_FAILURE;  | 
3772  |  |  | 
3773  | 0  |     if (m_poDS->GetLayerByName(pszNewName) != nullptr)  | 
3774  | 0  |     { | 
3775  | 0  |         CPLError(CE_Failure, CPLE_AppDefined, "Layer %s already exists",  | 
3776  | 0  |                  pszNewName);  | 
3777  | 0  |         return OGRERR_FAILURE;  | 
3778  | 0  |     }  | 
3779  |  |  | 
3780  | 0  |     if (!m_poDS->UncompressIfNeeded())  | 
3781  | 0  |         return OGRERR_FAILURE;  | 
3782  |  |  | 
3783  | 0  |     CPLStringList oFileList;  | 
3784  | 0  |     AddToFileList(oFileList);  | 
3785  |  | 
  | 
3786  | 0  |     const std::string osDirname = CPLGetPathSafe(m_osFullName.c_str());  | 
3787  | 0  |     for (int i = 0; i < oFileList.size(); ++i)  | 
3788  | 0  |     { | 
3789  | 0  |         const std::string osRenamedFile =  | 
3790  | 0  |             CPLFormFilenameSafe(osDirname.c_str(), pszNewName,  | 
3791  | 0  |                                 CPLGetExtensionSafe(oFileList[i]).c_str());  | 
3792  | 0  |         VSIStatBufL sStat;  | 
3793  | 0  |         if (VSIStatL(osRenamedFile.c_str(), &sStat) == 0)  | 
3794  | 0  |         { | 
3795  | 0  |             CPLError(CE_Failure, CPLE_AppDefined, "File %s already exists",  | 
3796  | 0  |                      osRenamedFile.c_str());  | 
3797  | 0  |             return OGRERR_FAILURE;  | 
3798  | 0  |         }  | 
3799  | 0  |     }  | 
3800  |  |  | 
3801  | 0  |     CloseUnderlyingLayer();  | 
3802  |  | 
  | 
3803  | 0  |     for (int i = 0; i < oFileList.size(); ++i)  | 
3804  | 0  |     { | 
3805  | 0  |         const std::string osRenamedFile =  | 
3806  | 0  |             CPLFormFilenameSafe(osDirname.c_str(), pszNewName,  | 
3807  | 0  |                                 CPLGetExtensionSafe(oFileList[i]).c_str());  | 
3808  | 0  |         if (VSIRename(oFileList[i], osRenamedFile.c_str()) != 0)  | 
3809  | 0  |         { | 
3810  | 0  |             CPLError(CE_Failure, CPLE_AppDefined, "Cannot rename %s to %s",  | 
3811  | 0  |                      oFileList[i], osRenamedFile.c_str());  | 
3812  | 0  |             return OGRERR_FAILURE;  | 
3813  | 0  |         }  | 
3814  | 0  |     }  | 
3815  |  |  | 
3816  | 0  |     if (GetSpatialRef() != nullptr)  | 
3817  | 0  |     { | 
3818  | 0  |         OGRShapeGeomFieldDefn *poGeomFieldDefn =  | 
3819  | 0  |             cpl::down_cast<OGRShapeGeomFieldDefn *>(  | 
3820  | 0  |                 GetLayerDefn()->GetGeomFieldDefn(0));  | 
3821  | 0  |         poGeomFieldDefn->SetPrjFilename(  | 
3822  | 0  |             CPLFormFilenameSafe(  | 
3823  | 0  |                 osDirname.c_str(), pszNewName,  | 
3824  | 0  |                 CPLGetExtensionSafe(poGeomFieldDefn->GetPrjFilename().c_str())  | 
3825  | 0  |                     .c_str())  | 
3826  | 0  |                 .c_str());  | 
3827  | 0  |     }  | 
3828  |  | 
  | 
3829  | 0  |     m_osFullName =  | 
3830  | 0  |         CPLFormFilenameSafe(osDirname.c_str(), pszNewName,  | 
3831  | 0  |                             CPLGetExtensionSafe(m_osFullName.c_str()).c_str());  | 
3832  |  | 
  | 
3833  | 0  |     if (!ReopenFileDescriptors())  | 
3834  | 0  |         return OGRERR_FAILURE;  | 
3835  |  |  | 
3836  | 0  |     SetDescription(pszNewName);  | 
3837  | 0  |     whileUnsealing(m_poFeatureDefn)->SetName(pszNewName);  | 
3838  |  | 
  | 
3839  | 0  |     return OGRERR_NONE;  | 
3840  | 0  | }  | 
3841  |  |  | 
3842  |  | /************************************************************************/  | 
3843  |  | /*                          GetDataset()                                */  | 
3844  |  | /************************************************************************/  | 
3845  |  |  | 
3846  |  | GDALDataset *OGRShapeLayer::GetDataset()  | 
3847  | 0  | { | 
3848  | 0  |     return m_poDS;  | 
3849  | 0  | }  | 
3850  |  |  | 
3851  |  | /************************************************************************/  | 
3852  |  | /*                        GetNextArrowArray()                           */  | 
3853  |  | /************************************************************************/  | 
3854  |  |  | 
3855  |  | // Specialized implementation restricted to situations where only retrieving  | 
3856  |  | // of FID values is asked (without filters)  | 
3857  |  | // In other cases, fall back to generic implementation.  | 
3858  |  | int OGRShapeLayer::GetNextArrowArray(struct ArrowArrayStream *stream,  | 
3859  |  |                                      struct ArrowArray *out_array)  | 
3860  | 0  | { | 
3861  | 0  |     m_bLastGetNextArrowArrayUsedOptimizedCodePath = false;  | 
3862  | 0  |     if (!TouchLayer())  | 
3863  | 0  |     { | 
3864  | 0  |         memset(out_array, 0, sizeof(*out_array));  | 
3865  | 0  |         return EIO;  | 
3866  | 0  |     }  | 
3867  |  |  | 
3868  | 0  |     if (!m_hDBF || m_poAttrQuery != nullptr || m_poFilterGeom != nullptr)  | 
3869  | 0  |     { | 
3870  | 0  |         return OGRLayer::GetNextArrowArray(stream, out_array);  | 
3871  | 0  |     }  | 
3872  |  |  | 
3873  |  |     // If any field is not ignored, use generic implementation  | 
3874  | 0  |     const int nFieldCount = m_poFeatureDefn->GetFieldCount();  | 
3875  | 0  |     for (int i = 0; i < nFieldCount; ++i)  | 
3876  | 0  |     { | 
3877  | 0  |         if (!m_poFeatureDefn->GetFieldDefn(i)->IsIgnored())  | 
3878  | 0  |             return OGRLayer::GetNextArrowArray(stream, out_array);  | 
3879  | 0  |     }  | 
3880  | 0  |     if (GetGeomType() != wkbNone &&  | 
3881  | 0  |         !m_poFeatureDefn->GetGeomFieldDefn(0)->IsIgnored())  | 
3882  | 0  |         return OGRLayer::GetNextArrowArray(stream, out_array);  | 
3883  |  |  | 
3884  | 0  |     OGRArrowArrayHelper sHelper(m_poDS, m_poFeatureDefn,  | 
3885  | 0  |                                 m_aosArrowArrayStreamOptions, out_array);  | 
3886  | 0  |     if (out_array->release == nullptr)  | 
3887  | 0  |     { | 
3888  | 0  |         return ENOMEM;  | 
3889  | 0  |     }  | 
3890  |  |  | 
3891  | 0  |     if (!sHelper.m_bIncludeFID)  | 
3892  | 0  |     { | 
3893  | 0  |         out_array->release(out_array);  | 
3894  | 0  |         return OGRLayer::GetNextArrowArray(stream, out_array);  | 
3895  | 0  |     }  | 
3896  |  |  | 
3897  | 0  |     m_bLastGetNextArrowArrayUsedOptimizedCodePath = true;  | 
3898  | 0  |     int nCount = 0;  | 
3899  | 0  |     while (m_iNextShapeId < m_nTotalShapeCount)  | 
3900  | 0  |     { | 
3901  | 0  |         const bool bIsDeleted =  | 
3902  | 0  |             CPL_TO_BOOL(DBFIsRecordDeleted(m_hDBF, m_iNextShapeId));  | 
3903  | 0  |         if (bIsDeleted)  | 
3904  | 0  |         { | 
3905  | 0  |             ++m_iNextShapeId;  | 
3906  | 0  |             continue;  | 
3907  | 0  |         }  | 
3908  | 0  |         if (VSIFEofL(VSI_SHP_GetVSIL(m_hDBF->fp)) ||  | 
3909  | 0  |             VSIFErrorL(VSI_SHP_GetVSIL(m_hDBF->fp)))  | 
3910  | 0  |         { | 
3911  | 0  |             out_array->release(out_array);  | 
3912  | 0  |             memset(out_array, 0, sizeof(*out_array));  | 
3913  | 0  |             return EIO;  | 
3914  | 0  |         }  | 
3915  | 0  |         sHelper.m_panFIDValues[nCount] = m_iNextShapeId;  | 
3916  | 0  |         ++m_iNextShapeId;  | 
3917  | 0  |         ++nCount;  | 
3918  | 0  |         if (nCount == sHelper.m_nMaxBatchSize)  | 
3919  | 0  |             break;  | 
3920  | 0  |     }  | 
3921  | 0  |     sHelper.Shrink(nCount);  | 
3922  | 0  |     if (nCount == 0)  | 
3923  | 0  |     { | 
3924  | 0  |         out_array->release(out_array);  | 
3925  | 0  |         memset(out_array, 0, sizeof(*out_array));  | 
3926  | 0  |     }  | 
3927  | 0  |     return 0;  | 
3928  | 0  | }  | 
3929  |  |  | 
3930  |  | /************************************************************************/  | 
3931  |  | /*                        GetMetadataItem()                             */  | 
3932  |  | /************************************************************************/  | 
3933  |  |  | 
3934  |  | const char *OGRShapeLayer::GetMetadataItem(const char *pszName,  | 
3935  |  |                                            const char *pszDomain)  | 
3936  | 0  | { | 
3937  | 0  |     if (pszName && pszDomain && EQUAL(pszDomain, "__DEBUG__") &&  | 
3938  | 0  |         EQUAL(pszName, "LAST_GET_NEXT_ARROW_ARRAY_USED_OPTIMIZED_CODE_PATH"))  | 
3939  | 0  |     { | 
3940  | 0  |         return m_bLastGetNextArrowArrayUsedOptimizedCodePath ? "YES" : "NO";  | 
3941  | 0  |     }  | 
3942  | 0  |     return OGRLayer::GetMetadataItem(pszName, pszDomain);  | 
3943  | 0  | }  |