LCOV - code coverage report
Current view: top level - frmts/hfa - hfatype.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 154 188 81.9 %
Date: 2025-01-18 12:42:00 Functions: 8 10 80.0 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  Erdas Imagine (.img) Translator
       4             :  * Purpose:  Implementation of the HFAType class, for managing one type
       5             :  *           defined in the HFA data dictionary.  Managed by HFADictionary.
       6             :  * Author:   Frank Warmerdam, warmerdam@pobox.com
       7             :  *
       8             :  ******************************************************************************
       9             :  * Copyright (c) 1999, Intergraph Corporation
      10             :  * Copyright (c) 2009-2011, Even Rouault <even dot rouault at spatialys.com>
      11             :  *
      12             :  * SPDX-License-Identifier: MIT
      13             :  ****************************************************************************/
      14             : 
      15             : #include "cpl_port.h"
      16             : #include "hfa_p.h"
      17             : 
      18             : #include <climits>
      19             : #include <cstdio>
      20             : #include <cstdlib>
      21             : #include <cstring>
      22             : 
      23             : #include "cpl_conv.h"
      24             : #include "cpl_error.h"
      25             : #include "cpl_vsi.h"
      26             : 
      27             : /************************************************************************/
      28             : /* ==================================================================== */
      29             : /*                             HFAType                                  */
      30             : /* ==================================================================== */
      31             : /************************************************************************/
      32             : 
      33             : /************************************************************************/
      34             : /*                              HFAType()                               */
      35             : /************************************************************************/
      36             : 
      37       27431 : HFAType::HFAType() : bInCompleteDefn(false), nBytes(0), pszTypeName(nullptr)
      38             : {
      39       27431 : }
      40             : 
      41             : /************************************************************************/
      42             : /*                              ~HFAType()                              */
      43             : /************************************************************************/
      44             : 
      45       27431 : HFAType::~HFAType()
      46             : 
      47             : {
      48       27431 :     CPLFree(pszTypeName);
      49       27431 : }
      50             : 
      51             : /************************************************************************/
      52             : /*                             Initialize()                             */
      53             : /************************************************************************/
      54             : 
      55       27431 : const char *HFAType::Initialize(const char *pszInput)
      56             : 
      57             : {
      58       27431 :     if (*pszInput != '{')
      59             :     {
      60           1 :         if (*pszInput != '\0')
      61           1 :             CPLDebug("HFAType", "Initialize(%60.60s) - unexpected input.",
      62             :                      pszInput);
      63             : 
      64          26 :         while (*pszInput != '{' && *pszInput != '\0')
      65          25 :             pszInput++;
      66             : 
      67           1 :         if (*pszInput == '\0')
      68           0 :             return nullptr;
      69             :     }
      70             : 
      71       27431 :     pszInput++;
      72             : 
      73             :     // Read the field definitions.
      74      131076 :     while (pszInput != nullptr && *pszInput != '}')
      75             :     {
      76      207290 :         auto poNewField = std::make_unique<HFAField>();
      77             : 
      78      103645 :         pszInput = poNewField->Initialize(pszInput);
      79      103645 :         if (pszInput != nullptr)
      80             :         {
      81      103644 :             apoFields.emplace_back(std::move(poNewField));
      82             :         }
      83             :     }
      84             : 
      85       27431 :     if (pszInput == nullptr)
      86           1 :         return nullptr;
      87             : 
      88             :     // Collect the name.
      89       27430 :     pszInput++;  // Skip `}'
      90       27430 :     int i = 0;   // Used after for.
      91      451096 :     for (; pszInput[i] != '\0' && pszInput[i] != ','; i++)
      92             :     {
      93             :     }
      94       27430 :     if (pszInput[i] == '\0')
      95             :     {
      96           6 :         pszTypeName = CPLStrdup(pszInput);
      97           6 :         return nullptr;
      98             :     }
      99             : 
     100       27424 :     pszTypeName = static_cast<char *>(CPLMalloc(i + 1));
     101       27424 :     strncpy(pszTypeName, pszInput, i);
     102       27424 :     pszTypeName[i] = '\0';
     103             : 
     104       27424 :     pszInput += i + 1;
     105             : 
     106       27424 :     return pszInput;
     107             : }
     108             : 
     109             : /************************************************************************/
     110             : /*                            CompleteDefn()                            */
     111             : /************************************************************************/
     112             : 
     113       38993 : bool HFAType::CompleteDefn(HFADictionary *poDict)
     114             : 
     115             : {
     116             :     // This may already be done, if an earlier object required this
     117             :     // object (as a field), and forced an early computation of the size.
     118       38993 :     if (nBytes != 0)
     119       11561 :         return true;
     120             : 
     121       27432 :     if (bInCompleteDefn)
     122             :     {
     123           2 :         CPLError(CE_Failure, CPLE_AppDefined,
     124             :                  "Recursion detected in HFAType::CompleteDefn()");
     125           2 :         return false;
     126             :     }
     127       27430 :     bInCompleteDefn = true;
     128             : 
     129             :     // Complete each of the fields, totaling up the sizes.  This
     130             :     // isn't really accurate for object with variable sized subobjects.
     131       27430 :     bool bRet = true;
     132      131066 :     for (auto &poField : apoFields)
     133             :     {
     134      103638 :         if (!poField->CompleteDefn(poDict))
     135             :         {
     136           2 :             bRet = false;
     137           2 :             break;
     138             :         }
     139      103636 :         if (poField->nBytes < 0 || nBytes == -1)
     140       44660 :             nBytes = -1;
     141       58976 :         else if (nBytes < INT_MAX - poField->nBytes)
     142       58976 :             nBytes += poField->nBytes;
     143             :         else
     144           0 :             nBytes = -1;
     145             :     }
     146             : 
     147       27430 :     bInCompleteDefn = false;
     148       27430 :     return bRet;
     149             : }
     150             : 
     151             : /************************************************************************/
     152             : /*                                Dump()                                */
     153             : /************************************************************************/
     154             : 
     155           0 : void HFAType::Dump(FILE *fp)
     156             : 
     157             : {
     158           0 :     CPL_IGNORE_RET_VAL(
     159           0 :         VSIFPrintf(fp, "HFAType %s/%d bytes\n", pszTypeName, nBytes));
     160             : 
     161           0 :     for (auto &poField : apoFields)
     162             :     {
     163           0 :         poField->Dump(fp);
     164             :     }
     165             : 
     166           0 :     CPL_IGNORE_RET_VAL(VSIFPrintf(fp, "\n"));
     167           0 : }
     168             : 
     169             : /************************************************************************/
     170             : /*                            SetInstValue()                            */
     171             : /************************************************************************/
     172             : 
     173       14312 : CPLErr HFAType::SetInstValue(const char *pszFieldPath, GByte *pabyData,
     174             :                              GUInt32 nDataOffset, int nDataSize, char chReqType,
     175             :                              void *pValue)
     176             : 
     177             : {
     178       14312 :     int nArrayIndex = 0;
     179       14312 :     int nNameLen = 0;
     180       14312 :     const char *pszRemainder = nullptr;
     181             : 
     182             :     // Parse end of field name, possible index value and
     183             :     // establish where the remaining fields (if any) would start.
     184       14312 :     if (strchr(pszFieldPath, '[') != nullptr)
     185             :     {
     186        3809 :         const char *pszEnd = strchr(pszFieldPath, '[');
     187             : 
     188        3809 :         nArrayIndex = atoi(pszEnd + 1);
     189        3809 :         nNameLen = static_cast<int>(pszEnd - pszFieldPath);
     190             : 
     191        3809 :         pszRemainder = strchr(pszFieldPath, '.');
     192        3809 :         if (pszRemainder != nullptr)
     193         151 :             pszRemainder++;
     194             :     }
     195       10503 :     else if (strchr(pszFieldPath, '.') != nullptr)
     196             :     {
     197        2740 :         const char *pszEnd = strchr(pszFieldPath, '.');
     198             : 
     199        2740 :         nNameLen = static_cast<int>(pszEnd - pszFieldPath);
     200             : 
     201        2740 :         pszRemainder = pszEnd + 1;
     202             :     }
     203             :     else
     204             :     {
     205        7763 :         nNameLen = static_cast<int>(strlen(pszFieldPath));
     206        7763 :         pszRemainder = pszFieldPath /* NULL */;
     207             :     }
     208             : 
     209             :     // Find this field within this type, if possible.
     210       14312 :     int nByteOffset = 0;
     211       14312 :     size_t iField = 0;
     212       14312 :     const size_t nFields = apoFields.size();
     213       48552 :     for (; iField < nFields && nByteOffset < nDataSize; iField++)
     214             :     {
     215       62828 :         if (EQUALN(pszFieldPath, apoFields[iField]->pszFieldName, nNameLen) &&
     216       14312 :             apoFields[iField]->pszFieldName[nNameLen] == '\0')
     217             :         {
     218       14276 :             break;
     219             :         }
     220             : 
     221       34240 :         std::set<HFAField *> oVisitedFields;
     222       34240 :         const int nInc = apoFields[iField]->GetInstBytes(
     223       34240 :             pabyData + nByteOffset, nDataSize - nByteOffset, oVisitedFields);
     224             : 
     225       34240 :         if (nInc <= 0 || nByteOffset > INT_MAX - nInc)
     226             :         {
     227           0 :             CPLError(CE_Failure, CPLE_AppDefined, "Invalid return value");
     228           0 :             return CE_Failure;
     229             :         }
     230             : 
     231       34240 :         nByteOffset += nInc;
     232             :     }
     233             : 
     234       14312 :     if (iField == nFields || nByteOffset >= nDataSize)
     235          36 :         return CE_Failure;
     236             : 
     237             :     // Extract this field value, and return.
     238       14276 :     return apoFields[iField]->SetInstValue(
     239       14276 :         pszRemainder, nArrayIndex, pabyData + nByteOffset,
     240       14276 :         nDataOffset + nByteOffset, nDataSize - nByteOffset, chReqType, pValue);
     241             : }
     242             : 
     243             : /************************************************************************/
     244             : /*                            GetInstCount()                            */
     245             : /************************************************************************/
     246             : 
     247         934 : int HFAType::GetInstCount(const char *pszFieldPath, GByte *pabyData,
     248             :                           GUInt32 /* nDataOffset */, int nDataSize)
     249             : {
     250             :     // int nArrayIndex = 0;
     251         934 :     int nNameLen = 0;
     252             :     // const char *pszRemainder;
     253             : 
     254             :     // Parse end of field name, possible index value and
     255             :     // establish where the remaining fields (if any) would start.
     256         934 :     if (strchr(pszFieldPath, '[') != nullptr)
     257             :     {
     258           0 :         const char *pszEnd = strchr(pszFieldPath, '[');
     259             : 
     260             :         // nArrayIndex = atoi(pszEnd+1);
     261           0 :         nNameLen = static_cast<int>(pszEnd - pszFieldPath);
     262             : 
     263             :         // pszRemainder = strchr(pszFieldPath, '.');
     264             :         // if( pszRemainder != NULL )
     265             :         //    pszRemainder++;
     266             :     }
     267         934 :     else if (strchr(pszFieldPath, '.') != nullptr)
     268             :     {
     269         219 :         const char *pszEnd = strchr(pszFieldPath, '.');
     270             : 
     271         219 :         nNameLen = static_cast<int>(pszEnd - pszFieldPath);
     272             : 
     273             :         // pszRemainder = pszEnd + 1;
     274             :     }
     275             :     else
     276             :     {
     277         715 :         nNameLen = static_cast<int>(strlen(pszFieldPath));
     278             :         // pszRemainder = NULL;
     279             :     }
     280             : 
     281             :     // Find this field within this type, if possible.
     282         934 :     int nByteOffset = 0;
     283         934 :     size_t iField = 0;
     284         934 :     const size_t nFields = apoFields.size();
     285        3836 :     for (; iField < nFields && nByteOffset < nDataSize; iField++)
     286             :     {
     287        4770 :         if (EQUALN(pszFieldPath, apoFields[iField]->pszFieldName, nNameLen) &&
     288         934 :             apoFields[iField]->pszFieldName[nNameLen] == '\0')
     289             :         {
     290         934 :             break;
     291             :         }
     292             : 
     293        2902 :         std::set<HFAField *> oVisitedFields;
     294        2902 :         const int nInc = apoFields[iField]->GetInstBytes(
     295        2902 :             pabyData + nByteOffset, nDataSize - nByteOffset, oVisitedFields);
     296             : 
     297        2902 :         if (nInc <= 0 || nByteOffset > INT_MAX - nInc)
     298             :         {
     299           0 :             CPLError(CE_Failure, CPLE_AppDefined, "Invalid return value");
     300           0 :             return -1;
     301             :         }
     302             : 
     303        2902 :         nByteOffset += nInc;
     304             :     }
     305             : 
     306         934 :     if (iField == nFields || nByteOffset >= nDataSize)
     307           0 :         return -1;
     308             : 
     309             :     // Extract this field value, and return.
     310         934 :     return apoFields[iField]->GetInstCount(pabyData + nByteOffset,
     311         934 :                                            nDataSize - nByteOffset);
     312             : }
     313             : 
     314             : /************************************************************************/
     315             : /*                          ExtractInstValue()                          */
     316             : /*                                                                      */
     317             : /*      Extract the value of a field instance within this type.         */
     318             : /*      Most of the work is done by the ExtractInstValue() for the      */
     319             : /*      HFAField, but this method does the field name parsing.          */
     320             : /*                                                                      */
     321             : /*      field names have the form:                                      */
     322             : /*                                                                      */
     323             : /*        fieldname{[index]}{.fieldname...}                             */
     324             : /*                                                                      */
     325             : /*      eg.                                                             */
     326             : /*        abc                                   - field abc[0]          */
     327             : /*        abc[3]                                - field abc[3]          */
     328             : /*        abc[2].def                            - field def[0] of       */
     329             : /*                                                the third abc struct. */
     330             : /************************************************************************/
     331             : 
     332       46224 : bool HFAType::ExtractInstValue(const char *pszFieldPath, GByte *pabyData,
     333             :                                GUInt32 nDataOffset, int nDataSize,
     334             :                                char chReqType, void *pReqReturn,
     335             :                                int *pnRemainingDataSize)
     336             : 
     337             : {
     338       46224 :     int nArrayIndex = 0;
     339       46224 :     int nNameLen = 0;
     340       46224 :     const char *pszRemainder = nullptr;
     341             : 
     342             :     // Parse end of field name, possible index value and
     343             :     // establish where the remaining fields (if any) would start.
     344       46224 :     const char *pszFirstArray = strchr(pszFieldPath, '[');
     345       46224 :     const char *pszFirstDot = strchr(pszFieldPath, '.');
     346             : 
     347       46224 :     if (pszFirstArray != nullptr &&
     348       10217 :         (pszFirstDot == nullptr || pszFirstDot > pszFirstArray))
     349             :     {
     350       15470 :         const char *pszEnd = pszFirstArray;
     351             : 
     352       15470 :         nArrayIndex = atoi(pszEnd + 1);
     353       15470 :         nNameLen = static_cast<int>(pszEnd - pszFieldPath);
     354             : 
     355       15470 :         pszRemainder = strchr(pszFieldPath, '.');
     356       15470 :         if (pszRemainder != nullptr)
     357       15470 :             pszRemainder++;
     358             :     }
     359       30754 :     else if (pszFirstDot != nullptr)
     360             :     {
     361        4054 :         const char *pszEnd = pszFirstDot;
     362             : 
     363        4054 :         nNameLen = static_cast<int>(pszEnd - pszFieldPath);
     364             : 
     365        4054 :         pszRemainder = pszEnd + 1;
     366             :     }
     367             :     else
     368             :     {
     369       26700 :         nNameLen = static_cast<int>(strlen(pszFieldPath));
     370       26700 :         pszRemainder = nullptr;
     371             :     }
     372             : 
     373             :     // Find this field within this type, if possible.
     374       46224 :     int nByteOffset = 0;
     375       46224 :     size_t iField = 0;
     376       46224 :     const size_t nFields = apoFields.size();
     377      172240 :     for (; iField < nFields && nByteOffset < nDataSize; iField++)
     378             :     {
     379      218384 :         if (EQUALN(pszFieldPath, apoFields[iField]->pszFieldName, nNameLen) &&
     380       46184 :             apoFields[iField]->pszFieldName[nNameLen] == '\0')
     381             :         {
     382       46184 :             break;
     383             :         }
     384             : 
     385      126016 :         std::set<HFAField *> oVisitedFields;
     386      126016 :         const int nInc = apoFields[iField]->GetInstBytes(
     387      126016 :             pabyData + nByteOffset, nDataSize - nByteOffset, oVisitedFields);
     388             : 
     389      126016 :         if (nInc <= 0 || nByteOffset > INT_MAX - nInc)
     390             :         {
     391           0 :             CPLError(CE_Failure, CPLE_AppDefined, "Invalid return value");
     392           0 :             return false;
     393             :         }
     394             : 
     395      126016 :         nByteOffset += nInc;
     396             :     }
     397             : 
     398       46224 :     if (iField == nFields || nByteOffset >= nDataSize)
     399          40 :         return false;
     400             : 
     401             :     // Extract this field value, and return.
     402       46184 :     return apoFields[iField]->ExtractInstValue(
     403       46184 :         pszRemainder, nArrayIndex, pabyData + nByteOffset,
     404       46184 :         nDataOffset + nByteOffset, nDataSize - nByteOffset, chReqType,
     405       46184 :         pReqReturn, pnRemainingDataSize);
     406             : }
     407             : 
     408             : /************************************************************************/
     409             : /*                           DumpInstValue()                            */
     410             : /************************************************************************/
     411             : 
     412           0 : void HFAType::DumpInstValue(FILE *fpOut, GByte *pabyData, GUInt32 nDataOffset,
     413             :                             int nDataSize, const char *pszPrefix) const
     414             : 
     415             : {
     416           0 :     const size_t nFields = apoFields.size();
     417           0 :     for (size_t iField = 0; iField < nFields && nDataSize > 0; iField++)
     418             :     {
     419           0 :         auto &poField = apoFields[iField];
     420             : 
     421           0 :         poField->DumpInstValue(fpOut, pabyData, nDataOffset, nDataSize,
     422             :                                pszPrefix);
     423             : 
     424           0 :         std::set<HFAField *> oVisitedFields;
     425             :         const int nInstBytes =
     426           0 :             poField->GetInstBytes(pabyData, nDataSize, oVisitedFields);
     427           0 :         if (nInstBytes <= 0 || nDataOffset > UINT_MAX - nInstBytes)
     428             :         {
     429           0 :             CPLError(CE_Failure, CPLE_AppDefined, "Invalid return value");
     430           0 :             return;
     431             :         }
     432             : 
     433           0 :         pabyData += nInstBytes;
     434           0 :         nDataOffset += nInstBytes;
     435           0 :         nDataSize -= nInstBytes;
     436             :     }
     437             : }
     438             : 
     439             : /************************************************************************/
     440             : /*                            GetInstBytes()                            */
     441             : /*                                                                      */
     442             : /*      How many bytes in this particular instance of this type?        */
     443             : /************************************************************************/
     444             : 
     445        2493 : int HFAType::GetInstBytes(GByte *pabyData, int nDataSize,
     446             :                           std::set<HFAField *> &oVisitedFields) const
     447             : 
     448             : {
     449        2493 :     if (nBytes >= 0)
     450           0 :         return nBytes;
     451             : 
     452        2493 :     int nTotal = 0;
     453        2493 :     const size_t nFields = apoFields.size();
     454        5598 :     for (size_t iField = 0; iField < nFields && nTotal < nDataSize; iField++)
     455             :     {
     456        3105 :         auto &poField = apoFields[iField];
     457             : 
     458             :         const int nInstBytes =
     459        3105 :             poField->GetInstBytes(pabyData, nDataSize - nTotal, oVisitedFields);
     460        3105 :         if (nInstBytes <= 0 || nTotal > INT_MAX - nInstBytes)
     461             :         {
     462           0 :             CPLError(CE_Failure, CPLE_AppDefined, "Invalid return value");
     463           0 :             return -1;
     464             :         }
     465             : 
     466        3105 :         pabyData += nInstBytes;
     467        3105 :         nTotal += nInstBytes;
     468             :     }
     469             : 
     470        2493 :     return nTotal;
     471             : }

Generated by: LCOV version 1.14