LCOV - code coverage report
Current view: top level - ogr/ogrsf_frmts/dxf - ogrdxfdatasource.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 406 495 82.0 %
Date: 2025-10-24 23:03:13 Functions: 22 22 100.0 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  DXF Translator
       4             :  * Purpose:  Implements OGRDXFDataSource class
       5             :  * Author:   Frank Warmerdam, warmerdam@pobox.com
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2009, Frank Warmerdam <warmerdam@pobox.com>
       9             :  * Copyright (c) 2010-2013, Even Rouault <even dot rouault at spatialys.com>
      10             :  *
      11             :  * SPDX-License-Identifier: MIT
      12             :  ****************************************************************************/
      13             : 
      14             : #include "ogr_dxf.h"
      15             : #include "cpl_conv.h"
      16             : #include "cpl_string.h"
      17             : 
      18             : #include <algorithm>
      19             : 
      20             : /************************************************************************/
      21             : /*                          OGRDXFDataSource()                          */
      22             : /************************************************************************/
      23             : 
      24         219 : OGRDXFDataSource::OGRDXFDataSource()
      25             :     : fp(nullptr), iEntitiesOffset(0), iEntitiesLineNumber(0),
      26             :       bInlineBlocks(false), bMergeBlockGeometries(false),
      27             :       bTranslateEscapeSequences(false), bIncludeRawCodeValues(false),
      28             :       b3DExtensibleMode(false), bHaveReadSolidData(false),
      29         219 :       poReader(std::make_unique<OGRDXFReaderASCII>())
      30             : {
      31         219 : }
      32             : 
      33             : /************************************************************************/
      34             : /*                         ~OGRDXFDataSource()                          */
      35             : /************************************************************************/
      36             : 
      37         590 : OGRDXFDataSource::~OGRDXFDataSource()
      38             : 
      39             : {
      40             :     /* -------------------------------------------------------------------- */
      41             :     /*      Destroy layers.                                                 */
      42             :     /* -------------------------------------------------------------------- */
      43         443 :     while (!apoLayers.empty())
      44             :     {
      45         224 :         delete apoLayers.back();
      46         224 :         apoLayers.pop_back();
      47             :     }
      48             : 
      49             :     /* -------------------------------------------------------------------- */
      50             :     /*      Close file.                                                     */
      51             :     /* -------------------------------------------------------------------- */
      52         219 :     if (fp != nullptr)
      53             :     {
      54         219 :         VSIFCloseL(fp);
      55         219 :         fp = nullptr;
      56             :     }
      57         371 : }
      58             : 
      59             : /************************************************************************/
      60             : /*                           TestCapability()                           */
      61             : /************************************************************************/
      62             : 
      63           1 : int OGRDXFDataSource::TestCapability(const char *pszCap) const
      64             : {
      65           1 :     if (EQUAL(pszCap, ODsCZGeometries))
      66           0 :         return true;
      67             : 
      68           1 :     return false;
      69             : }
      70             : 
      71             : /************************************************************************/
      72             : /*                              GetLayer()                              */
      73             : /************************************************************************/
      74             : 
      75         501 : const OGRLayer *OGRDXFDataSource::GetLayer(int iLayer) const
      76             : 
      77             : {
      78         501 :     if (iLayer < 0 || iLayer >= (int)apoLayers.size())
      79           0 :         return nullptr;
      80             :     else
      81         501 :         return apoLayers[iLayer];
      82             : }
      83             : 
      84             : /************************************************************************/
      85             : /*                                Open()                                */
      86             : /************************************************************************/
      87             : 
      88         219 : bool OGRDXFDataSource::Open(const char *pszFilename, VSILFILE *fpIn,
      89             :                             bool bHeaderOnly, CSLConstList papszOptionsIn)
      90             : 
      91             : {
      92         219 :     SetDescription(pszFilename);
      93             : 
      94         219 :     osEncoding = CPL_ENC_ISO8859_1;
      95             : 
      96         219 :     bInlineBlocks = CPLTestBool(
      97             :         CSLFetchNameValueDef(papszOptionsIn, "INLINE_BLOCKS",
      98             :                              CPLGetConfigOption("DXF_INLINE_BLOCKS", "TRUE")));
      99         219 :     bMergeBlockGeometries = CPLTestBool(CSLFetchNameValueDef(
     100             :         papszOptionsIn, "MERGE_BLOCK_GEOMETRIES",
     101             :         CPLGetConfigOption("DXF_MERGE_BLOCK_GEOMETRIES", "TRUE")));
     102             : 
     103         219 :     bTranslateEscapeSequences = CPLTestBool(CSLFetchNameValueDef(
     104             :         papszOptionsIn, "TRANSLATE_ESCAPE_SEQUENCES",
     105             :         CPLGetConfigOption("DXF_TRANSLATE_ESCAPE_SEQUENCES", "TRUE")));
     106             : 
     107         219 :     bIncludeRawCodeValues = CPLTestBool(CSLFetchNameValueDef(
     108             :         papszOptionsIn, "INCLUDE_RAW_CODE_VALUES",
     109             :         CPLGetConfigOption("DXF_INCLUDE_RAW_CODE_VALUES", "FALSE")));
     110             : 
     111         219 :     b3DExtensibleMode = CPLTestBool(CSLFetchNameValueDef(
     112             :         papszOptionsIn, "3D_EXTENSIBLE_MODE",
     113             :         CPLGetConfigOption("DXF_3D_EXTENSIBLE_MODE", "FALSE")));
     114             : 
     115         219 :     m_bClosedLineAsPolygon = CPLTestBool(CSLFetchNameValueDef(
     116             :         papszOptionsIn, "CLOSED_LINE_AS_POLYGON",
     117             :         CPLGetConfigOption("DXF_CLOSED_LINE_AS_POLYGON", "FALSE")));
     118             : 
     119         219 :     m_dfHatchTolerance = CPLAtof(
     120             :         CSLFetchNameValueDef(papszOptionsIn, "HATCH_TOLERANCE",
     121             :                              CPLGetConfigOption("DXF_HATCH_TOLERANCE", "-1")));
     122             : 
     123             :     // Only for debugging
     124         219 :     if (CPLTestBool(CPLGetConfigOption("DXF_HEADER_ONLY", "FALSE")))
     125           0 :         bHeaderOnly = true;
     126             : 
     127             :     /* -------------------------------------------------------------------- */
     128             :     /*      Open the file.                                                  */
     129             :     /* -------------------------------------------------------------------- */
     130         219 :     fp = fpIn;
     131         219 :     if (fp == nullptr)
     132          67 :         fp = VSIFOpenL(pszFilename, "rb");
     133         219 :     if (fp == nullptr)
     134           0 :         return false;
     135             : 
     136         219 :     VSIFSeekL(fp, 0, SEEK_SET);
     137         438 :     std::string osBuffer;
     138         219 :     constexpr size_t nBinarySignatureLen = AUTOCAD_BINARY_DXF_SIGNATURE.size();
     139         219 :     osBuffer.resize(nBinarySignatureLen);
     140         219 :     VSIFReadL(osBuffer.data(), 1, osBuffer.size(), fp);
     141         219 :     if (memcmp(osBuffer.data(), AUTOCAD_BINARY_DXF_SIGNATURE.data(),
     142         219 :                nBinarySignatureLen) == 0)
     143             :     {
     144           5 :         poReader = std::make_unique<OGRDXFReaderBinary>();
     145             :     }
     146             :     else
     147             :     {
     148         214 :         VSIFSeekL(fp, 0, SEEK_SET);
     149             :     }
     150             : 
     151         219 :     poReader->Initialize(fp);
     152             : 
     153             :     /* -------------------------------------------------------------------- */
     154             :     /*      Confirm we have a header section.                               */
     155             :     /* -------------------------------------------------------------------- */
     156             :     char szLineBuf[257];
     157         219 :     bool bEntitiesOnly = false;
     158             : 
     159         219 :     if (ReadValue(szLineBuf) != 0 || !EQUAL(szLineBuf, "SECTION"))
     160           0 :         return false;
     161             : 
     162         438 :     if (ReadValue(szLineBuf) != 2 ||
     163         219 :         (!EQUAL(szLineBuf, "HEADER") && !EQUAL(szLineBuf, "ENTITIES") &&
     164           9 :          !EQUAL(szLineBuf, "TABLES")))
     165           0 :         return false;
     166             : 
     167         219 :     if (EQUAL(szLineBuf, "ENTITIES"))
     168          19 :         bEntitiesOnly = true;
     169             : 
     170             :     /* Some files might have no header but begin directly with a TABLES section
     171             :      */
     172         200 :     else if (EQUAL(szLineBuf, "TABLES"))
     173             :     {
     174             :         osEncoding = CSLFetchNameValueDef(
     175             :             papszOptionsIn, "ENCODING",
     176           9 :             CPLGetConfigOption("DXF_ENCODING", osEncoding));
     177             : 
     178           9 :         if (!ReadTablesSection())
     179           0 :             return false;
     180           9 :         if (ReadValue(szLineBuf) < 0)
     181             :         {
     182           0 :             DXF_READER_ERROR();
     183           0 :             return false;
     184             :         }
     185             :     }
     186             : 
     187             :     /* -------------------------------------------------------------------- */
     188             :     /*      Process the header, picking up a few useful pieces of           */
     189             :     /*      information.                                                    */
     190             :     /* -------------------------------------------------------------------- */
     191             :     else /* if( EQUAL(szLineBuf,"HEADER") ) */
     192             :     {
     193         191 :         if (!ReadHeaderSection(papszOptionsIn))
     194           0 :             return false;
     195         191 :         if (ReadValue(szLineBuf) < 0)
     196             :         {
     197           0 :             DXF_READER_ERROR();
     198           0 :             return false;
     199             :         }
     200             : 
     201             :         /* --------------------------------------------------------------------
     202             :          */
     203             :         /*      Process the CLASSES section, if present. */
     204             :         /* --------------------------------------------------------------------
     205             :          */
     206         191 :         if (EQUAL(szLineBuf, "ENDSEC"))
     207             :         {
     208           0 :             if (ReadValue(szLineBuf) < 0)
     209             :             {
     210           0 :                 DXF_READER_ERROR();
     211           0 :                 return false;
     212             :             }
     213             :         }
     214             : 
     215         191 :         if (EQUAL(szLineBuf, "SECTION"))
     216             :         {
     217         190 :             if (ReadValue(szLineBuf) < 0)
     218             :             {
     219           0 :                 DXF_READER_ERROR();
     220           0 :                 return false;
     221             :             }
     222             :         }
     223             : 
     224         191 :         if (EQUAL(szLineBuf, "CLASSES"))
     225             :         {
     226             :             // int nCode = 0;
     227        2067 :             while ((/* nCode = */ ReadValue(szLineBuf, sizeof(szLineBuf))) >
     228        4134 :                        -1 &&
     229        2067 :                    !EQUAL(szLineBuf, "ENDSEC"))
     230             :             {
     231             :                 // printf("C:%d/%s\n", nCode, szLineBuf );
     232             :             }
     233             :         }
     234             : 
     235             :         /* --------------------------------------------------------------------
     236             :          */
     237             :         /*      Process the TABLES section, if present. */
     238             :         /* --------------------------------------------------------------------
     239             :          */
     240         191 :         if (EQUAL(szLineBuf, "ENDSEC"))
     241             :         {
     242         151 :             if (ReadValue(szLineBuf) < 0)
     243             :             {
     244           0 :                 DXF_READER_ERROR();
     245           0 :                 return false;
     246             :             }
     247             :         }
     248             : 
     249         191 :         if (EQUAL(szLineBuf, "SECTION"))
     250             :         {
     251         151 :             if (ReadValue(szLineBuf) < 0)
     252             :             {
     253           0 :                 DXF_READER_ERROR();
     254           0 :                 return false;
     255             :             }
     256             :         }
     257             : 
     258         191 :         if (EQUAL(szLineBuf, "TABLES"))
     259             :         {
     260         190 :             if (!ReadTablesSection())
     261           0 :                 return false;
     262         190 :             if (ReadValue(szLineBuf) < 0)
     263             :             {
     264           0 :                 DXF_READER_ERROR();
     265           0 :                 return false;
     266             :             }
     267             :         }
     268             :     }
     269             : 
     270             :     /* -------------------------------------------------------------------- */
     271             :     /*      Create a blocks layer if we are not in inlining mode.           */
     272             :     /* -------------------------------------------------------------------- */
     273         219 :     if (!bInlineBlocks)
     274           5 :         apoLayers.push_back(new OGRDXFBlocksLayer(this));
     275             : 
     276             :     /* -------------------------------------------------------------------- */
     277             :     /*      Create out layer object - we will need it when interpreting     */
     278             :     /*      blocks.                                                         */
     279             :     /* -------------------------------------------------------------------- */
     280         219 :     apoLayers.push_back(new OGRDXFLayer(this));
     281             : 
     282             :     /* -------------------------------------------------------------------- */
     283             :     /*      Process the BLOCKS section if present.                          */
     284             :     /* -------------------------------------------------------------------- */
     285         219 :     if (!bEntitiesOnly)
     286             :     {
     287         200 :         if (EQUAL(szLineBuf, "ENDSEC"))
     288             :         {
     289           0 :             if (ReadValue(szLineBuf) < 0)
     290             :             {
     291           0 :                 DXF_READER_ERROR();
     292           0 :                 return false;
     293             :             }
     294             :         }
     295             : 
     296         200 :         if (EQUAL(szLineBuf, "SECTION"))
     297             :         {
     298         197 :             if (ReadValue(szLineBuf) < 0)
     299             :             {
     300           0 :                 DXF_READER_ERROR();
     301           0 :                 return false;
     302             :             }
     303             :         }
     304             : 
     305         200 :         if (EQUAL(szLineBuf, "BLOCKS"))
     306             :         {
     307         197 :             if (!ReadBlocksSection())
     308           1 :                 return false;
     309         196 :             if (ReadValue(szLineBuf) < 0)
     310             :             {
     311           0 :                 DXF_READER_ERROR();
     312           0 :                 return false;
     313             :             }
     314             :         }
     315             :     }
     316             : 
     317         218 :     if (bHeaderOnly)
     318          67 :         return true;
     319             : 
     320             :     /* -------------------------------------------------------------------- */
     321             :     /*      Now we are at the entities section, hopefully.  Confirm.        */
     322             :     /* -------------------------------------------------------------------- */
     323         151 :     if (EQUAL(szLineBuf, "SECTION"))
     324             :     {
     325         129 :         if (ReadValue(szLineBuf) < 0)
     326             :         {
     327           0 :             DXF_READER_ERROR();
     328           0 :             return false;
     329             :         }
     330             :     }
     331             : 
     332         151 :     if (!EQUAL(szLineBuf, "ENTITIES"))
     333             :     {
     334           0 :         DXF_READER_ERROR();
     335           0 :         return false;
     336             :     }
     337             : 
     338         151 :     iEntitiesOffset = poReader->GetCurrentFilePos();
     339         151 :     iEntitiesLineNumber = poReader->nLineNumber;
     340         151 :     apoLayers[0]->ResetReading();
     341             : 
     342         151 :     return true;
     343             : }
     344             : 
     345             : /************************************************************************/
     346             : /*                         ReadTablesSection()                          */
     347             : /************************************************************************/
     348             : 
     349         199 : bool OGRDXFDataSource::ReadTablesSection()
     350             : 
     351             : {
     352             :     char szLineBuf[257];
     353         199 :     int nCode = 0;
     354             : 
     355        3768 :     while ((nCode = ReadValue(szLineBuf, sizeof(szLineBuf))) > -1 &&
     356        1884 :            !EQUAL(szLineBuf, "ENDSEC"))
     357             :     {
     358             :         // We are only interested in extracting tables.
     359        1685 :         if (nCode != 0 || !EQUAL(szLineBuf, "TABLE"))
     360           0 :             continue;
     361             : 
     362        1685 :         nCode = ReadValue(szLineBuf, sizeof(szLineBuf));
     363        1685 :         if (nCode < 0)
     364             :         {
     365           0 :             DXF_READER_ERROR();
     366           0 :             return false;
     367             :         }
     368             : 
     369        1685 :         if (nCode != 2)
     370           0 :             continue;
     371             : 
     372             :         // CPLDebug( "DXF", "Found table %s.", szLineBuf );
     373             : 
     374       48790 :         while ((nCode = ReadValue(szLineBuf, sizeof(szLineBuf))) > -1 &&
     375       24395 :                !EQUAL(szLineBuf, "ENDTAB"))
     376             :         {
     377       22710 :             if (nCode == 0 && EQUAL(szLineBuf, "LAYER"))
     378             :             {
     379         217 :                 if (!ReadLayerDefinition())
     380           0 :                     return false;
     381             :             }
     382       22710 :             if (nCode == 0 && EQUAL(szLineBuf, "LTYPE"))
     383             :             {
     384         749 :                 if (!ReadLineTypeDefinition())
     385           0 :                     return false;
     386             :             }
     387       22710 :             if (nCode == 0 && EQUAL(szLineBuf, "STYLE"))
     388             :             {
     389         209 :                 if (!ReadTextStyleDefinition())
     390           0 :                     return false;
     391             :             }
     392       22710 :             if (nCode == 0 && EQUAL(szLineBuf, "DIMSTYLE"))
     393             :             {
     394         194 :                 if (!ReadDimStyleDefinition())
     395           0 :                     return false;
     396             :             }
     397             :         }
     398             :     }
     399         199 :     if (nCode < 0)
     400             :     {
     401           0 :         DXF_READER_ERROR();
     402           0 :         return false;
     403             :     }
     404             : 
     405         199 :     CPLDebug("DXF", "Read %d layer definitions.", (int)oLayerTable.size());
     406         199 :     return true;
     407             : }
     408             : 
     409             : /************************************************************************/
     410             : /*                        ReadLayerDefinition()                         */
     411             : /************************************************************************/
     412             : 
     413         217 : bool OGRDXFDataSource::ReadLayerDefinition()
     414             : 
     415             : {
     416             :     char szLineBuf[257];
     417         217 :     int nCode = 0;
     418         434 :     std::map<CPLString, CPLString> oLayerProperties;
     419         434 :     CPLString osLayerName = "";
     420             : 
     421         217 :     oLayerProperties["Hidden"] = "0";
     422             : 
     423        2297 :     while ((nCode = ReadValue(szLineBuf, sizeof(szLineBuf))) > 0)
     424             :     {
     425        2080 :         switch (nCode)
     426             :         {
     427         217 :             case 2:
     428             :                 osLayerName =
     429         217 :                     CPLString(szLineBuf).Recode(GetEncoding(), CPL_ENC_UTF8);
     430         217 :                 oLayerProperties["Exists"] = "1";
     431         217 :                 break;
     432             : 
     433         216 :             case 6:
     434         432 :                 oLayerProperties["Linetype"] =
     435         648 :                     CPLString(szLineBuf).Recode(GetEncoding(), CPL_ENC_UTF8);
     436         216 :                 break;
     437             : 
     438         217 :             case 62:
     439         217 :                 oLayerProperties["Color"] = szLineBuf;
     440             : 
     441             :                 // Is layer off?
     442         217 :                 if (atoi(szLineBuf) < 0 && oLayerProperties["Hidden"] != "2")
     443           2 :                     oLayerProperties["Hidden"] = "1";
     444         217 :                 break;
     445             : 
     446           1 :             case 420:
     447           1 :                 oLayerProperties["TrueColor"] = szLineBuf;
     448           1 :                 break;
     449             : 
     450           0 :             case 440:
     451           0 :                 oLayerProperties["Transparency"] = szLineBuf;
     452           0 :                 break;
     453             : 
     454         217 :             case 70:
     455         217 :                 oLayerProperties["Flags"] = szLineBuf;
     456             : 
     457             :                 // Is layer frozen?
     458         217 :                 if (atoi(szLineBuf) & 0x01)
     459           5 :                     oLayerProperties["Hidden"] = "2";
     460         217 :                 break;
     461             : 
     462         192 :             case 370:
     463             :             case 39:
     464         192 :                 oLayerProperties["LineWeight"] = szLineBuf;
     465         192 :                 break;
     466             : 
     467        1020 :             default:
     468        1020 :                 break;
     469             :         }
     470             :     }
     471         217 :     if (nCode < 0)
     472             :     {
     473           0 :         DXF_READER_ERROR();
     474           0 :         return false;
     475             :     }
     476             : 
     477         217 :     if (!oLayerProperties.empty())
     478         217 :         oLayerTable[osLayerName] = std::move(oLayerProperties);
     479             : 
     480         217 :     if (nCode == 0)
     481         217 :         UnreadValue();
     482         217 :     return true;
     483             : }
     484             : 
     485             : /************************************************************************/
     486             : /*                        LookupLayerProperty()                         */
     487             : /************************************************************************/
     488             : 
     489             : std::optional<CPLString>
     490       13468 : OGRDXFDataSource::LookupLayerProperty(const char *pszLayer,
     491             :                                       const char *pszProperty) const
     492             : 
     493             : {
     494       13468 :     if (pszLayer)
     495             :     {
     496       13468 :         const auto oIterLayer = oLayerTable.find(pszLayer);
     497       13468 :         if (oIterLayer != oLayerTable.end())
     498             :         {
     499       12626 :             const auto &oTable = oIterLayer->second;
     500       12626 :             const auto oIterProperty = oTable.find(pszProperty);
     501       12626 :             if (oIterProperty != oTable.end())
     502        7091 :                 return oIterProperty->second;
     503             :         }
     504             :     }
     505        6377 :     return std::nullopt;
     506             : }
     507             : 
     508             : /************************************************************************/
     509             : /*                       ReadLineTypeDefinition()                       */
     510             : /************************************************************************/
     511             : 
     512         749 : bool OGRDXFDataSource::ReadLineTypeDefinition()
     513             : 
     514             : {
     515             :     char szLineBuf[257];
     516         749 :     int nCode = 0;
     517        1498 :     CPLString osLineTypeName;
     518        1498 :     std::vector<double> oLineTypeDef;
     519             :     double dfThisValue;
     520             : 
     521        9613 :     while ((nCode = ReadValue(szLineBuf, sizeof(szLineBuf))) > 0)
     522             :     {
     523        8864 :         switch (nCode)
     524             :         {
     525         749 :             case 2:
     526             :                 osLineTypeName =
     527         749 :                     CPLString(szLineBuf).Recode(GetEncoding(), CPL_ENC_UTF8);
     528         749 :                 break;
     529             : 
     530         840 :             case 49:
     531         840 :                 dfThisValue = CPLAtof(szLineBuf);
     532             : 
     533             :                 // Same sign as the previous entry? Continue the previous dash
     534             :                 // or gap by appending this length
     535        1463 :                 if (oLineTypeDef.size() > 0 &&
     536         623 :                     (dfThisValue < 0) == (oLineTypeDef.back() < 0))
     537             :                 {
     538           1 :                     oLineTypeDef.back() += dfThisValue;
     539             :                 }
     540             :                 // Otherwise, add a new entry
     541             :                 else
     542             :                 {
     543         839 :                     oLineTypeDef.push_back(dfThisValue);
     544             :                 }
     545             : 
     546         840 :                 break;
     547             : 
     548        7275 :             default:
     549        7275 :                 break;
     550             :         }
     551             :     }
     552         749 :     if (nCode < 0)
     553             :     {
     554           0 :         DXF_READER_ERROR();
     555           0 :         return false;
     556             :     }
     557             : 
     558             :     // Deal with an odd number of elements by adding the last element
     559             :     // onto the first
     560         749 :     if (oLineTypeDef.size() % 2 == 1)
     561             :     {
     562           1 :         oLineTypeDef.front() += oLineTypeDef.back();
     563           1 :         oLineTypeDef.pop_back();
     564             :     }
     565             : 
     566         749 :     if (oLineTypeDef.size())
     567             :     {
     568             :         // If the first element is a gap, rotate the elements so the first
     569             :         // element is a dash
     570         217 :         if (oLineTypeDef.front() < 0)
     571             :         {
     572           2 :             std::rotate(oLineTypeDef.begin(), oLineTypeDef.begin() + 1,
     573           2 :                         oLineTypeDef.end());
     574             :         }
     575             : 
     576         217 :         oLineTypeTable[osLineTypeName] = std::move(oLineTypeDef);
     577             :     }
     578             : 
     579         749 :     if (nCode == 0)
     580         749 :         UnreadValue();
     581         749 :     return true;
     582             : }
     583             : 
     584             : /************************************************************************/
     585             : /*                           LookupLineType()                           */
     586             : /************************************************************************/
     587             : 
     588        2734 : std::vector<double> OGRDXFDataSource::LookupLineType(const char *pszName)
     589             : 
     590             : {
     591        2734 :     if (pszName && oLineTypeTable.count(pszName) > 0)
     592          47 :         return oLineTypeTable[pszName];
     593             :     else
     594        2687 :         return std::vector<double>();  // empty, represents a continuous line
     595             : }
     596             : 
     597             : /************************************************************************/
     598             : /*                       ReadTextStyleDefinition()                      */
     599             : /************************************************************************/
     600             : 
     601         209 : bool OGRDXFDataSource::ReadTextStyleDefinition()
     602             : 
     603             : {
     604             :     char szLineBuf[257];
     605         209 :     int nCode = 0;
     606             : 
     607         418 :     CPLString osStyleHandle;
     608         418 :     CPLString osStyleName;
     609         209 :     bool bInsideAcadSection = false;
     610             : 
     611        2844 :     while ((nCode = ReadValue(szLineBuf, sizeof(szLineBuf))) > 0)
     612             :     {
     613        2635 :         switch (nCode)
     614             :         {
     615         185 :             case 5:
     616         185 :                 osStyleHandle = szLineBuf;
     617         185 :                 break;
     618             : 
     619         209 :             case 2:
     620         418 :                 osStyleName = CPLString(szLineBuf)
     621         209 :                                   .Recode(GetEncoding(), CPL_ENC_UTF8)
     622         209 :                                   .toupper();
     623         209 :                 break;
     624             : 
     625         207 :             case 70:
     626             :                 // If the LSB is set, this is not a text style
     627         207 :                 if (atoi(szLineBuf) & 1)
     628           0 :                     return true;
     629         207 :                 break;
     630             : 
     631             :                 // Note: 40 and 41 group codes do not propagate from a text
     632             :                 // style down to TEXT objects. However, 41 does propagate down
     633             :                 // for MTEXT.
     634             : 
     635         209 :             case 41:
     636         209 :                 oTextStyleTable[osStyleName]["Width"] = szLineBuf;
     637         209 :                 break;
     638             : 
     639          14 :             case 1001:
     640          14 :                 bInsideAcadSection = EQUAL(szLineBuf, "ACAD");
     641          14 :                 break;
     642             : 
     643          14 :             case 1000:
     644          14 :                 if (bInsideAcadSection)
     645          13 :                     oTextStyleTable[osStyleName]["Font"] = szLineBuf;
     646          14 :                 break;
     647             : 
     648          13 :             case 1071:
     649             :                 // bold and italic are kept in this undocumented bitfield
     650          13 :                 if (bInsideAcadSection)
     651             :                 {
     652          13 :                     const int nFontFlags = atoi(szLineBuf);
     653          26 :                     oTextStyleTable[osStyleName]["Bold"] =
     654          39 :                         (nFontFlags & 0x2000000) ? "1" : "0";
     655          26 :                     oTextStyleTable[osStyleName]["Italic"] =
     656          39 :                         (nFontFlags & 0x1000000) ? "1" : "0";
     657             :                 }
     658          13 :                 break;
     659             : 
     660        1784 :             default:
     661        1784 :                 break;
     662             :         }
     663             :     }
     664         209 :     if (nCode < 0)
     665             :     {
     666           0 :         DXF_READER_ERROR();
     667           0 :         return false;
     668             :     }
     669             : 
     670         209 :     if (nCode == 0)
     671         209 :         UnreadValue();
     672             : 
     673         209 :     if (osStyleHandle != "")
     674         185 :         oTextStyleHandles[osStyleHandle] = std::move(osStyleName);
     675             : 
     676         209 :     return true;
     677             : }
     678             : 
     679             : /************************************************************************/
     680             : /*                           TextStyleExists()                          */
     681             : /************************************************************************/
     682             : 
     683           1 : bool OGRDXFDataSource::TextStyleExists(const char *pszTextStyle)
     684             : 
     685             : {
     686           1 :     if (!pszTextStyle)
     687           0 :         return false;
     688             : 
     689           1 :     CPLString osTextStyleUpper = pszTextStyle;
     690           1 :     osTextStyleUpper.toupper();
     691             : 
     692           1 :     return oTextStyleTable.count(osTextStyleUpper) > 0;
     693             : }
     694             : 
     695             : /************************************************************************/
     696             : /*                       LookupTextStyleProperty()                      */
     697             : /************************************************************************/
     698             : 
     699         395 : const char *OGRDXFDataSource::LookupTextStyleProperty(const char *pszTextStyle,
     700             :                                                       const char *pszProperty,
     701             :                                                       const char *pszDefault)
     702             : 
     703             : {
     704         395 :     if (!pszTextStyle)
     705           0 :         return pszDefault;
     706             : 
     707         790 :     CPLString osTextStyleUpper = pszTextStyle;
     708         395 :     osTextStyleUpper.toupper();
     709             : 
     710         653 :     if (pszProperty && oTextStyleTable.count(osTextStyleUpper) > 0 &&
     711         653 :         oTextStyleTable[osTextStyleUpper].count(pszProperty) > 0)
     712             :     {
     713          54 :         return (oTextStyleTable[osTextStyleUpper])[pszProperty];
     714             :     }
     715             :     else
     716             :     {
     717         341 :         return pszDefault;
     718             :     }
     719             : }
     720             : 
     721             : /************************************************************************/
     722             : /*                      GetTextStyleNameByHandle()                      */
     723             : /*                                                                      */
     724             : /*      Find the name of the text style with the given STYLE table      */
     725             : /*      handle. If there is no such style, an empty string is returned. */
     726             : /************************************************************************/
     727             : 
     728          10 : CPLString OGRDXFDataSource::GetTextStyleNameByHandle(const char *pszID)
     729             : 
     730             : {
     731          20 :     CPLString l_osID = pszID;
     732             : 
     733          10 :     if (oTextStyleHandles.count(l_osID) == 0)
     734           6 :         return "";
     735             :     else
     736           4 :         return oTextStyleHandles[l_osID];
     737             : }
     738             : 
     739             : /************************************************************************/
     740             : /*                  PopulateDefaultDimStyleProperties()                 */
     741             : /************************************************************************/
     742             : 
     743         254 : void OGRDXFDataSource::PopulateDefaultDimStyleProperties(
     744             :     std::map<CPLString, CPLString> &oDimStyleProperties)
     745             : 
     746             : {
     747         254 :     const int *piCode = ACGetKnownDimStyleCodes();
     748        3048 :     do
     749             :     {
     750        3302 :         const char *pszProperty = ACGetDimStylePropertyName(*piCode);
     751        6604 :         oDimStyleProperties[pszProperty] =
     752        9906 :             ACGetDimStylePropertyDefault(*piCode);
     753        3302 :     } while (*(++piCode));
     754         254 : }
     755             : 
     756             : /************************************************************************/
     757             : /*                       ReadDimStyleDefinition()                       */
     758             : /************************************************************************/
     759             : 
     760         194 : bool OGRDXFDataSource::ReadDimStyleDefinition()
     761             : 
     762             : {
     763             :     char szLineBuf[257];
     764         194 :     int nCode = 0;
     765         388 :     std::map<CPLString, CPLString> oDimStyleProperties;
     766         388 :     CPLString osDimStyleName = "";
     767             : 
     768         194 :     PopulateDefaultDimStyleProperties(oDimStyleProperties);
     769             : 
     770        2510 :     while ((nCode = ReadValue(szLineBuf, sizeof(szLineBuf))) > 0)
     771             :     {
     772        2316 :         switch (nCode)
     773             :         {
     774         194 :             case 2:
     775             :                 osDimStyleName =
     776         194 :                     CPLString(szLineBuf).Recode(GetEncoding(), CPL_ENC_UTF8);
     777         194 :                 break;
     778             : 
     779        2122 :             default:
     780        2122 :                 const char *pszProperty = ACGetDimStylePropertyName(nCode);
     781        2122 :                 if (pszProperty)
     782         292 :                     oDimStyleProperties[pszProperty] = szLineBuf;
     783        2122 :                 break;
     784             :         }
     785             :     }
     786         194 :     if (nCode < 0)
     787             :     {
     788           0 :         DXF_READER_ERROR();
     789           0 :         return false;
     790             :     }
     791             : 
     792         194 :     if (!oDimStyleProperties.empty())
     793         194 :         oDimStyleTable[osDimStyleName] = std::move(oDimStyleProperties);
     794             : 
     795         194 :     if (nCode == 0)
     796         194 :         UnreadValue();
     797         194 :     return true;
     798             : }
     799             : 
     800             : /************************************************************************/
     801             : /*                           LookupDimStyle()                           */
     802             : /*                                                                      */
     803             : /*      If the specified DIMSTYLE does not exist, a default set of      */
     804             : /*      of style properties are copied into oDimStyleProperties and     */
     805             : /*      false is returned.  Otherwise true is returned.                 */
     806             : /************************************************************************/
     807             : 
     808          42 : bool OGRDXFDataSource::LookupDimStyle(
     809             :     const char *pszDimStyle,
     810             :     std::map<CPLString, CPLString> &oDimStyleProperties)
     811             : 
     812             : {
     813          42 :     if (pszDimStyle == nullptr || !oDimStyleTable.count(pszDimStyle))
     814             :     {
     815          16 :         PopulateDefaultDimStyleProperties(oDimStyleProperties);
     816          16 :         return false;
     817             :     }
     818             : 
     819             :     // make a copy of the DIMSTYLE properties, so no-one can mess around
     820             :     // with our original copy
     821          26 :     oDimStyleProperties = oDimStyleTable[pszDimStyle];
     822          26 :     return true;
     823             : }
     824             : 
     825             : /************************************************************************/
     826             : /*                         ReadHeaderSection()                          */
     827             : /************************************************************************/
     828             : 
     829         191 : bool OGRDXFDataSource::ReadHeaderSection(CSLConstList papszOpenOptionsIn)
     830             : 
     831             : {
     832             :     char szLineBuf[257];
     833         191 :     int nCode = 0;
     834             : 
     835       57172 :     while ((nCode = ReadValue(szLineBuf, sizeof(szLineBuf))) > -1 &&
     836       28586 :            !EQUAL(szLineBuf, "ENDSEC"))
     837             :     {
     838       28395 :         if (nCode != 9)
     839        7768 :             continue;
     840             : 
     841       20627 :         CPLString l_osName = szLineBuf;
     842             : 
     843       20627 :         if (ReadValue(szLineBuf, sizeof(szLineBuf)) < 0)
     844             :         {
     845           0 :             DXF_READER_ERROR();
     846           0 :             return false;
     847             :         }
     848             : 
     849       20627 :         oHeaderVariables[l_osName] = szLineBuf;
     850       20627 :         GDALDataset::SetMetadataItem(l_osName.c_str(), szLineBuf,
     851             :                                      "DXF_HEADER_VARIABLES");
     852             :     }
     853         191 :     if (nCode < 0)
     854             :     {
     855           0 :         DXF_READER_ERROR();
     856           0 :         return false;
     857             :     }
     858             : 
     859         191 :     nCode = ReadValue(szLineBuf, sizeof(szLineBuf));
     860         191 :     if (nCode < 0)
     861             :     {
     862           0 :         DXF_READER_ERROR();
     863           0 :         return false;
     864             :     }
     865         191 :     UnreadValue();
     866             : 
     867             :     /* Unusual DXF files produced by dxflib */
     868             :     /* such as http://www.ribbonsoft.com/library/architecture/plants/decd5.dxf
     869             :      */
     870             :     /* where there is a spurious ENDSEC in the middle of the header variables */
     871         191 :     if (nCode == 9 && STARTS_WITH_CI(szLineBuf, "$"))
     872             :     {
     873           0 :         while ((nCode = ReadValue(szLineBuf, sizeof(szLineBuf))) > -1 &&
     874           0 :                !EQUAL(szLineBuf, "ENDSEC"))
     875             :         {
     876           0 :             if (nCode != 9)
     877           0 :                 continue;
     878             : 
     879           0 :             CPLString l_osName = szLineBuf;
     880             : 
     881           0 :             if (ReadValue(szLineBuf, sizeof(szLineBuf)) < 0)
     882             :             {
     883           0 :                 DXF_READER_ERROR();
     884           0 :                 return false;
     885             :             }
     886             : 
     887           0 :             oHeaderVariables[l_osName] = szLineBuf;
     888           0 :             GDALDataset::SetMetadataItem(l_osName.c_str(), szLineBuf,
     889             :                                          "DXF_HEADER_VARIABLES");
     890             :         }
     891           0 :         if (nCode < 0)
     892             :         {
     893           0 :             DXF_READER_ERROR();
     894           0 :             return false;
     895             :         }
     896             :     }
     897             : 
     898         191 :     CPLDebug("DXF", "Read %d header variables.", (int)oHeaderVariables.size());
     899             : 
     900             :     /* -------------------------------------------------------------------- */
     901             :     /*      Decide on what CPLRecode() name to use for the files            */
     902             :     /*      encoding or allow the encoding to be overridden.                */
     903             :     /* -------------------------------------------------------------------- */
     904         191 :     CPLString osCodepage = GetVariable("$DWGCODEPAGE", "ANSI_1252");
     905             : 
     906             :     // not strictly accurate but works even without iconv.
     907         191 :     if (osCodepage == "ANSI_1252")
     908         186 :         osEncoding = CPL_ENC_ISO8859_1;
     909           5 :     else if (STARTS_WITH_CI(osCodepage, "ANSI_"))
     910             :     {
     911           5 :         osEncoding = "CP";
     912           5 :         osEncoding += osCodepage + 5;
     913             :     }
     914             :     else
     915             :     {
     916             :         // fallback to the default
     917           0 :         osEncoding = CPL_ENC_ISO8859_1;
     918             :     }
     919             : 
     920             :     const char *pszEncoding =
     921         191 :         CSLFetchNameValueDef(papszOpenOptionsIn, "ENCODING",
     922             :                              CPLGetConfigOption("DXF_ENCODING", nullptr));
     923         191 :     if (pszEncoding != nullptr)
     924           1 :         osEncoding = pszEncoding;
     925             : 
     926         191 :     if (osEncoding != CPL_ENC_ISO8859_1)
     927           6 :         CPLDebug("DXF", "Treating DXF as encoding '%s', $DWGCODEPAGE='%s'",
     928             :                  osEncoding.c_str(), osCodepage.c_str());
     929         191 :     return true;
     930             : }
     931             : 
     932             : /************************************************************************/
     933             : /*                            GetVariable()                             */
     934             : /*                                                                      */
     935             : /*      Fetch a variable that came from the HEADER section.             */
     936             : /************************************************************************/
     937             : 
     938        2918 : const char *OGRDXFDataSource::GetVariable(const char *pszName,
     939             :                                           const char *pszDefault)
     940             : 
     941             : {
     942        2918 :     if (oHeaderVariables.count(pszName) == 0)
     943        1117 :         return pszDefault;
     944             :     else
     945        1801 :         return oHeaderVariables[pszName];
     946             : }
     947             : 
     948             : /************************************************************************/
     949             : /*                         AddStandardFields()                          */
     950             : /************************************************************************/
     951             : 
     952         306 : void OGRDXFDataSource::AddStandardFields(OGRFeatureDefn *poFeatureDefn,
     953             :                                          const int nFieldModes)
     954             : {
     955         612 :     OGRFieldDefn oLayerField("Layer", OFTString);
     956         306 :     poFeatureDefn->AddFieldDefn(&oLayerField);
     957             : 
     958         612 :     OGRFieldDefn oPaperSpaceField("PaperSpace", OFTInteger);
     959         306 :     oPaperSpaceField.SetSubType(OFSTBoolean);
     960         306 :     poFeatureDefn->AddFieldDefn(&oPaperSpaceField);
     961             : 
     962         612 :     OGRFieldDefn oClassField("SubClasses", OFTString);
     963         306 :     poFeatureDefn->AddFieldDefn(&oClassField);
     964             : 
     965         306 :     if (nFieldModes & ODFM_IncludeRawCodeValues)
     966             :     {
     967           2 :         OGRFieldDefn oRawCodeField("RawCodeValues", OFTStringList);
     968           1 :         poFeatureDefn->AddFieldDefn(&oRawCodeField);
     969             :     }
     970             : 
     971         612 :     OGRFieldDefn oLinetypeField("Linetype", OFTString);
     972         306 :     poFeatureDefn->AddFieldDefn(&oLinetypeField);
     973             : 
     974         612 :     OGRFieldDefn oEntityHandleField("EntityHandle", OFTString);
     975         306 :     poFeatureDefn->AddFieldDefn(&oEntityHandleField);
     976             : 
     977         612 :     OGRFieldDefn oTextField("Text", OFTString);
     978         306 :     poFeatureDefn->AddFieldDefn(&oTextField);
     979             : 
     980         306 :     if (nFieldModes & ODFM_Include3DModeFields)
     981             :     {
     982           2 :         OGRFieldDefn oASMBinaryField("ASMData", OFTBinary);
     983           1 :         poFeatureDefn->AddFieldDefn(&oASMBinaryField);
     984             : 
     985           2 :         OGRFieldDefn oASMTransformField("ASMTransform", OFTRealList);
     986           1 :         poFeatureDefn->AddFieldDefn(&oASMTransformField);
     987             :     }
     988             : 
     989         306 :     if (nFieldModes & ODFM_IncludeBlockFields)
     990             :     {
     991         184 :         OGRFieldDefn oBlockNameField("BlockName", OFTString);
     992          92 :         poFeatureDefn->AddFieldDefn(&oBlockNameField);
     993             : 
     994         184 :         OGRFieldDefn oScaleField("BlockScale", OFTRealList);
     995          92 :         poFeatureDefn->AddFieldDefn(&oScaleField);
     996             : 
     997         184 :         OGRFieldDefn oBlockAngleField("BlockAngle", OFTReal);
     998          92 :         poFeatureDefn->AddFieldDefn(&oBlockAngleField);
     999             : 
    1000         184 :         OGRFieldDefn oBlockOCSNormalField("BlockOCSNormal", OFTRealList);
    1001          92 :         poFeatureDefn->AddFieldDefn(&oBlockOCSNormalField);
    1002             : 
    1003         184 :         OGRFieldDefn oBlockOCSCoordsField("BlockOCSCoords", OFTRealList);
    1004          92 :         poFeatureDefn->AddFieldDefn(&oBlockOCSCoordsField);
    1005             : 
    1006         184 :         OGRFieldDefn oBlockAttribsField("BlockAttributes", OFTStringList);
    1007          92 :         poFeatureDefn->AddFieldDefn(&oBlockAttribsField);
    1008             : 
    1009             :         // This field holds the name of the block on which the entity lies.
    1010             :         // The BlockName field was previously used for this purpose; this
    1011             :         // was changed because of the ambiguity with the BlockName field
    1012             :         // used by INSERT entities.
    1013         184 :         OGRFieldDefn oBlockField("Block", OFTString);
    1014          92 :         poFeatureDefn->AddFieldDefn(&oBlockField);
    1015             : 
    1016             :         // Extra field to use with ATTDEF entities
    1017         184 :         OGRFieldDefn oAttributeTagField("AttributeTag", OFTString);
    1018          92 :         poFeatureDefn->AddFieldDefn(&oAttributeTagField);
    1019             :     }
    1020         306 : }
    1021             : 
    1022             : /************************************************************************/
    1023             : /*                    GetEntryFromAcDsDataSection()                     */
    1024             : /************************************************************************/
    1025             : 
    1026             : size_t
    1027           2 : OGRDXFDataSource::GetEntryFromAcDsDataSection(const char *pszEntityHandle,
    1028             :                                               const GByte **pabyBuffer)
    1029             : 
    1030             : {
    1031           2 :     if (!pszEntityHandle || !pabyBuffer)
    1032           0 :         return 0;
    1033             : 
    1034           2 :     if (bHaveReadSolidData)
    1035             :     {
    1036           1 :         if (oSolidBinaryData.count(pszEntityHandle) > 0)
    1037             :         {
    1038           1 :             *pabyBuffer = oSolidBinaryData[pszEntityHandle].data();
    1039           1 :             return oSolidBinaryData[pszEntityHandle].size();
    1040             :         }
    1041           0 :         return 0;
    1042             :     }
    1043             : 
    1044             :     // Keep track of our current position and line number in the file so we can
    1045             :     // return here later
    1046           1 :     uint64_t iPrevOffset = poReader->GetCurrentFilePos();
    1047           1 :     int nPrevLineNumber = poReader->nLineNumber;
    1048             : 
    1049             :     char szLineBuf[4096];
    1050           1 :     int nCode = 0;
    1051           1 :     bool bFound = false;
    1052             : 
    1053             :     // Search for the ACDSDATA section
    1054         313 :     while ((nCode = ReadValue(szLineBuf, sizeof(szLineBuf))) >= 0)
    1055             :     {
    1056             :         // Check whether the ACDSDATA section starts here
    1057         313 :         if (nCode == 0 && EQUAL(szLineBuf, "SECTION"))
    1058             :         {
    1059           3 :             if ((nCode = ReadValue(szLineBuf, sizeof(szLineBuf))) < 0)
    1060             :             {
    1061           0 :                 break;
    1062             :             }
    1063             : 
    1064           3 :             if (nCode == 2 && EQUAL(szLineBuf, "ACDSDATA"))
    1065             :             {
    1066           1 :                 bFound = true;
    1067           1 :                 break;
    1068             :             }
    1069             :         }
    1070             :     }
    1071             : 
    1072           1 :     if (!bFound)
    1073             :     {
    1074           0 :         poReader->ResetReadPointer(iPrevOffset, nPrevLineNumber);
    1075           0 :         return 0;
    1076             :     }
    1077             : 
    1078           1 :     bool bInAcDsRecord = false;
    1079           1 :     bool bGotAsmData = false;
    1080           2 :     CPLString osThisHandle;
    1081             : 
    1082             :     // Search for the relevant ACDSRECORD and extract its binary data
    1083         129 :     while ((nCode = ReadValue(szLineBuf, sizeof(szLineBuf))) >= 0)
    1084             :     {
    1085         128 :         if (nCode == 0 && EQUAL(szLineBuf, "ENDSEC"))
    1086             :         {
    1087             :             // We've reached the end of the ACDSDATA section
    1088             :             break;
    1089             :         }
    1090         128 :         else if (nCode == 0)
    1091             :         {
    1092           9 :             bInAcDsRecord = EQUAL(szLineBuf, "ACDSRECORD");
    1093           9 :             bGotAsmData = false;
    1094           9 :             osThisHandle.clear();
    1095             :         }
    1096         119 :         else if (bInAcDsRecord && nCode == 320)
    1097             :         {
    1098           3 :             osThisHandle = szLineBuf;
    1099             :         }
    1100         116 :         else if (bInAcDsRecord && nCode == 2)
    1101             :         {
    1102           6 :             bGotAsmData = EQUAL(szLineBuf, "ASM_Data");
    1103             :         }
    1104         110 :         else if (bInAcDsRecord && bGotAsmData && nCode == 94)
    1105             :         {
    1106             :             // Group code 94 gives the length of the binary data that follows
    1107           2 :             int nLen = atoi(szLineBuf);
    1108             : 
    1109             :             // Enforce some limits (the upper limit is arbitrary)
    1110           2 :             if (nLen <= 0 || nLen > 1048576)
    1111             :             {
    1112           0 :                 CPLError(CE_Warning, CPLE_AppDefined,
    1113             :                          "ACDSRECORD data for entity %s is too long (more than "
    1114             :                          "1MB in size) and was skipped.",
    1115             :                          pszEntityHandle);
    1116           0 :                 continue;
    1117             :             }
    1118             : 
    1119           2 :             oSolidBinaryData[osThisHandle].resize(nLen);
    1120             : 
    1121             :             // Read the binary data into the buffer
    1122           2 :             int nPos = 0;
    1123          14 :             while (ReadValue(szLineBuf, sizeof(szLineBuf)) == 310)
    1124             :             {
    1125             :                 int nBytesRead;
    1126             :                 std::unique_ptr<GByte, VSIFreeReleaser> pabyHex(
    1127          12 :                     CPLHexToBinary(szLineBuf, &nBytesRead));
    1128             : 
    1129          12 :                 if (nPos + nBytesRead > nLen)
    1130             :                 {
    1131           0 :                     CPLError(CE_Warning, CPLE_AppDefined,
    1132             :                              "Too many bytes in ACDSRECORD data for entity %s. "
    1133             :                              "Is the length (group code 94) correct?",
    1134             :                              pszEntityHandle);
    1135           0 :                     break;
    1136             :                 }
    1137             :                 else
    1138             :                 {
    1139             :                     std::copy_n(pabyHex.get(), nBytesRead,
    1140          12 :                                 oSolidBinaryData[osThisHandle].begin() + nPos);
    1141          12 :                     nPos += nBytesRead;
    1142             :                 }
    1143             :             }
    1144             :         }
    1145             :     }
    1146             : 
    1147           1 :     poReader->ResetReadPointer(iPrevOffset, nPrevLineNumber);
    1148             : 
    1149           1 :     bHaveReadSolidData = true;
    1150             : 
    1151           1 :     if (oSolidBinaryData.count(pszEntityHandle) > 0)
    1152             :     {
    1153           1 :         *pabyBuffer = oSolidBinaryData[pszEntityHandle].data();
    1154           1 :         return oSolidBinaryData[pszEntityHandle].size();
    1155             :     }
    1156           0 :     return 0;
    1157             : }

Generated by: LCOV version 1.14