LCOV - code coverage report
Current view: top level - frmts/zmap - zmapdataset.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 277 340 81.5 %
Date: 2025-01-18 12:42:00 Functions: 14 14 100.0 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  ZMap driver
       4             :  * Purpose:  GDALDataset driver for ZMap dataset.
       5             :  * Author:   Even Rouault, <even dot rouault at spatialys.com>
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2011-2012, Even Rouault <even dot rouault at spatialys.com>
       9             :  *
      10             :  * SPDX-License-Identifier: MIT
      11             :  ****************************************************************************/
      12             : 
      13             : #include "cpl_string.h"
      14             : #include "cpl_vsi_virtual.h"
      15             : #include "gdal_frmts.h"
      16             : #include "gdal_pam.h"
      17             : 
      18             : #include <array>
      19             : #include <cmath>
      20             : #include <deque>
      21             : 
      22             : /************************************************************************/
      23             : /* ==================================================================== */
      24             : /*                              ZMapDataset                             */
      25             : /* ==================================================================== */
      26             : /************************************************************************/
      27             : 
      28             : class ZMapRasterBand;
      29             : 
      30          19 : class ZMapDataset final : public GDALPamDataset
      31             : {
      32             :     friend class ZMapRasterBand;
      33             : 
      34             :     VSIVirtualHandleUniquePtr m_fp{};
      35             :     int m_nValuesPerLine = 0;
      36             :     int m_nFieldSize = 0;
      37             :     int m_nDecimalCount = 0;
      38             :     int m_nColNum = -1;
      39             :     double m_dfNoDataValue = 0;
      40             :     vsi_l_offset m_nDataStartOff = 0;
      41             :     std::array<double, 6> m_adfGeoTransform = {{0, 1, 0, 0, 0, 1}};
      42             :     int m_nFirstDataLine = 0;
      43             :     int m_nCurLine = 0;
      44             :     std::deque<double> m_odfQueue{};
      45             : 
      46             :   public:
      47             :     ZMapDataset();
      48             :     virtual ~ZMapDataset();
      49             : 
      50             :     virtual CPLErr GetGeoTransform(double *) override;
      51             : 
      52             :     static GDALDataset *Open(GDALOpenInfo *);
      53             :     static int Identify(GDALOpenInfo *);
      54             :     static GDALDataset *CreateCopy(const char *pszFilename,
      55             :                                    GDALDataset *poSrcDS, int bStrict,
      56             :                                    char **papszOptions,
      57             :                                    GDALProgressFunc pfnProgress,
      58             :                                    void *pProgressData);
      59             : };
      60             : 
      61             : /************************************************************************/
      62             : /* ==================================================================== */
      63             : /*                            ZMapRasterBand                            */
      64             : /* ==================================================================== */
      65             : /************************************************************************/
      66             : 
      67             : class ZMapRasterBand final : public GDALPamRasterBand
      68             : {
      69             :     friend class ZMapDataset;
      70             : 
      71             :   public:
      72             :     explicit ZMapRasterBand(ZMapDataset *);
      73             : 
      74             :     virtual CPLErr IReadBlock(int, int, void *) override;
      75             :     virtual double GetNoDataValue(int *pbSuccess = nullptr) override;
      76             : };
      77             : 
      78             : /************************************************************************/
      79             : /*                           ZMapRasterBand()                           */
      80             : /************************************************************************/
      81             : 
      82          19 : ZMapRasterBand::ZMapRasterBand(ZMapDataset *poDSIn)
      83             : 
      84             : {
      85          19 :     poDS = poDSIn;
      86          19 :     nBand = 1;
      87             : 
      88          19 :     eDataType = GDT_Float64;
      89             : 
      90             :     // The format is column oriented! That is we have first the value of
      91             :     // pixel (col=0, line=0), then the one of (col=0, line=1), etc.
      92          19 :     nBlockXSize = 1;
      93          19 :     nBlockYSize = poDSIn->GetRasterYSize();
      94          19 : }
      95             : 
      96             : /************************************************************************/
      97             : /*                             IReadBlock()                             */
      98             : /************************************************************************/
      99             : 
     100          42 : CPLErr ZMapRasterBand::IReadBlock(int nBlockXOff, CPL_UNUSED int nBlockYOff,
     101             :                                   void *pImage)
     102             : {
     103          42 :     ZMapDataset *poGDS = cpl::down_cast<ZMapDataset *>(poDS);
     104             : 
     105             :     // If seeking backwards in term of columns, reset reading to the first
     106             :     // column
     107          42 :     if (nBlockXOff < poGDS->m_nColNum + 1)
     108             :     {
     109           0 :         poGDS->m_fp->Seek(poGDS->m_nDataStartOff, SEEK_SET);
     110           0 :         poGDS->m_nColNum = -1;
     111           0 :         poGDS->m_nCurLine = poGDS->m_nFirstDataLine;
     112           0 :         poGDS->m_odfQueue.clear();
     113             :     }
     114             : 
     115          42 :     if (nBlockXOff > poGDS->m_nColNum + 1)
     116             :     {
     117           0 :         for (int i = poGDS->m_nColNum + 1; i < nBlockXOff; i++)
     118             :         {
     119           0 :             if (IReadBlock(i, 0, nullptr) != CE_None)
     120           0 :                 return CE_Failure;
     121             :         }
     122             :     }
     123             : 
     124          42 :     int iRow = 0;
     125          42 :     const double dfExp = std::pow(10.0, poGDS->m_nDecimalCount);
     126          42 :     double *padfImage = reinterpret_cast<double *>(pImage);
     127             : 
     128             :     // If we have previously read too many values, start by consuming the
     129             :     // queue
     130          45 :     while (iRow < nRasterYSize && !poGDS->m_odfQueue.empty())
     131             :     {
     132           3 :         if (padfImage)
     133           3 :             padfImage[iRow] = poGDS->m_odfQueue.front();
     134           3 :         ++iRow;
     135           3 :         poGDS->m_odfQueue.pop_front();
     136             :     }
     137             : 
     138             :     // Now read as many lines as needed to finish filling the column buffer
     139         245 :     while (iRow < nRasterYSize)
     140             :     {
     141         203 :         constexpr int MARGIN = 16;  // Should be at least 2 for \r\n
     142         203 :         char *pszLine = const_cast<char *>(CPLReadLine2L(
     143             :             poGDS->m_fp.get(),
     144         203 :             poGDS->m_nValuesPerLine * poGDS->m_nFieldSize + MARGIN, nullptr));
     145         203 :         ++poGDS->m_nCurLine;
     146         203 :         if (pszLine == nullptr)
     147           0 :             return CE_Failure;
     148             : 
     149             :         // Each line should have at most m_nValuesPerLine values of size
     150             :         // m_nFieldSize
     151         203 :         const int nLineLen = static_cast<int>(strlen(pszLine));
     152         203 :         if ((nLineLen % poGDS->m_nFieldSize) != 0)
     153             :         {
     154           0 :             CPLError(CE_Failure, CPLE_AppDefined,
     155             :                      "Line %d has length %d, which is not a multiple of %d",
     156             :                      poGDS->m_nCurLine, nLineLen, poGDS->m_nFieldSize);
     157           0 :             return CE_Failure;
     158             :         }
     159             : 
     160         203 :         const int nValuesThisLine = nLineLen / poGDS->m_nFieldSize;
     161         203 :         if (nValuesThisLine > poGDS->m_nValuesPerLine)
     162             :         {
     163           0 :             CPLError(
     164             :                 CE_Failure, CPLE_AppDefined,
     165             :                 "Line %d has %d values, whereas the maximum expected is %d",
     166             :                 poGDS->m_nCurLine, nValuesThisLine, poGDS->m_nValuesPerLine);
     167           0 :             return CE_Failure;
     168             :         }
     169             : 
     170        1013 :         for (int iValueThisLine = 0; iValueThisLine < nValuesThisLine;
     171             :              iValueThisLine++)
     172             :         {
     173         810 :             char *pszValue = pszLine + iValueThisLine * poGDS->m_nFieldSize;
     174         810 :             const char chSaved = pszValue[poGDS->m_nFieldSize];
     175         810 :             pszValue[poGDS->m_nFieldSize] = 0;
     176         810 :             const double dfVal = strchr(pszValue, '.') != nullptr
     177         810 :                                      ? CPLAtofM(pszValue)
     178           0 :                                      : atoi(pszValue) * dfExp;
     179         810 :             pszValue[poGDS->m_nFieldSize] = chSaved;
     180         810 :             if (iRow < nRasterYSize)
     181             :             {
     182         807 :                 if (padfImage)
     183         807 :                     padfImage[iRow] = dfVal;
     184         807 :                 ++iRow;
     185             :             }
     186             :             else
     187             :             {
     188           3 :                 poGDS->m_odfQueue.push_back(dfVal);
     189             :             }
     190             :         }
     191             :     }
     192             : 
     193          42 :     poGDS->m_nColNum++;
     194             : 
     195          42 :     return CE_None;
     196             : }
     197             : 
     198             : /************************************************************************/
     199             : /*                          GetNoDataValue()                            */
     200             : /************************************************************************/
     201             : 
     202           3 : double ZMapRasterBand::GetNoDataValue(int *pbSuccess)
     203             : {
     204           3 :     ZMapDataset *poGDS = cpl::down_cast<ZMapDataset *>(poDS);
     205             : 
     206           3 :     if (pbSuccess)
     207           3 :         *pbSuccess = TRUE;
     208             : 
     209           3 :     return poGDS->m_dfNoDataValue;
     210             : }
     211             : 
     212             : /************************************************************************/
     213             : /*                            ~ZMapDataset()                            */
     214             : /************************************************************************/
     215             : 
     216             : ZMapDataset::ZMapDataset() = default;
     217             : 
     218             : /************************************************************************/
     219             : /*                            ~ZMapDataset()                            */
     220             : /************************************************************************/
     221             : 
     222          38 : ZMapDataset::~ZMapDataset()
     223             : 
     224             : {
     225          19 :     FlushCache(true);
     226          38 : }
     227             : 
     228             : /************************************************************************/
     229             : /*                             Identify()                               */
     230             : /************************************************************************/
     231             : 
     232       51253 : int ZMapDataset::Identify(GDALOpenInfo *poOpenInfo)
     233             : {
     234       51253 :     if (poOpenInfo->nHeaderBytes == 0)
     235       48178 :         return FALSE;
     236             : 
     237             :     /* -------------------------------------------------------------------- */
     238             :     /*      Check that it looks roughly as a ZMap dataset                   */
     239             :     /* -------------------------------------------------------------------- */
     240        3075 :     const char *pszData =
     241             :         reinterpret_cast<const char *>(poOpenInfo->pabyHeader);
     242             : 
     243             :     /* Skip comments line at the beginning */
     244        3075 :     int i = 0;
     245        3075 :     if (pszData[i] == '!')
     246             :     {
     247          33 :         i++;
     248         706 :         for (; i < poOpenInfo->nHeaderBytes; i++)
     249             :         {
     250         706 :             char ch = pszData[i];
     251         706 :             if (ch == 13 || ch == 10)
     252             :             {
     253          89 :                 i++;
     254          89 :                 if (ch == 13 && pszData[i] == 10)
     255           0 :                     i++;
     256          89 :                 if (pszData[i] != '!')
     257          33 :                     break;
     258             :             }
     259             :         }
     260             :     }
     261             : 
     262        3075 :     if (pszData[i] != '@')
     263        3047 :         return FALSE;
     264          28 :     i++;
     265             : 
     266          56 :     const CPLStringList aosTokens(CSLTokenizeString2(pszData + i, ",", 0));
     267          28 :     if (aosTokens.size() < 3)
     268             :     {
     269           4 :         return FALSE;
     270             :     }
     271             : 
     272          24 :     const char *pszToken = aosTokens[1];
     273          48 :     while (*pszToken == ' ')
     274          24 :         pszToken++;
     275             : 
     276          24 :     return STARTS_WITH(pszToken, "GRID");
     277             : }
     278             : 
     279             : /************************************************************************/
     280             : /*                                Open()                                */
     281             : /************************************************************************/
     282             : 
     283          19 : GDALDataset *ZMapDataset::Open(GDALOpenInfo *poOpenInfo)
     284             : 
     285             : {
     286          19 :     if (!Identify(poOpenInfo) || poOpenInfo->fpL == nullptr)
     287           0 :         return nullptr;
     288             : 
     289             :     /* -------------------------------------------------------------------- */
     290             :     /*      Confirm the requested access is supported.                      */
     291             :     /* -------------------------------------------------------------------- */
     292          19 :     if (poOpenInfo->eAccess == GA_Update)
     293             :     {
     294           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     295             :                  "The ZMAP driver does not support update access to existing"
     296             :                  " datasets.");
     297           0 :         return nullptr;
     298             :     }
     299             : 
     300          38 :     auto poDS = std::make_unique<ZMapDataset>();
     301          19 :     poDS->m_fp.reset(poOpenInfo->fpL);
     302          19 :     poOpenInfo->fpL = nullptr;
     303             : 
     304             :     /* -------------------------------------------------------------------- */
     305             :     /*      Find dataset characteristics                                    */
     306             :     /* -------------------------------------------------------------------- */
     307             : 
     308             :     const char *pszLine;
     309          19 :     int nLine = 0;
     310          19 :     constexpr int MAX_HEADER_LINE = 1024;
     311          76 :     while ((pszLine = CPLReadLine2L(poDS->m_fp.get(), MAX_HEADER_LINE,
     312          76 :                                     nullptr)) != nullptr)
     313             :     {
     314          76 :         ++nLine;
     315          76 :         if (*pszLine == '!')
     316             :         {
     317          57 :             continue;
     318             :         }
     319             :         else
     320          19 :             break;
     321             :     }
     322             :     // cppcheck-suppress knownConditionTrueFalse
     323          19 :     if (pszLine == nullptr)
     324             :     {
     325           0 :         return nullptr;
     326             :     }
     327             : 
     328             :     /* Parse first header line */
     329          38 :     CPLStringList aosTokensFirstLine(CSLTokenizeString2(pszLine, ",", 0));
     330          19 :     if (aosTokensFirstLine.size() != 3)
     331             :     {
     332           0 :         return nullptr;
     333             :     }
     334             : 
     335          19 :     const int nValuesPerLine = atoi(aosTokensFirstLine[2]);
     336          19 :     if (nValuesPerLine <= 0)
     337             :     {
     338           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     339             :                  "Invalid/unsupported value for nValuesPerLine = %d",
     340             :                  nValuesPerLine);
     341           0 :         return nullptr;
     342             :     }
     343             : 
     344             :     /* Parse second header line */
     345          19 :     pszLine = CPLReadLine2L(poDS->m_fp.get(), MAX_HEADER_LINE, nullptr);
     346          19 :     ++nLine;
     347          19 :     if (pszLine == nullptr)
     348             :     {
     349           0 :         return nullptr;
     350             :     }
     351             :     const CPLStringList aosTokensSecondLine(
     352          38 :         CSLTokenizeString2(pszLine, ",", 0));
     353          19 :     if (aosTokensSecondLine.size() != 5)
     354             :     {
     355           0 :         return nullptr;
     356             :     }
     357             : 
     358          19 :     const int nFieldSize = atoi(aosTokensSecondLine[0]);
     359          19 :     const double dfNoDataValue = CPLAtofM(aosTokensSecondLine[1]);
     360          19 :     const int nDecimalCount = atoi(aosTokensSecondLine[3]);
     361          19 :     const int nColumnNumber = atoi(aosTokensSecondLine[4]);
     362             : 
     363          19 :     if (nFieldSize <= 0 || nFieldSize >= 40)
     364             :     {
     365           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     366             :                  "Invalid/unsupported value for nFieldSize = %d", nFieldSize);
     367           0 :         return nullptr;
     368             :     }
     369             : 
     370          19 :     if (nDecimalCount <= 0 || nDecimalCount >= nFieldSize)
     371             :     {
     372           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     373             :                  "Invalid/unsupported value for nDecimalCount = %d",
     374             :                  nDecimalCount);
     375           0 :         return nullptr;
     376             :     }
     377             : 
     378          19 :     if (nColumnNumber != 1)
     379             :     {
     380           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     381             :                  "Invalid/unsupported value for nColumnNumber = %d",
     382             :                  nColumnNumber);
     383           0 :         return nullptr;
     384             :     }
     385             : 
     386          19 :     if (nValuesPerLine <= 0 || nFieldSize > 1024 * 1024 / nValuesPerLine)
     387             :     {
     388           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     389             :                  "Invalid/unsupported value for nFieldSize = %d x "
     390             :                  "nValuesPerLine = %d",
     391             :                  nFieldSize, nValuesPerLine);
     392           0 :         return nullptr;
     393             :     }
     394             : 
     395             :     /* Parse third header line */
     396          19 :     pszLine = CPLReadLine2L(poDS->m_fp.get(), MAX_HEADER_LINE, nullptr);
     397          19 :     ++nLine;
     398          19 :     if (pszLine == nullptr)
     399             :     {
     400           0 :         return nullptr;
     401             :     }
     402          38 :     const CPLStringList aosTokensThirdLine(CSLTokenizeString2(pszLine, ",", 0));
     403          19 :     if (aosTokensThirdLine.size() != 6)
     404             :     {
     405           0 :         return nullptr;
     406             :     }
     407             : 
     408          19 :     const int nRows = atoi(aosTokensThirdLine[0]);
     409          19 :     const int nCols = atoi(aosTokensThirdLine[1]);
     410          19 :     const double dfMinX = CPLAtofM(aosTokensThirdLine[2]);
     411          19 :     const double dfMaxX = CPLAtofM(aosTokensThirdLine[3]);
     412          19 :     const double dfMinY = CPLAtofM(aosTokensThirdLine[4]);
     413          19 :     const double dfMaxY = CPLAtofM(aosTokensThirdLine[5]);
     414             : 
     415          19 :     if (!GDALCheckDatasetDimensions(nCols, nRows) || nCols == 1 || nRows == 1)
     416             :     {
     417           0 :         return nullptr;
     418             :     }
     419             : 
     420             :     /* Ignore fourth header line */
     421          19 :     pszLine = CPLReadLine2L(poDS->m_fp.get(), MAX_HEADER_LINE, nullptr);
     422          19 :     ++nLine;
     423          19 :     if (pszLine == nullptr)
     424             :     {
     425           0 :         return nullptr;
     426             :     }
     427             : 
     428             :     /* Check fifth header line */
     429          19 :     pszLine = CPLReadLine2L(poDS->m_fp.get(), MAX_HEADER_LINE, nullptr);
     430          19 :     ++nLine;
     431          19 :     if (pszLine == nullptr || pszLine[0] != '@')
     432             :     {
     433           0 :         return nullptr;
     434             :     }
     435             : 
     436             :     /* -------------------------------------------------------------------- */
     437             :     /*      Create a corresponding GDALDataset.                             */
     438             :     /* -------------------------------------------------------------------- */
     439          19 :     poDS->m_nDataStartOff = VSIFTellL(poDS->m_fp.get());
     440          19 :     poDS->m_nValuesPerLine = nValuesPerLine;
     441          19 :     poDS->m_nFieldSize = nFieldSize;
     442          19 :     poDS->m_nDecimalCount = nDecimalCount;
     443          19 :     poDS->nRasterXSize = nCols;
     444          19 :     poDS->nRasterYSize = nRows;
     445          19 :     poDS->m_dfNoDataValue = dfNoDataValue;
     446          19 :     poDS->m_nFirstDataLine = nLine;
     447             : 
     448          19 :     if (CPLTestBool(CPLGetConfigOption("ZMAP_PIXEL_IS_POINT", "FALSE")))
     449             :     {
     450           0 :         const double dfStepX = (dfMaxX - dfMinX) / (nCols - 1);
     451           0 :         const double dfStepY = (dfMaxY - dfMinY) / (nRows - 1);
     452             : 
     453           0 :         poDS->m_adfGeoTransform[0] = dfMinX - dfStepX / 2;
     454           0 :         poDS->m_adfGeoTransform[1] = dfStepX;
     455           0 :         poDS->m_adfGeoTransform[3] = dfMaxY + dfStepY / 2;
     456           0 :         poDS->m_adfGeoTransform[5] = -dfStepY;
     457             :     }
     458             :     else
     459             :     {
     460          19 :         const double dfStepX = (dfMaxX - dfMinX) / nCols;
     461          19 :         const double dfStepY = (dfMaxY - dfMinY) / nRows;
     462             : 
     463          19 :         poDS->m_adfGeoTransform[0] = dfMinX;
     464          19 :         poDS->m_adfGeoTransform[1] = dfStepX;
     465          19 :         poDS->m_adfGeoTransform[3] = dfMaxY;
     466          19 :         poDS->m_adfGeoTransform[5] = -dfStepY;
     467             :     }
     468             : 
     469             :     /* -------------------------------------------------------------------- */
     470             :     /*      Create band information objects.                                */
     471             :     /* -------------------------------------------------------------------- */
     472          19 :     poDS->nBands = 1;
     473          19 :     poDS->SetBand(1, std::make_unique<ZMapRasterBand>(poDS.get()));
     474             : 
     475             :     /* -------------------------------------------------------------------- */
     476             :     /*      Initialize any PAM information.                                 */
     477             :     /* -------------------------------------------------------------------- */
     478          19 :     poDS->SetDescription(poOpenInfo->pszFilename);
     479          19 :     poDS->TryLoadXML();
     480             : 
     481             :     /* -------------------------------------------------------------------- */
     482             :     /*      Support overviews.                                              */
     483             :     /* -------------------------------------------------------------------- */
     484          19 :     poDS->oOvManager.Initialize(poDS.get(), poOpenInfo->pszFilename);
     485          19 :     return poDS.release();
     486             : }
     487             : 
     488             : /************************************************************************/
     489             : /*                       WriteRightJustified()                          */
     490             : /************************************************************************/
     491             : 
     492        1668 : static void WriteRightJustified(VSIVirtualHandleUniquePtr &fp,
     493             :                                 const char *pszValue, int nWidth)
     494             : {
     495        1668 :     int nLen = (int)strlen(pszValue);
     496        1668 :     CPLAssert(nLen <= nWidth);
     497       16550 :     for (int i = 0; i < nWidth - nLen; i++)
     498       14882 :         fp->Write(" ", 1, 1);
     499        1668 :     fp->Write(pszValue, 1, nLen);
     500        1668 : }
     501             : 
     502          70 : static void WriteRightJustified(VSIVirtualHandleUniquePtr &fp, int nValue,
     503             :                                 int nWidth)
     504             : {
     505         140 :     CPLString osValue(CPLSPrintf("%d", nValue));
     506          70 :     WriteRightJustified(fp, osValue.c_str(), nWidth);
     507          70 : }
     508             : 
     509        1584 : static void WriteRightJustified(VSIVirtualHandleUniquePtr &fp, double dfValue,
     510             :                                 int nWidth, int nDecimals = -1)
     511             : {
     512             :     char szFormat[32];
     513        1584 :     if (nDecimals >= 0)
     514        1584 :         snprintf(szFormat, sizeof(szFormat), "%%.%df", nDecimals);
     515             :     else
     516           0 :         snprintf(szFormat, sizeof(szFormat), "%%g");
     517        1584 :     char *pszValue = const_cast<char *>(CPLSPrintf(szFormat, dfValue));
     518        1584 :     char *pszE = strchr(pszValue, 'e');
     519        1584 :     if (pszE)
     520           0 :         *pszE = 'E';
     521             : 
     522        1584 :     if (static_cast<int>(strlen(pszValue)) > nWidth)
     523             :     {
     524          16 :         CPLAssert(nDecimals >= 0);
     525          16 :         snprintf(szFormat, sizeof(szFormat), "%%.%dg", nDecimals);
     526          16 :         pszValue = const_cast<char *>(CPLSPrintf(szFormat, dfValue));
     527          16 :         pszE = strchr(pszValue, 'e');
     528          16 :         if (pszE)
     529          14 :             *pszE = 'E';
     530             :     }
     531             : 
     532        3168 :     CPLString osValue(pszValue);
     533        1584 :     WriteRightJustified(fp, osValue.c_str(), nWidth);
     534        1584 : }
     535             : 
     536             : /************************************************************************/
     537             : /*                             CreateCopy()                             */
     538             : /************************************************************************/
     539             : 
     540          22 : GDALDataset *ZMapDataset::CreateCopy(const char *pszFilename,
     541             :                                      GDALDataset *poSrcDS, int bStrict,
     542             :                                      CPL_UNUSED char **papszOptions,
     543             :                                      GDALProgressFunc pfnProgress,
     544             :                                      void *pProgressData)
     545             : {
     546             :     /* -------------------------------------------------------------------- */
     547             :     /*      Some some rudimentary checks                                    */
     548             :     /* -------------------------------------------------------------------- */
     549          22 :     int nBands = poSrcDS->GetRasterCount();
     550          22 :     if (nBands == 0)
     551             :     {
     552           1 :         CPLError(
     553             :             CE_Failure, CPLE_NotSupported,
     554             :             "ZMap driver does not support source dataset with zero band.\n");
     555           1 :         return nullptr;
     556             :     }
     557             : 
     558          21 :     if (nBands != 1)
     559             :     {
     560           4 :         CPLError((bStrict) ? CE_Failure : CE_Warning, CPLE_NotSupported,
     561             :                  "ZMap driver only uses the first band of the dataset.\n");
     562           4 :         if (bStrict)
     563           4 :             return nullptr;
     564             :     }
     565             : 
     566          17 :     if (pfnProgress && !pfnProgress(0.0, nullptr, pProgressData))
     567           0 :         return nullptr;
     568             : 
     569             :     /* -------------------------------------------------------------------- */
     570             :     /*      Get source dataset info                                         */
     571             :     /* -------------------------------------------------------------------- */
     572             : 
     573          17 :     const int nXSize = poSrcDS->GetRasterXSize();
     574          17 :     const int nYSize = poSrcDS->GetRasterYSize();
     575          17 :     if (nXSize == 1 || nYSize == 1)
     576             :     {
     577           0 :         return nullptr;
     578             :     }
     579             : 
     580             :     double adfGeoTransform[6];
     581          17 :     poSrcDS->GetGeoTransform(adfGeoTransform);
     582          17 :     if (adfGeoTransform[2] != 0 || adfGeoTransform[4] != 0)
     583             :     {
     584           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     585             :                  "ZMap driver does not support CreateCopy() from skewed or "
     586             :                  "rotated dataset.\n");
     587           0 :         return nullptr;
     588             :     }
     589             : 
     590             :     /* -------------------------------------------------------------------- */
     591             :     /*      Create target file                                              */
     592             :     /* -------------------------------------------------------------------- */
     593             : 
     594          34 :     auto fp = VSIVirtualHandleUniquePtr(VSIFOpenL(pszFilename, "wb"));
     595          17 :     if (fp == nullptr)
     596             :     {
     597           3 :         CPLError(CE_Failure, CPLE_AppDefined, "Cannot create %s", pszFilename);
     598           3 :         return nullptr;
     599             :     }
     600             : 
     601          14 :     const int nFieldSize = 20;
     602          14 :     const int nValuesPerLine = 4;
     603          14 :     const int nDecimalCount = 7;
     604             : 
     605          14 :     int bHasNoDataValue = FALSE;
     606             :     double dfNoDataValue =
     607          14 :         poSrcDS->GetRasterBand(1)->GetNoDataValue(&bHasNoDataValue);
     608          14 :     if (!bHasNoDataValue)
     609          13 :         dfNoDataValue = 1.e30;
     610             : 
     611          14 :     fp->Printf("!\n");
     612          14 :     fp->Printf("! Created by GDAL.\n");
     613          14 :     fp->Printf("!\n");
     614          14 :     fp->Printf("@GRID FILE, GRID, %d\n", nValuesPerLine);
     615             : 
     616          14 :     WriteRightJustified(fp, nFieldSize, 10);
     617          14 :     fp->Printf(",");
     618          14 :     WriteRightJustified(fp, dfNoDataValue, nFieldSize, nDecimalCount);
     619          14 :     fp->Printf(",");
     620          14 :     WriteRightJustified(fp, "", 10);
     621          14 :     fp->Printf(",");
     622          14 :     WriteRightJustified(fp, nDecimalCount, 10);
     623          14 :     fp->Printf(",");
     624          14 :     WriteRightJustified(fp, 1, 10);
     625          14 :     fp->Printf("\n");
     626             : 
     627          14 :     WriteRightJustified(fp, nYSize, 10);
     628          14 :     fp->Printf(",");
     629          14 :     WriteRightJustified(fp, nXSize, 10);
     630          14 :     fp->Printf(",");
     631             : 
     632          14 :     if (CPLTestBool(CPLGetConfigOption("ZMAP_PIXEL_IS_POINT", "FALSE")))
     633             :     {
     634           0 :         WriteRightJustified(fp, adfGeoTransform[0] + adfGeoTransform[1] / 2, 14,
     635             :                             7);
     636           0 :         fp->Printf(",");
     637           0 :         WriteRightJustified(fp,
     638           0 :                             adfGeoTransform[0] + adfGeoTransform[1] * nXSize -
     639           0 :                                 adfGeoTransform[1] / 2,
     640             :                             14, 7);
     641           0 :         fp->Printf(",");
     642           0 :         WriteRightJustified(fp,
     643           0 :                             adfGeoTransform[3] + adfGeoTransform[5] * nYSize -
     644           0 :                                 adfGeoTransform[5] / 2,
     645             :                             14, 7);
     646           0 :         fp->Printf(",");
     647           0 :         WriteRightJustified(fp, adfGeoTransform[3] + adfGeoTransform[5] / 2, 14,
     648             :                             7);
     649             :     }
     650             :     else
     651             :     {
     652          14 :         WriteRightJustified(fp, adfGeoTransform[0], 14, 7);
     653          14 :         fp->Printf(",");
     654          14 :         WriteRightJustified(
     655          14 :             fp, adfGeoTransform[0] + adfGeoTransform[1] * nXSize, 14, 7);
     656          14 :         fp->Printf(",");
     657          14 :         WriteRightJustified(
     658          14 :             fp, adfGeoTransform[3] + adfGeoTransform[5] * nYSize, 14, 7);
     659          14 :         fp->Printf(",");
     660          14 :         WriteRightJustified(fp, adfGeoTransform[3], 14, 7);
     661             :     }
     662             : 
     663          14 :     fp->Printf("\n");
     664             : 
     665          14 :     fp->Printf("0.0, 0.0, 0.0\n");
     666          14 :     fp->Printf("@\n");
     667             : 
     668             :     /* -------------------------------------------------------------------- */
     669             :     /*      Copy imagery                                                    */
     670             :     /* -------------------------------------------------------------------- */
     671          28 :     std::vector<double> adfLineBuffer(nYSize);
     672             : 
     673          14 :     CPLErr eErr = CE_None;
     674          14 :     const bool bEmitEOLAtEndOfColumn = CPLTestBool(
     675             :         CPLGetConfigOption("ZMAP_EMIT_EOL_AT_END_OF_COLUMN", "YES"));
     676          14 :     bool bEOLPrinted = false;
     677          14 :     int nValuesThisLine = 0;
     678         148 :     for (int i = 0; i < nXSize && eErr == CE_None; i++)
     679             :     {
     680         268 :         eErr = poSrcDS->GetRasterBand(1)->RasterIO(
     681         134 :             GF_Read, i, 0, 1, nYSize, adfLineBuffer.data(), 1, nYSize,
     682             :             GDT_Float64, 0, 0, nullptr);
     683         134 :         if (eErr != CE_None)
     684           0 :             break;
     685        1648 :         for (int j = 0; j < nYSize; j++)
     686             :         {
     687        1514 :             WriteRightJustified(fp, adfLineBuffer[j], nFieldSize,
     688             :                                 nDecimalCount);
     689        1514 :             ++nValuesThisLine;
     690        1514 :             if (nValuesThisLine == nValuesPerLine)
     691             :             {
     692         322 :                 bEOLPrinted = true;
     693         322 :                 nValuesThisLine = 0;
     694         322 :                 fp->Printf("\n");
     695             :             }
     696             :             else
     697        1192 :                 bEOLPrinted = false;
     698             :         }
     699         134 :         if (bEmitEOLAtEndOfColumn && !bEOLPrinted)
     700             :         {
     701         112 :             bEOLPrinted = true;
     702         112 :             nValuesThisLine = 0;
     703         112 :             fp->Printf("\n");
     704             :         }
     705             : 
     706         268 :         if (pfnProgress != nullptr &&
     707         134 :             !pfnProgress((i + 1) * 1.0 / nXSize, nullptr, pProgressData))
     708             :         {
     709           0 :             eErr = CE_Failure;
     710           0 :             break;
     711             :         }
     712             :     }
     713          14 :     if (!bEOLPrinted)
     714           1 :         fp->Printf("\n");
     715             : 
     716          14 :     if (eErr != CE_None || fp->Close() != 0)
     717           0 :         return nullptr;
     718             : 
     719          14 :     fp.reset();
     720          28 :     GDALOpenInfo oOpenInfo(pszFilename, GA_ReadOnly);
     721          14 :     return ZMapDataset::Open(&oOpenInfo);
     722             : }
     723             : 
     724             : /************************************************************************/
     725             : /*                          GetGeoTransform()                           */
     726             : /************************************************************************/
     727             : 
     728           1 : CPLErr ZMapDataset::GetGeoTransform(double *padfTransform)
     729             : 
     730             : {
     731           1 :     memcpy(padfTransform, m_adfGeoTransform.data(), 6 * sizeof(double));
     732             : 
     733           1 :     return CE_None;
     734             : }
     735             : 
     736             : /************************************************************************/
     737             : /*                         GDALRegister_ZMap()                          */
     738             : /************************************************************************/
     739             : 
     740        1682 : void GDALRegister_ZMap()
     741             : 
     742             : {
     743        1682 :     if (GDALGetDriverByName("ZMap") != nullptr)
     744         301 :         return;
     745             : 
     746        2762 :     auto poDriver = std::make_unique<GDALDriver>();
     747             : 
     748        1381 :     poDriver->SetDescription("ZMap");
     749        1381 :     poDriver->SetMetadataItem(GDAL_DCAP_RASTER, "YES");
     750        1381 :     poDriver->SetMetadataItem(GDAL_DMD_LONGNAME, "ZMap Plus Grid");
     751        1381 :     poDriver->SetMetadataItem(GDAL_DMD_HELPTOPIC, "drivers/raster/zmap.html");
     752        1381 :     poDriver->SetMetadataItem(GDAL_DMD_EXTENSION, "dat");
     753        1381 :     poDriver->SetMetadataItem(GDAL_DCAP_VIRTUALIO, "YES");
     754             : 
     755        1381 :     poDriver->pfnOpen = ZMapDataset::Open;
     756        1381 :     poDriver->pfnIdentify = ZMapDataset::Identify;
     757        1381 :     poDriver->pfnCreateCopy = ZMapDataset::CreateCopy;
     758             : 
     759        1381 :     GetGDALDriverManager()->RegisterDriver(poDriver.release());
     760             : }

Generated by: LCOV version 1.14