LCOV - code coverage report
Current view: top level - frmts/terragen - terragendataset.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 275 358 76.8 %
Date: 2025-01-18 12:42:00 Functions: 31 34 91.2 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  * terragendataset.cpp,v 1.2
       3             :  *
       4             :  * Project:  Terragen(tm) TER Driver
       5             :  * Purpose:  Reader for Terragen TER documents
       6             :  * Author:   Ray Gardener, Daylon Graphics Ltd.
       7             :  *
       8             :  * Portions of this module derived from GDAL drivers by
       9             :  * Frank Warmerdam, see http://www.gdal.org
      10             : 
      11             :  rcg    apr 19/06       Fixed bug with hf size being misread by one
      12             :                         if xpts/ypts tags not included in file.
      13             :                         Added Create() support.
      14             :                         Treat pixels as points.
      15             : 
      16             :  rcg    jun  6/07       Better heightscale/baseheight determination
      17             :                     when writing.
      18             : 
      19             :  *
      20             :  ******************************************************************************
      21             :  * Copyright (c) 2006-2007 Daylon Graphics Ltd.
      22             :  *
      23             :  * SPDX-License-Identifier: MIT
      24             :  ******************************************************************************
      25             :  */
      26             : 
      27             : /*
      28             :         Terragen format notes:
      29             : 
      30             :         Based on official Planetside specs.
      31             : 
      32             :         All distances along all three axes are in
      33             :         terrain units, which are 30m by default.
      34             :         If a SCAL chunk is present, however, it
      35             :         can indicate something other than 30.
      36             :         Note that uniform scaling should be used.
      37             : 
      38             :         The offset (base height) in the ALTW chunk
      39             :         is in terrain units, and the scale (height scale)
      40             :         is a normalized value using unsigned 16-bit notation.
      41             :         The physical terrain value for a read pixel is
      42             :         hv' = hv * scale / 65536 + offset.
      43             :         It still needs to be scaled by SCAL to
      44             :         get to meters.
      45             : 
      46             :         For writing:
      47             : 
      48             :         SCAL = gridpost distance in meters
      49             :         hv_px = hv_m / SCAL
      50             :         span_px = span_m / SCAL
      51             :         offset = see TerragenDataset::write_header()
      52             :         scale = see TerragenDataset::write_header()
      53             :         physical hv =
      54             :           (hv_px - offset) * 65536.0/scale
      55             : 
      56             :         We tell callers that:
      57             : 
      58             :         Elevations are Int16 when reading,
      59             :         and Float32 when writing. We need logical
      60             :         elevations when writing so that we can
      61             :         encode them with as much precision as possible
      62             :         when going down to physical 16-bit ints.
      63             :         Implementing band::SetScale/SetOffset won't work because
      64             :         it requires callers to know format write details.
      65             :         So we've added two Create() options that let the
      66             :         caller tell us the span's logical extent, and with
      67             :         those two values we can convert to physical pixels.
      68             : 
      69             :         band::GetUnitType() returns meters.
      70             :         band::GetScale() returns SCAL * (scale/65536)
      71             :         band::GetOffset() returns SCAL * offset
      72             :         ds::GetSpatialRef() returns a local CS
      73             :                 using meters.
      74             :         ds::GetGeoTransform() returns a scale matrix
      75             :                 having SCAL sx,sy members.
      76             : 
      77             :         ds::SetGeoTransform() lets us establish the
      78             :                 size of ground pixels.
      79             :         ds::_SetProjection() lets us establish what
      80             :                 units ground measures are in (also needed
      81             :                 to calc the size of ground pixels).
      82             :         band::SetUnitType() tells us what units
      83             :                 the given Float32 elevations are in.
      84             :         band::SetScale() is unused.
      85             :         band::SetOffset() is unused.
      86             : */
      87             : 
      88             : #include "cpl_string.h"
      89             : #include "gdal_frmts.h"
      90             : #include "gdal_pam.h"
      91             : #include "ogr_spatialref.h"
      92             : 
      93             : #include <cmath>
      94             : 
      95             : #include <algorithm>
      96             : 
      97             : const double kdEarthCircumPolar = 40007849;
      98             : const double kdEarthCircumEquat = 40075004;
      99             : 
     100           1 : static double average(double a, double b)
     101             : {
     102           1 :     return 0.5 * (a + b);
     103             : }
     104             : 
     105           0 : static double degrees_to_radians(double d)
     106             : {
     107           0 :     return d * (M_PI / 180);
     108             : }
     109             : 
     110           2 : static bool approx_equal(double a, double b)
     111             : {
     112           2 :     const double epsilon = 1e-5;
     113           2 :     return std::abs(a - b) <= epsilon;
     114             : }
     115             : 
     116             : /************************************************************************/
     117             : /* ==================================================================== */
     118             : /*                              TerragenDataset                         */
     119             : /* ==================================================================== */
     120             : /************************************************************************/
     121             : 
     122             : class TerragenRasterBand;
     123             : 
     124             : class TerragenDataset final : public GDALPamDataset
     125             : {
     126             :     friend class TerragenRasterBand;
     127             : 
     128             :     double m_dScale, m_dOffset,
     129             :         m_dSCAL,  // 30.0 normally, from SCAL chunk
     130             :         m_adfTransform[6], m_dGroundScale, m_dMetersPerGroundUnit,
     131             :         m_dMetersPerElevUnit, m_dLogSpan[2], m_span_m[2], m_span_px[2];
     132             : 
     133             :     VSILFILE *m_fp;
     134             :     vsi_l_offset m_nDataOffset;
     135             : 
     136             :     GInt16 m_nHeightScale;
     137             :     GInt16 m_nBaseHeight;
     138             : 
     139             :     char *m_pszFilename;
     140             :     OGRSpatialReference m_oSRS{};
     141             :     char m_szUnits[32];
     142             : 
     143             :     bool m_bIsGeo;
     144             : 
     145             :     int LoadFromFile();
     146             : 
     147             :   public:
     148             :     TerragenDataset();
     149             :     virtual ~TerragenDataset();
     150             : 
     151             :     static GDALDataset *Open(GDALOpenInfo *);
     152             :     static GDALDataset *Create(const char *pszFilename, int nXSize, int nYSize,
     153             :                                int nBandsIn, GDALDataType eType,
     154             :                                char **papszOptions);
     155             : 
     156             :     virtual CPLErr GetGeoTransform(double *) override;
     157             :     virtual CPLErr SetGeoTransform(double *) override;
     158             :     const OGRSpatialReference *GetSpatialRef() const override;
     159             :     CPLErr SetSpatialRef(const OGRSpatialReference *poSRS) override;
     160             : 
     161             :   protected:
     162             :     bool get(GInt16 &);
     163             :     bool get(GUInt16 &);
     164             :     bool get(float &);
     165             :     bool put(GInt16);
     166             :     bool put(float);
     167             : 
     168           9 :     bool skip(size_t n)
     169             :     {
     170           9 :         return 0 == VSIFSeekL(m_fp, n, SEEK_CUR);
     171             :     }
     172             : 
     173           1 :     bool pad(size_t n)
     174             :     {
     175           1 :         return skip(n);
     176             :     }
     177             : 
     178             :     bool read_next_tag(char *);
     179             :     bool write_next_tag(const char *);
     180             :     static bool tag_is(const char *szTag, const char *);
     181             : 
     182             :     bool write_header(void);
     183             : };
     184             : 
     185             : /************************************************************************/
     186             : /* ==================================================================== */
     187             : /*                            TerragenRasterBand                        */
     188             : /* ==================================================================== */
     189             : /************************************************************************/
     190             : 
     191             : class TerragenRasterBand final : public GDALPamRasterBand
     192             : {
     193             :     friend class TerragenDataset;
     194             : 
     195             :     void *m_pvLine;
     196             :     bool m_bFirstTime;
     197             : 
     198             :   public:
     199             :     explicit TerragenRasterBand(TerragenDataset *);
     200             : 
     201          10 :     virtual ~TerragenRasterBand()
     202           5 :     {
     203           5 :         if (m_pvLine != nullptr)
     204           5 :             CPLFree(m_pvLine);
     205          10 :     }
     206             : 
     207             :     // Geomeasure support.
     208             :     virtual CPLErr IReadBlock(int, int, void *) override;
     209             :     virtual const char *GetUnitType() override;
     210             :     virtual double GetOffset(int *pbSuccess = nullptr) override;
     211             :     virtual double GetScale(int *pbSuccess = nullptr) override;
     212             : 
     213             :     virtual CPLErr IWriteBlock(int, int, void *) override;
     214             :     virtual CPLErr SetUnitType(const char *) override;
     215             : };
     216             : 
     217             : /************************************************************************/
     218             : /*                         TerragenRasterBand()                         */
     219             : /************************************************************************/
     220             : 
     221           5 : TerragenRasterBand::TerragenRasterBand(TerragenDataset *poDSIn)
     222           5 :     : m_pvLine(CPLMalloc(sizeof(GInt16) * poDSIn->GetRasterXSize())),
     223           5 :       m_bFirstTime(true)
     224             : {
     225           5 :     poDS = poDSIn;
     226           5 :     nBand = 1;
     227             : 
     228           5 :     eDataType = poDSIn->GetAccess() == GA_ReadOnly ? GDT_Int16 : GDT_Float32;
     229             : 
     230           5 :     nBlockXSize = poDSIn->GetRasterXSize();
     231           5 :     nBlockYSize = 1;
     232           5 : }
     233             : 
     234             : /************************************************************************/
     235             : /*                             IReadBlock()                             */
     236             : /************************************************************************/
     237             : 
     238          40 : CPLErr TerragenRasterBand::IReadBlock(CPL_UNUSED int nBlockXOff, int nBlockYOff,
     239             :                                       void *pImage)
     240             : {
     241             :     // CPLAssert( sizeof(float) == sizeof(GInt32) );
     242          40 :     CPLAssert(nBlockXOff == 0);
     243          40 :     CPLAssert(pImage != nullptr);
     244             : 
     245          40 :     TerragenDataset &ds = *reinterpret_cast<TerragenDataset *>(poDS);
     246             : 
     247             :     /* -------------------------------------------------------------------- */
     248             :     /*      Seek to scanline.
     249             :             Terragen is a bottom-top format, so we have to
     250             :             invert the row location.
     251             :      -------------------------------------------------------------------- */
     252          40 :     const size_t rowbytes = nBlockXSize * sizeof(GInt16);
     253             : 
     254          40 :     if (0 != VSIFSeekL(ds.m_fp,
     255          40 :                        ds.m_nDataOffset +
     256          40 :                            (ds.GetRasterYSize() - 1 - nBlockYOff) * rowbytes,
     257             :                        SEEK_SET))
     258             :     {
     259           0 :         CPLError(CE_Failure, CPLE_FileIO, "Terragen Seek failed:%s",
     260           0 :                  VSIStrerror(errno));
     261           0 :         return CE_Failure;
     262             :     }
     263             : 
     264             :     /* -------------------------------------------------------------------- */
     265             :     /*      Read the scanline into the line buffer.                        */
     266             :     /* -------------------------------------------------------------------- */
     267             : 
     268          40 :     if (VSIFReadL(pImage, rowbytes, 1, ds.m_fp) != 1)
     269             :     {
     270           0 :         CPLError(CE_Failure, CPLE_FileIO, "Terragen read failed:%s",
     271           0 :                  VSIStrerror(errno));
     272           0 :         return CE_Failure;
     273             :     }
     274             : 
     275             : /* -------------------------------------------------------------------- */
     276             : /*      Swap on MSB platforms.                                          */
     277             : /* -------------------------------------------------------------------- */
     278             : #ifdef CPL_MSB
     279             :     GDALSwapWords(pImage, sizeof(GInt16), nRasterXSize, sizeof(GInt16));
     280             : #endif
     281             : 
     282          40 :     return CE_None;
     283             : }
     284             : 
     285             : /************************************************************************/
     286             : /*                            GetUnitType()                             */
     287             : /************************************************************************/
     288           0 : const char *TerragenRasterBand::GetUnitType()
     289             : {
     290             :     // todo: Return elevation units.
     291             :     // For Terragen documents, it is the same as the ground units.
     292           0 :     TerragenDataset *poGDS = reinterpret_cast<TerragenDataset *>(poDS);
     293             : 
     294           0 :     return poGDS->m_szUnits;
     295             : }
     296             : 
     297             : /************************************************************************/
     298             : /*                              GetScale()                              */
     299             : /************************************************************************/
     300             : 
     301           1 : double TerragenRasterBand::GetScale(int *pbSuccess)
     302             : {
     303           1 :     const TerragenDataset &ds = *reinterpret_cast<TerragenDataset *>(poDS);
     304           1 :     if (pbSuccess != nullptr)
     305           0 :         *pbSuccess = TRUE;
     306             : 
     307           1 :     return ds.m_dScale;
     308             : }
     309             : 
     310             : /************************************************************************/
     311             : /*                             GetOffset()                              */
     312             : /************************************************************************/
     313             : 
     314           1 : double TerragenRasterBand::GetOffset(int *pbSuccess)
     315             : {
     316           1 :     const TerragenDataset &ds = *reinterpret_cast<TerragenDataset *>(poDS);
     317           1 :     if (pbSuccess != nullptr)
     318           0 :         *pbSuccess = TRUE;
     319             : 
     320           1 :     return ds.m_dOffset;
     321             : }
     322             : 
     323             : /************************************************************************/
     324             : /*                             IWriteBlock()                            */
     325             : /************************************************************************/
     326             : 
     327          20 : CPLErr TerragenRasterBand::IWriteBlock(CPL_UNUSED int nBlockXOff,
     328             :                                        int nBlockYOff, void *pImage)
     329             : {
     330          20 :     CPLAssert(nBlockXOff == 0);
     331          20 :     CPLAssert(pImage != nullptr);
     332          20 :     CPLAssert(m_pvLine != nullptr);
     333             : 
     334          20 :     const size_t pixelsize = sizeof(GInt16);
     335             : 
     336          20 :     TerragenDataset &ds = *reinterpret_cast<TerragenDataset *>(poDS);
     337          20 :     if (m_bFirstTime)
     338             :     {
     339           1 :         m_bFirstTime = false;
     340           1 :         ds.write_header();
     341           1 :         ds.m_nDataOffset = VSIFTellL(ds.m_fp);
     342             :     }
     343          20 :     const size_t rowbytes = nBlockXSize * pixelsize;
     344             : 
     345          20 :     GInt16 *pLine = reinterpret_cast<GInt16 *>(m_pvLine);
     346             : 
     347          20 :     if (0 == VSIFSeekL(ds.m_fp,
     348          20 :                        ds.m_nDataOffset +
     349             :                            // Terragen is Y inverted.
     350          20 :                            (ds.GetRasterYSize() - 1 - nBlockYOff) * rowbytes,
     351             :                        SEEK_SET))
     352             :     {
     353             :         // Convert each float32 to int16.
     354          20 :         float *pfImage = reinterpret_cast<float *>(pImage);
     355         420 :         for (size_t x = 0; x < static_cast<size_t>(nBlockXSize); x++)
     356             :         {
     357         400 :             const double f = pfImage[x] * ds.m_dMetersPerElevUnit / ds.m_dSCAL;
     358         400 :             const GInt16 hv = static_cast<GInt16>(
     359         400 :                 (f - ds.m_nBaseHeight) * 65536.0 / ds.m_nHeightScale
     360             :                 /*+ ds.m_nShift*/);
     361         400 :             pLine[x] = hv;
     362             :         }
     363             : 
     364             : #ifdef CPL_MSB
     365             :         GDALSwapWords(m_pvLine, pixelsize, nBlockXSize, pixelsize);
     366             : #endif
     367          20 :         if (1 == VSIFWriteL(m_pvLine, rowbytes, 1, ds.m_fp))
     368          20 :             return CE_None;
     369             :     }
     370             : 
     371           0 :     return CE_Failure;
     372             : }
     373             : 
     374           0 : CPLErr TerragenRasterBand::SetUnitType(const char *psz)
     375             : {
     376           0 :     TerragenDataset &ds = *reinterpret_cast<TerragenDataset *>(poDS);
     377             : 
     378           0 :     if (EQUAL(psz, "m"))
     379           0 :         ds.m_dMetersPerElevUnit = 1.0;
     380           0 :     else if (EQUAL(psz, "ft"))
     381           0 :         ds.m_dMetersPerElevUnit = 0.3048;
     382           0 :     else if (EQUAL(psz, "sft"))
     383           0 :         ds.m_dMetersPerElevUnit = 1200.0 / 3937.0;
     384             :     else
     385           0 :         return CE_Failure;
     386             : 
     387           0 :     return CE_None;
     388             : }
     389             : 
     390             : /************************************************************************/
     391             : /* ==================================================================== */
     392             : /*                          TerragenDataset                             */
     393             : /* ==================================================================== */
     394             : /************************************************************************/
     395             : 
     396             : /************************************************************************/
     397             : /*                          TerragenDataset()                           */
     398             : /************************************************************************/
     399             : 
     400          54 : TerragenDataset::TerragenDataset()
     401             :     : m_dScale(0.0), m_dOffset(0.0), m_dSCAL(30.0), m_dGroundScale(0.0),
     402             :       m_dMetersPerGroundUnit(1.0), m_dMetersPerElevUnit(1.0), m_fp(nullptr),
     403             :       m_nDataOffset(0), m_nHeightScale(0), m_nBaseHeight(0),
     404          54 :       m_pszFilename(nullptr), m_bIsGeo(false)
     405             : {
     406          54 :     m_oSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
     407             : 
     408          54 :     m_dLogSpan[0] = 0.0;
     409          54 :     m_dLogSpan[1] = 0.0;
     410             : 
     411          54 :     m_adfTransform[0] = 0.0;
     412          54 :     m_adfTransform[1] = m_dSCAL;
     413          54 :     m_adfTransform[2] = 0.0;
     414          54 :     m_adfTransform[3] = 0.0;
     415          54 :     m_adfTransform[4] = 0.0;
     416          54 :     m_adfTransform[5] = m_dSCAL;
     417          54 :     m_span_m[0] = 0.0;
     418          54 :     m_span_m[1] = 0.0;
     419          54 :     m_span_px[0] = 0.0;
     420          54 :     m_span_px[1] = 0.0;
     421          54 :     memset(m_szUnits, 0, sizeof(m_szUnits));
     422          54 : }
     423             : 
     424             : /************************************************************************/
     425             : /*                          ~TerragenDataset()                          */
     426             : /************************************************************************/
     427             : 
     428         108 : TerragenDataset::~TerragenDataset()
     429             : 
     430             : {
     431          54 :     FlushCache(true);
     432             : 
     433          54 :     CPLFree(m_pszFilename);
     434             : 
     435          54 :     if (m_fp != nullptr)
     436           5 :         VSIFCloseL(m_fp);
     437         108 : }
     438             : 
     439           1 : bool TerragenDataset::write_header()
     440             : {
     441             :     char szHeader[16];
     442             :     // cppcheck-suppress bufferNotZeroTerminated
     443           1 :     memcpy(szHeader, "TERRAGENTERRAIN ", sizeof(szHeader));
     444             : 
     445           1 :     if (1 != VSIFWriteL(reinterpret_cast<void *>(szHeader), sizeof(szHeader), 1,
     446             :                         m_fp))
     447             :     {
     448           0 :         CPLError(CE_Failure, CPLE_FileIO,
     449             :                  "Couldn't write to Terragen file %s.\n"
     450             :                  "Is file system full?",
     451             :                  m_pszFilename);
     452             : 
     453           0 :         return false;
     454             :     }
     455             : 
     456             :     // --------------------------------------------------------------------
     457             :     //      Write out the heightfield dimensions, etc.
     458             :     // --------------------------------------------------------------------
     459             : 
     460           1 :     const int nXSize = GetRasterXSize();
     461           1 :     const int nYSize = GetRasterYSize();
     462             : 
     463           1 :     write_next_tag("SIZE");
     464           1 :     put(static_cast<GInt16>(std::min(nXSize, nYSize) - 1));
     465           1 :     pad(sizeof(GInt16));
     466             : 
     467           1 :     if (nXSize != nYSize)
     468             :     {
     469           0 :         write_next_tag("XPTS");
     470           0 :         put(static_cast<GInt16>(nXSize));
     471           0 :         pad(sizeof(GInt16));
     472           0 :         write_next_tag("YPTS");
     473           0 :         put(static_cast<GInt16>(nYSize));
     474           0 :         pad(sizeof(GInt16));
     475             :     }
     476             : 
     477           1 :     if (m_bIsGeo)
     478             :     {
     479             :         /*
     480             :           With a geographic projection (degrees),
     481             :           m_dGroundScale will be in degrees and
     482             :           m_dMetersPerGroundUnit is undefined.
     483             :           So we're going to estimate a m_dMetersPerGroundUnit
     484             :           value here (i.e., meters per degree).
     485             : 
     486             :           We figure out the degree size of one
     487             :           pixel, and then the latitude degrees
     488             :           of the heightfield's center. The circumference of
     489             :           the latitude's great circle lets us know how
     490             :           wide the pixel is in meters, and we
     491             :           average that with the pixel's meter breadth,
     492             :           which is based on the polar circumference.
     493             :         */
     494             : 
     495             :         /* const double m_dDegLongPerPixel =
     496             :               fabs(m_adfTransform[1]); */
     497             : 
     498           0 :         const double m_dDegLatPerPixel = std::abs(m_adfTransform[5]);
     499             : 
     500             :         /* const double m_dCenterLongitude =
     501             :               m_adfTransform[0] +
     502             :               (0.5 * m_dDegLongPerPixel * (nXSize-1)); */
     503             : 
     504           0 :         const double m_dCenterLatitude =
     505           0 :             m_adfTransform[3] + (0.5 * m_dDegLatPerPixel * (nYSize - 1));
     506             : 
     507             :         const double dLatCircum =
     508             :             kdEarthCircumEquat *
     509           0 :             std::sin(degrees_to_radians(90.0 - m_dCenterLatitude));
     510             : 
     511           0 :         const double dMetersPerDegLongitude = dLatCircum / 360;
     512             :         /* const double dMetersPerPixelX =
     513             :               (m_dDegLongPerPixel / 360) * dLatCircum; */
     514             : 
     515           0 :         const double dMetersPerDegLatitude = kdEarthCircumPolar / 360;
     516             :         /* const double dMetersPerPixelY =
     517             :               (m_dDegLatPerPixel / 360) * kdEarthCircumPolar; */
     518             : 
     519           0 :         m_dMetersPerGroundUnit =
     520           0 :             average(dMetersPerDegLongitude, dMetersPerDegLatitude);
     521             :     }
     522             : 
     523           1 :     m_dSCAL = m_dGroundScale * m_dMetersPerGroundUnit;
     524             : 
     525           1 :     if (m_dSCAL != 30.0)
     526             :     {
     527           1 :         const float sc = static_cast<float>(m_dSCAL);
     528           1 :         write_next_tag("SCAL");
     529           1 :         put(sc);
     530           1 :         put(sc);
     531           1 :         put(sc);
     532             :     }
     533             : 
     534           1 :     if (!write_next_tag("ALTW"))
     535             :     {
     536           0 :         CPLError(CE_Failure, CPLE_FileIO,
     537             :                  "Couldn't write to Terragen file %s.\n"
     538             :                  "Is file system full?",
     539             :                  m_pszFilename);
     540             : 
     541           0 :         return false;
     542             :     }
     543             : 
     544             :     // Compute physical scales and offsets.
     545           1 :     m_span_m[0] = m_dLogSpan[0] * m_dMetersPerElevUnit;
     546           1 :     m_span_m[1] = m_dLogSpan[1] * m_dMetersPerElevUnit;
     547             : 
     548           1 :     m_span_px[0] = m_span_m[0] / m_dSCAL;
     549           1 :     m_span_px[1] = m_span_m[1] / m_dSCAL;
     550             : 
     551           1 :     const double span_px = m_span_px[1] - m_span_px[0];
     552           1 :     m_nHeightScale = static_cast<GInt16>(span_px);
     553           1 :     if (m_nHeightScale == 0)
     554           0 :         m_nHeightScale++;
     555             : 
     556             : // TODO(schwehr): Make static functions.
     557             : #define P2L_PX(n, hs, bh) (static_cast<double>(n) / 65536.0 * (hs) + (bh))
     558             : 
     559             : #define L2P_PX(n, hs, bh) (static_cast<int>(((n) - (bh)) * 65536.0 / (hs)))
     560             : 
     561             :     // Increase the heightscale until the physical span
     562             :     // fits within a 16-bit range. The smaller the logical span,
     563             :     // the more necessary this becomes.
     564           1 :     int hs = m_nHeightScale;
     565           1 :     int bh = 0;
     566           4 :     for (; hs <= 32767; hs++)
     567             :     {
     568           4 :         double prevdelta = 1.0e30;
     569      229383 :         for (bh = -32768; bh <= 32767; bh++)
     570             :         {
     571      229380 :             const int nValley = L2P_PX(m_span_px[0], hs, bh);
     572      229380 :             if (nValley < -32768)
     573       98293 :                 continue;
     574      131087 :             const int nPeak = L2P_PX(m_span_px[1], hs, bh);
     575      131087 :             if (nPeak > 32767)
     576      131082 :                 continue;
     577             : 
     578             :             // now see how closely the baseheight gets
     579             :             // to the pixel span.
     580           5 :             const double d = P2L_PX(nValley, hs, bh);
     581           5 :             const double delta = std::abs(d - m_span_px[0]);
     582           5 :             if (delta < prevdelta)  // Converging?
     583           4 :                 prevdelta = delta;
     584             :             else
     585             :             {
     586             :                 // We're diverging, so use the previous bh
     587             :                 // and stop looking.
     588           1 :                 bh--;
     589           1 :                 break;
     590             :             }
     591             :         }
     592           4 :         if (bh != 32768)
     593           1 :             break;
     594             :     }
     595           1 :     if (hs == 32768)
     596             :     {
     597           0 :         CPLError(CE_Failure, CPLE_FileIO,
     598             :                  "Couldn't write to Terragen file %s.\n"
     599             :                  "Cannot find adequate heightscale/baseheight combination.",
     600             :                  m_pszFilename);
     601             : 
     602           0 :         return false;
     603             :     }
     604             : 
     605           1 :     m_nHeightScale = static_cast<GInt16>(hs);
     606           1 :     m_nBaseHeight = static_cast<GInt16>(bh);
     607             : 
     608             :     // m_nHeightScale is the one that gives us the
     609             :     // widest use of the 16-bit space. However, there
     610             :     // might be larger heightscales that, even though
     611             :     // the reduce the space usage, give us a better fit
     612             :     // for preserving the span extents.
     613             : 
     614           1 :     return put(m_nHeightScale) && put(m_nBaseHeight);
     615             : }
     616             : 
     617             : /************************************************************************/
     618             : /*                                get()                                 */
     619             : /************************************************************************/
     620             : 
     621           8 : bool TerragenDataset::get(GInt16 &value)
     622             : {
     623           8 :     if (1 == VSIFReadL(&value, sizeof(value), 1, m_fp))
     624             :     {
     625           8 :         CPL_LSBPTR16(&value);
     626           8 :         return true;
     627             :     }
     628           0 :     return false;
     629             : }
     630             : 
     631           4 : bool TerragenDataset::get(GUInt16 &value)
     632             : {
     633           4 :     if (1 == VSIFReadL(&value, sizeof(value), 1, m_fp))
     634             :     {
     635           4 :         CPL_LSBPTR16(&value);
     636           4 :         return true;
     637             :     }
     638           0 :     return false;
     639             : }
     640             : 
     641          12 : bool TerragenDataset::get(float &value)
     642             : {
     643          12 :     if (1 == VSIFReadL(&value, sizeof(value), 1, m_fp))
     644             :     {
     645          12 :         CPL_LSBPTR32(&value);
     646          12 :         return true;
     647             :     }
     648           0 :     return false;
     649             : }
     650             : 
     651             : /************************************************************************/
     652             : /*                                put()                                 */
     653             : /************************************************************************/
     654             : 
     655           3 : bool TerragenDataset::put(GInt16 n)
     656             : {
     657           3 :     CPL_LSBPTR16(&n);
     658           3 :     return 1 == VSIFWriteL(&n, sizeof(n), 1, m_fp);
     659             : }
     660             : 
     661           3 : bool TerragenDataset::put(float f)
     662             : {
     663           3 :     CPL_LSBPTR32(&f);
     664           3 :     return 1 == VSIFWriteL(&f, sizeof(f), 1, m_fp);
     665             : }
     666             : 
     667             : /************************************************************************/
     668             : /*                              tag stuff                               */
     669             : /************************************************************************/
     670             : 
     671          16 : bool TerragenDataset::read_next_tag(char *szTag)
     672             : {
     673          16 :     return 1 == VSIFReadL(szTag, 4, 1, m_fp);
     674             : }
     675             : 
     676           3 : bool TerragenDataset::write_next_tag(const char *szTag)
     677             : {
     678           3 :     return 1 == VSIFWriteL(reinterpret_cast<void *>(const_cast<char *>(szTag)),
     679           3 :                            4, 1, m_fp);
     680             : }
     681             : 
     682          40 : bool TerragenDataset::tag_is(const char *szTag, const char *sz)
     683             : {
     684          40 :     return 0 == memcmp(szTag, sz, 4);
     685             : }
     686             : 
     687             : /************************************************************************/
     688             : /*                            LoadFromFile()                            */
     689             : /************************************************************************/
     690             : 
     691           4 : int TerragenDataset::LoadFromFile()
     692             : {
     693           4 :     m_dSCAL = 30.0;
     694           4 :     m_nDataOffset = 0;
     695             : 
     696           4 :     if (0 != VSIFSeekL(m_fp, 16, SEEK_SET))
     697           0 :         return FALSE;
     698             : 
     699             :     char szTag[4];
     700           4 :     if (!read_next_tag(szTag) || !tag_is(szTag, "SIZE"))
     701           0 :         return FALSE;
     702             : 
     703             :     GUInt16 nSize;
     704           4 :     if (!get(nSize) || !skip(2))
     705           0 :         return FALSE;
     706             : 
     707             :     // Set dimensions to SIZE chunk. If we don't
     708             :     // encounter XPTS/YPTS chunks, we can assume
     709             :     // the terrain to be square.
     710           4 :     GUInt16 xpts = nSize + 1;
     711           4 :     GUInt16 ypts = nSize + 1;
     712             : 
     713          12 :     while (read_next_tag(szTag))
     714             :     {
     715           8 :         if (tag_is(szTag, "XPTS"))
     716             :         {
     717           0 :             get(xpts);
     718           0 :             if (xpts < nSize || !skip(2))
     719           0 :                 return FALSE;
     720           0 :             continue;
     721             :         }
     722             : 
     723           8 :         if (tag_is(szTag, "YPTS"))
     724             :         {
     725           0 :             get(ypts);
     726           0 :             if (ypts < nSize || !skip(2))
     727           0 :                 return FALSE;
     728           0 :             continue;
     729             :         }
     730             : 
     731           8 :         if (tag_is(szTag, "SCAL"))
     732             :         {
     733           4 :             float sc[3] = {0.0f};
     734           4 :             get(sc[0]);
     735           4 :             get(sc[1]);
     736           4 :             get(sc[2]);
     737           4 :             m_dSCAL = sc[1];
     738           4 :             continue;
     739             :         }
     740             : 
     741           4 :         if (tag_is(szTag, "CRAD"))
     742             :         {
     743           0 :             if (!skip(sizeof(float)))
     744           0 :                 return FALSE;
     745           0 :             continue;
     746             :         }
     747           4 :         if (tag_is(szTag, "CRVM"))
     748             :         {
     749           0 :             if (!skip(sizeof(GUInt32)))
     750           0 :                 return FALSE;
     751           0 :             continue;
     752             :         }
     753           4 :         if (tag_is(szTag, "ALTW"))
     754             :         {
     755           4 :             get(m_nHeightScale);
     756           4 :             get(m_nBaseHeight);
     757           4 :             m_nDataOffset = VSIFTellL(m_fp);
     758           4 :             if (!skip(static_cast<size_t>(xpts) * static_cast<size_t>(ypts) *
     759             :                       sizeof(GInt16)))
     760           0 :                 return FALSE;
     761           4 :             continue;
     762             :         }
     763           0 :         if (tag_is(szTag, "EOF "))
     764             :         {
     765           0 :             break;
     766             :         }
     767             :     }
     768             : 
     769           4 :     if (xpts == 0 || ypts == 0 || m_nDataOffset == 0)
     770           0 :         return FALSE;
     771             : 
     772           4 :     nRasterXSize = xpts;
     773           4 :     nRasterYSize = ypts;
     774             : 
     775             :     // todo: sanity check: do we have enough pixels?
     776             : 
     777             :     // Cache realworld scaling and offset.
     778           4 :     m_dScale = m_dSCAL / 65536 * m_nHeightScale;
     779           4 :     m_dOffset = m_dSCAL * m_nBaseHeight;
     780           4 :     strcpy(m_szUnits, "m");
     781             : 
     782             :     // Make our projection to have origin at the
     783             :     // NW corner, and groundscale to match elev scale
     784             :     // (i.e., uniform voxels).
     785           4 :     m_adfTransform[0] = 0.0;
     786           4 :     m_adfTransform[1] = m_dSCAL;
     787           4 :     m_adfTransform[2] = 0.0;
     788           4 :     m_adfTransform[3] = 0.0;
     789           4 :     m_adfTransform[4] = 0.0;
     790           4 :     m_adfTransform[5] = m_dSCAL;
     791             : 
     792             :     /* -------------------------------------------------------------------- */
     793             :     /*      Set projection.                                                 */
     794             :     /* -------------------------------------------------------------------- */
     795             :     // Terragen files as of Apr 2006 are partially georeferenced,
     796             :     // we can declare a local coordsys that uses meters.
     797           4 :     m_oSRS.SetLocalCS("Terragen world space");
     798           4 :     m_oSRS.SetLinearUnits("m", 1.0);
     799             : 
     800           4 :     return TRUE;
     801             : }
     802             : 
     803             : /************************************************************************/
     804             : /*                           SetSpatialRef()                            */
     805             : /************************************************************************/
     806             : 
     807           1 : CPLErr TerragenDataset::SetSpatialRef(const OGRSpatialReference *poSRS)
     808             : {
     809             :     // Terragen files aren't really georeferenced, but
     810             :     // we should get the projection's linear units so
     811             :     // that we can scale elevations correctly.
     812             : 
     813             :     // m_dSCAL = 30.0; // default
     814             : 
     815           1 :     m_oSRS.Clear();
     816           1 :     if (poSRS)
     817           1 :         m_oSRS = *poSRS;
     818             : 
     819             :     /* -------------------------------------------------------------------- */
     820             :     /*      Linear units.                                                   */
     821             :     /* -------------------------------------------------------------------- */
     822           1 :     m_bIsGeo = poSRS != nullptr && m_oSRS.IsGeographic() != FALSE;
     823           1 :     if (m_bIsGeo)
     824             :     {
     825             :         // The caller is using degrees. We need to convert
     826             :         // to meters, otherwise we can't derive a SCAL
     827             :         // value to scale elevations with.
     828           0 :         m_bIsGeo = true;
     829             :     }
     830             :     else
     831             :     {
     832           1 :         const double dfLinear = m_oSRS.GetLinearUnits();
     833             : 
     834           1 :         if (approx_equal(dfLinear, 0.3048))
     835           0 :             m_dMetersPerGroundUnit = 0.3048;
     836           1 :         else if (approx_equal(dfLinear, CPLAtof(SRS_UL_US_FOOT_CONV)))
     837           0 :             m_dMetersPerGroundUnit = CPLAtof(SRS_UL_US_FOOT_CONV);
     838             :         else
     839           1 :             m_dMetersPerGroundUnit = 1.0;
     840             :     }
     841             : 
     842           1 :     return CE_None;
     843             : }
     844             : 
     845             : /************************************************************************/
     846             : /*                          GetSpatialRef()                             */
     847             : /************************************************************************/
     848             : 
     849           1 : const OGRSpatialReference *TerragenDataset::GetSpatialRef() const
     850             : {
     851           1 :     return m_oSRS.IsEmpty() ? nullptr : &m_oSRS;
     852             : }
     853             : 
     854             : /************************************************************************/
     855             : /*                          SetGeoTransform()                           */
     856             : /************************************************************************/
     857             : 
     858           1 : CPLErr TerragenDataset::SetGeoTransform(double *padfGeoTransform)
     859             : {
     860           1 :     memcpy(m_adfTransform, padfGeoTransform, sizeof(m_adfTransform));
     861             : 
     862             :     // Average the projection scales.
     863           1 :     m_dGroundScale = average(fabs(m_adfTransform[1]), fabs(m_adfTransform[5]));
     864           1 :     return CE_None;
     865             : }
     866             : 
     867             : /************************************************************************/
     868             : /*                          GetGeoTransform()                           */
     869             : /************************************************************************/
     870             : 
     871           1 : CPLErr TerragenDataset::GetGeoTransform(double *padfTransform)
     872             : {
     873           1 :     memcpy(padfTransform, m_adfTransform, sizeof(m_adfTransform));
     874           1 :     return CE_None;
     875             : }
     876             : 
     877             : /************************************************************************/
     878             : /*                                Create()                                */
     879             : /************************************************************************/
     880          50 : GDALDataset *TerragenDataset::Create(const char *pszFilename, int nXSize,
     881             :                                      int nYSize, int nBandsIn,
     882             :                                      GDALDataType eType, char **papszOptions)
     883             : {
     884          50 :     TerragenDataset *poDS = new TerragenDataset();
     885             : 
     886          50 :     poDS->eAccess = GA_Update;
     887             : 
     888          50 :     poDS->m_pszFilename = CPLStrdup(pszFilename);
     889             : 
     890             :     // --------------------------------------------------------------------
     891             :     //      Verify input options.
     892             :     // --------------------------------------------------------------------
     893          50 :     const char *pszValue = CSLFetchNameValue(papszOptions, "MINUSERPIXELVALUE");
     894          50 :     if (pszValue != nullptr)
     895           1 :         poDS->m_dLogSpan[0] = CPLAtof(pszValue);
     896             : 
     897          50 :     pszValue = CSLFetchNameValue(papszOptions, "MAXUSERPIXELVALUE");
     898          50 :     if (pszValue != nullptr)
     899           1 :         poDS->m_dLogSpan[1] = CPLAtof(pszValue);
     900             : 
     901          50 :     if (poDS->m_dLogSpan[1] <= poDS->m_dLogSpan[0])
     902             :     {
     903          49 :         CPLError(CE_Failure, CPLE_AppDefined,
     904             :                  "Inverted, flat, or unspecified span for Terragen file.");
     905             : 
     906          49 :         delete poDS;
     907          49 :         return nullptr;
     908             :     }
     909             : 
     910           1 :     if (eType != GDT_Float32)
     911             :     {
     912           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     913             :                  "Attempt to create Terragen dataset with a non-float32\n"
     914             :                  "data type (%s).\n",
     915             :                  GDALGetDataTypeName(eType));
     916             : 
     917           0 :         delete poDS;
     918           0 :         return nullptr;
     919             :     }
     920             : 
     921           1 :     if (nBandsIn != 1)
     922             :     {
     923           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     924             :                  "Terragen driver doesn't support %d bands. Must be 1.\n",
     925             :                  nBandsIn);
     926             : 
     927           0 :         delete poDS;
     928           0 :         return nullptr;
     929             :     }
     930             : 
     931             :     // --------------------------------------------------------------------
     932             :     //      Try to create the file.
     933             :     // --------------------------------------------------------------------
     934             : 
     935           1 :     poDS->m_fp = VSIFOpenL(pszFilename, "wb+");
     936             : 
     937           1 :     if (poDS->m_fp == nullptr)
     938             :     {
     939           0 :         CPLError(CE_Failure, CPLE_OpenFailed,
     940             :                  "Attempt to create file `%s' failed.\n", pszFilename);
     941           0 :         delete poDS;
     942           0 :         return nullptr;
     943             :     }
     944             : 
     945           1 :     poDS->nRasterXSize = nXSize;
     946           1 :     poDS->nRasterYSize = nYSize;
     947             : 
     948             :     // Don't bother writing the header here; the first
     949             :     // call to IWriteBlock will do that instead, since
     950             :     // the elevation data's location depends on the
     951             :     // header size.
     952             : 
     953             :     // --------------------------------------------------------------------
     954             :     //      Instance a band.
     955             :     // --------------------------------------------------------------------
     956           1 :     poDS->SetBand(1, new TerragenRasterBand(poDS));
     957             : 
     958             :     // VSIFClose( poDS->m_fp );
     959             : 
     960             :     // return (GDALDataset *) GDALOpen( pszFilename, GA_Update );
     961           1 :     return GDALDataset::FromHandle(poDS);
     962             : }
     963             : 
     964             : /************************************************************************/
     965             : /*                                Open()                                */
     966             : /************************************************************************/
     967             : 
     968       34605 : GDALDataset *TerragenDataset::Open(GDALOpenInfo *poOpenInfo)
     969             : 
     970             : {
     971             :     // The file should have at least 32 header bytes
     972       34605 :     if (poOpenInfo->nHeaderBytes < 32 || poOpenInfo->fpL == nullptr)
     973       28953 :         return nullptr;
     974             : 
     975        5652 :     if (!EQUALN(reinterpret_cast<const char *>(poOpenInfo->pabyHeader),
     976             :                 "TERRAGENTERRAIN ", 16))
     977        5648 :         return nullptr;
     978             : 
     979             :     /* -------------------------------------------------------------------- */
     980             :     /*      Create a corresponding GDALDataset.                             */
     981             :     /* -------------------------------------------------------------------- */
     982           4 :     TerragenDataset *poDS = new TerragenDataset();
     983           4 :     poDS->eAccess = poOpenInfo->eAccess;
     984           4 :     poDS->m_fp = poOpenInfo->fpL;
     985           4 :     poOpenInfo->fpL = nullptr;
     986             : 
     987             :     /* -------------------------------------------------------------------- */
     988             :     /*      Read the file.                                                  */
     989             :     /* -------------------------------------------------------------------- */
     990           4 :     if (!poDS->LoadFromFile())
     991             :     {
     992           0 :         delete poDS;
     993           0 :         return nullptr;
     994             :     }
     995             : 
     996             :     /* -------------------------------------------------------------------- */
     997             :     /*      Create band information objects.                                */
     998             :     /* -------------------------------------------------------------------- */
     999           4 :     poDS->SetBand(1, new TerragenRasterBand(poDS));
    1000             : 
    1001           4 :     poDS->SetMetadataItem(GDALMD_AREA_OR_POINT, GDALMD_AOP_POINT);
    1002             : 
    1003             :     /* -------------------------------------------------------------------- */
    1004             :     /*      Initialize any PAM information.                                 */
    1005             :     /* -------------------------------------------------------------------- */
    1006           4 :     poDS->SetDescription(poOpenInfo->pszFilename);
    1007           4 :     poDS->TryLoadXML();
    1008             : 
    1009             :     /* -------------------------------------------------------------------- */
    1010             :     /*      Support overviews.                                              */
    1011             :     /* -------------------------------------------------------------------- */
    1012           4 :     poDS->oOvManager.Initialize(poDS, poOpenInfo->pszFilename);
    1013             : 
    1014           4 :     return poDS;
    1015             : }
    1016             : 
    1017             : /************************************************************************/
    1018             : /*                        GDALRegister_Terragen()                       */
    1019             : /************************************************************************/
    1020             : 
    1021        1682 : void GDALRegister_Terragen()
    1022             : 
    1023             : {
    1024        1682 :     if (GDALGetDriverByName("Terragen") != nullptr)
    1025         301 :         return;
    1026             : 
    1027        1381 :     GDALDriver *poDriver = new GDALDriver();
    1028             : 
    1029        1381 :     poDriver->SetDescription("Terragen");
    1030        1381 :     poDriver->SetMetadataItem(GDAL_DCAP_RASTER, "YES");
    1031        1381 :     poDriver->SetMetadataItem(GDAL_DMD_EXTENSION, "ter");
    1032        1381 :     poDriver->SetMetadataItem(GDAL_DMD_LONGNAME, "Terragen heightfield");
    1033        1381 :     poDriver->SetMetadataItem(GDAL_DMD_HELPTOPIC,
    1034        1381 :                               "drivers/raster/terragen.html");
    1035             : 
    1036        1381 :     poDriver->SetMetadataItem(
    1037             :         GDAL_DMD_CREATIONOPTIONLIST,
    1038             :         "<CreationOptionList>"
    1039             :         "   <Option name='MINUSERPIXELVALUE' type='float' description='Lowest "
    1040             :         "logical elevation'/>"
    1041             :         "   <Option name='MAXUSERPIXELVALUE' type='float' description='Highest "
    1042             :         "logical elevation'/>"
    1043        1381 :         "</CreationOptionList>");
    1044        1381 :     poDriver->SetMetadataItem(GDAL_DCAP_VIRTUALIO, "YES");
    1045             : 
    1046        1381 :     poDriver->pfnOpen = TerragenDataset::Open;
    1047        1381 :     poDriver->pfnCreate = TerragenDataset::Create;
    1048             : 
    1049        1381 :     GetGDALDriverManager()->RegisterDriver(poDriver);
    1050             : }

Generated by: LCOV version 1.14