LCOV - code coverage report
Current view: top level - frmts/raw - gtxdataset.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 157 183 85.8 %
Date: 2025-01-18 12:42:00 Functions: 14 15 93.3 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  Vertical Datum Transformation
       4             :  * Purpose:  Implementation of NOAA .gtx vertical datum shift file format.
       5             :  * Author:   Frank Warmerdam, warmerdam@pobox.com
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2010, Frank Warmerdam
       9             :  * Copyright (c) 2010-2013, Even Rouault <even dot rouault at spatialys.com>
      10             :  *
      11             :  * SPDX-License-Identifier: MIT
      12             :  ****************************************************************************/
      13             : 
      14             : #include "cpl_string.h"
      15             : #include "gdal_frmts.h"
      16             : #include "ogr_srs_api.h"
      17             : #include "rawdataset.h"
      18             : 
      19             : #include <algorithm>
      20             : #include <limits>
      21             : 
      22             : /**
      23             : 
      24             : NOAA .GTX Vertical Datum Grid Shift Format
      25             : 
      26             : All values are bigendian
      27             : 
      28             : Header
      29             : ------
      30             : 
      31             : float64  latitude_of_origin
      32             : float64  longitude_of_origin (0-360)
      33             : float64  cell size (x?y?)
      34             : float64  cell size (x?y?)
      35             : int32    length in pixels
      36             : int32    width in pixels
      37             : 
      38             : Data
      39             : ----
      40             : 
      41             : float32  * width in pixels * length in pixels
      42             : 
      43             : Values are an offset in meters between two vertical datums.
      44             : 
      45             : **/
      46             : 
      47             : /************************************************************************/
      48             : /* ==================================================================== */
      49             : /*                              GTXDataset                              */
      50             : /* ==================================================================== */
      51             : /************************************************************************/
      52             : 
      53             : class GTXDataset final : public RawDataset
      54             : {
      55             :     VSILFILE *fpImage = nullptr;  // image data file.
      56             : 
      57             :     OGRSpatialReference m_oSRS{};
      58             :     double adfGeoTransform[6];
      59             : 
      60             :     CPL_DISALLOW_COPY_ASSIGN(GTXDataset)
      61             : 
      62             :     CPLErr Close() override;
      63             : 
      64             :   public:
      65          47 :     GTXDataset()
      66          47 :     {
      67          47 :         m_oSRS.SetFromUserInput(SRS_WKT_WGS84_LAT_LONG);
      68          47 :         m_oSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
      69             : 
      70          47 :         adfGeoTransform[0] = 0.0;
      71          47 :         adfGeoTransform[1] = 1.0;
      72          47 :         adfGeoTransform[2] = 0.0;
      73          47 :         adfGeoTransform[3] = 0.0;
      74          47 :         adfGeoTransform[4] = 0.0;
      75          47 :         adfGeoTransform[5] = 1.0;
      76          47 :     }
      77             : 
      78             :     ~GTXDataset() override;
      79             : 
      80             :     CPLErr GetGeoTransform(double *padfTransform) override;
      81             :     CPLErr SetGeoTransform(double *padfTransform) override;
      82             : 
      83           1 :     const OGRSpatialReference *GetSpatialRef() const override
      84             :     {
      85           1 :         return &m_oSRS;
      86             :     }
      87             : 
      88             :     static GDALDataset *Open(GDALOpenInfo *);
      89             :     static int Identify(GDALOpenInfo *);
      90             :     static GDALDataset *Create(const char *pszFilename, int nXSize, int nYSize,
      91             :                                int nBands, GDALDataType eType,
      92             :                                char **papszOptions);
      93             : };
      94             : 
      95             : /************************************************************************/
      96             : /* ==================================================================== */
      97             : /*                           GTXRasterBand                              */
      98             : /* ==================================================================== */
      99             : /************************************************************************/
     100             : 
     101             : class GTXRasterBand final : public RawRasterBand
     102             : {
     103             :     CPL_DISALLOW_COPY_ASSIGN(GTXRasterBand)
     104             : 
     105             :   public:
     106             :     GTXRasterBand(GDALDataset *poDS, int nBand, VSILFILE *fpRaw,
     107             :                   vsi_l_offset nImgOffset, int nPixelOffset, int nLineOffset,
     108             :                   GDALDataType eDataType, int bNativeOrder);
     109             : 
     110             :     ~GTXRasterBand() override;
     111             : 
     112             :     double GetNoDataValue(int *pbSuccess = nullptr) override;
     113             : };
     114             : 
     115             : /************************************************************************/
     116             : /*                            GTXRasterBand()                           */
     117             : /************************************************************************/
     118             : 
     119          47 : GTXRasterBand::GTXRasterBand(GDALDataset *poDSIn, int nBandIn,
     120             :                              VSILFILE *fpRawIn, vsi_l_offset nImgOffsetIn,
     121             :                              int nPixelOffsetIn, int nLineOffsetIn,
     122          47 :                              GDALDataType eDataTypeIn, int bNativeOrderIn)
     123             :     : RawRasterBand(poDSIn, nBandIn, fpRawIn, nImgOffsetIn, nPixelOffsetIn,
     124             :                     nLineOffsetIn, eDataTypeIn, bNativeOrderIn,
     125          47 :                     RawRasterBand::OwnFP::NO)
     126             : {
     127          47 : }
     128             : 
     129             : /************************************************************************/
     130             : /*                           ~GTXRasterBand()                           */
     131             : /************************************************************************/
     132             : 
     133          94 : GTXRasterBand::~GTXRasterBand()
     134             : {
     135          94 : }
     136             : 
     137             : /************************************************************************/
     138             : /*                           GetNoDataValue()                           */
     139             : /************************************************************************/
     140             : 
     141           0 : double GTXRasterBand::GetNoDataValue(int *pbSuccess)
     142             : {
     143           0 :     if (pbSuccess)
     144           0 :         *pbSuccess = TRUE;
     145           0 :     int bSuccess = FALSE;
     146           0 :     double dfNoData = GDALPamRasterBand::GetNoDataValue(&bSuccess);
     147           0 :     if (bSuccess)
     148             :     {
     149           0 :         return dfNoData;
     150             :     }
     151           0 :     return -88.8888;
     152             : }
     153             : 
     154             : /************************************************************************/
     155             : /* ==================================================================== */
     156             : /*                              GTXDataset                              */
     157             : /* ==================================================================== */
     158             : /************************************************************************/
     159             : 
     160             : /************************************************************************/
     161             : /*                            ~GTXDataset()                             */
     162             : /************************************************************************/
     163             : 
     164          94 : GTXDataset::~GTXDataset()
     165             : 
     166             : {
     167          47 :     GTXDataset::Close();
     168          94 : }
     169             : 
     170             : /************************************************************************/
     171             : /*                              Close()                                 */
     172             : /************************************************************************/
     173             : 
     174          85 : CPLErr GTXDataset::Close()
     175             : {
     176          85 :     CPLErr eErr = CE_None;
     177          85 :     if (nOpenFlags != OPEN_FLAGS_CLOSED)
     178             :     {
     179          47 :         if (GTXDataset::FlushCache(true) != CE_None)
     180           9 :             eErr = CE_Failure;
     181             : 
     182          47 :         if (fpImage)
     183             :         {
     184          47 :             if (VSIFCloseL(fpImage) != 0)
     185             :             {
     186           0 :                 CPLError(CE_Failure, CPLE_FileIO, "I/O error");
     187           0 :                 eErr = CE_Failure;
     188             :             }
     189             :         }
     190             : 
     191          47 :         if (GDALPamDataset::Close() != CE_None)
     192           0 :             eErr = CE_Failure;
     193             :     }
     194          85 :     return eErr;
     195             : }
     196             : 
     197             : /************************************************************************/
     198             : /*                              Identify()                              */
     199             : /************************************************************************/
     200             : 
     201       52130 : int GTXDataset::Identify(GDALOpenInfo *poOpenInfo)
     202             : 
     203             : {
     204       52130 :     if (poOpenInfo->nHeaderBytes < 40)
     205       48511 :         return FALSE;
     206             : 
     207        3619 :     if (!poOpenInfo->IsExtensionEqualToCI("gtx"))
     208        3525 :         return FALSE;
     209             : 
     210          94 :     return TRUE;
     211             : }
     212             : 
     213             : /************************************************************************/
     214             : /*                                Open()                                */
     215             : /************************************************************************/
     216             : 
     217          47 : GDALDataset *GTXDataset::Open(GDALOpenInfo *poOpenInfo)
     218             : 
     219             : {
     220          47 :     if (!Identify(poOpenInfo) || poOpenInfo->fpL == nullptr)
     221           0 :         return nullptr;
     222             : 
     223             :     /* -------------------------------------------------------------------- */
     224             :     /*      Create a corresponding GDALDataset.                             */
     225             :     /* -------------------------------------------------------------------- */
     226          94 :     auto poDS = std::make_unique<GTXDataset>();
     227             : 
     228          47 :     poDS->eAccess = poOpenInfo->eAccess;
     229          47 :     std::swap(poDS->fpImage, poOpenInfo->fpL);
     230             : 
     231             :     /* -------------------------------------------------------------------- */
     232             :     /*      Read the header.                                                */
     233             :     /* -------------------------------------------------------------------- */
     234          47 :     poDS->adfGeoTransform[2] = 0.0;
     235          47 :     poDS->adfGeoTransform[4] = 0.0;
     236             : 
     237          47 :     CPL_IGNORE_RET_VAL(
     238          47 :         VSIFReadL(poDS->adfGeoTransform + 3, 8, 1, poDS->fpImage));
     239          47 :     CPL_IGNORE_RET_VAL(
     240          47 :         VSIFReadL(poDS->adfGeoTransform + 0, 8, 1, poDS->fpImage));
     241          47 :     CPL_IGNORE_RET_VAL(
     242          47 :         VSIFReadL(poDS->adfGeoTransform + 5, 8, 1, poDS->fpImage));
     243          47 :     CPL_IGNORE_RET_VAL(
     244          47 :         VSIFReadL(poDS->adfGeoTransform + 1, 8, 1, poDS->fpImage));
     245             : 
     246          47 :     CPL_IGNORE_RET_VAL(VSIFReadL(&(poDS->nRasterYSize), 4, 1, poDS->fpImage));
     247          47 :     CPL_IGNORE_RET_VAL(VSIFReadL(&(poDS->nRasterXSize), 4, 1, poDS->fpImage));
     248             : 
     249          47 :     CPL_MSBPTR32(&(poDS->nRasterYSize));
     250          47 :     CPL_MSBPTR32(&(poDS->nRasterXSize));
     251             : 
     252          47 :     CPL_MSBPTR64(poDS->adfGeoTransform + 0);
     253          47 :     CPL_MSBPTR64(poDS->adfGeoTransform + 1);
     254          47 :     CPL_MSBPTR64(poDS->adfGeoTransform + 3);
     255          47 :     CPL_MSBPTR64(poDS->adfGeoTransform + 5);
     256             : 
     257          47 :     poDS->adfGeoTransform[3] += poDS->adfGeoTransform[5] *
     258          47 :                                 (static_cast<double>(poDS->nRasterYSize) - 1);
     259             : 
     260          47 :     poDS->adfGeoTransform[0] -= poDS->adfGeoTransform[1] * 0.5;
     261          47 :     poDS->adfGeoTransform[3] += poDS->adfGeoTransform[5] * 0.5;
     262             : 
     263          47 :     poDS->adfGeoTransform[5] *= -1;
     264             : 
     265          47 :     if (CPLFetchBool(poOpenInfo->papszOpenOptions,
     266             :                      "SHIFT_ORIGIN_IN_MINUS_180_PLUS_180", false))
     267             :     {
     268           0 :         if (poDS->adfGeoTransform[0] < -180.0 - poDS->adfGeoTransform[1])
     269           0 :             poDS->adfGeoTransform[0] += 360.0;
     270           0 :         else if (poDS->adfGeoTransform[0] > 180.0)
     271           0 :             poDS->adfGeoTransform[0] -= 360.0;
     272             :     }
     273             : 
     274          94 :     if (!GDALCheckDatasetDimensions(poDS->nRasterXSize, poDS->nRasterYSize) ||
     275          47 :         static_cast<vsi_l_offset>(poDS->nRasterXSize) * poDS->nRasterYSize >
     276          47 :             std::numeric_limits<vsi_l_offset>::max() / sizeof(double))
     277             :     {
     278           0 :         return nullptr;
     279             :     }
     280             : 
     281             :     /* -------------------------------------------------------------------- */
     282             :     /*      Guess the data type. Since October 1, 2009, it should be        */
     283             :     /*      Float32. Before it was double.                                  */
     284             :     /* -------------------------------------------------------------------- */
     285          47 :     CPL_IGNORE_RET_VAL(VSIFSeekL(poDS->fpImage, 0, SEEK_END));
     286          47 :     const vsi_l_offset nSize = VSIFTellL(poDS->fpImage);
     287             : 
     288          47 :     GDALDataType eDT = GDT_Float32;
     289          94 :     if (nSize - 40 == sizeof(double) *
     290          47 :                           static_cast<vsi_l_offset>(poDS->nRasterXSize) *
     291          47 :                           poDS->nRasterYSize)
     292           0 :         eDT = GDT_Float64;
     293          47 :     const int nDTSize = GDALGetDataTypeSizeBytes(eDT);
     294          47 :     if (nDTSize <= 0 || poDS->nRasterXSize > INT_MAX / nDTSize)
     295             :     {
     296           0 :         return nullptr;
     297             :     }
     298             : 
     299             :     /* -------------------------------------------------------------------- */
     300             :     /*      Create band information object.                                 */
     301             :     /* -------------------------------------------------------------------- */
     302             :     auto poBand = std::make_unique<GTXRasterBand>(
     303          47 :         poDS.get(), 1, poDS->fpImage,
     304          47 :         static_cast<vsi_l_offset>(poDS->nRasterYSize - 1) * poDS->nRasterXSize *
     305          47 :                 nDTSize +
     306             :             40,
     307         141 :         nDTSize, poDS->nRasterXSize * -nDTSize, eDT, !CPL_IS_LSB);
     308          47 :     if (!poBand->IsValid())
     309           0 :         return nullptr;
     310          47 :     poDS->SetBand(1, std::move(poBand));
     311             : 
     312             :     /* -------------------------------------------------------------------- */
     313             :     /*      Initialize any PAM information.                                 */
     314             :     /* -------------------------------------------------------------------- */
     315          47 :     poDS->SetDescription(poOpenInfo->pszFilename);
     316          47 :     poDS->TryLoadXML();
     317             : 
     318             :     /* -------------------------------------------------------------------- */
     319             :     /*      Check for overviews.                                            */
     320             :     /* -------------------------------------------------------------------- */
     321          47 :     poDS->oOvManager.Initialize(poDS.get(), poOpenInfo->pszFilename);
     322             : 
     323          47 :     return poDS.release();
     324             : }
     325             : 
     326             : /************************************************************************/
     327             : /*                          GetGeoTransform()                           */
     328             : /************************************************************************/
     329             : 
     330           3 : CPLErr GTXDataset::GetGeoTransform(double *padfTransform)
     331             : 
     332             : {
     333           3 :     memcpy(padfTransform, adfGeoTransform, sizeof(double) * 6);
     334           3 :     return CE_None;
     335             : }
     336             : 
     337             : /************************************************************************/
     338             : /*                          SetGeoTransform()                           */
     339             : /************************************************************************/
     340             : 
     341          33 : CPLErr GTXDataset::SetGeoTransform(double *padfTransform)
     342             : 
     343             : {
     344          33 :     if (padfTransform[2] != 0.0 || padfTransform[4] != 0.0)
     345             :     {
     346           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     347             :                  "Attempt to write skewed or rotated geotransform to gtx.");
     348           0 :         return CE_Failure;
     349             :     }
     350             : 
     351          33 :     memcpy(adfGeoTransform, padfTransform, sizeof(double) * 6);
     352             : 
     353          33 :     const double dfXOrigin = adfGeoTransform[0] + 0.5 * adfGeoTransform[1];
     354          33 :     const double dfYOrigin =
     355          33 :         adfGeoTransform[3] + (nRasterYSize - 0.5) * adfGeoTransform[5];
     356          33 :     const double dfWidth = adfGeoTransform[1];
     357          33 :     const double dfHeight = -adfGeoTransform[5];
     358             : 
     359          33 :     unsigned char header[32] = {'\0'};
     360          33 :     memcpy(header + 0, &dfYOrigin, 8);
     361          33 :     CPL_MSBPTR64(header + 0);
     362             : 
     363          33 :     memcpy(header + 8, &dfXOrigin, 8);
     364          33 :     CPL_MSBPTR64(header + 8);
     365             : 
     366          33 :     memcpy(header + 16, &dfHeight, 8);
     367          33 :     CPL_MSBPTR64(header + 16);
     368             : 
     369          33 :     memcpy(header + 24, &dfWidth, 8);
     370          33 :     CPL_MSBPTR64(header + 24);
     371             : 
     372          66 :     if (VSIFSeekL(fpImage, SEEK_SET, 0) != 0 ||
     373          33 :         VSIFWriteL(header, 32, 1, fpImage) != 1)
     374             :     {
     375           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     376             :                  "Attempt to write geotransform header to GTX failed.");
     377           0 :         return CE_Failure;
     378             :     }
     379             : 
     380          33 :     return CE_None;
     381             : }
     382             : 
     383             : /************************************************************************/
     384             : /*                               Create()                               */
     385             : /************************************************************************/
     386             : 
     387          81 : GDALDataset *GTXDataset::Create(const char *pszFilename, int nXSize, int nYSize,
     388             :                                 int /* nBands */, GDALDataType eType,
     389             :                                 char ** /* papszOptions */)
     390             : {
     391          81 :     if (eType != GDT_Float32)
     392             :     {
     393          44 :         CPLError(CE_Failure, CPLE_AppDefined,
     394             :                  "Attempt to create gtx file with unsupported data type '%s'.",
     395             :                  GDALGetDataTypeName(eType));
     396          44 :         return nullptr;
     397             :     }
     398             : 
     399          37 :     if (!EQUAL(CPLGetExtensionSafe(pszFilename).c_str(), "gtx"))
     400             :     {
     401           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     402             :                  "Attempt to create gtx file with extension other than gtx.");
     403           0 :         return nullptr;
     404             :     }
     405             : 
     406             :     /* -------------------------------------------------------------------- */
     407             :     /*      Try to create the file.                                         */
     408             :     /* -------------------------------------------------------------------- */
     409          37 :     VSILFILE *fp = VSIFOpenL(pszFilename, "wb");
     410          37 :     if (fp == nullptr)
     411             :     {
     412           3 :         CPLError(CE_Failure, CPLE_OpenFailed,
     413             :                  "Attempt to create file `%s' failed.\n", pszFilename);
     414           3 :         return nullptr;
     415             :     }
     416             : 
     417             :     /* -------------------------------------------------------------------- */
     418             :     /*      Write out the header with stub georeferencing.                  */
     419             :     /* -------------------------------------------------------------------- */
     420             : 
     421          34 :     unsigned char header[40] = {'\0'};
     422          34 :     double dfYOrigin = 0.0;
     423          34 :     memcpy(header + 0, &dfYOrigin, 8);
     424          34 :     CPL_MSBPTR64(header + 0);
     425             : 
     426          34 :     double dfXOrigin = 0.0;
     427          34 :     memcpy(header + 8, &dfXOrigin, 8);
     428          34 :     CPL_MSBPTR64(header + 8);
     429             : 
     430          34 :     double dfYSize = 0.01;
     431          34 :     memcpy(header + 16, &dfYSize, 8);
     432          34 :     CPL_MSBPTR64(header + 16);
     433             : 
     434          34 :     double dfXSize = 0.01;
     435          34 :     memcpy(header + 24, &dfXSize, 8);
     436          34 :     CPL_MSBPTR64(header + 24);
     437             : 
     438          34 :     GInt32 nYSize32 = nYSize;
     439          34 :     memcpy(header + 32, &nYSize32, 4);
     440          34 :     CPL_MSBPTR32(header + 32);
     441             : 
     442          34 :     GInt32 nXSize32 = nXSize;
     443          34 :     memcpy(header + 36, &nXSize32, 4);
     444          34 :     CPL_MSBPTR32(header + 36);
     445             : 
     446          34 :     CPL_IGNORE_RET_VAL(VSIFWriteL(header, 40, 1, fp));
     447          34 :     CPL_IGNORE_RET_VAL(VSIFCloseL(fp));
     448             : 
     449          34 :     return GDALDataset::FromHandle(GDALOpen(pszFilename, GA_Update));
     450             : }
     451             : 
     452             : /************************************************************************/
     453             : /*                          GDALRegister_GTX()                          */
     454             : /************************************************************************/
     455             : 
     456        1682 : void GDALRegister_GTX()
     457             : 
     458             : {
     459        1682 :     if (GDALGetDriverByName("GTX") != nullptr)
     460         301 :         return;
     461             : 
     462        1381 :     GDALDriver *poDriver = new GDALDriver();
     463             : 
     464        1381 :     poDriver->SetDescription("GTX");
     465        1381 :     poDriver->SetMetadataItem(GDAL_DCAP_RASTER, "YES");
     466        1381 :     poDriver->SetMetadataItem(GDAL_DMD_LONGNAME, "NOAA Vertical Datum .GTX");
     467        1381 :     poDriver->SetMetadataItem(GDAL_DMD_EXTENSION, "gtx");
     468        1381 :     poDriver->SetMetadataItem(GDAL_DCAP_VIRTUALIO, "YES");
     469             :     // poDriver->SetMetadataItem( GDAL_DMD_HELPTOPIC,
     470             :     //                            "frmt_various.html#GTX" );
     471        1381 :     poDriver->SetMetadataItem(
     472             :         GDAL_DMD_OPENOPTIONLIST,
     473             :         "<OpenOptionList>"
     474             :         "   <Option name='SHIFT_ORIGIN_IN_MINUS_180_PLUS_180' type='boolean' "
     475             :         "description='Whether to apply a +/-360 deg shift to the longitude of "
     476             :         "the top left corner so that it is in the [-180,180] range' "
     477             :         "default='NO'/>"
     478        1381 :         "</OpenOptionList>");
     479             : 
     480        1381 :     poDriver->SetMetadataItem(GDAL_DMD_CREATIONDATATYPES, "Float32");
     481             : 
     482        1381 :     poDriver->pfnOpen = GTXDataset::Open;
     483        1381 :     poDriver->pfnIdentify = GTXDataset::Identify;
     484        1381 :     poDriver->pfnCreate = GTXDataset::Create;
     485             : 
     486        1381 :     GetGDALDriverManager()->RegisterDriver(poDriver);
     487             : }

Generated by: LCOV version 1.14