LCOV - code coverage report
Current view: top level - ogr/ogrsf_frmts/mitab - mitab_rawbinblock.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 276 415 66.5 %
Date: 2025-01-18 12:42:00 Functions: 40 45 88.9 %

          Line data    Source code
       1             : /**********************************************************************
       2             :  *
       3             :  * Name:     mitab_rawbinblock.cpp
       4             :  * Project:  MapInfo TAB Read/Write library
       5             :  * Language: C++
       6             :  * Purpose:  Implementation of the TABRawBinBlock class used to handle
       7             :  *           reading/writing blocks in the .MAP files
       8             :  * Author:   Daniel Morissette, dmorissette@dmsolutions.ca
       9             :  *
      10             :  **********************************************************************
      11             :  * Copyright (c) 1999, 2000, 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 <cctype>
      21             : #include <cstddef>
      22             : #include <cstdio>
      23             : #include <cstring>
      24             : #if HAVE_FCNTL_H
      25             : #include <fcntl.h>
      26             : #endif
      27             : #include <algorithm>
      28             : 
      29             : #include "cpl_conv.h"
      30             : #include "cpl_error.h"
      31             : #include "cpl_vsi.h"
      32             : #include "mitab_priv.h"
      33             : 
      34             : /*=====================================================================
      35             :  *                      class TABRawBinBlock
      36             :  *====================================================================*/
      37             : 
      38             : /**********************************************************************
      39             :  *                   TABRawBinBlock::TABRawBinBlock()
      40             :  *
      41             :  * Constructor.
      42             :  **********************************************************************/
      43       83987 : TABRawBinBlock::TABRawBinBlock(TABAccess eAccessMode /*= TABRead*/,
      44       83987 :                                GBool bHardBlockSize /*= TRUE*/)
      45             :     : m_fp(nullptr), m_eAccess(eAccessMode), m_nBlockType(0),
      46             :       m_pabyBuf(nullptr), m_nBlockSize(0), m_nSizeUsed(0),
      47             :       m_bHardBlockSize(bHardBlockSize), m_nFileOffset(0), m_nCurPos(0),
      48       83987 :       m_nFirstBlockPtr(0), m_nFileSize(-1), m_bModified(FALSE)
      49             : {
      50       83987 : }
      51             : 
      52             : /**********************************************************************
      53             :  *                   TABRawBinBlock::~TABRawBinBlock()
      54             :  *
      55             :  * Destructor.
      56             :  **********************************************************************/
      57      172571 : TABRawBinBlock::~TABRawBinBlock()
      58             : {
      59       83987 :     if (m_pabyBuf)
      60       83963 :         CPLFree(m_pabyBuf);
      61       88584 : }
      62             : 
      63             : /**********************************************************************
      64             :  *                   TABRawBinBlock::ReadFromFile()
      65             :  *
      66             :  * Load data from the specified file location and initialize the block.
      67             :  *
      68             :  * Returns 0 if successful or -1 if an error happened, in which case
      69             :  * CPLError() will have been called.
      70             :  **********************************************************************/
      71      280787 : int TABRawBinBlock::ReadFromFile(VSILFILE *fpSrc, int nOffset, int nSize)
      72             : {
      73      280787 :     if (fpSrc == nullptr || nSize == 0)
      74             :     {
      75           0 :         CPLError(CE_Failure, CPLE_AssertionFailed,
      76             :                  "TABRawBinBlock::ReadFromFile(): Assertion Failed!");
      77           0 :         return -1;
      78             :     }
      79             : 
      80      280787 :     m_fp = fpSrc;
      81             : 
      82      280787 :     VSIFSeekL(fpSrc, 0, SEEK_END);
      83      280787 :     m_nFileSize = static_cast<int>(VSIFTellL(m_fp));
      84             : 
      85      280787 :     m_nFileOffset = nOffset;
      86      280787 :     m_nCurPos = 0;
      87      280787 :     m_bModified = FALSE;
      88             : 
      89             :     /*----------------------------------------------------------------
      90             :      * Alloc a buffer to contain the data
      91             :      *---------------------------------------------------------------*/
      92      280787 :     GByte *pabyBuf = static_cast<GByte *>(CPLMalloc(nSize * sizeof(GByte)));
      93             : 
      94             :     /*----------------------------------------------------------------
      95             :      * Read from the file
      96             :      *---------------------------------------------------------------*/
      97      280787 :     if (VSIFSeekL(fpSrc, nOffset, SEEK_SET) != 0 ||
      98      280787 :         (m_nSizeUsed = static_cast<int>(
      99      561574 :              VSIFReadL(pabyBuf, sizeof(GByte), nSize, fpSrc))) == 0 ||
     100      280787 :         (m_bHardBlockSize && m_nSizeUsed != nSize))
     101             :     {
     102           0 :         CPLError(CE_Failure, CPLE_FileIO,
     103             :                  "ReadFromFile() failed reading %d bytes at offset %d.", nSize,
     104             :                  nOffset);
     105           0 :         CPLFree(pabyBuf);
     106           0 :         return -1;
     107             :     }
     108             : 
     109             :     /*----------------------------------------------------------------
     110             :      * Init block with the data we just read
     111             :      *---------------------------------------------------------------*/
     112      280787 :     return InitBlockFromData(pabyBuf, nSize, m_nSizeUsed, FALSE, fpSrc,
     113      280787 :                              nOffset);
     114             : }
     115             : 
     116             : /**********************************************************************
     117             :  *                   TABRawBinBlock::CommitToFile()
     118             :  *
     119             :  * Commit the current state of the binary block to the file to which
     120             :  * it has been previously attached.
     121             :  *
     122             :  * Derived classes may want to (optionally) reimplement this method if
     123             :  * they need to do special processing before committing the block to disk.
     124             :  *
     125             :  * For files created with bHardBlockSize=TRUE, a complete block of
     126             :  * the specified size is always written, otherwise only the number of
     127             :  * used bytes in the block will be written to disk.
     128             :  *
     129             :  * Returns 0 if successful or -1 if an error happened, in which case
     130             :  * CPLError() will have been called.
     131             :  **********************************************************************/
     132       53748 : int TABRawBinBlock::CommitToFile()
     133             : {
     134       53748 :     int nStatus = 0;
     135             : 
     136       53748 :     if (m_fp == nullptr || m_nBlockSize <= 0 || m_pabyBuf == nullptr ||
     137       53748 :         m_nFileOffset < 0)
     138             :     {
     139           0 :         CPLError(CE_Failure, CPLE_AssertionFailed,
     140             :                  "TABRawBinBlock::CommitToFile(): Block has not been "
     141             :                  "initialized yet!");
     142           0 :         return -1;
     143             :     }
     144             : 
     145             :     /*----------------------------------------------------------------
     146             :      * If block has not been modified, then just return... nothing to do.
     147             :      *---------------------------------------------------------------*/
     148       53748 :     if (!m_bModified)
     149        7183 :         return 0;
     150             : 
     151             :     /*----------------------------------------------------------------
     152             :      * Move the output file pointer to the right position...
     153             :      *---------------------------------------------------------------*/
     154       46565 :     if (VSIFSeekL(m_fp, m_nFileOffset, SEEK_SET) != 0)
     155             :     {
     156             :         /*------------------------------------------------------------
     157             :          * Moving pointer failed... we may need to pad with zeros if
     158             :          * block destination is beyond current end of file.
     159             :          *-----------------------------------------------------------*/
     160           0 :         int nCurPos = static_cast<int>(VSIFTellL(m_fp));
     161             : 
     162           0 :         if (nCurPos < m_nFileOffset && VSIFSeekL(m_fp, 0L, SEEK_END) == 0 &&
     163           0 :             (nCurPos = static_cast<int>(VSIFTellL(m_fp))) < m_nFileOffset)
     164             :         {
     165           0 :             const GByte cZero = 0;
     166             : 
     167           0 :             while (nCurPos < m_nFileOffset && nStatus == 0)
     168             :             {
     169           0 :                 if (VSIFWriteL(&cZero, 1, 1, m_fp) != 1)
     170             :                 {
     171           0 :                     CPLError(CE_Failure, CPLE_FileIO,
     172             :                              "Failed writing 1 byte at offset %d.", nCurPos);
     173           0 :                     nStatus = -1;
     174           0 :                     break;
     175             :                 }
     176           0 :                 nCurPos++;
     177             :             }
     178             :         }
     179             : 
     180           0 :         if (nCurPos != m_nFileOffset)
     181           0 :             nStatus = -1;  // Error message will follow below
     182             :     }
     183             : 
     184             :     /*----------------------------------------------------------------
     185             :      * At this point we are ready to write to the file.
     186             :      *
     187             :      * If m_bHardBlockSize==FALSE, then we do not write a complete block;
     188             :      * we write only the part of the block that was used.
     189             :      *---------------------------------------------------------------*/
     190       46565 :     int numBytesToWrite = m_bHardBlockSize ? m_nBlockSize : m_nSizeUsed;
     191             : 
     192             :     /*CPLDebug("MITAB", "Committing to offset %d", m_nFileOffset);*/
     193             : 
     194       93130 :     if (nStatus != 0 ||
     195       46565 :         VSIFWriteL(m_pabyBuf, sizeof(GByte), numBytesToWrite, m_fp) !=
     196       46565 :             static_cast<size_t>(numBytesToWrite))
     197             :     {
     198          10 :         CPLError(CE_Failure, CPLE_FileIO,
     199             :                  "Failed writing %d bytes at offset %d.", numBytesToWrite,
     200             :                  m_nFileOffset);
     201          10 :         return -1;
     202             :     }
     203       46555 :     if (m_nFileOffset + numBytesToWrite > m_nFileSize)
     204             :     {
     205       31286 :         m_nFileSize = m_nFileOffset + numBytesToWrite;
     206             :     }
     207             : 
     208       46555 :     VSIFFlushL(m_fp);
     209             : 
     210       46555 :     m_bModified = FALSE;
     211             : 
     212       46555 :     return 0;
     213             : }
     214             : 
     215             : /**********************************************************************
     216             :  *                   TABRawBinBlock::CommitAsDeleted()
     217             :  *
     218             :  * Commit current block to file using block type 4 (garbage block)
     219             :  *
     220             :  * Returns 0 if successful or -1 if an error happened, in which case
     221             :  * CPLError() will have been called.
     222             :  **********************************************************************/
     223          13 : int TABRawBinBlock::CommitAsDeleted(GInt32 nNextBlockPtr)
     224             : {
     225          13 :     CPLErrorReset();
     226             : 
     227          13 :     if (m_pabyBuf == nullptr)
     228             :     {
     229           0 :         CPLError(CE_Failure, CPLE_AssertionFailed,
     230             :                  "CommitAsDeleted(): Block has not been initialized yet!");
     231           0 :         return -1;
     232             :     }
     233             : 
     234             :     /*-----------------------------------------------------------------
     235             :      * Create deleted block header
     236             :      *----------------------------------------------------------------*/
     237          13 :     GotoByteInBlock(0x000);
     238          13 :     WriteInt16(TABMAP_GARB_BLOCK);  // Block type code
     239          13 :     WriteInt32(nNextBlockPtr);
     240             : 
     241          13 :     int nStatus = CPLGetLastErrorType() == CE_Failure ? -1 : 0;
     242             : 
     243             :     /*-----------------------------------------------------------------
     244             :      * OK, call the base class to write the block to disk.
     245             :      *----------------------------------------------------------------*/
     246          13 :     if (nStatus == 0)
     247             :     {
     248             : #ifdef DEBUG_VERBOSE
     249             :         CPLDebug("MITAB", "Committing GARBAGE block to offset %d",
     250             :                  m_nFileOffset);
     251             : #endif
     252          13 :         nStatus = TABRawBinBlock::CommitToFile();
     253          13 :         m_nSizeUsed = 0;
     254             :     }
     255             : 
     256          13 :     return nStatus;
     257             : }
     258             : 
     259             : /**********************************************************************
     260             :  *                   TABRawBinBlock::InitBlockFromData()
     261             :  *
     262             :  * Set the binary data buffer and initialize the block.
     263             :  *
     264             :  * Calling ReadFromFile() will automatically call InitBlockFromData() to
     265             :  * complete the initialization of the block after the data is read from the
     266             :  * file.  Derived classes should implement their own version of
     267             :  * InitBlockFromData() if they need specific initialization... in this
     268             :  * case the derived InitBlockFromData() should call
     269             :  * TABRawBinBlock::InitBlockFromData() before doing anything else.
     270             :  *
     271             :  * By default, the buffer will be copied, but if bMakeCopy = FALSE then
     272             :  * it won't be copied, and the object will keep a reference to the
     273             :  * user's buffer... and this object will eventually free the user's buffer.
     274             :  *
     275             :  * Returns 0 if successful or -1 if an error happened, in which case
     276             :  * CPLError() will have been called.
     277             :  **********************************************************************/
     278      355831 : int TABRawBinBlock::InitBlockFromData(GByte *pabyBuf, int nBlockSize,
     279             :                                       int nSizeUsed,
     280             :                                       GBool bMakeCopy /* = TRUE */,
     281             :                                       VSILFILE *fpSrc /* = NULL */,
     282             :                                       int nOffset /* = 0 */)
     283             : {
     284      355831 :     m_fp = fpSrc;
     285      355831 :     m_nFileOffset = nOffset;
     286      355831 :     m_nCurPos = 0;
     287      355831 :     m_bModified = FALSE;
     288             : 
     289             :     /*----------------------------------------------------------------
     290             :      * Alloc or realloc the buffer to contain the data if necessary
     291             :      *---------------------------------------------------------------*/
     292      355831 :     if (!bMakeCopy)
     293             :     {
     294      355831 :         if (m_pabyBuf != nullptr)
     295      278064 :             CPLFree(m_pabyBuf);
     296      355831 :         m_pabyBuf = pabyBuf;
     297      355831 :         m_nBlockSize = nBlockSize;
     298      355831 :         m_nSizeUsed = nSizeUsed;
     299             :     }
     300           0 :     else if (m_pabyBuf == nullptr || nBlockSize != m_nBlockSize)
     301             :     {
     302           0 :         m_pabyBuf = static_cast<GByte *>(
     303           0 :             CPLRealloc(m_pabyBuf, nBlockSize * sizeof(GByte)));
     304           0 :         m_nBlockSize = nBlockSize;
     305           0 :         m_nSizeUsed = nSizeUsed;
     306           0 :         memcpy(m_pabyBuf, pabyBuf, m_nSizeUsed);
     307             :     }
     308             : 
     309             :     /*----------------------------------------------------------------
     310             :      * Extract block type... header block (first block in a file) has
     311             :      * no block type, so we assign one by default.
     312             :      *---------------------------------------------------------------*/
     313      355831 :     if (m_nFileOffset == 0)
     314        5612 :         m_nBlockType = TABMAP_HEADER_BLOCK;
     315             :     else
     316             :     {
     317             :         // Block type will be validated only if GetBlockType() is called
     318      350219 :         m_nBlockType = static_cast<int>(m_pabyBuf[0]);
     319             :     }
     320             : 
     321      355831 :     return 0;
     322             : }
     323             : 
     324             : /**********************************************************************
     325             :  *                   TABRawBinBlock::InitNewBlock()
     326             :  *
     327             :  * Initialize the block so that it knows to which file is attached,
     328             :  * its block size, etc.
     329             :  *
     330             :  * This is an alternative to calling ReadFromFile() or InitBlockFromData()
     331             :  * that puts the block in a stable state without loading any initial
     332             :  * data in it.
     333             :  *
     334             :  * Returns 0 if successful or -1 if an error happened, in which case
     335             :  * CPLError() will have been called.
     336             :  **********************************************************************/
     337       23134 : int TABRawBinBlock::InitNewBlock(VSILFILE *fpSrc, int nBlockSize,
     338             :                                  int nFileOffset /* = 0*/)
     339             : {
     340       23134 :     m_fp = fpSrc;
     341       23134 :     m_nBlockSize = nBlockSize;
     342       23134 :     m_nSizeUsed = 0;
     343       23134 :     m_nCurPos = 0;
     344       23134 :     m_bModified = FALSE;
     345             : 
     346       23134 :     if (nFileOffset > 0)
     347       17427 :         m_nFileOffset = nFileOffset;
     348             :     else
     349        5707 :         m_nFileOffset = 0;
     350             : 
     351       23134 :     if (m_fp != nullptr && m_nFileSize < 0 && m_eAccess == TABReadWrite)
     352             :     {
     353        4117 :         int nCurPos = static_cast<int>(VSIFTellL(m_fp));
     354        4117 :         VSIFSeekL(fpSrc, 0, SEEK_END);
     355        4117 :         m_nFileSize = static_cast<int>(VSIFTellL(m_fp));
     356        4117 :         VSIFSeekL(fpSrc, nCurPos, SEEK_SET);
     357             :     }
     358             : 
     359       23134 :     m_nBlockType = -1;
     360             : 
     361       23134 :     m_pabyBuf = static_cast<GByte *>(
     362       23134 :         CPLRealloc(m_pabyBuf, m_nBlockSize * sizeof(GByte)));
     363       23134 :     if (m_nBlockSize)
     364       23107 :         memset(m_pabyBuf, 0, m_nBlockSize);
     365             : 
     366       23134 :     return 0;
     367             : }
     368             : 
     369             : /**********************************************************************
     370             :  *                   TABRawBinBlock::GetBlockType()
     371             :  *
     372             :  * Return the block type for the current object.
     373             :  *
     374             :  * Returns a block type >= 0 if successful or -1 if an error happened, in
     375             :  * which case  CPLError() will have been called.
     376             :  **********************************************************************/
     377       93087 : int TABRawBinBlock::GetBlockType()
     378             : {
     379       93087 :     if (m_pabyBuf == nullptr)
     380             :     {
     381           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     382             :                  "GetBlockType(): Block has not been initialized.");
     383           0 :         return -1;
     384             :     }
     385             : 
     386       93087 :     if (m_nBlockType > TABMAP_LAST_VALID_BLOCK_TYPE)
     387             :     {
     388           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     389             :                  "GetBlockType(): Unsupported block type %d.", m_nBlockType);
     390           0 :         return -1;
     391             :     }
     392             : 
     393       93087 :     return m_nBlockType;
     394             : }
     395             : 
     396             : /**********************************************************************
     397             :  *                   TABRawBinBlock::GotoByteInBlock()
     398             :  *
     399             :  * Move the block pointer to the specified position relative to the
     400             :  * beginning of the block.
     401             :  *
     402             :  * Returns 0 if successful or -1 if an error happened, in which case
     403             :  * CPLError() will have been called.
     404             :  **********************************************************************/
     405      605064 : int TABRawBinBlock::GotoByteInBlock(int nOffset)
     406             : {
     407      605064 :     if ((m_eAccess == TABRead && nOffset > m_nSizeUsed) ||
     408      605064 :         (m_eAccess != TABRead && nOffset > m_nBlockSize))
     409             :     {
     410           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     411             :                  "GotoByteInBlock(): Attempt to go past end of data block.");
     412           0 :         return -1;
     413             :     }
     414             : 
     415      605064 :     if (nOffset < 0)
     416             :     {
     417           0 :         CPLError(
     418             :             CE_Failure, CPLE_AppDefined,
     419             :             "GotoByteInBlock(): Attempt to go before start of data block.");
     420           0 :         return -1;
     421             :     }
     422             : 
     423      605064 :     m_nCurPos = nOffset;
     424             : 
     425      605064 :     m_nSizeUsed = std::max(m_nSizeUsed, m_nCurPos);
     426             : 
     427      605064 :     return 0;
     428             : }
     429             : 
     430             : /**********************************************************************
     431             :  *                   TABRawBinBlock::GotoByteRel()
     432             :  *
     433             :  * Move the block pointer by the specified number of bytes relative
     434             :  * to its current position.
     435             :  *
     436             :  * Returns 0 if successful or -1 if an error happened, in which case
     437             :  * CPLError() will have been called.
     438             :  **********************************************************************/
     439          43 : int TABRawBinBlock::GotoByteRel(int nOffset)
     440             : {
     441          43 :     return GotoByteInBlock(m_nCurPos + nOffset);
     442             : }
     443             : 
     444             : /**********************************************************************
     445             :  *                   TABRawBinBlock::GotoByteInFile()
     446             :  *
     447             :  * Move the block pointer to the specified position relative to the
     448             :  * beginning of the file.
     449             :  *
     450             :  * In read access, the current block may be reloaded to contain a right
     451             :  * block of binary data if necessary.
     452             :  *
     453             :  * In write mode, the current block may automagically be committed to
     454             :  * disk and a new block initialized if necessary.
     455             :  *
     456             :  * bForceReadFromFile is used in write mode to read the new block data from
     457             :  * file instead of creating an empty block. (Useful for TABCollection
     458             :  * or other cases that need to do random access in the file in write mode.)
     459             :  *
     460             :  * bOffsetIsEndOfData is set to TRUE to indicate that the nOffset
     461             :  * to which we are attempting to go is the end of the used data in this
     462             :  * block (we are positioning ourselves to append data), so if the nOffset
     463             :  * corresponds to the beginning of a block then we should really
     464             :  * be positioning ourselves at the end of the block that ends at this
     465             :  * address instead of at the beginning of the blocks that starts at this
     466             :  * address. This case can happen when going back and forth to write collection
     467             :  * objects to a Coordblock and is documented in bug 1657.
     468             :  *
     469             :  * Returns 0 if successful or -1 if an error happened, in which case
     470             :  * CPLError() will have been called.
     471             :  **********************************************************************/
     472     1586200 : int TABRawBinBlock::GotoByteInFile(int nOffset,
     473             :                                    GBool bForceReadFromFile /*=FALSE*/,
     474             :                                    GBool bOffsetIsEndOfData /*=FALSE*/)
     475             : {
     476     1586200 :     if (nOffset < 0)
     477             :     {
     478           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     479             :                  "GotoByteInFile(): Attempt to go before start of file.");
     480           0 :         return -1;
     481             :     }
     482             : 
     483     1586200 :     int nNewBlockPtr =
     484     1586200 :         ((nOffset - m_nFirstBlockPtr) / m_nBlockSize) * m_nBlockSize +
     485     1586200 :         m_nFirstBlockPtr;
     486             : 
     487     1586200 :     if (m_eAccess == TABRead)
     488             :     {
     489     4458740 :         if ((nOffset < m_nFileOffset ||
     490     1753450 :              nOffset >= m_nFileOffset + m_nSizeUsed) &&
     491      260585 :             ReadFromFile(m_fp, nNewBlockPtr, m_nBlockSize) != 0)
     492             :         {
     493             :             // Failed reading new block... error has already been reported.
     494           0 :             return -1;
     495             :         }
     496             :     }
     497       93332 :     else if (m_eAccess == TABWrite)
     498             :     {
     499       10359 :         if ((nOffset < m_nFileOffset ||
     500        3453 :              nOffset >= m_nFileOffset + m_nBlockSize) &&
     501           0 :             (CommitToFile() != 0 ||
     502           0 :              InitNewBlock(m_fp, m_nBlockSize, nNewBlockPtr) != 0))
     503             :         {
     504             :             // Failed reading new block... error has already been reported.
     505           0 :             return -1;
     506             :         }
     507             :     }
     508       89879 :     else if (m_eAccess == TABReadWrite)
     509             :     {
     510             :         // TODO: THIS IS NOT REAL read/write access (it is more extended write)
     511             :         // Currently we try to read from file only if explicitly requested.
     512             :         // If we ever want true read/write mode we should implement
     513             :         // more smarts to detect whether the caller wants an existing block to
     514             :         // be read, or a new one to be created from scratch.
     515             :         // CommitToFile() should only be called only if something changed.
     516             :         //
     517       89879 :         if (bOffsetIsEndOfData && nOffset % m_nBlockSize == 0)
     518             :         {
     519             :             /* We're trying to go byte m_nBlockSize of a block that's full of
     520             :              * data. In this case it is okay to place the m_nCurPos at byte
     521             :              * m_nBlockSize which is past the end of the block.
     522             :              */
     523             : 
     524             :             /* Make sure we request the block that ends with requested
     525             :              * address and not the following block that doesn't exist
     526             :              * yet on disk */
     527           0 :             nNewBlockPtr -= m_nBlockSize;
     528             : 
     529           0 :             if ((nOffset < m_nFileOffset ||
     530           0 :                  nOffset > m_nFileOffset + m_nBlockSize) &&
     531           0 :                 (CommitToFile() != 0 ||
     532           0 :                  (!bForceReadFromFile &&
     533           0 :                   InitNewBlock(m_fp, m_nBlockSize, nNewBlockPtr) != 0) ||
     534           0 :                  (bForceReadFromFile &&
     535           0 :                   ReadFromFile(m_fp, nNewBlockPtr, m_nBlockSize) != 0)))
     536             :             {
     537             :                 // Failed reading new block... error has already been reported.
     538           0 :                 return -1;
     539             :             }
     540             :         }
     541             :         else
     542             :         {
     543       89879 :             if (!bForceReadFromFile && m_nFileSize > 0 && nOffset < m_nFileSize)
     544             :             {
     545       21185 :                 bForceReadFromFile = TRUE;
     546       21185 :                 if (!(nOffset < m_nFileOffset ||
     547       21093 :                       nOffset >= m_nFileOffset + m_nBlockSize))
     548             :                 {
     549       21194 :                     if ((nOffset >= m_nFileOffset + m_nSizeUsed) &&
     550        4932 :                         (CommitToFile() != 0 ||
     551        4932 :                          ReadFromFile(m_fp, nNewBlockPtr, m_nBlockSize) != 0))
     552             :                     {
     553             :                         // Failed reading new block... error has already been
     554             :                         // reported.
     555           0 :                         return -1;
     556             :                     }
     557             :                 }
     558             :             }
     559             : 
     560      268755 :             if ((nOffset < m_nFileOffset ||
     561      102749 :                  nOffset >= m_nFileOffset + m_nBlockSize) &&
     562       12870 :                 (CommitToFile() != 0 ||
     563         323 :                  (!bForceReadFromFile &&
     564       12870 :                   InitNewBlock(m_fp, m_nBlockSize, nNewBlockPtr) != 0) ||
     565       12547 :                  (bForceReadFromFile &&
     566       12547 :                   ReadFromFile(m_fp, nNewBlockPtr, m_nBlockSize) != 0)))
     567             :             {
     568             :                 // Failed reading new block... error has already been reported.
     569           0 :                 return -1;
     570             :             }
     571             :         }
     572             :     }
     573             :     else
     574             :     {
     575           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     576             :                  "Access mode not supported yet!");
     577           0 :         return -1;
     578             :     }
     579             : 
     580     1586200 :     m_nCurPos = nOffset - m_nFileOffset;
     581             : 
     582     1586200 :     m_nSizeUsed = std::max(m_nSizeUsed, m_nCurPos);
     583             : 
     584     1586200 :     return 0;
     585             : }
     586             : 
     587             : /**********************************************************************
     588             :  *                   TABRawBinBlock::SetFirstBlockPtr()
     589             :  *
     590             :  * Set the position in the file at which the first block starts.
     591             :  * This value will usually be the header size and needs to be specified
     592             :  * only if the header size is different from the other blocks size.
     593             :  *
     594             :  * This value will be used by GotoByteInFile() to properly align the data
     595             :  * blocks that it loads automatically when a requested position is outside
     596             :  * of the block currently in memory.
     597             :  **********************************************************************/
     598        1495 : void TABRawBinBlock::SetFirstBlockPtr(int nOffset)
     599             : {
     600        1495 :     m_nFirstBlockPtr = nOffset;
     601        1495 : }
     602             : 
     603             : /**********************************************************************
     604             :  *                   TABRawBinBlock::GetNumUnusedBytes()
     605             :  *
     606             :  * Return the number of unused bytes in this block.
     607             :  **********************************************************************/
     608       45552 : int TABRawBinBlock::GetNumUnusedBytes()
     609             : {
     610       45552 :     return m_nBlockSize - m_nSizeUsed;
     611             : }
     612             : 
     613             : /**********************************************************************
     614             :  *                   TABRawBinBlock::GetFirstUnusedByteOffset()
     615             :  *
     616             :  * Return the position of the first unused byte in this block relative
     617             :  * to the beginning of the file, or -1 if the block is full.
     618             :  **********************************************************************/
     619       26380 : int TABRawBinBlock::GetFirstUnusedByteOffset()
     620             : {
     621       26380 :     if (m_nSizeUsed < m_nBlockSize)
     622       26380 :         return m_nFileOffset + m_nSizeUsed;
     623             :     else
     624           0 :         return -1;
     625             : }
     626             : 
     627             : /**********************************************************************
     628             :  *                   TABRawBinBlock::GetCurAddress()
     629             :  *
     630             :  * Return the current pointer position, relative to beginning of file.
     631             :  **********************************************************************/
     632         517 : int TABRawBinBlock::GetCurAddress()
     633             : {
     634         517 :     return m_nFileOffset + m_nCurPos;
     635             : }
     636             : 
     637             : /**********************************************************************
     638             :  *                   TABRawBinBlock::ReadBytes()
     639             :  *
     640             :  * Copy the number of bytes from the data block's internal buffer to
     641             :  * the user's buffer pointed by pabyDstBuf.
     642             :  *
     643             :  * Passing pabyDstBuf = NULL will only move the read pointer by the
     644             :  * specified number of bytes as if the copy had happened... but it
     645             :  * won't crash.
     646             :  *
     647             :  * Returns 0 if successful or -1 if an error happened, in which case
     648             :  * CPLError() will have been called.
     649             :  **********************************************************************/
     650     8547400 : int TABRawBinBlock::ReadBytes(int numBytes, GByte *pabyDstBuf)
     651             : {
     652             :     /*----------------------------------------------------------------
     653             :      * Make sure block is initialized with Read access and that the
     654             :      * operation won't go beyond the buffer's size.
     655             :      *---------------------------------------------------------------*/
     656     8547400 :     if (m_pabyBuf == nullptr)
     657             :     {
     658           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     659             :                  "ReadBytes(): Block has not been initialized.");
     660           0 :         return -1;
     661             :     }
     662             : 
     663     8547400 :     if (m_nCurPos + numBytes > m_nSizeUsed)
     664             :     {
     665           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     666             :                  "ReadBytes(): Attempt to read past end of data block.");
     667           0 :         return -1;
     668             :     }
     669             : 
     670     8547400 :     if (pabyDstBuf)
     671             :     {
     672     8547400 :         memcpy(pabyDstBuf, m_pabyBuf + m_nCurPos, numBytes);
     673             :     }
     674             : 
     675     8547400 :     m_nCurPos += numBytes;
     676             : 
     677     8547400 :     return 0;
     678             : }
     679             : 
     680             : /**********************************************************************
     681             :  *                   TABRawBinBlock::Read<datatype>()
     682             :  *
     683             :  * MapInfo files are binary files with LSB first (Intel) byte
     684             :  * ordering.  The following functions will read from the input file
     685             :  * and return a value with the bytes ordered properly for the current
     686             :  * platform.
     687             :  **********************************************************************/
     688     2375840 : GByte TABRawBinBlock::ReadByte()
     689             : {
     690     2375840 :     GByte byValue = 0;
     691             : 
     692     2375840 :     ReadBytes(1, &byValue);
     693             : 
     694     2375840 :     return byValue;
     695             : }
     696             : 
     697      177098 : GInt16 TABRawBinBlock::ReadInt16()
     698             : {
     699      177098 :     GInt16 n16Value = 0;
     700             : 
     701      177098 :     ReadBytes(2, reinterpret_cast<GByte *>(&n16Value));
     702             : 
     703             : #ifdef CPL_MSB
     704             :     return static_cast<GInt16>(CPL_SWAP16(n16Value));
     705             : #else
     706      177098 :     return n16Value;
     707             : #endif
     708             : }
     709             : 
     710     5933250 : GInt32 TABRawBinBlock::ReadInt32()
     711             : {
     712     5933250 :     GInt32 n32Value = 0;
     713             : 
     714     5933250 :     ReadBytes(4, reinterpret_cast<GByte *>(&n32Value));
     715             : 
     716             : #ifdef CPL_MSB
     717             :     return static_cast<GInt32>(CPL_SWAP32(n32Value));
     718             : #else
     719     5933250 :     return n32Value;
     720             : #endif
     721             : }
     722             : 
     723           6 : GInt64 TABRawBinBlock::ReadInt64()
     724             : {
     725           6 :     GInt64 n64Value = 0;
     726             : 
     727           6 :     ReadBytes(8, reinterpret_cast<GByte *>(&n64Value));
     728             : 
     729             : #ifdef CPL_MSB
     730             :     CPL_LSBPTR64(&n64Value);
     731             : #endif
     732           6 :     return n64Value;
     733             : }
     734             : 
     735           0 : float TABRawBinBlock::ReadFloat()
     736             : {
     737           0 :     float fValue = 0.0f;
     738             : 
     739           0 :     ReadBytes(4, reinterpret_cast<GByte *>(&fValue));
     740             : 
     741             : #ifdef CPL_MSB
     742             :     CPL_LSBPTR32(&fValue);
     743             : #endif
     744           0 :     return fValue;
     745             : }
     746             : 
     747       50683 : double TABRawBinBlock::ReadDouble()
     748             : {
     749       50683 :     double dValue = 0.0;
     750             : 
     751       50683 :     ReadBytes(8, reinterpret_cast<GByte *>(&dValue));
     752             : 
     753             : #ifdef CPL_MSB
     754             :     CPL_SWAPDOUBLE(&dValue);
     755             : #endif
     756             : 
     757       50683 :     return dValue;
     758             : }
     759             : 
     760             : /**********************************************************************
     761             :  *                   TABRawBinBlock::WriteBytes()
     762             :  *
     763             :  * Copy the number of bytes from the user's buffer pointed by pabySrcBuf
     764             :  * to the data block's internal buffer.
     765             :  * Note that this call only writes to the memory buffer... nothing is
     766             :  * written to the file until WriteToFile() is called.
     767             :  *
     768             :  * Passing pabySrcBuf = NULL will only move the write pointer by the
     769             :  * specified number of bytes as if the copy had happened... but it
     770             :  * won't crash.
     771             :  *
     772             :  * Returns 0 if successful or -1 if an error happened, in which case
     773             :  * CPLError() will have been called.
     774             :  **********************************************************************/
     775      597429 : int TABRawBinBlock::WriteBytes(int nBytesToWrite, const GByte *pabySrcBuf)
     776             : {
     777             :     /*----------------------------------------------------------------
     778             :      * Make sure block is initialized with Write access and that the
     779             :      * operation won't go beyond the buffer's size.
     780             :      *---------------------------------------------------------------*/
     781      597429 :     if (m_pabyBuf == nullptr)
     782             :     {
     783           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     784             :                  "WriteBytes(): Block has not been initialized.");
     785           0 :         return -1;
     786             :     }
     787             : 
     788      597429 :     if (m_eAccess == TABRead)
     789             :     {
     790           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     791             :                  "WriteBytes(): Block does not support write operations.");
     792           0 :         return -1;
     793             :     }
     794             : 
     795      597429 :     if (m_nCurPos + nBytesToWrite > m_nBlockSize)
     796             :     {
     797           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     798             :                  "WriteBytes(): Attempt to write past end of data block.");
     799           0 :         return -1;
     800             :     }
     801             : 
     802             :     /*----------------------------------------------------------------
     803             :      * Everything is OK... copy the data
     804             :      *---------------------------------------------------------------*/
     805      597429 :     if (pabySrcBuf)
     806             :     {
     807      597429 :         memcpy(m_pabyBuf + m_nCurPos, pabySrcBuf, nBytesToWrite);
     808             :     }
     809             : 
     810      597429 :     m_nCurPos += nBytesToWrite;
     811             : 
     812      597429 :     m_nSizeUsed = std::max(m_nSizeUsed, m_nCurPos);
     813             : 
     814      597429 :     m_bModified = TRUE;
     815             : 
     816      597429 :     return 0;
     817             : }
     818             : 
     819             : /**********************************************************************
     820             :  *                    TABRawBinBlock::Write<datatype>()
     821             :  *
     822             :  * Arc/Info files are binary files with MSB first (Motorola) byte
     823             :  * ordering.  The following functions will reorder the byte for the
     824             :  * value properly and write that to the output file.
     825             :  *
     826             :  * If a problem happens, then CPLError() will be called and
     827             :  * CPLGetLastErrNo() can be used to test if a write operation was
     828             :  * successful.
     829             :  **********************************************************************/
     830      119484 : int TABRawBinBlock::WriteByte(GByte byValue)
     831             : {
     832      119484 :     return WriteBytes(1, &byValue);
     833             : }
     834             : 
     835       66482 : int TABRawBinBlock::WriteInt16(GInt16 n16Value)
     836             : {
     837             : #ifdef CPL_MSB
     838             :     n16Value = static_cast<GInt16>(CPL_SWAP16(n16Value));
     839             : #endif
     840             : 
     841       66482 :     return WriteBytes(2, reinterpret_cast<GByte *>(&n16Value));
     842             : }
     843             : 
     844      341124 : int TABRawBinBlock::WriteInt32(GInt32 n32Value)
     845             : {
     846             : #ifdef CPL_MSB
     847             :     n32Value = static_cast<GInt32>(CPL_SWAP32(n32Value));
     848             : #endif
     849             : 
     850      341124 :     return WriteBytes(4, reinterpret_cast<GByte *>(&n32Value));
     851             : }
     852             : 
     853           2 : int TABRawBinBlock::WriteInt64(GInt64 n64Value)
     854             : {
     855             : #ifdef CPL_MSB
     856             :     CPL_SWAP64PTR(&n64Value);
     857             : #endif
     858             : 
     859           2 :     return WriteBytes(8, reinterpret_cast<GByte *>(&n64Value));
     860             : }
     861             : 
     862         160 : int TABRawBinBlock::WriteFloat(float fValue)
     863             : {
     864             : #ifdef CPL_MSB
     865             :     CPL_LSBPTR32(&fValue);
     866             : #endif
     867             : 
     868         160 :     return WriteBytes(4, reinterpret_cast<GByte *>(&fValue));
     869             : }
     870             : 
     871       23121 : int TABRawBinBlock::WriteDouble(double dValue)
     872             : {
     873             : #ifdef CPL_MSB
     874             :     CPL_SWAPDOUBLE(&dValue);
     875             : #endif
     876             : 
     877       23121 :     return WriteBytes(8, reinterpret_cast<GByte *>(&dValue));
     878             : }
     879             : 
     880             : /**********************************************************************
     881             :  *                    TABRawBinBlock::WriteZeros()
     882             :  *
     883             :  * Write a number of zeros (specified in bytes) at the current position
     884             :  * in the file.
     885             :  *
     886             :  * If a problem happens, then CPLError() will be called and
     887             :  * CPLGetLastErrNo() can be used to test if a write operation was
     888             :  * successful.
     889             :  **********************************************************************/
     890        8775 : int TABRawBinBlock::WriteZeros(int nBytesToWrite)
     891             : {
     892        8775 :     const GByte acZeros[8] = {0, 0, 0, 0, 0, 0, 0, 0};
     893        8775 :     int nStatus = 0;
     894             : 
     895             :     // Write by 8 bytes chunks.  The last chunk may be less than 8 bytes.
     896       51677 :     for (int i = 0; nStatus == 0 && i < nBytesToWrite; i += 8)
     897             :     {
     898       42902 :         nStatus = WriteBytes(std::min(8, nBytesToWrite - i), acZeros);
     899             :     }
     900             : 
     901        8775 :     return nStatus;
     902             : }
     903             : 
     904             : /**********************************************************************
     905             :  *                   TABRawBinBlock::WritePaddedString()
     906             :  *
     907             :  * Write a string and pad the end of the field (up to nFieldSize) with
     908             :  * spaces number of spaces at the current position in the file.
     909             :  *
     910             :  * If a problem happens, then CPLError() will be called and
     911             :  * CPLGetLastErrNo() can be used to test if a write operation was
     912             :  * successful.
     913             :  **********************************************************************/
     914           0 : int TABRawBinBlock::WritePaddedString(int nFieldSize, const char *pszString)
     915             : {
     916           0 :     char acSpaces[8] = {' ', ' ', ' ', ' ', ' ', ' ', ' ', ' '};
     917             :     int i, nLen, numSpaces;
     918           0 :     int nStatus = 0;
     919             : 
     920           0 :     nLen = static_cast<int>(strlen(pszString));
     921           0 :     nLen = std::min(nLen, nFieldSize);
     922           0 :     numSpaces = nFieldSize - nLen;
     923             : 
     924           0 :     if (nLen > 0)
     925           0 :         nStatus = WriteBytes(nLen, reinterpret_cast<const GByte *>(pszString));
     926             : 
     927             :     /* Write spaces by 8 bytes chunks.  The last chunk may be less than 8 bytes
     928             :      */
     929           0 :     for (i = 0; nStatus == 0 && i < numSpaces; i += 8)
     930             :     {
     931           0 :         nStatus = WriteBytes(std::min(8, numSpaces - i),
     932           0 :                              reinterpret_cast<GByte *>(acSpaces));
     933             :     }
     934             : 
     935           0 :     return nStatus;
     936             : }
     937             : 
     938             : /**********************************************************************
     939             :  *                   TABRawBinBlock::Dump()
     940             :  *
     941             :  * Dump block contents... available only in DEBUG mode.
     942             :  **********************************************************************/
     943             : #ifdef DEBUG
     944             : 
     945           0 : void TABRawBinBlock::Dump(FILE *fpOut /*=NULL*/)
     946             : {
     947           0 :     if (fpOut == nullptr)
     948           0 :         fpOut = stdout;
     949             : 
     950           0 :     fprintf(fpOut, "----- TABRawBinBlock::Dump() -----\n");
     951           0 :     if (m_pabyBuf == nullptr)
     952             :     {
     953           0 :         fprintf(fpOut, "Block has not been initialized yet.");
     954             :     }
     955             :     else
     956             :     {
     957           0 :         if (m_nBlockType == TABMAP_GARB_BLOCK)
     958             :         {
     959           0 :             fprintf(fpOut, "Garbage Block (type %d) at offset %d.\n",
     960             :                     m_nBlockType, m_nFileOffset);
     961           0 :             int nNextGarbageBlock = 0;
     962           0 :             memcpy(&nNextGarbageBlock, m_pabyBuf + 2, 4);
     963           0 :             CPL_LSBPTR32(&nNextGarbageBlock);
     964           0 :             fprintf(fpOut, "  m_nNextGarbageBlock     = %d\n",
     965             :                     nNextGarbageBlock);
     966             :         }
     967             :         else
     968             :         {
     969           0 :             fprintf(fpOut,
     970             :                     "Block (type %d) size=%d bytes at offset %d in file.\n",
     971             :                     m_nBlockType, m_nBlockSize, m_nFileOffset);
     972           0 :             fprintf(fpOut, "Current pointer at byte %d\n", m_nCurPos);
     973             :         }
     974             :     }
     975             : 
     976           0 :     fflush(fpOut);
     977           0 : }
     978             : 
     979             : #endif  // DEBUG
     980             : 
     981             : /**********************************************************************
     982             :  *                          DumpBytes()
     983             :  *
     984             :  * Read and dump the contents of an Binary file.
     985             :  **********************************************************************/
     986           0 : void TABRawBinBlock::DumpBytes(GInt32 nValue, int nOffset /*=0*/,
     987             :                                FILE *fpOut /*=NULL*/)
     988             : {
     989           0 :     float fValue = 0.0f;
     990           0 :     memcpy(&fValue, &nValue, 4);
     991             : 
     992             :     char achValue[4];
     993           0 :     memcpy(achValue, &nValue, 4);
     994             : 
     995           0 :     GInt16 n16Val1 = 0;
     996           0 :     memcpy(&n16Val1, achValue + 2, sizeof(GInt16));
     997           0 :     GInt16 n16Val2 = 0;
     998           0 :     memcpy(&n16Val2, achValue, sizeof(GInt16));
     999             : 
    1000             :     /* For double precision values, we only use the first half
    1001             :      * of the height bytes... and leave the other 4 bytes as zeros!
    1002             :      * It's a bit of a hack, but it seems to be enough for the
    1003             :      * precision of the values we print!
    1004             :      */
    1005             : #ifdef CPL_MSB
    1006             :     const GInt32 anVal[2] = {nValue, 0};
    1007             : #else
    1008           0 :     const GInt32 anVal[2] = {0, nValue};
    1009             : #endif
    1010           0 :     double dValue = 0.0;
    1011           0 :     memcpy(&dValue, anVal, 8);
    1012             : 
    1013           0 :     if (fpOut == nullptr)
    1014           0 :         fpOut = stdout;
    1015             : 
    1016           0 :     fprintf(fpOut, "%d\t0x%8.8x  %-5d\t%-6d %-6d %5.3e  d=%5.3e", nOffset,
    1017             :             nValue, nValue, n16Val1, n16Val2, fValue, dValue);
    1018             : 
    1019           0 :     fprintf(fpOut, "\t[%c%c%c%c]\n", isprint(achValue[0]) ? achValue[0] : '.',
    1020           0 :             isprint(achValue[1]) ? achValue[1] : '.',
    1021           0 :             isprint(achValue[2]) ? achValue[2] : '.',
    1022           0 :             isprint(achValue[3]) ? achValue[3] : '.');
    1023           0 : }
    1024             : 
    1025             : /**********************************************************************
    1026             :  *                   TABCreateMAPBlockFromFile()
    1027             :  *
    1028             :  * Load data from the specified file location and create and initialize
    1029             :  * a TABMAP*Block of the right type to handle it.
    1030             :  *
    1031             :  * Returns the new object if successful or NULL if an error happened, in
    1032             :  * which case CPLError() will have been called.
    1033             :  **********************************************************************/
    1034       36102 : TABRawBinBlock *TABCreateMAPBlockFromFile(VSILFILE *fpSrc, int nOffset,
    1035             :                                           int nSize,
    1036             :                                           GBool bHardBlockSize /*= TRUE */,
    1037             :                                           TABAccess eAccessMode /*= TABRead*/)
    1038             : {
    1039       36102 :     if (fpSrc == nullptr || nSize == 0)
    1040             :     {
    1041           0 :         CPLError(CE_Failure, CPLE_AssertionFailed,
    1042             :                  "TABCreateMAPBlockFromFile(): Assertion Failed!");
    1043           0 :         return nullptr;
    1044             :     }
    1045             : 
    1046             :     /*----------------------------------------------------------------
    1047             :      * Alloc a buffer to contain the data
    1048             :      *---------------------------------------------------------------*/
    1049       36102 :     GByte *pabyBuf = static_cast<GByte *>(CPLMalloc(nSize * sizeof(GByte)));
    1050             : 
    1051             :     /*----------------------------------------------------------------
    1052             :      * Read from the file
    1053             :      *---------------------------------------------------------------*/
    1054       72204 :     if (VSIFSeekL(fpSrc, nOffset, SEEK_SET) != 0 ||
    1055       36102 :         VSIFReadL(pabyBuf, sizeof(GByte), nSize, fpSrc) !=
    1056       36102 :             static_cast<unsigned int>(nSize))
    1057             :     {
    1058          14 :         CPLError(
    1059             :             CE_Failure, CPLE_FileIO,
    1060             :             "TABCreateMAPBlockFromFile() failed reading %d bytes at offset %d.",
    1061             :             nSize, nOffset);
    1062          14 :         CPLFree(pabyBuf);
    1063          14 :         return nullptr;
    1064             :     }
    1065             : 
    1066             :     /*----------------------------------------------------------------
    1067             :      * Create an object of the right type
    1068             :      * Header block is different: it does not start with the object
    1069             :      * type byte but it is always the first block in a file
    1070             :      *---------------------------------------------------------------*/
    1071       36088 :     TABRawBinBlock *poBlock = nullptr;
    1072             : 
    1073       36088 :     if (nOffset == 0)
    1074             :     {
    1075        2636 :         poBlock = new TABMAPHeaderBlock(eAccessMode);
    1076             :     }
    1077             :     else
    1078             :     {
    1079       33452 :         switch (pabyBuf[0])
    1080             :         {
    1081       10654 :             case TABMAP_INDEX_BLOCK:
    1082       10654 :                 poBlock = new TABMAPIndexBlock(eAccessMode);
    1083       10654 :                 break;
    1084       22643 :             case TABMAP_OBJECT_BLOCK:
    1085       22643 :                 poBlock = new TABMAPObjectBlock(eAccessMode);
    1086       22643 :                 break;
    1087         153 :             case TABMAP_COORD_BLOCK:
    1088         153 :                 poBlock = new TABMAPCoordBlock(eAccessMode);
    1089         153 :                 break;
    1090           0 :             case TABMAP_TOOL_BLOCK:
    1091           0 :                 poBlock = new TABMAPToolBlock(eAccessMode);
    1092           0 :                 break;
    1093           2 :             case TABMAP_GARB_BLOCK:
    1094             :             default:
    1095           2 :                 poBlock = new TABRawBinBlock(eAccessMode, bHardBlockSize);
    1096           2 :                 break;
    1097             :         }
    1098             :     }
    1099             : 
    1100             :     /*----------------------------------------------------------------
    1101             :      * Init new object with the data we just read
    1102             :      *---------------------------------------------------------------*/
    1103       72176 :     if (poBlock->InitBlockFromData(pabyBuf, nSize, nSize, FALSE, fpSrc,
    1104       36088 :                                    nOffset) != 0)
    1105             :     {
    1106             :         // Some error happened... and CPLError() has been called
    1107           0 :         delete poBlock;
    1108           0 :         poBlock = nullptr;
    1109             :     }
    1110             : 
    1111       36088 :     return poBlock;
    1112             : }
    1113             : 
    1114             : /*=====================================================================
    1115             :  *                      class TABBinBlockManager
    1116             :  *====================================================================*/
    1117             : 
    1118             : /**********************************************************************
    1119             :  *                   TABBinBlockManager::TABBinBlockManager()
    1120             :  *
    1121             :  * Constructor.
    1122             :  **********************************************************************/
    1123        1452 : TABBinBlockManager::TABBinBlockManager()
    1124             :     : m_nBlockSize(0), m_nLastAllocatedBlock(-1),
    1125        1452 :       m_psGarbageBlocksFirst(nullptr), m_psGarbageBlocksLast(nullptr)
    1126             : {
    1127        1452 :     m_szName[0] = '\0';
    1128        1452 : }
    1129             : 
    1130             : /**********************************************************************
    1131             :  *                   TABBinBlockManager::~TABBinBlockManager()
    1132             :  *
    1133             :  * Destructor.
    1134             :  **********************************************************************/
    1135        2904 : TABBinBlockManager::~TABBinBlockManager()
    1136             : {
    1137        1452 :     Reset();
    1138        1452 : }
    1139             : 
    1140             : /**********************************************************************
    1141             :  *                   TABBinBlockManager::SetBlockSize()
    1142             :  **********************************************************************/
    1143        1475 : void TABBinBlockManager::SetBlockSize(int nBlockSize)
    1144             : {
    1145        1475 :     m_nBlockSize = nBlockSize;
    1146        1475 : }
    1147             : 
    1148             : /**********************************************************************
    1149             :  *                   TABBinBlockManager::SetName()
    1150             :  **********************************************************************/
    1151        1452 : void TABBinBlockManager::SetName(const char *pszName)
    1152             : {
    1153        1452 :     strncpy(m_szName, pszName, sizeof(m_szName));
    1154        1452 :     m_szName[sizeof(m_szName) - 1] = '\0';
    1155        1452 : }
    1156             : 
    1157             : /**********************************************************************
    1158             :  *                   TABBinBlockManager::AllocNewBlock()
    1159             :  *
    1160             :  * Returns and reserves the address of the next available block, either a
    1161             :  * brand new block at end of file, or recycle a garbage block if one is
    1162             :  * available.
    1163             :  **********************************************************************/
    1164         859 : GInt32 TABBinBlockManager::AllocNewBlock(CPL_UNUSED const char *pszReason)
    1165             : {
    1166             :     // Try to reuse garbage blocks first
    1167         859 :     if (GetFirstGarbageBlock() > 0)
    1168             :     {
    1169          13 :         int nRetValue = PopGarbageBlock();
    1170             : #ifdef DEBUG_VERBOSE
    1171             :         CPLDebug("MITAB",
    1172             :                  "AllocNewBlock(%s, %s) = %d (recycling garbage block)",
    1173             :                  m_szName, pszReason, nRetValue);
    1174             : #endif
    1175          13 :         return nRetValue;
    1176             :     }
    1177             : 
    1178             :     // ... or alloc a new block at EOF
    1179         846 :     if (m_nLastAllocatedBlock == -1)
    1180          48 :         m_nLastAllocatedBlock = 0;
    1181             :     else
    1182             :     {
    1183         798 :         CPLAssert(m_nBlockSize);
    1184         798 :         m_nLastAllocatedBlock += m_nBlockSize;
    1185             :     }
    1186             : 
    1187             : #ifdef DEBUG_VERBOSE
    1188             :     CPLDebug("MITAB", "AllocNewBlock(%s, %s) = %d", m_szName, pszReason,
    1189             :              m_nLastAllocatedBlock);
    1190             : #endif
    1191         846 :     return m_nLastAllocatedBlock;
    1192             : }
    1193             : 
    1194             : /**********************************************************************
    1195             :  *                   TABBinBlockManager::Reset()
    1196             :  *
    1197             :  **********************************************************************/
    1198        2943 : void TABBinBlockManager::Reset()
    1199             : {
    1200        2943 :     m_nLastAllocatedBlock = -1;
    1201             : 
    1202             :     // Flush list of garbage blocks
    1203        2943 :     while (m_psGarbageBlocksFirst != nullptr)
    1204             :     {
    1205           0 :         TABBlockRef *psNext = m_psGarbageBlocksFirst->psNext;
    1206           0 :         CPLFree(m_psGarbageBlocksFirst);
    1207           0 :         m_psGarbageBlocksFirst = psNext;
    1208             :     }
    1209        2943 :     m_psGarbageBlocksLast = nullptr;
    1210        2943 : }
    1211             : 
    1212             : /**********************************************************************
    1213             :  *                   TABBinBlockManager::PushGarbageBlockAsFirst()
    1214             :  *
    1215             :  * Insert a garbage block at the head of the list of garbage blocks.
    1216             :  **********************************************************************/
    1217          13 : void TABBinBlockManager::PushGarbageBlockAsFirst(GInt32 nBlockPtr)
    1218             : {
    1219             :     TABBlockRef *psNewBlockRef =
    1220          13 :         static_cast<TABBlockRef *>(CPLMalloc(sizeof(TABBlockRef)));
    1221             : 
    1222          13 :     psNewBlockRef->nBlockPtr = nBlockPtr;
    1223          13 :     psNewBlockRef->psPrev = nullptr;
    1224          13 :     psNewBlockRef->psNext = m_psGarbageBlocksFirst;
    1225             : 
    1226          13 :     if (m_psGarbageBlocksFirst != nullptr)
    1227           0 :         m_psGarbageBlocksFirst->psPrev = psNewBlockRef;
    1228          13 :     m_psGarbageBlocksFirst = psNewBlockRef;
    1229          13 :     if (m_psGarbageBlocksLast == nullptr)
    1230          13 :         m_psGarbageBlocksLast = m_psGarbageBlocksFirst;
    1231          13 : }
    1232             : 
    1233             : /**********************************************************************
    1234             :  *                   TABBinBlockManager::PushGarbageBlockAsLast()
    1235             :  *
    1236             :  * Insert a garbage block at the tail of the list of garbage blocks.
    1237             :  **********************************************************************/
    1238           0 : void TABBinBlockManager::PushGarbageBlockAsLast(GInt32 nBlockPtr)
    1239             : {
    1240             :     TABBlockRef *psNewBlockRef =
    1241           0 :         static_cast<TABBlockRef *>(CPLMalloc(sizeof(TABBlockRef)));
    1242             : 
    1243           0 :     psNewBlockRef->nBlockPtr = nBlockPtr;
    1244           0 :     psNewBlockRef->psPrev = m_psGarbageBlocksLast;
    1245           0 :     psNewBlockRef->psNext = nullptr;
    1246             : 
    1247           0 :     if (m_psGarbageBlocksLast != nullptr)
    1248           0 :         m_psGarbageBlocksLast->psNext = psNewBlockRef;
    1249           0 :     m_psGarbageBlocksLast = psNewBlockRef;
    1250           0 :     if (m_psGarbageBlocksFirst == nullptr)
    1251           0 :         m_psGarbageBlocksFirst = m_psGarbageBlocksLast;
    1252           0 : }
    1253             : 
    1254             : /**********************************************************************
    1255             :  *                   TABBinBlockManager::GetFirstGarbageBlock()
    1256             :  *
    1257             :  * Return address of the block at the head of the list of garbage blocks
    1258             :  * or 0 if the list is empty.
    1259             :  **********************************************************************/
    1260        2073 : GInt32 TABBinBlockManager::GetFirstGarbageBlock()
    1261             : {
    1262        2073 :     if (m_psGarbageBlocksFirst)
    1263          13 :         return m_psGarbageBlocksFirst->nBlockPtr;
    1264             : 
    1265        2060 :     return 0;
    1266             : }
    1267             : 
    1268             : /**********************************************************************
    1269             :  *                   TABBinBlockManager::PopGarbageBlock()
    1270             :  *
    1271             :  * Return address of the block at the head of the list of garbage blocks
    1272             :  * and remove that block from the list.
    1273             :  * Returns 0 if the list is empty.
    1274             :  **********************************************************************/
    1275          13 : GInt32 TABBinBlockManager::PopGarbageBlock()
    1276             : {
    1277          13 :     GInt32 nBlockPtr = 0;
    1278             : 
    1279          13 :     if (m_psGarbageBlocksFirst)
    1280             :     {
    1281          13 :         nBlockPtr = m_psGarbageBlocksFirst->nBlockPtr;
    1282          13 :         TABBlockRef *psNext = m_psGarbageBlocksFirst->psNext;
    1283          13 :         CPLFree(m_psGarbageBlocksFirst);
    1284          13 :         if (psNext != nullptr)
    1285           0 :             psNext->psPrev = nullptr;
    1286             :         else
    1287          13 :             m_psGarbageBlocksLast = nullptr;
    1288          13 :         m_psGarbageBlocksFirst = psNext;
    1289             :     }
    1290             : 
    1291          13 :     return nBlockPtr;
    1292             : }

Generated by: LCOV version 1.14