LCOV - code coverage report
Current view: top level - frmts/nitf - rpftocfile.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 246 382 64.4 %
Date: 2025-01-18 12:42:00 Functions: 3 4 75.0 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  RPF A.TOC read Library
       4             :  * Purpose:  Module responsible for opening a RPF TOC file, populating RPFToc
       5             :  *           structure
       6             :  * Author:   Even Rouault, even.rouault at spatialys.com
       7             :  *
       8             :  **********************************************************************
       9             :  * Copyright (c) 2007-2010, Even Rouault <even dot rouault at spatialys.com>
      10             :  *
      11             :  * SPDX-License-Identifier: MIT
      12             :  ****************************************************************************/
      13             : 
      14             : /* Portions of code are placed under the following copyright : */
      15             : /*
      16             :  ******************************************************************************
      17             :  * Copyright (C) 1995 Logiciels et Applications Scientifiques (L.A.S.) Inc
      18             :  * Permission to use, copy, modify and distribute this software and
      19             :  * its documentation for any purpose and without fee is hereby granted,
      20             :  * provided that the above copyright notice appear in all copies, that
      21             :  * both the copyright notice and this permission notice appear in
      22             :  * supporting documentation, and that the name of L.A.S. Inc not be used
      23             :  * in advertising or publicity pertaining to distribution of the software
      24             :  * without specific, written prior permission. L.A.S. Inc. makes no
      25             :  * representations about the suitability of this software for any purpose.
      26             :  * It is provided "as is" without express or implied warranty.
      27             :  ******************************************************************************
      28             :  */
      29             : 
      30             : #include "cpl_port.h"
      31             : #include "rpftoclib.h"
      32             : 
      33             : #include <climits>
      34             : #include <cmath>
      35             : #include <cstring>
      36             : #if HAVE_FCNTL_H
      37             : #include <fcntl.h>
      38             : #endif
      39             : 
      40             : #include "cpl_conv.h"
      41             : #include "cpl_error.h"
      42             : #include "cpl_string.h"
      43             : #include "cpl_vsi.h"
      44             : #include "nitflib.h"
      45             : 
      46             : /************************************************************************/
      47             : /*                        RPFTOCTrim()                                    */
      48             : /************************************************************************/
      49             : 
      50          50 : static void RPFTOCTrim(char *str)
      51             : {
      52          50 :     char *c = str;
      53          50 :     if (str == nullptr || *str == 0)
      54           0 :         return;
      55             : 
      56          50 :     while (*c == ' ')
      57             :     {
      58           0 :         c++;
      59             :     }
      60          50 :     if (c != str)
      61             :     {
      62           0 :         memmove(str, c, strlen(c) + 1);
      63             :     }
      64             : 
      65          50 :     int i = static_cast<int>(strlen(str)) - 1;
      66          70 :     while (i >= 0 && str[i] == ' ')
      67             :     {
      68          20 :         str[i] = 0;
      69          20 :         i--;
      70             :     }
      71             : }
      72             : 
      73             : /************************************************************************/
      74             : /*                        RPFTOCRead()                                 */
      75             : /************************************************************************/
      76             : 
      77           0 : RPFToc *RPFTOCRead(const char *pszFilename, NITFFile *psFile)
      78             : {
      79             :     int nTRESize;
      80             :     const char *pachTRE =
      81           0 :         NITFFindTRE(psFile->pachTRE, psFile->nTREBytes, "RPFHDR", &nTRESize);
      82           0 :     if (pachTRE == nullptr)
      83             :     {
      84           0 :         CPLError(CE_Failure, CPLE_NotSupported,
      85             :                  "Invalid TOC file. Can't find RPFHDR.");
      86           0 :         return nullptr;
      87             :     }
      88             : 
      89           0 :     if (nTRESize != 48)
      90             :     {
      91           0 :         CPLError(CE_Failure, CPLE_NotSupported, "RPFHDR TRE wrong size.");
      92           0 :         return nullptr;
      93             :     }
      94             : 
      95           0 :     return RPFTOCReadFromBuffer(pszFilename, psFile->fp, pachTRE);
      96             : }
      97             : 
      98             : /* This function is directly inspired by function parse_toc coming from
      99             :  * ogdi/driver/rpf/utils.c */
     100             : 
     101          10 : RPFToc *RPFTOCReadFromBuffer(const char *pszFilename, VSILFILE *fp,
     102             :                              const char *tocHeader)
     103             : {
     104          10 :     tocHeader += 1;  /* skip endian */
     105          10 :     tocHeader += 2;  /* skip header length */
     106          10 :     tocHeader += 12; /* skip file name : this should be A.TOC (padded) */
     107          10 :     tocHeader += 1;  /* skip new  */
     108          10 :     tocHeader += 15; /* skip standard_num  */
     109          10 :     tocHeader += 8;  /* skip standard_date  */
     110          10 :     tocHeader += 1;  /* skip classification  */
     111          10 :     tocHeader += 2;  /* skip country  */
     112          10 :     tocHeader += 2;  /* skip release  */
     113             : 
     114             :     unsigned int locationSectionPhysicalLocation;
     115          10 :     memcpy(&locationSectionPhysicalLocation, tocHeader, sizeof(unsigned int));
     116          10 :     CPL_MSBPTR32(&locationSectionPhysicalLocation);
     117             : 
     118          10 :     if (VSIFSeekL(fp, locationSectionPhysicalLocation, SEEK_SET) != 0)
     119             :     {
     120           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     121             :                  "Invalid TOC file. Unable to seek to "
     122             :                  "locationSectionPhysicalLocation at offset %d.",
     123             :                  locationSectionPhysicalLocation);
     124           0 :         return nullptr;
     125             :     }
     126             : 
     127             :     int nSections;
     128          10 :     NITFLocation *pasLocations = NITFReadRPFLocationTable(fp, &nSections);
     129             : 
     130          10 :     unsigned int boundaryRectangleSectionSubHeaderPhysIndex = 0;
     131          10 :     unsigned int boundaryRectangleTablePhysIndex = 0;
     132          10 :     unsigned int frameFileIndexSectionSubHeaderPhysIndex = 0;
     133          10 :     unsigned int frameFileIndexSubsectionPhysIndex = 0;
     134             : 
     135          50 :     for (int i = 0; i < nSections; i++)
     136             :     {
     137          40 :         if (pasLocations[i].nLocId == LID_BoundaryRectangleSectionSubheader)
     138             :         {
     139          10 :             boundaryRectangleSectionSubHeaderPhysIndex =
     140          10 :                 pasLocations[i].nLocOffset;
     141             :         }
     142          30 :         else if (pasLocations[i].nLocId == LID_BoundaryRectangleTable)
     143             :         {
     144          10 :             boundaryRectangleTablePhysIndex = pasLocations[i].nLocOffset;
     145             :         }
     146          20 :         else if (pasLocations[i].nLocId == LID_FrameFileIndexSectionSubHeader)
     147             :         {
     148          10 :             frameFileIndexSectionSubHeaderPhysIndex =
     149          10 :                 pasLocations[i].nLocOffset;
     150             :         }
     151          10 :         else if (pasLocations[i].nLocId == LID_FrameFileIndexSubsection)
     152             :         {
     153          10 :             frameFileIndexSubsectionPhysIndex = pasLocations[i].nLocOffset;
     154             :         }
     155             :     }
     156             : 
     157          10 :     CPLFree(pasLocations);
     158             : 
     159          10 :     if (boundaryRectangleSectionSubHeaderPhysIndex == 0)
     160             :     {
     161           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     162             :                  "Invalid TOC file. Can't find "
     163             :                  "LID_BoundaryRectangleSectionSubheader.");
     164           0 :         return nullptr;
     165             :     }
     166          10 :     if (boundaryRectangleTablePhysIndex == 0)
     167             :     {
     168           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     169             :                  "Invalid TOC file. Can't find LID_BoundaryRectangleTable.");
     170           0 :         return nullptr;
     171             :     }
     172          10 :     if (frameFileIndexSectionSubHeaderPhysIndex == 0)
     173             :     {
     174           0 :         CPLError(
     175             :             CE_Failure, CPLE_NotSupported,
     176             :             "Invalid TOC file. Can't find LID_FrameFileIndexSectionSubHeader.");
     177           0 :         return nullptr;
     178             :     }
     179          10 :     if (frameFileIndexSubsectionPhysIndex == 0)
     180             :     {
     181           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     182             :                  "Invalid TOC file. Can't find LID_FrameFileIndexSubsection.");
     183           0 :         return nullptr;
     184             :     }
     185             : 
     186          10 :     if (VSIFSeekL(fp, boundaryRectangleSectionSubHeaderPhysIndex, SEEK_SET) !=
     187             :         0)
     188             :     {
     189           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     190             :                  "Invalid TOC file. Unable to seek to "
     191             :                  "boundaryRectangleSectionSubHeaderPhysIndex at offset %d.",
     192             :                  boundaryRectangleSectionSubHeaderPhysIndex);
     193           0 :         return nullptr;
     194             :     }
     195             : 
     196             :     unsigned int boundaryRectangleTableOffset;
     197          10 :     bool bOK = VSIFReadL(&boundaryRectangleTableOffset,
     198          10 :                          sizeof(boundaryRectangleTableOffset), 1, fp) == 1;
     199          10 :     CPL_MSBPTR32(&boundaryRectangleTableOffset);
     200             : 
     201             :     unsigned short boundaryRectangleCount;
     202          10 :     bOK &= VSIFReadL(&boundaryRectangleCount, sizeof(boundaryRectangleCount), 1,
     203          10 :                      fp) == 1;
     204          10 :     CPL_MSBPTR16(&boundaryRectangleCount);
     205             : 
     206          10 :     if (!bOK || VSIFSeekL(fp, boundaryRectangleTablePhysIndex, SEEK_SET) != 0)
     207             :     {
     208           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     209             :                  "Invalid TOC file. Unable to seek to "
     210             :                  "boundaryRectangleTablePhysIndex at offset %d.",
     211             :                  boundaryRectangleTablePhysIndex);
     212           0 :         return nullptr;
     213             :     }
     214             : 
     215          10 :     RPFToc *toc = reinterpret_cast<RPFToc *>(CPLMalloc(sizeof(RPFToc)));
     216          10 :     toc->nEntries = boundaryRectangleCount;
     217          10 :     toc->entries = reinterpret_cast<RPFTocEntry *>(
     218          10 :         CPLMalloc(boundaryRectangleCount * sizeof(RPFTocEntry)));
     219          10 :     memset(toc->entries, 0, boundaryRectangleCount * sizeof(RPFTocEntry));
     220             : 
     221          20 :     for (int i = 0; i < toc->nEntries; i++)
     222             :     {
     223          10 :         toc->entries[i].isOverviewOrLegend = 0;
     224             : 
     225          10 :         bOK &= VSIFReadL(toc->entries[i].type, 1, 5, fp) == 5;
     226          10 :         toc->entries[i].type[5] = 0;
     227          10 :         RPFTOCTrim(toc->entries[i].type);
     228             : 
     229          10 :         bOK &= VSIFReadL(toc->entries[i].compression, 1, 5, fp) == 5;
     230          10 :         toc->entries[i].compression[5] = 0;
     231          10 :         RPFTOCTrim(toc->entries[i].compression);
     232             : 
     233          10 :         bOK &= VSIFReadL(toc->entries[i].scale, 1, 12, fp) == 12;
     234          10 :         toc->entries[i].scale[12] = 0;
     235          10 :         RPFTOCTrim(toc->entries[i].scale);
     236          10 :         if (toc->entries[i].scale[0] == '1' && toc->entries[i].scale[1] == ':')
     237             :         {
     238          10 :             memmove(toc->entries[i].scale, toc->entries[i].scale + 2,
     239          10 :                     strlen(toc->entries[i].scale + 2) + 1);
     240             :         }
     241             : 
     242          10 :         bOK &= VSIFReadL(toc->entries[i].zone, 1, 1, fp) == 1;
     243          10 :         toc->entries[i].zone[1] = 0;
     244          10 :         RPFTOCTrim(toc->entries[i].zone);
     245             : 
     246          10 :         bOK &= VSIFReadL(toc->entries[i].producer, 1, 5, fp) == 5;
     247          10 :         toc->entries[i].producer[5] = 0;
     248          10 :         RPFTOCTrim(toc->entries[i].producer);
     249             : 
     250          10 :         bOK &= VSIFReadL(&toc->entries[i].nwLat, sizeof(double), 1, fp) == 1;
     251          10 :         CPL_MSBPTR64(&toc->entries[i].nwLat);
     252             : 
     253          10 :         bOK &= VSIFReadL(&toc->entries[i].nwLong, sizeof(double), 1, fp) == 1;
     254          10 :         CPL_MSBPTR64(&toc->entries[i].nwLong);
     255             : 
     256          10 :         bOK &= VSIFReadL(&toc->entries[i].swLat, sizeof(double), 1, fp) == 1;
     257          10 :         CPL_MSBPTR64(&toc->entries[i].swLat);
     258             : 
     259          10 :         bOK &= VSIFReadL(&toc->entries[i].swLong, sizeof(double), 1, fp) == 1;
     260          10 :         CPL_MSBPTR64(&toc->entries[i].swLong);
     261             : 
     262          10 :         bOK &= VSIFReadL(&toc->entries[i].neLat, sizeof(double), 1, fp) == 1;
     263          10 :         CPL_MSBPTR64(&toc->entries[i].neLat);
     264             : 
     265          10 :         bOK &= VSIFReadL(&toc->entries[i].neLong, sizeof(double), 1, fp) == 1;
     266          10 :         CPL_MSBPTR64(&toc->entries[i].neLong);
     267             : 
     268          10 :         bOK &= VSIFReadL(&toc->entries[i].seLat, sizeof(double), 1, fp) == 1;
     269          10 :         CPL_MSBPTR64(&toc->entries[i].seLat);
     270             : 
     271          10 :         bOK &= VSIFReadL(&toc->entries[i].seLong, sizeof(double), 1, fp) == 1;
     272          10 :         CPL_MSBPTR64(&toc->entries[i].seLong);
     273             : 
     274          10 :         bOK &= VSIFReadL(&toc->entries[i].vertResolution, sizeof(double), 1,
     275          10 :                          fp) == 1;
     276          10 :         CPL_MSBPTR64(&toc->entries[i].vertResolution);
     277             : 
     278          10 :         bOK &= VSIFReadL(&toc->entries[i].horizResolution, sizeof(double), 1,
     279          10 :                          fp) == 1;
     280          10 :         CPL_MSBPTR64(&toc->entries[i].horizResolution);
     281             : 
     282          10 :         bOK &= VSIFReadL(&toc->entries[i].vertInterval, sizeof(double), 1,
     283          10 :                          fp) == 1;
     284          10 :         CPL_MSBPTR64(&toc->entries[i].vertInterval);
     285             : 
     286          10 :         bOK &= VSIFReadL(&toc->entries[i].horizInterval, sizeof(double), 1,
     287          10 :                          fp) == 1;
     288          10 :         CPL_MSBPTR64(&toc->entries[i].horizInterval);
     289             : 
     290          10 :         bOK &= VSIFReadL(&toc->entries[i].nVertFrames, sizeof(int), 1, fp) == 1;
     291          10 :         CPL_MSBPTR32(&toc->entries[i].nVertFrames);
     292             : 
     293          10 :         bOK &=
     294          10 :             VSIFReadL(&toc->entries[i].nHorizFrames, sizeof(int), 1, fp) == 1;
     295          10 :         CPL_MSBPTR32(&toc->entries[i].nHorizFrames);
     296             : 
     297          10 :         if (!bOK)
     298             :         {
     299           0 :             CPLError(CE_Failure, CPLE_FileIO, "I/O error");
     300           0 :             toc->entries[i].nVertFrames = 0;
     301           0 :             toc->entries[i].nHorizFrames = 0;
     302           0 :             RPFTOCFree(toc);
     303           0 :             return nullptr;
     304             :         }
     305             : 
     306             :         // do some basic plausibility checks for all entries
     307          30 :         if (toc->entries[i].vertInterval <= 1e-10 ||
     308          10 :             !std::isfinite(toc->entries[i].vertInterval) ||
     309          10 :             toc->entries[i].horizInterval <= 1e-10 ||
     310          10 :             !std::isfinite(toc->entries[i].horizInterval) ||
     311          10 :             toc->entries[i].nHorizFrames == 0 ||
     312          30 :             toc->entries[i].nVertFrames == 0 ||
     313          10 :             toc->entries[i].nHorizFrames >
     314          10 :                 INT_MAX / toc->entries[i].nVertFrames)
     315             :         {
     316           0 :             CPLError(CE_Failure, CPLE_FileIO, "Invalid TOC entry");
     317           0 :             toc->entries[i].nVertFrames = 0;
     318           0 :             toc->entries[i].nHorizFrames = 0;
     319           0 :             RPFTOCFree(toc);
     320           0 :             return nullptr;
     321             :         }
     322             : 
     323             :         // Overview has ZONE 'R' and Legend ZONE 'D' but because the Zone 'D' is
     324             :         // also a valid Zone we need an additional check. -> In all cases of
     325             :         // Overview/Legend the values of the BoundingBox are equal so we simply
     326             :         // check here that NW == SE is.
     327          10 :         toc->entries[i].isOverviewOrLegend =
     328          20 :             (toc->entries[i].zone[0] == 'R' ||   // Overview
     329          10 :              (toc->entries[i].zone[0] == 'D' &&  // Legend
     330           0 :               memcmp(&(toc->entries[i].seLong), &(toc->entries[i].nwLong),
     331           0 :                      sizeof(toc->entries[i].nwLong)) == 0 &&
     332           0 :               memcmp(&(toc->entries[i].seLat), &(toc->entries[i].nwLat),
     333             :                      sizeof(toc->entries[i].nwLat)) == 0));
     334             : 
     335          20 :         bool isPolarZone = (toc->entries[i].zone[0] == '9') ||
     336          10 :                            (toc->entries[i].zone[0] == 'J');
     337             : 
     338             :         // make additional checks of the bounding for charts (without Legends
     339             :         // and Overviews)
     340          10 :         if (!toc->entries[i].isOverviewOrLegend)
     341             :         {
     342          10 :             if (!(fabs(toc->entries[i].seLong) <= 360.0) ||
     343          10 :                 !(fabs(toc->entries[i].nwLong) <= 360.0) ||
     344          10 :                 !(fabs(toc->entries[i].nwLat) <= 90.0) ||
     345          10 :                 !(fabs(toc->entries[i].seLat) <= 90.0) ||
     346             :                 // check only for non-polar zones, because the values are not
     347             :                 // always correct here
     348          10 :                 (!isPolarZone &&
     349          10 :                  (toc->entries[i].seLong < toc->entries[i].nwLong ||
     350          10 :                   toc->entries[i].nwLat < toc->entries[i].seLat)))
     351             :             {
     352           0 :                 CPLError(CE_Failure, CPLE_FileIO, "Invalid TOC entry");
     353           0 :                 toc->entries[i].nVertFrames = 0;
     354           0 :                 toc->entries[i].nHorizFrames = 0;
     355           0 :                 RPFTOCFree(toc);
     356           0 :                 return nullptr;
     357             :             }
     358             :         }
     359             : 
     360             :         // TODO: We could probably use another data structure, like a list,
     361             :         // instead of an array referenced by the frame coordinate...
     362          20 :         if (static_cast<int>(toc->entries[i].nHorizFrames *
     363          20 :                              toc->entries[i].nVertFrames) >
     364          10 :             atoi(CPLGetConfigOption("RPFTOC_MAX_FRAME_COUNT", "1000000")))
     365             :         {
     366           0 :             CPLError(
     367             :                 CE_Failure, CPLE_AppDefined,
     368             :                 "nHorizFrames=%d x nVertFrames=%d > %d. Please raise "
     369             :                 "the value of the RPFTOC_MAX_FRAME_COUNT configuration "
     370             :                 "option to more than %d if this dataset is legitimate.",
     371           0 :                 toc->entries[i].nHorizFrames, toc->entries[i].nVertFrames,
     372             :                 atoi(CPLGetConfigOption("RPFTOC_MAX_FRAME_COUNT", "1000000")),
     373           0 :                 toc->entries[i].nHorizFrames * toc->entries[i].nVertFrames);
     374           0 :             toc->entries[i].frameEntries = nullptr;
     375             :         }
     376             :         else
     377             :         {
     378          20 :             toc->entries[i].frameEntries =
     379          10 :                 reinterpret_cast<RPFTocFrameEntry *>(VSI_CALLOC_VERBOSE(
     380             :                     static_cast<size_t>(toc->entries[i].nVertFrames) *
     381             :                         toc->entries[i].nHorizFrames,
     382             :                     sizeof(RPFTocFrameEntry)));
     383             :         }
     384          10 :         if (toc->entries[i].frameEntries == nullptr)
     385             :         {
     386           0 :             toc->entries[i].nVertFrames = 0;
     387           0 :             toc->entries[i].nHorizFrames = 0;
     388           0 :             RPFTOCFree(toc);
     389           0 :             return nullptr;
     390             :         }
     391             : 
     392          10 :         CPLDebug("RPFTOC",
     393             :                  "[%d] type=%s, compression=%s, scale=%s, zone=%s, "
     394             :                  "producer=%s, nVertFrames=%d, nHorizFrames=%d",
     395          10 :                  i, toc->entries[i].type, toc->entries[i].compression,
     396          10 :                  toc->entries[i].scale, toc->entries[i].zone,
     397          10 :                  toc->entries[i].producer, toc->entries[i].nVertFrames,
     398          10 :                  toc->entries[i].nHorizFrames);
     399             :     }
     400             : 
     401          10 :     if (VSIFSeekL(fp, frameFileIndexSectionSubHeaderPhysIndex, SEEK_SET) != 0)
     402             :     {
     403           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     404             :                  "Invalid TOC file. Unable to seek to "
     405             :                  "frameFileIndexSectionSubHeaderPhysIndex at offset %d.",
     406             :                  frameFileIndexSectionSubHeaderPhysIndex);
     407           0 :         RPFTOCFree(toc);
     408           0 :         return nullptr;
     409             :     }
     410             : 
     411             :     /* Skip 1 byte security classification */
     412          10 :     bOK &= VSIFSeekL(fp, 1, SEEK_CUR) == 0;
     413             : 
     414             :     unsigned int frameIndexTableOffset;
     415          10 :     bOK &= VSIFReadL(&frameIndexTableOffset, sizeof(frameIndexTableOffset), 1,
     416          10 :                      fp) == 1;
     417          10 :     CPL_MSBPTR32(&frameIndexTableOffset);
     418             : 
     419             :     unsigned int nFrameFileIndexRecords;
     420          10 :     bOK &= VSIFReadL(&nFrameFileIndexRecords, sizeof(nFrameFileIndexRecords), 1,
     421          10 :                      fp) == 1;
     422          10 :     CPL_MSBPTR32(&nFrameFileIndexRecords);
     423             : 
     424             :     unsigned short nFrameFilePathnameRecords;
     425          10 :     bOK &= VSIFReadL(&nFrameFilePathnameRecords,
     426          10 :                      sizeof(nFrameFilePathnameRecords), 1, fp) == 1;
     427          10 :     CPL_MSBPTR16(&nFrameFilePathnameRecords);
     428             : 
     429             :     unsigned short frameFileIndexRecordLength;
     430          10 :     bOK &= VSIFReadL(&frameFileIndexRecordLength,
     431          10 :                      sizeof(frameFileIndexRecordLength), 1, fp) == 1;
     432          10 :     CPL_MSBPTR16(&frameFileIndexRecordLength);
     433          10 :     if (frameFileIndexRecordLength < 3 * sizeof(short))
     434             :     {
     435           0 :         CPLError(CE_Failure, CPLE_FileIO, "Invalid file");
     436           0 :         RPFTOCFree(toc);
     437           0 :         return nullptr;
     438             :     }
     439             : 
     440          10 :     if (!bOK)
     441             :     {
     442           0 :         CPLError(CE_Failure, CPLE_FileIO, "I/O error");
     443           0 :         RPFTOCFree(toc);
     444           0 :         return nullptr;
     445             :     }
     446             : 
     447          10 :     int newBoundaryId = 0;
     448             : 
     449          20 :     for (int i = 0; i < static_cast<int>(nFrameFileIndexRecords); i++)
     450             :     {
     451          10 :         vsi_l_offset nFrameOffset =
     452          10 :             static_cast<vsi_l_offset>(frameFileIndexSubsectionPhysIndex) +
     453          10 :             static_cast<vsi_l_offset>(frameFileIndexRecordLength) * i;
     454          10 :         if (VSIFSeekL(fp, nFrameOffset, SEEK_SET) != 0)
     455             :         {
     456           0 :             CPLError(
     457             :                 CE_Failure, CPLE_NotSupported,
     458             :                 "Invalid TOC file. Unable to seek to "
     459             :                 "frameFileIndexSubsectionPhysIndex(%d) at offset " CPL_FRMT_GUIB
     460             :                 ".",
     461             :                 i, static_cast<GUIntBig>(nFrameOffset));
     462           0 :             RPFTOCFree(toc);
     463           0 :             return nullptr;
     464             :         }
     465             : 
     466             :         unsigned short boundaryId;
     467          10 :         if (VSIFReadL(&boundaryId, sizeof(boundaryId), 1, fp) != 1)
     468             :         {
     469           0 :             CPLError(CE_Failure, CPLE_FileIO, "I/O error");
     470           0 :             RPFTOCFree(toc);
     471           0 :             return nullptr;
     472             :         }
     473          10 :         CPL_MSBPTR16(&boundaryId);
     474             : 
     475          10 :         if (i == 0 && boundaryId == 0)
     476          10 :             newBoundaryId = 1;
     477          10 :         if (newBoundaryId == 0)
     478           0 :             boundaryId--;
     479             : 
     480          10 :         if (boundaryId >= toc->nEntries)
     481             :         {
     482           0 :             CPLError(CE_Failure, CPLE_NotSupported,
     483             :                      "Invalid TOC file. Bad boundary id (%d) for frame file "
     484             :                      "index %d.",
     485             :                      boundaryId, i);
     486           0 :             RPFTOCFree(toc);
     487           0 :             return nullptr;
     488             :         }
     489             : 
     490          10 :         RPFTocEntry *entry = &toc->entries[boundaryId];
     491          10 :         entry->boundaryId = boundaryId;
     492             : 
     493             :         unsigned short frameRow;
     494          10 :         bOK &= VSIFReadL(&frameRow, sizeof(frameRow), 1, fp) == 1;
     495          10 :         CPL_MSBPTR16(&frameRow);
     496             : 
     497             :         unsigned short frameCol;
     498          10 :         bOK &= VSIFReadL(&frameCol, sizeof(frameCol), 1, fp) == 1;
     499          10 :         CPL_MSBPTR16(&frameCol);
     500          10 :         if (!bOK)
     501             :         {
     502           0 :             CPLError(CE_Failure, CPLE_FileIO, "I/O error");
     503           0 :             RPFTOCFree(toc);
     504           0 :             return nullptr;
     505             :         }
     506             : 
     507          10 :         if (newBoundaryId == 0)
     508             :         {
     509           0 :             frameRow--;
     510           0 :             frameCol--;
     511             :         }
     512             :         else
     513             :         {
     514             :             /* Trick so that frames are numbered north to south */
     515          10 :             if (entry->nVertFrames - 1 < frameRow)
     516             :             {
     517           0 :                 CPLError(CE_Failure, CPLE_FileIO,
     518             :                          "Invalid nVertFrames vs frameRow");
     519           0 :                 RPFTOCFree(toc);
     520           0 :                 return nullptr;
     521             :             }
     522          10 :             frameRow = (unsigned short)((entry->nVertFrames - 1) - frameRow);
     523             :         }
     524             : 
     525          10 :         if (frameRow >= entry->nVertFrames)
     526             :         {
     527           0 :             CPLError(
     528             :                 CE_Failure, CPLE_NotSupported,
     529             :                 "Invalid TOC file. Bad row num (%d) for frame file index %d.",
     530             :                 frameRow, i);
     531           0 :             RPFTOCFree(toc);
     532           0 :             return nullptr;
     533             :         }
     534             : 
     535          10 :         if (frameCol >= entry->nHorizFrames)
     536             :         {
     537           0 :             CPLError(
     538             :                 CE_Failure, CPLE_NotSupported,
     539             :                 "Invalid TOC file. Bad col num (%d) for frame file index %d.",
     540             :                 frameCol, i);
     541           0 :             RPFTOCFree(toc);
     542           0 :             return nullptr;
     543             :         }
     544             : 
     545          10 :         RPFTocFrameEntry *frameEntry =
     546          10 :             &entry->frameEntries[frameRow * entry->nHorizFrames + frameCol];
     547          10 :         frameEntry->frameRow = frameRow;
     548          10 :         frameEntry->frameCol = frameCol;
     549             : 
     550          10 :         if (frameEntry->exists)
     551             :         {
     552           0 :             CPLError(
     553             :                 CE_Warning, CPLE_AppDefined,
     554             :                 "Frame entry(%d,%d) for frame file index %d was already found.",
     555             :                 frameRow, frameCol, i);
     556           0 :             CPLFree(frameEntry->directory);
     557           0 :             frameEntry->directory = nullptr;
     558           0 :             CPLFree(frameEntry->fullFilePath);
     559           0 :             frameEntry->fullFilePath = nullptr;
     560           0 :             frameEntry->exists = 0;
     561             :         }
     562             : 
     563             :         unsigned int offsetFrameFilePathName;
     564          10 :         bOK &= VSIFReadL(&offsetFrameFilePathName,
     565          10 :                          sizeof(offsetFrameFilePathName), 1, fp) == 1;
     566          10 :         CPL_MSBPTR32(&offsetFrameFilePathName);
     567             : 
     568          10 :         bOK &= VSIFReadL(frameEntry->filename, 1, 12, fp) == 12;
     569          10 :         if (!bOK)
     570             :         {
     571           0 :             CPLError(CE_Failure, CPLE_FileIO, "I/O error");
     572           0 :             RPFTOCFree(toc);
     573           0 :             return nullptr;
     574             :         }
     575          10 :         frameEntry->filename[12] = '\0';
     576          10 :         bOK &= strlen(frameEntry->filename) > 0;
     577             : 
     578             :         // Check (case insensitive) if the filename is an overview or legend
     579             :         // some CADRG maps have legend name smaller than 8.3 then the extension
     580             :         // has blanks (0x20) at the end -> check only the first 3 letters of the
     581             :         // extension.
     582          10 :         const std::string fileExt = CPLGetExtensionSafe(frameEntry->filename);
     583          20 :         if (EQUALN(fileExt.c_str(), "ovr", 3) ||
     584          10 :             EQUALN(fileExt.c_str(), "lgd", 3))
     585             :         {
     586           0 :             entry->isOverviewOrLegend = TRUE;
     587             :         }
     588             : 
     589             :         /* Extract series code */
     590          10 :         if (entry->seriesAbbreviation == nullptr)
     591             :         {
     592          10 :             const NITFSeries *series = NITFGetSeriesInfo(frameEntry->filename);
     593          10 :             if (series)
     594             :             {
     595          10 :                 entry->seriesAbbreviation = series->abbreviation;
     596          10 :                 entry->seriesName = series->name;
     597             :             }
     598             :         }
     599             : 
     600             :         /* Get file geo reference */
     601          10 :         bOK &= VSIFReadL(frameEntry->georef, 1, 6, fp) == 6;
     602          10 :         frameEntry->georef[6] = '\0';
     603             : 
     604             :         /* Go to start of pathname record */
     605             :         /* New path_off offset from start of frame file index section of TOC??
     606             :          */
     607             :         /* Add pathoffset wrt frame file index table subsection (loc[3]) */
     608          20 :         if (!bOK || VSIFSeekL(fp,
     609          10 :                               static_cast<vsi_l_offset>(
     610             :                                   frameFileIndexSubsectionPhysIndex) +
     611          10 :                                   offsetFrameFilePathName,
     612             :                               SEEK_SET) != 0)
     613             :         {
     614           0 :             CPLError(CE_Failure, CPLE_NotSupported,
     615             :                      "Invalid TOC file. Unable to seek to "
     616             :                      "frameFileIndexSubsectionPhysIndex + "
     617             :                      "offsetFrameFilePathName(%d) at offset " CPL_FRMT_GUIB ".",
     618             :                      i,
     619           0 :                      static_cast<GUIntBig>(frameFileIndexSubsectionPhysIndex) +
     620           0 :                          offsetFrameFilePathName);
     621           0 :             RPFTOCFree(toc);
     622           0 :             return nullptr;
     623             :         }
     624             : 
     625             :         unsigned short pathLength;
     626          10 :         bOK &= VSIFReadL(&pathLength, sizeof(pathLength), 1, fp) == 1;
     627          10 :         CPL_MSBPTR16(&pathLength);
     628             : 
     629             :         /* if nFrameFileIndexRecords == 65535 and pathLength == 65535 for each
     630             :            record, this leads to 4 GB allocation... Protect against this case */
     631          10 :         if (!bOK || pathLength == 0 || pathLength > 256)
     632             :         {
     633           0 :             CPLError(
     634             :                 CE_Failure, CPLE_NotSupported,
     635             :                 "Path length is invalid : %d. Probably corrupted TOC file.",
     636             :                 static_cast<int>(pathLength));
     637           0 :             RPFTOCFree(toc);
     638           0 :             return nullptr;
     639             :         }
     640             : 
     641          10 :         frameEntry->directory =
     642          10 :             reinterpret_cast<char *>(CPLMalloc(pathLength + 1));
     643          10 :         bOK &=
     644          10 :             VSIFReadL(frameEntry->directory, 1, pathLength, fp) == pathLength;
     645          10 :         if (!bOK)
     646             :         {
     647           0 :             CPLError(CE_Failure, CPLE_FileIO, "I/O error");
     648           0 :             RPFTOCFree(toc);
     649           0 :             return nullptr;
     650             :         }
     651          10 :         frameEntry->directory[pathLength] = 0;
     652          10 :         if (frameEntry->directory[pathLength - 1] == '/')
     653          10 :             frameEntry->directory[pathLength - 1] = 0;
     654             : 
     655          10 :         if (frameEntry->directory[0] == '.' && frameEntry->directory[1] == '/')
     656             :         {
     657           0 :             memmove(frameEntry->directory, frameEntry->directory + 2,
     658           0 :                     strlen(frameEntry->directory + 2) + 1);
     659             : 
     660             :             // Some A.TOC have subdirectory names like ".//X/" ... (#5979)
     661             :             // Check if it was not intended to be "./X/" instead.
     662             :             VSIStatBufL sStatBuf;
     663           0 :             if (frameEntry->directory[0] == '/' &&
     664           0 :                 VSIStatL(
     665           0 :                     CPLFormFilenameSafe(CPLGetDirnameSafe(pszFilename).c_str(),
     666           0 :                                         frameEntry->directory + 1, nullptr)
     667             :                         .c_str(),
     668           0 :                     &sStatBuf) == 0 &&
     669           0 :                 VSI_ISDIR(sStatBuf.st_mode))
     670             :             {
     671           0 :                 memmove(frameEntry->directory, frameEntry->directory + 1,
     672           0 :                         strlen(frameEntry->directory + 1) + 1);
     673             :             }
     674             :         }
     675             : 
     676             :         {
     677          10 :             char *baseDir = CPLStrdup(CPLGetDirnameSafe(pszFilename).c_str());
     678             :             VSIStatBufL sStatBuf;
     679          10 :             char *subdir = nullptr;
     680          10 :             if (CPLIsFilenameRelative(frameEntry->directory) == FALSE)
     681           0 :                 subdir = CPLStrdup(frameEntry->directory);
     682          10 :             else if (frameEntry->directory[0] == '.' &&
     683          10 :                      frameEntry->directory[1] == 0)
     684          10 :                 subdir = CPLStrdup(baseDir);
     685             :             else
     686           0 :                 subdir = CPLStrdup(
     687           0 :                     CPLFormFilenameSafe(baseDir, frameEntry->directory, nullptr)
     688             :                         .c_str());
     689             : #if !defined(_WIN32) && !defined(_WIN32_CE)
     690          10 :             if (VSIStatL(subdir, &sStatBuf) != 0 &&
     691           0 :                 strlen(subdir) > strlen(baseDir))
     692             :             {
     693           0 :                 char *c = subdir + strlen(baseDir) + 1;
     694           0 :                 while (*c)
     695             :                 {
     696           0 :                     if (*c >= 'A' && *c <= 'Z')
     697           0 :                         *c += 'a' - 'A';
     698           0 :                     c++;
     699             :                 }
     700             :             }
     701             : #endif
     702          10 :             frameEntry->fullFilePath = CPLStrdup(
     703          20 :                 CPLFormFilenameSafe(subdir, frameEntry->filename, nullptr)
     704             :                     .c_str());
     705          10 :             if (VSIStatL(frameEntry->fullFilePath, &sStatBuf) != 0)
     706             :             {
     707             : #if !defined(_WIN32) && !defined(_WIN32_CE)
     708           0 :                 if (strlen(frameEntry->fullFilePath) > strlen(subdir))
     709             :                 {
     710           0 :                     char *c = frameEntry->fullFilePath + strlen(subdir) + 1;
     711           0 :                     while (*c)
     712             :                     {
     713           0 :                         if (*c >= 'A' && *c <= 'Z')
     714           0 :                             *c += 'a' - 'A';
     715           0 :                         c++;
     716             :                     }
     717             :                 }
     718           0 :                 if (VSIStatL(frameEntry->fullFilePath, &sStatBuf) != 0)
     719             : #endif
     720             :                 {
     721           0 :                     frameEntry->fileExists = 0;
     722           0 :                     CPLError(CE_Warning, CPLE_AppDefined,
     723             :                              "File %s does not exist.",
     724             :                              frameEntry->fullFilePath);
     725             :                 }
     726             : #if !defined(_WIN32) && !defined(_WIN32_CE)
     727             :                 else
     728             :                 {
     729           0 :                     frameEntry->fileExists = 1;
     730             :                 }
     731             : #endif
     732             :             }
     733             :             else
     734             :             {
     735          10 :                 frameEntry->fileExists = 1;
     736             :             }
     737          10 :             CPLFree(subdir);
     738          10 :             CPLFree(baseDir);
     739             :         }
     740             : 
     741          10 :         CPLDebug("RPFTOC", "Entry %d : %s,%s (%d, %d)", boundaryId,
     742          10 :                  frameEntry->directory, frameEntry->filename, frameRow,
     743             :                  frameCol);
     744             : 
     745          10 :         frameEntry->exists = 1;
     746             :     }
     747             : 
     748          10 :     return toc;
     749             : }
     750             : 
     751             : /************************************************************************/
     752             : /*                        RPFTOCFree()                                 */
     753             : /************************************************************************/
     754             : 
     755          10 : void RPFTOCFree(RPFToc *toc)
     756             : {
     757          10 :     if (!toc)
     758           0 :         return;
     759             : 
     760          20 :     for (int i = 0; i < toc->nEntries; i++)
     761             :     {
     762          20 :         for (int j = 0; j < static_cast<int>(toc->entries[i].nVertFrames *
     763          20 :                                              toc->entries[i].nHorizFrames);
     764             :              j++)
     765             :         {
     766          10 :             CPLFree(toc->entries[i].frameEntries[j].fullFilePath);
     767          10 :             CPLFree(toc->entries[i].frameEntries[j].directory);
     768             :         }
     769          10 :         CPLFree(toc->entries[i].frameEntries);
     770             :     }
     771             : 
     772          10 :     CPLFree(toc->entries);
     773          10 :     CPLFree(toc);
     774             : }

Generated by: LCOV version 1.14