LCOV - code coverage report
Current view: top level - frmts/ers - ersdataset.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 555 646 85.9 %
Date: 2025-01-18 12:42:00 Functions: 30 32 93.8 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  ERMapper .ers Driver
       4             :  * Purpose:  Implementation of .ers driver.
       5             :  * Author:   Frank Warmerdam, warmerdam@pobox.com
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2007, Frank Warmerdam <warmerdam@pobox.com>
       9             :  * Copyright (c) 2008-2013, Even Rouault <even dot rouault at spatialys.com>
      10             :  *
      11             :  * SPDX-License-Identifier: MIT
      12             :  ****************************************************************************/
      13             : 
      14             : #include "cpl_string.h"
      15             : #include "ershdrnode.h"
      16             : #include "gdal_frmts.h"
      17             : #include "gdal_proxy.h"
      18             : #include "ogr_spatialref.h"
      19             : #include "rawdataset.h"
      20             : 
      21             : #include <limits>
      22             : 
      23             : /************************************************************************/
      24             : /* ==================================================================== */
      25             : /*                              ERSDataset                              */
      26             : /* ==================================================================== */
      27             : /************************************************************************/
      28             : 
      29             : class ERSRasterBand;
      30             : 
      31             : class ERSDataset final : public RawDataset
      32             : {
      33             :     friend class ERSRasterBand;
      34             : 
      35             :     VSILFILE *fpImage;  // Image data file.
      36             :     GDALDataset *poDepFile;
      37             : 
      38             :     int bGotTransform;
      39             :     double adfGeoTransform[6];
      40             :     OGRSpatialReference m_oSRS{};
      41             : 
      42             :     CPLString osRawFilename;
      43             : 
      44             :     int bHDRDirty;
      45             :     ERSHdrNode *poHeader;
      46             : 
      47             :     const char *Find(const char *, const char *);
      48             : 
      49             :     int nGCPCount;
      50             :     GDAL_GCP *pasGCPList;
      51             :     OGRSpatialReference m_oGCPSRS{};
      52             : 
      53             :     void ReadGCPs();
      54             : 
      55             :     int bHasNoDataValue;
      56             :     double dfNoDataValue;
      57             : 
      58             :     CPLString osProj, osProjForced;
      59             :     CPLString osDatum, osDatumForced;
      60             :     CPLString osUnits, osUnitsForced;
      61             :     void WriteProjectionInfo(const char *pszProj, const char *pszDatum,
      62             :                              const char *pszUnits);
      63             : 
      64             :     CPLStringList oERSMetadataList;
      65             : 
      66             :   protected:
      67             :     int CloseDependentDatasets() override;
      68             : 
      69             :     CPLErr Close() override;
      70             : 
      71             :   public:
      72             :     ERSDataset();
      73             :     ~ERSDataset() override;
      74             : 
      75             :     CPLErr FlushCache(bool bAtClosing) override;
      76             :     CPLErr GetGeoTransform(double *padfTransform) override;
      77             :     CPLErr SetGeoTransform(double *padfTransform) override;
      78             :     const OGRSpatialReference *GetSpatialRef() const override;
      79             :     CPLErr SetSpatialRef(const OGRSpatialReference *poSRS) override;
      80             :     char **GetFileList(void) override;
      81             : 
      82             :     int GetGCPCount() override;
      83             :     const OGRSpatialReference *GetGCPSpatialRef() const override;
      84             :     const GDAL_GCP *GetGCPs() override;
      85             :     CPLErr SetGCPs(int nGCPCountIn, const GDAL_GCP *pasGCPListIn,
      86             :                    const OGRSpatialReference *poSRS) override;
      87             : 
      88             :     char **GetMetadataDomainList() override;
      89             :     const char *GetMetadataItem(const char *pszName,
      90             :                                 const char *pszDomain = "") override;
      91             :     char **GetMetadata(const char *pszDomain = "") override;
      92             : 
      93             :     static GDALDataset *Open(GDALOpenInfo *);
      94             :     static int Identify(GDALOpenInfo *);
      95             :     static GDALDataset *Create(const char *pszFilename, int nXSize, int nYSize,
      96             :                                int nBandsIn, GDALDataType eType,
      97             :                                char **papszParamList);
      98             : };
      99             : 
     100             : /************************************************************************/
     101             : /*                            ERSDataset()                             */
     102             : /************************************************************************/
     103             : 
     104          62 : ERSDataset::ERSDataset()
     105             :     : fpImage(nullptr), poDepFile(nullptr), bGotTransform(FALSE),
     106             :       bHDRDirty(FALSE), poHeader(nullptr), nGCPCount(0), pasGCPList(nullptr),
     107          62 :       bHasNoDataValue(FALSE), dfNoDataValue(0.0)
     108             : {
     109          62 :     m_oSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
     110          62 :     m_oGCPSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
     111          62 :     adfGeoTransform[0] = 0.0;
     112          62 :     adfGeoTransform[1] = 1.0;
     113          62 :     adfGeoTransform[2] = 0.0;
     114          62 :     adfGeoTransform[3] = 0.0;
     115          62 :     adfGeoTransform[4] = 0.0;
     116          62 :     adfGeoTransform[5] = 1.0;
     117          62 : }
     118             : 
     119             : /************************************************************************/
     120             : /*                            ~ERSDataset()                            */
     121             : /************************************************************************/
     122             : 
     123         124 : ERSDataset::~ERSDataset()
     124             : 
     125             : {
     126          62 :     ERSDataset::Close();
     127         124 : }
     128             : 
     129             : /************************************************************************/
     130             : /*                              Close()                                 */
     131             : /************************************************************************/
     132             : 
     133         123 : CPLErr ERSDataset::Close()
     134             : {
     135         123 :     CPLErr eErr = CE_None;
     136         123 :     if (nOpenFlags != OPEN_FLAGS_CLOSED)
     137             :     {
     138          62 :         if (ERSDataset::FlushCache(true) != CE_None)
     139           0 :             eErr = CE_Failure;
     140             : 
     141          62 :         if (fpImage != nullptr)
     142             :         {
     143          60 :             VSIFCloseL(fpImage);
     144             :         }
     145             : 
     146          62 :         ERSDataset::CloseDependentDatasets();
     147             : 
     148          62 :         if (nGCPCount > 0)
     149             :         {
     150           3 :             GDALDeinitGCPs(nGCPCount, pasGCPList);
     151           3 :             CPLFree(pasGCPList);
     152             :         }
     153             : 
     154          62 :         if (poHeader != nullptr)
     155          62 :             delete poHeader;
     156             : 
     157          62 :         if (GDALPamDataset::Close() != CE_None)
     158           0 :             eErr = CE_Failure;
     159             :     }
     160         123 :     return eErr;
     161             : }
     162             : 
     163             : /************************************************************************/
     164             : /*                      CloseDependentDatasets()                        */
     165             : /************************************************************************/
     166             : 
     167          62 : int ERSDataset::CloseDependentDatasets()
     168             : {
     169          62 :     int bHasDroppedRef = GDALPamDataset::CloseDependentDatasets();
     170             : 
     171          62 :     if (poDepFile != nullptr)
     172             :     {
     173           1 :         bHasDroppedRef = TRUE;
     174             : 
     175           2 :         for (int iBand = 0; iBand < nBands; iBand++)
     176             :         {
     177           1 :             delete papoBands[iBand];
     178           1 :             papoBands[iBand] = nullptr;
     179             :         }
     180           1 :         nBands = 0;
     181             : 
     182           1 :         GDALClose((GDALDatasetH)poDepFile);
     183           1 :         poDepFile = nullptr;
     184             :     }
     185             : 
     186          62 :     return bHasDroppedRef;
     187             : }
     188             : 
     189             : /************************************************************************/
     190             : /*                             FlushCache()                             */
     191             : /************************************************************************/
     192             : 
     193          63 : CPLErr ERSDataset::FlushCache(bool bAtClosing)
     194             : 
     195             : {
     196          63 :     CPLErr eErr = CE_None;
     197          63 :     if (bHDRDirty)
     198             :     {
     199          37 :         VSILFILE *fpERS = VSIFOpenL(GetDescription(), "w");
     200          37 :         if (fpERS == nullptr)
     201             :         {
     202           0 :             eErr = CE_Failure;
     203           0 :             CPLError(CE_Failure, CPLE_OpenFailed,
     204           0 :                      "Unable to rewrite %s header.", GetDescription());
     205             :         }
     206             :         else
     207             :         {
     208          37 :             if (VSIFPrintfL(fpERS, "DatasetHeader Begin\n") <= 0)
     209           0 :                 eErr = CE_Failure;
     210          37 :             poHeader->WriteSelf(fpERS, 1);
     211          37 :             if (VSIFPrintfL(fpERS, "DatasetHeader End\n") <= 0)
     212           0 :                 eErr = CE_Failure;
     213          37 :             if (VSIFCloseL(fpERS) != 0)
     214           0 :                 eErr = CE_Failure;
     215             :         }
     216             :     }
     217             : 
     218          63 :     if (RawDataset::FlushCache(bAtClosing) != CE_None)
     219           0 :         eErr = CE_Failure;
     220          63 :     return eErr;
     221             : }
     222             : 
     223             : /************************************************************************/
     224             : /*                      GetMetadataDomainList()                         */
     225             : /************************************************************************/
     226             : 
     227           0 : char **ERSDataset::GetMetadataDomainList()
     228             : {
     229           0 :     return BuildMetadataDomainList(GDALPamDataset::GetMetadataDomainList(),
     230           0 :                                    TRUE, "ERS", nullptr);
     231             : }
     232             : 
     233             : /************************************************************************/
     234             : /*                           GetMetadataItem()                          */
     235             : /************************************************************************/
     236             : 
     237          60 : const char *ERSDataset::GetMetadataItem(const char *pszName,
     238             :                                         const char *pszDomain)
     239             : {
     240          60 :     if (pszDomain != nullptr && EQUAL(pszDomain, "ERS") && pszName != nullptr)
     241             :     {
     242          15 :         if (EQUAL(pszName, "PROJ"))
     243           5 :             return osProj.size() ? osProj.c_str() : nullptr;
     244          10 :         if (EQUAL(pszName, "DATUM"))
     245           5 :             return osDatum.size() ? osDatum.c_str() : nullptr;
     246           5 :         if (EQUAL(pszName, "UNITS"))
     247           5 :             return osUnits.size() ? osUnits.c_str() : nullptr;
     248             :     }
     249          45 :     return GDALPamDataset::GetMetadataItem(pszName, pszDomain);
     250             : }
     251             : 
     252             : /************************************************************************/
     253             : /*                            GetMetadata()                             */
     254             : /************************************************************************/
     255             : 
     256          17 : char **ERSDataset::GetMetadata(const char *pszDomain)
     257             : 
     258             : {
     259          17 :     if (pszDomain != nullptr && EQUAL(pszDomain, "ERS"))
     260             :     {
     261           1 :         oERSMetadataList.Clear();
     262           1 :         if (!osProj.empty())
     263             :             oERSMetadataList.AddString(
     264           1 :                 CPLSPrintf("%s=%s", "PROJ", osProj.c_str()));
     265           1 :         if (!osDatum.empty())
     266             :             oERSMetadataList.AddString(
     267           1 :                 CPLSPrintf("%s=%s", "DATUM", osDatum.c_str()));
     268           1 :         if (!osUnits.empty())
     269             :             oERSMetadataList.AddString(
     270           1 :                 CPLSPrintf("%s=%s", "UNITS", osUnits.c_str()));
     271           1 :         return oERSMetadataList.List();
     272             :     }
     273             : 
     274          16 :     return GDALPamDataset::GetMetadata(pszDomain);
     275             : }
     276             : 
     277             : /************************************************************************/
     278             : /*                            GetGCPCount()                             */
     279             : /************************************************************************/
     280             : 
     281           3 : int ERSDataset::GetGCPCount()
     282             : 
     283             : {
     284           3 :     return nGCPCount;
     285             : }
     286             : 
     287             : /************************************************************************/
     288             : /*                          GetGCPSpatialRef()                          */
     289             : /************************************************************************/
     290             : 
     291           1 : const OGRSpatialReference *ERSDataset::GetGCPSpatialRef() const
     292             : 
     293             : {
     294           1 :     return m_oGCPSRS.IsEmpty() ? nullptr : &m_oGCPSRS;
     295             : }
     296             : 
     297             : /************************************************************************/
     298             : /*                               GetGCPs()                              */
     299             : /************************************************************************/
     300             : 
     301           1 : const GDAL_GCP *ERSDataset::GetGCPs()
     302             : 
     303             : {
     304           1 :     return pasGCPList;
     305             : }
     306             : 
     307             : /************************************************************************/
     308             : /*                              SetGCPs()                               */
     309             : /************************************************************************/
     310             : 
     311           1 : CPLErr ERSDataset::SetGCPs(int nGCPCountIn, const GDAL_GCP *pasGCPListIn,
     312             :                            const OGRSpatialReference *poSRS)
     313             : 
     314             : {
     315             :     /* -------------------------------------------------------------------- */
     316             :     /*      Clean old gcps.                                                 */
     317             :     /* -------------------------------------------------------------------- */
     318           1 :     m_oGCPSRS.Clear();
     319             : 
     320           1 :     if (nGCPCount > 0)
     321             :     {
     322           0 :         GDALDeinitGCPs(nGCPCount, pasGCPList);
     323           0 :         CPLFree(pasGCPList);
     324             : 
     325           0 :         pasGCPList = nullptr;
     326           0 :         nGCPCount = 0;
     327             :     }
     328             : 
     329             :     /* -------------------------------------------------------------------- */
     330             :     /*      Copy new ones.                                                  */
     331             :     /* -------------------------------------------------------------------- */
     332           1 :     nGCPCount = nGCPCountIn;
     333           1 :     pasGCPList = GDALDuplicateGCPs(nGCPCount, pasGCPListIn);
     334           1 :     if (poSRS)
     335           1 :         m_oGCPSRS = *poSRS;
     336             : 
     337             :     /* -------------------------------------------------------------------- */
     338             :     /*      Setup the header contents corresponding to these GCPs.          */
     339             :     /* -------------------------------------------------------------------- */
     340           1 :     bHDRDirty = TRUE;
     341             : 
     342           1 :     poHeader->Set("RasterInfo.WarpControl.WarpType", "Polynomial");
     343           1 :     if (nGCPCount > 6)
     344           0 :         poHeader->Set("RasterInfo.WarpControl.WarpOrder", "2");
     345             :     else
     346           1 :         poHeader->Set("RasterInfo.WarpControl.WarpOrder", "1");
     347           1 :     poHeader->Set("RasterInfo.WarpControl.WarpSampling", "Nearest");
     348             : 
     349             :     /* -------------------------------------------------------------------- */
     350             :     /*      Translate the projection.                                       */
     351             :     /* -------------------------------------------------------------------- */
     352             :     char szERSProj[32], szERSDatum[32], szERSUnits[32];
     353             : 
     354           1 :     m_oGCPSRS.exportToERM(szERSProj, szERSDatum, szERSUnits);
     355             : 
     356             :     /* Write the above computed values, unless they have been overridden by */
     357             :     /* the creation options PROJ, DATUM or UNITS */
     358             : 
     359           1 :     poHeader->Set("RasterInfo.WarpControl.CoordinateSpace.Datum",
     360           2 :                   CPLString().Printf("\"%s\"", (osDatum.size())
     361           0 :                                                    ? osDatum.c_str()
     362           1 :                                                    : szERSDatum));
     363           1 :     poHeader->Set("RasterInfo.WarpControl.CoordinateSpace.Projection",
     364           2 :                   CPLString().Printf("\"%s\"", (osProj.size()) ? osProj.c_str()
     365           1 :                                                                : szERSProj));
     366           1 :     poHeader->Set("RasterInfo.WarpControl.CoordinateSpace.CoordinateType",
     367           2 :                   CPLString().Printf("EN"));
     368           1 :     poHeader->Set("RasterInfo.WarpControl.CoordinateSpace.Units",
     369           2 :                   CPLString().Printf("\"%s\"", (osUnits.size())
     370           0 :                                                    ? osUnits.c_str()
     371           1 :                                                    : szERSUnits));
     372           1 :     poHeader->Set("RasterInfo.WarpControl.CoordinateSpace.Rotation", "0:0:0.0");
     373             : 
     374             :     /* -------------------------------------------------------------------- */
     375             :     /*      Translate the GCPs.                                             */
     376             :     /* -------------------------------------------------------------------- */
     377           1 :     CPLString osControlPoints = "{\n";
     378             : 
     379           5 :     for (int iGCP = 0; iGCP < nGCPCount; iGCP++)
     380             :     {
     381           8 :         CPLString osLine;
     382             : 
     383           8 :         CPLString osId = pasGCPList[iGCP].pszId;
     384           4 :         if (osId.empty())
     385           4 :             osId.Printf("%d", iGCP + 1);
     386             : 
     387             :         osLine.Printf(
     388             :             "\t\t\t\t\"%s\"\tYes\tYes\t%.6f\t%.6f\t%.15g\t%.15g\t%.15g\n",
     389           4 :             osId.c_str(), pasGCPList[iGCP].dfGCPPixel,
     390           4 :             pasGCPList[iGCP].dfGCPLine, pasGCPList[iGCP].dfGCPX,
     391           4 :             pasGCPList[iGCP].dfGCPY, pasGCPList[iGCP].dfGCPZ);
     392           4 :         osControlPoints += osLine;
     393             :     }
     394           1 :     osControlPoints += "\t\t}";
     395             : 
     396           1 :     poHeader->Set("RasterInfo.WarpControl.ControlPoints", osControlPoints);
     397             : 
     398           2 :     return CE_None;
     399             : }
     400             : 
     401             : /************************************************************************/
     402             : /*                          GetSpatialRef()                             */
     403             : /************************************************************************/
     404             : 
     405           3 : const OGRSpatialReference *ERSDataset::GetSpatialRef() const
     406             : 
     407             : {
     408             :     // try xml first
     409           3 :     const auto poSRS = GDALPamDataset::GetSpatialRef();
     410           3 :     if (poSRS)
     411           0 :         return poSRS;
     412             : 
     413           3 :     return m_oSRS.IsEmpty() ? nullptr : &m_oSRS;
     414             : }
     415             : 
     416             : /************************************************************************/
     417             : /*                           SetSpatialRef()                            */
     418             : /************************************************************************/
     419             : 
     420          34 : CPLErr ERSDataset::SetSpatialRef(const OGRSpatialReference *poSRS)
     421             : 
     422             : {
     423          34 :     if (poSRS == nullptr && m_oSRS.IsEmpty())
     424           0 :         return CE_None;
     425          34 :     if (poSRS != nullptr && poSRS->IsSame(&m_oSRS))
     426           0 :         return CE_None;
     427             : 
     428          34 :     m_oSRS.Clear();
     429          34 :     if (poSRS)
     430          34 :         m_oSRS = *poSRS;
     431             : 
     432             :     char szERSProj[32], szERSDatum[32], szERSUnits[32];
     433             : 
     434          34 :     m_oSRS.exportToERM(szERSProj, szERSDatum, szERSUnits);
     435             : 
     436             :     /* Write the above computed values, unless they have been overridden by */
     437             :     /* the creation options PROJ, DATUM or UNITS */
     438          34 :     if (!osProjForced.empty())
     439           1 :         osProj = osProjForced;
     440             :     else
     441          33 :         osProj = szERSProj;
     442          34 :     if (!osDatumForced.empty())
     443           1 :         osDatum = osDatumForced;
     444             :     else
     445          33 :         osDatum = szERSDatum;
     446          34 :     if (!osUnitsForced.empty())
     447           1 :         osUnits = osUnitsForced;
     448             :     else
     449          33 :         osUnits = szERSUnits;
     450             : 
     451          34 :     WriteProjectionInfo(osProj, osDatum, osUnits);
     452             : 
     453          34 :     return CE_None;
     454             : }
     455             : 
     456             : /************************************************************************/
     457             : /*                         WriteProjectionInfo()                        */
     458             : /************************************************************************/
     459             : 
     460          36 : void ERSDataset::WriteProjectionInfo(const char *pszProj, const char *pszDatum,
     461             :                                      const char *pszUnits)
     462             : {
     463          36 :     bHDRDirty = TRUE;
     464          36 :     poHeader->Set("CoordinateSpace.Datum",
     465          72 :                   CPLString().Printf("\"%s\"", pszDatum));
     466          36 :     poHeader->Set("CoordinateSpace.Projection",
     467          72 :                   CPLString().Printf("\"%s\"", pszProj));
     468          36 :     poHeader->Set("CoordinateSpace.CoordinateType", CPLString().Printf("EN"));
     469          36 :     poHeader->Set("CoordinateSpace.Units",
     470          72 :                   CPLString().Printf("\"%s\"", pszUnits));
     471          36 :     poHeader->Set("CoordinateSpace.Rotation", "0:0:0.0");
     472             : 
     473             :     /* -------------------------------------------------------------------- */
     474             :     /*      It seems that CoordinateSpace needs to come before              */
     475             :     /*      RasterInfo.  Try moving it up manually.                         */
     476             :     /* -------------------------------------------------------------------- */
     477          36 :     int iRasterInfo = -1;
     478          36 :     int iCoordSpace = -1;
     479             : 
     480         250 :     for (int i = 0; i < poHeader->nItemCount; i++)
     481             :     {
     482         250 :         if (EQUAL(poHeader->papszItemName[i], "RasterInfo"))
     483          34 :             iRasterInfo = i;
     484             : 
     485         250 :         if (EQUAL(poHeader->papszItemName[i], "CoordinateSpace"))
     486             :         {
     487          36 :             iCoordSpace = i;
     488          36 :             break;
     489             :         }
     490             :     }
     491             : 
     492          36 :     if (iCoordSpace > iRasterInfo && iRasterInfo != -1)
     493             :     {
     494          68 :         for (int i = iCoordSpace; i > 0 && i != iRasterInfo; i--)
     495             :         {
     496             : 
     497          34 :             ERSHdrNode *poTemp = poHeader->papoItemChild[i];
     498          34 :             poHeader->papoItemChild[i] = poHeader->papoItemChild[i - 1];
     499          34 :             poHeader->papoItemChild[i - 1] = poTemp;
     500             : 
     501          34 :             char *pszTemp = poHeader->papszItemName[i];
     502          34 :             poHeader->papszItemName[i] = poHeader->papszItemName[i - 1];
     503          34 :             poHeader->papszItemName[i - 1] = pszTemp;
     504             : 
     505          34 :             pszTemp = poHeader->papszItemValue[i];
     506          34 :             poHeader->papszItemValue[i] = poHeader->papszItemValue[i - 1];
     507          34 :             poHeader->papszItemValue[i - 1] = pszTemp;
     508             :         }
     509             :     }
     510          36 : }
     511             : 
     512             : /************************************************************************/
     513             : /*                          GetGeoTransform()                           */
     514             : /************************************************************************/
     515             : 
     516          22 : CPLErr ERSDataset::GetGeoTransform(double *padfTransform)
     517             : 
     518             : {
     519          22 :     if (bGotTransform)
     520             :     {
     521          22 :         memcpy(padfTransform, adfGeoTransform, sizeof(double) * 6);
     522          22 :         return CE_None;
     523             :     }
     524             : 
     525           0 :     return GDALPamDataset::GetGeoTransform(padfTransform);
     526             : }
     527             : 
     528             : /************************************************************************/
     529             : /*                          SetGeoTransform()                           */
     530             : /************************************************************************/
     531             : 
     532          32 : CPLErr ERSDataset::SetGeoTransform(double *padfTransform)
     533             : 
     534             : {
     535          32 :     if (memcmp(padfTransform, adfGeoTransform, sizeof(double) * 6) == 0)
     536           0 :         return CE_None;
     537             : 
     538          32 :     if (adfGeoTransform[2] != 0 || adfGeoTransform[4] != 0)
     539             :     {
     540           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     541             :                  "Rotated and skewed geotransforms not currently supported for "
     542             :                  "ERS driver.");
     543           0 :         return CE_Failure;
     544             :     }
     545             : 
     546          32 :     bGotTransform = TRUE;
     547          32 :     memcpy(adfGeoTransform, padfTransform, sizeof(double) * 6);
     548             : 
     549          32 :     bHDRDirty = TRUE;
     550             : 
     551          32 :     poHeader->Set("RasterInfo.CellInfo.Xdimension",
     552          64 :                   CPLString().Printf("%.15g", fabs(adfGeoTransform[1])));
     553          32 :     poHeader->Set("RasterInfo.CellInfo.Ydimension",
     554          64 :                   CPLString().Printf("%.15g", fabs(adfGeoTransform[5])));
     555          32 :     poHeader->Set("RasterInfo.RegistrationCoord.Eastings",
     556          64 :                   CPLString().Printf("%.15g", adfGeoTransform[0]));
     557          32 :     poHeader->Set("RasterInfo.RegistrationCoord.Northings",
     558          64 :                   CPLString().Printf("%.15g", adfGeoTransform[3]));
     559             : 
     560          64 :     if (CPLAtof(poHeader->Find("RasterInfo.RegistrationCellX", "0")) != 0.0 ||
     561          32 :         CPLAtof(poHeader->Find("RasterInfo.RegistrationCellY", "0")) != 0.0)
     562             :     {
     563             :         // Reset RegistrationCellX/Y to 0 if the header gets rewritten (#5493)
     564           0 :         poHeader->Set("RasterInfo.RegistrationCellX", "0");
     565           0 :         poHeader->Set("RasterInfo.RegistrationCellY", "0");
     566             :     }
     567             : 
     568          32 :     return CE_None;
     569             : }
     570             : 
     571             : /************************************************************************/
     572             : /*                             ERSDMS2Dec()                             */
     573             : /*                                                                      */
     574             : /*      Convert ERS DMS format to decimal degrees.   Input is like      */
     575             : /*      "-180:00:00".                                                   */
     576             : /************************************************************************/
     577             : 
     578          10 : static double ERSDMS2Dec(const char *pszDMS)
     579             : 
     580             : {
     581          10 :     char **papszTokens = CSLTokenizeStringComplex(pszDMS, ":", FALSE, FALSE);
     582             : 
     583          10 :     if (CSLCount(papszTokens) != 3)
     584             :     {
     585           0 :         CSLDestroy(papszTokens);
     586           0 :         return CPLAtof(pszDMS);
     587             :     }
     588             : 
     589          10 :     double dfResult = fabs(CPLAtof(papszTokens[0])) +
     590          10 :                       CPLAtof(papszTokens[1]) / 60.0 +
     591          10 :                       CPLAtof(papszTokens[2]) / 3600.0;
     592             : 
     593          10 :     if (CPLAtof(papszTokens[0]) < 0)
     594           8 :         dfResult *= -1;
     595             : 
     596          10 :     CSLDestroy(papszTokens);
     597          10 :     return dfResult;
     598             : }
     599             : 
     600             : /************************************************************************/
     601             : /*                            GetFileList()                             */
     602             : /************************************************************************/
     603             : 
     604          10 : char **ERSDataset::GetFileList()
     605             : 
     606             : {
     607             :     static thread_local int nRecLevel = 0;
     608          10 :     if (nRecLevel > 0)
     609           0 :         return nullptr;
     610             : 
     611             :     // Main data file, etc.
     612          10 :     char **papszFileList = GDALPamDataset::GetFileList();
     613             : 
     614             :     // Add raw data file if we have one.
     615          10 :     if (!osRawFilename.empty())
     616          10 :         papszFileList = CSLAddString(papszFileList, osRawFilename);
     617             : 
     618             :     // If we have a dependent file, merge its list of files in.
     619          10 :     if (poDepFile)
     620             :     {
     621           0 :         nRecLevel++;
     622           0 :         char **papszDepFiles = poDepFile->GetFileList();
     623           0 :         nRecLevel--;
     624           0 :         papszFileList = CSLInsertStrings(papszFileList, -1, papszDepFiles);
     625           0 :         CSLDestroy(papszDepFiles);
     626             :     }
     627             : 
     628          10 :     return papszFileList;
     629             : }
     630             : 
     631             : /************************************************************************/
     632             : /*                              ReadGCPs()                              */
     633             : /*                                                                      */
     634             : /*      Read the GCPs from the header.                                  */
     635             : /************************************************************************/
     636             : 
     637           2 : void ERSDataset::ReadGCPs()
     638             : 
     639             : {
     640             :     const char *pszCP =
     641           2 :         poHeader->Find("RasterInfo.WarpControl.ControlPoints", nullptr);
     642             : 
     643           2 :     if (pszCP == nullptr)
     644           0 :         return;
     645             : 
     646             :     /* -------------------------------------------------------------------- */
     647             :     /*      Parse the control points.  They will look something like:       */
     648             :     /*                                                                      */
     649             :     /*   "1035" Yes No 2344.650885 3546.419458 483270.73 3620906.21 3.105   */
     650             :     /* -------------------------------------------------------------------- */
     651           2 :     char **papszTokens = CSLTokenizeStringComplex(pszCP, "{ \t}", TRUE, FALSE);
     652           2 :     int nItemCount = CSLCount(papszTokens);
     653             : 
     654             :     /* -------------------------------------------------------------------- */
     655             :     /*      Work out if we have elevation values or not.                    */
     656             :     /* -------------------------------------------------------------------- */
     657             :     int nItemsPerLine;
     658             : 
     659           2 :     if (nItemCount == 7)
     660           0 :         nItemsPerLine = 7;
     661           2 :     else if (nItemCount == 8)
     662           0 :         nItemsPerLine = 8;
     663           2 :     else if (nItemCount < 14)
     664             :     {
     665           0 :         CPLDebug("ERS", "Invalid item count for ControlPoints");
     666           0 :         CSLDestroy(papszTokens);
     667           0 :         return;
     668             :     }
     669           2 :     else if (EQUAL(papszTokens[8], "Yes") || EQUAL(papszTokens[8], "No"))
     670           0 :         nItemsPerLine = 7;
     671           2 :     else if (EQUAL(papszTokens[9], "Yes") || EQUAL(papszTokens[9], "No"))
     672           2 :         nItemsPerLine = 8;
     673             :     else
     674             :     {
     675           0 :         CPLDebug("ERS", "Invalid format for ControlPoints");
     676           0 :         CSLDestroy(papszTokens);
     677           0 :         return;
     678             :     }
     679             : 
     680             :     /* -------------------------------------------------------------------- */
     681             :     /*      Setup GCPs.                                                     */
     682             :     /* -------------------------------------------------------------------- */
     683           2 :     CPLAssert(nGCPCount == 0);
     684             : 
     685           2 :     nGCPCount = nItemCount / nItemsPerLine;
     686           2 :     pasGCPList = (GDAL_GCP *)CPLCalloc(nGCPCount, sizeof(GDAL_GCP));
     687           2 :     GDALInitGCPs(nGCPCount, pasGCPList);
     688             : 
     689          10 :     for (int iGCP = 0; iGCP < nGCPCount; iGCP++)
     690             :     {
     691           8 :         GDAL_GCP *psGCP = pasGCPList + iGCP;
     692             : 
     693           8 :         CPLFree(psGCP->pszId);
     694           8 :         psGCP->pszId = CPLStrdup(papszTokens[iGCP * nItemsPerLine + 0]);
     695           8 :         psGCP->dfGCPPixel = CPLAtof(papszTokens[iGCP * nItemsPerLine + 3]);
     696           8 :         psGCP->dfGCPLine = CPLAtof(papszTokens[iGCP * nItemsPerLine + 4]);
     697           8 :         psGCP->dfGCPX = CPLAtof(papszTokens[iGCP * nItemsPerLine + 5]);
     698           8 :         psGCP->dfGCPY = CPLAtof(papszTokens[iGCP * nItemsPerLine + 6]);
     699           8 :         if (nItemsPerLine == 8)
     700           8 :             psGCP->dfGCPZ = CPLAtof(papszTokens[iGCP * nItemsPerLine + 7]);
     701             :     }
     702             : 
     703           2 :     CSLDestroy(papszTokens);
     704             : 
     705             :     /* -------------------------------------------------------------------- */
     706             :     /*      Parse the GCP projection.                                       */
     707             :     /* -------------------------------------------------------------------- */
     708             :     osProj =
     709           2 :         poHeader->Find("RasterInfo.WarpControl.CoordinateSpace.Projection", "");
     710             :     osDatum =
     711           2 :         poHeader->Find("RasterInfo.WarpControl.CoordinateSpace.Datum", "");
     712             :     osUnits =
     713           2 :         poHeader->Find("RasterInfo.WarpControl.CoordinateSpace.Units", "");
     714             : 
     715           2 :     m_oGCPSRS.importFromERM(!osProj.empty() ? osProj.c_str() : "RAW",
     716           2 :                             !osDatum.empty() ? osDatum.c_str() : "WGS84",
     717           2 :                             !osUnits.empty() ? osUnits.c_str() : "METERS");
     718             : }
     719             : 
     720             : /************************************************************************/
     721             : /* ==================================================================== */
     722             : /*                             ERSRasterBand                            */
     723             : /* ==================================================================== */
     724             : /************************************************************************/
     725             : 
     726             : class ERSRasterBand final : public RawRasterBand
     727             : {
     728             :   public:
     729             :     ERSRasterBand(GDALDataset *poDS, int nBand, VSILFILE *fpRaw,
     730             :                   vsi_l_offset nImgOffset, int nPixelOffset, int nLineOffset,
     731             :                   GDALDataType eDataType, int bNativeOrder);
     732             : 
     733             :     double GetNoDataValue(int *pbSuccess = nullptr) override;
     734             :     CPLErr SetNoDataValue(double) override;
     735             : };
     736             : 
     737             : /************************************************************************/
     738             : /*                           ERSRasterBand()                            */
     739             : /************************************************************************/
     740             : 
     741         102 : ERSRasterBand::ERSRasterBand(GDALDataset *poDSIn, int nBandIn,
     742             :                              VSILFILE *fpRawIn, vsi_l_offset nImgOffsetIn,
     743             :                              int nPixelOffsetIn, int nLineOffsetIn,
     744         102 :                              GDALDataType eDataTypeIn, int bNativeOrderIn)
     745             :     : RawRasterBand(poDSIn, nBandIn, fpRawIn, nImgOffsetIn, nPixelOffsetIn,
     746             :                     nLineOffsetIn, eDataTypeIn, bNativeOrderIn,
     747         102 :                     RawRasterBand::OwnFP::NO)
     748             : {
     749         102 : }
     750             : 
     751             : /************************************************************************/
     752             : /*                           GetNoDataValue()                           */
     753             : /************************************************************************/
     754             : 
     755          15 : double ERSRasterBand::GetNoDataValue(int *pbSuccess)
     756             : {
     757          15 :     ERSDataset *poGDS = cpl::down_cast<ERSDataset *>(poDS);
     758          15 :     if (poGDS->bHasNoDataValue)
     759             :     {
     760           1 :         if (pbSuccess)
     761           1 :             *pbSuccess = TRUE;
     762           1 :         return poGDS->dfNoDataValue;
     763             :     }
     764             : 
     765          14 :     return RawRasterBand::GetNoDataValue(pbSuccess);
     766             : }
     767             : 
     768             : /************************************************************************/
     769             : /*                           SetNoDataValue()                           */
     770             : /************************************************************************/
     771             : 
     772           1 : CPLErr ERSRasterBand::SetNoDataValue(double dfNoDataValue)
     773             : {
     774           1 :     ERSDataset *poGDS = cpl::down_cast<ERSDataset *>(poDS);
     775           1 :     if (!poGDS->bHasNoDataValue || poGDS->dfNoDataValue != dfNoDataValue)
     776             :     {
     777           1 :         poGDS->bHasNoDataValue = TRUE;
     778           1 :         poGDS->dfNoDataValue = dfNoDataValue;
     779             : 
     780           1 :         poGDS->bHDRDirty = TRUE;
     781           1 :         poGDS->poHeader->Set("RasterInfo.NullCellValue",
     782           2 :                              CPLString().Printf("%.16g", dfNoDataValue));
     783             :     }
     784           1 :     return CE_None;
     785             : }
     786             : 
     787             : /************************************************************************/
     788             : /*                              Identify()                              */
     789             : /************************************************************************/
     790             : 
     791       54465 : int ERSDataset::Identify(GDALOpenInfo *poOpenInfo)
     792             : 
     793             : {
     794             :     /* -------------------------------------------------------------------- */
     795             :     /*      We assume the user selects the .ers file.                       */
     796             :     /* -------------------------------------------------------------------- */
     797       54465 :     CPLString osHeader((const char *)poOpenInfo->pabyHeader,
     798      108927 :                        poOpenInfo->nHeaderBytes);
     799             : 
     800       54462 :     if (osHeader.ifind("Algorithm Begin") != std::string::npos)
     801             :     {
     802           0 :         CPLError(CE_Failure, CPLE_OpenFailed,
     803             :                  "%s appears to be an algorithm ERS file, which is not "
     804             :                  "currently supported.",
     805             :                  poOpenInfo->pszFilename);
     806           0 :         return FALSE;
     807             :     }
     808             : 
     809       54456 :     if (osHeader.ifind("DatasetHeader ") != std::string::npos)
     810          89 :         return TRUE;
     811             : 
     812       54373 :     return FALSE;
     813             : }
     814             : 
     815             : /************************************************************************/
     816             : /*                         ERSProxyRasterBand                           */
     817             : /************************************************************************/
     818             : 
     819             : namespace
     820             : {
     821             : 
     822          63 : static int &GetRecLevel()
     823             : {
     824             :     static thread_local int nRecLevel = 0;
     825          63 :     return nRecLevel;
     826             : }
     827             : 
     828             : class ERSProxyRasterBand final : public GDALProxyRasterBand
     829             : {
     830             :   public:
     831           1 :     explicit ERSProxyRasterBand(GDALRasterBand *poUnderlyingBand)
     832           1 :         : m_poUnderlyingBand(poUnderlyingBand)
     833             :     {
     834           1 :         poUnderlyingBand->GetBlockSize(&nBlockXSize, &nBlockYSize);
     835           1 :         eDataType = poUnderlyingBand->GetRasterDataType();
     836           1 :     }
     837             : 
     838             :     int GetOverviewCount() override;
     839             : 
     840             :   protected:
     841           1 :     GDALRasterBand *RefUnderlyingRasterBand(bool /*bForceOpen*/) const override
     842             :     {
     843           1 :         return m_poUnderlyingBand;
     844             :     }
     845             : 
     846             :   private:
     847             :     GDALRasterBand *m_poUnderlyingBand;
     848             : };
     849             : 
     850           0 : int ERSProxyRasterBand::GetOverviewCount()
     851             : {
     852           0 :     int &nRecLevel = GetRecLevel();
     853           0 :     nRecLevel++;
     854           0 :     int nRet = GDALProxyRasterBand::GetOverviewCount();
     855           0 :     nRecLevel--;
     856           0 :     return nRet;
     857             : }
     858             : 
     859             : }  // namespace
     860             : 
     861             : /************************************************************************/
     862             : /*                                Open()                                */
     863             : /************************************************************************/
     864             : 
     865          63 : GDALDataset *ERSDataset::Open(GDALOpenInfo *poOpenInfo)
     866             : 
     867             : {
     868          63 :     if (!Identify(poOpenInfo) || poOpenInfo->fpL == nullptr)
     869           0 :         return nullptr;
     870             : 
     871          63 :     int &nRecLevel = GetRecLevel();
     872             :     // cppcheck-suppress knownConditionTrueFalse
     873          63 :     if (nRecLevel)
     874             :     {
     875           1 :         CPLError(CE_Failure, CPLE_AppDefined,
     876             :                  "Attempt at recursively opening ERS dataset");
     877           1 :         return nullptr;
     878             :     }
     879             : 
     880             :     /* -------------------------------------------------------------------- */
     881             :     /*      Ingest the file as a tree of header nodes.                      */
     882             :     /* -------------------------------------------------------------------- */
     883          62 :     ERSHdrNode *poHeader = new ERSHdrNode();
     884             : 
     885          62 :     if (!poHeader->ParseHeader(poOpenInfo->fpL))
     886             :     {
     887           0 :         delete poHeader;
     888           0 :         VSIFCloseL(poOpenInfo->fpL);
     889           0 :         poOpenInfo->fpL = nullptr;
     890           0 :         return nullptr;
     891             :     }
     892             : 
     893          62 :     VSIFCloseL(poOpenInfo->fpL);
     894          62 :     poOpenInfo->fpL = nullptr;
     895             : 
     896             :     /* -------------------------------------------------------------------- */
     897             :     /*      Do we have the minimum required information from this header?   */
     898             :     /* -------------------------------------------------------------------- */
     899          62 :     if (poHeader->Find("RasterInfo.NrOfLines") == nullptr ||
     900         124 :         poHeader->Find("RasterInfo.NrOfCellsPerLine") == nullptr ||
     901          62 :         poHeader->Find("RasterInfo.NrOfBands") == nullptr)
     902             :     {
     903           0 :         if (poHeader->FindNode("Algorithm") != nullptr)
     904             :         {
     905           0 :             CPLError(CE_Failure, CPLE_OpenFailed,
     906             :                      "%s appears to be an algorithm ERS file, which is not "
     907             :                      "currently supported.",
     908             :                      poOpenInfo->pszFilename);
     909             :         }
     910           0 :         delete poHeader;
     911           0 :         return nullptr;
     912             :     }
     913             : 
     914             :     /* -------------------------------------------------------------------- */
     915             :     /*      Create a corresponding GDALDataset.                             */
     916             :     /* -------------------------------------------------------------------- */
     917         124 :     auto poDS = std::make_unique<ERSDataset>();
     918          62 :     poDS->poHeader = poHeader;
     919          62 :     poDS->eAccess = poOpenInfo->eAccess;
     920             : 
     921             :     /* -------------------------------------------------------------------- */
     922             :     /*      Capture some information from the file that is of interest.     */
     923             :     /* -------------------------------------------------------------------- */
     924          62 :     int nBands = atoi(poHeader->Find("RasterInfo.NrOfBands"));
     925          62 :     poDS->nRasterXSize = atoi(poHeader->Find("RasterInfo.NrOfCellsPerLine"));
     926          62 :     poDS->nRasterYSize = atoi(poHeader->Find("RasterInfo.NrOfLines"));
     927             : 
     928         124 :     if (!GDALCheckDatasetDimensions(poDS->nRasterXSize, poDS->nRasterYSize) ||
     929          62 :         !GDALCheckBandCount(nBands, FALSE))
     930             :     {
     931           0 :         return nullptr;
     932             :     }
     933             : 
     934             :     /* -------------------------------------------------------------------- */
     935             :     /*     Get the HeaderOffset if it exists in the header                  */
     936             :     /* -------------------------------------------------------------------- */
     937          62 :     GIntBig nHeaderOffset = 0;
     938          62 :     const char *pszHeaderOffset = poHeader->Find("HeaderOffset");
     939          62 :     if (pszHeaderOffset != nullptr)
     940             :     {
     941           2 :         nHeaderOffset = CPLAtoGIntBig(pszHeaderOffset);
     942           2 :         if (nHeaderOffset < 0)
     943             :         {
     944           0 :             CPLError(CE_Failure, CPLE_AppDefined,
     945             :                      "Illegal value for HeaderOffset: %s", pszHeaderOffset);
     946           0 :             return nullptr;
     947             :         }
     948             :     }
     949             : 
     950             :     /* -------------------------------------------------------------------- */
     951             :     /*      Establish the data type.                                        */
     952             :     /* -------------------------------------------------------------------- */
     953             :     CPLString osCellType =
     954         124 :         poHeader->Find("RasterInfo.CellType", "Unsigned8BitInteger");
     955             :     GDALDataType eType;
     956          62 :     if (EQUAL(osCellType, "Unsigned8BitInteger"))
     957          29 :         eType = GDT_Byte;
     958          33 :     else if (EQUAL(osCellType, "Signed8BitInteger"))
     959           6 :         eType = GDT_Int8;
     960          27 :     else if (EQUAL(osCellType, "Unsigned16BitInteger"))
     961           3 :         eType = GDT_UInt16;
     962          24 :     else if (EQUAL(osCellType, "Signed16BitInteger"))
     963           6 :         eType = GDT_Int16;
     964          18 :     else if (EQUAL(osCellType, "Unsigned32BitInteger"))
     965           3 :         eType = GDT_UInt32;
     966          15 :     else if (EQUAL(osCellType, "Signed32BitInteger"))
     967           3 :         eType = GDT_Int32;
     968          12 :     else if (EQUAL(osCellType, "IEEE4ByteReal"))
     969           9 :         eType = GDT_Float32;
     970           3 :     else if (EQUAL(osCellType, "IEEE8ByteReal"))
     971           3 :         eType = GDT_Float64;
     972             :     else
     973             :     {
     974           0 :         CPLDebug("ERS", "Unknown CellType '%s'", osCellType.c_str());
     975           0 :         eType = GDT_Byte;
     976             :     }
     977             : 
     978             :     /* -------------------------------------------------------------------- */
     979             :     /*      Pick up the word order.                                         */
     980             :     /* -------------------------------------------------------------------- */
     981             :     const int bNative =
     982             : #ifdef CPL_LSB
     983          62 :         EQUAL(poHeader->Find("ByteOrder", "LSBFirst"), "LSBFirst")
     984             : #else
     985             :         EQUAL(poHeader->Find("ByteOrder", "MSBFirst"), "MSBFirst")
     986             : #endif
     987             :         ;
     988             :     /* -------------------------------------------------------------------- */
     989             :     /*      Figure out the name of the target file.                         */
     990             :     /* -------------------------------------------------------------------- */
     991         124 :     CPLString osPath = CPLGetPathSafe(poOpenInfo->pszFilename);
     992         124 :     CPLString osDataFile = poHeader->Find("DataFile", "");
     993             : 
     994          62 :     if (osDataFile.length() == 0)  // just strip off extension.
     995             :     {
     996          61 :         osDataFile = CPLGetFilename(poOpenInfo->pszFilename);
     997          61 :         osDataFile = osDataFile.substr(0, osDataFile.find_last_of('.'));
     998             :     }
     999             : 
    1000         124 :     CPLString osDataFilePath = CPLFormFilenameSafe(osPath, osDataFile, nullptr);
    1001             : 
    1002             :     /* -------------------------------------------------------------------- */
    1003             :     /*      DataSetType = Translated files are links to things like ecw     */
    1004             :     /*      files.                                                          */
    1005             :     /* -------------------------------------------------------------------- */
    1006          62 :     if (EQUAL(poHeader->Find("DataSetType", ""), "Translated"))
    1007             :     {
    1008           2 :         nRecLevel++;
    1009           2 :         poDS->poDepFile = GDALDataset::FromHandle(
    1010             :             GDALOpen(osDataFilePath, poOpenInfo->eAccess));
    1011           2 :         nRecLevel--;
    1012             : 
    1013           2 :         if (poDS->poDepFile != nullptr &&
    1014           1 :             poDS->poDepFile->GetRasterXSize() == poDS->GetRasterXSize() &&
    1015           4 :             poDS->poDepFile->GetRasterYSize() == poDS->GetRasterYSize() &&
    1016           1 :             poDS->poDepFile->GetRasterCount() >= nBands)
    1017             :         {
    1018           2 :             for (int iBand = 0; iBand < nBands; iBand++)
    1019             :             {
    1020             :                 // Assume pixel interleaved.
    1021           2 :                 poDS->SetBand(iBand + 1,
    1022             :                               new ERSProxyRasterBand(
    1023           1 :                                   poDS->poDepFile->GetRasterBand(iBand + 1)));
    1024             :             }
    1025             :         }
    1026             :         else
    1027             :         {
    1028           1 :             delete poDS->poDepFile;
    1029           1 :             poDS->poDepFile = nullptr;
    1030             :         }
    1031             :     }
    1032             : 
    1033             :     /* ==================================================================== */
    1034             :     /*      While ERStorage indicates a raw file.                           */
    1035             :     /* ==================================================================== */
    1036          60 :     else if (EQUAL(poHeader->Find("DataSetType", ""), "ERStorage"))
    1037             :     {
    1038             :         // Open data file.
    1039          60 :         if (poOpenInfo->eAccess == GA_Update)
    1040          38 :             poDS->fpImage = VSIFOpenL(osDataFilePath, "r+");
    1041             :         else
    1042          22 :             poDS->fpImage = VSIFOpenL(osDataFilePath, "r");
    1043             : 
    1044          60 :         poDS->osRawFilename = std::move(osDataFilePath);
    1045             : 
    1046          60 :         if (poDS->fpImage != nullptr && nBands > 0)
    1047             :         {
    1048          60 :             int iWordSize = GDALGetDataTypeSizeBytes(eType);
    1049             : 
    1050          60 :             const auto knIntMax = std::numeric_limits<int>::max();
    1051         120 :             if (nBands > knIntMax / iWordSize ||
    1052          60 :                 poDS->nRasterXSize > knIntMax / (nBands * iWordSize))
    1053             :             {
    1054           0 :                 CPLError(CE_Failure, CPLE_AppDefined,
    1055             :                          "int overflow: too large nBands and/or nRasterXSize");
    1056           0 :                 return nullptr;
    1057             :             }
    1058             : 
    1059          60 :             if (!RAWDatasetCheckMemoryUsage(
    1060          60 :                     poDS->nRasterXSize, poDS->nRasterYSize, nBands, iWordSize,
    1061          60 :                     iWordSize, iWordSize * nBands * poDS->nRasterXSize,
    1062             :                     nHeaderOffset,
    1063          60 :                     static_cast<vsi_l_offset>(iWordSize) * poDS->nRasterXSize,
    1064          60 :                     poDS->fpImage))
    1065             :             {
    1066           0 :                 return nullptr;
    1067             :             }
    1068          60 :             if (nHeaderOffset > std::numeric_limits<GIntBig>::max() -
    1069          60 :                                     static_cast<GIntBig>(nBands - 1) *
    1070          60 :                                         iWordSize * poDS->nRasterXSize)
    1071             :             {
    1072           0 :                 CPLError(CE_Failure, CPLE_AppDefined,
    1073             :                          "int overflow: too large nHeaderOffset");
    1074           0 :                 return nullptr;
    1075             :             }
    1076             : 
    1077         162 :             for (int iBand = 0; iBand < nBands; iBand++)
    1078             :             {
    1079             :                 // Assume pixel interleaved.
    1080             :                 auto poBand = std::make_unique<ERSRasterBand>(
    1081         102 :                     poDS.get(), iBand + 1, poDS->fpImage,
    1082           0 :                     nHeaderOffset + static_cast<vsi_l_offset>(iWordSize) *
    1083         102 :                                         iBand * poDS->nRasterXSize,
    1084         102 :                     iWordSize, iWordSize * nBands * poDS->nRasterXSize, eType,
    1085         102 :                     bNative);
    1086         102 :                 if (!poBand->IsValid())
    1087           0 :                     return nullptr;
    1088         102 :                 poDS->SetBand(iBand + 1, std::move(poBand));
    1089             :             }
    1090             :         }
    1091             :     }
    1092             : 
    1093             :     /* -------------------------------------------------------------------- */
    1094             :     /*      Otherwise we have an error!                                     */
    1095             :     /* -------------------------------------------------------------------- */
    1096          62 :     if (poDS->nBands == 0)
    1097             :     {
    1098           1 :         return nullptr;
    1099             :     }
    1100             : 
    1101             :     /* -------------------------------------------------------------------- */
    1102             :     /*      Look for band descriptions.                                     */
    1103             :     /* -------------------------------------------------------------------- */
    1104          61 :     ERSHdrNode *poRI = poHeader->FindNode("RasterInfo");
    1105             : 
    1106         357 :     for (int iChild = 0, iBand = 0;
    1107         357 :          poRI != nullptr && iChild < poRI->nItemCount && iBand < poDS->nBands;
    1108             :          iChild++)
    1109             :     {
    1110         296 :         if (poRI->papoItemChild[iChild] != nullptr &&
    1111          33 :             EQUAL(poRI->papszItemName[iChild], "BandId"))
    1112             :         {
    1113             :             const char *pszValue =
    1114          12 :                 poRI->papoItemChild[iChild]->Find("Value", nullptr);
    1115             : 
    1116          12 :             iBand++;
    1117          12 :             if (pszValue)
    1118             :             {
    1119          12 :                 CPLPushErrorHandler(CPLQuietErrorHandler);
    1120          12 :                 poDS->GetRasterBand(iBand)->SetDescription(pszValue);
    1121          12 :                 CPLPopErrorHandler();
    1122             :             }
    1123             : 
    1124          12 :             pszValue = poRI->papoItemChild[iChild]->Find("Units", nullptr);
    1125          12 :             if (pszValue)
    1126             :             {
    1127           4 :                 CPLPushErrorHandler(CPLQuietErrorHandler);
    1128           4 :                 poDS->GetRasterBand(iBand)->SetUnitType(pszValue);
    1129           4 :                 CPLPopErrorHandler();
    1130             :             }
    1131             :         }
    1132             :     }
    1133             : 
    1134             :     /* -------------------------------------------------------------------- */
    1135             :     /*      Look for projection.                                            */
    1136             :     /* -------------------------------------------------------------------- */
    1137          61 :     poDS->osProj = poHeader->Find("CoordinateSpace.Projection", "");
    1138          61 :     poDS->osDatum = poHeader->Find("CoordinateSpace.Datum", "");
    1139          61 :     poDS->osUnits = poHeader->Find("CoordinateSpace.Units", "");
    1140             : 
    1141         219 :     poDS->m_oSRS.importFromERM(
    1142          61 :         !poDS->osProj.empty() ? poDS->osProj.c_str() : "RAW",
    1143          61 :         !poDS->osDatum.empty() ? poDS->osDatum.c_str() : "WGS84",
    1144          61 :         !poDS->osUnits.empty() ? poDS->osUnits.c_str() : "METERS");
    1145             : 
    1146             :     /* -------------------------------------------------------------------- */
    1147             :     /*      Look for the geotransform.                                      */
    1148             :     /* -------------------------------------------------------------------- */
    1149          61 :     if (poHeader->Find("RasterInfo.RegistrationCoord.Eastings", nullptr))
    1150             :     {
    1151           6 :         poDS->bGotTransform = TRUE;
    1152           6 :         poDS->adfGeoTransform[0] = CPLAtof(
    1153             :             poHeader->Find("RasterInfo.RegistrationCoord.Eastings", ""));
    1154          12 :         poDS->adfGeoTransform[1] =
    1155           6 :             CPLAtof(poHeader->Find("RasterInfo.CellInfo.Xdimension", "1.0"));
    1156           6 :         poDS->adfGeoTransform[2] = 0.0;
    1157           6 :         poDS->adfGeoTransform[3] = CPLAtof(
    1158             :             poHeader->Find("RasterInfo.RegistrationCoord.Northings", ""));
    1159           6 :         poDS->adfGeoTransform[4] = 0.0;
    1160           6 :         poDS->adfGeoTransform[5] =
    1161           6 :             -CPLAtof(poHeader->Find("RasterInfo.CellInfo.Ydimension", "1.0"));
    1162             :     }
    1163          60 :     else if (poHeader->Find("RasterInfo.RegistrationCoord.Latitude", nullptr) &&
    1164           5 :              poHeader->Find("RasterInfo.CellInfo.Xdimension", nullptr))
    1165             :     {
    1166           5 :         poDS->bGotTransform = TRUE;
    1167           5 :         poDS->adfGeoTransform[0] = ERSDMS2Dec(
    1168             :             poHeader->Find("RasterInfo.RegistrationCoord.Longitude", ""));
    1169          10 :         poDS->adfGeoTransform[1] =
    1170           5 :             CPLAtof(poHeader->Find("RasterInfo.CellInfo.Xdimension", ""));
    1171           5 :         poDS->adfGeoTransform[2] = 0.0;
    1172           5 :         poDS->adfGeoTransform[3] = ERSDMS2Dec(
    1173             :             poHeader->Find("RasterInfo.RegistrationCoord.Latitude", ""));
    1174           5 :         poDS->adfGeoTransform[4] = 0.0;
    1175           5 :         poDS->adfGeoTransform[5] =
    1176           5 :             -CPLAtof(poHeader->Find("RasterInfo.CellInfo.Ydimension", ""));
    1177             :     }
    1178             : 
    1179             :     /* -------------------------------------------------------------------- */
    1180             :     /*      Adjust if we have a registration cell.                          */
    1181             :     /* -------------------------------------------------------------------- */
    1182             : 
    1183             :     /* http://geospatial.intergraph.com/Libraries/Tech_Docs/ERDAS_ER_Mapper_Customization_Guide.sflb.ashx
    1184             :      */
    1185             :     /* Page 27 : */
    1186             :     /* RegistrationCellX and RegistrationCellY : The image X and Y
    1187             :        coordinates of the cell which corresponds to the Registration
    1188             :        Coordinate. Note that the RegistrationCellX and
    1189             :        RegistrationCellY can be fractional values. If
    1190             :        RegistrationCellX and RegistrationCellY are not specified,
    1191             :        they are assumed to be (0,0), which is the top left corner of the
    1192             :        image.
    1193             :        */
    1194             :     double dfCellX =
    1195          61 :         CPLAtof(poHeader->Find("RasterInfo.RegistrationCellX", "0"));
    1196             :     double dfCellY =
    1197          61 :         CPLAtof(poHeader->Find("RasterInfo.RegistrationCellY", "0"));
    1198             : 
    1199          61 :     if (poDS->bGotTransform)
    1200             :     {
    1201          11 :         poDS->adfGeoTransform[0] -= dfCellX * poDS->adfGeoTransform[1] +
    1202          11 :                                     dfCellY * poDS->adfGeoTransform[2];
    1203          22 :         poDS->adfGeoTransform[3] -= dfCellX * poDS->adfGeoTransform[4] +
    1204          11 :                                     dfCellY * poDS->adfGeoTransform[5];
    1205             :     }
    1206             : 
    1207             :     /* -------------------------------------------------------------------- */
    1208             :     /*      Check for null values.                                          */
    1209             :     /* -------------------------------------------------------------------- */
    1210          61 :     if (poHeader->Find("RasterInfo.NullCellValue", nullptr))
    1211             :     {
    1212           8 :         poDS->bHasNoDataValue = TRUE;
    1213          16 :         poDS->dfNoDataValue =
    1214           8 :             CPLAtofM(poHeader->Find("RasterInfo.NullCellValue"));
    1215             : 
    1216           8 :         if (poDS->poDepFile != nullptr)
    1217             :         {
    1218           0 :             CPLPushErrorHandler(CPLQuietErrorHandler);
    1219             : 
    1220           0 :             for (int iBand = 1; iBand <= poDS->nBands; iBand++)
    1221           0 :                 poDS->GetRasterBand(iBand)->SetNoDataValue(poDS->dfNoDataValue);
    1222             : 
    1223           0 :             CPLPopErrorHandler();
    1224             :         }
    1225             :     }
    1226             : 
    1227             :     /* -------------------------------------------------------------------- */
    1228             :     /*      Do we have an "All" region?                                     */
    1229             :     /* -------------------------------------------------------------------- */
    1230          61 :     ERSHdrNode *poAll = nullptr;
    1231             : 
    1232         366 :     for (int iChild = 0; poRI != nullptr && iChild < poRI->nItemCount; iChild++)
    1233             :     {
    1234         305 :         if (poRI->papoItemChild[iChild] != nullptr &&
    1235          39 :             EQUAL(poRI->papszItemName[iChild], "RegionInfo"))
    1236             :         {
    1237           3 :             if (EQUAL(poRI->papoItemChild[iChild]->Find("RegionName", ""),
    1238             :                       "All"))
    1239           3 :                 poAll = poRI->papoItemChild[iChild];
    1240             :         }
    1241             :     }
    1242             : 
    1243             :     /* -------------------------------------------------------------------- */
    1244             :     /*      Do we have statistics?                                          */
    1245             :     /* -------------------------------------------------------------------- */
    1246          61 :     if (poAll && poAll->FindNode("Stats"))
    1247             :     {
    1248           3 :         CPLPushErrorHandler(CPLQuietErrorHandler);
    1249             : 
    1250           6 :         for (int iBand = 1; iBand <= poDS->nBands; iBand++)
    1251             :         {
    1252             :             const char *pszValue =
    1253           3 :                 poAll->FindElem("Stats.MinimumValue", iBand - 1);
    1254             : 
    1255           3 :             if (pszValue)
    1256           3 :                 poDS->GetRasterBand(iBand)->SetMetadataItem(
    1257           3 :                     "STATISTICS_MINIMUM", pszValue);
    1258             : 
    1259           3 :             pszValue = poAll->FindElem("Stats.MaximumValue", iBand - 1);
    1260             : 
    1261           3 :             if (pszValue)
    1262           3 :                 poDS->GetRasterBand(iBand)->SetMetadataItem(
    1263           3 :                     "STATISTICS_MAXIMUM", pszValue);
    1264             : 
    1265           3 :             pszValue = poAll->FindElem("Stats.MeanValue", iBand - 1);
    1266             : 
    1267           3 :             if (pszValue)
    1268           3 :                 poDS->GetRasterBand(iBand)->SetMetadataItem("STATISTICS_MEAN",
    1269           3 :                                                             pszValue);
    1270             : 
    1271           3 :             pszValue = poAll->FindElem("Stats.MedianValue", iBand - 1);
    1272             : 
    1273           3 :             if (pszValue)
    1274           3 :                 poDS->GetRasterBand(iBand)->SetMetadataItem("STATISTICS_MEDIAN",
    1275           3 :                                                             pszValue);
    1276             :         }
    1277             : 
    1278           3 :         CPLPopErrorHandler();
    1279             :     }
    1280             : 
    1281             :     /* -------------------------------------------------------------------- */
    1282             :     /*      Do we have GCPs.                                                */
    1283             :     /* -------------------------------------------------------------------- */
    1284          61 :     if (poHeader->FindNode("RasterInfo.WarpControl"))
    1285           2 :         poDS->ReadGCPs();
    1286             : 
    1287             :     /* -------------------------------------------------------------------- */
    1288             :     /*      Initialize any PAM information.                                 */
    1289             :     /* -------------------------------------------------------------------- */
    1290          61 :     poDS->SetDescription(poOpenInfo->pszFilename);
    1291          61 :     poDS->TryLoadXML();
    1292             : 
    1293             :     // if no SR in xml, try aux
    1294          61 :     const OGRSpatialReference *poSRS = poDS->GDALPamDataset::GetSpatialRef();
    1295          61 :     if (poSRS == nullptr)
    1296             :     {
    1297             :         // try aux
    1298             :         auto poAuxDS = std::unique_ptr<GDALDataset>(GDALFindAssociatedAuxFile(
    1299         122 :             poOpenInfo->pszFilename, GA_ReadOnly, poDS.get()));
    1300          61 :         if (poAuxDS)
    1301             :         {
    1302           0 :             poSRS = poAuxDS->GetSpatialRef();
    1303           0 :             if (poSRS)
    1304             :             {
    1305           0 :                 poDS->m_oSRS = *poSRS;
    1306             :             }
    1307             :         }
    1308             :     }
    1309             :     /* -------------------------------------------------------------------- */
    1310             :     /*      Check for overviews.                                            */
    1311             :     /* -------------------------------------------------------------------- */
    1312          61 :     poDS->oOvManager.Initialize(poDS.get(), poOpenInfo->pszFilename);
    1313             : 
    1314          61 :     return poDS.release();
    1315             : }
    1316             : 
    1317             : /************************************************************************/
    1318             : /*                               Create()                               */
    1319             : /************************************************************************/
    1320             : 
    1321          67 : GDALDataset *ERSDataset::Create(const char *pszFilename, int nXSize, int nYSize,
    1322             :                                 int nBandsIn, GDALDataType eType,
    1323             :                                 char **papszOptions)
    1324             : 
    1325             : {
    1326             :     /* -------------------------------------------------------------------- */
    1327             :     /*      Verify settings.                                                */
    1328             :     /* -------------------------------------------------------------------- */
    1329          67 :     if (nBandsIn <= 0)
    1330             :     {
    1331           1 :         CPLError(CE_Failure, CPLE_NotSupported,
    1332             :                  "ERS driver does not support %d bands.\n", nBandsIn);
    1333           1 :         return nullptr;
    1334             :     }
    1335             : 
    1336          66 :     if (eType != GDT_Byte && eType != GDT_Int8 && eType != GDT_Int16 &&
    1337          29 :         eType != GDT_UInt16 && eType != GDT_Int32 && eType != GDT_UInt32 &&
    1338          19 :         eType != GDT_Float32 && eType != GDT_Float64)
    1339             :     {
    1340          16 :         CPLError(
    1341             :             CE_Failure, CPLE_AppDefined,
    1342             :             "The ERS driver does not supporting creating files of types %s.",
    1343             :             GDALGetDataTypeName(eType));
    1344          16 :         return nullptr;
    1345             :     }
    1346             : 
    1347             :     /* -------------------------------------------------------------------- */
    1348             :     /*      Work out the name we want to use for the .ers and binary        */
    1349             :     /*      data files.                                                     */
    1350             :     /* -------------------------------------------------------------------- */
    1351         100 :     CPLString osBinFile, osErsFile;
    1352             : 
    1353          50 :     if (EQUAL(CPLGetExtensionSafe(pszFilename).c_str(), "ers"))
    1354             :     {
    1355           7 :         osErsFile = pszFilename;
    1356           7 :         osBinFile = osErsFile.substr(0, osErsFile.length() - 4);
    1357             :     }
    1358             :     else
    1359             :     {
    1360          43 :         osBinFile = pszFilename;
    1361          43 :         osErsFile = osBinFile + ".ers";
    1362             :     }
    1363             : 
    1364             :     /* -------------------------------------------------------------------- */
    1365             :     /*      Work out some values we will write.                             */
    1366             :     /* -------------------------------------------------------------------- */
    1367          50 :     const char *pszCellType = "Unsigned8BitInteger";
    1368          50 :     CPL_IGNORE_RET_VAL(pszCellType);  // Make CSA happy
    1369             : 
    1370          50 :     if (eType == GDT_Byte)
    1371          28 :         pszCellType = "Unsigned8BitInteger";
    1372          22 :     else if (eType == GDT_Int8)
    1373           3 :         pszCellType = "Signed8BitInteger";
    1374          19 :     else if (eType == GDT_Int16)
    1375           3 :         pszCellType = "Signed16BitInteger";
    1376          16 :     else if (eType == GDT_UInt16)
    1377           3 :         pszCellType = "Unsigned16BitInteger";
    1378          13 :     else if (eType == GDT_Int32)
    1379           3 :         pszCellType = "Signed32BitInteger";
    1380          10 :     else if (eType == GDT_UInt32)
    1381           3 :         pszCellType = "Unsigned32BitInteger";
    1382           7 :     else if (eType == GDT_Float32)
    1383           4 :         pszCellType = "IEEE4ByteReal";
    1384           3 :     else if (eType == GDT_Float64)
    1385           3 :         pszCellType = "IEEE8ByteReal";
    1386             :     else
    1387             :     {
    1388           0 :         CPLAssert(false);
    1389             :     }
    1390             : 
    1391             :     /* -------------------------------------------------------------------- */
    1392             :     /*      Handling for signed eight bit data.                             */
    1393             :     /* -------------------------------------------------------------------- */
    1394          50 :     const char *pszPixelType = CSLFetchNameValue(papszOptions, "PIXELTYPE");
    1395          50 :     if (pszPixelType && EQUAL(pszPixelType, "SIGNEDBYTE") && eType == GDT_Byte)
    1396           0 :         pszCellType = "Signed8BitInteger";
    1397             : 
    1398             :     /* -------------------------------------------------------------------- */
    1399             :     /*      Write binary file.                                              */
    1400             :     /* -------------------------------------------------------------------- */
    1401          50 :     VSILFILE *fpBin = VSIFOpenL(osBinFile, "w");
    1402             : 
    1403          50 :     if (fpBin == nullptr)
    1404             :     {
    1405           3 :         CPLError(CE_Failure, CPLE_FileIO, "Failed to create %s:\n%s",
    1406           3 :                  osBinFile.c_str(), VSIStrerror(errno));
    1407           3 :         return nullptr;
    1408             :     }
    1409             : 
    1410             :     GUIntBig nSize =
    1411          47 :         nXSize * (GUIntBig)nYSize * nBandsIn * (GDALGetDataTypeSize(eType) / 8);
    1412          47 :     GByte byZero = 0;
    1413          94 :     if (VSIFSeekL(fpBin, nSize - 1, SEEK_SET) != 0 ||
    1414          47 :         VSIFWriteL(&byZero, 1, 1, fpBin) != 1)
    1415             :     {
    1416          10 :         CPLError(CE_Failure, CPLE_FileIO, "Failed to write %s:\n%s",
    1417          10 :                  osBinFile.c_str(), VSIStrerror(errno));
    1418          10 :         VSIFCloseL(fpBin);
    1419          10 :         return nullptr;
    1420             :     }
    1421          37 :     VSIFCloseL(fpBin);
    1422             : 
    1423             :     /* -------------------------------------------------------------------- */
    1424             :     /*      Try writing header file.                                        */
    1425             :     /* -------------------------------------------------------------------- */
    1426          37 :     VSILFILE *fpERS = VSIFOpenL(osErsFile, "w");
    1427             : 
    1428          37 :     if (fpERS == nullptr)
    1429             :     {
    1430           0 :         CPLError(CE_Failure, CPLE_FileIO, "Failed to create %s:\n%s",
    1431           0 :                  osErsFile.c_str(), VSIStrerror(errno));
    1432           0 :         return nullptr;
    1433             :     }
    1434             : 
    1435          37 :     VSIFPrintfL(fpERS, "DatasetHeader Begin\n");
    1436          37 :     VSIFPrintfL(fpERS, "\tVersion\t\t = \"6.0\"\n");
    1437          37 :     VSIFPrintfL(fpERS, "\tName\t\t= \"%s\"\n", CPLGetFilename(osErsFile));
    1438             : 
    1439             :     // Last updated requires timezone info which we don't necessarily get
    1440             :     // get from VSICTime() so perhaps it is better to omit this.
    1441             :     //    VSIFPrintfL( fpERS, "\tLastUpdated\t= %s",
    1442             :     //                 VSICTime( VSITime( NULL ) ) );
    1443             : 
    1444          37 :     VSIFPrintfL(fpERS, "\tDataSetType\t= ERStorage\n");
    1445          37 :     VSIFPrintfL(fpERS, "\tDataType\t= Raster\n");
    1446          37 :     VSIFPrintfL(fpERS, "\tByteOrder\t= LSBFirst\n");
    1447          37 :     VSIFPrintfL(fpERS, "\tRasterInfo Begin\n");
    1448          37 :     VSIFPrintfL(fpERS, "\t\tCellType\t= %s\n", pszCellType);
    1449          37 :     VSIFPrintfL(fpERS, "\t\tNrOfLines\t= %d\n", nYSize);
    1450          37 :     VSIFPrintfL(fpERS, "\t\tNrOfCellsPerLine\t= %d\n", nXSize);
    1451          37 :     VSIFPrintfL(fpERS, "\t\tNrOfBands\t= %d\n", nBandsIn);
    1452          37 :     VSIFPrintfL(fpERS, "\tRasterInfo End\n");
    1453          37 :     if (VSIFPrintfL(fpERS, "DatasetHeader End\n") < 17)
    1454             :     {
    1455           0 :         CPLError(CE_Failure, CPLE_FileIO, "Failed to write %s:\n%s",
    1456           0 :                  osErsFile.c_str(), VSIStrerror(errno));
    1457           0 :         return nullptr;
    1458             :     }
    1459             : 
    1460          37 :     VSIFCloseL(fpERS);
    1461             : 
    1462             :     /* -------------------------------------------------------------------- */
    1463             :     /*      Reopen.                                                         */
    1464             :     /* -------------------------------------------------------------------- */
    1465          74 :     GDALOpenInfo oOpenInfo(osErsFile, GA_Update);
    1466          37 :     ERSDataset *poDS = (ERSDataset *)Open(&oOpenInfo);
    1467          37 :     if (poDS == nullptr)
    1468           0 :         return nullptr;
    1469             : 
    1470             :     /* -------------------------------------------------------------------- */
    1471             :     /*      Fetch DATUM, PROJ and UNITS creation option                     */
    1472             :     /* -------------------------------------------------------------------- */
    1473          37 :     const char *pszDatum = CSLFetchNameValue(papszOptions, "DATUM");
    1474          37 :     if (pszDatum)
    1475             :     {
    1476           2 :         poDS->osDatumForced = pszDatum;
    1477           2 :         poDS->osDatum = pszDatum;
    1478             :     }
    1479          37 :     const char *pszProj = CSLFetchNameValue(papszOptions, "PROJ");
    1480          37 :     if (pszProj)
    1481             :     {
    1482           2 :         poDS->osProjForced = pszProj;
    1483           2 :         poDS->osProj = pszProj;
    1484             :     }
    1485          37 :     const char *pszUnits = CSLFetchNameValue(papszOptions, "UNITS");
    1486          37 :     if (pszUnits)
    1487             :     {
    1488           2 :         poDS->osUnitsForced = pszUnits;
    1489           2 :         poDS->osUnits = pszUnits;
    1490             :     }
    1491             : 
    1492          37 :     if (pszDatum || pszProj || pszUnits)
    1493             :     {
    1494           2 :         poDS->WriteProjectionInfo(pszProj ? pszProj : "RAW",
    1495             :                                   pszDatum ? pszDatum : "RAW",
    1496             :                                   pszUnits ? pszUnits : "METERS");
    1497             :     }
    1498             : 
    1499          37 :     return poDS;
    1500             : }
    1501             : 
    1502             : /************************************************************************/
    1503             : /*                         GDALRegister_ERS()                           */
    1504             : /************************************************************************/
    1505             : 
    1506        1682 : void GDALRegister_ERS()
    1507             : 
    1508             : {
    1509        1682 :     if (GDALGetDriverByName("ERS") != nullptr)
    1510         301 :         return;
    1511             : 
    1512        1381 :     GDALDriver *poDriver = new GDALDriver();
    1513             : 
    1514        1381 :     poDriver->SetDescription("ERS");
    1515        1381 :     poDriver->SetMetadataItem(GDAL_DCAP_RASTER, "YES");
    1516        1381 :     poDriver->SetMetadataItem(GDAL_DMD_LONGNAME, "ERMapper .ers Labelled");
    1517        1381 :     poDriver->SetMetadataItem(GDAL_DMD_HELPTOPIC, "drivers/raster/ers.html");
    1518        1381 :     poDriver->SetMetadataItem(GDAL_DMD_EXTENSION, "ers");
    1519        1381 :     poDriver->SetMetadataItem(GDAL_DMD_CREATIONDATATYPES,
    1520             :                               "Byte Int8 Int16 UInt16 Int32 UInt32 "
    1521        1381 :                               "Float32 Float64");
    1522             : 
    1523        1381 :     poDriver->SetMetadataItem(
    1524             :         GDAL_DMD_CREATIONOPTIONLIST,
    1525             :         "<CreationOptionList>"
    1526             :         "   <Option name='PIXELTYPE' type='string' description='(deprecated, "
    1527             :         "use Int8 datatype) By setting this to SIGNEDBYTE, a new Byte file can "
    1528             :         "be forced to be written as signed byte'/>"
    1529             :         "   <Option name='PROJ' type='string' description='ERS Projection "
    1530             :         "Name'/>"
    1531             :         "   <Option name='DATUM' type='string' description='ERS Datum Name' />"
    1532             :         "   <Option name='UNITS' type='string-select' description='ERS "
    1533             :         "Projection Units'>"
    1534             :         "       <Value>METERS</Value>"
    1535             :         "       <Value>FEET</Value>"
    1536             :         "   </Option>"
    1537        1381 :         "</CreationOptionList>");
    1538             : 
    1539        1381 :     poDriver->SetMetadataItem(GDAL_DCAP_VIRTUALIO, "YES");
    1540             : 
    1541        1381 :     poDriver->pfnOpen = ERSDataset::Open;
    1542        1381 :     poDriver->pfnIdentify = ERSDataset::Identify;
    1543        1381 :     poDriver->pfnCreate = ERSDataset::Create;
    1544             : 
    1545        1381 :     GetGDALDriverManager()->RegisterDriver(poDriver);
    1546             : }

Generated by: LCOV version 1.14