LCOV - code coverage report
Current view: top level - frmts/leveller - levellerdataset.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 180 518 34.7 %
Date: 2025-01-18 12:42:00 Functions: 20 57 35.1 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  * levellerdataset.cpp,v 1.22
       3             :  *
       4             :  * Project:  Leveller TER Driver
       5             :  * Purpose:  Reader for Leveller 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             :  ******************************************************************************
      12             :  * Copyright (c) 2005-2007 Daylon Graphics Ltd.
      13             :  * Copyright (c) 2007-2013, Even Rouault <even dot rouault at spatialys.com>
      14             :  *
      15             :  * SPDX-License-Identifier: MIT
      16             :  ****************************************************************************/
      17             : 
      18             : #include "gdal_frmts.h"
      19             : #include "gdal_pam.h"
      20             : #include "ogr_spatialref.h"
      21             : 
      22        1540 : static bool str_equal(const char *_s1, const char *_s2)
      23             : {
      24        1540 :     return 0 == strcmp(_s1, _s2);
      25             : }
      26             : 
      27             : /*GDALDataset *LevellerCreateCopy( const char *, GDALDataset *, int, char **,
      28             :                                 GDALProgressFunc pfnProgress,
      29             :                                 void * pProgressData );
      30             : */
      31             : 
      32             : /************************************************************************/
      33             : /* ==================================================================== */
      34             : /*                              LevellerDataset                         */
      35             : /* ==================================================================== */
      36             : /************************************************************************/
      37             : 
      38             : constexpr size_t kMaxTagNameLen = 63;
      39             : 
      40             : enum
      41             : {
      42             :     // Leveller coordsys types.
      43             :     LEV_COORDSYS_RASTER = 0,
      44             :     LEV_COORDSYS_LOCAL,
      45             :     LEV_COORDSYS_GEO
      46             : };
      47             : 
      48             : enum
      49             : {
      50             :     // Leveller digital axis extent styles.
      51             :     LEV_DA_POSITIONED = 0,
      52             :     LEV_DA_SIZED,
      53             :     LEV_DA_PIXEL_SIZED
      54             : };
      55             : 
      56             : typedef enum
      57             : {
      58             :     // Measurement unit IDs, OEM version.
      59             :     UNITLABEL_UNKNOWN = 0x00000000,
      60             :     UNITLABEL_PIXEL = 0x70780000,
      61             :     UNITLABEL_PERCENT = 0x25000000,
      62             : 
      63             :     UNITLABEL_RADIAN = 0x72616400,
      64             :     UNITLABEL_DEGREE = 0x64656700,
      65             :     UNITLABEL_ARCMINUTE = 0x6172636D,
      66             :     UNITLABEL_ARCSECOND = 0x61726373,
      67             : 
      68             :     UNITLABEL_YM = 0x796D0000,
      69             :     UNITLABEL_ZM = 0x7A6D0000,
      70             :     UNITLABEL_AM = 0x616D0000,
      71             :     UNITLABEL_FM = 0x666D0000,
      72             :     UNITLABEL_PM = 0x706D0000,
      73             :     UNITLABEL_A = 0x41000000,
      74             :     UNITLABEL_NM = 0x6E6D0000,
      75             :     UNITLABEL_U = 0x75000000,
      76             :     UNITLABEL_UM = 0x756D0000,
      77             :     UNITLABEL_PPT = 0x70707400,
      78             :     UNITLABEL_PT = 0x70740000,
      79             :     UNITLABEL_MM = 0x6D6D0000,
      80             :     UNITLABEL_P = 0x70000000,
      81             :     UNITLABEL_CM = 0x636D0000,
      82             :     UNITLABEL_IN = 0x696E0000,
      83             :     UNITLABEL_DFT = 0x64667400,
      84             :     UNITLABEL_DM = 0x646D0000,
      85             :     UNITLABEL_LI = 0x6C690000,
      86             :     UNITLABEL_SLI = 0x736C6900,
      87             :     UNITLABEL_SP = 0x73700000,
      88             :     UNITLABEL_FT = 0x66740000,
      89             :     UNITLABEL_SFT = 0x73667400,
      90             :     UNITLABEL_YD = 0x79640000,
      91             :     UNITLABEL_SYD = 0x73796400,
      92             :     UNITLABEL_M = 0x6D000000,
      93             :     UNITLABEL_FATH = 0x66617468,
      94             :     UNITLABEL_R = 0x72000000,
      95             :     UNITLABEL_RD = UNITLABEL_R,
      96             :     UNITLABEL_DAM = 0x64416D00,
      97             :     UNITLABEL_DKM = UNITLABEL_DAM,
      98             :     UNITLABEL_CH = 0x63680000,
      99             :     UNITLABEL_SCH = 0x73636800,
     100             :     UNITLABEL_HM = 0x686D0000,
     101             :     UNITLABEL_F = 0x66000000,
     102             :     UNITLABEL_KM = 0x6B6D0000,
     103             :     UNITLABEL_MI = 0x6D690000,
     104             :     UNITLABEL_SMI = 0x736D6900,
     105             :     UNITLABEL_NMI = 0x6E6D6900,
     106             :     UNITLABEL_MEGAM = 0x4D6D0000,
     107             :     UNITLABEL_LS = 0x6C730000,
     108             :     UNITLABEL_GM = 0x476D0000,
     109             :     UNITLABEL_LM = 0x6C6D0000,
     110             :     UNITLABEL_AU = 0x41550000,
     111             :     UNITLABEL_TM = 0x546D0000,
     112             :     UNITLABEL_LHR = 0x6C687200,
     113             :     UNITLABEL_LD = 0x6C640000,
     114             :     UNITLABEL_PETAM = 0x506D0000,
     115             :     UNITLABEL_LY = 0x6C790000,
     116             :     UNITLABEL_PC = 0x70630000,
     117             :     UNITLABEL_EXAM = 0x456D0000,
     118             :     UNITLABEL_KLY = 0x6B6C7900,
     119             :     UNITLABEL_KPC = 0x6B706300,
     120             :     UNITLABEL_ZETTAM = 0x5A6D0000,
     121             :     UNITLABEL_MLY = 0x4D6C7900,
     122             :     UNITLABEL_MPC = 0x4D706300,
     123             :     UNITLABEL_YOTTAM = 0x596D0000
     124             : } UNITLABEL;
     125             : 
     126             : typedef struct
     127             : {
     128             :     const char *pszID;
     129             :     double dScale;
     130             :     UNITLABEL oemCode;
     131             : } measurement_unit;
     132             : 
     133             : constexpr double kdays_per_year = 365.25;
     134             : constexpr double kdLStoM = 299792458.0;
     135             : constexpr double kdLYtoM = kdLStoM * kdays_per_year * 24 * 60 * 60;
     136             : constexpr double kdInch = 0.3048 / 12;
     137             : constexpr double kPI = M_PI;
     138             : 
     139             : constexpr int kFirstLinearMeasureIdx = 9;
     140             : 
     141             : static const measurement_unit kUnits[] = {
     142             :     {"", 1.0, UNITLABEL_UNKNOWN},
     143             :     {"px", 1.0, UNITLABEL_PIXEL},
     144             :     {"%", 1.0, UNITLABEL_PERCENT},  // not actually used
     145             : 
     146             :     {"rad", 1.0, UNITLABEL_RADIAN},
     147             :     {"\xB0", kPI / 180.0, UNITLABEL_DEGREE},  // \xB0 is Unicode degree symbol
     148             :     {"d", kPI / 180.0, UNITLABEL_DEGREE},
     149             :     {"deg", kPI / 180.0, UNITLABEL_DEGREE},
     150             :     {"'", kPI / (60.0 * 180.0), UNITLABEL_ARCMINUTE},
     151             :     {"\"", kPI / (3600.0 * 180.0), UNITLABEL_ARCSECOND},
     152             : 
     153             :     {"ym", 1.0e-24, UNITLABEL_YM},
     154             :     {"zm", 1.0e-21, UNITLABEL_ZM},
     155             :     {"am", 1.0e-18, UNITLABEL_AM},
     156             :     {"fm", 1.0e-15, UNITLABEL_FM},
     157             :     {"pm", 1.0e-12, UNITLABEL_PM},
     158             :     {"A", 1.0e-10, UNITLABEL_A},
     159             :     {"nm", 1.0e-9, UNITLABEL_NM},
     160             :     {"u", 1.0e-6, UNITLABEL_U},
     161             :     {"um", 1.0e-6, UNITLABEL_UM},
     162             :     {"ppt", kdInch / 72.27, UNITLABEL_PPT},
     163             :     {"pt", kdInch / 72.0, UNITLABEL_PT},
     164             :     {"mm", 1.0e-3, UNITLABEL_MM},
     165             :     {"p", kdInch / 6.0, UNITLABEL_P},
     166             :     {"cm", 1.0e-2, UNITLABEL_CM},
     167             :     {"in", kdInch, UNITLABEL_IN},
     168             :     {"dft", 0.03048, UNITLABEL_DFT},
     169             :     {"dm", 0.1, UNITLABEL_DM},
     170             :     {"li", 0.2011684 /* GDAL 0.20116684023368047 ? */, UNITLABEL_LI},
     171             :     {"sli", 0.201168402336805, UNITLABEL_SLI},
     172             :     {"sp", 0.2286, UNITLABEL_SP},
     173             :     {"ft", 0.3048, UNITLABEL_FT},
     174             :     {"sft", 1200.0 / 3937.0, UNITLABEL_SFT},
     175             :     {"yd", 0.9144, UNITLABEL_YD},
     176             :     {"syd", 0.914401828803658, UNITLABEL_SYD},
     177             :     {"m", 1.0, UNITLABEL_M},
     178             :     {"fath", 1.8288, UNITLABEL_FATH},
     179             :     {"rd", 5.02921, UNITLABEL_RD},
     180             :     {"dam", 10.0, UNITLABEL_DAM},
     181             :     {"dkm", 10.0, UNITLABEL_DKM},
     182             :     {"ch", 20.1168 /* GDAL: 2.0116684023368047 ? */, UNITLABEL_CH},
     183             :     {"sch", 20.1168402336805, UNITLABEL_SCH},
     184             :     {"hm", 100.0, UNITLABEL_HM},
     185             :     {"f", 201.168, UNITLABEL_F},
     186             :     {"km", 1000.0, UNITLABEL_KM},
     187             :     {"mi", 1609.344, UNITLABEL_MI},
     188             :     {"smi", 1609.34721869444, UNITLABEL_SMI},
     189             :     {"nmi", 1853.0, UNITLABEL_NMI},
     190             :     {"Mm", 1.0e+6, UNITLABEL_MEGAM},
     191             :     {"ls", kdLStoM, UNITLABEL_LS},
     192             :     {"Gm", 1.0e+9, UNITLABEL_GM},
     193             :     {"lm", kdLStoM * 60, UNITLABEL_LM},
     194             :     {"AU", 8.317 * kdLStoM * 60, UNITLABEL_AU},
     195             :     {"Tm", 1.0e+12, UNITLABEL_TM},
     196             :     {"lhr", 60.0 * 60.0 * kdLStoM, UNITLABEL_LHR},
     197             :     {"ld", 24 * 60.0 * 60.0 * kdLStoM, UNITLABEL_LD},
     198             :     {"Pm", 1.0e+15, UNITLABEL_PETAM},
     199             :     {"ly", kdLYtoM, UNITLABEL_LY},
     200             :     {"pc", 3.2616 * kdLYtoM, UNITLABEL_PC},
     201             :     {"Em", 1.0e+18, UNITLABEL_EXAM},
     202             :     {"kly", 1.0e+3 * kdLYtoM, UNITLABEL_KLY},
     203             :     {"kpc", 3.2616 * 1.0e+3 * kdLYtoM, UNITLABEL_KPC},
     204             :     {"Zm", 1.0e+21, UNITLABEL_ZETTAM},
     205             :     {"Mly", 1.0e+6 * kdLYtoM, UNITLABEL_MLY},
     206             :     {"Mpc", 3.2616 * 1.0e+6 * kdLYtoM, UNITLABEL_MPC},
     207             :     {"Ym", 1.0e+24, UNITLABEL_YOTTAM}};
     208             : 
     209             : // ----------------------------------------------------------------
     210             : 
     211           0 : static bool approx_equal(double a, double b)
     212             : {
     213           0 :     const double epsilon = 1e-5;
     214           0 :     return fabs(a - b) <= epsilon;
     215             : }
     216             : 
     217             : // ----------------------------------------------------------------
     218             : 
     219             : class LevellerRasterBand;
     220             : 
     221             : class LevellerDataset final : public GDALPamDataset
     222             : {
     223             :     friend class LevellerRasterBand;
     224             :     friend class digital_axis;
     225             : 
     226             :     int m_version;
     227             : 
     228             :     char *m_pszFilename;
     229             :     OGRSpatialReference m_oSRS{};
     230             : 
     231             :     // char              m_szUnits[8];
     232             :     char m_szElevUnits[8];
     233             :     double m_dElevScale;  // physical-to-logical scaling.
     234             :     double m_dElevBase;   // logical offset.
     235             :     double m_adfTransform[6];
     236             :     // double            m_dMeasurePerPixel;
     237             :     double m_dLogSpan[2];
     238             : 
     239             :     VSILFILE *m_fp;
     240             :     vsi_l_offset m_nDataOffset;
     241             : 
     242             :     bool load_from_file(VSILFILE *, const char *);
     243             : 
     244             :     static bool locate_data(vsi_l_offset &, size_t &, VSILFILE *, const char *);
     245             :     static bool get(int &, VSILFILE *, const char *);
     246             : 
     247           0 :     static bool get(size_t &n, VSILFILE *fp, const char *psz)
     248             :     {
     249           0 :         return get((int &)n, fp, psz);
     250             :     }
     251             : 
     252             :     static bool get(double &, VSILFILE *, const char *);
     253             :     static bool get(char *, size_t, VSILFILE *, const char *);
     254             : 
     255             :     bool write_header();
     256             :     bool write_tag(const char *, int);
     257             :     bool write_tag(const char *, size_t);
     258             :     bool write_tag(const char *, double);
     259             :     bool write_tag(const char *, const char *);
     260             :     bool write_tag_start(const char *, size_t);
     261             :     bool write(int);
     262             :     bool write(size_t);
     263             :     bool write(double);
     264             :     bool write_byte(size_t);
     265             : 
     266             :     static const measurement_unit *get_uom(const char *);
     267             :     static const measurement_unit *get_uom(UNITLABEL);
     268             :     static const measurement_unit *get_uom(double);
     269             : 
     270             :     static bool convert_measure(double, double &, const char *pszUnitsFrom);
     271             :     bool make_local_coordsys(const char *pszName, const char *pszUnits);
     272             :     bool make_local_coordsys(const char *pszName, UNITLABEL);
     273             :     const char *code_to_id(UNITLABEL) const;
     274             :     UNITLABEL id_to_code(const char *) const;
     275             :     UNITLABEL meter_measure_to_code(double) const;
     276             :     bool compute_elev_scaling(const OGRSpatialReference &);
     277             :     void raw_to_proj(double, double, double &, double &) const;
     278             : 
     279             :   public:
     280             :     LevellerDataset();
     281             :     virtual ~LevellerDataset();
     282             : 
     283             :     static GDALDataset *Open(GDALOpenInfo *);
     284             :     static int Identify(GDALOpenInfo *);
     285             :     static GDALDataset *Create(const char *pszFilename, int nXSize, int nYSize,
     286             :                                int nBandsIn, GDALDataType eType,
     287             :                                char **papszOptions);
     288             : 
     289             :     virtual CPLErr GetGeoTransform(double *) override;
     290             : 
     291             :     virtual CPLErr SetGeoTransform(double *) override;
     292             : 
     293             :     const OGRSpatialReference *GetSpatialRef() const override;
     294             :     CPLErr SetSpatialRef(const OGRSpatialReference *poSRS) override;
     295             : };
     296             : 
     297             : class digital_axis
     298             : {
     299             :   public:
     300           0 :     digital_axis() : m_eStyle(LEV_DA_PIXEL_SIZED), m_fixedEnd(0)
     301             :     {
     302           0 :         m_d[0] = 0.0;
     303           0 :         m_d[1] = 0.0;
     304           0 :     }
     305             : 
     306           0 :     bool get(LevellerDataset &ds, VSILFILE *fp, int n)
     307             :     {
     308             :         char szTag[32];
     309           0 :         snprintf(szTag, sizeof(szTag), "coordsys_da%d_style", n);
     310           0 :         if (!ds.get(m_eStyle, fp, szTag))
     311           0 :             return false;
     312           0 :         snprintf(szTag, sizeof(szTag), "coordsys_da%d_fixedend", n);
     313           0 :         if (!ds.get(m_fixedEnd, fp, szTag))
     314           0 :             return false;
     315           0 :         snprintf(szTag, sizeof(szTag), "coordsys_da%d_v0", n);
     316           0 :         if (!ds.get(m_d[0], fp, szTag))
     317           0 :             return false;
     318           0 :         snprintf(szTag, sizeof(szTag), "coordsys_da%d_v1", n);
     319           0 :         if (!ds.get(m_d[1], fp, szTag))
     320           0 :             return false;
     321           0 :         return true;
     322             :     }
     323             : 
     324           0 :     double origin(size_t pixels) const
     325             :     {
     326           0 :         if (m_fixedEnd == 1)
     327             :         {
     328           0 :             switch (m_eStyle)
     329             :             {
     330           0 :                 case LEV_DA_SIZED:
     331           0 :                     return m_d[1] + m_d[0];
     332             : 
     333           0 :                 case LEV_DA_PIXEL_SIZED:
     334           0 :                     return m_d[1] + (m_d[0] * (pixels - 1));
     335             :             }
     336             :         }
     337           0 :         return m_d[0];
     338             :     }
     339             : 
     340           0 :     double scaling(size_t pixels) const
     341             :     {
     342           0 :         CPLAssert(pixels > 1);
     343           0 :         if (m_eStyle == LEV_DA_PIXEL_SIZED)
     344           0 :             return m_d[1 - m_fixedEnd];
     345             : 
     346           0 :         return this->length(static_cast<int>(pixels)) / (pixels - 1);
     347             :     }
     348             : 
     349           0 :     double length(int pixels) const
     350             :     {
     351             :         // Return the signed length of the axis.
     352             : 
     353           0 :         switch (m_eStyle)
     354             :         {
     355           0 :             case LEV_DA_POSITIONED:
     356           0 :                 return m_d[1] - m_d[0];
     357             : 
     358           0 :             case LEV_DA_SIZED:
     359           0 :                 return m_d[1 - m_fixedEnd];
     360             : 
     361           0 :             case LEV_DA_PIXEL_SIZED:
     362           0 :                 return m_d[1 - m_fixedEnd] * (pixels - 1);
     363             :         }
     364           0 :         CPLAssert(false);
     365             :         return 0.0;
     366             :     }
     367             : 
     368             :   protected:
     369             :     int m_eStyle;
     370             :     size_t m_fixedEnd;
     371             :     double m_d[2];
     372             : };
     373             : 
     374             : /************************************************************************/
     375             : /* ==================================================================== */
     376             : /*                            LevellerRasterBand                        */
     377             : /* ==================================================================== */
     378             : /************************************************************************/
     379             : 
     380             : class LevellerRasterBand final : public GDALPamRasterBand
     381             : {
     382             :     friend class LevellerDataset;
     383             : 
     384             :     float *m_pLine;
     385             :     bool m_bFirstTime;
     386             : 
     387             :   public:
     388             :     explicit LevellerRasterBand(LevellerDataset *);
     389             :     virtual ~LevellerRasterBand();
     390             : 
     391             :     bool Init();
     392             : 
     393             :     // Geomeasure support.
     394             :     virtual const char *GetUnitType() override;
     395             :     virtual double GetScale(int *pbSuccess = nullptr) override;
     396             :     virtual double GetOffset(int *pbSuccess = nullptr) override;
     397             : 
     398             :     virtual CPLErr IReadBlock(int, int, void *) override;
     399             :     virtual CPLErr IWriteBlock(int, int, void *) override;
     400             :     virtual CPLErr SetUnitType(const char *) override;
     401             : };
     402             : 
     403             : /************************************************************************/
     404             : /*                         LevellerRasterBand()                         */
     405             : /************************************************************************/
     406             : 
     407           2 : LevellerRasterBand::LevellerRasterBand(LevellerDataset *poDSIn)
     408           2 :     : m_pLine(nullptr), m_bFirstTime(true)
     409             : {
     410           2 :     poDS = poDSIn;
     411           2 :     nBand = 1;
     412             : 
     413           2 :     eDataType = GDT_Float32;
     414             : 
     415           2 :     nBlockXSize = poDS->GetRasterXSize();
     416           2 :     nBlockYSize = 1;  // poDS->GetRasterYSize();
     417           2 : }
     418             : 
     419             : /************************************************************************/
     420             : /*                           Init()                                     */
     421             : /************************************************************************/
     422             : 
     423           2 : bool LevellerRasterBand::Init()
     424             : {
     425           2 :     m_pLine = reinterpret_cast<float *>(
     426           2 :         VSI_MALLOC2_VERBOSE(sizeof(float), nBlockXSize));
     427           2 :     return m_pLine != nullptr;
     428             : }
     429             : 
     430           4 : LevellerRasterBand::~LevellerRasterBand()
     431             : {
     432           2 :     CPLFree(m_pLine);
     433           4 : }
     434             : 
     435             : /************************************************************************/
     436             : /*                             IWriteBlock()                            */
     437             : /************************************************************************/
     438             : 
     439           0 : CPLErr LevellerRasterBand::IWriteBlock(CPL_UNUSED int nBlockXOff,
     440             :                                        int nBlockYOff, void *pImage)
     441             : {
     442           0 :     CPLAssert(nBlockXOff == 0);
     443           0 :     CPLAssert(pImage != nullptr);
     444           0 :     CPLAssert(m_pLine != nullptr);
     445             : 
     446             :     /*  #define sgn(_n) ((_n) < 0 ? -1 : ((_n) > 0 ? 1 : 0) )
     447             :         #define sround(_f)                          \
     448             :         (int)((_f) + (0.5 * sgn(_f)))
     449             :     */
     450           0 :     const size_t pixelsize = sizeof(float);
     451             : 
     452           0 :     LevellerDataset &ds = *reinterpret_cast<LevellerDataset *>(poDS);
     453           0 :     if (m_bFirstTime)
     454             :     {
     455           0 :         m_bFirstTime = false;
     456           0 :         if (!ds.write_header())
     457           0 :             return CE_Failure;
     458           0 :         ds.m_nDataOffset = VSIFTellL(ds.m_fp);
     459             :     }
     460           0 :     const size_t rowbytes = nBlockXSize * pixelsize;
     461           0 :     const float *pfImage = reinterpret_cast<float *>(pImage);
     462             : 
     463           0 :     if (0 ==
     464           0 :         VSIFSeekL(ds.m_fp, ds.m_nDataOffset + nBlockYOff * rowbytes, SEEK_SET))
     465             :     {
     466           0 :         for (size_t x = 0; x < (size_t)nBlockXSize; x++)
     467             :         {
     468             :             // Convert logical elevations to physical.
     469           0 :             m_pLine[x] = static_cast<float>((pfImage[x] - ds.m_dElevBase) /
     470           0 :                                             ds.m_dElevScale);
     471             :         }
     472             : 
     473             : #ifdef CPL_MSB
     474             :         GDALSwapWords(m_pLine, pixelsize, nBlockXSize, pixelsize);
     475             : #endif
     476           0 :         if (1 == VSIFWriteL(m_pLine, rowbytes, 1, ds.m_fp))
     477           0 :             return CE_None;
     478             :     }
     479             : 
     480           0 :     return CE_Failure;
     481             : }
     482             : 
     483           0 : CPLErr LevellerRasterBand::SetUnitType(const char *psz)
     484             : {
     485           0 :     LevellerDataset &ds = *reinterpret_cast<LevellerDataset *>(poDS);
     486             : 
     487           0 :     if (strlen(psz) >= sizeof(ds.m_szElevUnits))
     488           0 :         return CE_Failure;
     489             : 
     490           0 :     strcpy(ds.m_szElevUnits, psz);
     491             : 
     492           0 :     return CE_None;
     493             : }
     494             : 
     495             : /************************************************************************/
     496             : /*                             IReadBlock()                             */
     497             : /************************************************************************/
     498             : 
     499          96 : CPLErr LevellerRasterBand::IReadBlock(CPL_UNUSED int nBlockXOff, int nBlockYOff,
     500             :                                       void *pImage)
     501             : 
     502             : {
     503             :     CPLAssert(sizeof(float) == sizeof(GInt32));
     504          96 :     CPLAssert(nBlockXOff == 0);
     505          96 :     CPLAssert(pImage != nullptr);
     506             : 
     507          96 :     LevellerDataset *poGDS = reinterpret_cast<LevellerDataset *>(poDS);
     508             : 
     509             :     /* -------------------------------------------------------------------- */
     510             :     /*      Seek to scanline.                                               */
     511             :     /* -------------------------------------------------------------------- */
     512          96 :     const size_t rowbytes = nBlockXSize * sizeof(float);
     513             : 
     514          96 :     if (0 != VSIFSeekL(poGDS->m_fp,
     515          96 :                        poGDS->m_nDataOffset + nBlockYOff * rowbytes, SEEK_SET))
     516             :     {
     517           0 :         CPLError(CE_Failure, CPLE_FileIO, "Leveller seek failed: %s",
     518           0 :                  VSIStrerror(errno));
     519           0 :         return CE_Failure;
     520             :     }
     521             : 
     522             :     /* -------------------------------------------------------------------- */
     523             :     /*      Read the scanline into the image buffer.                        */
     524             :     /* -------------------------------------------------------------------- */
     525             : 
     526          96 :     if (VSIFReadL(pImage, rowbytes, 1, poGDS->m_fp) != 1)
     527             :     {
     528           0 :         CPLError(CE_Failure, CPLE_FileIO, "Leveller read failed: %s",
     529           0 :                  VSIStrerror(errno));
     530           0 :         return CE_Failure;
     531             :     }
     532             : 
     533             : /* -------------------------------------------------------------------- */
     534             : /*      Swap on MSB platforms.                                          */
     535             : /* -------------------------------------------------------------------- */
     536             : #ifdef CPL_MSB
     537             :     GDALSwapWords(pImage, 4, nRasterXSize, 4);
     538             : #endif
     539             : 
     540             :     /* -------------------------------------------------------------------- */
     541             :     /*      Convert from legacy-format fixed-point if necessary.            */
     542             :     /* -------------------------------------------------------------------- */
     543          96 :     float *pf = reinterpret_cast<float *>(pImage);
     544             : 
     545          96 :     if (poGDS->m_version < 6)
     546             :     {
     547           0 :         GInt32 *pi = reinterpret_cast<int *>(pImage);
     548           0 :         for (size_t i = 0; i < (size_t)nBlockXSize; i++)
     549           0 :             pf[i] = static_cast<float>(pi[i]) / 65536;
     550             :     }
     551             : 
     552             : /* -------------------------------------------------------------------- */
     553             : /*      Convert raw elevations to realworld elevs.                      */
     554             : /* -------------------------------------------------------------------- */
     555             : #if 0
     556             :     for(size_t i = 0; i < nBlockXSize; i++)
     557             :         pf[i] *= poGDS->m_dWorldscale; //this->GetScale();
     558             : #endif
     559             : 
     560          96 :     return CE_None;
     561             : }
     562             : 
     563             : /************************************************************************/
     564             : /*                            GetUnitType()                             */
     565             : /************************************************************************/
     566           0 : const char *LevellerRasterBand::GetUnitType()
     567             : {
     568             :     // Return elevation units.
     569             : 
     570           0 :     LevellerDataset *poGDS = reinterpret_cast<LevellerDataset *>(poDS);
     571             : 
     572           0 :     return poGDS->m_szElevUnits;
     573             : }
     574             : 
     575             : /************************************************************************/
     576             : /*                              GetScale()                              */
     577             : /************************************************************************/
     578             : 
     579           0 : double LevellerRasterBand::GetScale(int *pbSuccess)
     580             : {
     581           0 :     LevellerDataset *poGDS = reinterpret_cast<LevellerDataset *>(poDS);
     582           0 :     if (pbSuccess != nullptr)
     583           0 :         *pbSuccess = TRUE;
     584           0 :     return poGDS->m_dElevScale;
     585             : }
     586             : 
     587             : /************************************************************************/
     588             : /*                             GetOffset()                              */
     589             : /************************************************************************/
     590             : 
     591           0 : double LevellerRasterBand::GetOffset(int *pbSuccess)
     592             : {
     593           0 :     LevellerDataset *poGDS = reinterpret_cast<LevellerDataset *>(poDS);
     594           0 :     if (pbSuccess != nullptr)
     595           0 :         *pbSuccess = TRUE;
     596           0 :     return poGDS->m_dElevBase;
     597             : }
     598             : 
     599             : /************************************************************************/
     600             : /* ==================================================================== */
     601             : /*                              LevellerDataset                         */
     602             : /* ==================================================================== */
     603             : /************************************************************************/
     604             : 
     605             : /************************************************************************/
     606             : /*                          LevellerDataset()                           */
     607             : /************************************************************************/
     608             : 
     609           6 : LevellerDataset::LevellerDataset()
     610             :     : m_version(0), m_pszFilename(nullptr), m_dElevScale(), m_dElevBase(),
     611           6 :       m_fp(nullptr), m_nDataOffset()
     612             : {
     613           6 :     m_oSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
     614           6 :     memset(m_szElevUnits, 0, sizeof(m_szElevUnits));
     615           6 :     memset(m_adfTransform, 0, sizeof(m_adfTransform));
     616           6 :     memset(m_dLogSpan, 0, sizeof(m_dLogSpan));
     617           6 : }
     618             : 
     619             : /************************************************************************/
     620             : /*                          ~LevellerDataset()                          */
     621             : /************************************************************************/
     622             : 
     623          12 : LevellerDataset::~LevellerDataset()
     624             : {
     625           6 :     FlushCache(true);
     626             : 
     627           6 :     CPLFree(m_pszFilename);
     628             : 
     629           6 :     if (m_fp != nullptr)
     630           4 :         VSIFCloseL(m_fp);
     631          12 : }
     632             : 
     633           0 : static double degrees_to_radians(double d)
     634             : {
     635           0 :     return d * (M_PI / 180);
     636             : }
     637             : 
     638           0 : static double average(double a, double b)
     639             : {
     640           0 :     return 0.5 * (a + b);
     641             : }
     642             : 
     643           0 : void LevellerDataset::raw_to_proj(double x, double y, double &xp,
     644             :                                   double &yp) const
     645             : {
     646           0 :     xp = x * m_adfTransform[1] + m_adfTransform[0];
     647           0 :     yp = y * m_adfTransform[5] + m_adfTransform[3];
     648           0 : }
     649             : 
     650           0 : bool LevellerDataset::compute_elev_scaling(const OGRSpatialReference &sr)
     651             : {
     652           0 :     const char *pszGroundUnits = nullptr;
     653             : 
     654           0 :     if (!sr.IsGeographic())
     655             :     {
     656             :         // For projected or local CS, the elev scale is
     657             :         // the average ground scale.
     658           0 :         m_dElevScale = average(m_adfTransform[1], m_adfTransform[5]);
     659             : 
     660           0 :         const double dfLinear = sr.GetLinearUnits();
     661           0 :         const measurement_unit *pu = this->get_uom(dfLinear);
     662           0 :         if (pu == nullptr)
     663           0 :             return false;
     664             : 
     665           0 :         pszGroundUnits = pu->pszID;
     666             :     }
     667             :     else
     668             :     {
     669           0 :         pszGroundUnits = "m";
     670             : 
     671           0 :         const double kdEarthCircumPolar = 40007849;
     672           0 :         const double kdEarthCircumEquat = 40075004;
     673             : 
     674           0 :         const double xr = 0.5 * nRasterXSize;
     675           0 :         const double yr = 0.5 * nRasterYSize;
     676             : 
     677             :         double xg[2], yg[2];
     678           0 :         raw_to_proj(xr, yr, xg[0], yg[0]);
     679           0 :         raw_to_proj(xr + 1, yr + 1, xg[1], yg[1]);
     680             : 
     681             :         // The earths' circumference shrinks using a sin()
     682             :         // curve as we go up in latitude.
     683             :         const double dLatCircum =
     684           0 :             kdEarthCircumEquat * sin(degrees_to_radians(90.0 - yg[0]));
     685             : 
     686             :         // Derive meter distance between geolongitudes
     687             :         // in xg[0] and xg[1].
     688           0 :         const double dx = fabs(xg[1] - xg[0]) / 360.0 * dLatCircum;
     689           0 :         const double dy = fabs(yg[1] - yg[0]) / 360.0 * kdEarthCircumPolar;
     690             : 
     691           0 :         m_dElevScale = average(dx, dy);
     692             :     }
     693             : 
     694           0 :     m_dElevBase = m_dLogSpan[0];
     695             : 
     696             :     // Convert from ground units to elev units.
     697           0 :     const measurement_unit *puG = this->get_uom(pszGroundUnits);
     698           0 :     const measurement_unit *puE = this->get_uom(m_szElevUnits);
     699             : 
     700           0 :     if (puG == nullptr || puE == nullptr)
     701           0 :         return false;
     702             : 
     703           0 :     const double g_to_e = puG->dScale / puE->dScale;
     704             : 
     705           0 :     m_dElevScale *= g_to_e;
     706           0 :     return true;
     707             : }
     708             : 
     709           0 : bool LevellerDataset::write_header()
     710             : {
     711             :     char szHeader[5];
     712           0 :     strcpy(szHeader, "trrn");
     713           0 :     szHeader[4] = 7;  // TER v7 introduced w/ Lev 2.6.
     714             : 
     715           0 :     if (1 != VSIFWriteL(szHeader, 5, 1, m_fp) ||
     716           0 :         !this->write_tag("hf_w", (size_t)nRasterXSize) ||
     717           0 :         !this->write_tag("hf_b", (size_t)nRasterYSize))
     718             :     {
     719           0 :         CPLError(CE_Failure, CPLE_FileIO, "Could not write header");
     720           0 :         return false;
     721             :     }
     722             : 
     723           0 :     m_dElevBase = 0.0;
     724           0 :     m_dElevScale = 1.0;
     725             : 
     726           0 :     if (m_oSRS.IsEmpty())
     727             :     {
     728           0 :         write_tag("csclass", LEV_COORDSYS_RASTER);
     729             :     }
     730             :     else
     731             :     {
     732           0 :         char *pszWkt = nullptr;
     733           0 :         m_oSRS.exportToWkt(&pszWkt);
     734           0 :         if (pszWkt)
     735           0 :             write_tag("coordsys_wkt", pszWkt);
     736           0 :         CPLFree(pszWkt);
     737           0 :         const UNITLABEL units_elev = this->id_to_code(m_szElevUnits);
     738             : 
     739           0 :         const int bHasECS =
     740           0 :             (units_elev != UNITLABEL_PIXEL && units_elev != UNITLABEL_UNKNOWN);
     741             : 
     742           0 :         write_tag("coordsys_haselevm", bHasECS);
     743             : 
     744           0 :         if (bHasECS)
     745             :         {
     746           0 :             if (!this->compute_elev_scaling(m_oSRS))
     747           0 :                 return false;
     748             : 
     749             :             // Raw-to-real units scaling.
     750           0 :             write_tag("coordsys_em_scale", m_dElevScale);
     751             : 
     752             :             // Elev offset, in real units.
     753           0 :             write_tag("coordsys_em_base", m_dElevBase);
     754           0 :             write_tag("coordsys_em_units", units_elev);
     755             :         }
     756             : 
     757           0 :         if (m_oSRS.IsLocal())
     758             :         {
     759           0 :             write_tag("csclass", LEV_COORDSYS_LOCAL);
     760             : 
     761           0 :             const double dfLinear = m_oSRS.GetLinearUnits();
     762           0 :             const int n = this->meter_measure_to_code(dfLinear);
     763           0 :             write_tag("coordsys_units", n);
     764             :         }
     765             :         else
     766             :         {
     767           0 :             write_tag("csclass", LEV_COORDSYS_GEO);
     768             :         }
     769             : 
     770           0 :         if (m_adfTransform[2] != 0.0 || m_adfTransform[4] != 0.0)
     771             :         {
     772           0 :             CPLError(CE_Failure, CPLE_IllegalArg,
     773             :                      "Cannot handle rotated geotransform");
     774           0 :             return false;
     775             :         }
     776             : 
     777             :         // todo: GDAL gridpost spacing is based on extent / rastersize
     778             :         // instead of extent / (rastersize-1) like Leveller.
     779             :         // We need to look into this and adjust accordingly.
     780             : 
     781             :         // Write north-south digital axis.
     782           0 :         write_tag("coordsys_da0_style", LEV_DA_PIXEL_SIZED);
     783           0 :         write_tag("coordsys_da0_fixedend", 0);
     784           0 :         write_tag("coordsys_da0_v0", m_adfTransform[3]);
     785           0 :         write_tag("coordsys_da0_v1", m_adfTransform[5]);
     786             : 
     787             :         // Write east-west digital axis.
     788           0 :         write_tag("coordsys_da1_style", LEV_DA_PIXEL_SIZED);
     789           0 :         write_tag("coordsys_da1_fixedend", 0);
     790           0 :         write_tag("coordsys_da1_v0", m_adfTransform[0]);
     791           0 :         write_tag("coordsys_da1_v1", m_adfTransform[1]);
     792             :     }
     793             : 
     794           0 :     this->write_tag_start("hf_data",
     795           0 :                           sizeof(float) * nRasterXSize * nRasterYSize);
     796             : 
     797           0 :     return true;
     798             : }
     799             : 
     800             : /************************************************************************/
     801             : /*                          SetGeoTransform()                           */
     802             : /************************************************************************/
     803             : 
     804           0 : CPLErr LevellerDataset::SetGeoTransform(double *padfGeoTransform)
     805             : {
     806           0 :     memcpy(m_adfTransform, padfGeoTransform, sizeof(m_adfTransform));
     807             : 
     808           0 :     return CE_None;
     809             : }
     810             : 
     811             : /************************************************************************/
     812             : /*                           SetSpatialRef()                            */
     813             : /************************************************************************/
     814             : 
     815           0 : CPLErr LevellerDataset::SetSpatialRef(const OGRSpatialReference *poSRS)
     816             : {
     817           0 :     m_oSRS.Clear();
     818           0 :     if (poSRS)
     819           0 :         m_oSRS = *poSRS;
     820             : 
     821           0 :     return CE_None;
     822             : }
     823             : 
     824             : /************************************************************************/
     825             : /*                           Create()                                   */
     826             : /************************************************************************/
     827          49 : GDALDataset *LevellerDataset::Create(const char *pszFilename, int nXSize,
     828             :                                      int nYSize, int nBandsIn,
     829             :                                      GDALDataType eType, char **papszOptions)
     830             : {
     831          49 :     if (nBandsIn != 1)
     832             :     {
     833          22 :         CPLError(CE_Failure, CPLE_IllegalArg, "Band count must be 1");
     834          22 :         return nullptr;
     835             :     }
     836             : 
     837          27 :     if (eType != GDT_Float32)
     838             :     {
     839          23 :         CPLError(CE_Failure, CPLE_IllegalArg, "Pixel type must be Float32");
     840          23 :         return nullptr;
     841             :     }
     842             : 
     843           4 :     if (nXSize < 2 || nYSize < 2)
     844             :     {
     845           0 :         CPLError(CE_Failure, CPLE_IllegalArg,
     846             :                  "One or more raster dimensions too small");
     847           0 :         return nullptr;
     848             :     }
     849             : 
     850           4 :     LevellerDataset *poDS = new LevellerDataset;
     851             : 
     852           4 :     poDS->eAccess = GA_Update;
     853             : 
     854           4 :     poDS->m_pszFilename = CPLStrdup(pszFilename);
     855             : 
     856           4 :     poDS->m_fp = VSIFOpenL(pszFilename, "wb+");
     857             : 
     858           4 :     if (poDS->m_fp == nullptr)
     859             :     {
     860           2 :         CPLError(CE_Failure, CPLE_OpenFailed,
     861             :                  "Attempt to create file `%s' failed.", pszFilename);
     862           2 :         delete poDS;
     863           2 :         return nullptr;
     864             :     }
     865             : 
     866             :     // Header will be written the first time IWriteBlock
     867             :     // is called.
     868             : 
     869           2 :     poDS->nRasterXSize = nXSize;
     870           2 :     poDS->nRasterYSize = nYSize;
     871             : 
     872           2 :     const char *pszValue = CSLFetchNameValue(papszOptions, "MINUSERPIXELVALUE");
     873           2 :     if (pszValue != nullptr)
     874           0 :         poDS->m_dLogSpan[0] = CPLAtof(pszValue);
     875             :     else
     876             :     {
     877           2 :         delete poDS;
     878           2 :         CPLError(CE_Failure, CPLE_IllegalArg,
     879             :                  "MINUSERPIXELVALUE must be specified.");
     880           2 :         return nullptr;
     881             :     }
     882             : 
     883           0 :     pszValue = CSLFetchNameValue(papszOptions, "MAXUSERPIXELVALUE");
     884           0 :     if (pszValue != nullptr)
     885           0 :         poDS->m_dLogSpan[1] = CPLAtof(pszValue);
     886             : 
     887           0 :     if (poDS->m_dLogSpan[1] < poDS->m_dLogSpan[0])
     888             :     {
     889           0 :         double t = poDS->m_dLogSpan[0];
     890           0 :         poDS->m_dLogSpan[0] = poDS->m_dLogSpan[1];
     891           0 :         poDS->m_dLogSpan[1] = t;
     892             :     }
     893             : 
     894             :     // --------------------------------------------------------------------
     895             :     //      Instance a band.
     896             :     // --------------------------------------------------------------------
     897           0 :     LevellerRasterBand *poBand = new LevellerRasterBand(poDS);
     898           0 :     poDS->SetBand(1, poBand);
     899             : 
     900           0 :     if (!poBand->Init())
     901             :     {
     902           0 :         delete poDS;
     903           0 :         return nullptr;
     904             :     }
     905             : 
     906           0 :     return poDS;
     907             : }
     908             : 
     909           0 : bool LevellerDataset::write_byte(size_t n)
     910             : {
     911           0 :     unsigned char uch = static_cast<unsigned char>(n);
     912           0 :     return 1 == VSIFWriteL(&uch, 1, 1, m_fp);
     913             : }
     914             : 
     915           0 : bool LevellerDataset::write(int n)
     916             : {
     917           0 :     CPL_LSBPTR32(&n);
     918           0 :     return 1 == VSIFWriteL(&n, sizeof(n), 1, m_fp);
     919             : }
     920             : 
     921           0 : bool LevellerDataset::write(size_t n)
     922             : {
     923           0 :     GUInt32 n32 = (GUInt32)n;
     924           0 :     CPL_LSBPTR32(&n32);
     925           0 :     return 1 == VSIFWriteL(&n32, sizeof(n32), 1, m_fp);
     926             : }
     927             : 
     928           0 : bool LevellerDataset::write(double d)
     929             : {
     930           0 :     CPL_LSBPTR64(&d);
     931           0 :     return 1 == VSIFWriteL(&d, sizeof(d), 1, m_fp);
     932             : }
     933             : 
     934           0 : bool LevellerDataset::write_tag_start(const char *pszTag, size_t n)
     935             : {
     936           0 :     if (this->write_byte(strlen(pszTag)))
     937             :     {
     938           0 :         return (1 == VSIFWriteL(pszTag, strlen(pszTag), 1, m_fp) &&
     939           0 :                 this->write(n));
     940             :     }
     941             : 
     942           0 :     return false;
     943             : }
     944             : 
     945           0 : bool LevellerDataset::write_tag(const char *pszTag, int n)
     946             : {
     947           0 :     return (this->write_tag_start(pszTag, sizeof(n)) && this->write(n));
     948             : }
     949             : 
     950           0 : bool LevellerDataset::write_tag(const char *pszTag, size_t n)
     951             : {
     952           0 :     return (this->write_tag_start(pszTag, sizeof(n)) && this->write(n));
     953             : }
     954             : 
     955           0 : bool LevellerDataset::write_tag(const char *pszTag, double d)
     956             : {
     957           0 :     return (this->write_tag_start(pszTag, sizeof(d)) && this->write(d));
     958             : }
     959             : 
     960           0 : bool LevellerDataset::write_tag(const char *pszTag, const char *psz)
     961             : {
     962           0 :     CPLAssert(strlen(pszTag) <= kMaxTagNameLen);
     963             : 
     964             :     char sz[kMaxTagNameLen + 1];
     965           0 :     snprintf(sz, sizeof(sz), "%sl", pszTag);
     966           0 :     const size_t len = strlen(psz);
     967             : 
     968           0 :     if (len > 0 && this->write_tag(sz, len))
     969             :     {
     970           0 :         snprintf(sz, sizeof(sz), "%sd", pszTag);
     971           0 :         this->write_tag_start(sz, len);
     972           0 :         return 1 == VSIFWriteL(psz, len, 1, m_fp);
     973             :     }
     974           0 :     return false;
     975             : }
     976             : 
     977          10 : bool LevellerDataset::locate_data(vsi_l_offset &offset, size_t &len,
     978             :                                   VSILFILE *fp, const char *pszTag)
     979             : {
     980             :     // Locate the file offset of the desired tag's data.
     981             :     // If it is not available, return false.
     982             :     // If the tag is found, leave the filemark at the
     983             :     // start of its data.
     984             : 
     985          10 :     if (0 != VSIFSeekL(fp, 5, SEEK_SET))
     986           0 :         return false;
     987             : 
     988          10 :     const int kMaxDescLen = 64;
     989             :     for (;;)
     990             :     {
     991             :         unsigned char c;
     992        1498 :         if (1 != VSIFReadL(&c, sizeof(c), 1, fp))
     993          10 :             return false;
     994             : 
     995        1498 :         const size_t descriptorLen = c;
     996        1498 :         if (descriptorLen == 0 || descriptorLen > (size_t)kMaxDescLen)
     997           0 :             return false;
     998             : 
     999             :         char descriptor[kMaxDescLen + 1];
    1000        1498 :         if (1 != VSIFReadL(descriptor, descriptorLen, 1, fp))
    1001           0 :             return false;
    1002             : 
    1003             :         GUInt32 datalen;
    1004        1498 :         if (1 != VSIFReadL(&datalen, sizeof(datalen), 1, fp))
    1005           0 :             return false;
    1006             : 
    1007        1498 :         CPL_LSBPTR32(&datalen);
    1008        1498 :         descriptor[descriptorLen] = 0;
    1009        1498 :         if (str_equal(descriptor, pszTag))
    1010             :         {
    1011          10 :             len = (size_t)datalen;
    1012          10 :             offset = VSIFTellL(fp);
    1013          10 :             return true;
    1014             :         }
    1015             :         else
    1016             :         {
    1017             :             // Seek to next tag.
    1018        1488 :             if (0 != VSIFSeekL(fp, (vsi_l_offset)datalen, SEEK_CUR))
    1019           0 :                 return false;
    1020             :         }
    1021        1488 :     }
    1022             : }
    1023             : 
    1024             : /************************************************************************/
    1025             : /*                                get()                                 */
    1026             : /************************************************************************/
    1027             : 
    1028           4 : bool LevellerDataset::get(int &n, VSILFILE *fp, const char *psz)
    1029             : {
    1030             :     vsi_l_offset offset;
    1031             :     size_t len;
    1032             : 
    1033           4 :     if (locate_data(offset, len, fp, psz))
    1034             :     {
    1035             :         GInt32 value;
    1036           4 :         if (1 == VSIFReadL(&value, sizeof(value), 1, fp))
    1037             :         {
    1038           4 :             CPL_LSBPTR32(&value);
    1039           4 :             n = static_cast<int>(value);
    1040           4 :             return true;
    1041             :         }
    1042             :     }
    1043           0 :     return false;
    1044             : }
    1045             : 
    1046             : /************************************************************************/
    1047             : /*                                get()                                 */
    1048             : /************************************************************************/
    1049             : 
    1050           2 : bool LevellerDataset::get(double &d, VSILFILE *fp, const char *pszTag)
    1051             : {
    1052             :     vsi_l_offset offset;
    1053             :     size_t len;
    1054             : 
    1055           2 :     if (locate_data(offset, len, fp, pszTag))
    1056             :     {
    1057           2 :         if (1 == VSIFReadL(&d, sizeof(d), 1, fp))
    1058             :         {
    1059           2 :             CPL_LSBPTR64(&d);
    1060           2 :             return true;
    1061             :         }
    1062             :     }
    1063           0 :     return false;
    1064             : }
    1065             : 
    1066             : /************************************************************************/
    1067             : /*                                get()                                 */
    1068             : /************************************************************************/
    1069           2 : bool LevellerDataset::get(char *pszValue, size_t maxchars, VSILFILE *fp,
    1070             :                           const char *pszTag)
    1071             : {
    1072             :     char szTag[65];
    1073             : 
    1074             :     // We can assume 8-bit encoding, so just go straight
    1075             :     // to the *_d tag.
    1076           2 :     snprintf(szTag, sizeof(szTag), "%sd", pszTag);
    1077             : 
    1078             :     vsi_l_offset offset;
    1079             :     size_t len;
    1080             : 
    1081           2 :     if (locate_data(offset, len, fp, szTag))
    1082             :     {
    1083           2 :         if (len > maxchars)
    1084           0 :             return false;
    1085             : 
    1086           2 :         if (1 == VSIFReadL(pszValue, len, 1, fp))
    1087             :         {
    1088           2 :             pszValue[len] = 0;  // terminate C-string
    1089           2 :             return true;
    1090             :         }
    1091             :     }
    1092             : 
    1093           0 :     return false;
    1094             : }
    1095             : 
    1096           0 : UNITLABEL LevellerDataset::meter_measure_to_code(double dM) const
    1097             : {
    1098             :     // Convert a meter conversion factor to its UOM OEM code.
    1099             :     // If the factor is close to the approximation margin, then
    1100             :     // require exact equality, otherwise be loose.
    1101             : 
    1102           0 :     const measurement_unit *pu = this->get_uom(dM);
    1103           0 :     return pu != nullptr ? pu->oemCode : UNITLABEL_UNKNOWN;
    1104             : }
    1105             : 
    1106           0 : UNITLABEL LevellerDataset::id_to_code(const char *pszUnits) const
    1107             : {
    1108             :     // Convert a readable UOM to its OEM code.
    1109             : 
    1110           0 :     const measurement_unit *pu = this->get_uom(pszUnits);
    1111           0 :     return pu != nullptr ? pu->oemCode : UNITLABEL_UNKNOWN;
    1112             : }
    1113             : 
    1114           0 : const char *LevellerDataset::code_to_id(UNITLABEL code) const
    1115             : {
    1116             :     // Convert a measurement unit's OEM ID to its readable ID.
    1117             : 
    1118           0 :     const measurement_unit *pu = this->get_uom(code);
    1119           0 :     return pu != nullptr ? pu->pszID : nullptr;
    1120             : }
    1121             : 
    1122           0 : const measurement_unit *LevellerDataset::get_uom(const char *pszUnits)
    1123             : {
    1124           0 :     for (size_t i = 0; i < CPL_ARRAYSIZE(kUnits); i++)
    1125             :     {
    1126           0 :         if (strcmp(pszUnits, kUnits[i].pszID) == 0)
    1127           0 :             return &kUnits[i];
    1128             :     }
    1129           0 :     CPLError(CE_Failure, CPLE_AppDefined, "Unknown measurement units: %s",
    1130             :              pszUnits);
    1131           0 :     return nullptr;
    1132             : }
    1133             : 
    1134           0 : const measurement_unit *LevellerDataset::get_uom(UNITLABEL code)
    1135             : {
    1136           0 :     for (size_t i = 0; i < CPL_ARRAYSIZE(kUnits); i++)
    1137             :     {
    1138           0 :         if (kUnits[i].oemCode == code)
    1139           0 :             return &kUnits[i];
    1140             :     }
    1141           0 :     CPLError(CE_Failure, CPLE_AppDefined, "Unknown measurement unit code: %08x",
    1142             :              code);
    1143           0 :     return nullptr;
    1144             : }
    1145             : 
    1146           0 : const measurement_unit *LevellerDataset::get_uom(double dM)
    1147             : {
    1148           0 :     for (size_t i = kFirstLinearMeasureIdx; i < CPL_ARRAYSIZE(kUnits); i++)
    1149             :     {
    1150           0 :         if (dM >= 1.0e-4)
    1151             :         {
    1152           0 :             if (approx_equal(dM, kUnits[i].dScale))
    1153           0 :                 return &kUnits[i];
    1154             :         }
    1155           0 :         else if (dM == kUnits[i].dScale)
    1156           0 :             return &kUnits[i];
    1157             :     }
    1158           0 :     CPLError(CE_Failure, CPLE_AppDefined,
    1159             :              "Unknown measurement conversion factor: %f", dM);
    1160           0 :     return nullptr;
    1161             : }
    1162             : 
    1163             : /************************************************************************/
    1164             : /*                          convert_measure()                           */
    1165             : /************************************************************************/
    1166             : 
    1167           2 : bool LevellerDataset::convert_measure(double d, double &dResult,
    1168             :                                       const char *pszSpace)
    1169             : {
    1170             :     // Convert a measure to meters.
    1171             : 
    1172          42 :     for (size_t i = kFirstLinearMeasureIdx; i < CPL_ARRAYSIZE(kUnits); i++)
    1173             :     {
    1174          42 :         if (str_equal(pszSpace, kUnits[i].pszID))
    1175             :         {
    1176           2 :             dResult = d * kUnits[i].dScale;
    1177           2 :             return true;
    1178             :         }
    1179             :     }
    1180           0 :     CPLError(CE_Failure, CPLE_FileIO, "Unknown linear measurement unit: '%s'",
    1181             :              pszSpace);
    1182           0 :     return false;
    1183             : }
    1184             : 
    1185           2 : bool LevellerDataset::make_local_coordsys(const char *pszName,
    1186             :                                           const char *pszUnits)
    1187             : {
    1188           2 :     m_oSRS.SetLocalCS(pszName);
    1189             :     double d;
    1190           4 :     return (convert_measure(1.0, d, pszUnits) &&
    1191           4 :             OGRERR_NONE == m_oSRS.SetLinearUnits(pszUnits, d));
    1192             : }
    1193             : 
    1194           0 : bool LevellerDataset::make_local_coordsys(const char *pszName, UNITLABEL code)
    1195             : {
    1196           0 :     const char *pszUnitID = code_to_id(code);
    1197           0 :     return pszUnitID != nullptr && make_local_coordsys(pszName, pszUnitID);
    1198             : }
    1199             : 
    1200             : /************************************************************************/
    1201             : /*                            load_from_file()                            */
    1202             : /************************************************************************/
    1203             : 
    1204           2 : bool LevellerDataset::load_from_file(VSILFILE *file, const char *pszFilename)
    1205             : {
    1206             :     // get hf dimensions
    1207           2 :     if (!get(nRasterXSize, file, "hf_w"))
    1208             :     {
    1209           0 :         CPLError(CE_Failure, CPLE_OpenFailed,
    1210             :                  "Cannot determine heightfield width.");
    1211           0 :         return false;
    1212             :     }
    1213             : 
    1214           2 :     if (!get(nRasterYSize, file, "hf_b"))
    1215             :     {
    1216           0 :         CPLError(CE_Failure, CPLE_OpenFailed,
    1217             :                  "Cannot determine heightfield breadth.");
    1218           0 :         return false;
    1219             :     }
    1220             : 
    1221           2 :     if (nRasterXSize < 2 || nRasterYSize < 2)
    1222             :     {
    1223           0 :         CPLError(CE_Failure, CPLE_OpenFailed,
    1224             :                  "Heightfield raster dimensions too small.");
    1225           0 :         return false;
    1226             :     }
    1227             : 
    1228             :     // Record start of pixel data
    1229             :     size_t datalen;
    1230           2 :     if (!locate_data(m_nDataOffset, datalen, file, "hf_data"))
    1231             :     {
    1232           0 :         CPLError(CE_Failure, CPLE_OpenFailed, "Cannot locate elevation data.");
    1233           0 :         return false;
    1234             :     }
    1235             : 
    1236             :     // Sanity check: do we have enough pixels?
    1237           2 :     if (static_cast<GUIntBig>(datalen) !=
    1238           2 :         static_cast<GUIntBig>(nRasterXSize) *
    1239           2 :             static_cast<GUIntBig>(nRasterYSize) * sizeof(float))
    1240             :     {
    1241           0 :         CPLError(CE_Failure, CPLE_OpenFailed,
    1242             :                  "File does not have enough data.");
    1243           0 :         return false;
    1244             :     }
    1245             : 
    1246             :     // Defaults for raster coordsys.
    1247           2 :     m_adfTransform[0] = 0.0;
    1248           2 :     m_adfTransform[1] = 1.0;
    1249           2 :     m_adfTransform[2] = 0.0;
    1250           2 :     m_adfTransform[3] = 0.0;
    1251           2 :     m_adfTransform[4] = 0.0;
    1252           2 :     m_adfTransform[5] = 1.0;
    1253             : 
    1254           2 :     m_dElevScale = 1.0;
    1255           2 :     m_dElevBase = 0.0;
    1256           2 :     strcpy(m_szElevUnits, "");
    1257             : 
    1258           2 :     if (m_version >= 7)
    1259             :     {
    1260             :         // Read coordsys info.
    1261           0 :         int csclass = LEV_COORDSYS_RASTER;
    1262           0 :         /* (void) */ get(csclass, file, "csclass");
    1263             : 
    1264           0 :         if (csclass != LEV_COORDSYS_RASTER)
    1265             :         {
    1266             :             // Get projection details and units.
    1267           0 :             if (csclass == LEV_COORDSYS_LOCAL)
    1268             :             {
    1269             :                 UNITLABEL unitcode;
    1270             :                 // char szLocalUnits[8];
    1271             :                 int unitcode_int;
    1272           0 :                 if (!get(unitcode_int, file, "coordsys_units"))
    1273           0 :                     unitcode_int = UNITLABEL_M;
    1274           0 :                 unitcode = static_cast<UNITLABEL>(unitcode_int);
    1275             : 
    1276           0 :                 if (!make_local_coordsys("Leveller", unitcode))
    1277             :                 {
    1278           0 :                     CPLError(CE_Failure, CPLE_OpenFailed,
    1279             :                              "Cannot define local coordinate system.");
    1280           0 :                     return false;
    1281             :                 }
    1282             :             }
    1283           0 :             else if (csclass == LEV_COORDSYS_GEO)
    1284             :             {
    1285             :                 char szWKT[1024];
    1286           0 :                 if (!get(szWKT, 1023, file, "coordsys_wkt"))
    1287           0 :                     return false;
    1288             : 
    1289           0 :                 m_oSRS.importFromWkt(szWKT);
    1290             :             }
    1291             :             else
    1292             :             {
    1293           0 :                 CPLError(CE_Failure, CPLE_OpenFailed,
    1294             :                          "Unknown coordinate system type in %s.", pszFilename);
    1295           0 :                 return false;
    1296             :             }
    1297             : 
    1298             :             // Get ground extents.
    1299           0 :             digital_axis axis_ns, axis_ew;
    1300             : 
    1301           0 :             if (axis_ns.get(*this, file, 0) && axis_ew.get(*this, file, 1))
    1302             :             {
    1303           0 :                 m_adfTransform[0] = axis_ew.origin(nRasterXSize);
    1304           0 :                 m_adfTransform[1] = axis_ew.scaling(nRasterXSize);
    1305           0 :                 m_adfTransform[2] = 0.0;
    1306             : 
    1307           0 :                 m_adfTransform[3] = axis_ns.origin(nRasterYSize);
    1308           0 :                 m_adfTransform[4] = 0.0;
    1309           0 :                 m_adfTransform[5] = axis_ns.scaling(nRasterYSize);
    1310             :             }
    1311             :         }
    1312             : 
    1313             :         // Get vertical (elev) coordsys.
    1314           0 :         int bHasVertCS = FALSE;
    1315           0 :         if (get(bHasVertCS, file, "coordsys_haselevm") && bHasVertCS)
    1316             :         {
    1317           0 :             get(m_dElevScale, file, "coordsys_em_scale");
    1318           0 :             get(m_dElevBase, file, "coordsys_em_base");
    1319             :             UNITLABEL unitcode;
    1320             :             int unitcode_int;
    1321           0 :             if (get(unitcode_int, file, "coordsys_em_units"))
    1322             :             {
    1323           0 :                 unitcode = static_cast<UNITLABEL>(unitcode_int);
    1324           0 :                 const char *pszUnitID = code_to_id(unitcode);
    1325           0 :                 if (pszUnitID != nullptr)
    1326             :                 {
    1327           0 :                     strncpy(m_szElevUnits, pszUnitID, sizeof(m_szElevUnits));
    1328           0 :                     m_szElevUnits[sizeof(m_szElevUnits) - 1] = '\0';
    1329             :                 }
    1330             :                 else
    1331             :                 {
    1332           0 :                     CPLError(CE_Failure, CPLE_OpenFailed,
    1333             :                              "Unknown OEM elevation unit of measure (%d)",
    1334             :                              unitcode);
    1335           0 :                     return false;
    1336             :                 }
    1337             :             }
    1338             :             // datum and localcs are currently unused.
    1339             :         }
    1340             :     }
    1341             :     else
    1342             :     {
    1343             :         // Legacy files use world units.
    1344             :         char szWorldUnits[32];
    1345           2 :         strcpy(szWorldUnits, "m");
    1346             : 
    1347           2 :         double dWorldscale = 1.0;
    1348             : 
    1349           2 :         if (get(dWorldscale, file, "hf_worldspacing"))
    1350             :         {
    1351             :             // m_bHasWorldscale = true;
    1352           2 :             if (get(szWorldUnits, sizeof(szWorldUnits) - 1, file,
    1353             :                     "hf_worldspacinglabel"))
    1354             :             {
    1355             :                 // Drop long name, if present.
    1356           2 :                 char *p = strchr(szWorldUnits, ' ');
    1357           2 :                 if (p != nullptr)
    1358           2 :                     *p = 0;
    1359             :             }
    1360             : 
    1361             : #if 0
    1362             :           // If the units are something besides m/ft/sft,
    1363             :           // then convert them to meters.
    1364             : 
    1365             :           if(!str_equal("m", szWorldUnits)
    1366             :              && !str_equal("ft", szWorldUnits)
    1367             :              && !str_equal("sft", szWorldUnits))
    1368             :           {
    1369             :               dWorldscale = this->convert_measure(dWorldscale, szWorldUnits);
    1370             :               strcpy(szWorldUnits, "m");
    1371             :           }
    1372             : #endif
    1373             : 
    1374             :             // Our extents are such that the origin is at the
    1375             :             // center of the heightfield.
    1376           2 :             m_adfTransform[0] = -0.5 * dWorldscale * (nRasterXSize - 1);
    1377           2 :             m_adfTransform[3] = -0.5 * dWorldscale * (nRasterYSize - 1);
    1378           2 :             m_adfTransform[1] = dWorldscale;
    1379           2 :             m_adfTransform[5] = dWorldscale;
    1380             :         }
    1381           2 :         m_dElevScale = dWorldscale;  // this was 1.0 before because
    1382             :         // we were converting to real elevs ourselves, but
    1383             :         // some callers may want both the raw pixels and the
    1384             :         // transform to get real elevs.
    1385             : 
    1386           2 :         if (!make_local_coordsys("Leveller world space", szWorldUnits))
    1387             :         {
    1388           0 :             CPLError(CE_Failure, CPLE_OpenFailed,
    1389             :                      "Cannot define local coordinate system.");
    1390           0 :             return false;
    1391             :         }
    1392             :     }
    1393             : 
    1394           2 :     return true;
    1395             : }
    1396             : 
    1397             : /************************************************************************/
    1398             : /*                          GetSpatialRef()                             */
    1399             : /************************************************************************/
    1400             : 
    1401           0 : const OGRSpatialReference *LevellerDataset::GetSpatialRef() const
    1402             : {
    1403           0 :     return m_oSRS.IsEmpty() ? nullptr : &m_oSRS;
    1404             : }
    1405             : 
    1406             : /************************************************************************/
    1407             : /*                          GetGeoTransform()                           */
    1408             : /************************************************************************/
    1409             : 
    1410           0 : CPLErr LevellerDataset::GetGeoTransform(double *padfTransform)
    1411             : 
    1412             : {
    1413           0 :     memcpy(padfTransform, m_adfTransform, sizeof(m_adfTransform));
    1414           0 :     return CE_None;
    1415             : }
    1416             : 
    1417             : /************************************************************************/
    1418             : /*                              Identify()                              */
    1419             : /************************************************************************/
    1420             : 
    1421       55946 : int LevellerDataset::Identify(GDALOpenInfo *poOpenInfo)
    1422             : {
    1423       55946 :     if (poOpenInfo->nHeaderBytes < 4)
    1424       49666 :         return FALSE;
    1425             : 
    1426        6280 :     return STARTS_WITH_CI(
    1427             :         reinterpret_cast<const char *>(poOpenInfo->pabyHeader), "trrn");
    1428             : }
    1429             : 
    1430             : /************************************************************************/
    1431             : /*                                Open()                                */
    1432             : /************************************************************************/
    1433             : 
    1434           2 : GDALDataset *LevellerDataset::Open(GDALOpenInfo *poOpenInfo)
    1435             : {
    1436             :     // The file should have at least 5 header bytes
    1437             :     // and hf_w, hf_b, and hf_data tags.
    1438             : 
    1439           2 :     if (poOpenInfo->nHeaderBytes < 5 + 13 + 13 + 16 ||
    1440           2 :         poOpenInfo->fpL == nullptr)
    1441           0 :         return nullptr;
    1442             : 
    1443           2 :     if (!LevellerDataset::Identify(poOpenInfo))
    1444           0 :         return nullptr;
    1445             : 
    1446           2 :     const int version = poOpenInfo->pabyHeader[4];
    1447           2 :     if (version < 4 || version > 9)
    1448           0 :         return nullptr;
    1449             : 
    1450             :     /* -------------------------------------------------------------------- */
    1451             :     /*      Create a corresponding GDALDataset.                             */
    1452             :     /* -------------------------------------------------------------------- */
    1453             : 
    1454           2 :     LevellerDataset *poDS = new LevellerDataset();
    1455             : 
    1456           2 :     poDS->m_version = version;
    1457             : 
    1458           2 :     poDS->m_fp = poOpenInfo->fpL;
    1459           2 :     poOpenInfo->fpL = nullptr;
    1460           2 :     poDS->eAccess = poOpenInfo->eAccess;
    1461             : 
    1462             :     /* -------------------------------------------------------------------- */
    1463             :     /*      Read the file.                                                  */
    1464             :     /* -------------------------------------------------------------------- */
    1465           2 :     if (!poDS->load_from_file(poDS->m_fp, poOpenInfo->pszFilename))
    1466             :     {
    1467           0 :         delete poDS;
    1468           0 :         return nullptr;
    1469             :     }
    1470             : 
    1471             :     /* -------------------------------------------------------------------- */
    1472             :     /*      Create band information objects.                                */
    1473             :     /* -------------------------------------------------------------------- */
    1474           2 :     LevellerRasterBand *poBand = new LevellerRasterBand(poDS);
    1475           2 :     poDS->SetBand(1, poBand);
    1476           2 :     if (!poBand->Init())
    1477             :     {
    1478           0 :         delete poDS;
    1479           0 :         return nullptr;
    1480             :     }
    1481             : 
    1482           2 :     poDS->SetMetadataItem(GDALMD_AREA_OR_POINT, GDALMD_AOP_POINT);
    1483             : 
    1484             :     /* -------------------------------------------------------------------- */
    1485             :     /*      Initialize any PAM information.                                 */
    1486             :     /* -------------------------------------------------------------------- */
    1487           2 :     poDS->SetDescription(poOpenInfo->pszFilename);
    1488           2 :     poDS->TryLoadXML();
    1489             : 
    1490             :     /* -------------------------------------------------------------------- */
    1491             :     /*      Check for external overviews.                                   */
    1492             :     /* -------------------------------------------------------------------- */
    1493           4 :     poDS->oOvManager.Initialize(poDS, poOpenInfo->pszFilename,
    1494           2 :                                 poOpenInfo->GetSiblingFiles());
    1495             : 
    1496           2 :     return poDS;
    1497             : }
    1498             : 
    1499             : /************************************************************************/
    1500             : /*                        GDALRegister_Leveller()                       */
    1501             : /************************************************************************/
    1502             : 
    1503        1682 : void GDALRegister_Leveller()
    1504             : 
    1505             : {
    1506        1682 :     if (GDALGetDriverByName("Leveller") != nullptr)
    1507         301 :         return;
    1508             : 
    1509        1381 :     GDALDriver *poDriver = new GDALDriver();
    1510             : 
    1511        1381 :     poDriver->SetDescription("Leveller");
    1512        1381 :     poDriver->SetMetadataItem(GDAL_DCAP_RASTER, "YES");
    1513        1381 :     poDriver->SetMetadataItem(GDAL_DMD_EXTENSION, "ter");
    1514        1381 :     poDriver->SetMetadataItem(GDAL_DMD_LONGNAME, "Leveller heightfield");
    1515        1381 :     poDriver->SetMetadataItem(GDAL_DMD_HELPTOPIC,
    1516        1381 :                               "drivers/raster/leveller.html");
    1517        1381 :     poDriver->SetMetadataItem(GDAL_DCAP_VIRTUALIO, "YES");
    1518             : 
    1519        1381 :     poDriver->pfnIdentify = LevellerDataset::Identify;
    1520        1381 :     poDriver->pfnOpen = LevellerDataset::Open;
    1521        1381 :     poDriver->pfnCreate = LevellerDataset::Create;
    1522             : 
    1523        1381 :     GetGDALDriverManager()->RegisterDriver(poDriver);
    1524             : }

Generated by: LCOV version 1.14