LCOV - code coverage report
Current view: top level - ogr/ogrsf_frmts/sxf - ogrsxfdatasource.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 365 619 59.0 %
Date: 2025-01-18 12:42:00 Functions: 15 15 100.0 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  SXF Translator
       4             :  * Purpose:  Definition of classes for OGR SXF Datasource.
       5             :  * Author:   Ben Ahmed Daho Ali, bidandou(at)yahoo(dot)fr
       6             :  *           Dmitry Baryshnikov, polimax@mail.ru
       7             :  *           Alexandr Lisovenko, alexander.lisovenko@gmail.com
       8             :  *
       9             :  ******************************************************************************
      10             :  * Copyright (c) 2011, Ben Ahmed Daho Ali
      11             :  * Copyright (c) 2013, NextGIS
      12             :  * Copyright (c) 2014, Even Rouault <even dot rouault at spatialys.com>
      13             :  * Copyright (c) 2019, NextGIS, <info@nextgis.com>
      14             :  *
      15             :  * SPDX-License-Identifier: MIT
      16             :  ****************************************************************************/
      17             : 
      18             : #include "cpl_conv.h"
      19             : #include "ogr_sxf.h"
      20             : #include "cpl_string.h"
      21             : #include "cpl_multiproc.h"
      22             : 
      23             : #include <math.h>
      24             : #include <map>
      25             : #include <string>
      26             : 
      27             : #ifdef EMBED_RESOURCE_FILES
      28             : #include "embedded_resources.h"
      29             : #endif
      30             : 
      31             : // EPSG code range http://gis.stackexchange.com/a/18676/9904
      32             : constexpr int MIN_EPSG = 1000;
      33             : constexpr int MAX_EPSG = 32768;
      34             : 
      35             : /************************************************************************/
      36             : /*                         OGRSXFDataSource()                           */
      37             : /************************************************************************/
      38             : 
      39           6 : OGRSXFDataSource::OGRSXFDataSource()
      40             : {
      41           6 :     memset(&oSXFPassport.informationFlags, 0,
      42             :            sizeof(oSXFPassport.informationFlags));
      43           6 : }
      44             : 
      45             : /************************************************************************/
      46             : /*                          ~OGRSXFDataSource()                         */
      47             : /************************************************************************/
      48             : 
      49          12 : OGRSXFDataSource::~OGRSXFDataSource()
      50             : 
      51             : {
      52           6 :     m_apoLayers.clear();
      53             : 
      54           6 :     if (nullptr != oSXFPassport.stMapDescription.pSpatRef)
      55             :     {
      56           5 :         oSXFPassport.stMapDescription.pSpatRef->Release();
      57             :     }
      58             : 
      59           6 :     CloseFile();
      60             : 
      61           6 :     if (hIOMutex != nullptr)
      62             :     {
      63           1 :         CPLDestroyMutex(hIOMutex);
      64           1 :         hIOMutex = nullptr;
      65             :     }
      66          12 : }
      67             : 
      68             : /************************************************************************/
      69             : /*                     CloseFile()                                      */
      70             : /************************************************************************/
      71           6 : void OGRSXFDataSource::CloseFile()
      72             : {
      73           6 :     if (nullptr != fpSXF)
      74             :     {
      75           5 :         VSIFCloseL(fpSXF);
      76           5 :         fpSXF = nullptr;
      77             :     }
      78           6 : }
      79             : 
      80             : /************************************************************************/
      81             : /*                           TestCapability()                           */
      82             : /************************************************************************/
      83             : 
      84          39 : int OGRSXFDataSource::TestCapability(const char *pszCap)
      85             : {
      86          39 :     if (EQUAL(pszCap, ODsCZGeometries))
      87          18 :         return true;
      88             : 
      89          21 :     return false;
      90             : }
      91             : 
      92             : /************************************************************************/
      93             : /*                              GetLayer()                              */
      94             : /************************************************************************/
      95             : 
      96         942 : OGRLayer *OGRSXFDataSource::GetLayer(int iLayer)
      97             : 
      98             : {
      99         942 :     if (iLayer < 0 || iLayer >= GetLayerCount())
     100           2 :         return nullptr;
     101             :     else
     102         940 :         return m_apoLayers[iLayer].get();
     103             : }
     104             : 
     105             : /************************************************************************/
     106             : /*                                Open()                                */
     107             : /************************************************************************/
     108             : 
     109           6 : int OGRSXFDataSource::Open(const char *pszFilename, bool bUpdateIn,
     110             :                            const char *const *papszOpenOpts)
     111             : {
     112           6 :     if (bUpdateIn)
     113             :     {
     114           1 :         return FALSE;
     115             :     }
     116             : 
     117           5 :     fpSXF = VSIFOpenL(pszFilename, "rb");
     118           5 :     if (fpSXF == nullptr)
     119             :     {
     120           0 :         CPLError(CE_Warning, CPLE_OpenFailed, "SXF open file %s failed",
     121             :                  pszFilename);
     122           0 :         return FALSE;
     123             :     }
     124             : 
     125             :     // read header
     126           5 :     const int nFileHeaderSize = sizeof(SXFHeader);
     127             :     SXFHeader stSXFFileHeader;
     128             :     const size_t nObjectsRead =
     129           5 :         VSIFReadL(&stSXFFileHeader, nFileHeaderSize, 1, fpSXF);
     130             : 
     131           5 :     if (nObjectsRead != 1)
     132             :     {
     133           0 :         CPLError(CE_Failure, CPLE_None, "SXF head read failed");
     134           0 :         CloseFile();
     135           0 :         return FALSE;
     136             :     }
     137           5 :     CPL_LSBPTR32(&(stSXFFileHeader.nHeaderLength));
     138           5 :     CPL_LSBPTR32(&(stSXFFileHeader.nCheckSum));
     139             : 
     140             :     // check version
     141           5 :     oSXFPassport.version = 0;
     142           5 :     if (stSXFFileHeader.nHeaderLength > 256)  // if size == 400 then version >=
     143             :                                               // 4
     144             :     {
     145           5 :         oSXFPassport.version = stSXFFileHeader.nFormatVersion[2];
     146             :     }
     147             :     else
     148             :     {
     149           0 :         oSXFPassport.version = stSXFFileHeader.nFormatVersion[1];
     150             :     }
     151             : 
     152           5 :     if (oSXFPassport.version < 3)
     153             :     {
     154           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     155             :                  "SXF File version not supported");
     156           0 :         CloseFile();
     157           0 :         return FALSE;
     158             :     }
     159             : 
     160             :     // read description
     161           5 :     if (ReadSXFDescription(fpSXF, oSXFPassport) != OGRERR_NONE)
     162             :     {
     163           0 :         CPLError(CE_Failure, CPLE_NotSupported, "SXF. Wrong description.");
     164           0 :         CloseFile();
     165           0 :         return FALSE;
     166             :     }
     167             : 
     168             :     // read flags
     169           5 :     if (ReadSXFInformationFlags(fpSXF, oSXFPassport) != OGRERR_NONE)
     170             :     {
     171           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     172             :                  "SXF. Wrong state of the data.");
     173           0 :         CloseFile();
     174           0 :         return FALSE;
     175             :     }
     176             : 
     177           5 :     if (oSXFPassport.version == 3 &&
     178           0 :         oSXFPassport.informationFlags.bProjectionDataCompliance == false)
     179             :     {
     180           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     181             :                  "SXF. Data does not correspond to the projection.");
     182           0 :         CloseFile();
     183           0 :         return FALSE;
     184             :     }
     185             : 
     186             :     // read spatial data
     187           5 :     if (ReadSXFMapDescription(fpSXF, oSXFPassport, papszOpenOpts) !=
     188             :         OGRERR_NONE)
     189             :     {
     190           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     191             :                  "SXF. Wrong state of the data.");
     192           0 :         CloseFile();
     193           0 :         return FALSE;
     194             :     }
     195             : 
     196           5 :     if (oSXFPassport.informationFlags.bRealCoordinatesCompliance == false)
     197             :     {
     198           5 :         CPLError(CE_Warning, CPLE_NotSupported,
     199             :                  "SXF. Given material may be rotated in the conditional system "
     200             :                  "of coordinates");
     201             :     }
     202             : 
     203             :     /*---------------- TRY READ THE RSC FILE HEADER  -----------------------*/
     204             : 
     205          10 :     CPLString soRSCRileName;
     206             :     std::string osRSCRileNameCandidate =
     207             :         CSLFetchNameValueDef(papszOpenOpts, "SXF_RSC_FILENAME",
     208           5 :                              CPLGetConfigOption("SXF_RSC_FILENAME", ""));
     209           6 :     if (!osRSCRileNameCandidate.empty() &&
     210           1 :         CPLCheckForFile(osRSCRileNameCandidate.data(), nullptr) == TRUE)
     211             :     {
     212           1 :         soRSCRileName = osRSCRileNameCandidate;
     213             :     }
     214             : 
     215           5 :     if (soRSCRileName.empty())
     216             :     {
     217           4 :         osRSCRileNameCandidate = CPLResetExtensionSafe(pszFilename, "rsc");
     218           4 :         if (CPLCheckForFile(osRSCRileNameCandidate.data(), nullptr) == TRUE)
     219             :         {
     220           0 :             soRSCRileName = osRSCRileNameCandidate;
     221             :         }
     222             :     }
     223             : 
     224           5 :     if (soRSCRileName.empty())
     225             :     {
     226           4 :         osRSCRileNameCandidate = CPLResetExtensionSafe(pszFilename, "RSC");
     227           4 :         if (CPLCheckForFile(osRSCRileNameCandidate.data(), nullptr) == TRUE)
     228             :         {
     229           0 :             soRSCRileName = osRSCRileNameCandidate;
     230             :         }
     231             :     }
     232             : 
     233             :     // 1. Create layers from RSC file or create default set of layers from
     234             :     // gdal_data/default.rsc.
     235             : 
     236           5 :     VSILFILE *fpRSC = nullptr;
     237           5 :     if (soRSCRileName.empty())
     238             :     {
     239             : #if defined(USE_ONLY_EMBEDDED_RESOURCE_FILES)
     240             :         const char *pszRSCRileName = nullptr;
     241             : #else
     242           4 :         const char *pszRSCRileName = CPLFindFile("gdal", "default.rsc");
     243             : #endif
     244             : #ifdef EMBED_RESOURCE_FILES
     245             :         if (!pszRSCRileName || EQUAL(pszRSCRileName, "default.rsc"))
     246             :         {
     247             :             static const bool bOnce [[maybe_unused]] = []()
     248             :             {
     249             :                 CPLDebug("SXF", "Using embedded default.rsc");
     250             :                 return true;
     251             :             }();
     252             :             int sxf_default_rsc_size = 0;
     253             :             const unsigned char *sxf_default_rsc =
     254             :                 SXFGetDefaultRSC(&sxf_default_rsc_size);
     255             :             fpRSC = VSIFileFromMemBuffer(
     256             :                 nullptr,
     257             :                 const_cast<GByte *>(
     258             :                     reinterpret_cast<const GByte *>(sxf_default_rsc)),
     259             :                 sxf_default_rsc_size,
     260             :                 /* bTakeOwnership = */ false);
     261             :         }
     262             : #endif
     263           4 :         if (nullptr != pszRSCRileName)
     264             :         {
     265           4 :             soRSCRileName = pszRSCRileName;
     266             :         }
     267             :         else
     268             :         {
     269           0 :             CPLDebug("OGRSXFDataSource", "Default RSC file not found");
     270             :         }
     271             :     }
     272             : 
     273           5 :     if (soRSCRileName.empty()
     274             : #ifdef EMBED_RESOURCE_FILES
     275             :         && !fpRSC
     276             : #endif
     277             :     )
     278             :     {
     279           0 :         CPLError(CE_Warning, CPLE_None, "RSC file for %s not exist",
     280             :                  pszFilename);
     281             :     }
     282             :     else
     283             :     {
     284             : #ifdef EMBED_RESOURCE_FILES
     285             :         if (!fpRSC)
     286             : #endif
     287             :         {
     288           5 :             fpRSC = VSIFOpenL(soRSCRileName, "rb");
     289             :         }
     290           5 :         if (fpRSC == nullptr)
     291             :         {
     292           0 :             CPLError(CE_Warning, CPLE_OpenFailed, "RSC file %s open failed",
     293             :                      soRSCRileName.c_str());
     294             :         }
     295             :         else
     296             :         {
     297           5 :             CPLDebug("OGRSXFDataSource", "RSC Filename: %s",
     298             :                      soRSCRileName.c_str());
     299           5 :             CreateLayers(fpRSC, papszOpenOpts);
     300           5 :             VSIFCloseL(fpRSC);
     301             :         }
     302             :     }
     303             : 
     304           5 :     if (m_apoLayers.empty())  // create default set of layers
     305             :     {
     306           1 :         CreateLayers();
     307             :     }
     308             : 
     309           5 :     FillLayers();
     310             : 
     311           5 :     return TRUE;
     312             : }
     313             : 
     314           5 : OGRErr OGRSXFDataSource::ReadSXFDescription(VSILFILE *fpSXFIn,
     315             :                                             SXFPassport &passport)
     316             : {
     317             :     // int nObjectsRead = 0;
     318             : 
     319           5 :     if (passport.version == 3)
     320             :     {
     321             :         // 78
     322             :         GByte buff[62];
     323           0 :         /* nObjectsRead = */ VSIFReadL(&buff, 62, 1, fpSXFIn);
     324           0 :         char date[3] = {0};
     325             : 
     326             :         // read year
     327           0 :         memcpy(date, buff, 2);
     328           0 :         passport.dtCrateDate.nYear = static_cast<GUInt16>(atoi(date));
     329           0 :         if (passport.dtCrateDate.nYear < 50)
     330           0 :             passport.dtCrateDate.nYear += 2000;
     331             :         else
     332           0 :             passport.dtCrateDate.nYear += 1900;
     333             : 
     334           0 :         memcpy(date, buff + 2, 2);
     335             : 
     336           0 :         passport.dtCrateDate.nMonth = static_cast<GUInt16>(atoi(date));
     337             : 
     338           0 :         memcpy(date, buff + 4, 2);
     339             : 
     340           0 :         passport.dtCrateDate.nDay = static_cast<GUInt16>(atoi(date));
     341             : 
     342           0 :         char szName[26] = {0};
     343           0 :         memcpy(szName, buff + 8, 24);
     344           0 :         szName[sizeof(szName) - 1] = '\0';
     345             :         char *pszRecoded =
     346           0 :             CPLRecode(szName, "CP1251", CPL_ENC_UTF8);  // szName + 2
     347             :         passport.sMapSheet =
     348           0 :             pszRecoded;  // TODO: check the encoding in SXF created in Linux
     349           0 :         CPLFree(pszRecoded);
     350             : 
     351           0 :         memcpy(&passport.nScale, buff + 32, 4);
     352           0 :         CPL_LSBPTR32(&passport.nScale);
     353             : 
     354           0 :         memcpy(szName, buff + 36, 26);
     355           0 :         szName[sizeof(szName) - 1] = '\0';
     356           0 :         pszRecoded = CPLRecode(szName, "CP866", CPL_ENC_UTF8);
     357             :         passport.sMapSheetName =
     358           0 :             pszRecoded;  // TODO: check the encoding in SXF created in Linux
     359           0 :         CPLFree(pszRecoded);
     360             :     }
     361           5 :     else if (passport.version == 4)
     362             :     {
     363             :         // 96
     364             :         GByte buff[80];
     365           5 :         /* nObjectsRead = */ VSIFReadL(&buff, 80, 1, fpSXFIn);
     366           5 :         char date[5] = {0};
     367             : 
     368             :         // read year
     369           5 :         memcpy(date, buff, 4);
     370           5 :         passport.dtCrateDate.nYear = static_cast<GUInt16>(atoi(date));
     371             : 
     372           5 :         memcpy(date, buff + 4, 2);
     373           5 :         memset(date + 2, 0, 3);
     374             : 
     375           5 :         passport.dtCrateDate.nMonth = static_cast<GUInt16>(atoi(date));
     376             : 
     377           5 :         memcpy(date, buff + 6, 2);
     378             : 
     379           5 :         passport.dtCrateDate.nDay = static_cast<GUInt16>(atoi(date));
     380             : 
     381           5 :         char szName[32] = {0};
     382           5 :         memcpy(szName, buff + 12, 32);
     383           5 :         szName[sizeof(szName) - 1] = '\0';
     384             :         char *pszRecoded =
     385           5 :             CPLRecode(szName, "CP1251", CPL_ENC_UTF8);  // szName + 2
     386             :         passport.sMapSheet =
     387           5 :             pszRecoded;  // TODO: check the encoding in SXF created in Linux
     388           5 :         CPLFree(pszRecoded);
     389             : 
     390           5 :         memcpy(&passport.nScale, buff + 44, 4);
     391           5 :         CPL_LSBPTR32(&passport.nScale);
     392             : 
     393           5 :         memcpy(szName, buff + 48, 32);
     394           5 :         szName[sizeof(szName) - 1] = '\0';
     395           5 :         pszRecoded = CPLRecode(szName, "CP1251", CPL_ENC_UTF8);
     396             :         passport.sMapSheetName =
     397           5 :             pszRecoded;  // TODO: check the encoding in SXF created in Linux
     398           5 :         CPLFree(pszRecoded);
     399             :     }
     400             : 
     401           5 :     SetMetadataItem("SHEET", passport.sMapSheet);
     402           5 :     SetMetadataItem("SHEET_NAME", passport.sMapSheetName);
     403           5 :     SetMetadataItem("SHEET_CREATE_DATE",
     404           5 :                     CPLSPrintf("%.2u-%.2u-%.4u", passport.dtCrateDate.nDay,
     405           5 :                                passport.dtCrateDate.nMonth,
     406           5 :                                passport.dtCrateDate.nYear));
     407           5 :     SetMetadataItem("SXF_VERSION", CPLSPrintf("%u", passport.version));
     408           5 :     SetMetadataItem("SCALE", CPLSPrintf("1 : %u", passport.nScale));
     409             : 
     410           5 :     return OGRERR_NONE;
     411             : }
     412             : 
     413           5 : OGRErr OGRSXFDataSource::ReadSXFInformationFlags(VSILFILE *fpSXFIn,
     414             :                                                  SXFPassport &passport)
     415             : {
     416             :     // int nObjectsRead = 0;
     417             :     GByte val[4];
     418           5 :     /* nObjectsRead = */ VSIFReadL(&val, 4, 1, fpSXFIn);
     419             : 
     420           5 :     if (!(CHECK_BIT(val[0], 0) && CHECK_BIT(val[0], 1)))  // xxxxxx11
     421             :     {
     422           0 :         return OGRERR_UNSUPPORTED_OPERATION;
     423             :     }
     424             : 
     425           5 :     if (CHECK_BIT(val[0], 2))  // xxxxx0xx or xxxxx1xx
     426             :     {
     427           5 :         passport.informationFlags.bProjectionDataCompliance = true;
     428             :     }
     429             :     else
     430             :     {
     431           0 :         passport.informationFlags.bProjectionDataCompliance = false;
     432             :     }
     433             : 
     434           5 :     if (CHECK_BIT(val[0], 4))
     435             :     {
     436           0 :         passport.informationFlags.bRealCoordinatesCompliance = true;
     437             :     }
     438             :     else
     439             :     {
     440           5 :         passport.informationFlags.bRealCoordinatesCompliance = false;
     441             :     }
     442             : 
     443           5 :     if (CHECK_BIT(val[0], 6))
     444             :     {
     445           0 :         passport.informationFlags.stCodingType = SXF_SEM_TXT;
     446             :     }
     447             :     else
     448             :     {
     449           5 :         if (CHECK_BIT(val[0], 5))
     450             :         {
     451           0 :             passport.informationFlags.stCodingType = SXF_SEM_HEX;
     452             :         }
     453             :         else
     454             :         {
     455           5 :             passport.informationFlags.stCodingType = SXF_SEM_DEC;
     456             :         }
     457             :     }
     458             : 
     459           5 :     if (CHECK_BIT(val[0], 7))
     460             :     {
     461           0 :         passport.informationFlags.stGenType = SXF_GT_LARGE_SCALE;
     462             :     }
     463             :     else
     464             :     {
     465           5 :         passport.informationFlags.stGenType = SXF_GT_SMALL_SCALE;
     466             :     }
     467             : 
     468             :     // version specific
     469             : 
     470           5 :     if (passport.version == 3)
     471             :     {
     472             :         // degrees are ints * 100 000 000
     473             :         // meters are ints / 10
     474           0 :         passport.informationFlags.stEnc = SXF_ENC_DOS;
     475           0 :         passport.informationFlags.stCoordAcc = SXF_COORD_ACC_DM;
     476           0 :         passport.informationFlags.bSort = false;
     477             :     }
     478           5 :     else if (passport.version == 4)
     479             :     {
     480           5 :         if (val[1] <= SXF_ENC_LAST)
     481             :         {
     482           5 :             passport.informationFlags.stEnc =
     483           5 :                 static_cast<SXFTextEncoding>(val[1]);
     484             :         }
     485             :         else
     486             :         {
     487           0 :             CPLDebug("SXF",
     488             :                      "Invalid passport.informationFlags.stEnc = %d. "
     489             :                      "Defaulting to SXF_ENC_DOS",
     490           0 :                      val[1]);
     491           0 :             passport.informationFlags.stEnc = SXF_ENC_DOS;
     492             :         }
     493             : 
     494           5 :         if (val[2] <= SXF_COORD_ACC_LAST)
     495             :         {
     496           5 :             passport.informationFlags.stCoordAcc =
     497           5 :                 static_cast<SXFCoordinatesAccuracy>(val[2]);
     498             :         }
     499             :         else
     500             :         {
     501           0 :             CPLDebug("SXF",
     502             :                      "Invalid passport.informationFlags.stCoordAcc = %d. "
     503             :                      "Defaulting to SXF_COORD_ACC_UNDEFINED",
     504           0 :                      val[1]);
     505           0 :             passport.informationFlags.stCoordAcc = SXF_COORD_ACC_UNDEFINED;
     506             :         }
     507             : 
     508           5 :         if (CHECK_BIT(val[3], 0))
     509             :         {
     510           0 :             passport.informationFlags.bSort = true;
     511             :         }
     512             :         else
     513             :         {
     514           5 :             passport.informationFlags.bSort = false;
     515             :         }
     516             :     }
     517             : 
     518           5 :     return OGRERR_NONE;
     519             : }
     520             : 
     521           5 : void OGRSXFDataSource::SetVertCS(const long iVCS, SXFPassport &passport,
     522             :                                  const char *const *papszOpenOpts)
     523             : {
     524             :     const char *pszSetVertCS =
     525           5 :         CSLFetchNameValueDef(papszOpenOpts, "SXF_SET_VERTCS",
     526             :                              CPLGetConfigOption("SXF_SET_VERTCS", "NO"));
     527           5 :     if (!CPLTestBool(pszSetVertCS))
     528           5 :         return;
     529             : 
     530           0 :     passport.stMapDescription.pSpatRef->importVertCSFromPanorama(
     531             :         static_cast<int>(iVCS));
     532             : }
     533             : 
     534           5 : OGRErr OGRSXFDataSource::ReadSXFMapDescription(VSILFILE *fpSXFIn,
     535             :                                                SXFPassport &passport,
     536             :                                                const char *const *papszOpenOpts)
     537             : {
     538             :     // int nObjectsRead = 0;
     539           5 :     passport.stMapDescription.Env.MaxX = -100000000;
     540           5 :     passport.stMapDescription.Env.MinX = 100000000;
     541           5 :     passport.stMapDescription.Env.MaxY = -100000000;
     542           5 :     passport.stMapDescription.Env.MinY = 100000000;
     543             : 
     544           5 :     bool bIsX = true;  // passport.informationFlags.bRealCoordinatesCompliance;
     545             :                        // //if real coordinates we need to swap x & y
     546             : 
     547             :     // version specific
     548           5 :     if (passport.version == 3)
     549             :     {
     550             :         short nNoObjClass, nNoSemClass;
     551           0 :         if (VSIFReadL(&nNoObjClass, 2, 1, fpSXFIn) != 1)
     552           0 :             return OGRERR_FAILURE;
     553           0 :         if (VSIFReadL(&nNoSemClass, 2, 1, fpSXFIn) != 1)
     554           0 :             return OGRERR_FAILURE;
     555             :         GByte byMask[8];
     556           0 :         if (VSIFReadL(&byMask, 8, 1, fpSXFIn) != 1)
     557           0 :             return OGRERR_FAILURE;
     558             : 
     559             :         int nCorners[8];
     560             : 
     561             :         // get projected corner coords
     562           0 :         if (VSIFReadL(&nCorners, 32, 1, fpSXFIn) != 1)
     563           0 :             return OGRERR_FAILURE;
     564             : 
     565           0 :         for (int i = 0; i < 8; i++)
     566             :         {
     567           0 :             CPL_LSBPTR32(&nCorners[i]);
     568           0 :             passport.stMapDescription.stProjCoords[i] =
     569           0 :                 double(nCorners[i]) / 10.0;
     570           0 :             if (bIsX)  // X
     571             :             {
     572           0 :                 if (passport.stMapDescription.Env.MaxY <
     573           0 :                     passport.stMapDescription.stProjCoords[i])
     574           0 :                     passport.stMapDescription.Env.MaxY =
     575           0 :                         passport.stMapDescription.stProjCoords[i];
     576           0 :                 if (passport.stMapDescription.Env.MinY >
     577           0 :                     passport.stMapDescription.stProjCoords[i])
     578           0 :                     passport.stMapDescription.Env.MinY =
     579           0 :                         passport.stMapDescription.stProjCoords[i];
     580             :             }
     581             :             else
     582             :             {
     583           0 :                 if (passport.stMapDescription.Env.MaxX <
     584           0 :                     passport.stMapDescription.stProjCoords[i])
     585           0 :                     passport.stMapDescription.Env.MaxX =
     586           0 :                         passport.stMapDescription.stProjCoords[i];
     587           0 :                 if (passport.stMapDescription.Env.MinX >
     588           0 :                     passport.stMapDescription.stProjCoords[i])
     589           0 :                     passport.stMapDescription.Env.MinX =
     590           0 :                         passport.stMapDescription.stProjCoords[i];
     591             :             }
     592           0 :             bIsX = !bIsX;
     593             :         }
     594             :         // get geographic corner coords
     595           0 :         if (VSIFReadL(&nCorners, 32, 1, fpSXFIn) != 1)
     596           0 :             return OGRERR_FAILURE;
     597             : 
     598           0 :         for (int i = 0; i < 8; i++)
     599             :         {
     600           0 :             CPL_LSBPTR32(&nCorners[i]);
     601           0 :             passport.stMapDescription.stGeoCoords[i] =
     602           0 :                 double(nCorners[i]) *
     603             :                 0.00000057295779513082;  // from radians to degree * 100 000 000
     604             :         }
     605             :     }
     606           5 :     else if (passport.version == 4)
     607             :     {
     608           5 :         int nEPSG = 0;
     609           5 :         if (VSIFReadL(&nEPSG, 4, 1, fpSXFIn) != 1)
     610           0 :             return OGRERR_FAILURE;
     611           5 :         CPL_LSBPTR32(&nEPSG);
     612             : 
     613           5 :         if (nEPSG >= MIN_EPSG &&
     614           0 :             nEPSG <= MAX_EPSG)  // TODO: check epsg valid range
     615             :         {
     616           0 :             passport.stMapDescription.pSpatRef = new OGRSpatialReference();
     617           0 :             passport.stMapDescription.pSpatRef->importFromEPSG(nEPSG);
     618             :         }
     619             : 
     620             :         double dfCorners[8];
     621           5 :         if (VSIFReadL(&dfCorners, 64, 1, fpSXFIn) != 1)
     622           0 :             return OGRERR_FAILURE;
     623             : 
     624          45 :         for (int i = 0; i < 8; i++)
     625             :         {
     626          40 :             CPL_LSBPTR64(&dfCorners[i]);
     627          40 :             passport.stMapDescription.stProjCoords[i] = dfCorners[i];
     628          40 :             if (bIsX)  // X
     629             :             {
     630          20 :                 if (passport.stMapDescription.Env.MaxY <
     631          20 :                     passport.stMapDescription.stProjCoords[i])
     632          10 :                     passport.stMapDescription.Env.MaxY =
     633          10 :                         passport.stMapDescription.stProjCoords[i];
     634          20 :                 if (passport.stMapDescription.Env.MinY >
     635          20 :                     passport.stMapDescription.stProjCoords[i])
     636          10 :                     passport.stMapDescription.Env.MinY =
     637          10 :                         passport.stMapDescription.stProjCoords[i];
     638             :             }
     639             :             else
     640             :             {
     641          20 :                 if (passport.stMapDescription.Env.MaxX <
     642          20 :                     passport.stMapDescription.stProjCoords[i])
     643          15 :                     passport.stMapDescription.Env.MaxX =
     644          15 :                         passport.stMapDescription.stProjCoords[i];
     645          20 :                 if (passport.stMapDescription.Env.MinX >
     646          20 :                     passport.stMapDescription.stProjCoords[i])
     647           5 :                     passport.stMapDescription.Env.MinX =
     648           5 :                         passport.stMapDescription.stProjCoords[i];
     649             :             }
     650          40 :             bIsX = !bIsX;
     651             :         }
     652             :         // get geographic corner coords
     653           5 :         if (VSIFReadL(&dfCorners, 64, 1, fpSXFIn) != 1)
     654           0 :             return OGRERR_FAILURE;
     655             : 
     656          45 :         for (int i = 0; i < 8; i++)
     657             :         {
     658          40 :             CPL_LSBPTR64(&dfCorners[i]);
     659          40 :             passport.stMapDescription.stGeoCoords[i] =
     660          40 :                 dfCorners[i] * TO_DEGREES;  // to degree
     661             :         }
     662             :     }
     663             : 
     664           5 :     if (nullptr != passport.stMapDescription.pSpatRef)
     665             :     {
     666           0 :         return OGRERR_NONE;
     667             :     }
     668             : 
     669           5 :     GByte anData[8] = {0};
     670           5 :     if (VSIFReadL(&anData, 8, 1, fpSXFIn) != 1)
     671           0 :         return OGRERR_FAILURE;
     672           5 :     long iEllips = anData[0];
     673           5 :     long iVCS = anData[1];
     674           5 :     long iProjSys = anData[2];
     675             :     /* long iDatum = anData[3]; Unused. */
     676           5 :     double dfProjScale = 1;
     677             : 
     678           5 :     double adfPrjParams[8] = {0};
     679             : 
     680           5 :     if (passport.version == 3)
     681             :     {
     682           0 :         switch (anData[4])
     683             :         {
     684           0 :             case 1:
     685           0 :                 passport.stMapDescription.eUnitInPlan = SXF_COORD_MU_DECIMETRE;
     686           0 :                 break;
     687           0 :             case 2:
     688           0 :                 passport.stMapDescription.eUnitInPlan = SXF_COORD_MU_CENTIMETRE;
     689           0 :                 break;
     690           0 :             case 3:
     691           0 :                 passport.stMapDescription.eUnitInPlan = SXF_COORD_MU_MILLIMETRE;
     692           0 :                 break;
     693           0 :             case 130:
     694           0 :                 passport.stMapDescription.eUnitInPlan = SXF_COORD_MU_RADIAN;
     695           0 :                 break;
     696           0 :             case 129:
     697           0 :                 passport.stMapDescription.eUnitInPlan = SXF_COORD_MU_DEGREE;
     698           0 :                 break;
     699           0 :             default:
     700           0 :                 passport.stMapDescription.eUnitInPlan = SXF_COORD_MU_METRE;
     701           0 :                 break;
     702             :         }
     703             : 
     704           0 :         VSIFSeekL(fpSXFIn, 212, SEEK_SET);
     705             : 
     706             :         struct _buff
     707             :         {
     708             :             GUInt32 nRes;
     709             :             GInt16 anFrame[8];
     710             :             // cppcheck-suppress unusedStructMember
     711             :             GUInt32 nFrameCode;
     712             :         } buff;
     713             : 
     714           0 :         if (VSIFReadL(&buff, 20, 1, fpSXFIn) != 1)
     715           0 :             return OGRERR_FAILURE;
     716           0 :         CPL_LSBPTR32(&buff.nRes);
     717           0 :         CPL_LSBPTR32(&buff.nFrameCode);
     718           0 :         passport.stMapDescription.nResolution = buff.nRes;  // resolution
     719             : 
     720           0 :         for (int i = 0; i < 8; i++)
     721             :         {
     722           0 :             CPL_LSBPTR16(&(buff.anFrame[i]));
     723           0 :             passport.stMapDescription.stFrameCoords[i] = buff.anFrame[i];
     724             :         }
     725             : 
     726             :         int anParams[5];
     727           0 :         if (VSIFReadL(&anParams, 20, 1, fpSXFIn) != 1)
     728           0 :             return OGRERR_FAILURE;
     729           0 :         for (int i = 0; i < 5; i++)
     730             :         {
     731           0 :             CPL_LSBPTR32(&anParams[i]);
     732             :         }
     733             : 
     734           0 :         if (anParams[0] != -1)
     735           0 :             dfProjScale = double(anParams[0]) / 100000000.0;
     736             : 
     737           0 :         if (anParams[2] != -1)
     738           0 :             passport.stMapDescription.dfXOr =
     739           0 :                 double(anParams[2]) / 100000000.0 * TO_DEGREES;
     740             :         else
     741           0 :             passport.stMapDescription.dfXOr = 0;
     742             : 
     743           0 :         if (anParams[3] != -1)
     744           0 :             passport.stMapDescription.dfYOr =
     745           0 :                 double(anParams[2]) / 100000000.0 * TO_DEGREES;
     746             :         else
     747           0 :             passport.stMapDescription.dfYOr = 0;
     748             : 
     749           0 :         passport.stMapDescription.dfFalseNorthing = 0;
     750           0 :         passport.stMapDescription.dfFalseEasting = 0;
     751             : 
     752             :         // adfPrjParams[0] = double(anParams[0]) / 100000000.0; // to radians
     753             :         // adfPrjParams[1] = double(anParams[1]) / 100000000.0;
     754             :         // adfPrjParams[2] = double(anParams[2]) / 100000000.0;
     755             :         // adfPrjParams[3] = double(anParams[3]) / 100000000.0;
     756           0 :         adfPrjParams[4] = dfProjScale;  //?
     757             :         // adfPrjParams[5] = 0;//?
     758             :         // adfPrjParams[6] = 0;//?
     759             :         // adfPrjParams[7] = 0;// importFromPanorama calc it by itself
     760             :     }
     761           5 :     else if (passport.version == 4)
     762             :     {
     763           5 :         switch (anData[4])
     764             :         {
     765           0 :             case 64:
     766           0 :                 passport.stMapDescription.eUnitInPlan = SXF_COORD_MU_RADIAN;
     767           0 :                 break;
     768           0 :             case 65:
     769           0 :                 passport.stMapDescription.eUnitInPlan = SXF_COORD_MU_DEGREE;
     770           0 :                 break;
     771           5 :             default:
     772           5 :                 passport.stMapDescription.eUnitInPlan = SXF_COORD_MU_METRE;
     773           5 :                 break;
     774             :         }
     775             : 
     776           5 :         VSIFSeekL(fpSXFIn, 312, SEEK_SET);
     777             :         GUInt32 buff[10];
     778           5 :         if (VSIFReadL(&buff, 40, 1, fpSXFIn) != 1)
     779           0 :             return OGRERR_FAILURE;
     780          55 :         for (int i = 0; i < 10; i++)
     781             :         {
     782          50 :             CPL_LSBPTR32(&buff[i]);
     783             :         }
     784             : 
     785           5 :         passport.stMapDescription.nResolution = buff[0];  // resolution
     786          45 :         for (int i = 0; i < 8; i++)
     787          40 :             passport.stMapDescription.stFrameCoords[i] = buff[1 + i];
     788             : 
     789           5 :         double adfParams[6] = {};
     790           5 :         if (VSIFReadL(&adfParams, 48, 1, fpSXFIn) != 1)
     791           0 :             return OGRERR_FAILURE;
     792          35 :         for (int i = 0; i < 6; i++)
     793             :         {
     794          30 :             CPL_LSBPTR64(&adfParams[i]);
     795             :         }
     796             : 
     797           5 :         if (adfParams[1] != -1)
     798           5 :             dfProjScale = adfParams[1];
     799           5 :         passport.stMapDescription.dfXOr = adfParams[2] * TO_DEGREES;
     800           5 :         passport.stMapDescription.dfYOr = adfParams[3] * TO_DEGREES;
     801           5 :         passport.stMapDescription.dfFalseNorthing = adfParams[4];
     802           5 :         passport.stMapDescription.dfFalseEasting = adfParams[5];
     803             : 
     804             :         // adfPrjParams[0] = adfParams[0]; // to radians
     805             :         // adfPrjParams[1] = adfParams[1];
     806             :         // adfPrjParams[2] = adfParams[2];
     807             :         // adfPrjParams[3] = adfParams[3];
     808           5 :         adfPrjParams[4] = dfProjScale;  //?
     809             :         // adfPrjParams[5] = adfParams[4];
     810             :         // adfPrjParams[6] = adfParams[5];
     811             :         // adfPrjParams[7] = 0;// importFromPanorama calc it by itself
     812             :     }
     813             : 
     814           5 :     passport.stMapDescription.dfScale = passport.nScale;
     815             : 
     816           5 :     if (passport.stMapDescription.nResolution == 0)
     817             :     {
     818           0 :         return OGRERR_FAILURE;
     819             :     }
     820           5 :     const double dfCoeff = double(passport.stMapDescription.dfScale) /
     821           5 :                            passport.stMapDescription.nResolution;
     822           5 :     passport.stMapDescription.bIsRealCoordinates =
     823           5 :         passport.informationFlags.bRealCoordinatesCompliance;
     824           5 :     passport.stMapDescription.stCoordAcc = passport.informationFlags.stCoordAcc;
     825             : 
     826           5 :     if (!passport.stMapDescription.bIsRealCoordinates)
     827             :     {
     828           5 :         if (passport.stMapDescription.stFrameCoords[0] == 0 &&
     829           5 :             passport.stMapDescription.stFrameCoords[1] == 0 &&
     830           5 :             passport.stMapDescription.stFrameCoords[2] == 0 &&
     831           5 :             passport.stMapDescription.stFrameCoords[3] == 0 &&
     832           5 :             passport.stMapDescription.stFrameCoords[4] == 0 &&
     833           5 :             passport.stMapDescription.stFrameCoords[5] == 0 &&
     834           5 :             passport.stMapDescription.stFrameCoords[6] == 0 &&
     835           5 :             passport.stMapDescription.stFrameCoords[7] == 0)
     836             :         {
     837           5 :             passport.stMapDescription.bIsRealCoordinates = true;
     838             :         }
     839             :         else
     840             :         {
     841             :             // origin
     842           0 :             passport.stMapDescription.dfXOr =
     843           0 :                 passport.stMapDescription.stProjCoords[1] -
     844           0 :                 passport.stMapDescription.stFrameCoords[1] * dfCoeff;
     845           0 :             passport.stMapDescription.dfYOr =
     846           0 :                 passport.stMapDescription.stProjCoords[0] -
     847           0 :                 passport.stMapDescription.stFrameCoords[0] * dfCoeff;
     848             :         }
     849             :     }
     850             : 
     851             :     // normalize some coordintatessystems
     852           5 :     if ((iEllips == 1 || iEllips == 0) &&
     853             :         iProjSys == 1)  // Pulkovo 1942 / Gauss-Kruger
     854             :     {
     855           5 :         double dfCenterLongEnv =
     856           5 :             passport.stMapDescription.stGeoCoords[1] +
     857           5 :             fabs(passport.stMapDescription.stGeoCoords[5] -
     858           5 :                  passport.stMapDescription.stGeoCoords[1]) /
     859             :                 2;
     860             : 
     861           5 :         int nZoneEnv = (int)((dfCenterLongEnv + 3.0) / 6.0 + 0.5);
     862             : 
     863           5 :         if (nZoneEnv > 1 && nZoneEnv < 33)
     864             :         {
     865           5 :             int nEPSG = 28400 + nZoneEnv;
     866           5 :             passport.stMapDescription.pSpatRef = new OGRSpatialReference();
     867           5 :             passport.stMapDescription.pSpatRef->SetAxisMappingStrategy(
     868             :                 OAMS_TRADITIONAL_GIS_ORDER);
     869             :             OGRErr eErr =
     870           5 :                 passport.stMapDescription.pSpatRef->importFromEPSG(nEPSG);
     871           5 :             SetVertCS(iVCS, passport, papszOpenOpts);
     872           5 :             return eErr;
     873             :         }
     874             :         else
     875             :         {
     876           0 :             adfPrjParams[7] = nZoneEnv;
     877             : 
     878           0 :             if (adfPrjParams[5] == 0)  // False Easting
     879             :             {
     880           0 :                 if (passport.stMapDescription.Env.MaxX < 500000)
     881           0 :                     adfPrjParams[5] = 500000;
     882             :                 else
     883             :                 {
     884           0 :                     if (nZoneEnv >= -60 && nZoneEnv <= 60)
     885           0 :                         adfPrjParams[5] = nZoneEnv * 1000000 + 500000;
     886             :                     else
     887             :                     {
     888           0 :                         CPLError(CE_Failure, CPLE_AppDefined,
     889             :                                  "Wrong nZoneEnv = %d value", nZoneEnv);
     890           0 :                         return OGRERR_FAILURE;
     891             :                     }
     892             :                 }
     893             :             }
     894           0 :         }
     895             :     }
     896           0 :     else if (iEllips == 9 && iProjSys == 17)  // WGS84 / UTM
     897             :     {
     898           0 :         double dfCenterLongEnv =
     899           0 :             passport.stMapDescription.stGeoCoords[1] +
     900           0 :             fabs(passport.stMapDescription.stGeoCoords[5] -
     901           0 :                  passport.stMapDescription.stGeoCoords[1]) /
     902             :                 2;
     903           0 :         int nZoneEnv = (int)(30 + (dfCenterLongEnv + 3.0) / 6.0 + 0.5);
     904           0 :         bool bNorth = passport.stMapDescription.stGeoCoords[6] +
     905           0 :                           (passport.stMapDescription.stGeoCoords[2] -
     906           0 :                            passport.stMapDescription.stGeoCoords[6]) /
     907             :                               2 <
     908             :                       0;
     909           0 :         int nEPSG = 0;
     910           0 :         if (bNorth)
     911             :         {
     912           0 :             nEPSG = 32600 + nZoneEnv;
     913             :         }
     914             :         else
     915             :         {
     916           0 :             nEPSG = 32700 + nZoneEnv;
     917             :         }
     918           0 :         passport.stMapDescription.pSpatRef = new OGRSpatialReference();
     919           0 :         passport.stMapDescription.pSpatRef->SetAxisMappingStrategy(
     920             :             OAMS_TRADITIONAL_GIS_ORDER);
     921           0 :         OGRErr eErr = passport.stMapDescription.pSpatRef->importFromEPSG(nEPSG);
     922           0 :         SetVertCS(iVCS, passport, papszOpenOpts);
     923           0 :         return eErr;
     924             :     }
     925           0 :     else if (iEllips == 45 && iProjSys == 35)  // Mercator 3857 on sphere wgs84
     926             :     {
     927           0 :         passport.stMapDescription.pSpatRef = new OGRSpatialReference(
     928             :             "PROJCS[\"WGS 84 / Pseudo-Mercator\",GEOGCS[\"WGS "
     929             :             "84\",DATUM[\"WGS_1984\",SPHEROID[\"WGS "
     930             :             "84\",6378137,298.257223563,AUTHORITY[\"EPSG\",\"7030\"]],"
     931             :             "AUTHORITY[\"EPSG\",\"6326\"]],PRIMEM[\"Greenwich\",0,AUTHORITY["
     932             :             "\"EPSG\",\"8901\"]],UNIT[\"degree\",0.0174532925199433,AUTHORITY["
     933             :             "\"EPSG\",\"9122\"]],AUTHORITY[\"EPSG\",\"4326\"]],PROJECTION["
     934             :             "\"Mercator_1SP\"],PARAMETER[\"central_meridian\",0],PARAMETER["
     935             :             "\"scale_factor\",1],PARAMETER[\"false_easting\",0],PARAMETER["
     936             :             "\"false_northing\",0],UNIT[\"metre\",1,AUTHORITY[\"EPSG\","
     937             :             "\"9001\"]],AXIS[\"X\",EAST],AXIS[\"Y\",NORTH],EXTENSION[\"PROJ4\","
     938             :             "\"+proj=merc +a=6378137 +b=6378137 +lat_ts=0.0 +lon_0=0.0 "
     939             :             "+x_0=0.0 +y_0=0 +k=1.0 +units=m +nadgrids=@null +wktext  "
     940           0 :             "+no_defs\"],AUTHORITY[\"EPSG\",\"3857\"]]");
     941           0 :         passport.stMapDescription.pSpatRef->SetAxisMappingStrategy(
     942             :             OAMS_TRADITIONAL_GIS_ORDER);
     943           0 :         OGRErr eErr =
     944             :             OGRERR_NONE;  // passport.stMapDescription.pSpatRef->importFromEPSG(3857);
     945           0 :         SetVertCS(iVCS, passport, papszOpenOpts);
     946           0 :         return eErr;
     947             :     }
     948           0 :     else if (iEllips == 9 && iProjSys == 35)  // Mercator 3395 on ellips wgs84
     949             :     {
     950           0 :         passport.stMapDescription.pSpatRef = new OGRSpatialReference();
     951           0 :         passport.stMapDescription.pSpatRef->SetAxisMappingStrategy(
     952             :             OAMS_TRADITIONAL_GIS_ORDER);
     953           0 :         OGRErr eErr = passport.stMapDescription.pSpatRef->importFromEPSG(3395);
     954           0 :         SetVertCS(iVCS, passport, papszOpenOpts);
     955           0 :         return eErr;
     956             :     }
     957           0 :     else if (iEllips == 9 && iProjSys == 34)  // Miller 54003 on sphere wgs84
     958             :     {
     959           0 :         passport.stMapDescription.pSpatRef = new OGRSpatialReference(
     960             :             "PROJCS[\"World_Miller_Cylindrical\",GEOGCS[\"GCS_GLOBE\", "
     961             :             "DATUM[\"GLOBE\", SPHEROID[\"GLOBE\", 6367444.6571, "
     962             :             "0.0]],PRIMEM[\"Greenwich\",0],UNIT[\"Degree\",0."
     963             :             "017453292519943295]],PROJECTION[\"Miller_Cylindrical\"],PARAMETER["
     964             :             "\"False_Easting\",0],PARAMETER[\"False_Northing\",0],PARAMETER["
     965             :             "\"Central_Meridian\",0],UNIT[\"Meter\",1],AUTHORITY[\"EPSG\","
     966           0 :             "\"54003\"]]");
     967           0 :         passport.stMapDescription.pSpatRef->SetAxisMappingStrategy(
     968             :             OAMS_TRADITIONAL_GIS_ORDER);
     969           0 :         OGRErr eErr =
     970             :             OGRERR_NONE;  // passport.stMapDescription.pSpatRef->importFromEPSG(3395);
     971             :         // OGRErr eErr =
     972             :         // passport.stMapDescription.pSpatRef->importFromEPSG(54003);
     973             : 
     974           0 :         SetVertCS(iVCS, passport, papszOpenOpts);
     975           0 :         return eErr;
     976             :     }
     977           0 :     else if (iEllips == 9 && iProjSys == 33 &&
     978           0 :              passport.stMapDescription.eUnitInPlan == SXF_COORD_MU_DEGREE)
     979             :     {
     980           0 :         passport.stMapDescription.pSpatRef =
     981           0 :             new OGRSpatialReference(SRS_WKT_WGS84_LAT_LONG);
     982           0 :         passport.stMapDescription.pSpatRef->SetAxisMappingStrategy(
     983             :             OAMS_TRADITIONAL_GIS_ORDER);
     984           0 :         OGRErr eErr = OGRERR_NONE;
     985           0 :         SetVertCS(iVCS, passport, papszOpenOpts);
     986           0 :         return eErr;
     987             :     }
     988             : 
     989             :     // TODO: Need to normalize more SRS:
     990             :     // PAN_PROJ_WAG1
     991             :     // PAN_PROJ_MERCAT
     992             :     // PAN_PROJ_PS
     993             :     // PAN_PROJ_POLYC
     994             :     // PAN_PROJ_EC
     995             :     // PAN_PROJ_LCC
     996             :     // PAN_PROJ_STEREO
     997             :     // PAN_PROJ_AE
     998             :     // PAN_PROJ_GNOMON
     999             :     // PAN_PROJ_MOLL
    1000             :     // PAN_PROJ_LAEA
    1001             :     // PAN_PROJ_EQC
    1002             :     // PAN_PROJ_CEA
    1003             :     // PAN_PROJ_IMWP
    1004             :     //
    1005             : 
    1006           0 :     passport.stMapDescription.pSpatRef = new OGRSpatialReference();
    1007           0 :     passport.stMapDescription.pSpatRef->SetAxisMappingStrategy(
    1008             :         OAMS_TRADITIONAL_GIS_ORDER);
    1009           0 :     OGRErr eErr = passport.stMapDescription.pSpatRef->importFromPanorama(
    1010           0 :         anData[2], anData[3], anData[0], adfPrjParams);
    1011           0 :     SetVertCS(iVCS, passport, papszOpenOpts);
    1012           0 :     return eErr;
    1013             : }
    1014             : 
    1015           5 : void OGRSXFDataSource::FillLayers()
    1016             : {
    1017           5 :     CPLDebug("SXF", "Create layers");
    1018             : 
    1019             :     // 2. Read all records (only classify code and offset) and add this to
    1020             :     // correspondence layer
    1021           5 :     int nObjectsRead = 0;
    1022           5 :     vsi_l_offset nOffset = 0;
    1023             : 
    1024             :     // get record count
    1025           5 :     GUInt32 nRecordCountMax = 0;
    1026           5 :     if (oSXFPassport.version == 3)
    1027             :     {
    1028           0 :         VSIFSeekL(fpSXF, 288, SEEK_SET);
    1029           0 :         nObjectsRead =
    1030           0 :             static_cast<int>(VSIFReadL(&nRecordCountMax, 4, 1, fpSXF));
    1031           0 :         nOffset = 300;
    1032           0 :         CPL_LSBPTR32(&nRecordCountMax);
    1033             :     }
    1034           5 :     else if (oSXFPassport.version == 4)
    1035             :     {
    1036           5 :         VSIFSeekL(fpSXF, 440, SEEK_SET);
    1037           5 :         nObjectsRead =
    1038           5 :             static_cast<int>(VSIFReadL(&nRecordCountMax, 4, 1, fpSXF));
    1039           5 :         nOffset = 452;
    1040           5 :         CPL_LSBPTR32(&nRecordCountMax);
    1041             :     }
    1042             :     /* else nOffset and nObjectsRead will be 0 */
    1043             : 
    1044           5 :     if (nObjectsRead != 1)
    1045             :     {
    1046           0 :         CPLError(CE_Failure, CPLE_FileIO, "Get record count failed");
    1047           0 :         return;
    1048             :     }
    1049             : 
    1050           5 :     VSIFSeekL(fpSXF, nOffset, SEEK_SET);
    1051             : 
    1052         395 :     for (GUInt32 nFID = 0; nFID < nRecordCountMax; nFID++)
    1053             :     {
    1054             :         GInt32 buff[6];
    1055         390 :         nObjectsRead = static_cast<int>(VSIFReadL(&buff, 24, 1, fpSXF));
    1056        2730 :         for (int i = 0; i < 6; i++)
    1057             :         {
    1058        2340 :             CPL_LSBPTR32(&buff[i]);
    1059             :         }
    1060             : 
    1061         390 :         if (nObjectsRead != 1 || buff[0] != IDSXFOBJ)
    1062             :         {
    1063           0 :             CPLError(CE_Failure, CPLE_FileIO, "Read record %d failed", nFID);
    1064           0 :             return;
    1065             :         }
    1066             : 
    1067         390 :         bool bHasSemantic = CHECK_BIT(buff[5], 9);
    1068         390 :         if (bHasSemantic)  // check has attributes
    1069             :         {
    1070             :             // we have already 24 byte read
    1071         250 :             vsi_l_offset nOffsetSemantic = 8 + buff[2];
    1072         250 :             VSIFSeekL(fpSXF, nOffsetSemantic, SEEK_CUR);
    1073             :         }
    1074             : 
    1075         390 :         int nSemanticSize = buff[1] - 32 - buff[2];
    1076         390 :         if (nSemanticSize < 0)
    1077             :         {
    1078           0 :             CPLError(CE_Failure, CPLE_AppDefined, "Invalid value");
    1079           0 :             break;
    1080             :         }
    1081             : 
    1082        4643 :         for (const auto &poLayer : m_apoLayers)
    1083             :         {
    1084        4643 :             if (poLayer->AddRecord(nFID, buff[3], nOffset, bHasSemantic,
    1085        4643 :                                    nSemanticSize) == TRUE)
    1086             :             {
    1087         390 :                 break;
    1088             :             }
    1089             :         }
    1090         390 :         nOffset += buff[1];
    1091         390 :         VSIFSeekL(fpSXF, nOffset, SEEK_SET);
    1092             :     }
    1093             :     // 3. delete empty layers
    1094          87 :     for (size_t i = 0; i < m_apoLayers.size(); /* increment in loop */)
    1095             :     {
    1096          82 :         OGRSXFLayer *pOGRSXFLayer = m_apoLayers[i].get();
    1097          82 :         if (pOGRSXFLayer->GetFeatureCount() == 0)
    1098             :         {
    1099          44 :             m_apoLayers.erase(m_apoLayers.begin() + i);
    1100             :         }
    1101             :         else
    1102             :         {
    1103          38 :             pOGRSXFLayer->ResetReading();
    1104          38 :             ++i;
    1105             :         }
    1106             :     }
    1107             : }
    1108             : 
    1109        2140 : OGRSXFLayer *OGRSXFDataSource::GetLayerById(GByte nID)
    1110             : {
    1111       18820 :     for (const auto &poLayer : m_apoLayers)
    1112             :     {
    1113       18820 :         if (poLayer->GetId() == nID)
    1114             :         {
    1115        2140 :             return poLayer.get();
    1116             :         }
    1117             :     }
    1118           0 :     return nullptr;
    1119             : }
    1120             : 
    1121           1 : void OGRSXFDataSource::CreateLayers()
    1122             : {
    1123             :     // default layers set
    1124           1 :     m_apoLayers.emplace_back(std::make_unique<OGRSXFLayer>(
    1125           2 :         fpSXF, &hIOMutex, static_cast<GByte>(0), CPLString("SYSTEM"),
    1126           2 :         oSXFPassport.version, oSXFPassport.stMapDescription));
    1127           1 :     auto pLayer = m_apoLayers.back().get();
    1128             : 
    1129             :     // default codes
    1130          15 :     for (unsigned int i = 1000000001; i < 1000000015; i++)
    1131             :     {
    1132          14 :         pLayer->AddClassifyCode(i);
    1133             :     }
    1134           1 :     pLayer->AddClassifyCode(91000000);
    1135             : 
    1136           1 :     m_apoLayers.emplace_back(std::make_unique<OGRSXFLayer>(
    1137           2 :         fpSXF, &hIOMutex, static_cast<GByte>(255), CPLString("Not_Classified"),
    1138           2 :         oSXFPassport.version, oSXFPassport.stMapDescription));
    1139           1 : }
    1140             : 
    1141           5 : void OGRSXFDataSource::CreateLayers(VSILFILE *fpRSC,
    1142             :                                     const char *const *papszOpenOpts)
    1143             : {
    1144             : 
    1145             :     RSCHeader stRSCFileHeader;
    1146             :     int nObjectsRead = static_cast<int>(
    1147           5 :         VSIFReadL(&stRSCFileHeader, sizeof(stRSCFileHeader), 1, fpRSC));
    1148             : 
    1149           5 :     if (nObjectsRead != 1)
    1150             :     {
    1151           1 :         CPLError(CE_Warning, CPLE_None, "RSC head read failed");
    1152           1 :         return;
    1153             :     }
    1154             : 
    1155           4 :     CPL_LSBPTR32(&(stRSCFileHeader.nFileLength));
    1156           4 :     CPL_LSBPTR32(&(stRSCFileHeader.nVersion));
    1157           4 :     CPL_LSBPTR32(&(stRSCFileHeader.nEncoding));
    1158           4 :     CPL_LSBPTR32(&(stRSCFileHeader.nFileState));
    1159           4 :     CPL_LSBPTR32(&(stRSCFileHeader.nFileModState));
    1160           4 :     CPL_LSBPTR32(&(stRSCFileHeader.nLang));
    1161           4 :     CPL_LSBPTR32(&(stRSCFileHeader.nNextID));
    1162           4 :     CPL_LSBPTR32(&(stRSCFileHeader.nScale));
    1163             : 
    1164             : #define SWAP_SECTION(x)                                                        \
    1165             :     CPL_LSBPTR32(&(x.nOffset));                                                \
    1166             :     CPL_LSBPTR32(&(x.nLength));                                                \
    1167             :     CPL_LSBPTR32(&(x.nRecordCount));
    1168             : 
    1169           4 :     SWAP_SECTION(stRSCFileHeader.Objects);
    1170           4 :     SWAP_SECTION(stRSCFileHeader.Semantic);
    1171           4 :     SWAP_SECTION(stRSCFileHeader.ClassifySemantic);
    1172           4 :     SWAP_SECTION(stRSCFileHeader.Defaults);
    1173           4 :     SWAP_SECTION(stRSCFileHeader.Semantics);
    1174           4 :     SWAP_SECTION(stRSCFileHeader.Layers);
    1175           4 :     SWAP_SECTION(stRSCFileHeader.Limits);
    1176           4 :     SWAP_SECTION(stRSCFileHeader.Parameters);
    1177           4 :     SWAP_SECTION(stRSCFileHeader.Print);
    1178           4 :     SWAP_SECTION(stRSCFileHeader.Palettes);
    1179           4 :     SWAP_SECTION(stRSCFileHeader.Fonts);
    1180           4 :     SWAP_SECTION(stRSCFileHeader.Libs);
    1181           4 :     SWAP_SECTION(stRSCFileHeader.ImageParams);
    1182           4 :     SWAP_SECTION(stRSCFileHeader.Tables);
    1183           4 :     CPL_LSBPTR32(&(stRSCFileHeader.nFontEnc));
    1184           4 :     CPL_LSBPTR32(&(stRSCFileHeader.nColorsInPalette));
    1185             : 
    1186             :     GByte szLayersID[4];
    1187             : 
    1188             :     struct _layer
    1189             :     {
    1190             :         GUInt32 nLength;
    1191             :         char szName[32];
    1192             :         char szShortName[16];
    1193             :         GByte nNo;
    1194             :         // cppcheck-suppress unusedStructMember
    1195             :         GByte nPos;
    1196             :         // cppcheck-suppress unusedStructMember
    1197             :         GUInt16 nSemanticCount;
    1198             :     };
    1199             : 
    1200           4 :     VSIFSeekL(fpRSC, stRSCFileHeader.Layers.nOffset - sizeof(szLayersID),
    1201             :               SEEK_SET);
    1202           4 :     VSIFReadL(&szLayersID, sizeof(szLayersID), 1, fpRSC);
    1203           4 :     vsi_l_offset nOffset = stRSCFileHeader.Layers.nOffset;
    1204             :     _layer LAYER;
    1205             : 
    1206          80 :     for (GUInt32 i = 0; i < stRSCFileHeader.Layers.nRecordCount; ++i)
    1207             :     {
    1208          76 :         VSIFReadL(&LAYER, sizeof(LAYER), 1, fpRSC);
    1209          76 :         CPL_LSBPTR32(&(LAYER.nLength));
    1210          76 :         CPL_LSBPTR16(&(LAYER.nSemanticCount));
    1211          76 :         bool bLayerFullName = CPLTestBool(CSLFetchNameValueDef(
    1212             :             papszOpenOpts, "SXF_LAYER_FULLNAME",
    1213             :             CPLGetConfigOption("SXF_LAYER_FULLNAME", "NO")));
    1214          76 :         char *pszRecoded = nullptr;
    1215          76 :         if (bLayerFullName)
    1216             :         {
    1217          19 :             if (LAYER.szName[0] == 0)
    1218           0 :                 pszRecoded = CPLStrdup("Unnamed");
    1219          19 :             else if (stRSCFileHeader.nFontEnc == 125)
    1220           0 :                 pszRecoded = CPLRecode(LAYER.szName, "KOI8-R", CPL_ENC_UTF8);
    1221          19 :             else if (stRSCFileHeader.nFontEnc == 126)
    1222          19 :                 pszRecoded = CPLRecode(LAYER.szName, "CP1251", CPL_ENC_UTF8);
    1223             :             else
    1224           0 :                 pszRecoded = CPLStrdup(LAYER.szName);
    1225             : 
    1226          19 :             m_apoLayers.emplace_back(std::make_unique<OGRSXFLayer>(
    1227          38 :                 fpSXF, &hIOMutex, LAYER.nNo, CPLString(pszRecoded),
    1228          38 :                 oSXFPassport.version, oSXFPassport.stMapDescription));
    1229             :         }
    1230             :         else
    1231             :         {
    1232          57 :             if (LAYER.szShortName[0] == 0)
    1233           0 :                 pszRecoded = CPLStrdup("Unnamed");
    1234          57 :             else if (stRSCFileHeader.nFontEnc == 125)
    1235             :                 pszRecoded =
    1236           0 :                     CPLRecode(LAYER.szShortName, "KOI8-R", CPL_ENC_UTF8);
    1237          57 :             else if (stRSCFileHeader.nFontEnc == 126)
    1238             :                 pszRecoded =
    1239          57 :                     CPLRecode(LAYER.szShortName, "CP1251", CPL_ENC_UTF8);
    1240             :             else
    1241           0 :                 pszRecoded = CPLStrdup(LAYER.szShortName);
    1242             : 
    1243          57 :             m_apoLayers.emplace_back(std::make_unique<OGRSXFLayer>(
    1244         114 :                 fpSXF, &hIOMutex, LAYER.nNo, CPLString(pszRecoded),
    1245         114 :                 oSXFPassport.version, oSXFPassport.stMapDescription));
    1246             :         }
    1247          76 :         CPLFree(pszRecoded);
    1248             : 
    1249          76 :         nOffset += LAYER.nLength;
    1250          76 :         VSIFSeekL(fpRSC, nOffset, SEEK_SET);
    1251             :     }
    1252             : 
    1253           4 :     m_apoLayers.emplace_back(std::make_unique<OGRSXFLayer>(
    1254           8 :         fpSXF, &hIOMutex, static_cast<GByte>(255), CPLString("Not_Classified"),
    1255           8 :         oSXFPassport.version, oSXFPassport.stMapDescription));
    1256             : 
    1257             :     char szObjectsID[4];
    1258             : 
    1259             :     struct _object
    1260             :     {
    1261             :         unsigned nLength;
    1262             :         unsigned nClassifyCode;
    1263             :         // cppcheck-suppress unusedStructMember
    1264             :         unsigned nObjectNumber;
    1265             :         // cppcheck-suppress unusedStructMember
    1266             :         unsigned nObjectCode;
    1267             :         char szShortName[32];
    1268             :         char szName[32];
    1269             :         // cppcheck-suppress unusedStructMember
    1270             :         char szGeomType;
    1271             :         char szLayernNo;
    1272             :         // cppcheck-suppress unusedStructMember
    1273             :         char szUnimportantSeg[14];
    1274             :     };
    1275             : 
    1276           4 :     VSIFSeekL(fpRSC, stRSCFileHeader.Objects.nOffset - sizeof(szObjectsID),
    1277             :               SEEK_SET);
    1278           4 :     VSIFReadL(&szObjectsID, sizeof(szObjectsID), 1, fpRSC);
    1279           4 :     nOffset = stRSCFileHeader.Objects.nOffset;
    1280             :     _object OBJECT;
    1281             : 
    1282        2144 :     for (GUInt32 i = 0; i < stRSCFileHeader.Objects.nRecordCount; ++i)
    1283             :     {
    1284        2140 :         VSIFReadL(&OBJECT, sizeof(_object), 1, fpRSC);
    1285        2140 :         CPL_LSBPTR32(&(OBJECT.nLength));
    1286        2140 :         CPL_LSBPTR32(&(OBJECT.nClassifyCode));
    1287        2140 :         CPL_LSBPTR32(&(OBJECT.nObjectNumber));
    1288        2140 :         CPL_LSBPTR32(&(OBJECT.nObjectCode));
    1289             : 
    1290        2140 :         OGRSXFLayer *pLayer = GetLayerById(OBJECT.szLayernNo);
    1291        2140 :         if (nullptr != pLayer)
    1292             :         {
    1293        2140 :             char *pszRecoded = nullptr;
    1294        2140 :             if (OBJECT.szName[0] == 0)
    1295           0 :                 pszRecoded = CPLStrdup("Unnamed");
    1296        2140 :             else if (stRSCFileHeader.nFontEnc == 125)
    1297           0 :                 pszRecoded = CPLRecode(OBJECT.szName, "KOI8-R", CPL_ENC_UTF8);
    1298        2140 :             else if (stRSCFileHeader.nFontEnc == 126)
    1299        2140 :                 pszRecoded = CPLRecode(OBJECT.szName, "CP1251", CPL_ENC_UTF8);
    1300             :             else
    1301             :                 pszRecoded =
    1302           0 :                     CPLStrdup(OBJECT.szName);  // already in  CPL_ENC_UTF8
    1303             : 
    1304        2140 :             pLayer->AddClassifyCode(OBJECT.nClassifyCode, pszRecoded);
    1305             :             // printf("%d;%s\n", OBJECT.nClassifyCode, OBJECT.szName);
    1306        2140 :             CPLFree(pszRecoded);
    1307             :         }
    1308             : 
    1309        2140 :         nOffset += OBJECT.nLength;
    1310        2140 :         VSIFSeekL(fpRSC, nOffset, SEEK_SET);
    1311             :     }
    1312             : }

Generated by: LCOV version 1.14