Line data Source code
1 : /**********************************************************************
2 : *
3 : * Name: mitab_mapfile.cpp
4 : * Project: MapInfo TAB Read/Write library
5 : * Language: C++
6 : * Purpose: Implementation of the TABMAPFile class used to handle
7 : * reading/writing of the .MAP files at the MapInfo object level
8 : * Author: Daniel Morissette, dmorissette@dmsolutions.ca
9 : *
10 : **********************************************************************
11 : * Copyright (c) 1999-2002, Daniel Morissette
12 : * Copyright (c) 2014, Even Rouault <even.rouault at spatialys.com>
13 : *
14 : * SPDX-License-Identifier: MIT
15 : **********************************************************************/
16 :
17 : #include "cpl_port.h"
18 : #include "mitab.h"
19 :
20 : #include <cassert>
21 : #include <cstddef>
22 : #if HAVE_FCNTL_H
23 : #include <fcntl.h>
24 : #endif
25 : #include <algorithm>
26 : #include <utility>
27 :
28 : #include "cpl_conv.h"
29 : #include "cpl_error.h"
30 : #include "cpl_vsi.h"
31 : #include "mitab_priv.h"
32 : #include "ogr_feature.h"
33 :
34 : /*=====================================================================
35 : * class TABMAPFile
36 : *====================================================================*/
37 :
38 : /**********************************************************************
39 : * TABMAPFile::TABMAPFile()
40 : *
41 : * Constructor.
42 : **********************************************************************/
43 1417 : TABMAPFile::TABMAPFile(const char *pszEncoding)
44 : : m_nMinTABVersion(300), m_pszFname(nullptr), m_fp(nullptr),
45 : m_eAccessMode(TABRead), m_poHeader(nullptr), m_poSpIndex(nullptr),
46 : // See bug 1732: Optimized spatial index produces broken files because of
47 : // the way CoordBlocks are split. For now we have to force using the quick
48 : // (old) spatial index mode by default until bug 1732 is fixed.
49 : m_bQuickSpatialIndexMode(TRUE), m_poIdIndex(nullptr),
50 : m_poCurObjBlock(nullptr), m_nCurObjPtr(-1), m_nCurObjType(TAB_GEOM_UNSET),
51 : m_nCurObjId(-1), m_poCurCoordBlock(nullptr), m_poToolDefTable(nullptr),
52 : m_XMinFilter(0), m_YMinFilter(0), m_XMaxFilter(0), m_YMaxFilter(0),
53 : m_bUpdated(FALSE), m_bLastOpWasRead(FALSE), m_bLastOpWasWrite(FALSE),
54 1417 : m_poSpIndexLeaf(nullptr), m_osEncoding(pszEncoding)
55 : {
56 1417 : m_sMinFilter.x = 0;
57 1417 : m_sMinFilter.y = 0;
58 1417 : m_sMaxFilter.x = 0;
59 1417 : m_sMaxFilter.y = 0;
60 :
61 1417 : m_oBlockManager.SetName("MAP");
62 1417 : }
63 :
64 : /**********************************************************************
65 : * TABMAPFile::~TABMAPFile()
66 : *
67 : * Destructor.
68 : **********************************************************************/
69 1417 : TABMAPFile::~TABMAPFile()
70 : {
71 1417 : Close();
72 1417 : }
73 :
74 : /**********************************************************************
75 : * TABMAPFile::Open()
76 : *
77 : * Compatibility layer with new interface.
78 : * Return 0 on success, -1 in case of failure.
79 : **********************************************************************/
80 :
81 0 : int TABMAPFile::Open(const char *pszFname, const char *pszAccess,
82 : GBool bNoErrorMsg, int nBlockSizeForCreate)
83 : {
84 : // cppcheck-suppress nullPointer
85 0 : if (STARTS_WITH_CI(pszAccess, "r"))
86 0 : return Open(pszFname, TABRead, bNoErrorMsg, nBlockSizeForCreate);
87 0 : else if (STARTS_WITH_CI(pszAccess, "w"))
88 0 : return Open(pszFname, TABWrite, bNoErrorMsg, nBlockSizeForCreate);
89 : else
90 : {
91 0 : CPLError(CE_Failure, CPLE_FileIO,
92 : "Open() failed: access mode \"%s\" not supported", pszAccess);
93 0 : return -1;
94 : }
95 : }
96 :
97 : /**********************************************************************
98 : * TABMAPFile::Open()
99 : *
100 : * Open a .MAP file, and initialize the structures to be ready to read
101 : * objects from it.
102 : *
103 : * Since .MAP and .ID files are optional, you can set bNoErrorMsg=TRUE to
104 : * disable the error message and receive an return value of 1 if file
105 : * cannot be opened.
106 : * In this case, only the methods MoveToObjId() and GetCurObjType() can
107 : * be used. They will behave as if the .ID file contained only null
108 : * references, so all object will look like they have NONE geometries.
109 : *
110 : * Returns 0 on success, 1 when the .map file does not exist, -1 on error.
111 : **********************************************************************/
112 1444 : int TABMAPFile::Open(const char *pszFname, TABAccess eAccess,
113 : GBool bNoErrorMsg /* = FALSE */,
114 : int nBlockSizeForCreate /* = 512 */)
115 : {
116 1444 : CPLErrorReset();
117 :
118 1444 : VSILFILE *fp = nullptr;
119 1444 : TABRawBinBlock *poBlock = nullptr;
120 :
121 1444 : if (m_fp)
122 : {
123 0 : CPLError(CE_Failure, CPLE_FileIO,
124 : "Open() failed: object already contains an open file");
125 0 : return -1;
126 : }
127 :
128 1444 : m_nMinTABVersion = 300;
129 1444 : m_fp = nullptr;
130 1444 : m_poHeader = nullptr;
131 1444 : m_poIdIndex = nullptr;
132 1444 : m_poSpIndex = nullptr;
133 1444 : m_poToolDefTable = nullptr;
134 1444 : m_eAccessMode = eAccess;
135 1444 : m_bUpdated = FALSE;
136 1444 : m_bLastOpWasRead = FALSE;
137 1444 : m_bLastOpWasWrite = FALSE;
138 :
139 1444 : if (m_eAccessMode == TABWrite &&
140 123 : (nBlockSizeForCreate < TAB_MIN_BLOCK_SIZE ||
141 122 : nBlockSizeForCreate > TAB_MAX_BLOCK_SIZE ||
142 122 : (nBlockSizeForCreate % TAB_MIN_BLOCK_SIZE) != 0))
143 : {
144 1 : CPLError(CE_Failure, CPLE_NotSupported,
145 : "Open() failed: invalid block size: %d", nBlockSizeForCreate);
146 1 : return -1;
147 : }
148 :
149 : /*-----------------------------------------------------------------
150 : * Open file
151 : *----------------------------------------------------------------*/
152 1443 : const char *pszAccess = (eAccess == TABRead) ? "rb"
153 : : (eAccess == TABWrite) ? "wb+"
154 : : "rb+";
155 1443 : fp = VSIFOpenL(pszFname, pszAccess);
156 :
157 1443 : m_oBlockManager.Reset();
158 :
159 1443 : if (fp != nullptr &&
160 1440 : (m_eAccessMode == TABRead || m_eAccessMode == TABReadWrite))
161 : {
162 : /*-----------------------------------------------------------------
163 : * Read access: try to read header block
164 : * First try with a 512 bytes block to check the .map version.
165 : * If it is version 500 or more then read again a 1024 bytes block
166 : *----------------------------------------------------------------*/
167 1318 : poBlock = TABCreateMAPBlockFromFile(fp, 0, 512, TRUE, m_eAccessMode);
168 :
169 2636 : if (poBlock && poBlock->GetBlockClass() == TABMAP_HEADER_BLOCK &&
170 1318 : cpl::down_cast<TABMAPHeaderBlock *>(poBlock)->m_nMAPVersionNumber >=
171 : 500)
172 : {
173 : // Version 500 or higher. Read 1024 bytes block instead of 512
174 1318 : delete poBlock;
175 : poBlock =
176 1318 : TABCreateMAPBlockFromFile(fp, 0, 1024, TRUE, m_eAccessMode);
177 : }
178 :
179 2636 : if (poBlock == nullptr ||
180 1318 : poBlock->GetBlockClass() != TABMAP_HEADER_BLOCK)
181 : {
182 0 : if (poBlock)
183 0 : delete poBlock;
184 0 : poBlock = nullptr;
185 0 : VSIFCloseL(fp);
186 0 : CPLError(
187 : CE_Failure, CPLE_FileIO,
188 : "Open() failed: %s does not appear to be a valid .MAP file",
189 : pszFname);
190 0 : return -1;
191 : }
192 2636 : m_oBlockManager.SetBlockSize(
193 1318 : cpl::down_cast<TABMAPHeaderBlock *>(poBlock)->m_nRegularBlockSize);
194 : }
195 125 : else if (fp != nullptr && m_eAccessMode == TABWrite)
196 : {
197 : /*-----------------------------------------------------------------
198 : * Write access: create a new header block
199 : * .MAP files of Version 500 and up appear to have a 1024 bytes
200 : * header. The last 512 bytes are usually all zeros.
201 : *----------------------------------------------------------------*/
202 122 : m_poHeader = new TABMAPHeaderBlock(m_eAccessMode);
203 122 : poBlock = m_poHeader;
204 122 : poBlock->InitNewBlock(fp, nBlockSizeForCreate, 0);
205 :
206 122 : m_oBlockManager.SetBlockSize(m_poHeader->m_nRegularBlockSize);
207 122 : if (m_poHeader->m_nRegularBlockSize == 512)
208 121 : m_oBlockManager.SetLastPtr(512);
209 : else
210 1 : m_oBlockManager.SetLastPtr(0);
211 :
212 122 : m_bUpdated = TRUE;
213 : }
214 3 : else if (bNoErrorMsg)
215 : {
216 : /*-----------------------------------------------------------------
217 : * .MAP does not exist... produce no error message, but set
218 : * the class members so that MoveToObjId() and GetCurObjType()
219 : * can be used to return only NONE geometries.
220 : *----------------------------------------------------------------*/
221 3 : m_fp = nullptr;
222 3 : m_nCurObjType = TAB_GEOM_NONE;
223 :
224 : /* Create a false header block that will return default
225 : * values for projection and coordsys conversion stuff...
226 : */
227 3 : m_poHeader = new TABMAPHeaderBlock(m_eAccessMode);
228 3 : m_poHeader->InitNewBlock(nullptr, 512, 0);
229 :
230 3 : return 1;
231 : }
232 : else
233 : {
234 0 : CPLError(CE_Failure, CPLE_FileIO, "Open() failed for %s", pszFname);
235 0 : return -1;
236 : }
237 :
238 : /*-----------------------------------------------------------------
239 : * File appears to be valid... set the various class members
240 : *----------------------------------------------------------------*/
241 1440 : m_fp = fp;
242 1440 : m_poHeader = cpl::down_cast<TABMAPHeaderBlock *>(poBlock);
243 1440 : m_pszFname = CPLStrdup(pszFname);
244 :
245 : /*-----------------------------------------------------------------
246 : * Create a TABMAPObjectBlock, in READ mode only or in UPDATE mode
247 : * if there's an object
248 : *
249 : * In WRITE mode, the object block will be created only when needed.
250 : * We do not create the object block in the open() call because
251 : * files that contained only "NONE" geometries ended up with empty
252 : * object and spatial index blocks.
253 : *----------------------------------------------------------------*/
254 :
255 1440 : if (m_eAccessMode == TABRead ||
256 1197 : (m_eAccessMode == TABReadWrite && m_poHeader->m_nFirstIndexBlock != 0))
257 : {
258 1291 : m_poCurObjBlock = new TABMAPObjectBlock(m_eAccessMode);
259 1291 : m_poCurObjBlock->InitNewBlock(m_fp, m_poHeader->m_nRegularBlockSize);
260 : }
261 : else
262 : {
263 149 : m_poCurObjBlock = nullptr;
264 : }
265 :
266 : /*-----------------------------------------------------------------
267 : * Open associated .ID (object id index) file
268 : *----------------------------------------------------------------*/
269 1440 : m_poIdIndex = new TABIDFile;
270 1440 : if (m_poIdIndex->Open(pszFname, m_eAccessMode) != 0)
271 : {
272 : // Failed... an error has already been reported
273 0 : Close();
274 0 : return -1;
275 : }
276 :
277 : /*-----------------------------------------------------------------
278 : * Default Coord filter is the MBR of the whole file
279 : * This is currently unused but could eventually be used to handle
280 : * spatial filters more efficiently.
281 : *----------------------------------------------------------------*/
282 1440 : if (m_eAccessMode == TABRead || m_eAccessMode == TABReadWrite)
283 : {
284 1318 : ResetCoordFilter();
285 : }
286 :
287 : /*-----------------------------------------------------------------
288 : * We could scan a file through its quad tree index... but we don't!
289 : *
290 : * In read mode, we just ignore the spatial index.
291 : *
292 : * In write mode the index is created and maintained as new object
293 : * blocks are added inside CommitObjBlock().
294 : *----------------------------------------------------------------*/
295 1440 : m_poSpIndex = nullptr;
296 :
297 1440 : if (m_eAccessMode == TABReadWrite)
298 : {
299 : /* We don't allow quick mode in read/write mode */
300 1075 : m_bQuickSpatialIndexMode = FALSE;
301 :
302 1075 : if (m_poHeader->m_nFirstIndexBlock != 0)
303 : {
304 1048 : poBlock = GetIndexObjectBlock(m_poHeader->m_nFirstIndexBlock);
305 2096 : if (poBlock == nullptr ||
306 1048 : (poBlock->GetBlockType() != TABMAP_INDEX_BLOCK &&
307 4 : poBlock->GetBlockType() != TABMAP_OBJECT_BLOCK))
308 : {
309 0 : CPLError(CE_Failure, CPLE_AppDefined,
310 : "Cannot find first index block at offset %d",
311 0 : m_poHeader->m_nFirstIndexBlock);
312 0 : delete poBlock;
313 : }
314 1048 : else if (poBlock->GetBlockType() == TABMAP_INDEX_BLOCK)
315 : {
316 1044 : m_poSpIndex = cpl::down_cast<TABMAPIndexBlock *>(poBlock);
317 1044 : m_poSpIndex->SetMBR(m_poHeader->m_nXMin, m_poHeader->m_nYMin,
318 1044 : m_poHeader->m_nXMax, m_poHeader->m_nYMax);
319 : }
320 : else /* if( poBlock->GetBlockType() == TABMAP_OBJECT_BLOCK ) */
321 : {
322 : /* This can happen if the file created by MapInfo contains just
323 : */
324 : /* a few objects */
325 4 : delete poBlock;
326 : }
327 : }
328 : }
329 :
330 : /*-----------------------------------------------------------------
331 : * Initialization of the Drawing Tools table will be done automatically
332 : * as Read/Write calls are done later.
333 : *----------------------------------------------------------------*/
334 1440 : m_poToolDefTable = nullptr;
335 :
336 1440 : if (m_eAccessMode == TABReadWrite)
337 : {
338 1075 : InitDrawingTools();
339 : }
340 :
341 1440 : if (m_eAccessMode == TABReadWrite)
342 : {
343 : VSIStatBufL sStatBuf;
344 1075 : if (VSIStatL(m_pszFname, &sStatBuf) != 0)
345 : {
346 0 : Close();
347 0 : return -1;
348 : }
349 1075 : m_oBlockManager.SetLastPtr(static_cast<int>(
350 1075 : ((sStatBuf.st_size - 1) / m_poHeader->m_nRegularBlockSize) *
351 1075 : m_poHeader->m_nRegularBlockSize));
352 :
353 : /* Read chain of garbage blocks */
354 1075 : if (m_poHeader->m_nFirstGarbageBlock != 0)
355 : {
356 0 : int nCurGarbBlock = m_poHeader->m_nFirstGarbageBlock;
357 0 : m_oBlockManager.PushGarbageBlockAsLast(nCurGarbBlock);
358 : while (true)
359 : {
360 0 : GUInt16 nBlockType = 0;
361 0 : int nNextGarbBlockPtr = 0;
362 0 : if (VSIFSeekL(fp, nCurGarbBlock, SEEK_SET) != 0 ||
363 0 : VSIFReadL(&nBlockType, sizeof(nBlockType), 1, fp) != 1 ||
364 0 : VSIFReadL(&nNextGarbBlockPtr, sizeof(nNextGarbBlockPtr), 1,
365 : fp) != 1)
366 : {
367 0 : CPLError(CE_Failure, CPLE_AppDefined,
368 : "Cannot read garbage block at offset %d",
369 : nCurGarbBlock);
370 0 : break;
371 : }
372 0 : if (nBlockType != TABMAP_GARB_BLOCK)
373 : {
374 0 : CPLError(CE_Failure, CPLE_AppDefined,
375 : "Got block type (%d) instead of %d at offset %d",
376 : nBlockType, TABMAP_GARB_BLOCK, nCurGarbBlock);
377 : }
378 0 : if (nNextGarbBlockPtr == 0)
379 0 : break;
380 0 : nCurGarbBlock = nNextGarbBlockPtr;
381 0 : m_oBlockManager.PushGarbageBlockAsLast(nCurGarbBlock);
382 0 : }
383 : }
384 : }
385 :
386 : /*-----------------------------------------------------------------
387 : * Make sure all previous calls succeeded.
388 : *----------------------------------------------------------------*/
389 1440 : if (CPLGetLastErrorType() == CE_Failure)
390 : {
391 : // Open Failed... an error has already been reported
392 0 : Close();
393 0 : return -1;
394 : }
395 :
396 1440 : return 0;
397 : }
398 :
399 : /**********************************************************************
400 : * TABMAPFile::Close()
401 : *
402 : * Close current file, and release all memory used.
403 : *
404 : * Returns 0 on success, -1 on error.
405 : **********************************************************************/
406 2861 : int TABMAPFile::Close()
407 : {
408 : // Check if file is opened... it is possible to have a fake header
409 : // without an actual file attached to it.
410 2861 : if (m_fp == nullptr && m_poHeader == nullptr)
411 1418 : return 0;
412 :
413 : /*----------------------------------------------------------------
414 : * Write access: commit latest changes to the file.
415 : *---------------------------------------------------------------*/
416 1443 : if (m_eAccessMode != TABRead)
417 : {
418 1198 : SyncToDisk();
419 : }
420 :
421 : // Delete all structures
422 1443 : if (m_poHeader)
423 1443 : delete m_poHeader;
424 1443 : m_poHeader = nullptr;
425 :
426 1443 : if (m_poIdIndex)
427 : {
428 1440 : m_poIdIndex->Close();
429 1440 : delete m_poIdIndex;
430 1440 : m_poIdIndex = nullptr;
431 : }
432 :
433 1443 : if (m_poCurObjBlock)
434 : {
435 1381 : delete m_poCurObjBlock;
436 1381 : m_poCurObjBlock = nullptr;
437 1381 : m_nCurObjPtr = -1;
438 1381 : m_nCurObjType = TAB_GEOM_UNSET;
439 1381 : m_nCurObjId = -1;
440 : }
441 :
442 1443 : if (m_poCurCoordBlock)
443 : {
444 55 : delete m_poCurCoordBlock;
445 55 : m_poCurCoordBlock = nullptr;
446 : }
447 :
448 1443 : if (m_poSpIndex)
449 : {
450 1142 : delete m_poSpIndex;
451 1142 : m_poSpIndex = nullptr;
452 1142 : m_poSpIndexLeaf = nullptr;
453 : }
454 :
455 1443 : if (m_poToolDefTable)
456 : {
457 1325 : delete m_poToolDefTable;
458 1325 : m_poToolDefTable = nullptr;
459 : }
460 :
461 : // Close file
462 1443 : if (m_fp)
463 1440 : VSIFCloseL(m_fp);
464 1443 : m_fp = nullptr;
465 :
466 1443 : CPLFree(m_pszFname);
467 1443 : m_pszFname = nullptr;
468 :
469 1443 : return 0;
470 : }
471 :
472 : /************************************************************************/
473 : /* GetFileSize() */
474 : /************************************************************************/
475 :
476 0 : GUInt32 TABMAPFile::GetFileSize()
477 : {
478 0 : if (!m_fp)
479 0 : return 0;
480 0 : vsi_l_offset nCurPos = VSIFTellL(m_fp);
481 0 : VSIFSeekL(m_fp, 0, SEEK_END);
482 0 : vsi_l_offset nSize = VSIFTellL(m_fp);
483 0 : VSIFSeekL(m_fp, nCurPos, SEEK_SET);
484 0 : return nSize > UINT_MAX ? UINT_MAX : static_cast<GUInt32>(nSize);
485 : }
486 :
487 : /************************************************************************/
488 : /* SyncToDisk() */
489 : /************************************************************************/
490 :
491 1317 : int TABMAPFile::SyncToDisk()
492 : {
493 1317 : if (m_eAccessMode == TABRead)
494 : {
495 0 : CPLError(CE_Failure, CPLE_NotSupported,
496 : "SyncToDisk() can be used only with Write access.");
497 0 : return -1;
498 : }
499 :
500 1317 : if (!m_bUpdated)
501 116 : return 0;
502 :
503 : // Start by committing current object and coord blocks
504 : // Nothing happens if none has been created yet.
505 1201 : if (CommitObjAndCoordBlocks(FALSE) != 0)
506 0 : return -1;
507 :
508 : // Write the drawing tools definitions now.
509 1201 : if (CommitDrawingTools() != 0)
510 0 : return -1;
511 :
512 : // Commit spatial index blocks
513 1201 : if (CommitSpatialIndex() != 0)
514 0 : return -1;
515 :
516 : // Update header fields and commit
517 1201 : if (m_poHeader)
518 : {
519 : // OK, with V450 files, objects are not limited to 32k nodes
520 : // any more, and this means that m_nMaxCoordBufSize can become
521 : // huge, and actually more huge than can be held in memory.
522 : // MapInfo counts m_nMaxCoordBufSize=0 for V450 objects, but
523 : // until this is cleanly implemented, we will just prevent
524 : // m_nMaxCoordBufSizefrom going beyond 512k in V450 files.
525 1201 : if (m_nMinTABVersion >= 450)
526 : {
527 0 : m_poHeader->m_nMaxCoordBufSize =
528 0 : std::min(m_poHeader->m_nMaxCoordBufSize, 512 * 1024);
529 : }
530 :
531 : // Write Ref to beginning of the chain of garbage blocks
532 2402 : m_poHeader->m_nFirstGarbageBlock =
533 1201 : m_oBlockManager.GetFirstGarbageBlock();
534 :
535 1201 : if (m_poHeader->CommitToFile() != 0)
536 0 : return -1;
537 : }
538 :
539 : // Check for overflow of internal coordinates and produce a warning
540 : // if that happened...
541 1201 : if (m_poHeader && m_poHeader->m_bIntBoundsOverflow)
542 : {
543 0 : double dBoundsMinX = 0.0;
544 0 : double dBoundsMinY = 0.0;
545 0 : double dBoundsMaxX = 0.0;
546 0 : double dBoundsMaxY = 0.0;
547 0 : Int2Coordsys(-1000000000, -1000000000, dBoundsMinX, dBoundsMinY);
548 0 : Int2Coordsys(1000000000, 1000000000, dBoundsMaxX, dBoundsMaxY);
549 :
550 0 : CPLError(CE_Warning,
551 : static_cast<CPLErrorNum>(TAB_WarningBoundsOverflow),
552 : "Some objects were written outside of the file's "
553 : "predefined bounds.\n"
554 : "These objects may have invalid coordinates when the file "
555 : "is reopened.\n"
556 : "Predefined bounds: (%.15g,%.15g)-(%.15g,%.15g)\n",
557 : dBoundsMinX, dBoundsMinY, dBoundsMaxX, dBoundsMaxY);
558 : }
559 :
560 1201 : if (m_poIdIndex != nullptr && m_poIdIndex->SyncToDisk() != 0)
561 0 : return -1;
562 :
563 1201 : m_bUpdated = FALSE;
564 :
565 1201 : return 0;
566 : }
567 :
568 : /**********************************************************************
569 : * TABMAPFile::ReOpenReadWrite()
570 : **********************************************************************/
571 27 : int TABMAPFile::ReOpenReadWrite()
572 : {
573 27 : char *pszFname = m_pszFname;
574 27 : m_pszFname = nullptr;
575 27 : Close();
576 27 : if (Open(pszFname, TABReadWrite) < 0)
577 : {
578 0 : CPLFree(pszFname);
579 0 : return -1;
580 : }
581 27 : CPLFree(pszFname);
582 27 : return 0;
583 : }
584 :
585 : /**********************************************************************
586 : * TABMAPFile::SetQuickSpatialIndexMode()
587 : *
588 : * Select "quick spatial index mode".
589 : *
590 : * The default behavior of MITAB is to generate an optimized spatial index,
591 : * but this results in slower write speed.
592 : *
593 : * Applications that want faster write speed and do not care
594 : * about the performance of spatial queries on the resulting file can
595 : * use SetQuickSpatialIndexMode() to require the creation of a non-optimal
596 : * spatial index (actually emulating the type of spatial index produced
597 : * by MITAB before version 1.6.0). In this mode writing files can be
598 : * about 5 times faster, but spatial queries can be up to 30 times slower.
599 : *
600 : * Returns 0 on success, -1 on error.
601 : **********************************************************************/
602 0 : int TABMAPFile::SetQuickSpatialIndexMode(GBool bQuickSpatialIndexMode /*=TRUE*/)
603 : {
604 0 : if (m_eAccessMode != TABWrite)
605 : {
606 0 : CPLError(CE_Failure, CPLE_AssertionFailed,
607 : "SetQuickSpatialIndexMode() failed: file not opened for write "
608 : "access.");
609 0 : return -1;
610 : }
611 :
612 0 : if (m_poCurObjBlock != nullptr || m_poSpIndex != nullptr)
613 : {
614 0 : CPLError(CE_Failure, CPLE_AssertionFailed,
615 : "SetQuickSpatialIndexMode() must be called before writing the "
616 : "first object.");
617 0 : return -1;
618 : }
619 :
620 0 : m_bQuickSpatialIndexMode = bQuickSpatialIndexMode;
621 :
622 0 : return 0;
623 : }
624 :
625 : /************************************************************************/
626 : /* PushBlock() */
627 : /* */
628 : /* Install a new block (object or spatial) as being current - */
629 : /* whatever that means. This method is only intended to ever */
630 : /* be called from LoadNextMatchingObjectBlock(). */
631 : /************************************************************************/
632 :
633 37905 : TABRawBinBlock *TABMAPFile::PushBlock(int nFileOffset)
634 :
635 : {
636 37905 : TABRawBinBlock *poBlock = GetIndexObjectBlock(nFileOffset);
637 37905 : if (poBlock == nullptr)
638 0 : return nullptr;
639 :
640 37905 : if (poBlock->GetBlockType() == TABMAP_INDEX_BLOCK)
641 : {
642 : auto poIndex = std::unique_ptr<TABMAPIndexBlock>(
643 45452 : cpl::down_cast<TABMAPIndexBlock *>(poBlock));
644 :
645 22726 : if (m_poSpIndexLeaf == nullptr)
646 : {
647 5 : delete m_poSpIndex;
648 5 : m_poSpIndex = poIndex.release();
649 5 : m_poSpIndexLeaf = m_poSpIndex;
650 : }
651 : else
652 : {
653 22721 : CPLAssert(
654 : m_poSpIndexLeaf->GetEntry(m_poSpIndexLeaf->GetCurChildIndex())
655 : ->nBlockPtr == nFileOffset);
656 :
657 22721 : m_poSpIndexLeaf->SetCurChild(std::move(poIndex),
658 22721 : m_poSpIndexLeaf->GetCurChildIndex());
659 22721 : m_poSpIndexLeaf = m_poSpIndexLeaf->GetCurChild();
660 : }
661 : }
662 : else
663 : {
664 15179 : CPLAssert(poBlock->GetBlockType() == TABMAP_OBJECT_BLOCK);
665 :
666 15179 : if (m_poCurObjBlock != nullptr)
667 15179 : delete m_poCurObjBlock;
668 :
669 15179 : m_poCurObjBlock = cpl::down_cast<TABMAPObjectBlock *>(poBlock);
670 :
671 15179 : m_nCurObjPtr = nFileOffset;
672 15179 : m_nCurObjType = TAB_GEOM_NONE;
673 15179 : m_nCurObjId = -1;
674 : }
675 :
676 37905 : return poBlock;
677 : }
678 :
679 : /************************************************************************/
680 : /* LoadNextMatchingObjectBlock() */
681 : /* */
682 : /* Advance through the spatial indices till the next object */
683 : /* block is loaded that matching the spatial query extents. */
684 : /************************************************************************/
685 :
686 25303 : int TABMAPFile::LoadNextMatchingObjectBlock(int bFirstObject)
687 :
688 : {
689 : // If we are just starting, verify the stack is empty.
690 25303 : if (bFirstObject)
691 : {
692 10624 : CPLAssert(m_poSpIndexLeaf == nullptr);
693 :
694 : /* m_nFirstIndexBlock set to 0 means that there is no feature */
695 10624 : if (m_poHeader->m_nFirstIndexBlock == 0)
696 0 : return FALSE;
697 :
698 10624 : if (m_poSpIndex != nullptr)
699 : {
700 10619 : m_poSpIndex->UnsetCurChild();
701 10619 : m_poSpIndexLeaf = m_poSpIndex;
702 : }
703 : else
704 : {
705 5 : if (PushBlock(m_poHeader->m_nFirstIndexBlock) == nullptr)
706 0 : return FALSE;
707 :
708 5 : if (m_poSpIndex == nullptr)
709 : {
710 0 : CPLAssert(m_poCurObjBlock != nullptr);
711 0 : return TRUE;
712 : }
713 : }
714 : }
715 :
716 443990 : while (m_poSpIndexLeaf != nullptr)
717 : {
718 433866 : int iEntry = m_poSpIndexLeaf->GetCurChildIndex();
719 :
720 433866 : if (iEntry >= m_poSpIndexLeaf->GetNumEntries() - 1)
721 : {
722 32845 : TABMAPIndexBlock *poParent = m_poSpIndexLeaf->GetParentRef();
723 32845 : if (m_poSpIndexLeaf == m_poSpIndex)
724 10124 : m_poSpIndex->UnsetCurChild();
725 32845 : m_poSpIndexLeaf = poParent;
726 :
727 32845 : if (poParent != nullptr)
728 : {
729 22721 : poParent->SetCurChild(nullptr, poParent->GetCurChildIndex());
730 : }
731 32845 : continue;
732 : }
733 :
734 401021 : m_poSpIndexLeaf->SetCurChild(nullptr, ++iEntry);
735 :
736 401021 : TABMAPIndexEntry *psEntry = m_poSpIndexLeaf->GetEntry(iEntry);
737 401021 : if (!psEntry)
738 : {
739 0 : CPLAssert(false);
740 : continue;
741 : }
742 401021 : if (psEntry->XMax < m_XMinFilter || psEntry->YMax < m_YMinFilter ||
743 167880 : psEntry->XMin > m_XMaxFilter || psEntry->YMin > m_YMaxFilter)
744 363121 : continue;
745 :
746 37900 : TABRawBinBlock *poBlock = PushBlock(psEntry->nBlockPtr);
747 37900 : if (poBlock == nullptr)
748 0 : return FALSE;
749 37900 : else if (poBlock->GetBlockType() == TABMAP_OBJECT_BLOCK)
750 15179 : return TRUE;
751 : else
752 : {
753 : /* continue processing new index block */
754 : }
755 : }
756 :
757 10124 : return false;
758 : }
759 :
760 : /************************************************************************/
761 : /* ResetReading() */
762 : /* */
763 : /* Ensure that any resources related to a spatial traversal of */
764 : /* the file are recovered, and the state reinitialized to the */
765 : /* initial conditions. */
766 : /************************************************************************/
767 :
768 33821 : void TABMAPFile::ResetReading()
769 :
770 : {
771 33821 : if (m_bLastOpWasWrite)
772 235 : CommitObjAndCoordBlocks(FALSE);
773 :
774 33821 : if (m_poSpIndex)
775 : {
776 33328 : m_poSpIndex->UnsetCurChild();
777 : }
778 33821 : m_poSpIndexLeaf = nullptr;
779 :
780 33821 : m_bLastOpWasWrite = FALSE;
781 33821 : m_bLastOpWasRead = FALSE;
782 33821 : }
783 :
784 : /************************************************************************/
785 : /* GetNextFeatureId() */
786 : /* */
787 : /* Fetch the next feature id based on a traversal of the */
788 : /* spatial index. */
789 : /************************************************************************/
790 :
791 384348 : int TABMAPFile::GetNextFeatureId(int nPrevId)
792 :
793 : {
794 384348 : if (m_bLastOpWasWrite)
795 : {
796 0 : CPLError(CE_Failure, CPLE_AppDefined,
797 : "GetNextFeatureId() cannot be called after write operation");
798 0 : return -1;
799 : }
800 384348 : if (m_eAccessMode == TABWrite)
801 : {
802 0 : if (ReOpenReadWrite() < 0)
803 0 : return -1;
804 : }
805 384348 : m_bLastOpWasRead = TRUE;
806 :
807 : /* -------------------------------------------------------------------- */
808 : /* m_fp is NULL when all geometry are NONE and/or there's */
809 : /* no .map file and/or there's no spatial indexes */
810 : /* -------------------------------------------------------------------- */
811 384348 : if (m_fp == nullptr)
812 0 : return -1;
813 :
814 384348 : if (nPrevId == 0)
815 10624 : nPrevId = -1;
816 :
817 : /* -------------------------------------------------------------------- */
818 : /* This should always be true if we are being called properly. */
819 : /* -------------------------------------------------------------------- */
820 384348 : if (nPrevId != -1 && m_nCurObjId != nPrevId)
821 : {
822 0 : CPLError(CE_Failure, CPLE_AppDefined,
823 : "TABMAPFile::GetNextFeatureId(%d) called out of sequence.",
824 : nPrevId);
825 0 : return -1;
826 : }
827 :
828 384348 : CPLAssert(nPrevId == -1 || m_poCurObjBlock != nullptr);
829 :
830 : /* -------------------------------------------------------------------- */
831 : /* Ensure things are initialized properly if this is a request */
832 : /* for the first feature. */
833 : /* -------------------------------------------------------------------- */
834 384348 : if (nPrevId == -1)
835 : {
836 10624 : m_nCurObjId = -1;
837 : }
838 :
839 : /* -------------------------------------------------------------------- */
840 : /* Try to advance to the next object in the current object */
841 : /* block. */
842 : /* -------------------------------------------------------------------- */
843 384348 : if (nPrevId == -1 || m_poCurObjBlock->AdvanceToNextObject(m_poHeader) == -1)
844 : {
845 : // If not, try to advance to the next object block, and get
846 : // first object from it. Note that some object blocks actually
847 : // have no objects, so we may have to advance to additional
848 : // object blocks till we find a non-empty one.
849 25275 : GBool bFirstCall = (nPrevId == -1);
850 28 : do
851 : {
852 25303 : if (!LoadNextMatchingObjectBlock(bFirstCall))
853 10124 : return -1;
854 :
855 15179 : bFirstCall = FALSE;
856 15179 : } while (m_poCurObjBlock->AdvanceToNextObject(m_poHeader) == -1);
857 : }
858 :
859 374224 : m_nCurObjType = m_poCurObjBlock->GetCurObjectType();
860 374224 : m_nCurObjId = m_poCurObjBlock->GetCurObjectId();
861 374224 : m_nCurObjPtr = m_poCurObjBlock->GetStartAddress() +
862 374224 : m_poCurObjBlock->GetCurObjectOffset();
863 :
864 374224 : CPLAssert(m_nCurObjId != -1);
865 :
866 374224 : return m_nCurObjId;
867 : }
868 :
869 : /**********************************************************************
870 : * TABMAPFile::Int2Coordsys()
871 : *
872 : * Convert from long integer (internal) to coordinates system units
873 : * as defined in the file's coordsys clause.
874 : *
875 : * Note that the false easting/northing and the conversion factor from
876 : * datum to coordsys units are not included in the calculation.
877 : *
878 : * Returns 0 on success, -1 on error.
879 : **********************************************************************/
880 626382 : int TABMAPFile::Int2Coordsys(GInt32 nX, GInt32 nY, double &dX, double &dY)
881 : {
882 626382 : if (m_poHeader == nullptr)
883 0 : return -1;
884 :
885 626382 : return m_poHeader->Int2Coordsys(nX, nY, dX, dY);
886 : }
887 :
888 : /**********************************************************************
889 : * TABMAPFile::Coordsys2Int()
890 : *
891 : * Convert from coordinates system units as defined in the file's
892 : * coordsys clause to long integer (internal) coordinates.
893 : *
894 : * Note that the false easting/northing and the conversion factor from
895 : * datum to coordsys units are not included in the calculation.
896 : *
897 : * Returns 0 on success, -1 on error.
898 : **********************************************************************/
899 115525 : int TABMAPFile::Coordsys2Int(double dX, double dY, GInt32 &nX, GInt32 &nY,
900 : GBool bIgnoreOverflow /*=FALSE*/)
901 : {
902 115525 : if (m_poHeader == nullptr)
903 0 : return -1;
904 :
905 115525 : return m_poHeader->Coordsys2Int(dX, dY, nX, nY, bIgnoreOverflow);
906 : }
907 :
908 : /**********************************************************************
909 : * TABMAPFile::Int2CoordsysDist()
910 : *
911 : * Convert a pair of X,Y size (or distance) values from long integer
912 : * (internal) to coordinates system units as defined in the file's coordsys
913 : * clause.
914 : *
915 : * The difference with Int2Coordsys() is that this function only applies
916 : * the scaling factor: it does not apply the displacement.
917 : *
918 : * Since the calculations on the X and Y values are independent, either
919 : * one can be omitted (i.e. passed as 0)
920 : *
921 : * Returns 0 on success, -1 on error.
922 : **********************************************************************/
923 12 : int TABMAPFile::Int2CoordsysDist(GInt32 nX, GInt32 nY, double &dX, double &dY)
924 : {
925 12 : if (m_poHeader == nullptr)
926 0 : return -1;
927 :
928 12 : return m_poHeader->Int2CoordsysDist(nX, nY, dX, dY);
929 : }
930 :
931 : /**********************************************************************
932 : * TABMAPFile::Coordsys2IntDist()
933 : *
934 : * Convert a pair of X,Y size (or distance) values from coordinates
935 : * system units as defined in the file's coordsys clause to long
936 : * integer (internal) coordinate units.
937 : *
938 : * The difference with Int2Coordsys() is that this function only applies
939 : * the scaling factor: it does not apply the displacement.
940 : *
941 : * Since the calculations on the X and Y values are independent, either
942 : * one can be omitted (i.e. passed as 0)
943 : *
944 : * Returns 0 on success, -1 on error.
945 : **********************************************************************/
946 4 : int TABMAPFile::Coordsys2IntDist(double dX, double dY, GInt32 &nX, GInt32 &nY)
947 : {
948 4 : if (m_poHeader == nullptr)
949 0 : return -1;
950 :
951 4 : return m_poHeader->Coordsys2IntDist(dX, dY, nX, nY);
952 : }
953 :
954 : /**********************************************************************
955 : * TABMAPFile::SetCoordsysBounds()
956 : *
957 : * Set projection coordinates bounds of the newly created dataset.
958 : *
959 : * This function must be called after creating a new dataset and before any
960 : * feature can be written to it.
961 : *
962 : * Returns 0 on success, -1 on error.
963 : **********************************************************************/
964 122 : int TABMAPFile::SetCoordsysBounds(double dXMin, double dYMin, double dXMax,
965 : double dYMax)
966 : {
967 122 : if (m_poHeader == nullptr)
968 0 : return -1;
969 :
970 : const int nStatus =
971 122 : m_poHeader->SetCoordsysBounds(dXMin, dYMin, dXMax, dYMax);
972 :
973 122 : if (nStatus == 0)
974 122 : ResetCoordFilter();
975 :
976 122 : return nStatus;
977 : }
978 :
979 : /**********************************************************************
980 : * TABMAPFile::GetMaxObjId()
981 : *
982 : * Return the value of the biggest valid object id.
983 : *
984 : * Note that object ids are positive and start at 1.
985 : *
986 : * Returns a value >= 0 on success, -1 on error.
987 : **********************************************************************/
988 0 : GInt32 TABMAPFile::GetMaxObjId()
989 : {
990 0 : if (m_poIdIndex)
991 0 : return m_poIdIndex->GetMaxObjId();
992 :
993 0 : return -1;
994 : }
995 :
996 : /**********************************************************************
997 : * TABMAPFile::MoveToObjId()
998 : *
999 : * Get ready to work with the object with the specified id. The object
1000 : * data pointer (inside m_poCurObjBlock) will be moved to the first byte
1001 : * of data for this map object.
1002 : *
1003 : * The object type and id (i.e. table row number) will be accessible
1004 : * using GetCurObjType() and GetCurObjId().
1005 : *
1006 : * Note that object ids are positive and start at 1.
1007 : *
1008 : * Returns 0 on success, -1 on error.
1009 : **********************************************************************/
1010 679215 : int TABMAPFile::MoveToObjId(int nObjId)
1011 : {
1012 679215 : if (m_bLastOpWasWrite)
1013 : {
1014 0 : CPLError(CE_Failure, CPLE_AppDefined,
1015 : "MoveToObjId() cannot be called after write operation");
1016 0 : return -1;
1017 : }
1018 679215 : if (m_eAccessMode == TABWrite)
1019 : {
1020 27 : if (ReOpenReadWrite() < 0)
1021 0 : return -1;
1022 : }
1023 679215 : m_bLastOpWasRead = TRUE;
1024 :
1025 : /*-----------------------------------------------------------------
1026 : * In non creation mode, since the .MAP/.ID are optional, if the
1027 : * file is not opened then we can still act as if one existed and
1028 : * make any object id look like a TAB_GEOM_NONE
1029 : *----------------------------------------------------------------*/
1030 679215 : if (m_fp == nullptr && m_eAccessMode != TABWrite)
1031 : {
1032 22 : CPLAssert(m_poIdIndex == nullptr && m_poCurObjBlock == nullptr);
1033 22 : m_nCurObjPtr = 0;
1034 22 : m_nCurObjId = nObjId;
1035 22 : m_nCurObjType = TAB_GEOM_NONE;
1036 :
1037 22 : return 0;
1038 : }
1039 :
1040 679193 : if (m_poIdIndex == nullptr)
1041 : {
1042 0 : CPLError(CE_Failure, CPLE_AssertionFailed,
1043 : "MoveToObjId(): file not opened!");
1044 0 : m_nCurObjPtr = -1;
1045 0 : m_nCurObjId = -1;
1046 0 : m_nCurObjType = TAB_GEOM_UNSET;
1047 0 : return -1;
1048 : }
1049 :
1050 : /*-----------------------------------------------------------------
1051 : * Move map object pointer to the right location. Fetch location
1052 : * from the index file, unless we are already pointing at it.
1053 : *----------------------------------------------------------------*/
1054 : int nFileOffset =
1055 679193 : m_nCurObjId == nObjId ? m_nCurObjPtr : m_poIdIndex->GetObjPtr(nObjId);
1056 :
1057 679193 : if (nFileOffset != 0 && m_poCurObjBlock == nullptr)
1058 : {
1059 0 : CPLError(CE_Failure, CPLE_AssertionFailed,
1060 : "MoveToObjId(): no current object block!");
1061 0 : m_nCurObjPtr = -1;
1062 0 : m_nCurObjId = -1;
1063 0 : m_nCurObjType = TAB_GEOM_UNSET;
1064 0 : return -1;
1065 : }
1066 :
1067 679193 : if (nFileOffset == 0)
1068 : {
1069 : /*---------------------------------------------------------
1070 : * Object with no geometry... this is a valid case.
1071 : *--------------------------------------------------------*/
1072 1287 : m_nCurObjPtr = 0;
1073 1287 : m_nCurObjId = nObjId;
1074 1287 : m_nCurObjType = TAB_GEOM_NONE;
1075 : }
1076 677906 : else if (m_poCurObjBlock->GotoByteInFile(nFileOffset, TRUE) == 0)
1077 : {
1078 : /*-------------------------------------------------------------
1079 : * OK, it worked, read the object type and row id.
1080 : *------------------------------------------------------------*/
1081 677906 : m_nCurObjPtr = nFileOffset;
1082 :
1083 677906 : const GByte byVal = m_poCurObjBlock->ReadByte();
1084 677906 : if (IsValidObjType(byVal))
1085 : {
1086 677906 : m_nCurObjType = static_cast<TABGeomType>(byVal);
1087 : }
1088 : else
1089 : {
1090 0 : CPLError(
1091 : CE_Warning,
1092 : static_cast<CPLErrorNum>(TAB_WarningFeatureTypeNotSupported),
1093 : "Unsupported object type %d (0x%2.2x). Feature will be "
1094 : "returned with NONE geometry.",
1095 : byVal, byVal);
1096 0 : m_nCurObjType = TAB_GEOM_NONE;
1097 : }
1098 677906 : m_nCurObjId = m_poCurObjBlock->ReadInt32();
1099 :
1100 : // Do a consistency check...
1101 677906 : if (m_nCurObjId != nObjId)
1102 : {
1103 0 : if (m_nCurObjId == (nObjId | 0x40000000))
1104 : {
1105 0 : CPLError(CE_Failure, CPLE_FileIO,
1106 : "Object %d is marked as deleted in the .MAP file but "
1107 : "not in the .ID file."
1108 : "File may be corrupt.",
1109 : nObjId);
1110 : }
1111 : else
1112 : {
1113 0 : CPLError(
1114 : CE_Failure, CPLE_FileIO,
1115 : "Object ID from the .ID file (%d) differs from the value "
1116 : "in the .MAP file (%d). File may be corrupt.",
1117 : nObjId, m_nCurObjId);
1118 : }
1119 0 : m_nCurObjPtr = -1;
1120 0 : m_nCurObjId = -1;
1121 0 : m_nCurObjType = TAB_GEOM_UNSET;
1122 0 : return -1;
1123 : }
1124 : }
1125 : else
1126 : {
1127 : /*---------------------------------------------------------
1128 : * Failed positioning input file... CPLError has been called.
1129 : *--------------------------------------------------------*/
1130 0 : m_nCurObjPtr = -1;
1131 0 : m_nCurObjId = -1;
1132 0 : m_nCurObjType = TAB_GEOM_UNSET;
1133 0 : return -1;
1134 : }
1135 :
1136 679193 : return 0;
1137 : }
1138 :
1139 : /**********************************************************************
1140 : * TABMAPFile::MarkAsDeleted()
1141 : *
1142 : * Returns 0 on success, -1 on error.
1143 : **********************************************************************/
1144 715 : int TABMAPFile::MarkAsDeleted()
1145 : {
1146 715 : if (m_eAccessMode == TABRead)
1147 0 : return -1;
1148 :
1149 715 : if (m_nCurObjPtr <= 0)
1150 2 : return 0;
1151 :
1152 713 : int ret = 0;
1153 713 : if (m_nCurObjType != TAB_GEOM_NONE)
1154 : {
1155 : /* Goto offset for object id */
1156 1426 : if (m_poCurObjBlock == nullptr ||
1157 713 : m_poCurObjBlock->GotoByteInFile(m_nCurObjPtr + 1, TRUE) != 0)
1158 0 : return -1;
1159 :
1160 : /* Mark object as deleted */
1161 713 : m_poCurObjBlock->WriteInt32(m_nCurObjId | 0x40000000);
1162 :
1163 713 : if (m_poCurObjBlock->CommitToFile() != 0)
1164 0 : ret = -1;
1165 : }
1166 :
1167 : /* Update index entry to reflect delete state as well */
1168 713 : if (m_poIdIndex->SetObjPtr(m_nCurObjId, 0) != 0)
1169 0 : ret = -1;
1170 :
1171 713 : m_nCurObjPtr = -1;
1172 713 : m_nCurObjId = -1;
1173 713 : m_nCurObjType = TAB_GEOM_UNSET;
1174 713 : m_bUpdated = TRUE;
1175 :
1176 713 : return ret;
1177 : }
1178 :
1179 : /**********************************************************************
1180 : * TABMAPFile::UpdateMapHeaderInfo()
1181 : *
1182 : * Update .map header information (counter of objects by type and minimum
1183 : * required version) in light of a new object to be written to the file.
1184 : *
1185 : * Called only by PrepareNewObj() and by the TABCollection class.
1186 : **********************************************************************/
1187 15021 : void TABMAPFile::UpdateMapHeaderInfo(TABGeomType nObjType)
1188 : {
1189 : /*-----------------------------------------------------------------
1190 : * Update count of objects by type in the header block
1191 : *----------------------------------------------------------------*/
1192 15021 : if (nObjType == TAB_GEOM_SYMBOL || nObjType == TAB_GEOM_FONTSYMBOL ||
1193 330 : nObjType == TAB_GEOM_CUSTOMSYMBOL || nObjType == TAB_GEOM_MULTIPOINT ||
1194 330 : nObjType == TAB_GEOM_V800_MULTIPOINT || nObjType == TAB_GEOM_SYMBOL_C ||
1195 330 : nObjType == TAB_GEOM_FONTSYMBOL_C ||
1196 330 : nObjType == TAB_GEOM_CUSTOMSYMBOL_C ||
1197 330 : nObjType == TAB_GEOM_MULTIPOINT_C ||
1198 : nObjType == TAB_GEOM_V800_MULTIPOINT_C)
1199 : {
1200 14691 : m_poHeader->m_numPointObjects++;
1201 : }
1202 330 : else if (nObjType == TAB_GEOM_LINE || nObjType == TAB_GEOM_PLINE ||
1203 304 : nObjType == TAB_GEOM_MULTIPLINE ||
1204 304 : nObjType == TAB_GEOM_V450_MULTIPLINE ||
1205 304 : nObjType == TAB_GEOM_V800_MULTIPLINE || nObjType == TAB_GEOM_ARC ||
1206 304 : nObjType == TAB_GEOM_LINE_C || nObjType == TAB_GEOM_PLINE_C ||
1207 92 : nObjType == TAB_GEOM_MULTIPLINE_C ||
1208 92 : nObjType == TAB_GEOM_V450_MULTIPLINE_C ||
1209 92 : nObjType == TAB_GEOM_V800_MULTIPLINE_C ||
1210 : nObjType == TAB_GEOM_ARC_C)
1211 : {
1212 238 : m_poHeader->m_numLineObjects++;
1213 : }
1214 92 : else if (nObjType == TAB_GEOM_REGION || nObjType == TAB_GEOM_V450_REGION ||
1215 86 : nObjType == TAB_GEOM_V800_REGION || nObjType == TAB_GEOM_RECT ||
1216 86 : nObjType == TAB_GEOM_ROUNDRECT || nObjType == TAB_GEOM_ELLIPSE ||
1217 4 : nObjType == TAB_GEOM_REGION_C ||
1218 4 : nObjType == TAB_GEOM_V450_REGION_C ||
1219 4 : nObjType == TAB_GEOM_V800_REGION_C ||
1220 4 : nObjType == TAB_GEOM_RECT_C || nObjType == TAB_GEOM_ROUNDRECT_C ||
1221 : nObjType == TAB_GEOM_ELLIPSE_C)
1222 : {
1223 88 : m_poHeader->m_numRegionObjects++;
1224 : }
1225 4 : else if (nObjType == TAB_GEOM_TEXT || nObjType == TAB_GEOM_TEXT_C)
1226 : {
1227 4 : m_poHeader->m_numTextObjects++;
1228 : }
1229 :
1230 : /*-----------------------------------------------------------------
1231 : * Check for minimum TAB file version number
1232 : *----------------------------------------------------------------*/
1233 15021 : int nVersion = TAB_GEOM_GET_VERSION(nObjType);
1234 :
1235 15021 : if (nVersion > m_nMinTABVersion)
1236 : {
1237 0 : m_nMinTABVersion = nVersion;
1238 : }
1239 15021 : }
1240 :
1241 : /**********************************************************************
1242 : * TABMAPFile::PrepareNewObj()
1243 : *
1244 : * Get ready to write a new object described by poObjHdr (using the
1245 : * poObjHdr's m_nId (featureId), m_nType and IntMBR members which must
1246 : * have been set by the caller).
1247 : *
1248 : * Depending on whether "quick spatial index mode" is selected, we either:
1249 : *
1250 : * 1- Walk through the spatial index to find the best place to insert the
1251 : * new object, update the spatial index references, and prepare the object
1252 : * data block to be ready to write the object to it.
1253 : * ... or ...
1254 : * 2- prepare the current object data block to be ready to write the
1255 : * object to it. If the object block is full then it is inserted in the
1256 : * spatial index and committed to disk, and a new obj block is created.
1257 : *
1258 : * m_poCurObjBlock will be set to be ready to receive the new object, and
1259 : * a new block will be created if necessary (in which case the current
1260 : * block contents will be committed to disk, etc.) The actual ObjHdr
1261 : * data won't be written to m_poCurObjBlock until CommitNewObj() is called.
1262 : *
1263 : * If this object type uses coordinate blocks, then the coordinate block
1264 : * will be prepared to receive coordinates.
1265 : *
1266 : * This function will also take care of updating the .ID index entry for
1267 : * the new object.
1268 : *
1269 : * Note that object ids are positive and start at 1.
1270 : *
1271 : * Returns 0 on success, -1 on error.
1272 : **********************************************************************/
1273 15081 : int TABMAPFile::PrepareNewObj(TABMAPObjHdr *poObjHdr)
1274 : {
1275 15081 : m_nCurObjPtr = -1;
1276 15081 : m_nCurObjId = -1;
1277 15081 : m_nCurObjType = TAB_GEOM_UNSET;
1278 :
1279 15081 : if (m_eAccessMode == TABRead || m_poIdIndex == nullptr ||
1280 15081 : m_poHeader == nullptr)
1281 : {
1282 0 : CPLError(CE_Failure, CPLE_AssertionFailed,
1283 : "PrepareNewObj() failed: file not opened for write access.");
1284 0 : return -1;
1285 : }
1286 :
1287 15081 : if (m_bLastOpWasRead)
1288 : {
1289 114 : m_bLastOpWasRead = FALSE;
1290 114 : if (m_poSpIndex)
1291 : {
1292 112 : m_poSpIndex->UnsetCurChild();
1293 : }
1294 : }
1295 :
1296 : /*-----------------------------------------------------------------
1297 : * For objects with no geometry, we just update the .ID file and return
1298 : *----------------------------------------------------------------*/
1299 15081 : if (poObjHdr->m_nType == TAB_GEOM_NONE)
1300 : {
1301 60 : m_nCurObjType = poObjHdr->m_nType;
1302 60 : m_nCurObjId = poObjHdr->m_nId;
1303 60 : m_nCurObjPtr = 0;
1304 60 : m_poIdIndex->SetObjPtr(m_nCurObjId, 0);
1305 :
1306 60 : return 0;
1307 : }
1308 :
1309 : /*-----------------------------------------------------------------
1310 : * Update count of objects by type in the header block and minimum
1311 : * required version.
1312 : *----------------------------------------------------------------*/
1313 15021 : UpdateMapHeaderInfo(poObjHdr->m_nType);
1314 :
1315 : /*-----------------------------------------------------------------
1316 : * Depending on the selected spatial index mode, we will either insert
1317 : * new objects via the spatial index (slower write but results in optimal
1318 : * spatial index) or directly in the current ObjBlock (faster write
1319 : * but non-optimal spatial index)
1320 : *----------------------------------------------------------------*/
1321 15021 : if (!m_bQuickSpatialIndexMode)
1322 : {
1323 11568 : if (PrepareNewObjViaSpatialIndex(poObjHdr) != 0)
1324 0 : return -1; /* Error already reported */
1325 : }
1326 : else
1327 : {
1328 3453 : if (PrepareNewObjViaObjBlock(poObjHdr) != 0)
1329 0 : return -1; /* Error already reported */
1330 : }
1331 :
1332 : /*-----------------------------------------------------------------
1333 : * Prepare ObjBlock for this new object.
1334 : * Real data won't be written to the object block until CommitNewObj()
1335 : * is called.
1336 : *----------------------------------------------------------------*/
1337 15021 : m_nCurObjPtr = m_poCurObjBlock->PrepareNewObject(poObjHdr);
1338 15021 : if (m_nCurObjPtr < 0)
1339 : {
1340 0 : CPLError(CE_Failure, CPLE_FileIO,
1341 : "Failed writing object header for feature id %d",
1342 : poObjHdr->m_nId);
1343 0 : return -1;
1344 : }
1345 :
1346 15021 : m_nCurObjType = poObjHdr->m_nType;
1347 15021 : m_nCurObjId = poObjHdr->m_nId;
1348 :
1349 : /*-----------------------------------------------------------------
1350 : * Update .ID Index
1351 : *----------------------------------------------------------------*/
1352 15021 : m_poIdIndex->SetObjPtr(m_nCurObjId, m_nCurObjPtr);
1353 :
1354 : /*-----------------------------------------------------------------
1355 : * Prepare Coords block...
1356 : * create a new TABMAPCoordBlock if it was not done yet.
1357 : *----------------------------------------------------------------*/
1358 15021 : PrepareCoordBlock(m_nCurObjType, m_poCurObjBlock, &m_poCurCoordBlock);
1359 :
1360 15021 : if (CPLGetLastErrorType() == CE_Failure)
1361 0 : return -1;
1362 :
1363 15021 : m_bUpdated = TRUE;
1364 15021 : m_bLastOpWasWrite = TRUE;
1365 :
1366 15021 : return 0;
1367 : }
1368 :
1369 : /**********************************************************************
1370 : * TABMAPFile::PrepareNewObjViaSpatialIndex()
1371 : *
1372 : * Used by TABMAPFile::PrepareNewObj() to walk through the spatial index
1373 : * to find the best place to insert the new object, update the spatial
1374 : * index references, and prepare the object data block to be ready to
1375 : * write the object to it.
1376 : *
1377 : * This method is used when "quick spatial index mode" is NOT selected,
1378 : * i.e. when we want to produce a file with an optimal spatial index
1379 : *
1380 : * Returns 0 on success, -1 on error.
1381 : **********************************************************************/
1382 11568 : int TABMAPFile::PrepareNewObjViaSpatialIndex(TABMAPObjHdr *poObjHdr)
1383 : {
1384 11568 : GInt32 nObjBlockForInsert = -1;
1385 :
1386 : /*-----------------------------------------------------------------
1387 : * Create spatial index if we don't have one yet.
1388 : * We do not create the index and object data blocks in the open()
1389 : * call because files that contained only "NONE" geometries ended up
1390 : * with empty object and spatial index blocks.
1391 : *----------------------------------------------------------------*/
1392 11568 : if (m_poSpIndex == nullptr)
1393 : {
1394 : // Spatial Index not created yet...
1395 6 : m_poSpIndex = new TABMAPIndexBlock(m_eAccessMode);
1396 :
1397 6 : m_poSpIndex->InitNewBlock(m_fp, m_poHeader->m_nRegularBlockSize,
1398 : m_oBlockManager.AllocNewBlock("INDEX"));
1399 6 : m_poSpIndex->SetMAPBlockManagerRef(&m_oBlockManager);
1400 :
1401 6 : if (m_eAccessMode == TABReadWrite &&
1402 6 : m_poHeader->m_nFirstIndexBlock != 0)
1403 : {
1404 : /* This can happen if the file created by MapInfo contains just */
1405 : /* a few objects */
1406 : TABRawBinBlock *poBlock =
1407 3 : GetIndexObjectBlock(m_poHeader->m_nFirstIndexBlock);
1408 3 : CPLAssert(poBlock != nullptr &&
1409 : poBlock->GetBlockType() == TABMAP_OBJECT_BLOCK);
1410 3 : delete poBlock;
1411 :
1412 6 : if (m_poSpIndex->AddEntry(m_poHeader->m_nXMin, m_poHeader->m_nYMin,
1413 3 : m_poHeader->m_nXMax, m_poHeader->m_nYMax,
1414 3 : m_poHeader->m_nFirstIndexBlock) != 0)
1415 0 : return -1;
1416 :
1417 3 : delete m_poCurObjBlock;
1418 3 : m_poCurObjBlock = nullptr;
1419 3 : delete m_poCurCoordBlock;
1420 3 : m_poCurCoordBlock = nullptr;
1421 : }
1422 :
1423 6 : m_poHeader->m_nFirstIndexBlock = m_poSpIndex->GetNodeBlockPtr();
1424 :
1425 : /* We'll also need to create an object data block (later) */
1426 : // nObjBlockForInsert = -1;
1427 :
1428 6 : CPLAssert(m_poCurObjBlock == nullptr);
1429 : }
1430 : else
1431 : /*-----------------------------------------------------------------
1432 : * Search the spatial index to find the best place to insert this
1433 : * new object.
1434 : *----------------------------------------------------------------*/
1435 : {
1436 11562 : nObjBlockForInsert = m_poSpIndex->ChooseLeafForInsert(
1437 : poObjHdr->m_nMinX, poObjHdr->m_nMinY, poObjHdr->m_nMaxX,
1438 : poObjHdr->m_nMaxY);
1439 11562 : if (nObjBlockForInsert == -1)
1440 : {
1441 : /* ChooseLeafForInsert() should not fail unless file is corrupt*/
1442 0 : CPLError(CE_Failure, CPLE_AssertionFailed,
1443 : "ChooseLeafForInsert() Failed?!?!");
1444 0 : return -1;
1445 : }
1446 : }
1447 :
1448 11568 : if (nObjBlockForInsert == -1)
1449 : {
1450 : /*-------------------------------------------------------------
1451 : * Create a new object data block from scratch
1452 : *------------------------------------------------------------*/
1453 6 : m_poCurObjBlock = new TABMAPObjectBlock(TABReadWrite);
1454 :
1455 6 : int nBlockOffset = m_oBlockManager.AllocNewBlock("OBJECT");
1456 :
1457 6 : m_poCurObjBlock->InitNewBlock(m_fp, m_poHeader->m_nRegularBlockSize,
1458 : nBlockOffset);
1459 :
1460 : /*-------------------------------------------------------------
1461 : * Insert new object block in index, based on MBR of poObjHdr
1462 : *------------------------------------------------------------*/
1463 6 : if (m_poSpIndex->AddEntry(poObjHdr->m_nMinX, poObjHdr->m_nMinY,
1464 : poObjHdr->m_nMaxX, poObjHdr->m_nMaxY,
1465 12 : m_poCurObjBlock->GetStartAddress()) != 0)
1466 0 : return -1;
1467 :
1468 6 : m_poCurObjBlock->SetMBR(poObjHdr->m_nMinX, poObjHdr->m_nMinY,
1469 : poObjHdr->m_nMaxX, poObjHdr->m_nMaxY);
1470 :
1471 6 : const int nNextDepth = m_poSpIndex->GetCurMaxDepth() + 1;
1472 6 : m_poHeader->m_nMaxSpIndexDepth = static_cast<GByte>(std::max(
1473 6 : static_cast<int>(m_poHeader->m_nMaxSpIndexDepth), nNextDepth));
1474 : }
1475 : else
1476 : {
1477 : /*-------------------------------------------------------------
1478 : * Load existing object and Coord blocks, unless we've already
1479 : * got the right object block in memory
1480 : *------------------------------------------------------------*/
1481 23124 : if (m_poCurObjBlock &&
1482 11562 : m_poCurObjBlock->GetStartAddress() != nObjBlockForInsert)
1483 : {
1484 : /* Got a block in memory but it is not the right one, flush it */
1485 10716 : if (CommitObjAndCoordBlocks(TRUE) != 0)
1486 0 : return -1;
1487 : }
1488 :
1489 11562 : if (m_poCurObjBlock == nullptr)
1490 : {
1491 10716 : if (LoadObjAndCoordBlocks(nObjBlockForInsert) != 0)
1492 0 : return -1;
1493 : }
1494 :
1495 : /* If we have compressed objects, we don't want to change the center */
1496 11562 : m_poCurObjBlock->LockCenter();
1497 :
1498 : // Check if the ObjBlock know its MBR. If not (new block, or the current
1499 : // block was the good one but retrieved without the index), get the
1500 : // value from the index and set it.
1501 : GInt32 nMinX, nMinY, nMaxX, nMaxY;
1502 11562 : m_poCurObjBlock->GetMBR(nMinX, nMinY, nMaxX, nMaxY);
1503 11562 : if (nMinX > nMaxX)
1504 : {
1505 10720 : m_poSpIndex->GetCurLeafEntryMBR(m_poCurObjBlock->GetStartAddress(),
1506 : nMinX, nMinY, nMaxX, nMaxY);
1507 10720 : m_poCurObjBlock->SetMBR(nMinX, nMinY, nMaxX, nMaxY);
1508 : }
1509 : }
1510 :
1511 : /*-----------------------------------------------------------------
1512 : * Fetch new object size, make sure there is enough room in obj.
1513 : * block for new object, update spatial index and split if necessary.
1514 : *----------------------------------------------------------------*/
1515 11568 : int nObjSize = m_poHeader->GetMapObjectSize(poObjHdr->m_nType);
1516 :
1517 : /*-----------------------------------------------------------------
1518 : * But first check if we can recover space from this block in case
1519 : * there are deleted objects in it.
1520 : *----------------------------------------------------------------*/
1521 11568 : if (m_poCurObjBlock->GetNumUnusedBytes() < nObjSize)
1522 : {
1523 531 : std::vector<std::unique_ptr<TABMAPObjHdr>> apoSrcObjHdrs;
1524 531 : int nObjectSpace = 0;
1525 :
1526 : /* First pass to enumerate valid objects and compute their accumulated
1527 : required size. */
1528 531 : m_poCurObjBlock->Rewind();
1529 : while (auto poExistingObjHdr =
1530 11890 : TABMAPObjHdr::ReadNextObj(m_poCurObjBlock, m_poHeader))
1531 : {
1532 11359 : nObjectSpace +=
1533 11359 : m_poHeader->GetMapObjectSize(poExistingObjHdr->m_nType);
1534 11359 : apoSrcObjHdrs.emplace_back(poExistingObjHdr);
1535 11359 : }
1536 :
1537 : /* Check that there's really some place that can be recovered */
1538 1062 : if (nObjectSpace < m_poHeader->m_nRegularBlockSize - 20 -
1539 531 : m_poCurObjBlock->GetNumUnusedBytes())
1540 : {
1541 : #ifdef DEBUG_VERBOSE
1542 : CPLDebug("MITAB",
1543 : "Compacting block at offset %d, %d objects valid, "
1544 : "recovering %d bytes",
1545 : m_poCurObjBlock->GetStartAddress(),
1546 : static_cast<int>(apoSrcObjHdrs.size()),
1547 : (m_poHeader->m_nRegularBlockSize - 20 -
1548 : m_poCurObjBlock->GetNumUnusedBytes()) -
1549 : nObjectSpace);
1550 : #endif
1551 271 : m_poCurObjBlock->ClearObjects();
1552 :
1553 2749 : for (auto &poSrcObjHdrs : apoSrcObjHdrs)
1554 : {
1555 : /*-----------------------------------------------------------------
1556 : * Prepare and Write ObjHdr to this ObjBlock
1557 : *----------------------------------------------------------------*/
1558 : int nObjPtr =
1559 2478 : m_poCurObjBlock->PrepareNewObject(poSrcObjHdrs.get());
1560 4956 : if (nObjPtr < 0 ||
1561 2478 : m_poCurObjBlock->CommitNewObject(poSrcObjHdrs.get()) != 0)
1562 : {
1563 0 : CPLError(CE_Failure, CPLE_FileIO,
1564 : "Failed writing object header for feature id %d",
1565 0 : poSrcObjHdrs->m_nId);
1566 0 : return -1;
1567 : }
1568 :
1569 : /*-----------------------------------------------------------------
1570 : * Update .ID Index
1571 : *----------------------------------------------------------------*/
1572 2478 : m_poIdIndex->SetObjPtr(poSrcObjHdrs->m_nId, nObjPtr);
1573 : }
1574 : }
1575 : }
1576 :
1577 11568 : if (m_poCurObjBlock->GetNumUnusedBytes() >= nObjSize)
1578 : {
1579 : /*-------------------------------------------------------------
1580 : * New object fits in current block, just update the spatial index
1581 : *------------------------------------------------------------*/
1582 : GInt32 nMinX, nMinY, nMaxX, nMaxY;
1583 11308 : m_poCurObjBlock->GetMBR(nMinX, nMinY, nMaxX, nMaxY);
1584 :
1585 : /* Need to calculate the enlarged MBR that includes new object */
1586 11308 : nMinX = std::min(nMinX, poObjHdr->m_nMinX);
1587 11308 : nMinY = std::min(nMinY, poObjHdr->m_nMinY);
1588 11308 : nMaxX = std::max(nMaxX, poObjHdr->m_nMaxX);
1589 11308 : nMaxY = std::max(nMaxY, poObjHdr->m_nMaxY);
1590 :
1591 11308 : m_poCurObjBlock->SetMBR(nMinX, nMinY, nMaxX, nMaxY);
1592 :
1593 11308 : if (m_poSpIndex->UpdateLeafEntry(m_poCurObjBlock->GetStartAddress(),
1594 11308 : nMinX, nMinY, nMaxX, nMaxY) != 0)
1595 0 : return -1;
1596 : }
1597 : else
1598 : {
1599 : /*-------------------------------------------------------------
1600 : * OK, the new object won't fit in the current block, need to split
1601 : * and update index.
1602 : * Split() does its job so that the current obj block will remain
1603 : * the best candidate to receive the new object. It also flushes
1604 : * everything to disk and will update m_poCurCoordBlock to point to
1605 : * the last coord block in the chain, ready to accept new data
1606 : *------------------------------------------------------------*/
1607 : auto poNewObjBlock = std::unique_ptr<TABMAPObjectBlock>(
1608 260 : SplitObjBlock(poObjHdr, nObjSize));
1609 :
1610 260 : if (poNewObjBlock == nullptr)
1611 0 : return -1; /* Split failed, error already reported. */
1612 :
1613 : /*-------------------------------------------------------------
1614 : * Update index with info about m_poCurObjectBlock *first*
1615 : * This is important since UpdateLeafEntry() needs the chain of
1616 : * index nodes preloaded by ChooseLeafEntry() in order to do its job
1617 : *------------------------------------------------------------*/
1618 260 : GInt32 nMinX = 0;
1619 260 : GInt32 nMinY = 0;
1620 260 : GInt32 nMaxX = 0;
1621 260 : GInt32 nMaxY = 0;
1622 260 : m_poCurObjBlock->GetMBR(nMinX, nMinY, nMaxX, nMaxY);
1623 260 : CPLAssert(nMinX <= nMaxX);
1624 :
1625 : /* Need to calculate the enlarged MBR that includes new object */
1626 260 : nMinX = std::min(nMinX, poObjHdr->m_nMinX);
1627 260 : nMinY = std::min(nMinY, poObjHdr->m_nMinY);
1628 260 : nMaxX = std::max(nMaxX, poObjHdr->m_nMaxX);
1629 260 : nMaxY = std::max(nMaxY, poObjHdr->m_nMaxY);
1630 :
1631 260 : m_poCurObjBlock->SetMBR(nMinX, nMinY, nMaxX, nMaxY);
1632 :
1633 260 : if (m_poSpIndex->UpdateLeafEntry(m_poCurObjBlock->GetStartAddress(),
1634 260 : nMinX, nMinY, nMaxX, nMaxY) != 0)
1635 0 : return -1;
1636 :
1637 : /*-------------------------------------------------------------
1638 : * Add new obj block to index
1639 : *------------------------------------------------------------*/
1640 260 : poNewObjBlock->GetMBR(nMinX, nMinY, nMaxX, nMaxY);
1641 260 : CPLAssert(nMinX <= nMaxX);
1642 :
1643 260 : if (m_poSpIndex->AddEntry(nMinX, nMinY, nMaxX, nMaxY,
1644 520 : poNewObjBlock->GetStartAddress()) != 0)
1645 0 : return -1;
1646 260 : const int nNextDepth = m_poSpIndex->GetCurMaxDepth() + 1;
1647 260 : m_poHeader->m_nMaxSpIndexDepth = static_cast<GByte>(std::max(
1648 260 : static_cast<int>(m_poHeader->m_nMaxSpIndexDepth), nNextDepth));
1649 :
1650 : /*-------------------------------------------------------------
1651 : * Implicitly delete second object block, no need to commit to file
1652 : *first since it is already been committed to disk by Split()
1653 : *------------------------------------------------------------*/
1654 : }
1655 :
1656 11568 : return 0;
1657 : }
1658 :
1659 : /**********************************************************************
1660 : * TABMAPFile::PrepareNewObjViaObjBlock()
1661 : *
1662 : * Used by TABMAPFile::PrepareNewObj() to prepare the current object
1663 : * data block to be ready to write the object to it. If the object block
1664 : * is full then it is inserted in the spatial index and committed to disk,
1665 : * and a new obj block is created.
1666 : *
1667 : * This method is used when "quick spatial index mode" is selected,
1668 : * i.e. faster write, but non-optimal spatial index.
1669 : *
1670 : * Returns 0 on success, -1 on error.
1671 : **********************************************************************/
1672 3453 : int TABMAPFile::PrepareNewObjViaObjBlock(TABMAPObjHdr *poObjHdr)
1673 : {
1674 : /*-------------------------------------------------------------
1675 : * We will need an object block... check if it exists and
1676 : * create it if it has not been created yet (first time for this file).
1677 : * We do not create the object block in the open() call because
1678 : * files that contained only "NONE" geometries ended up with empty
1679 : * object and spatial index blocks.
1680 : * Note: A coord block will be created only if needed later.
1681 : *------------------------------------------------------------*/
1682 3453 : if (m_poCurObjBlock == nullptr)
1683 : {
1684 87 : m_poCurObjBlock = new TABMAPObjectBlock(m_eAccessMode);
1685 :
1686 87 : int nBlockOffset = m_oBlockManager.AllocNewBlock("OBJECT");
1687 :
1688 87 : m_poCurObjBlock->InitNewBlock(m_fp, m_poHeader->m_nRegularBlockSize,
1689 : nBlockOffset);
1690 :
1691 : // The reference to the first object block should
1692 : // actually go through the index blocks... this will be
1693 : // updated when file is closed.
1694 87 : m_poHeader->m_nFirstIndexBlock = nBlockOffset;
1695 : }
1696 :
1697 : /*-----------------------------------------------------------------
1698 : * Fetch new object size, make sure there is enough room in obj.
1699 : * block for new object, and save/create a new one if necessary.
1700 : *----------------------------------------------------------------*/
1701 3453 : const int nObjSize = m_poHeader->GetMapObjectSize(poObjHdr->m_nType);
1702 3453 : if (m_poCurObjBlock->GetNumUnusedBytes() < nObjSize)
1703 : {
1704 : /*-------------------------------------------------------------
1705 : * OK, the new object won't fit in the current block. Add the
1706 : * current block to the spatial index, commit it to disk and init
1707 : * a new block
1708 : *------------------------------------------------------------*/
1709 90 : CommitObjAndCoordBlocks(FALSE);
1710 :
1711 90 : if (m_poCurObjBlock->InitNewBlock(
1712 90 : m_fp, m_poHeader->m_nRegularBlockSize,
1713 90 : m_oBlockManager.AllocNewBlock("OBJECT")) != 0)
1714 0 : return -1; /* Error already reported */
1715 :
1716 : /*-------------------------------------------------------------
1717 : * Coord block has been committed to disk but not deleted.
1718 : * Delete it to require the creation of a new coord block chain
1719 : * as needed.
1720 : *-------------------------------------------------------------*/
1721 90 : if (m_poCurCoordBlock)
1722 : {
1723 1 : delete m_poCurCoordBlock;
1724 1 : m_poCurCoordBlock = nullptr;
1725 : }
1726 : }
1727 :
1728 3453 : return 0;
1729 : }
1730 :
1731 : /**********************************************************************
1732 : * TABMAPFile::CommitNewObj()
1733 : *
1734 : * Commit object header data to the ObjBlock. Should be called after
1735 : * PrepareNewObj, once all members of the ObjHdr have been set.
1736 : *
1737 : * Returns 0 on success, -1 on error.
1738 : **********************************************************************/
1739 15081 : int TABMAPFile::CommitNewObj(TABMAPObjHdr *poObjHdr)
1740 : {
1741 : // Nothing to do for NONE objects
1742 15081 : if (poObjHdr->m_nType == TAB_GEOM_NONE)
1743 : {
1744 60 : return 0;
1745 : }
1746 :
1747 : /* Update this now so that PrepareCoordBlock() doesn't try to old an older
1748 : */
1749 : /* block */
1750 15021 : if (m_poCurCoordBlock != nullptr)
1751 326 : m_poCurObjBlock->AddCoordBlockRef(m_poCurCoordBlock->GetStartAddress());
1752 :
1753 : /* So that GetExtent() is up-to-date */
1754 15021 : if (m_poSpIndex != nullptr)
1755 : {
1756 14611 : m_poSpIndex->GetMBR(m_poHeader->m_nXMin, m_poHeader->m_nYMin,
1757 14611 : m_poHeader->m_nXMax, m_poHeader->m_nYMax);
1758 : }
1759 :
1760 15021 : return m_poCurObjBlock->CommitNewObject(poObjHdr);
1761 : }
1762 :
1763 : /**********************************************************************
1764 : * TABMAPFile::CommitObjAndCoordBlocks()
1765 : *
1766 : * Commit the TABMAPObjBlock and TABMAPCoordBlock to disk.
1767 : *
1768 : * The objects are deleted from memory if bDeleteObjects==TRUE.
1769 : *
1770 : * Returns 0 on success, -1 on error.
1771 : **********************************************************************/
1772 12242 : int TABMAPFile::CommitObjAndCoordBlocks(GBool bDeleteObjects /*=FALSE*/)
1773 : {
1774 12242 : int nStatus = 0;
1775 :
1776 : /*-----------------------------------------------------------------
1777 : * First check that a objBlock has been created. It is possible to have
1778 : * no object block in files that contain only "NONE" geometries.
1779 : *----------------------------------------------------------------*/
1780 12242 : if (m_poCurObjBlock == nullptr)
1781 36 : return 0;
1782 :
1783 12206 : if (m_eAccessMode == TABRead)
1784 : {
1785 0 : CPLError(CE_Failure, CPLE_AssertionFailed,
1786 : "CommitObjAndCoordBlocks() failed: file not opened for write "
1787 : "access.");
1788 0 : return -1;
1789 : }
1790 :
1791 12206 : if (!m_bLastOpWasWrite)
1792 : {
1793 1131 : if (bDeleteObjects)
1794 : {
1795 1093 : delete m_poCurCoordBlock;
1796 1093 : m_poCurCoordBlock = nullptr;
1797 1093 : delete m_poCurObjBlock;
1798 1093 : m_poCurObjBlock = nullptr;
1799 : }
1800 1131 : return 0;
1801 : }
1802 11075 : m_bLastOpWasWrite = FALSE;
1803 :
1804 : /*-----------------------------------------------------------------
1805 : * We need to flush the coord block if there was one
1806 : * since a list of coord blocks can belong to only one obj. block
1807 : *----------------------------------------------------------------*/
1808 11075 : if (m_poCurCoordBlock)
1809 : {
1810 : // Update the m_nMaxCoordBufSize member in the header block
1811 : //
1812 204 : int nTotalCoordSize = m_poCurCoordBlock->GetNumBlocksInChain() *
1813 204 : m_poHeader->m_nRegularBlockSize;
1814 204 : if (nTotalCoordSize > m_poHeader->m_nMaxCoordBufSize)
1815 35 : m_poHeader->m_nMaxCoordBufSize = nTotalCoordSize;
1816 :
1817 : // Update the references to this coord block in the MAPObjBlock
1818 : //
1819 204 : m_poCurObjBlock->AddCoordBlockRef(m_poCurCoordBlock->GetStartAddress());
1820 204 : nStatus = m_poCurCoordBlock->CommitToFile();
1821 :
1822 204 : if (bDeleteObjects)
1823 : {
1824 63 : delete m_poCurCoordBlock;
1825 63 : m_poCurCoordBlock = nullptr;
1826 : }
1827 : }
1828 :
1829 : /*-----------------------------------------------------------------
1830 : * Commit the obj block
1831 : *----------------------------------------------------------------*/
1832 11075 : if (nStatus == 0)
1833 : {
1834 11075 : nStatus = m_poCurObjBlock->CommitToFile();
1835 : }
1836 :
1837 : /*-----------------------------------------------------------------
1838 : * Update the spatial index ** only in "quick spatial index" mode **
1839 : * In the (default) optimized spatial index mode, the spatial index
1840 : * is already maintained up to date as part of inserting the objects in
1841 : * PrepareNewObj().
1842 : *
1843 : * Spatial index will be created here if it was not done yet.
1844 : *----------------------------------------------------------------*/
1845 11075 : if (nStatus == 0 && m_bQuickSpatialIndexMode)
1846 : {
1847 205 : if (m_poSpIndex == nullptr)
1848 : {
1849 : // Spatial Index not created yet...
1850 87 : m_poSpIndex = new TABMAPIndexBlock(m_eAccessMode);
1851 :
1852 87 : m_poSpIndex->InitNewBlock(m_fp, m_poHeader->m_nRegularBlockSize,
1853 : m_oBlockManager.AllocNewBlock("INDEX"));
1854 87 : m_poSpIndex->SetMAPBlockManagerRef(&m_oBlockManager);
1855 :
1856 87 : m_poHeader->m_nFirstIndexBlock = m_poSpIndex->GetNodeBlockPtr();
1857 : }
1858 :
1859 : GInt32 nXMin, nYMin, nXMax, nYMax;
1860 205 : m_poCurObjBlock->GetMBR(nXMin, nYMin, nXMax, nYMax);
1861 205 : nStatus = m_poSpIndex->AddEntry(nXMin, nYMin, nXMax, nYMax,
1862 205 : m_poCurObjBlock->GetStartAddress());
1863 :
1864 205 : const int nNextDepth = m_poSpIndex->GetCurMaxDepth() + 1;
1865 205 : m_poHeader->m_nMaxSpIndexDepth = static_cast<GByte>(std::max(
1866 205 : static_cast<int>(m_poHeader->m_nMaxSpIndexDepth), nNextDepth));
1867 : }
1868 :
1869 : /*-----------------------------------------------------------------
1870 : * Delete obj block only if requested
1871 : *----------------------------------------------------------------*/
1872 11075 : if (bDeleteObjects)
1873 : {
1874 9623 : delete m_poCurObjBlock;
1875 9623 : m_poCurObjBlock = nullptr;
1876 : }
1877 :
1878 11075 : return nStatus;
1879 : }
1880 :
1881 : /**********************************************************************
1882 : * TABMAPFile::LoadObjAndCoordBlocks()
1883 : *
1884 : * Load the TABMAPObjBlock at specified address and corresponding
1885 : * TABMAPCoordBlock, ready to write new objects to them.
1886 : *
1887 : * It is assumed that pre-existing m_poCurObjBlock and m_poCurCoordBlock
1888 : * have been flushed to disk already using CommitObjAndCoordBlocks()
1889 : *
1890 : * Returns 0 on success, -1 on error.
1891 : **********************************************************************/
1892 10716 : int TABMAPFile::LoadObjAndCoordBlocks(GInt32 nBlockPtr)
1893 : {
1894 : /*-----------------------------------------------------------------
1895 : * In Write mode, if an object block is already in memory then flush it
1896 : *----------------------------------------------------------------*/
1897 10716 : if (m_eAccessMode != TABRead && m_poCurObjBlock != nullptr)
1898 : {
1899 0 : int nStatus = CommitObjAndCoordBlocks(TRUE);
1900 0 : if (nStatus != 0)
1901 0 : return nStatus;
1902 : }
1903 :
1904 : /*-----------------------------------------------------------------
1905 : * Load Obj Block
1906 : *----------------------------------------------------------------*/
1907 21432 : TABRawBinBlock *poBlock = TABCreateMAPBlockFromFile(
1908 10716 : m_fp, nBlockPtr, m_poHeader->m_nRegularBlockSize, TRUE, TABReadWrite);
1909 10716 : if (poBlock != nullptr && poBlock->GetBlockClass() == TABMAP_OBJECT_BLOCK)
1910 : {
1911 10716 : m_poCurObjBlock = cpl::down_cast<TABMAPObjectBlock *>(poBlock);
1912 10716 : poBlock = nullptr;
1913 : }
1914 : else
1915 : {
1916 0 : CPLError(CE_Failure, CPLE_FileIO,
1917 : "LoadObjAndCoordBlocks() failed for object block at %d.",
1918 : nBlockPtr);
1919 0 : return -1;
1920 : }
1921 :
1922 : /*-----------------------------------------------------------------
1923 : * Load the last coord block in the chain
1924 : *----------------------------------------------------------------*/
1925 10716 : if (m_poCurObjBlock->GetLastCoordBlockAddress() == 0)
1926 : {
1927 10566 : m_poCurCoordBlock = nullptr;
1928 10566 : return 0;
1929 : }
1930 :
1931 150 : poBlock = TABCreateMAPBlockFromFile(
1932 150 : m_fp, m_poCurObjBlock->GetLastCoordBlockAddress(),
1933 150 : m_poHeader->m_nRegularBlockSize, TRUE, TABReadWrite);
1934 150 : if (poBlock != nullptr && poBlock->GetBlockClass() == TABMAP_COORD_BLOCK)
1935 : {
1936 150 : m_poCurCoordBlock = cpl::down_cast<TABMAPCoordBlock *>(poBlock);
1937 150 : m_poCurCoordBlock->SetMAPBlockManagerRef(&m_oBlockManager);
1938 150 : poBlock = nullptr;
1939 : }
1940 : else
1941 : {
1942 0 : CPLError(CE_Failure, CPLE_FileIO,
1943 : "LoadObjAndCoordBlocks() failed for coord block at %d.",
1944 0 : m_poCurObjBlock->GetLastCoordBlockAddress());
1945 0 : return -1;
1946 : }
1947 :
1948 150 : return 0;
1949 : }
1950 :
1951 : /**********************************************************************
1952 : * TABMAPFile::SplitObjBlock()
1953 : *
1954 : * Split m_poCurObjBlock using Guttman algorithm.
1955 : *
1956 : * SplitObjBlock() doe its job so that the current obj block will remain
1957 : * the best candidate to receive the new object to add. It also flushes
1958 : * everything to disk and will update m_poCurCoordBlock to point to the
1959 : * last coord block in the chain, ready to accept new data
1960 : *
1961 : * Updates to the spatial index are left to the caller.
1962 : *
1963 : * Returns the TABMAPObjBlock of the second block for use by the caller
1964 : * in updating the spatial index, or NULL in case of error.
1965 : **********************************************************************/
1966 260 : TABMAPObjectBlock *TABMAPFile::SplitObjBlock(TABMAPObjHdr *poObjHdrToAdd,
1967 : int nSizeOfObjToAdd)
1968 : {
1969 520 : std::vector<std::unique_ptr<TABMAPObjHdr>> apoSrcObjHdrs;
1970 :
1971 : /*-----------------------------------------------------------------
1972 : * Read all object headers
1973 : *----------------------------------------------------------------*/
1974 260 : m_poCurObjBlock->Rewind();
1975 : while (auto poObjHdr =
1976 9141 : TABMAPObjHdr::ReadNextObj(m_poCurObjBlock, m_poHeader))
1977 : {
1978 8881 : apoSrcObjHdrs.emplace_back(poObjHdr);
1979 8881 : }
1980 : /* PickSeedsForSplit (reasonably) assumes at least 2 nodes */
1981 260 : CPLAssert(apoSrcObjHdrs.size() > 1);
1982 :
1983 : /*-----------------------------------------------------------------
1984 : * Reset current obj and coord block
1985 : *----------------------------------------------------------------*/
1986 260 : GInt32 nFirstSrcCoordBlock = m_poCurObjBlock->GetFirstCoordBlockAddress();
1987 :
1988 260 : m_poCurObjBlock->InitNewBlock(m_fp, m_poHeader->m_nRegularBlockSize,
1989 260 : m_poCurObjBlock->GetStartAddress());
1990 :
1991 520 : std::unique_ptr<TABMAPCoordBlock> poSrcCoordBlock(m_poCurCoordBlock);
1992 260 : m_poCurCoordBlock = nullptr;
1993 :
1994 : /*-----------------------------------------------------------------
1995 : * Create new obj and coord block
1996 : *----------------------------------------------------------------*/
1997 520 : auto poNewObjBlock = std::make_unique<TABMAPObjectBlock>(m_eAccessMode);
1998 260 : poNewObjBlock->InitNewBlock(m_fp, m_poHeader->m_nRegularBlockSize,
1999 : m_oBlockManager.AllocNewBlock("OBJECT"));
2000 :
2001 : /* Use existing center of other block in case we have compressed objects
2002 : and freeze it */
2003 260 : poNewObjBlock->SetCenterFromOtherBlock(m_poCurObjBlock);
2004 :
2005 : /* Coord block will be alloc'd automatically*/
2006 260 : TABMAPCoordBlock *poNewCoordBlock = nullptr;
2007 :
2008 : /*-----------------------------------------------------------------
2009 : * Pick Seeds for each block
2010 : *----------------------------------------------------------------*/
2011 520 : std::vector<TABMAPIndexEntry> asSrcEntries;
2012 260 : asSrcEntries.reserve(apoSrcObjHdrs.size());
2013 9141 : for (const auto &poSrcObjHdrs : apoSrcObjHdrs)
2014 : {
2015 : TABMAPIndexEntry sEntry;
2016 8881 : sEntry.nBlockPtr = 0;
2017 8881 : sEntry.XMin = poSrcObjHdrs->m_nMinX;
2018 8881 : sEntry.YMin = poSrcObjHdrs->m_nMinY;
2019 8881 : sEntry.XMax = poSrcObjHdrs->m_nMaxX;
2020 8881 : sEntry.YMax = poSrcObjHdrs->m_nMaxY;
2021 8881 : asSrcEntries.emplace_back(sEntry);
2022 : }
2023 :
2024 : int nSeed1, nSeed2;
2025 260 : TABMAPIndexBlock::PickSeedsForSplit(
2026 260 : asSrcEntries.data(), static_cast<int>(asSrcEntries.size()), -1,
2027 : poObjHdrToAdd->m_nMinX, poObjHdrToAdd->m_nMinY, poObjHdrToAdd->m_nMaxX,
2028 : poObjHdrToAdd->m_nMaxY, nSeed1, nSeed2);
2029 :
2030 : /*-----------------------------------------------------------------
2031 : * Assign the seeds to their respective block
2032 : *----------------------------------------------------------------*/
2033 : // Insert nSeed1 in this block
2034 260 : if (MoveObjToBlock(apoSrcObjHdrs[nSeed1].get(), poSrcCoordBlock.get(),
2035 260 : m_poCurObjBlock, &m_poCurCoordBlock) <= 0)
2036 : {
2037 0 : return nullptr;
2038 : }
2039 :
2040 : // Move nSeed2 to 2nd block
2041 260 : if (MoveObjToBlock(apoSrcObjHdrs[nSeed2].get(), poSrcCoordBlock.get(),
2042 260 : poNewObjBlock.get(), &poNewCoordBlock) <= 0)
2043 : {
2044 0 : return nullptr;
2045 : }
2046 :
2047 : /*-----------------------------------------------------------------
2048 : * Go through the rest of the entries and assign them to one
2049 : * of the 2 blocks
2050 : *
2051 : * Criteria is minimal area difference.
2052 : * Resolve ties by adding the entry to the block with smaller total
2053 : * area, then to the one with fewer entries, then to either.
2054 : *----------------------------------------------------------------*/
2055 9141 : for (int iEntry = 0; iEntry < static_cast<int>(apoSrcObjHdrs.size());
2056 : iEntry++)
2057 : {
2058 8881 : if (iEntry == nSeed1 || iEntry == nSeed2)
2059 520 : continue;
2060 :
2061 8361 : TABMAPObjHdr *poObjHdr = apoSrcObjHdrs[iEntry].get();
2062 :
2063 8361 : int nObjSize = m_poHeader->GetMapObjectSize(poObjHdr->m_nType);
2064 :
2065 : // If one of the two blocks is almost full then all remaining
2066 : // entries should go to the other block
2067 8361 : if (m_poCurObjBlock->GetNumUnusedBytes() < nObjSize + nSizeOfObjToAdd)
2068 : {
2069 0 : if (MoveObjToBlock(poObjHdr, poSrcCoordBlock.get(),
2070 0 : poNewObjBlock.get(), &poNewCoordBlock) <= 0)
2071 0 : return nullptr;
2072 0 : continue;
2073 : }
2074 8361 : else if (poNewObjBlock->GetNumUnusedBytes() <
2075 8361 : nObjSize + nSizeOfObjToAdd)
2076 : {
2077 0 : if (MoveObjToBlock(poObjHdr, poSrcCoordBlock.get(), m_poCurObjBlock,
2078 0 : &m_poCurCoordBlock) <= 0)
2079 0 : return nullptr;
2080 0 : continue;
2081 : }
2082 :
2083 : // Decide which of the two blocks to put this entry in
2084 : GInt32 nXMin, nYMin, nXMax, nYMax;
2085 8361 : m_poCurObjBlock->GetMBR(nXMin, nYMin, nXMax, nYMax);
2086 8361 : CPLAssert(nXMin <= nXMax);
2087 8361 : double dAreaDiff1 = TABMAPIndexBlock::ComputeAreaDiff(
2088 : nXMin, nYMin, nXMax, nYMax, poObjHdr->m_nMinX, poObjHdr->m_nMinY,
2089 : poObjHdr->m_nMaxX, poObjHdr->m_nMaxY);
2090 :
2091 8361 : poNewObjBlock->GetMBR(nXMin, nYMin, nXMax, nYMax);
2092 8361 : CPLAssert(nXMin <= nXMax);
2093 8361 : double dAreaDiff2 = TABMAPIndexBlock::ComputeAreaDiff(
2094 : nXMin, nYMin, nXMax, nYMax, poObjHdr->m_nMinX, poObjHdr->m_nMinY,
2095 : poObjHdr->m_nMaxX, poObjHdr->m_nMaxY);
2096 :
2097 8361 : if (dAreaDiff1 < dAreaDiff2)
2098 : {
2099 : // This entry stays in this block
2100 3528 : if (MoveObjToBlock(poObjHdr, poSrcCoordBlock.get(), m_poCurObjBlock,
2101 3528 : &m_poCurCoordBlock) <= 0)
2102 0 : return nullptr;
2103 : }
2104 : else
2105 : {
2106 : // This entry goes to new block
2107 4833 : if (MoveObjToBlock(poObjHdr, poSrcCoordBlock.get(),
2108 4833 : poNewObjBlock.get(), &poNewCoordBlock) <= 0)
2109 0 : return nullptr;
2110 : }
2111 : }
2112 :
2113 : /*-----------------------------------------------------------------
2114 : * Delete second coord block if one was created
2115 : * Refs to coord block were kept up to date by MoveObjToBlock()
2116 : * We just need to commit to file and delete the object now.
2117 : *----------------------------------------------------------------*/
2118 260 : if (poNewCoordBlock)
2119 : {
2120 13 : if (poNewCoordBlock->CommitToFile() != 0)
2121 : {
2122 0 : return nullptr;
2123 : }
2124 13 : delete poNewCoordBlock;
2125 : }
2126 :
2127 : /*-----------------------------------------------------------------
2128 : * Release unused coord. data blocks
2129 : *----------------------------------------------------------------*/
2130 260 : if (poSrcCoordBlock)
2131 : {
2132 13 : if (poSrcCoordBlock->GetStartAddress() != nFirstSrcCoordBlock)
2133 : {
2134 0 : if (poSrcCoordBlock->GotoByteInFile(nFirstSrcCoordBlock, TRUE) != 0)
2135 : {
2136 0 : return nullptr;
2137 : }
2138 : }
2139 :
2140 13 : int nNextCoordBlock = poSrcCoordBlock->GetNextCoordBlock();
2141 26 : while (poSrcCoordBlock != nullptr)
2142 : {
2143 : // Mark this block as deleted
2144 13 : if (poSrcCoordBlock->CommitAsDeleted(
2145 13 : m_oBlockManager.GetFirstGarbageBlock()) != 0)
2146 : {
2147 0 : return nullptr;
2148 : }
2149 13 : m_oBlockManager.PushGarbageBlockAsFirst(
2150 13 : poSrcCoordBlock->GetStartAddress());
2151 :
2152 : // Advance to next
2153 13 : if (nNextCoordBlock > 0)
2154 : {
2155 0 : if (poSrcCoordBlock->GotoByteInFile(nNextCoordBlock, TRUE) != 0)
2156 0 : return nullptr;
2157 :
2158 0 : nNextCoordBlock = poSrcCoordBlock->GetNextCoordBlock();
2159 : }
2160 : else
2161 : {
2162 : // end of chain
2163 13 : poSrcCoordBlock.reset();
2164 : }
2165 : }
2166 : }
2167 :
2168 260 : if (poNewObjBlock->CommitToFile() != 0)
2169 0 : return nullptr;
2170 :
2171 260 : return poNewObjBlock.release();
2172 : }
2173 :
2174 : /**********************************************************************
2175 : * TABMAPFile::MoveObjToBlock()
2176 : *
2177 : * Moves an object and its coord data to a new ObjBlock. Used when
2178 : * splitting Obj Blocks.
2179 : *
2180 : * May update the value of ppoCoordBlock if a new coord block had to
2181 : * be created.
2182 : *
2183 : * Returns the address where new object is stored on success, -1 on error.
2184 : **********************************************************************/
2185 8881 : int TABMAPFile::MoveObjToBlock(TABMAPObjHdr *poObjHdr,
2186 : TABMAPCoordBlock *poSrcCoordBlock,
2187 : TABMAPObjectBlock *poDstObjBlock,
2188 : TABMAPCoordBlock **ppoDstCoordBlock)
2189 : {
2190 : /*-----------------------------------------------------------------
2191 : * Copy Coord data if applicable
2192 : * We use a temporary TABFeature object to handle the reading/writing
2193 : * of coord block data.
2194 : *----------------------------------------------------------------*/
2195 8881 : if (m_poHeader->MapObjectUsesCoordBlock(poObjHdr->m_nType))
2196 : {
2197 : TABMAPObjHdrWithCoord *poObjHdrCoord =
2198 182 : cpl::down_cast<TABMAPObjHdrWithCoord *>(poObjHdr);
2199 182 : OGRFeatureDefn *poDummyDefn = new OGRFeatureDefn;
2200 : // Ref count defaults to 0... set it to 1
2201 182 : poDummyDefn->Reference();
2202 :
2203 : TABFeature *poFeature =
2204 182 : TABFeature::CreateFromMapInfoType(poObjHdr->m_nType, poDummyDefn);
2205 :
2206 182 : if (PrepareCoordBlock(poObjHdrCoord->m_nType, poDstObjBlock,
2207 182 : ppoDstCoordBlock) != 0)
2208 0 : return -1;
2209 :
2210 182 : GInt32 nSrcCoordPtr = poObjHdrCoord->m_nCoordBlockPtr;
2211 :
2212 : /* Copy Coord data
2213 : * poObjHdrCoord->m_nCoordBlockPtr will be set by WriteGeometry...
2214 : * We pass second arg to GotoByteInFile() to force reading from file
2215 : * if nSrcCoordPtr is not in current block
2216 : */
2217 182 : if (poSrcCoordBlock->GotoByteInFile(nSrcCoordPtr, TRUE) != 0 ||
2218 182 : poFeature->ReadGeometryFromMAPFile(this, poObjHdr,
2219 : TRUE /* bCoordDataOnly */,
2220 364 : &poSrcCoordBlock) != 0 ||
2221 182 : poFeature->WriteGeometryToMAPFile(this, poObjHdr,
2222 : TRUE /* bCoordDataOnly */,
2223 182 : ppoDstCoordBlock) != 0)
2224 : {
2225 0 : delete poFeature;
2226 0 : delete poDummyDefn;
2227 0 : return -1;
2228 : }
2229 :
2230 : // Update the references to dest coord block in the MAPObjBlock
2231 : // in case new block has been alloc'd since PrepareCoordBlock()
2232 : //
2233 182 : poDstObjBlock->AddCoordBlockRef((*ppoDstCoordBlock)->GetStartAddress());
2234 : /* Cleanup */
2235 182 : delete poFeature;
2236 182 : poDummyDefn->Release();
2237 : }
2238 :
2239 : /*-----------------------------------------------------------------
2240 : * Prepare and Write ObjHdr to this ObjBlock
2241 : *----------------------------------------------------------------*/
2242 8881 : int nObjPtr = poDstObjBlock->PrepareNewObject(poObjHdr);
2243 8881 : if (nObjPtr < 0 || poDstObjBlock->CommitNewObject(poObjHdr) != 0)
2244 : {
2245 0 : CPLError(CE_Failure, CPLE_FileIO,
2246 : "Failed writing object header for feature id %d",
2247 : poObjHdr->m_nId);
2248 0 : return -1;
2249 : }
2250 :
2251 : /*-----------------------------------------------------------------
2252 : * Update .ID Index
2253 : *----------------------------------------------------------------*/
2254 8881 : m_poIdIndex->SetObjPtr(poObjHdr->m_nId, nObjPtr);
2255 :
2256 8881 : return nObjPtr;
2257 : }
2258 :
2259 : /**********************************************************************
2260 : * TABMAPFile::PrepareCoordBlock()
2261 : *
2262 : * Prepare the coord block to receive an object of specified type if one
2263 : * is needed, and update corresponding members in ObjBlock.
2264 : *
2265 : * May update the value of ppoCoordBlock and Returns 0 on success, -1 on error.
2266 : **********************************************************************/
2267 15203 : int TABMAPFile::PrepareCoordBlock(int nObjType, TABMAPObjectBlock *poObjBlock,
2268 : TABMAPCoordBlock **ppoCoordBlock)
2269 : {
2270 :
2271 : /*-----------------------------------------------------------------
2272 : * Prepare Coords block...
2273 : * create a new TABMAPCoordBlock if it was not done yet.
2274 : * Note that in write mode, TABCollections require read/write access
2275 : * to the coord block.
2276 : *----------------------------------------------------------------*/
2277 15203 : if (m_poHeader->MapObjectUsesCoordBlock(nObjType))
2278 : {
2279 489 : if (*ppoCoordBlock == nullptr)
2280 : {
2281 61 : *ppoCoordBlock = new TABMAPCoordBlock(
2282 61 : m_eAccessMode == TABWrite ? TABReadWrite : m_eAccessMode);
2283 : (*ppoCoordBlock)
2284 61 : ->InitNewBlock(m_fp, m_poHeader->m_nRegularBlockSize,
2285 : m_oBlockManager.AllocNewBlock("COORD"));
2286 61 : (*ppoCoordBlock)->SetMAPBlockManagerRef(&m_oBlockManager);
2287 :
2288 : // Set the references to this coord block in the MAPObjBlock
2289 61 : poObjBlock->AddCoordBlockRef((*ppoCoordBlock)->GetStartAddress());
2290 : }
2291 : /* If we are not at the end of the chain of coordinate blocks, then */
2292 : /* reload us */
2293 856 : else if ((*ppoCoordBlock)->GetStartAddress() !=
2294 428 : poObjBlock->GetLastCoordBlockAddress())
2295 : {
2296 3 : TABRawBinBlock *poBlock = TABCreateMAPBlockFromFile(
2297 : m_fp, poObjBlock->GetLastCoordBlockAddress(),
2298 3 : m_poHeader->m_nRegularBlockSize, TRUE, TABReadWrite);
2299 6 : if (poBlock != nullptr &&
2300 3 : poBlock->GetBlockClass() == TABMAP_COORD_BLOCK)
2301 : {
2302 3 : delete *ppoCoordBlock;
2303 3 : *ppoCoordBlock = cpl::down_cast<TABMAPCoordBlock *>(poBlock);
2304 3 : (*ppoCoordBlock)->SetMAPBlockManagerRef(&m_oBlockManager);
2305 : }
2306 : else
2307 : {
2308 0 : delete poBlock;
2309 0 : CPLError(
2310 : CE_Failure, CPLE_FileIO,
2311 : "LoadObjAndCoordBlocks() failed for coord block at %d.",
2312 : poObjBlock->GetLastCoordBlockAddress());
2313 0 : return -1;
2314 : }
2315 : }
2316 :
2317 489 : if ((*ppoCoordBlock)->GetNumUnusedBytes() < 4)
2318 : {
2319 0 : int nNewBlockOffset = m_oBlockManager.AllocNewBlock("COORD");
2320 0 : (*ppoCoordBlock)->SetNextCoordBlock(nNewBlockOffset);
2321 0 : CPL_IGNORE_RET_VAL((*ppoCoordBlock)->CommitToFile());
2322 : (*ppoCoordBlock)
2323 0 : ->InitNewBlock(m_fp, m_poHeader->m_nRegularBlockSize,
2324 : nNewBlockOffset);
2325 0 : poObjBlock->AddCoordBlockRef((*ppoCoordBlock)->GetStartAddress());
2326 : }
2327 :
2328 : // Make sure read/write pointer is at the end of the block
2329 489 : (*ppoCoordBlock)->SeekEnd();
2330 :
2331 489 : if (CPLGetLastErrorType() == CE_Failure)
2332 0 : return -1;
2333 : }
2334 :
2335 15203 : return 0;
2336 : }
2337 :
2338 : /**********************************************************************
2339 : * TABMAPFile::GetCurObjType()
2340 : *
2341 : * Return the MapInfo object type of the object that the m_poCurObjBlock
2342 : * is pointing to. This value is set after a call to MoveToObjId().
2343 : *
2344 : * Returns a value >= 0 on success, -1 on error.
2345 : **********************************************************************/
2346 1204490 : TABGeomType TABMAPFile::GetCurObjType()
2347 : {
2348 1204490 : return m_nCurObjType;
2349 : }
2350 :
2351 : /**********************************************************************
2352 : * TABMAPFile::GetCurObjId()
2353 : *
2354 : * Return the MapInfo object id of the object that the m_poCurObjBlock
2355 : * is pointing to. This value is set after a call to MoveToObjId().
2356 : *
2357 : * Returns a value >= 0 on success, -1 on error.
2358 : **********************************************************************/
2359 525991 : int TABMAPFile::GetCurObjId()
2360 : {
2361 525991 : return m_nCurObjId;
2362 : }
2363 :
2364 : /**********************************************************************
2365 : * TABMAPFile::GetCurObjBlock()
2366 : *
2367 : * Return the m_poCurObjBlock. If MoveToObjId() has previously been
2368 : * called then m_poCurObjBlock points to the beginning of the current
2369 : * object data.
2370 : *
2371 : * Returns a reference to an object owned by this TABMAPFile object, or
2372 : * NULL on error.
2373 : **********************************************************************/
2374 525991 : TABMAPObjectBlock *TABMAPFile::GetCurObjBlock()
2375 : {
2376 525991 : return m_poCurObjBlock;
2377 : }
2378 :
2379 : /**********************************************************************
2380 : * TABMAPFile::GetCurCoordBlock()
2381 : *
2382 : * Return the m_poCurCoordBlock. This function should be used after
2383 : * PrepareNewObj() to get the reference to the coord block that has
2384 : * just been initialized.
2385 : *
2386 : * Returns a reference to an object owned by this TABMAPFile object, or
2387 : * NULL on error.
2388 : **********************************************************************/
2389 307 : TABMAPCoordBlock *TABMAPFile::GetCurCoordBlock()
2390 : {
2391 307 : return m_poCurCoordBlock;
2392 : }
2393 :
2394 : /**********************************************************************
2395 : * TABMAPFile::GetCoordBlock()
2396 : *
2397 : * Return a TABMAPCoordBlock object ready to read coordinates from it.
2398 : * The block that contains nFileOffset will automatically be
2399 : * loaded, and if nFileOffset is the beginning of a new block then the
2400 : * pointer will be moved to the beginning of the data.
2401 : *
2402 : * The contents of the returned object is only valid until the next call
2403 : * to GetCoordBlock().
2404 : *
2405 : * Returns a reference to an object owned by this TABMAPFile object, or
2406 : * NULL on error.
2407 : **********************************************************************/
2408 2238 : TABMAPCoordBlock *TABMAPFile::GetCoordBlock(int nFileOffset)
2409 : {
2410 2238 : if (m_poCurCoordBlock == nullptr)
2411 : {
2412 22 : m_poCurCoordBlock = new TABMAPCoordBlock(m_eAccessMode);
2413 22 : m_poCurCoordBlock->InitNewBlock(m_fp, m_poHeader->m_nRegularBlockSize);
2414 22 : m_poCurCoordBlock->SetMAPBlockManagerRef(&m_oBlockManager);
2415 : }
2416 :
2417 : /*-----------------------------------------------------------------
2418 : * Use GotoByteInFile() to go to the requested location. This will
2419 : * force loading the block if necessary and reading its header.
2420 : * If nFileOffset is at the beginning of the requested block, then
2421 : * we make sure to move the read pointer past the 8 bytes header
2422 : * to be ready to read coordinates data
2423 : *----------------------------------------------------------------*/
2424 2238 : if (m_poCurCoordBlock->GotoByteInFile(nFileOffset, TRUE) != 0)
2425 : {
2426 : // Failed... an error has already been reported.
2427 0 : return nullptr;
2428 : }
2429 :
2430 2238 : if (nFileOffset % m_poHeader->m_nRegularBlockSize == 0)
2431 0 : m_poCurCoordBlock->GotoByteInBlock(8); // Skip Header
2432 :
2433 2238 : return m_poCurCoordBlock;
2434 : }
2435 :
2436 : /**********************************************************************
2437 : * TABMAPFile::GetHeaderBlock()
2438 : *
2439 : * Return a reference to the MAP file's header block.
2440 : *
2441 : * The returned pointer is a reference to an object owned by this TABMAPFile
2442 : * object and should not be deleted by the caller.
2443 : *
2444 : * Return NULL if file has not been opened yet.
2445 : **********************************************************************/
2446 2673 : TABMAPHeaderBlock *TABMAPFile::GetHeaderBlock()
2447 : {
2448 2673 : return m_poHeader;
2449 : }
2450 :
2451 : /**********************************************************************
2452 : * TABMAPFile::GetIDFileRef()
2453 : *
2454 : * Return a reference to the .ID file attached to this .MAP file
2455 : *
2456 : * The returned pointer is a reference to an object owned by this TABMAPFile
2457 : * object and should not be deleted by the caller.
2458 : *
2459 : * Return NULL if file has not been opened yet.
2460 : **********************************************************************/
2461 0 : TABIDFile *TABMAPFile::GetIDFileRef()
2462 : {
2463 0 : return m_poIdIndex;
2464 : }
2465 :
2466 : /**********************************************************************
2467 : * TABMAPFile::GetIndexBlock()
2468 : *
2469 : * Return a reference to the requested index or object block..
2470 : *
2471 : * Ownership of the returned block is turned over to the caller, who should
2472 : * delete it when no longer needed. The type of the block can be determined
2473 : * with the GetBlockType() method.
2474 : *
2475 : * @param nFileOffset the offset in the map file of the spatial index
2476 : * block or object block to load.
2477 : *
2478 : * @return The requested TABMAPIndexBlock, TABMAPObjectBlock or NULL if the
2479 : * read fails for some reason.
2480 : **********************************************************************/
2481 38956 : TABRawBinBlock *TABMAPFile::GetIndexObjectBlock(int nFileOffset)
2482 : {
2483 : /*----------------------------------------------------------------
2484 : * Read from the file
2485 : *---------------------------------------------------------------*/
2486 : GByte *pabyData =
2487 38956 : static_cast<GByte *>(CPLMalloc(m_poHeader->m_nRegularBlockSize));
2488 :
2489 77912 : if (VSIFSeekL(m_fp, nFileOffset, SEEK_SET) != 0 ||
2490 77912 : static_cast<int>(VSIFReadL(pabyData, sizeof(GByte),
2491 38956 : m_poHeader->m_nRegularBlockSize, m_fp)) !=
2492 38956 : m_poHeader->m_nRegularBlockSize)
2493 : {
2494 0 : CPLError(CE_Failure, CPLE_FileIO,
2495 : "GetIndexBlock() failed reading %d bytes at offset %d.",
2496 0 : m_poHeader->m_nRegularBlockSize, nFileOffset);
2497 0 : CPLFree(pabyData);
2498 0 : return nullptr;
2499 : }
2500 :
2501 : /* -------------------------------------------------------------------- */
2502 : /* Create and initialize depending on the block type. */
2503 : /* -------------------------------------------------------------------- */
2504 38956 : int nBlockType = pabyData[0];
2505 38956 : TABRawBinBlock *poBlock = nullptr;
2506 :
2507 38956 : if (nBlockType == TABMAP_INDEX_BLOCK)
2508 : {
2509 23770 : TABMAPIndexBlock *poIndexBlock = new TABMAPIndexBlock(m_eAccessMode);
2510 23770 : poBlock = poIndexBlock;
2511 23770 : poIndexBlock->SetMAPBlockManagerRef(&m_oBlockManager);
2512 : }
2513 : else
2514 15186 : poBlock = new TABMAPObjectBlock(m_eAccessMode);
2515 :
2516 38956 : poBlock->InitBlockFromData(pabyData, m_poHeader->m_nRegularBlockSize,
2517 38956 : m_poHeader->m_nRegularBlockSize, FALSE, m_fp,
2518 38956 : nFileOffset);
2519 :
2520 38956 : return poBlock;
2521 : }
2522 :
2523 : /**********************************************************************
2524 : * TABMAPFile::InitDrawingTools()
2525 : *
2526 : * Init the drawing tools for this file.
2527 : *
2528 : * In Read mode, this will load the drawing tools from the file.
2529 : *
2530 : * In Write mode, this function will init an empty the tool def table.
2531 : *
2532 : * Returns 0 on success, -1 on error.
2533 : **********************************************************************/
2534 1325 : int TABMAPFile::InitDrawingTools()
2535 : {
2536 1325 : int nStatus = 0;
2537 :
2538 1325 : if (m_poHeader == nullptr)
2539 0 : return -1; // File not opened yet!
2540 :
2541 : /*-------------------------------------------------------------
2542 : * We want to perform this initialization only once
2543 : *------------------------------------------------------------*/
2544 1325 : if (m_poToolDefTable != nullptr)
2545 0 : return 0;
2546 :
2547 : /*-------------------------------------------------------------
2548 : * Create a new ToolDefTable... no more initialization is required
2549 : * unless we want to read tool blocks from file.
2550 : *------------------------------------------------------------*/
2551 1325 : m_poToolDefTable = new TABToolDefTable;
2552 :
2553 1325 : if ((m_eAccessMode == TABRead || m_eAccessMode == TABReadWrite) &&
2554 1238 : m_poHeader->m_nFirstToolBlock != 0)
2555 : {
2556 1211 : TABMAPToolBlock *poBlock = new TABMAPToolBlock(TABRead);
2557 1211 : poBlock->InitNewBlock(m_fp, m_poHeader->m_nRegularBlockSize);
2558 :
2559 : /*-------------------------------------------------------------
2560 : * Use GotoByteInFile() to go to the first block's location. This will
2561 : * force loading the block if necessary and reading its header.
2562 : * Also make sure to move the read pointer past the 8 bytes header
2563 : * to be ready to read drawing tools data
2564 : *------------------------------------------------------------*/
2565 1211 : if (poBlock->GotoByteInFile(m_poHeader->m_nFirstToolBlock) != 0)
2566 : {
2567 : // Failed... an error has already been reported.
2568 0 : delete poBlock;
2569 0 : return -1;
2570 : }
2571 :
2572 1211 : poBlock->GotoByteInBlock(8);
2573 :
2574 1211 : nStatus = m_poToolDefTable->ReadAllToolDefs(poBlock);
2575 1211 : delete poBlock;
2576 : }
2577 :
2578 1325 : return nStatus;
2579 : }
2580 :
2581 : /**********************************************************************
2582 : * TABMAPFile::CommitDrawingTools()
2583 : *
2584 : * Write the drawing tools for this file.
2585 : *
2586 : * This function applies only to write access mode.
2587 : *
2588 : * Returns 0 on success, -1 on error.
2589 : **********************************************************************/
2590 1201 : int TABMAPFile::CommitDrawingTools()
2591 : {
2592 1201 : int nStatus = 0;
2593 :
2594 1201 : if (m_eAccessMode == TABRead || m_poHeader == nullptr)
2595 : {
2596 0 : CPLError(
2597 : CE_Failure, CPLE_AssertionFailed,
2598 : "CommitDrawingTools() failed: file not opened for write access.");
2599 0 : return -1;
2600 : }
2601 :
2602 2366 : if (m_poToolDefTable == nullptr ||
2603 1165 : (m_poToolDefTable->GetNumPen() + m_poToolDefTable->GetNumBrushes() +
2604 1165 : m_poToolDefTable->GetNumFonts() + m_poToolDefTable->GetNumSymbols()) ==
2605 : 0)
2606 : {
2607 36 : return 0; // Nothing to do!
2608 : }
2609 :
2610 : /*-------------------------------------------------------------
2611 : * Create a new TABMAPToolBlock and update header fields
2612 : *------------------------------------------------------------*/
2613 1165 : TABMAPToolBlock *poBlock = new TABMAPToolBlock(m_eAccessMode);
2614 1165 : if (m_poHeader->m_nFirstToolBlock != 0)
2615 1075 : poBlock->InitNewBlock(m_fp, m_poHeader->m_nRegularBlockSize,
2616 1075 : m_poHeader->m_nFirstToolBlock);
2617 : else
2618 90 : poBlock->InitNewBlock(m_fp, m_poHeader->m_nRegularBlockSize,
2619 : m_oBlockManager.AllocNewBlock("TOOL"));
2620 1165 : poBlock->SetMAPBlockManagerRef(&m_oBlockManager);
2621 :
2622 1165 : m_poHeader->m_nFirstToolBlock = poBlock->GetStartAddress();
2623 :
2624 1165 : m_poHeader->m_numPenDefs =
2625 1165 : static_cast<GByte>(m_poToolDefTable->GetNumPen());
2626 1165 : m_poHeader->m_numBrushDefs =
2627 1165 : static_cast<GByte>(m_poToolDefTable->GetNumBrushes());
2628 1165 : m_poHeader->m_numFontDefs =
2629 1165 : static_cast<GByte>(m_poToolDefTable->GetNumFonts());
2630 1165 : m_poHeader->m_numSymbolDefs =
2631 1165 : static_cast<GByte>(m_poToolDefTable->GetNumSymbols());
2632 :
2633 : /*-------------------------------------------------------------
2634 : * Do the actual work and delete poBlock
2635 : * (Note that poBlock will have already been committed to the file
2636 : * by WriteAllToolDefs() )
2637 : *------------------------------------------------------------*/
2638 1165 : nStatus = m_poToolDefTable->WriteAllToolDefs(poBlock);
2639 :
2640 1165 : m_poHeader->m_numMapToolBlocks =
2641 1165 : static_cast<GByte>(poBlock->GetNumBlocksInChain());
2642 :
2643 1165 : delete poBlock;
2644 :
2645 1165 : return nStatus;
2646 : }
2647 :
2648 : /**********************************************************************
2649 : * TABMAPFile::ReadPenDef()
2650 : *
2651 : * Fill the TABPenDef structure with the definition of the specified pen
2652 : * index... (1-based pen index)
2653 : *
2654 : * If nPenIndex==0 or is invalid, then the structure is cleared.
2655 : *
2656 : * Returns 0 on success, -1 on error (i.e. Pen not found).
2657 : **********************************************************************/
2658 2279 : int TABMAPFile::ReadPenDef(int nPenIndex, TABPenDef *psDef)
2659 : {
2660 2279 : if (m_poToolDefTable == nullptr && InitDrawingTools() != 0)
2661 0 : return -1;
2662 :
2663 2279 : TABPenDef *psTmp = nullptr;
2664 4558 : if (psDef && m_poToolDefTable &&
2665 2279 : (psTmp = m_poToolDefTable->GetPenDefRef(nPenIndex)) != nullptr)
2666 : {
2667 2277 : *psDef = *psTmp;
2668 : }
2669 2 : else if (psDef)
2670 : {
2671 : /* Init to MapInfo default */
2672 : static const TABPenDef csDefaultPen = MITAB_PEN_DEFAULT;
2673 2 : *psDef = csDefaultPen;
2674 2 : return -1;
2675 : }
2676 2277 : return 0;
2677 : }
2678 :
2679 : /**********************************************************************
2680 : * TABMAPFile::WritePenDef()
2681 : *
2682 : * Write a Pen Tool to the map file and return the pen index that has
2683 : * been attributed to this Pen tool definition, or -1 if something went
2684 : * wrong
2685 : *
2686 : * Note that the returned index is a 1-based index. A value of 0
2687 : * indicates "none" in MapInfo.
2688 :
2689 : * Returns a value >= 0 on success, -1 on error
2690 : **********************************************************************/
2691 330 : int TABMAPFile::WritePenDef(TABPenDef *psDef)
2692 : {
2693 660 : if (psDef == nullptr ||
2694 660 : (m_poToolDefTable == nullptr && InitDrawingTools() != 0) ||
2695 330 : m_poToolDefTable == nullptr)
2696 : {
2697 0 : return -1;
2698 : }
2699 :
2700 330 : return m_poToolDefTable->AddPenDefRef(psDef);
2701 : }
2702 :
2703 : /**********************************************************************
2704 : * TABMAPFile::ReadBrushDef()
2705 : *
2706 : * Fill the TABBrushDef structure with the definition of the specified Brush
2707 : * index... (1-based Brush index)
2708 : *
2709 : * If nBrushIndex==0 or is invalid, then the structure is cleared.
2710 : *
2711 : * Returns 0 on success, -1 on error (i.e. Brush not found).
2712 : **********************************************************************/
2713 526 : int TABMAPFile::ReadBrushDef(int nBrushIndex, TABBrushDef *psDef)
2714 : {
2715 526 : if (m_poToolDefTable == nullptr && InitDrawingTools() != 0)
2716 0 : return -1;
2717 :
2718 526 : TABBrushDef *psTmp = nullptr;
2719 1052 : if (psDef && m_poToolDefTable &&
2720 526 : (psTmp = m_poToolDefTable->GetBrushDefRef(nBrushIndex)) != nullptr)
2721 : {
2722 521 : *psDef = *psTmp;
2723 : }
2724 5 : else if (psDef)
2725 : {
2726 : /* Init to MapInfo default */
2727 : static const TABBrushDef csDefaultBrush = MITAB_BRUSH_DEFAULT;
2728 5 : *psDef = csDefaultBrush;
2729 5 : return -1;
2730 : }
2731 521 : return 0;
2732 : }
2733 :
2734 : /**********************************************************************
2735 : * TABMAPFile::WriteBrushDef()
2736 : *
2737 : * Write a Brush Tool to the map file and return the Brush index that has
2738 : * been attributed to this Brush tool definition, or -1 if something went
2739 : * wrong
2740 : *
2741 : * Note that the returned index is a 1-based index. A value of 0
2742 : * indicates "none" in MapInfo.
2743 :
2744 : * Returns a value >= 0 on success, -1 on error
2745 : **********************************************************************/
2746 88 : int TABMAPFile::WriteBrushDef(TABBrushDef *psDef)
2747 : {
2748 176 : if (psDef == nullptr ||
2749 176 : (m_poToolDefTable == nullptr && InitDrawingTools() != 0) ||
2750 88 : m_poToolDefTable == nullptr)
2751 : {
2752 0 : return -1;
2753 : }
2754 :
2755 88 : return m_poToolDefTable->AddBrushDefRef(psDef);
2756 : }
2757 :
2758 : /**********************************************************************
2759 : * TABMAPFile::ReadFontDef()
2760 : *
2761 : * Fill the TABFontDef structure with the definition of the specified Font
2762 : * index... (1-based Font index)
2763 : *
2764 : * If nFontIndex==0 or is invalid, then the structure is cleared.
2765 : *
2766 : * Returns 0 on success, -1 on error (i.e. Font not found).
2767 : **********************************************************************/
2768 20 : int TABMAPFile::ReadFontDef(int nFontIndex, TABFontDef *psDef)
2769 : {
2770 20 : if (m_poToolDefTable == nullptr && InitDrawingTools() != 0)
2771 0 : return -1;
2772 :
2773 20 : TABFontDef *psTmp = nullptr;
2774 40 : if (psDef && m_poToolDefTable &&
2775 20 : (psTmp = m_poToolDefTable->GetFontDefRef(nFontIndex)) != nullptr)
2776 : {
2777 20 : *psDef = *psTmp;
2778 : }
2779 0 : else if (psDef)
2780 : {
2781 : /* Init to MapInfo default */
2782 : static const TABFontDef csDefaultFont = MITAB_FONT_DEFAULT;
2783 0 : *psDef = csDefaultFont;
2784 0 : return -1;
2785 : }
2786 20 : return 0;
2787 : }
2788 :
2789 : /**********************************************************************
2790 : * TABMAPFile::WriteFontDef()
2791 : *
2792 : * Write a Font Tool to the map file and return the Font index that has
2793 : * been attributed to this Font tool definition, or -1 if something went
2794 : * wrong
2795 : *
2796 : * Note that the returned index is a 1-based index. A value of 0
2797 : * indicates "none" in MapInfo.
2798 :
2799 : * Returns a value >= 0 on success, -1 on error
2800 : **********************************************************************/
2801 8 : int TABMAPFile::WriteFontDef(TABFontDef *psDef)
2802 : {
2803 16 : if (psDef == nullptr ||
2804 16 : (m_poToolDefTable == nullptr && InitDrawingTools() != 0) ||
2805 8 : m_poToolDefTable == nullptr)
2806 : {
2807 0 : return -1;
2808 : }
2809 :
2810 8 : return m_poToolDefTable->AddFontDefRef(psDef);
2811 : }
2812 :
2813 : /**********************************************************************
2814 : * TABMAPFile::ReadSymbolDef()
2815 : *
2816 : * Fill the TABSymbolDef structure with the definition of the specified Symbol
2817 : * index... (1-based Symbol index)
2818 : *
2819 : * If nSymbolIndex==0 or is invalid, then the structure is cleared.
2820 : *
2821 : * Returns 0 on success, -1 on error (i.e. Symbol not found).
2822 : **********************************************************************/
2823 523559 : int TABMAPFile::ReadSymbolDef(int nSymbolIndex, TABSymbolDef *psDef)
2824 : {
2825 523559 : if (m_poToolDefTable == nullptr && InitDrawingTools() != 0)
2826 0 : return -1;
2827 :
2828 523559 : TABSymbolDef *psTmp = nullptr;
2829 1047120 : if (psDef && m_poToolDefTable &&
2830 523559 : (psTmp = m_poToolDefTable->GetSymbolDefRef(nSymbolIndex)) != nullptr)
2831 : {
2832 523559 : *psDef = *psTmp;
2833 : }
2834 0 : else if (psDef)
2835 : {
2836 : /* Init to MapInfo default */
2837 : static const TABSymbolDef csDefaultSymbol = MITAB_SYMBOL_DEFAULT;
2838 0 : *psDef = csDefaultSymbol;
2839 0 : return -1;
2840 : }
2841 523559 : return 0;
2842 : }
2843 :
2844 : /**********************************************************************
2845 : * TABMAPFile::WriteSymbolDef()
2846 : *
2847 : * Write a Symbol Tool to the map file and return the Symbol index that has
2848 : * been attributed to this Symbol tool definition, or -1 if something went
2849 : * wrong
2850 : *
2851 : * Note that the returned index is a 1-based index. A value of 0
2852 : * indicates "none" in MapInfo.
2853 :
2854 : * Returns a value >= 0 on success, -1 on error
2855 : **********************************************************************/
2856 14689 : int TABMAPFile::WriteSymbolDef(TABSymbolDef *psDef)
2857 : {
2858 29378 : if (psDef == nullptr ||
2859 29378 : (m_poToolDefTable == nullptr && InitDrawingTools() != 0) ||
2860 14689 : m_poToolDefTable == nullptr)
2861 : {
2862 0 : return -1;
2863 : }
2864 :
2865 14689 : return m_poToolDefTable->AddSymbolDefRef(psDef);
2866 : }
2867 :
2868 133250 : static void ORDER_MIN_MAX(double &min, double &max)
2869 : {
2870 133250 : if (max < min)
2871 1 : std::swap(min, max);
2872 133250 : }
2873 :
2874 133250 : static void ORDER_MIN_MAX(int &min, int &max)
2875 : {
2876 133250 : if (max < min)
2877 0 : std::swap(min, max);
2878 133250 : }
2879 :
2880 : /**********************************************************************
2881 : * TABMAPFile::SetCoordFilter()
2882 : *
2883 : * Set the MBR of the area of interest... only objects that at least
2884 : * overlap with that area will be returned.
2885 : *
2886 : * @param sMin minimum x/y the file's projection coord.
2887 : * @param sMax maximum x/y the file's projection coord.
2888 : **********************************************************************/
2889 31364 : void TABMAPFile::SetCoordFilter(TABVertex sMin, TABVertex sMax)
2890 : {
2891 31364 : m_sMinFilter = sMin;
2892 31364 : m_sMaxFilter = sMax;
2893 :
2894 31364 : Coordsys2Int(sMin.x, sMin.y, m_XMinFilter, m_YMinFilter, TRUE);
2895 31364 : Coordsys2Int(sMax.x, sMax.y, m_XMaxFilter, m_YMaxFilter, TRUE);
2896 :
2897 31364 : ORDER_MIN_MAX(m_XMinFilter, m_XMaxFilter);
2898 31364 : ORDER_MIN_MAX(m_YMinFilter, m_YMaxFilter);
2899 31364 : ORDER_MIN_MAX(m_sMinFilter.x, m_sMaxFilter.x);
2900 31364 : ORDER_MIN_MAX(m_sMinFilter.y, m_sMaxFilter.y);
2901 31364 : }
2902 :
2903 : /**********************************************************************
2904 : * TABMAPFile::ResetCoordFilter()
2905 : *
2906 : * Reset the MBR of the area of interest to be the extents as defined
2907 : * in the header.
2908 : **********************************************************************/
2909 :
2910 35261 : void TABMAPFile::ResetCoordFilter()
2911 :
2912 : {
2913 35261 : m_XMinFilter = m_poHeader->m_nXMin;
2914 35261 : m_YMinFilter = m_poHeader->m_nYMin;
2915 35261 : m_XMaxFilter = m_poHeader->m_nXMax;
2916 35261 : m_YMaxFilter = m_poHeader->m_nYMax;
2917 35261 : Int2Coordsys(m_XMinFilter, m_YMinFilter, m_sMinFilter.x, m_sMinFilter.y);
2918 35261 : Int2Coordsys(m_XMaxFilter, m_YMaxFilter, m_sMaxFilter.x, m_sMaxFilter.y);
2919 :
2920 35261 : ORDER_MIN_MAX(m_XMinFilter, m_XMaxFilter);
2921 35261 : ORDER_MIN_MAX(m_YMinFilter, m_YMaxFilter);
2922 35261 : ORDER_MIN_MAX(m_sMinFilter.x, m_sMaxFilter.x);
2923 35261 : ORDER_MIN_MAX(m_sMinFilter.y, m_sMaxFilter.y);
2924 35261 : }
2925 :
2926 : /**********************************************************************
2927 : * TABMAPFile::GetCoordFilter()
2928 : *
2929 : * Get the MBR of the area of interest, as previously set by
2930 : * SetCoordFilter().
2931 : *
2932 : * @param sMin vertex into which the minimum x/y values put in coordsys space.
2933 : * @param sMax vertex into which the maximum x/y values put in coordsys space.
2934 : **********************************************************************/
2935 31399 : void TABMAPFile::GetCoordFilter(TABVertex &sMin, TABVertex &sMax) const
2936 : {
2937 31399 : sMin = m_sMinFilter;
2938 31399 : sMax = m_sMaxFilter;
2939 31399 : }
2940 :
2941 : /**********************************************************************
2942 : * TABMAPFile::CommitSpatialIndex()
2943 : *
2944 : * Write the spatial index blocks tree for this file.
2945 : *
2946 : * This function applies only to write access mode.
2947 : *
2948 : * Returns 0 on success, -1 on error.
2949 : **********************************************************************/
2950 1201 : int TABMAPFile::CommitSpatialIndex()
2951 : {
2952 1201 : if (m_eAccessMode == TABRead || m_poHeader == nullptr)
2953 : {
2954 0 : CPLError(
2955 : CE_Failure, CPLE_AssertionFailed,
2956 : "CommitSpatialIndex() failed: file not opened for write access.");
2957 0 : return -1;
2958 : }
2959 :
2960 1201 : if (m_poSpIndex == nullptr)
2961 : {
2962 36 : return 0; // Nothing to do!
2963 : }
2964 :
2965 : /*-------------------------------------------------------------
2966 : * Update header fields and commit index block
2967 : * (its children will be recursively committed as well)
2968 : *------------------------------------------------------------*/
2969 : // Add 1 to Spatial Index Depth to account to the MapObjectBlocks
2970 1165 : const int nNextDepth = m_poSpIndex->GetCurMaxDepth() + 1;
2971 1165 : m_poHeader->m_nMaxSpIndexDepth = static_cast<GByte>(
2972 1165 : std::max(static_cast<int>(m_poHeader->m_nMaxSpIndexDepth), nNextDepth));
2973 :
2974 1165 : m_poSpIndex->GetMBR(m_poHeader->m_nXMin, m_poHeader->m_nYMin,
2975 1165 : m_poHeader->m_nXMax, m_poHeader->m_nYMax);
2976 :
2977 1165 : return m_poSpIndex->CommitToFile();
2978 : }
2979 :
2980 : /**********************************************************************
2981 : * TABMAPFile::GetMinTABFileVersion()
2982 : *
2983 : * Returns the minimum TAB file version number that can contain all the
2984 : * objects stored in this file.
2985 : **********************************************************************/
2986 144 : int TABMAPFile::GetMinTABFileVersion()
2987 : {
2988 144 : int nToolVersion = 0;
2989 :
2990 144 : if (m_poToolDefTable)
2991 108 : nToolVersion = m_poToolDefTable->GetMinVersionNumber();
2992 :
2993 144 : return std::max(nToolVersion, m_nMinTABVersion);
2994 : }
2995 :
2996 13 : const CPLString &TABMAPFile::GetEncoding() const
2997 : {
2998 13 : return m_osEncoding;
2999 : }
3000 :
3001 2 : void TABMAPFile::SetEncoding(const CPLString &osEncoding)
3002 : {
3003 2 : m_osEncoding = osEncoding;
3004 2 : }
3005 :
3006 1088790 : bool TABMAPFile::IsValidObjType(int nObjType)
3007 : {
3008 1088790 : switch (nObjType)
3009 : {
3010 1088790 : case TAB_GEOM_NONE:
3011 : case TAB_GEOM_SYMBOL_C:
3012 : case TAB_GEOM_SYMBOL:
3013 : case TAB_GEOM_LINE_C:
3014 : case TAB_GEOM_LINE:
3015 : case TAB_GEOM_PLINE_C:
3016 : case TAB_GEOM_PLINE:
3017 : case TAB_GEOM_ARC_C:
3018 : case TAB_GEOM_ARC:
3019 : case TAB_GEOM_REGION_C:
3020 : case TAB_GEOM_REGION:
3021 : case TAB_GEOM_TEXT_C:
3022 : case TAB_GEOM_TEXT:
3023 : case TAB_GEOM_RECT_C:
3024 : case TAB_GEOM_RECT:
3025 : case TAB_GEOM_ROUNDRECT_C:
3026 : case TAB_GEOM_ROUNDRECT:
3027 : case TAB_GEOM_ELLIPSE_C:
3028 : case TAB_GEOM_ELLIPSE:
3029 : case TAB_GEOM_MULTIPLINE_C:
3030 : case TAB_GEOM_MULTIPLINE:
3031 : case TAB_GEOM_FONTSYMBOL_C:
3032 : case TAB_GEOM_FONTSYMBOL:
3033 : case TAB_GEOM_CUSTOMSYMBOL_C:
3034 : case TAB_GEOM_CUSTOMSYMBOL:
3035 : case TAB_GEOM_V450_REGION_C:
3036 : case TAB_GEOM_V450_REGION:
3037 : case TAB_GEOM_V450_MULTIPLINE_C:
3038 : case TAB_GEOM_V450_MULTIPLINE:
3039 : case TAB_GEOM_MULTIPOINT_C:
3040 : case TAB_GEOM_MULTIPOINT:
3041 : case TAB_GEOM_COLLECTION_C:
3042 : case TAB_GEOM_COLLECTION:
3043 : case TAB_GEOM_UNKNOWN1_C:
3044 : case TAB_GEOM_UNKNOWN1:
3045 : case TAB_GEOM_V800_REGION_C:
3046 : case TAB_GEOM_V800_REGION:
3047 : case TAB_GEOM_V800_MULTIPLINE_C:
3048 : case TAB_GEOM_V800_MULTIPLINE:
3049 : case TAB_GEOM_V800_MULTIPOINT_C:
3050 : case TAB_GEOM_V800_MULTIPOINT:
3051 : case TAB_GEOM_V800_COLLECTION_C:
3052 : case TAB_GEOM_V800_COLLECTION:
3053 1088790 : return true;
3054 :
3055 0 : default:
3056 0 : return false;
3057 : }
3058 : }
3059 :
3060 : /**********************************************************************
3061 : * TABMAPFile::Dump()
3062 : *
3063 : * Dump block contents... available only in DEBUG mode.
3064 : **********************************************************************/
3065 : #ifdef DEBUG
3066 :
3067 0 : void TABMAPFile::Dump(FILE *fpOut /*=NULL*/)
3068 : {
3069 0 : if (fpOut == nullptr)
3070 0 : fpOut = stdout;
3071 :
3072 0 : fprintf(fpOut, "----- TABMAPFile::Dump() -----\n");
3073 :
3074 0 : if (m_fp == nullptr)
3075 : {
3076 0 : fprintf(fpOut, "File is not opened.\n");
3077 : }
3078 : else
3079 : {
3080 0 : fprintf(fpOut, "File is opened: %s\n", m_pszFname);
3081 0 : fprintf(fpOut, "Coordsys filter = (%g,%g)-(%g,%g)\n", m_sMinFilter.x,
3082 : m_sMinFilter.y, m_sMaxFilter.x, m_sMaxFilter.y);
3083 0 : fprintf(fpOut, "Int coord filter = (%d,%d)-(%d,%d)\n", m_XMinFilter,
3084 : m_YMinFilter, m_XMaxFilter, m_YMaxFilter);
3085 :
3086 0 : fprintf(fpOut, "\nFile Header follows ...\n\n");
3087 0 : m_poHeader->Dump(fpOut);
3088 0 : fprintf(fpOut, "... end of file header.\n\n");
3089 :
3090 0 : fprintf(fpOut, "Associated .ID file ...\n\n");
3091 0 : m_poIdIndex->Dump(fpOut);
3092 0 : fprintf(fpOut, "... end of ID file dump.\n\n");
3093 : }
3094 :
3095 0 : fflush(fpOut);
3096 0 : }
3097 :
3098 : #endif // DEBUG
3099 :
3100 : /**********************************************************************
3101 : * TABMAPFile::DumpSpatialIndexToMIF()
3102 : *
3103 : * Dump the spatial index tree... available only in DEBUG mode.
3104 : **********************************************************************/
3105 : #ifdef DEBUG
3106 :
3107 0 : void TABMAPFile::DumpSpatialIndexToMIF(TABMAPIndexBlock *poNode, FILE *fpMIF,
3108 : FILE *fpMID, int nParentId /*=-1*/,
3109 : int nIndexInNode /*=-1*/,
3110 : int nCurDepth /*=0*/,
3111 : int nMaxDepth /*=-1*/)
3112 : {
3113 0 : if (poNode == nullptr)
3114 : {
3115 0 : if (m_poHeader && m_poHeader->m_nFirstIndexBlock != 0)
3116 : {
3117 : TABRawBinBlock *poBlock =
3118 0 : GetIndexObjectBlock(m_poHeader->m_nFirstIndexBlock);
3119 0 : if (poBlock && poBlock->GetBlockType() == TABMAP_INDEX_BLOCK)
3120 0 : poNode = cpl::down_cast<TABMAPIndexBlock *>(poBlock);
3121 : }
3122 :
3123 0 : if (poNode == nullptr)
3124 0 : return;
3125 : }
3126 :
3127 : /*-------------------------------------------------------------
3128 : * Report info on current tree node
3129 : *------------------------------------------------------------*/
3130 0 : const int numEntries = poNode->GetNumEntries();
3131 0 : GInt32 nXMin = 0;
3132 0 : GInt32 nYMin = 0;
3133 0 : GInt32 nXMax = 0;
3134 0 : GInt32 nYMax = 0;
3135 :
3136 0 : poNode->RecomputeMBR();
3137 0 : poNode->GetMBR(nXMin, nYMin, nXMax, nYMax);
3138 :
3139 0 : double dXMin = 0.0;
3140 0 : double dYMin = 0.0;
3141 0 : double dXMax = 0.0;
3142 0 : double dYMax = 0.0;
3143 0 : Int2Coordsys(nXMin, nYMin, dXMin, dYMin);
3144 0 : Int2Coordsys(nXMax, nYMax, dXMax, dYMax);
3145 :
3146 0 : VSIFPrintf(fpMIF, "RECT %g %g %g %g\n", dXMin, dYMin, dXMax, dYMax);
3147 0 : VSIFPrintf(fpMIF, " Brush(1, 0)\n"); /* No fill */
3148 :
3149 0 : VSIFPrintf(fpMID, "%d,%d,%d,%d,%g,%d,%d,%d,%d\n", poNode->GetStartAddress(),
3150 : nParentId, nIndexInNode, nCurDepth,
3151 0 : MITAB_AREA(nXMin, nYMin, nXMax, nYMax), nXMin, nYMin, nXMax,
3152 : nYMax);
3153 :
3154 0 : if (nMaxDepth != 0)
3155 : {
3156 : /*-------------------------------------------------------------
3157 : * Loop through all entries, dumping each of them
3158 : *------------------------------------------------------------*/
3159 0 : for (int i = 0; i < numEntries; i++)
3160 : {
3161 0 : TABMAPIndexEntry *psEntry = poNode->GetEntry(i);
3162 :
3163 0 : TABRawBinBlock *poBlock = GetIndexObjectBlock(psEntry->nBlockPtr);
3164 0 : if (poBlock == nullptr)
3165 0 : continue;
3166 :
3167 0 : if (poBlock->GetBlockType() == TABMAP_INDEX_BLOCK)
3168 : {
3169 : /* Index block, dump recursively */
3170 0 : DumpSpatialIndexToMIF(
3171 : cpl::down_cast<TABMAPIndexBlock *>(poBlock), fpMIF, fpMID,
3172 : poNode->GetStartAddress(), i, nCurDepth + 1, nMaxDepth - 1);
3173 : }
3174 : else
3175 : {
3176 : /* Object block, dump directly */
3177 0 : CPLAssert(poBlock->GetBlockType() == TABMAP_OBJECT_BLOCK);
3178 :
3179 0 : Int2Coordsys(psEntry->XMin, psEntry->YMin, dXMin, dYMin);
3180 0 : Int2Coordsys(psEntry->XMax, psEntry->YMax, dXMax, dYMax);
3181 :
3182 0 : VSIFPrintf(fpMIF, "RECT %g %g %g %g\n", dXMin, dYMin, dXMax,
3183 : dYMax);
3184 0 : VSIFPrintf(fpMIF, " Brush(1, 0)\n"); /* No fill */
3185 :
3186 0 : VSIFPrintf(
3187 : fpMID, "%d,%d,%d,%d,%g,%d,%d,%d,%d\n", psEntry->nBlockPtr,
3188 : poNode->GetStartAddress(), i, nCurDepth + 1,
3189 0 : MITAB_AREA(psEntry->XMin, psEntry->YMin, psEntry->XMax,
3190 : psEntry->YMax),
3191 : psEntry->XMin, psEntry->YMin, psEntry->XMax, psEntry->YMax);
3192 : }
3193 :
3194 0 : delete poBlock;
3195 : }
3196 : }
3197 : }
3198 :
3199 : #endif // DEBUG
|