LCOV - code coverage report
Current view: top level - ogr/ogrsf_frmts/mitab - mitab_mapcoordblock.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 181 259 69.9 %
Date: 2025-01-18 12:42:00 Functions: 18 20 90.0 %

          Line data    Source code
       1             : /**********************************************************************
       2             :  *
       3             :  * Name:     mitab_mapcoordblock.cpp
       4             :  * Project:  MapInfo TAB Read/Write library
       5             :  * Language: C++
       6             :  * Purpose:  Implementation of the TABMAPCoordBlock class used to handle
       7             :  *           reading/writing of the .MAP files' coordinate blocks
       8             :  * Author:   Daniel Morissette, dmorissette@dmsolutions.ca
       9             :  *
      10             :  **********************************************************************
      11             :  * Copyright (c) 1999-2001, 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 <climits>
      21             : #include <cstddef>
      22             : #include <algorithm>
      23             : 
      24             : #include "mitab_priv.h"
      25             : #include "mitab_utils.h"
      26             : #include "cpl_conv.h"
      27             : #include "cpl_error.h"
      28             : #include "cpl_vsi.h"
      29             : 
      30             : /*=====================================================================
      31             :  *                      class TABMAPCoordBlock
      32             :  *====================================================================*/
      33             : 
      34             : constexpr int MAP_COORD_HEADER_SIZE = 8;
      35             : 
      36             : /**********************************************************************
      37             :  *                   TABMAPCoordBlock::TABMAPCoordBlock()
      38             :  *
      39             :  * Constructor.
      40             :  **********************************************************************/
      41         236 : TABMAPCoordBlock::TABMAPCoordBlock(TABAccess eAccessMode /*= TABRead*/)
      42             :     : TABRawBinBlock(eAccessMode, TRUE), m_numDataBytes(0),
      43             :       m_nNextCoordBlock(0), m_numBlocksInChain(1),  // Current block counts as 1
      44             :       m_nComprOrgX(0), m_nComprOrgY(0), m_nMinX(1000000000),
      45             :       m_nMinY(1000000000), m_nMaxX(-1000000000), m_nMaxY(-1000000000),
      46             :       m_poBlockManagerRef(nullptr), m_nTotalDataSize(0), m_nFeatureDataSize(0),
      47             :       m_nFeatureXMin(1000000000), m_nFeatureYMin(1000000000),
      48         236 :       m_nFeatureXMax(-1000000000), m_nFeatureYMax(-1000000000)
      49             : {
      50         236 : }
      51             : 
      52             : /**********************************************************************
      53             :  *                   TABMAPCoordBlock::~TABMAPCoordBlock()
      54             :  *
      55             :  * Destructor.
      56             :  **********************************************************************/
      57         472 : TABMAPCoordBlock::~TABMAPCoordBlock()
      58             : {
      59         472 : }
      60             : 
      61             : /**********************************************************************
      62             :  *                   TABMAPCoordBlock::InitBlockFromData()
      63             :  *
      64             :  * Perform some initialization on the block after its binary data has
      65             :  * been set or changed (or loaded from a file).
      66             :  *
      67             :  * Returns 0 if successful or -1 if an error happened, in which case
      68             :  * CPLError() will have been called.
      69             :  **********************************************************************/
      70         650 : int TABMAPCoordBlock::InitBlockFromData(GByte *pabyBuf, int nBlockSize,
      71             :                                         int nSizeUsed,
      72             :                                         GBool bMakeCopy /* = TRUE */,
      73             :                                         VSILFILE *fpSrc /* = NULL */,
      74             :                                         int nOffset /* = 0 */)
      75             : {
      76             : #ifdef DEBUG_VERBOSE
      77             :     CPLDebug("MITAB", "Instantiating COORD block to/from offset %d", nOffset);
      78             : #endif
      79             :     /*-----------------------------------------------------------------
      80             :      * First of all, we must call the base class' InitBlockFromData()
      81             :      *----------------------------------------------------------------*/
      82         650 :     const int nStatus = TABRawBinBlock::InitBlockFromData(
      83             :         pabyBuf, nBlockSize, nSizeUsed, bMakeCopy, fpSrc, nOffset);
      84         650 :     if (nStatus != 0)
      85           0 :         return nStatus;
      86             : 
      87             :     /*-----------------------------------------------------------------
      88             :      * Validate block type
      89             :      *----------------------------------------------------------------*/
      90         650 :     if (m_nBlockType != TABMAP_COORD_BLOCK)
      91             :     {
      92           0 :         CPLError(CE_Failure, CPLE_FileIO,
      93             :                  "InitBlockFromData(): Invalid Block Type: got %d expected %d",
      94             :                  m_nBlockType, TABMAP_COORD_BLOCK);
      95           0 :         CPLFree(m_pabyBuf);
      96           0 :         m_pabyBuf = nullptr;
      97           0 :         return -1;
      98             :     }
      99             : 
     100             :     /*-----------------------------------------------------------------
     101             :      * Init member variables
     102             :      *----------------------------------------------------------------*/
     103         650 :     GotoByteInBlock(0x002);
     104         650 :     m_numDataBytes = ReadInt16(); /* Excluding 8 bytes header */
     105         650 :     if (m_numDataBytes < 0 ||
     106         650 :         m_numDataBytes + MAP_COORD_HEADER_SIZE > nBlockSize)
     107             :     {
     108           0 :         CPLError(CE_Failure, CPLE_FileIO,
     109             :                  "TABMAPCoordBlock::InitBlockFromData(): m_numDataBytes=%d "
     110             :                  "incompatible with block size %d",
     111             :                  m_numDataBytes, nBlockSize);
     112           0 :         CPLFree(m_pabyBuf);
     113           0 :         m_pabyBuf = nullptr;
     114           0 :         return -1;
     115             :     }
     116             : 
     117         650 :     m_nNextCoordBlock = ReadInt32();
     118             : 
     119             :     // Set the real SizeUsed based on numDataBytes
     120         650 :     m_nSizeUsed = m_numDataBytes + MAP_COORD_HEADER_SIZE;
     121             : 
     122             :     /*-----------------------------------------------------------------
     123             :      * The read ptr is now located at the beginning of the data part.
     124             :      *----------------------------------------------------------------*/
     125         650 :     GotoByteInBlock(MAP_COORD_HEADER_SIZE);
     126             : 
     127         650 :     return 0;
     128             : }
     129             : 
     130             : /**********************************************************************
     131             :  *                   TABMAPCoordBlock::CommitToFile()
     132             :  *
     133             :  * Commit the current state of the binary block to the file to which
     134             :  * it has been previously attached.
     135             :  *
     136             :  * This method makes sure all values are properly set in the map object
     137             :  * block header and then calls TABRawBinBlock::CommitToFile() to do
     138             :  * the actual writing to disk.
     139             :  *
     140             :  * Returns 0 if successful or -1 if an error happened, in which case
     141             :  * CPLError() will have been called.
     142             :  **********************************************************************/
     143         477 : int TABMAPCoordBlock::CommitToFile()
     144             : {
     145         477 :     int nStatus = 0;
     146             : 
     147         477 :     CPLErrorReset();
     148             : 
     149         477 :     if (m_pabyBuf == nullptr)
     150             :     {
     151           0 :         CPLError(CE_Failure, CPLE_AssertionFailed,
     152             :                  "CommitToFile(): Block has not been initialized yet!");
     153           0 :         return -1;
     154             :     }
     155             : 
     156             :     /*-----------------------------------------------------------------
     157             :      * Nothing to do here if block has not been modified
     158             :      *----------------------------------------------------------------*/
     159         477 :     if (!m_bModified)
     160         192 :         return 0;
     161             : 
     162             :     /*-----------------------------------------------------------------
     163             :      * Make sure 8 bytes block header is up to date.
     164             :      *----------------------------------------------------------------*/
     165         285 :     GotoByteInBlock(0x000);
     166             : 
     167         285 :     WriteInt16(TABMAP_COORD_BLOCK);  // Block type code
     168         285 :     CPLAssert(m_nSizeUsed >= MAP_COORD_HEADER_SIZE &&
     169             :               m_nSizeUsed < MAP_COORD_HEADER_SIZE + 32768);
     170         285 :     WriteInt16(static_cast<GInt16>(m_nSizeUsed -
     171             :                                    MAP_COORD_HEADER_SIZE));  // num. bytes used
     172         285 :     WriteInt32(m_nNextCoordBlock);
     173             : 
     174         285 :     if (CPLGetLastErrorType() == CE_Failure)
     175           0 :         nStatus = -1;
     176             : 
     177             :     /*-----------------------------------------------------------------
     178             :      * OK, call the base class to write the block to disk.
     179             :      *----------------------------------------------------------------*/
     180         285 :     if (nStatus == 0)
     181             :     {
     182             : #ifdef DEBUG_VERBOSE
     183             :         CPLDebug("MITAB", "Committing COORD block to offset %d", m_nFileOffset);
     184             : #endif
     185         285 :         nStatus = TABRawBinBlock::CommitToFile();
     186             :     }
     187             : 
     188         285 :     return nStatus;
     189             : }
     190             : 
     191             : /**********************************************************************
     192             :  *                   TABMAPCoordBlock::InitNewBlock()
     193             :  *
     194             :  * Initialize a newly created block so that it knows to which file it
     195             :  * is attached, its block size, etc . and then perform any specific
     196             :  * initialization for this block type, including writing a default
     197             :  * block header, etc. and leave the block ready to receive data.
     198             :  *
     199             :  * This is an alternative to calling ReadFromFile() or InitBlockFromData()
     200             :  * that puts the block in a stable state without loading any initial
     201             :  * data in it.
     202             :  *
     203             :  * Returns 0 if successful or -1 if an error happened, in which case
     204             :  * CPLError() will have been called.
     205             :  **********************************************************************/
     206         151 : int TABMAPCoordBlock::InitNewBlock(VSILFILE *fpSrc, int nBlockSize,
     207             :                                    int nFileOffset /* = 0*/)
     208             : {
     209         151 :     CPLErrorReset();
     210             : #ifdef DEBUG_VERBOSE
     211             :     CPLDebug("MITAB", "Instantiating new COORD block at offset %d",
     212             :              nFileOffset);
     213             : #endif
     214             :     /*-----------------------------------------------------------------
     215             :      * Start with the default initialization
     216             :      *----------------------------------------------------------------*/
     217         151 :     if (TABRawBinBlock::InitNewBlock(fpSrc, nBlockSize, nFileOffset) != 0)
     218           0 :         return -1;
     219             : 
     220             :     /*-----------------------------------------------------------------
     221             :      * And then set default values for the block header.
     222             :      *
     223             :      * IMPORTANT: Do not reset m_nComprOrg here because its value needs to be
     224             :      * maintained between blocks in the same chain.
     225             :      *----------------------------------------------------------------*/
     226         151 :     m_nNextCoordBlock = 0;
     227             : 
     228         151 :     m_numDataBytes = 0;
     229             : 
     230             :     // m_nMin/Max are used to keep track of current block MBR
     231             :     // FeatureMin/Max should not be reset here since feature coords can
     232             :     // be split on several blocks
     233         151 :     m_nMinX = 1000000000;
     234         151 :     m_nMinY = 1000000000;
     235         151 :     m_nMaxX = -1000000000;
     236         151 :     m_nMaxY = -1000000000;
     237             : 
     238         151 :     if (m_eAccess != TABRead && nFileOffset != 0)
     239             :     {
     240         129 :         GotoByteInBlock(0x000);
     241             : 
     242         129 :         WriteInt16(TABMAP_COORD_BLOCK);  // Block type code
     243         129 :         WriteInt16(0);                   // num. bytes used, excluding header
     244         129 :         WriteInt32(0);                   // Pointer to next coord block
     245             :     }
     246             : 
     247         151 :     if (CPLGetLastErrorType() == CE_Failure)
     248           0 :         return -1;
     249             : 
     250         151 :     return 0;
     251             : }
     252             : 
     253             : /**********************************************************************
     254             :  *                   TABMAPObjectBlock::SetNextCoordBlock()
     255             :  *
     256             :  * Set the address (offset from beginning of file) of the coord. block
     257             :  * that follows the current one.
     258             :  **********************************************************************/
     259          68 : void TABMAPCoordBlock::SetNextCoordBlock(GInt32 nNextCoordBlockAddress)
     260             : {
     261          68 :     m_nNextCoordBlock = nNextCoordBlockAddress;
     262          68 :     m_bModified = TRUE;
     263          68 : }
     264             : 
     265             : /**********************************************************************
     266             :  *                   TABMAPObjectBlock::SetComprCoordOrigin()
     267             :  *
     268             :  * Set the Compressed integer coordinates space origin to be used when
     269             :  * reading compressed coordinates using ReadIntCoord().
     270             :  **********************************************************************/
     271        2905 : void TABMAPCoordBlock::SetComprCoordOrigin(GInt32 nX, GInt32 nY)
     272             : {
     273        2905 :     m_nComprOrgX = nX;
     274        2905 :     m_nComprOrgY = nY;
     275        2905 : }
     276             : 
     277             : /**********************************************************************
     278             :  *                   TABMAPObjectBlock::ReadIntCoord()
     279             :  *
     280             :  * Read the next pair of integer coordinates value from the block, and
     281             :  * apply the translation relative to the origin of the coord. space
     282             :  * previously set using SetComprCoordOrigin() if bCompressed=TRUE.
     283             :  *
     284             :  * This means that the returned coordinates are always absolute integer
     285             :  * coordinates, even when the source coords are in compressed form.
     286             :  *
     287             :  * Returns 0 if successful or -1 if an error happened, in which case
     288             :  * CPLError() will have been called.
     289             :  **********************************************************************/
     290       12442 : int TABMAPCoordBlock::ReadIntCoord(GBool bCompressed, GInt32 &nX, GInt32 &nY)
     291             : {
     292       12442 :     if (bCompressed)
     293             :     {
     294        7332 :         nX = ReadInt16();
     295        7332 :         nY = ReadInt16();
     296        7332 :         TABSaturatedAdd(nX, m_nComprOrgX);
     297        7332 :         TABSaturatedAdd(nY, m_nComprOrgY);
     298             :     }
     299             :     else
     300             :     {
     301        5110 :         nX = ReadInt32();
     302        5110 :         nY = ReadInt32();
     303             :     }
     304             : 
     305       12442 :     if (CPLGetLastErrorType() == CE_Failure)
     306           0 :         return -1;
     307             : 
     308       12442 :     return 0;
     309             : }
     310             : 
     311             : /**********************************************************************
     312             :  *                   TABMAPObjectBlock::ReadIntCoords()
     313             :  *
     314             :  * Read the specified number of pairs of X,Y integer coordinates values
     315             :  * from the block, and apply the translation relative to the origin of
     316             :  * the coord. space previously set using SetComprCoordOrigin() if
     317             :  * bCompressed=TRUE.
     318             :  *
     319             :  * This means that the returned coordinates are always absolute integer
     320             :  * coordinates, even when the source coords are in compressed form.
     321             :  *
     322             :  * panXY should point to an array big enough to receive the specified
     323             :  * number of coordinates.
     324             :  *
     325             :  * Returns 0 if successful or -1 if an error happened, in which case
     326             :  * CPLError() will have been called.
     327             :  **********************************************************************/
     328         524 : int TABMAPCoordBlock::ReadIntCoords(GBool bCompressed, int numCoordPairs,
     329             :                                     GInt32 *panXY)
     330             : {
     331         524 :     int i, numValues = numCoordPairs * 2;
     332             : 
     333         524 :     if (bCompressed)
     334             :     {
     335        9987 :         for (i = 0; i < numValues; i += 2)
     336             :         {
     337        9518 :             panXY[i] = ReadInt16();
     338        9518 :             panXY[i + 1] = ReadInt16();
     339        9518 :             TABSaturatedAdd(panXY[i], m_nComprOrgX);
     340        9518 :             TABSaturatedAdd(panXY[i + 1], m_nComprOrgY);
     341        9518 :             if (CPLGetLastErrorType() == CE_Failure)
     342           0 :                 return -1;
     343             :         }
     344             :     }
     345             :     else
     346             :     {
     347        4040 :         for (i = 0; i < numValues; i += 2)
     348             :         {
     349        3985 :             panXY[i] = ReadInt32();
     350        3985 :             panXY[i + 1] = ReadInt32();
     351        3985 :             if (CPLGetLastErrorType() == CE_Failure)
     352           0 :                 return -1;
     353             :         }
     354             :     }
     355             : 
     356         524 :     return 0;
     357             : }
     358             : 
     359             : /**********************************************************************
     360             :  *                   TABMAPObjectBlock::ReadCoordSecHdrs()
     361             :  *
     362             :  * Read a set of coordinate section headers for PLINE MULTIPLE or REGIONs
     363             :  * and store the result in the array of structures pasHdrs[].  It is assumed
     364             :  * that pasHdrs points to an allocated array of at least numSections
     365             :  * TABMAPCoordSecHdr structures.
     366             :  *
     367             :  * The function will also set the values of numVerticesTotal to the
     368             :  * total number of coordinates in the object (the sum of all sections
     369             :  * headers read).
     370             :  *
     371             :  * At the end of the call, this TABMAPCoordBlock object will be located
     372             :  * at the beginning of the coordinate data.
     373             :  *
     374             :  * In V450 the numVertices is stored on an int32 instead of an int16
     375             :  *
     376             :  * In V800 the numHoles is stored on an int32 instead of an int16
     377             :  *
     378             :  * IMPORTANT: This function makes the assumption that coordinates for all
     379             :  *            the sections are grouped together immediately after the
     380             :  *            last section header block (i.e. that the coord. data is not
     381             :  *            located all over the place).  If it is not the case then
     382             :  *            an error will be produced and the code to read region and
     383             :  *            multipline objects will have to be updated.
     384             :  *
     385             :  * Returns 0 if successful or -1 if an error happened, in which case
     386             :  * CPLError() will have been called.
     387             :  **********************************************************************/
     388         524 : int TABMAPCoordBlock::ReadCoordSecHdrs(GBool bCompressed, int nVersion,
     389             :                                        int numSections,
     390             :                                        TABMAPCoordSecHdr *pasHdrs,
     391             :                                        GInt32 &numVerticesTotal)
     392             : {
     393         524 :     CPLErrorReset();
     394             : 
     395             :     /*-------------------------------------------------------------
     396             :      * Note about header+vertices size vs compressed coordinates:
     397             :      * The uncompressed header sections are actually 16 bytes, but the
     398             :      * offset calculations are based on prior decompression of the
     399             :      * coordinates.  Our coordinate offset calculations have
     400             :      * to take this fact into account.
     401             :      * Also, V450 header section uses int32 instead of int16 for numVertices
     402             :      * and we add another 2 bytes to align with a 4 bytes boundary.
     403             :      * V800 header section uses int32 for numHoles but there is no need
     404             :      * for the 2 alignment bytes so the size is the same as V450
     405             :      *------------------------------------------------------------*/
     406         524 :     const int nSectionSize = (nVersion >= 450) ? 28 : 24;
     407         524 :     if (numSections > INT_MAX / nSectionSize)
     408             :     {
     409           0 :         CPLError(CE_Failure, CPLE_AssertionFailed, "Invalid numSections");
     410           0 :         return -1;
     411             :     }
     412         524 :     const int nTotalHdrSizeUncompressed = nSectionSize * numSections;
     413             : 
     414         524 :     const int nVertexSize =
     415             :         bCompressed ? 2 * sizeof(GUInt16) : 2 * sizeof(GUInt32);
     416         524 :     numVerticesTotal = 0;
     417             : 
     418        1054 :     for (int i = 0; i < numSections; i++)
     419             :     {
     420             :         /*-------------------------------------------------------------
     421             :          * Read the coord. section header blocks
     422             :          *------------------------------------------------------------*/
     423             : #ifdef TABDUMP
     424             :         int nHdrAddress = GetCurAddress();
     425             : #endif
     426         530 :         if (nVersion >= 450)
     427           8 :             pasHdrs[i].numVertices = ReadInt32();
     428             :         else
     429         522 :             pasHdrs[i].numVertices = ReadInt16();
     430             : 
     431         530 :         if (pasHdrs[i].numVertices < 0 ||
     432         530 :             pasHdrs[i].numVertices > INT_MAX / nVertexSize)
     433             :         {
     434           0 :             CPLError(CE_Failure, CPLE_AssertionFailed,
     435             :                      "Invalid number of vertices for section %d", i);
     436           0 :             return -1;
     437             :         }
     438         530 :         if (nVersion >= 800)
     439           0 :             pasHdrs[i].numHoles = ReadInt32();
     440             :         else
     441         530 :             pasHdrs[i].numHoles = ReadInt16();
     442         530 :         if (pasHdrs[i].numHoles < 0)
     443             :         {
     444           0 :             CPLError(CE_Failure, CPLE_AssertionFailed,
     445             :                      "Invalid number of holes for section %d", i);
     446           0 :             return -1;
     447             :         }
     448         530 :         ReadIntCoord(bCompressed, pasHdrs[i].nXMin, pasHdrs[i].nYMin);
     449         530 :         ReadIntCoord(bCompressed, pasHdrs[i].nXMax, pasHdrs[i].nYMax);
     450         530 :         pasHdrs[i].nDataOffset = ReadInt32();
     451         530 :         if (pasHdrs[i].nDataOffset < nTotalHdrSizeUncompressed)
     452             :         {
     453           0 :             CPLError(CE_Failure, CPLE_AssertionFailed,
     454             :                      "Invalid data offset for section %d", i);
     455           0 :             return -1;
     456             :         }
     457             : 
     458         530 :         if (CPLGetLastErrorType() != 0)
     459           0 :             return -1;
     460             : 
     461         530 :         if (numVerticesTotal > INT_MAX / nVertexSize - pasHdrs[i].numVertices)
     462             :         {
     463           0 :             CPLError(CE_Failure, CPLE_AssertionFailed,
     464             :                      "Invalid number of vertices for section %d", i);
     465           0 :             return -1;
     466             :         }
     467         530 :         numVerticesTotal += pasHdrs[i].numVertices;
     468             : 
     469         530 :         pasHdrs[i].nVertexOffset =
     470         530 :             (pasHdrs[i].nDataOffset - nTotalHdrSizeUncompressed) / 8;
     471             : #ifdef TABDUMP
     472             :         printf("READING pasHdrs[%d] @ %d = \n" /*ok*/
     473             :                "              { numVertices = %d, numHoles = %d, \n"
     474             :                "                nXMin=%d, nYMin=%d, nXMax=%d, nYMax=%d,\n"
     475             :                "                nDataOffset=%d, nVertexOffset=%d }\n",
     476             :                i, nHdrAddress, pasHdrs[i].numVertices, pasHdrs[i].numHoles,
     477             :                pasHdrs[i].nXMin, pasHdrs[i].nYMin, pasHdrs[i].nXMax,
     478             :                pasHdrs[i].nYMax, pasHdrs[i].nDataOffset,
     479             :                pasHdrs[i].nVertexOffset);
     480             :         printf("                dX = %d, dY = %d  (center = %d , %d)\n", /*ok*/
     481             :                pasHdrs[i].nXMax - pasHdrs[i].nXMin,
     482             :                pasHdrs[i].nYMax - pasHdrs[i].nYMin, m_nComprOrgX, m_nComprOrgY);
     483             : #endif
     484             :     }
     485             : 
     486        1054 :     for (int i = 0; i < numSections; i++)
     487             :     {
     488             :         /*-------------------------------------------------------------
     489             :          * Make sure all coordinates are grouped together
     490             :          * (Well... at least check that all the vertex indices are enclosed
     491             :          * inside the [0..numVerticesTotal] range.)
     492             :          *------------------------------------------------------------*/
     493         530 :         if (pasHdrs[i].nVertexOffset < 0 ||
     494         530 :             pasHdrs[i].nVertexOffset > INT_MAX - pasHdrs[i].numVertices ||
     495         530 :             (pasHdrs[i].nVertexOffset + pasHdrs[i].numVertices) >
     496         530 :                 numVerticesTotal)
     497             :         {
     498           0 :             CPLError(CE_Failure, CPLE_AssertionFailed,
     499             :                      "Unsupported case or corrupt file: MULTIPLINE/REGION "
     500             :                      "object vertices do not appear to be grouped together.");
     501           0 :             return -1;
     502             :         }
     503             :     }
     504             : 
     505         524 :     return 0;
     506             : }
     507             : 
     508             : /**********************************************************************
     509             :  *                   TABMAPObjectBlock::WriteCoordSecHdrs()
     510             :  *
     511             :  * Write a set of coordinate section headers for PLINE MULTIPLE or REGIONs.
     512             :  * pasHdrs should point to an array of numSections TABMAPCoordSecHdr
     513             :  * structures that have been properly initialized.
     514             :  *
     515             :  * In V450 the numVertices is stored on an int32 instead of an int16
     516             :  *
     517             :  * In V800 the numHoles is stored on an int32 instead of an int16
     518             :  *
     519             :  * At the end of the call, this TABMAPCoordBlock object will be ready to
     520             :  * receive the coordinate data.
     521             :  *
     522             :  * Returns 0 if successful or -1 if an error happened, in which case
     523             :  * CPLError() will have been called.
     524             :  **********************************************************************/
     525          96 : int TABMAPCoordBlock::WriteCoordSecHdrs(int nVersion, int numSections,
     526             :                                         TABMAPCoordSecHdr *pasHdrs,
     527             :                                         GBool bCompressed /*=FALSE*/)
     528             : {
     529          96 :     CPLErrorReset();
     530             : 
     531         194 :     for (int i = 0; i < numSections; i++)
     532             :     {
     533             :         /*-------------------------------------------------------------
     534             :          * Write the coord. section header blocks
     535             :          *------------------------------------------------------------*/
     536             : #ifdef TABDUMP
     537             :         printf("WRITING pasHdrs[%d] @ %d = \n" /*ok*/
     538             :                "              { numVertices = %d, numHoles = %d, \n"
     539             :                "                nXMin=%d, nYMin=%d, nXMax=%d, nYMax=%d,\n"
     540             :                "                nDataOffset=%d, nVertexOffset=%d }\n",
     541             :                i, GetCurAddress(), pasHdrs[i].numVertices, pasHdrs[i].numHoles,
     542             :                pasHdrs[i].nXMin, pasHdrs[i].nYMin, pasHdrs[i].nXMax,
     543             :                pasHdrs[i].nYMax, pasHdrs[i].nDataOffset,
     544             :                pasHdrs[i].nVertexOffset);
     545             :         printf("                dX = %d, dY = %d  (center = %d , %d)\n", /*ok*/
     546             :                pasHdrs[i].nXMax - pasHdrs[i].nXMin,
     547             :                pasHdrs[i].nYMax - pasHdrs[i].nYMin, m_nComprOrgX, m_nComprOrgY);
     548             : #endif
     549             : 
     550          98 :         if (nVersion >= 450)
     551           0 :             WriteInt32(pasHdrs[i].numVertices);
     552             :         else
     553          98 :             WriteInt16(static_cast<GInt16>(pasHdrs[i].numVertices));
     554          98 :         if (nVersion >= 800)
     555           0 :             WriteInt32(pasHdrs[i].numHoles);
     556             :         else
     557          98 :             WriteInt16(static_cast<GInt16>(pasHdrs[i].numHoles));
     558          98 :         WriteIntCoord(pasHdrs[i].nXMin, pasHdrs[i].nYMin, bCompressed);
     559          98 :         WriteIntCoord(pasHdrs[i].nXMax, pasHdrs[i].nYMax, bCompressed);
     560          98 :         WriteInt32(pasHdrs[i].nDataOffset);
     561             : 
     562          98 :         if (CPLGetLastErrorType() == CE_Failure)
     563           0 :             return -1;
     564             :     }
     565             : 
     566          96 :     return 0;
     567             : }
     568             : 
     569             : /**********************************************************************
     570             :  *                   TABMAPCoordBlock::WriteIntCoord()
     571             :  *
     572             :  * Write a pair of integer coordinates values to the current position in the
     573             :  * the block.
     574             :  *
     575             :  * Returns 0 if successful or -1 if an error happened, in which case
     576             :  * CPLError() will have been called.
     577             :  **********************************************************************/
     578             : 
     579        7517 : int TABMAPCoordBlock::WriteIntCoord(GInt32 nX, GInt32 nY,
     580             :                                     GBool bCompressed /*=FALSE*/)
     581             : {
     582             : 
     583       11439 :     if ((!bCompressed && (WriteInt32(nX) != 0 || WriteInt32(nY) != 0)) ||
     584        3922 :         (bCompressed && (WriteInt16(TABInt16Diff(nX, m_nComprOrgX)) != 0 ||
     585        3922 :                          WriteInt16(TABInt16Diff(nY, m_nComprOrgY)) != 0)))
     586             :     {
     587           0 :         return -1;
     588             :     }
     589             : 
     590             :     /*-----------------------------------------------------------------
     591             :      * Update block MBR
     592             :      *----------------------------------------------------------------*/
     593             :     //__TODO__ Do we still need to track the block MBR???
     594        7517 :     if (nX < m_nMinX)
     595         639 :         m_nMinX = nX;
     596        7517 :     if (nX > m_nMaxX)
     597        3985 :         m_nMaxX = nX;
     598             : 
     599        7517 :     if (nY < m_nMinY)
     600         648 :         m_nMinY = nY;
     601        7517 :     if (nY > m_nMaxY)
     602        3544 :         m_nMaxY = nY;
     603             : 
     604             :     /*-------------------------------------------------------------
     605             :      * Also keep track of current feature MBR.
     606             :      *------------------------------------------------------------*/
     607        7517 :     if (nX < m_nFeatureXMin)
     608         733 :         m_nFeatureXMin = nX;
     609        7517 :     if (nX > m_nFeatureXMax)
     610        4345 :         m_nFeatureXMax = nX;
     611             : 
     612        7517 :     if (nY < m_nFeatureYMin)
     613         641 :         m_nFeatureYMin = nY;
     614        7517 :     if (nY > m_nFeatureYMax)
     615        2737 :         m_nFeatureYMax = nY;
     616             : 
     617        7517 :     return 0;
     618             : }
     619             : 
     620             : /**********************************************************************
     621             :  *                   TABMAPCoordBlock::SetMAPBlockManagerRef()
     622             :  *
     623             :  * Pass a reference to the block manager object for the file this
     624             :  * block belongs to.  The block manager will be used by this object
     625             :  * when it needs to automatically allocate a new block.
     626             :  **********************************************************************/
     627         236 : void TABMAPCoordBlock::SetMAPBlockManagerRef(TABBinBlockManager *poBlockMgr)
     628             : {
     629         236 :     m_poBlockManagerRef = poBlockMgr;
     630         236 : }
     631             : 
     632             : /**********************************************************************
     633             :  *                   TABMAPCoordBlock::ReadBytes()
     634             :  *
     635             :  * Cover function for TABRawBinBlock::ReadBytes() that will automagically
     636             :  * load the next coordinate block in the chain before reading the
     637             :  * requested bytes if we are at the end of the current block and if
     638             :  * m_nNextCoordBlock is a valid block.
     639             :  *
     640             :  * Then the control is passed to TABRawBinBlock::ReadBytes() to finish the
     641             :  * work:
     642             :  * Copy the number of bytes from the data block's internal buffer to
     643             :  * the user's buffer pointed by pabyDstBuf.
     644             :  *
     645             :  * Passing pabyDstBuf = NULL will only move the read pointer by the
     646             :  * specified number of bytes as if the copy had happened... but it
     647             :  * won't crash.
     648             :  *
     649             :  * Returns 0 if successful or -1 if an error happened, in which case
     650             :  * CPLError() will have been called.
     651             :  **********************************************************************/
     652       54860 : int TABMAPCoordBlock::ReadBytes(int numBytes, GByte *pabyDstBuf)
     653             : {
     654             : 
     655       54860 :     if (m_pabyBuf && m_nCurPos >= (m_numDataBytes + MAP_COORD_HEADER_SIZE) &&
     656         334 :         m_nNextCoordBlock > 0)
     657             :     {
     658             :         // We're at end of current block... advance to next block.
     659         191 :         int nStatus = GotoByteInFile(m_nNextCoordBlock, TRUE);
     660             : 
     661         191 :         if (nStatus != 0)
     662             :         {
     663             :             // Failed.... an error has already been reported.
     664           0 :             return nStatus;
     665             :         }
     666             : 
     667         191 :         GotoByteInBlock(MAP_COORD_HEADER_SIZE);  // Move pointer past header
     668         191 :         m_numBlocksInChain++;
     669             :     }
     670             : 
     671       54860 :     if (m_pabyBuf && m_nCurPos < (m_numDataBytes + MAP_COORD_HEADER_SIZE) &&
     672       54717 :         m_nCurPos + numBytes > (m_numDataBytes + MAP_COORD_HEADER_SIZE) &&
     673           0 :         m_nNextCoordBlock > 0)
     674             :     {
     675             :         // Data overlaps on more than one block
     676             :         // Read until end of this block and then recursively call ReadBytes()
     677             :         // for the rest.
     678           0 :         int numBytesInThisBlock =
     679           0 :             (m_numDataBytes + MAP_COORD_HEADER_SIZE) - m_nCurPos;
     680             :         int nStatus =
     681           0 :             TABRawBinBlock::ReadBytes(numBytesInThisBlock, pabyDstBuf);
     682           0 :         if (nStatus == 0)
     683             :             nStatus =
     684           0 :                 TABMAPCoordBlock::ReadBytes(numBytes - numBytesInThisBlock,
     685           0 :                                             pabyDstBuf + numBytesInThisBlock);
     686           0 :         return nStatus;
     687             :     }
     688             : 
     689       54860 :     return TABRawBinBlock::ReadBytes(numBytes, pabyDstBuf);
     690             : }
     691             : 
     692             : /**********************************************************************
     693             :  *                   TABMAPCoordBlock::WriteBytes()
     694             :  *
     695             :  * Cover function for TABRawBinBlock::WriteBytes() that will automagically
     696             :  * CommitToFile() the current block and create a new one if we are at
     697             :  * the end of the current block.
     698             :  *
     699             :  * Then the control is passed to TABRawBinBlock::WriteBytes() to finish the
     700             :  * work.
     701             :  *
     702             :  * Passing pabySrcBuf = NULL will only move the write pointer by the
     703             :  * specified number of bytes as if the copy had happened... but it
     704             :  * won't crash.
     705             :  *
     706             :  * Returns 0 if successful or -1 if an error happened, in which case
     707             :  * CPLError() will have been called.
     708             :  **********************************************************************/
     709       16599 : int TABMAPCoordBlock::WriteBytes(int nBytesToWrite, const GByte *pabySrcBuf)
     710             : {
     711       16599 :     if (m_eAccess != TABWrite && m_eAccess != TABReadWrite)
     712             :     {
     713           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     714             :                  "WriteBytes(): Block does not support write operations.");
     715           0 :         return -1;
     716             :     }
     717             : 
     718       16599 :     if (m_poBlockManagerRef && (m_nBlockSize - m_nCurPos) < nBytesToWrite)
     719             :     {
     720          68 :         if (nBytesToWrite <= (m_nBlockSize - MAP_COORD_HEADER_SIZE))
     721             :         {
     722             :             // Data won't fit in this block but can fit inside a single
     723             :             // block, so we'll allocate a new block for it.  This will
     724             :             // prevent us from overlapping coordinate values on 2 blocks, but
     725             :             // still allows strings longer than one block (see 'else' below).
     726             :             //
     727             : 
     728          68 :             if (m_nNextCoordBlock != 0)
     729             :             {
     730             :                 // We're in read/write mode and there is already an allocated
     731             :                 // block following this one in the chain ... just reload it
     732             :                 // and continue writing to it
     733             : 
     734           0 :                 CPLAssert(m_eAccess == TABReadWrite);
     735             : 
     736           0 :                 if (CommitToFile() != 0 ||
     737           0 :                     ReadFromFile(m_fp, m_nNextCoordBlock, m_nBlockSize) != 0)
     738             :                 {
     739             :                     // An error message should have already been reported.
     740           0 :                     return -1;
     741             :                 }
     742             :             }
     743             :             else
     744             :             {
     745             :                 // Need to alloc a new block.
     746             : 
     747             :                 int nNewBlockOffset =
     748          68 :                     m_poBlockManagerRef->AllocNewBlock("COORD");
     749          68 :                 SetNextCoordBlock(nNewBlockOffset);
     750             : 
     751         136 :                 if (CommitToFile() != 0 ||
     752          68 :                     InitNewBlock(m_fp, m_nBlockSize, nNewBlockOffset) != 0)
     753             :                 {
     754             :                     // An error message should have already been reported.
     755           0 :                     return -1;
     756             :                 }
     757             : 
     758          68 :                 m_numBlocksInChain++;
     759             :             }
     760             :         }
     761             :         else
     762             :         {
     763             :             // Data to write is longer than one block... so we'll have to
     764             :             // split it over multiple block through multiple calls.
     765             :             //
     766           0 :             int nStatus = 0;
     767           0 :             while (nStatus == 0 && nBytesToWrite > 0)
     768             :             {
     769           0 :                 int nBytes = m_nBlockSize - MAP_COORD_HEADER_SIZE;
     770           0 :                 if ((m_nBlockSize - m_nCurPos) > 0)
     771             :                 {
     772             :                     // Use free room in current block
     773           0 :                     nBytes = (m_nBlockSize - m_nCurPos);
     774             :                 }
     775             : 
     776           0 :                 nBytes = std::min(nBytes, nBytesToWrite);
     777             : 
     778             :                 // The following call will result in a new block being
     779             :                 // allocated in the if() block above.
     780           0 :                 nStatus = TABMAPCoordBlock::WriteBytes(nBytes, pabySrcBuf);
     781             : 
     782           0 :                 nBytesToWrite -= nBytes;
     783           0 :                 pabySrcBuf += nBytes;
     784             :             }
     785           0 :             return nStatus;
     786             :         }
     787             :     }
     788             : 
     789       16599 :     if (m_nCurPos >= MAP_COORD_HEADER_SIZE)
     790             :     {
     791             :         // Keep track of Coordinate data... this means ignore header bytes
     792             :         // that could be written.
     793       15331 :         m_nTotalDataSize += nBytesToWrite;
     794       15331 :         m_nFeatureDataSize += nBytesToWrite;
     795             :     }
     796             : 
     797       16599 :     return TABRawBinBlock::WriteBytes(nBytesToWrite, pabySrcBuf);
     798             : }
     799             : 
     800             : /**********************************************************************
     801             :  *                   TABMAPObjectBlock::SeekEnd()
     802             :  *
     803             :  * Move read/write pointer to end of used part of the block
     804             :  **********************************************************************/
     805         489 : void TABMAPCoordBlock::SeekEnd()
     806             : {
     807         489 :     m_nCurPos = m_nSizeUsed;
     808         489 : }
     809             : 
     810             : /**********************************************************************
     811             :  *                   TABMAPCoordBlock::StartNewFeature()
     812             :  *
     813             :  * Reset all member vars that are used to keep track of data size
     814             :  * and MBR for the current feature.  This is info is not needed by
     815             :  * the coord blocks themselves, but it helps a lot the callers to
     816             :  * have this class take care of that for them.
     817             :  *
     818             :  * See Also: GetFeatureDataSize() and GetFeatureMBR()
     819             :  **********************************************************************/
     820         489 : void TABMAPCoordBlock::StartNewFeature()
     821             : {
     822         489 :     m_nFeatureDataSize = 0;
     823             : 
     824         489 :     m_nFeatureXMin = 1000000000;
     825         489 :     m_nFeatureYMin = 1000000000;
     826         489 :     m_nFeatureXMax = -1000000000;
     827         489 :     m_nFeatureYMax = -1000000000;
     828         489 : }
     829             : 
     830             : /**********************************************************************
     831             :  *                   TABMAPCoordBlock::GetFeatureMBR()
     832             :  *
     833             :  * Return the MBR of all the coords written using WriteIntCoord() since
     834             :  * the last call to StartNewFeature().
     835             :  **********************************************************************/
     836           0 : void TABMAPCoordBlock::GetFeatureMBR(GInt32 &nXMin, GInt32 &nYMin,
     837             :                                      GInt32 &nXMax, GInt32 &nYMax)
     838             : {
     839           0 :     nXMin = m_nFeatureXMin;
     840           0 :     nYMin = m_nFeatureYMin;
     841           0 :     nXMax = m_nFeatureXMax;
     842           0 :     nYMax = m_nFeatureYMax;
     843           0 : }
     844             : 
     845             : /**********************************************************************
     846             :  *                   TABMAPCoordBlock::Dump()
     847             :  *
     848             :  * Dump block contents... available only in DEBUG mode.
     849             :  **********************************************************************/
     850             : #ifdef DEBUG
     851             : 
     852           0 : void TABMAPCoordBlock::Dump(FILE *fpOut /*=NULL*/)
     853             : {
     854           0 :     if (fpOut == nullptr)
     855           0 :         fpOut = stdout;
     856             : 
     857           0 :     fprintf(fpOut, "----- TABMAPCoordBlock::Dump() -----\n");
     858           0 :     if (m_pabyBuf == nullptr)
     859             :     {
     860           0 :         fprintf(fpOut, "Block has not been initialized yet.");
     861             :     }
     862             :     else
     863             :     {
     864           0 :         fprintf(fpOut, "Coordinate Block (type %d) at offset %d.\n",
     865             :                 m_nBlockType, m_nFileOffset);
     866           0 :         fprintf(fpOut, "  m_numDataBytes        = %d\n", m_numDataBytes);
     867           0 :         fprintf(fpOut, "  m_nNextCoordBlock     = %d\n", m_nNextCoordBlock);
     868             :     }
     869             : 
     870           0 :     fflush(fpOut);
     871           0 : }
     872             : 
     873             : #endif  // DEBUG

Generated by: LCOV version 1.14