LCOV - code coverage report
Current view: top level - ogr/ogrsf_frmts/openfilegdb - gdalopenfilegdbrasterband.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 669 918 72.9 %
Date: 2025-01-18 12:42:00 Functions: 13 19 68.4 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  OpenGIS Simple Features Reference Implementation
       4             :  * Purpose:  Implements Open FileGDB raster driver.
       5             :  * Author:   Even Rouault, <even dot rouault at spatialys.com>
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2023, Even Rouault <even dot rouault at spatialys.com>
       9             :  *
      10             :  * SPDX-License-Identifier: MIT
      11             :  ****************************************************************************/
      12             : 
      13             : #include "cpl_port.h"
      14             : #include "cpl_conv.h"
      15             : #include "cpl_minixml.h"
      16             : 
      17             : #include "ogr_openfilegdb.h"
      18             : 
      19             : #include "gdal_rat.h"
      20             : #include "filegdbtable_priv.h"
      21             : 
      22             : #include <algorithm>
      23             : #include <cassert>
      24             : #include <cmath>
      25             : #include <limits>
      26             : #include <new>
      27             : #include <utility>
      28             : 
      29             : using namespace OpenFileGDB;
      30             : 
      31             : /***********************************************************************/
      32             : /*                         OpenRaster()                                */
      33             : /***********************************************************************/
      34             : 
      35          29 : bool OGROpenFileGDBDataSource::OpenRaster(const GDALOpenInfo *poOpenInfo,
      36             :                                           const std::string &osLayerName,
      37             :                                           const std::string &osDefinition,
      38             :                                           const std::string &osDocumentation)
      39             : {
      40          29 :     m_osRasterLayerName = osLayerName;
      41             : 
      42             :     const std::string osBndTableName(
      43          87 :         std::string("fras_bnd_").append(osLayerName).c_str());
      44          29 :     const auto oIter = m_osMapNameToIdx.find(osBndTableName);
      45          29 :     if (oIter == m_osMapNameToIdx.end())
      46             :     {
      47           0 :         CPLError(CE_Failure, CPLE_AppDefined, "Cannot find table %s",
      48             :                  osBndTableName.c_str());
      49           0 :         return false;
      50             :     }
      51          29 :     const int nBndIdx = oIter->second;
      52             : 
      53          58 :     FileGDBTable oTable;
      54             : 
      55             :     const std::string osBndFilename(CPLFormFilenameSafe(
      56          58 :         m_osDirName, CPLSPrintf("a%08x.gdbtable", nBndIdx), nullptr));
      57          29 :     if (!oTable.Open(osBndFilename.c_str(), false))
      58             :     {
      59           0 :         CPLError(CE_Failure, CPLE_AppDefined, "Cannot open table %s",
      60             :                  osBndTableName.c_str());
      61           0 :         return false;
      62             :     }
      63             : 
      64          29 :     const int i_rasterband_id = oTable.GetFieldIdx("rasterband_id");
      65          29 :     const int i_sequence_nbr = oTable.GetFieldIdx("sequence_nbr");
      66          29 :     const int i_raster_id = oTable.GetFieldIdx("raster_id");
      67          29 :     const int i_band_width = oTable.GetFieldIdx("band_width");
      68          29 :     const int i_band_height = oTable.GetFieldIdx("band_height");
      69          29 :     const int i_band_types = oTable.GetFieldIdx("band_types");
      70          29 :     const int i_block_width = oTable.GetFieldIdx("block_width");
      71          29 :     const int i_block_height = oTable.GetFieldIdx("block_height");
      72          29 :     const int i_block_origin_x = oTable.GetFieldIdx("block_origin_x");
      73          29 :     const int i_block_origin_y = oTable.GetFieldIdx("block_origin_y");
      74          29 :     const int i_eminx = oTable.GetFieldIdx("eminx");
      75          29 :     const int i_eminy = oTable.GetFieldIdx("eminy");
      76          29 :     const int i_emaxx = oTable.GetFieldIdx("emaxx");
      77          29 :     const int i_emaxy = oTable.GetFieldIdx("emaxy");
      78          29 :     const int i_srid = oTable.GetFieldIdx("srid");
      79          29 :     if (i_rasterband_id < 0 || i_sequence_nbr < 0 || i_raster_id < 0 ||
      80          29 :         i_band_width < 0 || i_band_height < 0 || i_band_types < 0 ||
      81          29 :         i_block_width < 0 || i_block_height < 0 || i_block_origin_x < 0 ||
      82          29 :         i_block_origin_y < 0 || i_eminx < 0 || i_eminy < 0 || i_emaxx < 0 ||
      83          58 :         i_emaxy < 0 || i_srid < 0 ||
      84          58 :         oTable.GetField(i_rasterband_id)->GetType() != FGFT_OBJECTID ||
      85          58 :         oTable.GetField(i_sequence_nbr)->GetType() != FGFT_INT32 ||
      86          58 :         oTable.GetField(i_raster_id)->GetType() != FGFT_INT32 ||
      87          58 :         oTable.GetField(i_band_width)->GetType() != FGFT_INT32 ||
      88          58 :         oTable.GetField(i_band_height)->GetType() != FGFT_INT32 ||
      89          58 :         oTable.GetField(i_band_types)->GetType() != FGFT_INT32 ||
      90          58 :         oTable.GetField(i_block_width)->GetType() != FGFT_INT32 ||
      91          58 :         oTable.GetField(i_block_height)->GetType() != FGFT_INT32 ||
      92          58 :         oTable.GetField(i_block_origin_x)->GetType() != FGFT_FLOAT64 ||
      93          58 :         oTable.GetField(i_block_origin_y)->GetType() != FGFT_FLOAT64 ||
      94          58 :         oTable.GetField(i_eminx)->GetType() != FGFT_FLOAT64 ||
      95          58 :         oTable.GetField(i_eminy)->GetType() != FGFT_FLOAT64 ||
      96          58 :         oTable.GetField(i_emaxx)->GetType() != FGFT_FLOAT64 ||
      97          87 :         oTable.GetField(i_emaxy)->GetType() != FGFT_FLOAT64 ||
      98          29 :         oTable.GetField(i_srid)->GetType() != FGFT_INT32)
      99             :     {
     100           0 :         CPLError(CE_Failure, CPLE_AppDefined, "Wrong structure for %s table",
     101             :                  osBndTableName.c_str());
     102           0 :         return false;
     103             :     }
     104             : 
     105          29 :     int64_t iRow = 0;
     106         111 :     while (iRow < oTable.GetTotalRecordCount() &&
     107          41 :            (iRow = oTable.GetAndSelectNextNonEmptyRow(iRow)) >= 0)
     108             :     {
     109          41 :         if (iRow >= INT32_MAX)
     110             :         {
     111           0 :             return false;
     112             :         }
     113          41 :         auto psField = oTable.GetFieldValue(i_raster_id);
     114          41 :         if (!psField)
     115             :         {
     116           0 :             CPLError(CE_Failure, CPLE_AppDefined,
     117             :                      "Cannot read field %s in %s table", "raster_id",
     118             :                      osBndTableName.c_str());
     119           0 :             return false;
     120             :         }
     121          41 :         if (psField->Integer != 1)
     122             :         {
     123           0 :             CPLError(CE_Warning, CPLE_AppDefined,
     124             :                      "Raster with raster_id = %d (!= 1) ignored",
     125           0 :                      psField->Integer);
     126           0 :             continue;
     127             :         }
     128             : 
     129          41 :         const int nGDBRasterBandId = static_cast<int>(iRow) + 1;
     130             : 
     131          41 :         psField = oTable.GetFieldValue(i_sequence_nbr);
     132          41 :         if (!psField)
     133             :         {
     134           0 :             CPLError(CE_Failure, CPLE_AppDefined,
     135             :                      "Cannot read field %s in %s table", "sequence_nbr",
     136             :                      osBndTableName.c_str());
     137           0 :             return false;
     138             :         }
     139          41 :         const int nSequenceNr = psField->Integer;
     140             : 
     141          41 :         m_oMapGDALBandToGDBBandId[nSequenceNr] = nGDBRasterBandId;
     142             : 
     143          41 :         ++iRow;
     144             :     }
     145             : 
     146          29 :     if (m_oMapGDALBandToGDBBandId.empty())
     147             :     {
     148           0 :         CPLError(CE_Failure, CPLE_AppDefined, "Cannot read record in %s table",
     149             :                  osBndTableName.c_str());
     150           0 :         return false;
     151             :     }
     152             : 
     153          29 :     auto psField = oTable.GetFieldValue(i_band_width);
     154          29 :     if (!psField)
     155             :     {
     156           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     157             :                  "Cannot read field %s in %s table", "band_width",
     158             :                  osBndTableName.c_str());
     159           0 :         return false;
     160             :     }
     161          29 :     int nWidth = psField->Integer;
     162             : 
     163          29 :     psField = oTable.GetFieldValue(i_band_height);
     164          29 :     if (!psField)
     165             :     {
     166           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     167             :                  "Cannot read field %s in %s table", "band_height",
     168             :                  osBndTableName.c_str());
     169           0 :         return false;
     170             :     }
     171          29 :     int nHeight = psField->Integer;
     172             : 
     173          29 :     const int l_nBands = static_cast<int>(m_oMapGDALBandToGDBBandId.size());
     174          58 :     if (!GDALCheckDatasetDimensions(nWidth, nHeight) ||
     175          29 :         !GDALCheckBandCount(l_nBands, /*bIsZeroAllowed=*/false))
     176             :     {
     177           0 :         return false;
     178             :     }
     179             : 
     180          29 :     psField = oTable.GetFieldValue(i_block_width);
     181          29 :     if (!psField)
     182             :     {
     183           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     184             :                  "Cannot read field %s in %s table", "block_width",
     185             :                  osBndTableName.c_str());
     186           0 :         return false;
     187             :     }
     188          29 :     const int nBlockWidth = psField->Integer;
     189             : 
     190             :     // 32768 somewhat arbitrary
     191          29 :     if (nBlockWidth <= 0 || nBlockWidth > 32768)
     192             :     {
     193           0 :         CPLError(CE_Failure, CPLE_AppDefined, "Invalid %s in %s table",
     194             :                  "block_width", osBndTableName.c_str());
     195           0 :         return false;
     196             :     }
     197             : 
     198          29 :     psField = oTable.GetFieldValue(i_block_height);
     199          29 :     if (!psField)
     200             :     {
     201           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     202             :                  "Cannot read field %s in %s table", "block_height",
     203             :                  osBndTableName.c_str());
     204           0 :         return false;
     205             :     }
     206          29 :     const int nBlockHeight = psField->Integer;
     207             : 
     208             :     // 32768 somewhat arbitrary
     209          29 :     if (nBlockHeight <= 0 || nBlockHeight > 32768)
     210             :     {
     211           0 :         CPLError(CE_Failure, CPLE_AppDefined, "Invalid %s in %s table",
     212             :                  "block_height", osBndTableName.c_str());
     213           0 :         return false;
     214             :     }
     215             : 
     216          29 :     psField = oTable.GetFieldValue(i_band_types);
     217          29 :     if (!psField)
     218             :     {
     219           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     220             :                  "Cannot read field %s in %s table", "band_types",
     221             :                  osBndTableName.c_str());
     222           0 :         return false;
     223             :     }
     224          29 :     const int nBandTypes = psField->Integer;
     225             : 
     226          29 :     psField = oTable.GetFieldValue(i_eminx);
     227          29 :     if (!psField)
     228             :     {
     229           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     230             :                  "Cannot read field %s in %s table", "eminx",
     231             :                  osBndTableName.c_str());
     232           0 :         return false;
     233             :     }
     234          29 :     const double dfMinX = psField->Real;
     235             : 
     236          29 :     psField = oTable.GetFieldValue(i_eminy);
     237          29 :     if (!psField)
     238             :     {
     239           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     240             :                  "Cannot read field %s in %s table", "eminy",
     241             :                  osBndTableName.c_str());
     242           0 :         return false;
     243             :     }
     244          29 :     const double dfMinY = psField->Real;
     245             : 
     246          29 :     psField = oTable.GetFieldValue(i_emaxx);
     247          29 :     if (!psField)
     248             :     {
     249           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     250             :                  "Cannot read field %s in %s table", "emaxx",
     251             :                  osBndTableName.c_str());
     252           0 :         return false;
     253             :     }
     254          29 :     const double dfMaxX = psField->Real;
     255             : 
     256          29 :     psField = oTable.GetFieldValue(i_emaxy);
     257          29 :     if (!psField)
     258             :     {
     259           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     260             :                  "Cannot read field %s in %s table", "emaxy",
     261             :                  osBndTableName.c_str());
     262           0 :         return false;
     263             :     }
     264          29 :     const double dfMaxY = psField->Real;
     265             : 
     266          29 :     psField = oTable.GetFieldValue(i_block_origin_x);
     267          29 :     if (!psField)
     268             :     {
     269           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     270             :                  "Cannot read field %s in %s table", "block_origin_x",
     271             :                  osBndTableName.c_str());
     272           0 :         return false;
     273             :     }
     274          29 :     const double dfBlockOriginX = psField->Real;
     275             : 
     276          29 :     psField = oTable.GetFieldValue(i_block_origin_y);
     277          29 :     if (!psField)
     278             :     {
     279           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     280             :                  "Cannot read field %s in %s table", "block_origin_y",
     281             :                  osBndTableName.c_str());
     282           0 :         return false;
     283             :     }
     284          29 :     const double dfBlockOriginY = psField->Real;
     285             : 
     286             :     // Figure out data type
     287          29 :     GDALDataType eDT = GDT_Byte;
     288          29 :     const int nBitWidth = (nBandTypes >> 19) & ((1 << 7) - 1);
     289          29 :     const int nBitType = (nBandTypes >> 16) & ((1 << 2) - 1);
     290          29 :     constexpr int IS_UNSIGNED = 0;
     291          29 :     constexpr int IS_SIGNED = 1;
     292          29 :     constexpr int IS_FLOATING_POINT = 2;
     293          29 :     if ((nBitWidth >= 1 && nBitWidth < 8) && nBitType == IS_UNSIGNED)
     294             :     {
     295           2 :         eDT = GDT_Byte;
     296             :     }
     297          27 :     else if (nBitWidth == 8 && nBitType <= IS_SIGNED)
     298             :     {
     299          13 :         eDT = nBitType == IS_SIGNED ? GDT_Int8 : GDT_Byte;
     300             :     }
     301          14 :     else if (nBitWidth == 16 && nBitType <= IS_SIGNED)
     302             :     {
     303           4 :         eDT = nBitType == IS_SIGNED ? GDT_Int16 : GDT_UInt16;
     304             :     }
     305          10 :     else if (nBitWidth == 32 && nBitType <= IS_FLOATING_POINT)
     306             :     {
     307          13 :         eDT = nBitType == IS_FLOATING_POINT ? GDT_Float32
     308           4 :               : nBitType == IS_SIGNED       ? GDT_Int32
     309             :                                             : GDT_UInt32;
     310             :     }
     311           1 :     else if (nBitWidth == 64 && nBitType == 0)
     312             :     {
     313           1 :         eDT = GDT_Float64;
     314             :     }
     315             :     else
     316             :     {
     317           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     318             :                  "Unhandled nBitWidth=%d, nBitType=%d in %s table", nBitWidth,
     319             :                  nBitType, osBndTableName.c_str());
     320           0 :         return false;
     321             :     }
     322             : 
     323             :     // To avoid potential integer overflows in IReadBlock()
     324          58 :     if (nBlockWidth * nBlockHeight >
     325          29 :         std::numeric_limits<int>::max() / nBitWidth)
     326             :     {
     327           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     328             :                  "Too large block %dx%d in %s table", nBlockWidth, nBlockHeight,
     329             :                  osBndTableName.c_str());
     330           0 :         return false;
     331             :     }
     332             : 
     333             :     // Figure out compression
     334          29 :     const int nCompression = (nBandTypes >> 8) & 0xff;
     335          29 :     switch (nCompression)
     336             :     {
     337           6 :         case 0:
     338           6 :             m_eRasterCompression = Compression::NONE;
     339           6 :             break;
     340          19 :         case 4:
     341          19 :             m_eRasterCompression = Compression::LZ77;
     342          19 :             SetMetadataItem("COMPRESSION", "DEFLATE", "IMAGE_STRUCTURE");
     343          19 :             break;
     344           2 :         case 8:
     345           2 :             m_eRasterCompression = Compression::JPEG;
     346           2 :             SetMetadataItem("COMPRESSION", "JPEG", "IMAGE_STRUCTURE");
     347           2 :             break;
     348           2 :         case 12:
     349           2 :             m_eRasterCompression = Compression::JPEG2000;
     350           2 :             SetMetadataItem("COMPRESSION", "JPEG2000", "IMAGE_STRUCTURE");
     351           2 :             break;
     352           0 :         default:
     353             :         {
     354           0 :             CPLError(CE_Failure, CPLE_AppDefined,
     355             :                      "Unhandled compression %d in %s table", nCompression,
     356             :                      osBndTableName.c_str());
     357           0 :             return false;
     358             :         }
     359             :     }
     360             : 
     361             :     // Figure out geotransform
     362             : 
     363          29 :     if (!(dfMaxX > dfMinX && dfMaxY > dfMinY))
     364             :     {
     365           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     366             :                  "!(dfMaxX > dfMinX && dfMaxY > dfMinY)");
     367           0 :         return false;
     368             :     }
     369          29 :     else if (nWidth == 1 || nHeight == 1)
     370             :     {
     371           0 :         CPLError(CE_Warning, CPLE_AppDefined,
     372             :                  "nWidth == 1 || nHeight == 1: cannot determine geotransform");
     373             :     }
     374             :     else
     375             :     {
     376             :         // FileGDB uses a center-of-pixel convention for georeferencing
     377             :         // Transform to GDAL's corner-of-pixel convention.
     378          29 :         const double dfResX = (dfMaxX - dfMinX) / (nWidth - 1);
     379          29 :         const double dfResY = (dfMaxY - dfMinY) / (nHeight - 1);
     380          29 :         m_bHasGeoTransform = true;
     381          29 :         const double dfBlockGeorefWidth = dfResX * nBlockWidth;
     382          29 :         if (dfMinX != dfBlockOriginX)
     383             :         {
     384             :             // Take into account MinX by making sure the raster origin is
     385             :             // close to it, while being shifted from an integer number of blocks
     386             :             // from BlockOriginX
     387             :             const double dfTmp =
     388           1 :                 std::floor((dfMinX - dfBlockOriginX) / dfBlockGeorefWidth);
     389           1 :             if (std::fabs(dfTmp) > std::numeric_limits<int>::max())
     390             :             {
     391           0 :                 CPLError(CE_Warning, CPLE_AppDefined,
     392             :                          "Inconsistent eminx=%g and block_origin_x=%g", dfMinX,
     393             :                          dfBlockOriginX);
     394           0 :                 return false;
     395             :             }
     396           1 :             m_nShiftBlockX = static_cast<int>(dfTmp);
     397           1 :             CPLDebug("OpenFileGDB", "m_nShiftBlockX = %d", m_nShiftBlockX);
     398           1 :             const double dfMinXAdjusted =
     399           1 :                 dfBlockOriginX + m_nShiftBlockX * dfBlockGeorefWidth;
     400           1 :             nWidth = 1 + static_cast<int>(
     401           1 :                              std::round((dfMaxX - dfMinXAdjusted) / dfResX));
     402             :         }
     403          58 :         m_adfGeoTransform[0] =
     404          29 :             (dfBlockOriginX + m_nShiftBlockX * dfBlockGeorefWidth) - dfResX / 2;
     405          29 :         m_adfGeoTransform[1] = dfResX;
     406          29 :         m_adfGeoTransform[2] = 0.0;
     407          29 :         const double dfBlockGeorefHeight = dfResY * nBlockHeight;
     408          29 :         if (dfMaxY != dfBlockOriginY)
     409             :         {
     410             :             // Take into account MaxY by making sure the raster origin is
     411             :             // close to it, while being shifted from an integer number of blocks
     412             :             // from BlockOriginY
     413             :             const double dfTmp =
     414           3 :                 std::floor((dfBlockOriginY - dfMaxY) / dfBlockGeorefHeight);
     415           3 :             if (std::fabs(dfTmp) > std::numeric_limits<int>::max())
     416             :             {
     417           0 :                 CPLError(CE_Warning, CPLE_AppDefined,
     418             :                          "Inconsistent emaxy=%g and block_origin_y=%g", dfMaxY,
     419             :                          dfBlockOriginY);
     420           0 :                 return false;
     421             :             }
     422           3 :             m_nShiftBlockY = static_cast<int>(dfTmp);
     423           3 :             CPLDebug("OpenFileGDB", "m_nShiftBlockY = %d", m_nShiftBlockY);
     424           3 :             const double dfMaxYAdjusted =
     425           3 :                 dfBlockOriginY - m_nShiftBlockY * dfBlockGeorefHeight;
     426           3 :             nHeight = 1 + static_cast<int>(
     427           3 :                               std::round((dfMaxYAdjusted - dfMinY) / dfResY));
     428             :         }
     429          58 :         m_adfGeoTransform[3] =
     430          29 :             (dfBlockOriginY - m_nShiftBlockY * dfBlockGeorefHeight) +
     431          29 :             dfResY / 2;
     432          29 :         m_adfGeoTransform[4] = 0.0;
     433          29 :         m_adfGeoTransform[5] = -dfResY;
     434             :     }
     435             : 
     436             :     // Two cases:
     437             :     // - osDefinition is empty (that is FileGDB v9): find the SRS by looking
     438             :     //   at the SRS attached to the RASTER field definition of the .gdbtable
     439             :     //   file of the main table of the raster (that is the one without fras_XXX
     440             :     //   prefixes)
     441             :     // - or osDefinition is not empty (that is FileGDB v10): get SRID from the
     442             :     //   "srid" field of the _fras_bnd table, and use that has the key to
     443             :     //   lookup the corresponding WKT from the GDBSpatialRefs table.
     444             :     //   In some cases srid might be 0 (invalid), then we try to get it from
     445             :     //   Definition column of the GDB_Items table, stored in osDefinition
     446          29 :     psField = oTable.GetFieldValue(i_srid);
     447          29 :     if (osDefinition.empty())
     448             :     {
     449             :         // osDefinition empty for FileGDB v9
     450           2 :         const auto oIter2 = m_osMapNameToIdx.find(osLayerName);
     451           2 :         if (oIter2 != m_osMapNameToIdx.end())
     452             :         {
     453           2 :             const int nTableIdx = oIter2->second;
     454             : 
     455           4 :             FileGDBTable oTableMain;
     456             : 
     457             :             const std::string osTableMain(CPLFormFilenameSafe(
     458           4 :                 m_osDirName, CPLSPrintf("a%08x.gdbtable", nTableIdx), nullptr));
     459           2 :             if (oTableMain.Open(osTableMain.c_str(), false))
     460             :             {
     461           2 :                 const int iRasterFieldIdx = oTableMain.GetFieldIdx("RASTER");
     462           2 :                 if (iRasterFieldIdx >= 0)
     463             :                 {
     464           2 :                     const auto poField = oTableMain.GetField(iRasterFieldIdx);
     465           2 :                     if (poField->GetType() == FGFT_RASTER)
     466             :                     {
     467           2 :                         const auto poFieldRaster =
     468             :                             static_cast<FileGDBRasterField *>(poField);
     469           2 :                         const auto &osWKT = poFieldRaster->GetWKT();
     470           2 :                         if (!osWKT.empty() && osWKT[0] != '{')
     471             :                         {
     472           2 :                             auto poSRS = BuildSRS(osWKT.c_str());
     473           2 :                             if (poSRS)
     474             :                             {
     475           2 :                                 m_oRasterSRS = *poSRS;
     476           2 :                                 poSRS->Release();
     477             :                             }
     478             :                         }
     479             :                     }
     480             :                 }
     481             :             }
     482             :         }
     483             :     }
     484          27 :     else if (!psField)
     485             :     {
     486           0 :         CPLError(CE_Warning, CPLE_AppDefined,
     487             :                  "Cannot read field %s in %s table", "srid",
     488             :                  osBndTableName.c_str());
     489             :     }
     490          27 :     else if (m_osGDBSpatialRefsFilename.empty())
     491             :     {
     492           0 :         CPLError(CE_Warning, CPLE_AppDefined, "No GDBSpatialRefs table");
     493             :     }
     494             :     else
     495             :     {
     496             :         // FileGDB v10 case
     497          27 :         const int nSRID = psField->Integer;
     498          54 :         FileGDBTable oTableSRS;
     499          27 :         if (oTableSRS.Open(m_osGDBSpatialRefsFilename.c_str(), false))
     500             :         {
     501          27 :             const int iSRTEXT = oTableSRS.GetFieldIdx("SRTEXT");
     502          54 :             if (iSRTEXT < 0 ||
     503          27 :                 oTableSRS.GetField(iSRTEXT)->GetType() != FGFT_STRING)
     504             :             {
     505           0 :                 CPLError(CE_Warning, CPLE_AppDefined,
     506             :                          "Could not find field %s in table %s", "SRTEXT",
     507           0 :                          oTableSRS.GetFilename().c_str());
     508             :             }
     509          27 :             else if (nSRID == 0)
     510             :             {
     511             :                 // BldgHeights.gdb is such. We must fetch the SRS from the
     512             :                 // Definition column of the GDB_Items table
     513             :                 CPLXMLTreeCloser psTree(
     514           0 :                     CPLParseXMLString(osDefinition.c_str()));
     515           0 :                 if (psTree == nullptr)
     516             :                 {
     517           0 :                     CPLError(
     518             :                         CE_Warning, CPLE_AppDefined,
     519             :                         "Cannot parse XML definition. SRS will be missing");
     520             :                 }
     521             :                 else
     522             :                 {
     523           0 :                     CPLStripXMLNamespace(psTree.get(), nullptr, TRUE);
     524             :                     const CPLXMLNode *psInfo =
     525           0 :                         CPLSearchXMLNode(psTree.get(), "=DERasterDataset");
     526           0 :                     if (psInfo)
     527             :                     {
     528           0 :                         auto poSRS = BuildSRS(psInfo);
     529           0 :                         if (poSRS)
     530           0 :                             m_oRasterSRS = *poSRS;
     531             :                     }
     532           0 :                     if (m_oRasterSRS.IsEmpty())
     533             :                     {
     534           0 :                         CPLError(CE_Warning, CPLE_AppDefined,
     535             :                                  "Cannot get SRS from XML definition");
     536             :                     }
     537             :                 }
     538             :             }
     539          54 :             else if (nSRID < 0 || !oTableSRS.SelectRow(nSRID - 1) ||
     540          27 :                      oTableSRS.HasGotError())
     541             :             {
     542           0 :                 CPLError(CE_Warning, CPLE_AppDefined,
     543             :                          "Cannot find record corresponding to SRID = %d",
     544             :                          nSRID);
     545             :             }
     546             :             else
     547             :             {
     548          27 :                 const auto psSRTEXT = oTableSRS.GetFieldValue(iSRTEXT);
     549          27 :                 if (psSRTEXT && psSRTEXT->String)
     550             :                 {
     551          27 :                     if (psSRTEXT->String[0] != '{')
     552             :                     {
     553          26 :                         auto poSRS = BuildSRS(psSRTEXT->String);
     554          26 :                         if (poSRS)
     555             :                         {
     556          26 :                             m_oRasterSRS = *poSRS;
     557          26 :                             poSRS->Release();
     558             :                         }
     559          27 :                     }
     560             :                 }
     561             :                 else
     562             :                 {
     563           0 :                     CPLError(CE_Warning, CPLE_AppDefined,
     564             :                              "Cannot get SRTEXT corresponding to SRID = %d",
     565             :                              nSRID);
     566             :                 }
     567             :             }
     568             :         }
     569             :     }
     570             : 
     571             :     // Open the fras_blk_XXX table, which contains pixel data, as a OGR layer
     572             :     const std::string osBlkTableName(
     573          87 :         std::string("fras_blk_").append(osLayerName).c_str());
     574          29 :     m_poBlkLayer = BuildLayerFromName(osBlkTableName.c_str());
     575          29 :     if (!m_poBlkLayer)
     576             :     {
     577           0 :         CPLError(CE_Failure, CPLE_AppDefined, "Cannot find table %s",
     578             :                  osBlkTableName.c_str());
     579           0 :         return false;
     580             :     }
     581          29 :     auto poFDefn = m_poBlkLayer->GetLayerDefn();
     582          29 :     if (poFDefn->GetFieldIndex("rasterband_id") < 0 ||
     583          29 :         poFDefn->GetFieldIndex("rrd_factor") < 0 ||
     584          29 :         poFDefn->GetFieldIndex("row_nbr") < 0 ||
     585          29 :         poFDefn->GetFieldIndex("col_nbr") < 0 ||
     586          87 :         poFDefn->GetFieldIndex("block_data") < 0 ||
     587          29 :         poFDefn->GetFieldIndex("block_key") < 0)
     588             :     {
     589           0 :         CPLError(CE_Failure, CPLE_AppDefined, "Wrong structure for %s table",
     590             :                  osBlkTableName.c_str());
     591           0 :         return false;
     592             :     }
     593             : 
     594          29 :     nRasterXSize = nWidth;
     595          29 :     nRasterYSize = nHeight;
     596             : 
     597          29 :     if (m_oMapGDALBandToGDBBandId.size() > 1)
     598             :     {
     599           6 :         SetMetadataItem("INTERLEAVE", "BAND", "IMAGE_STRUCTURE");
     600             :     }
     601             : 
     602             :     // Figure out number of overviews by looking at the biggest block_key
     603             :     // (should only involve looking in the corresponding index).
     604          29 :     int nOverviewCount = 0;
     605          58 :     CPLString osSQL;
     606          29 :     osSQL.Printf("SELECT MAX(block_key) FROM \"%s\"", osBlkTableName.c_str());
     607          29 :     auto poSQLLyr = ExecuteSQL(osSQL.c_str(), nullptr, nullptr);
     608          29 :     if (poSQLLyr)
     609             :     {
     610          58 :         auto poFeat = std::unique_ptr<OGRFeature>(poSQLLyr->GetNextFeature());
     611          29 :         if (poFeat)
     612             :         {
     613          29 :             const char *pszMaxKey = poFeat->GetFieldAsString(0);
     614          29 :             if (strlen(pszMaxKey) == strlen("0000BANDOVYYYYXXXX    ") ||
     615           1 :                 strlen(pszMaxKey) == strlen("0000BANDOV-YYYYXXXX    ") ||
     616           1 :                 strlen(pszMaxKey) == strlen("0000BANDOVYYYY-XXXX    ") ||
     617           1 :                 strlen(pszMaxKey) == strlen("0000BANDOV-YYYY-XXXX    "))
     618             :             {
     619          28 :                 char szHex[3] = {0};
     620          28 :                 memcpy(szHex, pszMaxKey + 8, 2);
     621          28 :                 unsigned nMaxRRD = 0;
     622          28 :                 sscanf(szHex, "%02X", &nMaxRRD);
     623          28 :                 nOverviewCount =
     624          28 :                     static_cast<int>(std::min<unsigned>(31, nMaxRRD));
     625             :             }
     626             :         }
     627          29 :         ReleaseResultSet(poSQLLyr);
     628             :     }
     629             : 
     630          29 :     if (m_eRasterCompression == Compression::JPEG)
     631             :     {
     632           2 :         GuessJPEGQuality(nOverviewCount);
     633             :     }
     634             : 
     635             :     // It seems that the top left corner of overviews is registered against
     636             :     // (eminx, emaxy), contrary to the full resolution layer which is registered
     637             :     // against (block_origin_x, block_origin_y).
     638             :     // At least, that's what was observed on the dataset
     639             :     // ftp://ftp.gisdata.mn.gov/pub/gdrs/data/pub/us_mn_state_dnr/water_lake_bathymetry/fgdb_water_lake_bathymetry.zip
     640          29 :     if ((dfBlockOriginX != dfMinX || dfBlockOriginY != dfMaxY) &&
     641             :         nOverviewCount > 0)
     642             :     {
     643           0 :         CPLDebug("OpenFileGDB",
     644             :                  "Ignoring overviews as block origin != (minx, maxy)");
     645           0 :         nOverviewCount = 0;
     646             :     }
     647             : 
     648             :     // Create raster bands
     649             : 
     650             :     // Create mask band of full resolution, if we don't assign a nodata value
     651          29 :     std::unique_ptr<GDALOpenFileGDBRasterBand> poMaskBand;
     652             : 
     653             :     // Default "nodata" padding in areas whose validity mask is 0 ?
     654             :     // Not reliable on integer data types.
     655             :     // Byte -> 0
     656             :     // Int8 -> -128 ?
     657             :     // Int16 -> 32767
     658             :     // UInt16 -> 0
     659             :     // (u)int10 -> 65535
     660             :     // (u)int12 -> 65535
     661             :     // Int32 -> 2147483647
     662             :     // UInt32 -> 2147483647
     663             :     // Float32 -> 3.4e+38
     664             :     // Float64 -> 1.79e+308
     665             : 
     666          29 :     bool bHasNoData = false;
     667          29 :     double dfNoData = 0.0;
     668          58 :     const char *pszNoDataOrMask = CSLFetchNameValueDef(
     669          29 :         poOpenInfo->papszOpenOptions, "NODATA_OR_MASK", "AUTO");
     670          29 :     if (EQUAL(pszNoDataOrMask, "AUTO"))
     671             :     {
     672             :         // In AUTO mode, we only set nodata for Float32/Float64
     673             :         // For other data types, report a mask band.
     674          29 :         if (eDT == GDT_Float32)
     675             :         {
     676           5 :             bHasNoData = true;
     677           5 :             dfNoData = static_cast<double>(static_cast<float>(3.4e+38));
     678             :         }
     679          24 :         else if (eDT == GDT_Float64)
     680             :         {
     681           1 :             bHasNoData = true;
     682           1 :             dfNoData = 1.79e+308;
     683             :         }
     684             :         else
     685             :         {
     686          23 :             poMaskBand = std::make_unique<GDALOpenFileGDBRasterBand>(
     687          46 :                 this, 1, GDT_Byte, 8, nBlockWidth, nBlockHeight, 0, true);
     688             :         }
     689             :     }
     690           0 :     else if (EQUAL(pszNoDataOrMask, "MASK"))
     691             :     {
     692           0 :         poMaskBand = std::make_unique<GDALOpenFileGDBRasterBand>(
     693           0 :             this, 1, GDT_Byte, 8, nBlockWidth, nBlockHeight, 0, true);
     694             :     }
     695           0 :     else if (!EQUAL(pszNoDataOrMask, "NONE"))
     696             :     {
     697           0 :         dfNoData = CPLAtof(pszNoDataOrMask);
     698           0 :         if (eDT == GDT_Float64)
     699             :         {
     700           0 :             bHasNoData = true;
     701             :         }
     702           0 :         else if (eDT == GDT_Float32)
     703             :         {
     704           0 :             if (std::fabs(dfNoData) > std::numeric_limits<float>::max())
     705             :             {
     706           0 :                 CPLError(CE_Failure, CPLE_AppDefined,
     707             :                          "Invalid nodata value %.17g for Float32", dfNoData);
     708           0 :                 return false;
     709             :             }
     710           0 :             bHasNoData = true;
     711             :         }
     712           0 :         else if (GDALDataTypeIsInteger(eDT))
     713             :         {
     714           0 :             double dfMin = 0, dfMax = 0;
     715           0 :             switch (eDT)
     716             :             {
     717           0 :                 case GDT_Int8:
     718           0 :                     dfMin = std::numeric_limits<int8_t>::min();
     719           0 :                     dfMax = std::numeric_limits<int8_t>::max();
     720           0 :                     break;
     721           0 :                 case GDT_Byte:
     722           0 :                     dfMin = std::numeric_limits<uint8_t>::min();
     723           0 :                     dfMax = std::numeric_limits<uint8_t>::max();
     724           0 :                     break;
     725           0 :                 case GDT_Int16:
     726           0 :                     dfMin = std::numeric_limits<int16_t>::min();
     727           0 :                     dfMax = std::numeric_limits<int16_t>::max();
     728           0 :                     break;
     729           0 :                 case GDT_UInt16:
     730           0 :                     dfMin = std::numeric_limits<uint16_t>::min();
     731           0 :                     dfMax = std::numeric_limits<uint16_t>::max();
     732           0 :                     break;
     733           0 :                 case GDT_Int32:
     734           0 :                     dfMin = std::numeric_limits<int32_t>::min();
     735           0 :                     dfMax = std::numeric_limits<int32_t>::max();
     736           0 :                     break;
     737           0 :                 case GDT_UInt32:
     738           0 :                     dfMin = std::numeric_limits<uint32_t>::min();
     739           0 :                     dfMax = std::numeric_limits<uint32_t>::max();
     740           0 :                     break;
     741           0 :                 default:
     742           0 :                     CPLAssert(false);
     743             :                     return false;
     744             :             }
     745           0 :             if (!std::isfinite(dfNoData) || dfNoData < dfMin ||
     746           0 :                 dfNoData > dfMax ||
     747           0 :                 dfNoData != static_cast<double>(static_cast<int64_t>(dfNoData)))
     748             :             {
     749           0 :                 CPLError(CE_Failure, CPLE_AppDefined,
     750             :                          "Invalid nodata value %.17g for %s", dfNoData,
     751             :                          GDALGetDataTypeName(eDT));
     752           0 :                 return false;
     753             :             }
     754           0 :             bHasNoData = true;
     755             :         }
     756             :     }
     757             : 
     758          29 :     GDALOpenFileGDBRasterBand *poMaskBandRef = poMaskBand.get();
     759             : 
     760          70 :     for (int iBand = 1; iBand <= l_nBands; ++iBand)
     761             :     {
     762             :         auto poBand = new GDALOpenFileGDBRasterBand(
     763          41 :             this, iBand, eDT, nBitWidth, nBlockWidth, nBlockHeight, 0, false);
     764          41 :         if (poMaskBandRef)
     765             :         {
     766          35 :             if (iBand == 1)
     767             :             {
     768             :                 // Make the mask band owned by the first raster band
     769          23 :                 poBand->m_poMaskBandOwned = std::move(poMaskBand);
     770          23 :                 poMaskBandRef = poBand->m_poMaskBandOwned.get();
     771          23 :                 poMaskBandRef->m_poMainBand = poBand;
     772             :             }
     773          35 :             poBand->m_poMaskBand = poMaskBandRef;
     774             :         }
     775           6 :         else if (bHasNoData)
     776             :         {
     777           6 :             poBand->m_dfNoData = dfNoData;
     778           6 :             poBand->m_bHasNoData = true;
     779             :         }
     780             : 
     781             :         // Create overview bands
     782         116 :         for (int iOvr = 0; iOvr < nOverviewCount; ++iOvr)
     783             :         {
     784             :             auto poOvrBand = std::make_unique<GDALOpenFileGDBRasterBand>(
     785             :                 this, iBand, eDT, nBitWidth, nBlockWidth, nBlockHeight,
     786         150 :                 iOvr + 1, false);
     787          75 :             if (poBand->m_bHasNoData)
     788             :             {
     789           9 :                 poOvrBand->m_dfNoData = dfNoData;
     790           9 :                 poOvrBand->m_bHasNoData = true;
     791             :             }
     792          75 :             poBand->m_apoOverviewBands.emplace_back(std::move(poOvrBand));
     793             :         }
     794             : 
     795          41 :         SetBand(iBand, poBand);
     796             :     }
     797             : 
     798             :     // Create mask band of overview bands
     799          29 :     if (poMaskBandRef)
     800             :     {
     801          59 :         for (int iOvr = 0; iOvr < nOverviewCount; ++iOvr)
     802             :         {
     803         102 :             for (int iBand = 1; iBand <= l_nBands; ++iBand)
     804             :             {
     805          66 :                 auto poOvrBand = cpl::down_cast<GDALOpenFileGDBRasterBand *>(
     806             :                                      GetRasterBand(iBand))
     807          66 :                                      ->m_apoOverviewBands[iOvr]
     808          66 :                                      .get();
     809          66 :                 if (iBand == 1)
     810             :                 {
     811             :                     // Make the mask band owned by the first raster band
     812             :                     poOvrBand->m_poMaskBandOwned =
     813          36 :                         std::make_unique<GDALOpenFileGDBRasterBand>(
     814           0 :                             this, 1, GDT_Byte, 8, nBlockWidth, nBlockHeight,
     815          36 :                             iOvr + 1, true);
     816          36 :                     poMaskBandRef = poOvrBand->m_poMaskBandOwned.get();
     817          36 :                     poMaskBandRef->m_poMainBand = poOvrBand;
     818             :                 }
     819          66 :                 poOvrBand->m_poMaskBand = poMaskBandRef;
     820             :             }
     821             :         }
     822             :     }
     823             : 
     824          29 :     ReadAuxTable(osLayerName);
     825             : 
     826          29 :     SetMetadataItem("RASTER_DATASET", m_osRasterLayerName.c_str());
     827             : 
     828          29 :     if (!osDefinition.empty())
     829             :     {
     830          27 :         const char *const apszMD[] = {osDefinition.c_str(), nullptr};
     831          27 :         SetMetadata(const_cast<char **>(apszMD), "xml:definition");
     832             :     }
     833             : 
     834          29 :     if (!osDocumentation.empty())
     835             :     {
     836          26 :         const char *const apszMD[] = {osDocumentation.c_str(), nullptr};
     837          26 :         SetMetadata(const_cast<char **>(apszMD), "xml:documentation");
     838             :     }
     839             : 
     840             :     // We are all fine after all those preliminary checks and setups !
     841          29 :     return true;
     842             : }
     843             : 
     844             : /************************************************************************/
     845             : /*                       GuessJPEGQuality()                             */
     846             : /************************************************************************/
     847             : 
     848           2 : void OGROpenFileGDBDataSource::GuessJPEGQuality(int nOverviewCount)
     849             : {
     850             :     // For JPEG, fetch JPEG_QUALITY from the data of the smallest overview level
     851           4 :     CPLString osFilter;
     852             :     osFilter.Printf("block_key = '0000%04X%02X%04X%04X'",
     853             :                     1,  // band
     854             :                     nOverviewCount,
     855             :                     0,  // nBlockYOff
     856             :                     0   // nBlockXOff
     857           2 :     );
     858             : 
     859           2 :     CPLAssert(m_poBlkLayer);
     860           2 :     m_poBlkLayer->SetAttributeFilter(osFilter.c_str());
     861             :     auto poFeature =
     862           4 :         std::unique_ptr<OGRFeature>(m_poBlkLayer->GetNextFeature());
     863           2 :     if (poFeature)
     864             :     {
     865           2 :         const int nFieldIdx = poFeature->GetFieldIndex("block_data");
     866           2 :         CPLAssert(nFieldIdx >= 0);
     867           2 :         if (poFeature->IsFieldSetAndNotNull(nFieldIdx))
     868             :         {
     869           2 :             int nInBytes = 0;
     870             :             const GByte *pabyData =
     871           2 :                 poFeature->GetFieldAsBinary(nFieldIdx, &nInBytes);
     872           2 :             if (nInBytes >= 5)
     873             :             {
     874           2 :                 uint32_t nJPEGSize = nInBytes - 1;
     875           2 :                 uint32_t nJPEGOffset = 1;
     876           2 :                 if (pabyData[0] == 0xFE)
     877             :                 {
     878             :                     // JPEG followed by binary mask
     879           2 :                     memcpy(&nJPEGSize, pabyData + 1, sizeof(uint32_t));
     880           2 :                     CPL_LSBPTR32(&nJPEGSize);
     881           2 :                     if (nJPEGSize > static_cast<unsigned>(nInBytes - 5))
     882             :                     {
     883           0 :                         nJPEGSize = 0;
     884             :                     }
     885           2 :                     nJPEGOffset = 5;
     886             :                 }
     887           0 :                 else if (pabyData[0] != 1)
     888             :                 {
     889           0 :                     nJPEGSize = 0;
     890             :                 }
     891           2 :                 if (nJPEGSize)
     892             :                 {
     893             :                     const CPLString osTmpFilename(
     894           4 :                         VSIMemGenerateHiddenFilename("openfilegdb.jpg"));
     895           2 :                     VSIFCloseL(VSIFileFromMemBuffer(
     896             :                         osTmpFilename.c_str(),
     897           2 :                         const_cast<GByte *>(pabyData + nJPEGOffset), nJPEGSize,
     898             :                         false));
     899           2 :                     const char *const apszDrivers[] = {"JPEG", nullptr};
     900             :                     auto poJPEGDS = std::unique_ptr<GDALDataset>(
     901             :                         GDALDataset::Open(osTmpFilename.c_str(), GDAL_OF_RASTER,
     902           4 :                                           apszDrivers));
     903           2 :                     if (poJPEGDS)
     904             :                     {
     905           2 :                         const char *pszQuality = poJPEGDS->GetMetadataItem(
     906           2 :                             "JPEG_QUALITY", "IMAGE_STRUCTURE");
     907           2 :                         if (pszQuality)
     908             :                         {
     909           2 :                             SetMetadataItem("JPEG_QUALITY", pszQuality,
     910           2 :                                             "IMAGE_STRUCTURE");
     911             :                         }
     912             :                     }
     913           2 :                     VSIUnlink(osTmpFilename);
     914             :                 }
     915             :             }
     916             :         }
     917             :     }
     918           2 : }
     919             : 
     920             : /************************************************************************/
     921             : /*                        ReadAuxTable()                                */
     922             : /************************************************************************/
     923             : 
     924             : // Record type=9 of table fras_ras_XXXX contains a PropertySet object,
     925             : // which may contain statistics
     926             : // For example on
     927             : // https://listdata.thelist.tas.gov.au/opendata/data/NCH_ES_WATER_LOGGING_HAZARD_STATEWIDE.zip
     928          29 : void OGROpenFileGDBDataSource::ReadAuxTable(const std::string &osLayerName)
     929             : {
     930             :     const std::string osAuxTableName(
     931          58 :         std::string("fras_aux_").append(osLayerName).c_str());
     932          29 :     auto poLayer = BuildLayerFromName(osAuxTableName.c_str());
     933          29 :     if (!poLayer)
     934             :     {
     935           0 :         CPLDebug("OpenFileGDB", "Cannot find table %s", osAuxTableName.c_str());
     936           0 :         return;
     937             :     }
     938          29 :     auto poFDefn = poLayer->GetLayerDefn();
     939          29 :     const int iFieldObjectIdx = poFDefn->GetFieldIndex("object");
     940          29 :     if (poFDefn->GetFieldIndex("type") < 0 || iFieldObjectIdx < 0)
     941             :     {
     942           0 :         CPLDebug("OpenFileGDB", "Wrong structure for %s table",
     943             :                  osAuxTableName.c_str());
     944           0 :         return;
     945             :     }
     946          29 :     poLayer->SetAttributeFilter("type = 9");
     947          29 :     auto poFeature = std::unique_ptr<OGRFeature>(poLayer->GetNextFeature());
     948          29 :     if (!poFeature)
     949           3 :         return;
     950          26 :     if (!poFeature->IsFieldSetAndNotNull(iFieldObjectIdx))
     951           0 :         return;
     952          26 :     int nBytes = 0;
     953             :     const GByte *pabyData =
     954          26 :         poFeature->GetFieldAsBinary(iFieldObjectIdx, &nBytes);
     955          26 :     if (!pabyData || nBytes == 0)
     956           0 :         return;
     957          26 :     int iOffset = 0;
     958             : 
     959         545 :     const auto ReadString = [pabyData, &iOffset, nBytes](std::string &osStr)
     960             :     {
     961         110 :         if (iOffset > nBytes - 4)
     962          21 :             return false;
     963             :         int nStrLength;
     964          89 :         memcpy(&nStrLength, pabyData + iOffset, 4);
     965          89 :         CPL_LSBPTR32(&nStrLength);
     966          89 :         iOffset += 4;
     967          89 :         if (nStrLength <= 2 || iOffset > nBytes - nStrLength)
     968           5 :             return false;
     969          84 :         if ((nStrLength % 2) != 0)
     970           0 :             return false;
     971             :         // nStrLength / 2 to get the number of characters
     972             :         // and - 1 to remove the null terminating one
     973          84 :         osStr = ReadUTF16String(pabyData + iOffset, nStrLength / 2 - 1);
     974          84 :         iOffset += nStrLength;
     975          84 :         return true;
     976          26 :     };
     977             : 
     978             :     // pabyData is an ArcObject "PropertySet" object, which is key/value
     979             :     // dictionary. This is hard to parse given there are variable-length value
     980             :     // whose size is not explicit. So let's use a heuristics by looking for
     981             :     // the beginning of a inner PropertySet with band properties that starts
     982             :     // with a KIND=BAND key value pair.
     983          26 :     constexpr GByte abyNeedle[] = {
     984             :         'K', 0, 'I', 0, 'N', 0, 'D', 0, 0, 0, 8, 0,  // 8 = string
     985             :         10,  0, 0,   0,  // number of bytes of following value
     986             :         'B', 0, 'A', 0, 'N', 0, 'D', 0, 0, 0};
     987          26 :     constexpr int nNeedleSize = static_cast<int>(sizeof(abyNeedle));
     988             : 
     989          26 :     for (int iBand = 1; iBand <= nBands; ++iBand)
     990             :     {
     991          26 :         int iNewOffset = -1;
     992       10218 :         for (int i = iOffset; i < nBytes - nNeedleSize; ++i)
     993             :         {
     994       10218 :             if (pabyData[i] == 'K' &&
     995          78 :                 memcmp(pabyData + i, abyNeedle, nNeedleSize) == 0)
     996             :             {
     997          26 :                 iNewOffset = i + nNeedleSize;
     998          26 :                 break;
     999             :             }
    1000             :         }
    1001          26 :         if (iNewOffset < 0)
    1002           0 :             return;
    1003          26 :         iOffset = iNewOffset;
    1004             : 
    1005             :         // Try to read as many key/value pairs as possible
    1006             :         while (true)
    1007             :         {
    1008             :             // Read key
    1009          94 :             std::string osKey;
    1010          94 :             if (!ReadString(osKey))
    1011          26 :                 return;
    1012             : 
    1013             :             // Read value type as a short
    1014             :             uint16_t nValueType;
    1015          68 :             if (iOffset > nBytes - 2)
    1016           0 :                 return;
    1017          68 :             memcpy(&nValueType, pabyData + iOffset, 2);
    1018          68 :             CPL_LSBPTR16(&nValueType);
    1019          68 :             iOffset += 2;
    1020             : 
    1021             :             // Skip over non-string values
    1022          68 :             if (nValueType == 0 || nValueType == 1)  // null / empty value
    1023             :             {
    1024           0 :                 continue;
    1025             :             }
    1026          68 :             if (nValueType == 2)  // short value
    1027             :             {
    1028          26 :                 if (iOffset > nBytes - 2)
    1029           0 :                     return;
    1030          26 :                 iOffset += 2;
    1031          26 :                 continue;
    1032             :             }
    1033             : 
    1034          42 :             if (nValueType == 3 || nValueType == 4)  // int or long value
    1035             :             {
    1036          26 :                 if (iOffset > nBytes - 4)
    1037           0 :                     return;
    1038          26 :                 iOffset += 4;
    1039          26 :                 continue;
    1040             :             }
    1041             : 
    1042          16 :             if (nValueType == 5 || nValueType == 7)  // double or date value
    1043             :             {
    1044           0 :                 if (iOffset > nBytes - 8)
    1045           0 :                     return;
    1046           0 :                 iOffset += 8;
    1047           0 :                 continue;
    1048             :             }
    1049             : 
    1050          16 :             if (nValueType != 8)  // 8 = string
    1051             :             {
    1052             :                 // Give up with this band as the value type is not handled,
    1053             :                 // and we can't skip over it.
    1054           0 :                 break;
    1055             :             }
    1056             : 
    1057             :             // Read string value
    1058          16 :             std::string osValue;
    1059          16 :             if (!ReadString(osValue))
    1060           0 :                 return;
    1061             : 
    1062          32 :             GetRasterBand(iBand)->SetMetadataItem(osKey.c_str(),
    1063          16 :                                                   osValue.c_str());
    1064          68 :         }
    1065             :     }
    1066             : }
    1067             : 
    1068             : /************************************************************************/
    1069             : /*                         GetGeoTransform()                            */
    1070             : /************************************************************************/
    1071             : 
    1072           4 : CPLErr OGROpenFileGDBDataSource::GetGeoTransform(double *padfGeoTransform)
    1073             : {
    1074           4 :     memcpy(padfGeoTransform, m_adfGeoTransform.data(),
    1075             :            sizeof(m_adfGeoTransform));
    1076           4 :     return m_bHasGeoTransform ? CE_None : CE_Failure;
    1077             : }
    1078             : 
    1079             : /************************************************************************/
    1080             : /*                         GetSpatialRef()                              */
    1081             : /************************************************************************/
    1082             : 
    1083           4 : const OGRSpatialReference *OGROpenFileGDBDataSource::GetSpatialRef() const
    1084             : {
    1085           4 :     return m_oRasterSRS.IsEmpty() ? nullptr : &m_oRasterSRS;
    1086             : }
    1087             : 
    1088             : /************************************************************************/
    1089             : /*                     GDALOpenFileGDBRasterBand()                      */
    1090             : /************************************************************************/
    1091             : 
    1092         175 : GDALOpenFileGDBRasterBand::GDALOpenFileGDBRasterBand(
    1093             :     OGROpenFileGDBDataSource *poDSIn, int nBandIn, GDALDataType eDT,
    1094             :     int nBitWidth, int nBlockWidth, int nBlockHeight, int nOverviewLevel,
    1095         175 :     bool bIsMask)
    1096             :     : m_nBitWidth(nBitWidth), m_nOverviewLevel(nOverviewLevel),
    1097         175 :       m_bIsMask(bIsMask)
    1098             : {
    1099         175 :     poDS = poDSIn;
    1100         175 :     nBand = nBandIn;
    1101         175 :     eDataType = eDT;
    1102         175 :     nRasterXSize = std::max(1, poDSIn->GetRasterXSize() >> nOverviewLevel);
    1103         175 :     nRasterYSize = std::max(1, poDSIn->GetRasterYSize() >> nOverviewLevel);
    1104         175 :     nBlockXSize = nBlockWidth;
    1105         175 :     nBlockYSize = nBlockHeight;
    1106         175 :     if (nBitWidth < 8)
    1107             :     {
    1108           8 :         SetMetadataItem("NBITS", CPLSPrintf("%d", nBitWidth),
    1109             :                         "IMAGE_STRUCTURE");
    1110             :     }
    1111         175 : }
    1112             : 
    1113             : /************************************************************************/
    1114             : /*                         SetNoDataFromMask()                          */
    1115             : /************************************************************************/
    1116             : 
    1117             : template <class T>
    1118           0 : static void SetNoDataFromMask(void *pImage, const GByte *pabyMask,
    1119             :                               size_t nPixels, double dfNoData)
    1120             : {
    1121           0 :     const T noData = static_cast<T>(dfNoData);
    1122           0 :     const T noDataReplacement =
    1123           0 :         noData == std::numeric_limits<T>::max() ? noData - 1 : noData + 1;
    1124           0 :     bool bHasWarned = false;
    1125           0 :     for (size_t i = 0; i < nPixels; ++i)
    1126             :     {
    1127           0 :         if (pabyMask && !(pabyMask[i / 8] & (0x80 >> (i & 7))))
    1128             :         {
    1129           0 :             static_cast<T *>(pImage)[i] = noData;
    1130             :         }
    1131           0 :         else if (static_cast<T *>(pImage)[i] == noData)
    1132             :         {
    1133           0 :             static_cast<T *>(pImage)[i] = noDataReplacement;
    1134           0 :             if (!bHasWarned)
    1135             :             {
    1136           0 :                 bHasWarned = true;
    1137           0 :                 CPLError(CE_Warning, CPLE_AppDefined,
    1138             :                          "Valid data found with value equal to nodata (%.0f). "
    1139             :                          "Got substituted with %.0f",
    1140             :                          static_cast<double>(noData),
    1141             :                          static_cast<double>(noDataReplacement));
    1142             :             }
    1143             :         }
    1144             :     }
    1145           0 : }
    1146             : 
    1147             : /************************************************************************/
    1148             : /*                         IReadBlock()                                 */
    1149             : /************************************************************************/
    1150             : 
    1151         118 : CPLErr GDALOpenFileGDBRasterBand::IReadBlock(int nBlockXOff, int nBlockYOff,
    1152             :                                              void *pImage)
    1153             : {
    1154         118 :     auto poGDS = cpl::down_cast<OGROpenFileGDBDataSource *>(poDS);
    1155         118 :     auto &poLyr = poGDS->m_poBlkLayer;
    1156             : 
    1157             :     // Return (pointer to image data, owner block). Works when called from main band
    1158             :     // or mask band. owner block must be DropLock() once done (if not null)
    1159         238 :     const auto GetImageData = [this, nBlockXOff, nBlockYOff, pImage]()
    1160             :     {
    1161         118 :         void *pImageData = nullptr;
    1162         118 :         GDALRasterBlock *poBlock = nullptr;
    1163         118 :         if (m_bIsMask)
    1164             :         {
    1165           1 :             CPLAssert(m_poMainBand);
    1166           1 :             poBlock =
    1167           1 :                 m_poMainBand->TryGetLockedBlockRef(nBlockXOff, nBlockYOff);
    1168           1 :             if (poBlock)
    1169             :             {
    1170             :                 // The block is already in cache. Return (null, null)
    1171           0 :                 poBlock->DropLock();
    1172           0 :                 poBlock = nullptr;
    1173             :             }
    1174             :             else
    1175             :             {
    1176           2 :                 poBlock = m_poMainBand->GetLockedBlockRef(nBlockXOff,
    1177           1 :                                                           nBlockYOff, true);
    1178           1 :                 if (poBlock)
    1179           1 :                     pImageData = poBlock->GetDataRef();
    1180             :             }
    1181             :         }
    1182             :         else
    1183             :         {
    1184         117 :             pImageData = pImage;
    1185             :         }
    1186         236 :         return std::make_pair(pImageData, poBlock);
    1187         118 :     };
    1188             : 
    1189             :     // Return (pointer to mask data, owner block). Works when called from main band
    1190             :     // or mask band. owner block must be DropLock() once done (if not null)
    1191         404 :     const auto GetMaskData = [this, nBlockXOff, nBlockYOff, pImage]()
    1192             :     {
    1193         114 :         void *pMaskData = nullptr;
    1194         114 :         GDALRasterBlock *poBlock = nullptr;
    1195         114 :         if (m_bIsMask)
    1196             :         {
    1197           1 :             pMaskData = pImage;
    1198             :         }
    1199             :         else
    1200             :         {
    1201         113 :             CPLAssert(m_poMaskBand);
    1202         113 :             poBlock =
    1203         113 :                 m_poMaskBand->TryGetLockedBlockRef(nBlockXOff, nBlockYOff);
    1204         113 :             if (poBlock)
    1205             :             {
    1206             :                 // The block is already in cache. Return (null, null)
    1207          50 :                 poBlock->DropLock();
    1208          50 :                 poBlock = nullptr;
    1209             :             }
    1210             :             else
    1211             :             {
    1212         126 :                 poBlock = m_poMaskBand->GetLockedBlockRef(nBlockXOff,
    1213          63 :                                                           nBlockYOff, true);
    1214          63 :                 if (poBlock)
    1215          63 :                     pMaskData = poBlock->GetDataRef();
    1216             :             }
    1217             :         }
    1218         228 :         return std::make_pair(pMaskData, poBlock);
    1219         118 :     };
    1220             : 
    1221             :     const GDALDataType eImageDT =
    1222         118 :         m_poMainBand ? m_poMainBand->GetRasterDataType() : eDataType;
    1223         118 :     const size_t nPixels = static_cast<size_t>(nBlockXSize) * nBlockYSize;
    1224             : 
    1225             :     const auto FillMissingBlock =
    1226           6 :         [this, eImageDT, nPixels, &GetImageData, &GetMaskData]()
    1227             :     {
    1228             :         // Set image data to nodata / 0
    1229             :         {
    1230           1 :             auto imageDataAndBlock = GetImageData();
    1231           1 :             auto pImageData = imageDataAndBlock.first;
    1232           1 :             auto poBlock = imageDataAndBlock.second;
    1233           1 :             if (pImageData)
    1234             :             {
    1235           1 :                 const int nDTSize = GDALGetDataTypeSizeBytes(eImageDT);
    1236           1 :                 if (m_bHasNoData)
    1237             :                 {
    1238           1 :                     GDALCopyWords64(&m_dfNoData, GDT_Float64, 0, pImageData,
    1239             :                                     eImageDT, nDTSize, nPixels);
    1240             :                 }
    1241             :                 else
    1242             :                 {
    1243           0 :                     memset(pImageData, 0, nPixels * nDTSize);
    1244             :                 }
    1245             :             }
    1246           1 :             if (poBlock)
    1247           0 :                 poBlock->DropLock();
    1248             :         }
    1249             : 
    1250             :         // Set mask band to 0 (when it exists)
    1251           1 :         if (m_poMaskBand || m_bIsMask)
    1252             :         {
    1253           0 :             auto maskDataAndBlock = GetMaskData();
    1254           0 :             auto pMaskData = maskDataAndBlock.first;
    1255           0 :             auto poBlock = maskDataAndBlock.second;
    1256           0 :             if (pMaskData)
    1257             :             {
    1258           0 :                 const size_t nSize =
    1259           0 :                     static_cast<size_t>(nBlockXSize) * nBlockYSize;
    1260           0 :                 memset(pMaskData, 0, nSize);
    1261             :             }
    1262           0 :             if (poBlock)
    1263           0 :                 poBlock->DropLock();
    1264             :         }
    1265           1 :     };
    1266             : 
    1267             :     // Fetch block data from fras_blk_XXX layer
    1268         118 :     const int nGDALBandId = m_bIsMask ? 1 : nBand;
    1269         118 :     auto oIter = poGDS->m_oMapGDALBandToGDBBandId.find(nGDALBandId);
    1270         118 :     if (oIter == poGDS->m_oMapGDALBandToGDBBandId.end())
    1271             :     {
    1272           0 :         CPLError(CE_Failure, CPLE_AppDefined,
    1273             :                  "poGDS->m_oMapGDALBandToGDBBandId.find(%d) failed",
    1274             :                  nGDALBandId);
    1275           0 :         return CE_Failure;
    1276             :     }
    1277         118 :     const int nGDBRasterBandId = oIter->second;
    1278             : 
    1279         236 :     CPLString osFilter;
    1280             :     /* osFilter.Printf("rasterband_id = %d AND rrd_factor = %d AND row_nbr = %d "
    1281             :                     "AND col_nbr = %d",
    1282             :                     nGDBRasterBandId,
    1283             :                     m_nOverviewLevel, nBlockYOff, nBlockXOff);
    1284             :     */
    1285         118 :     const int nColNbr = nBlockXOff + poGDS->m_nShiftBlockX;
    1286         118 :     const int nRowNbr = nBlockYOff + poGDS->m_nShiftBlockY;
    1287         118 :     if (nRowNbr >= 0 && nColNbr >= 0)
    1288             :     {
    1289         118 :         osFilter.Printf("block_key = '0000%04X%02X%04X%04X'", nGDBRasterBandId,
    1290         118 :                         m_nOverviewLevel, nRowNbr, nColNbr);
    1291             :     }
    1292           0 :     else if (nRowNbr < 0 && nColNbr >= 0)
    1293             :     {
    1294           0 :         osFilter.Printf("block_key = '0000%04X%02X-%04X%04X'", nGDBRasterBandId,
    1295           0 :                         m_nOverviewLevel, -nRowNbr, nColNbr);
    1296             :     }
    1297           0 :     else if (nRowNbr >= 0 && nColNbr < 0)
    1298             :     {
    1299           0 :         osFilter.Printf("block_key = '0000%04X%02X%04X-%04X'", nGDBRasterBandId,
    1300           0 :                         m_nOverviewLevel, nRowNbr, -nColNbr);
    1301             :     }
    1302             :     else /* if( nRowNbr < 0 && nColNbr < 0 ) */
    1303             :     {
    1304             :         osFilter.Printf("block_key = '0000%04X%02X-%04X-%04X'",
    1305           0 :                         nGDBRasterBandId, m_nOverviewLevel, -nRowNbr, -nColNbr);
    1306             :     }
    1307             :     // CPLDebug("OpenFileGDB", "Request %s", osFilter.c_str());
    1308         118 :     poLyr->SetAttributeFilter(osFilter.c_str());
    1309         236 :     auto poFeature = std::unique_ptr<OGRFeature>(poLyr->GetNextFeature());
    1310         118 :     const int nImageDTSize = GDALGetDataTypeSizeBytes(eImageDT);
    1311         118 :     if (!poFeature)
    1312             :     {
    1313             :         // Missing blocks are legit
    1314           1 :         FillMissingBlock();
    1315           1 :         return CE_None;
    1316             :     }
    1317         117 :     const int nFieldIdx = poFeature->GetFieldIndex("block_data");
    1318         117 :     CPLAssert(nFieldIdx >= 0);
    1319         117 :     int nInBytes = 0;
    1320         117 :     if (!poFeature->IsFieldSetAndNotNull(nFieldIdx))
    1321             :     {
    1322             :         // block_data unset found on ForestFalls.gdb
    1323           0 :         FillMissingBlock();
    1324           0 :         return CE_None;
    1325             :     }
    1326         117 :     const GByte *pabyData = poFeature->GetFieldAsBinary(nFieldIdx, &nInBytes);
    1327         117 :     if (nInBytes == 0)
    1328             :     {
    1329           0 :         CPLError(CE_Failure, CPLE_AppDefined, "Image block is empty");
    1330           0 :         return CE_Failure;
    1331             :     }
    1332             : 
    1333             :     // The input buffer may be concatenated with a 1-bit binary mask
    1334         117 :     const size_t nImageSize = nPixels * nImageDTSize;
    1335         117 :     const int nImageBitWidth =
    1336         117 :         m_poMainBand ? m_poMainBand->m_nBitWidth : m_nBitWidth;
    1337         117 :     const size_t nImageSizePacked = (nPixels * nImageBitWidth + 7) / 8;
    1338         117 :     const size_t nBinaryMaskSize = (nPixels + 7) / 8;
    1339         117 :     const size_t nImageSizeWithBinaryMask = nImageSizePacked + nBinaryMaskSize;
    1340             : 
    1341             :     // Unpack 1-bit, 2-bit, 4-bit data to full byte
    1342             :     const auto ExpandSubByteData =
    1343     2175500 :         [nPixels, nImageBitWidth](const GByte *pabyInput, void *pDstBuffer)
    1344             :     {
    1345          24 :         CPLAssert(nImageBitWidth < 8);
    1346             : 
    1347          24 :         size_t iBitOffset = 0;
    1348      393240 :         for (size_t i = 0; i < nPixels; ++i)
    1349             :         {
    1350      393216 :             unsigned nOutWord = 0;
    1351             : 
    1352     1523710 :             for (int iBit = 0; iBit < nImageBitWidth; ++iBit)
    1353             :             {
    1354     1130500 :                 if (pabyInput[iBitOffset >> 3] & (0x80 >> (iBitOffset & 7)))
    1355             :                 {
    1356      258524 :                     nOutWord |= (1 << (nImageBitWidth - 1 - iBit));
    1357             :                 }
    1358     1130500 :                 ++iBitOffset;
    1359             :             }
    1360             : 
    1361      393216 :             static_cast<GByte *>(pDstBuffer)[i] = static_cast<GByte>(nOutWord);
    1362             :         }
    1363         141 :     };
    1364             : 
    1365         117 :     const GByte *pabyMask = nullptr;
    1366         116 :     auto &abyTmpBuffer =
    1367         117 :         m_poMainBand ? m_poMainBand->m_abyTmpBuffer : m_abyTmpBuffer;
    1368             : 
    1369         117 :     switch (poGDS->m_eRasterCompression)
    1370             :     {
    1371           6 :         case OGROpenFileGDBDataSource::Compression::NONE:
    1372             :         {
    1373           6 :             if (static_cast<unsigned>(nInBytes) != nImageSizePacked &&
    1374           6 :                 static_cast<unsigned>(nInBytes) != nImageSizeWithBinaryMask)
    1375             :             {
    1376           0 :                 CPLError(CE_Failure, CPLE_AppDefined,
    1377             :                          "Not expected number of input bytes: %d", nInBytes);
    1378           0 :                 return CE_Failure;
    1379             :             }
    1380             : 
    1381           6 :             auto imageDataAndBlock = GetImageData();
    1382           6 :             auto pImageData = imageDataAndBlock.first;
    1383           6 :             auto poBlock = imageDataAndBlock.second;
    1384             : 
    1385           6 :             if (pImageData)
    1386             :             {
    1387           6 :                 if (nImageSizePacked == nImageSize)
    1388             :                 {
    1389           6 :                     memcpy(pImageData, pabyData, nImageSize);
    1390             : #ifdef CPL_LSB
    1391           6 :                     if (nImageDTSize > 1)
    1392             :                     {
    1393           5 :                         GDALSwapWordsEx(pImageData, nImageDTSize, nPixels,
    1394             :                                         nImageDTSize);
    1395             :                     }
    1396             : #endif
    1397             :                 }
    1398             :                 else
    1399             :                 {
    1400           0 :                     ExpandSubByteData(pabyData, pImageData);
    1401             :                 }
    1402             :             }
    1403           6 :             if (poBlock)
    1404           0 :                 poBlock->DropLock();
    1405             : 
    1406           6 :             if (static_cast<unsigned>(nInBytes) == nImageSizeWithBinaryMask)
    1407           6 :                 pabyMask = pabyData + nImageSizePacked;
    1408           6 :             break;
    1409             :         }
    1410             : 
    1411          63 :         case OGROpenFileGDBDataSource::Compression::LZ77:
    1412             :         {
    1413          63 :             if (abyTmpBuffer.empty())
    1414             :             {
    1415             :                 try
    1416             :                 {
    1417          18 :                     abyTmpBuffer.resize(nImageSizeWithBinaryMask);
    1418             :                 }
    1419           0 :                 catch (const std::bad_alloc &e)
    1420             :                 {
    1421           0 :                     CPLError(CE_Failure, CPLE_OutOfMemory, "%s", e.what());
    1422           0 :                     return CE_Failure;
    1423             :                 }
    1424             :             }
    1425             : 
    1426          63 :             size_t nOutBytes = 0;
    1427          63 :             GByte *outPtr = abyTmpBuffer.data();
    1428          63 :             assert(outPtr != nullptr);  // For Coverity Scan
    1429          63 :             if (!CPLZLibInflate(pabyData, nInBytes, outPtr, abyTmpBuffer.size(),
    1430         126 :                                 &nOutBytes) ||
    1431          63 :                 !(nOutBytes == nImageSizePacked ||
    1432          48 :                   nOutBytes == nImageSizeWithBinaryMask))
    1433             :             {
    1434           0 :                 CPLError(
    1435             :                     CE_Failure, CPLE_AppDefined,
    1436             :                     "CPLZLibInflate() failed: nInBytes = %u, nOutBytes = %u, "
    1437             :                     "nImageSizePacked = %u, "
    1438             :                     "nImageSizeWithBinaryMask = %u",
    1439             :                     unsigned(nInBytes), unsigned(nOutBytes),
    1440             :                     unsigned(nImageSizePacked),
    1441             :                     unsigned(nImageSizeWithBinaryMask));
    1442           0 :                 return CE_Failure;
    1443             :             }
    1444             : 
    1445          63 :             auto imageDataAndBlock = GetImageData();
    1446          63 :             auto pImageData = imageDataAndBlock.first;
    1447          63 :             auto poBlock = imageDataAndBlock.second;
    1448             : 
    1449          63 :             if (pImageData)
    1450             :             {
    1451          63 :                 if (nImageSizePacked == nImageSize)
    1452             :                 {
    1453          39 :                     memcpy(pImageData, abyTmpBuffer.data(), nImageSize);
    1454             : #ifdef CPL_LSB
    1455          39 :                     if (nImageDTSize > 1)
    1456             :                     {
    1457           6 :                         GDALSwapWordsEx(pImageData, nImageDTSize, nPixels,
    1458             :                                         nImageDTSize);
    1459             :                     }
    1460             : #endif
    1461             :                 }
    1462             :                 else
    1463             :                 {
    1464          24 :                     ExpandSubByteData(abyTmpBuffer.data(), pImageData);
    1465             :                 }
    1466             :             }
    1467          63 :             if (poBlock)
    1468           1 :                 poBlock->DropLock();
    1469             : 
    1470          63 :             if (nOutBytes == nImageSizeWithBinaryMask)
    1471          48 :                 pabyMask = abyTmpBuffer.data() + nImageSizePacked;
    1472          63 :             break;
    1473             :         }
    1474             : 
    1475          24 :         case OGROpenFileGDBDataSource::Compression::JPEG:
    1476             :         {
    1477          24 :             if (GDALGetDriverByName("JPEG") == nullptr)
    1478             :             {
    1479           0 :                 CPLError(CE_Failure, CPLE_AppDefined, "JPEG driver missing");
    1480           0 :                 return CE_Failure;
    1481             :             }
    1482             : 
    1483          24 :             if (static_cast<unsigned>(nInBytes) < 5)
    1484             :             {
    1485           0 :                 CPLError(CE_Failure, CPLE_AppDefined,
    1486             :                          "Not expected number of input bytes: %d", nInBytes);
    1487           0 :                 return CE_Failure;
    1488             :             }
    1489          24 :             uint32_t nJPEGSize = nInBytes - 1;
    1490          24 :             uint32_t nJPEGOffset = 1;
    1491          24 :             if (pabyData[0] == 0xFE)
    1492             :             {
    1493             :                 // JPEG followed by binary mask
    1494          15 :                 memcpy(&nJPEGSize, pabyData + 1, sizeof(uint32_t));
    1495          15 :                 CPL_LSBPTR32(&nJPEGSize);
    1496          15 :                 if (nJPEGSize > static_cast<unsigned>(nInBytes - 5))
    1497             :                 {
    1498           0 :                     CPLError(CE_Failure, CPLE_AppDefined,
    1499             :                              "Invalid nJPEGSize = %u", nJPEGSize);
    1500           0 :                     return CE_Failure;
    1501             :                 }
    1502          15 :                 nJPEGOffset = 5;
    1503             : 
    1504          15 :                 if (abyTmpBuffer.empty())
    1505             :                 {
    1506             :                     try
    1507             :                     {
    1508           3 :                         abyTmpBuffer.resize(nBinaryMaskSize);
    1509             :                     }
    1510           0 :                     catch (const std::bad_alloc &e)
    1511             :                     {
    1512           0 :                         CPLError(CE_Failure, CPLE_OutOfMemory, "%s", e.what());
    1513           0 :                         return CE_Failure;
    1514             :                     }
    1515             :                 }
    1516          15 :                 size_t nOutBytes = 0;
    1517          15 :                 GByte *outPtr = abyTmpBuffer.data();
    1518          15 :                 assert(outPtr != nullptr);  // For Coverity Scan
    1519          45 :                 if (CPLZLibInflate(pabyData + 5 + nJPEGSize,
    1520          15 :                                    nInBytes - 5 - nJPEGSize, outPtr,
    1521          30 :                                    nBinaryMaskSize, &nOutBytes) &&
    1522          15 :                     nOutBytes == nBinaryMaskSize)
    1523             :                 {
    1524          15 :                     pabyMask = abyTmpBuffer.data();
    1525             :                 }
    1526             :                 else
    1527             :                 {
    1528           0 :                     CPLError(CE_Warning, CPLE_AppDefined,
    1529             :                              "Cannot decompress binary mask");
    1530             :                 }
    1531             :             }
    1532           9 :             else if (pabyData[0] != 1)
    1533             :             {
    1534           0 :                 CPLError(CE_Failure, CPLE_AppDefined, "Invalid JPEG blob");
    1535           0 :                 return CE_Failure;
    1536             :             }
    1537             : 
    1538             :             const CPLString osTmpFilename(
    1539          24 :                 VSIMemGenerateHiddenFilename("openfilegdb.jpg"));
    1540          24 :             VSIFCloseL(VSIFileFromMemBuffer(
    1541             :                 osTmpFilename.c_str(),
    1542          24 :                 const_cast<GByte *>(pabyData + nJPEGOffset), nJPEGSize, false));
    1543          24 :             const char *const apszDrivers[] = {"JPEG", nullptr};
    1544             :             auto poJPEGDS = std::unique_ptr<GDALDataset>(GDALDataset::Open(
    1545          24 :                 osTmpFilename.c_str(), GDAL_OF_RASTER, apszDrivers));
    1546          24 :             if (!poJPEGDS)
    1547             :             {
    1548           0 :                 VSIUnlink(osTmpFilename.c_str());
    1549           0 :                 CPLError(CE_Failure, CPLE_AppDefined, "Cannot open JPEG blob");
    1550           0 :                 return CE_Failure;
    1551             :             }
    1552          24 :             if (poJPEGDS->GetRasterCount() != 1 ||
    1553          48 :                 poJPEGDS->GetRasterXSize() != nBlockXSize ||
    1554          24 :                 poJPEGDS->GetRasterYSize() != nBlockYSize)
    1555             :             {
    1556           0 :                 VSIUnlink(osTmpFilename.c_str());
    1557           0 :                 CPLError(CE_Failure, CPLE_AppDefined,
    1558             :                          "Inconsistent characteristics of JPEG blob");
    1559           0 :                 return CE_Failure;
    1560             :             }
    1561             : 
    1562          24 :             auto imageDataAndBlock = GetImageData();
    1563          24 :             auto pImageData = imageDataAndBlock.first;
    1564          24 :             auto poBlock = imageDataAndBlock.second;
    1565             : 
    1566             :             const CPLErr eErr =
    1567             :                 pImageData
    1568          24 :                     ? poJPEGDS->GetRasterBand(1)->RasterIO(
    1569             :                           GF_Read, 0, 0, nBlockXSize, nBlockYSize, pImageData,
    1570             :                           nBlockXSize, nBlockYSize, eImageDT, 0, 0, nullptr)
    1571          24 :                     : CE_None;
    1572          24 :             VSIUnlink(osTmpFilename.c_str());
    1573          24 :             if (poBlock)
    1574           0 :                 poBlock->DropLock();
    1575             : 
    1576          24 :             if (eErr != CE_None)
    1577             :             {
    1578           0 :                 CPLError(CE_Failure, CPLE_AppDefined, "Cannot read JPEG blob");
    1579           0 :                 return CE_Failure;
    1580             :             }
    1581             : 
    1582          24 :             break;
    1583             :         }
    1584             : 
    1585          24 :         case OGROpenFileGDBDataSource::Compression::JPEG2000:
    1586             :         {
    1587          24 :             const char *const apszDrivers[] = {"JP2KAK",      "JP2ECW",
    1588             :                                                "JP2OpenJPEG", "JP2MrSID",
    1589             :                                                "JP2Lura",     nullptr};
    1590          24 :             bool bFoundJP2Driver = false;
    1591          48 :             for (const char *pszDriver : apszDrivers)
    1592             :             {
    1593          48 :                 if (pszDriver && GDALGetDriverByName(pszDriver))
    1594             :                 {
    1595          24 :                     bFoundJP2Driver = true;
    1596          24 :                     break;
    1597             :                 }
    1598             :             }
    1599          24 :             if (!bFoundJP2Driver)
    1600             :             {
    1601           0 :                 CPLError(CE_Failure, CPLE_AppDefined,
    1602             :                          "Did not find any JPEG2000 capable driver");
    1603           0 :                 return CE_Failure;
    1604             :             }
    1605             : 
    1606          24 :             if (static_cast<unsigned>(nInBytes) < 5)
    1607             :             {
    1608           0 :                 CPLError(CE_Failure, CPLE_AppDefined,
    1609             :                          "Not expected number of input bytes: %d", nInBytes);
    1610           0 :                 return CE_Failure;
    1611             :             }
    1612          24 :             uint32_t nJPEGSize = nInBytes - 1;
    1613          24 :             uint32_t nJPEGOffset = 1;
    1614          24 :             if (pabyData[0] == 0xFF)
    1615             :             {
    1616             :                 // JPEG2000 followed by binary mask
    1617          15 :                 memcpy(&nJPEGSize, pabyData + 1, sizeof(uint32_t));
    1618          15 :                 CPL_LSBPTR32(&nJPEGSize);
    1619          15 :                 if (nJPEGSize > static_cast<unsigned>(nInBytes - 5))
    1620             :                 {
    1621           0 :                     CPLError(CE_Failure, CPLE_AppDefined,
    1622             :                              "Invalid nJPEGSize = %u", nJPEGSize);
    1623           0 :                     return CE_Failure;
    1624             :                 }
    1625          15 :                 nJPEGOffset = 5;
    1626             : 
    1627          15 :                 if (abyTmpBuffer.empty())
    1628             :                 {
    1629             :                     try
    1630             :                     {
    1631           3 :                         abyTmpBuffer.resize(nBinaryMaskSize);
    1632             :                     }
    1633           0 :                     catch (const std::bad_alloc &e)
    1634             :                     {
    1635           0 :                         CPLError(CE_Failure, CPLE_OutOfMemory, "%s", e.what());
    1636           0 :                         return CE_Failure;
    1637             :                     }
    1638             :                 }
    1639          15 :                 size_t nOutBytes = 0;
    1640          15 :                 GByte *outPtr = abyTmpBuffer.data();
    1641          15 :                 assert(outPtr != nullptr);  // For Coverity Scan
    1642          45 :                 if (CPLZLibInflate(pabyData + 5 + nJPEGSize,
    1643          15 :                                    nInBytes - 5 - nJPEGSize, outPtr,
    1644          30 :                                    nBinaryMaskSize, &nOutBytes) &&
    1645          15 :                     nOutBytes == nBinaryMaskSize)
    1646             :                 {
    1647          15 :                     pabyMask = outPtr;
    1648             :                 }
    1649             :                 else
    1650             :                 {
    1651           0 :                     CPLError(CE_Warning, CPLE_AppDefined,
    1652             :                              "Cannot decompress binary mask");
    1653             :                 }
    1654             :             }
    1655           9 :             else if (pabyData[0] != 0)
    1656             :             {
    1657           0 :                 CPLError(CE_Failure, CPLE_AppDefined, "Invalid JPEG2000 blob");
    1658           0 :                 return CE_Failure;
    1659             :             }
    1660             : 
    1661             :             const CPLString osTmpFilename(
    1662          24 :                 VSIMemGenerateHiddenFilename("openfilegdb.j2k"));
    1663          24 :             VSIFCloseL(VSIFileFromMemBuffer(
    1664             :                 osTmpFilename.c_str(),
    1665          24 :                 const_cast<GByte *>(pabyData + nJPEGOffset), nJPEGSize, false));
    1666             :             auto poJP2KDS = std::unique_ptr<GDALDataset>(GDALDataset::Open(
    1667          24 :                 osTmpFilename.c_str(), GDAL_OF_RASTER, apszDrivers));
    1668          24 :             if (!poJP2KDS)
    1669             :             {
    1670           0 :                 VSIUnlink(osTmpFilename.c_str());
    1671           0 :                 CPLError(CE_Failure, CPLE_AppDefined,
    1672             :                          "Cannot open JPEG2000 blob");
    1673           0 :                 return CE_Failure;
    1674             :             }
    1675          24 :             if (poJP2KDS->GetRasterCount() != 1 ||
    1676          48 :                 poJP2KDS->GetRasterXSize() != nBlockXSize ||
    1677          24 :                 poJP2KDS->GetRasterYSize() != nBlockYSize)
    1678             :             {
    1679           0 :                 VSIUnlink(osTmpFilename.c_str());
    1680           0 :                 CPLError(CE_Failure, CPLE_AppDefined,
    1681             :                          "Inconsistent characteristics of JPEG2000 blob");
    1682           0 :                 return CE_Failure;
    1683             :             }
    1684             : 
    1685          24 :             auto imageDataAndBlock = GetImageData();
    1686          24 :             auto pImageData = imageDataAndBlock.first;
    1687          24 :             auto poBlock = imageDataAndBlock.second;
    1688             : 
    1689             :             const CPLErr eErr =
    1690             :                 pImageData
    1691          24 :                     ? poJP2KDS->GetRasterBand(1)->RasterIO(
    1692             :                           GF_Read, 0, 0, nBlockXSize, nBlockYSize, pImageData,
    1693             :                           nBlockXSize, nBlockYSize, eImageDT, 0, 0, nullptr)
    1694          24 :                     : CE_None;
    1695          24 :             VSIUnlink(osTmpFilename.c_str());
    1696          24 :             if (poBlock)
    1697           0 :                 poBlock->DropLock();
    1698             : 
    1699          24 :             if (eErr != CE_None)
    1700             :             {
    1701           0 :                 CPLError(CE_Failure, CPLE_AppDefined,
    1702             :                          "Cannot read JPEG2000 blob");
    1703           0 :                 return CE_Failure;
    1704             :             }
    1705             : 
    1706          24 :             break;
    1707             :         }
    1708             :     }
    1709             : 
    1710         117 :     if (m_bIsMask || m_poMaskBand)
    1711             :     {
    1712         114 :         auto maskDataAndBlock = GetMaskData();
    1713         114 :         auto pMaskData = maskDataAndBlock.first;
    1714         114 :         auto poBlock = maskDataAndBlock.second;
    1715             : 
    1716         114 :         if (pMaskData)
    1717             :         {
    1718          64 :             if (pabyMask)
    1719             :             {
    1720             :                 // Unpack 1-bit array
    1721      802865 :                 for (size_t i = 0; i < nPixels; ++i)
    1722             :                 {
    1723      802816 :                     static_cast<GByte *>(pMaskData)[i] =
    1724      802816 :                         (pabyMask[i / 8] & (0x80 >> (i & 7))) ? 255 : 0;
    1725             :                 }
    1726             :             }
    1727             :             else
    1728             :             {
    1729             :                 // No explicit mask in source block --> all valid
    1730          15 :                 memset(pMaskData, 255, nPixels);
    1731             :             }
    1732             :         }
    1733             : 
    1734         114 :         if (poBlock)
    1735         114 :             poBlock->DropLock();
    1736             :     }
    1737           3 :     else if (m_bHasNoData)
    1738             :     {
    1739           3 :         if (eImageDT == GDT_Byte)
    1740             :         {
    1741           0 :             SetNoDataFromMask<uint8_t>(pImage, pabyMask, nPixels, m_dfNoData);
    1742             :         }
    1743           3 :         else if (eImageDT == GDT_Int8)
    1744             :         {
    1745           0 :             SetNoDataFromMask<int8_t>(pImage, pabyMask, nPixels, m_dfNoData);
    1746             :         }
    1747           3 :         else if (eImageDT == GDT_UInt16)
    1748             :         {
    1749           0 :             SetNoDataFromMask<uint16_t>(pImage, pabyMask, nPixels, m_dfNoData);
    1750             :         }
    1751           3 :         else if (eImageDT == GDT_Int16)
    1752             :         {
    1753           0 :             SetNoDataFromMask<int16_t>(pImage, pabyMask, nPixels, m_dfNoData);
    1754             :         }
    1755           3 :         else if (eImageDT == GDT_UInt32)
    1756             :         {
    1757           0 :             SetNoDataFromMask<uint32_t>(pImage, pabyMask, nPixels, m_dfNoData);
    1758             :         }
    1759           3 :         else if (eImageDT == GDT_Int32)
    1760             :         {
    1761           0 :             SetNoDataFromMask<int32_t>(pImage, pabyMask, nPixels, m_dfNoData);
    1762             :         }
    1763           3 :         else if (eImageDT == GDT_Float32)
    1764             :         {
    1765           2 :             if (pabyMask)
    1766             :             {
    1767       32770 :                 for (size_t i = 0; i < nPixels; ++i)
    1768             :                 {
    1769       32768 :                     if (!(pabyMask[i / 8] & (0x80 >> (i & 7))))
    1770             :                     {
    1771       31968 :                         static_cast<float *>(pImage)[i] =
    1772       31968 :                             static_cast<float>(m_dfNoData);
    1773             :                     }
    1774             :                 }
    1775             :             }
    1776             :         }
    1777           1 :         else if (eImageDT == GDT_Float64)
    1778             :         {
    1779           1 :             if (pabyMask)
    1780             :             {
    1781       16385 :                 for (size_t i = 0; i < nPixels; ++i)
    1782             :                 {
    1783       16384 :                     if (!(pabyMask[i / 8] & (0x80 >> (i & 7))))
    1784             :                     {
    1785       15984 :                         static_cast<double *>(pImage)[i] = m_dfNoData;
    1786             :                     }
    1787             :                 }
    1788             :             }
    1789             :         }
    1790             :         else
    1791             :         {
    1792           0 :             CPLAssert(false);
    1793             :         }
    1794             :     }
    1795             : 
    1796             : #if 0
    1797             :     printf("Data:\n"); // ok
    1798             :     if (eDataType == GDT_Byte)
    1799             :     {
    1800             :         for (int y = 0; y < nBlockYSize; ++y)
    1801             :         {
    1802             :             for (int x = 0; x < nBlockXSize; ++x)
    1803             :             {
    1804             :                 printf("%d ", // ok
    1805             :                        static_cast<GByte *>(pImage)[y * nBlockXSize + x]);
    1806             :             }
    1807             :             printf("\n"); // ok
    1808             :         }
    1809             :     }
    1810             :     else if (eDataType == GDT_Int8)
    1811             :     {
    1812             :         for (int y = 0; y < nBlockYSize; ++y)
    1813             :         {
    1814             :             for (int x = 0; x < nBlockXSize; ++x)
    1815             :             {
    1816             :                 printf("%d ", // ok
    1817             :                        static_cast<int8_t *>(pImage)[y * nBlockXSize + x]);
    1818             :             }
    1819             :             printf("\n"); // ok
    1820             :         }
    1821             :     }
    1822             :     else if (eDataType == GDT_UInt16)
    1823             :     {
    1824             :         for (int y = 0; y < nBlockYSize; ++y)
    1825             :         {
    1826             :             for (int x = 0; x < nBlockXSize; ++x)
    1827             :             {
    1828             :                 printf("%d ", // ok
    1829             :                        static_cast<uint16_t *>(pImage)[y * nBlockXSize + x]);
    1830             :             }
    1831             :             printf("\n"); // ok
    1832             :         }
    1833             :     }
    1834             :     else if (eDataType == GDT_Int16)
    1835             :     {
    1836             :         for (int y = 0; y < nBlockYSize; ++y)
    1837             :         {
    1838             :             for (int x = 0; x < nBlockXSize; ++x)
    1839             :             {
    1840             :                 printf("%d ", // ok
    1841             :                        static_cast<int16_t *>(pImage)[y * nBlockXSize + x]);
    1842             :             }
    1843             :             printf("\n"); // ok
    1844             :         }
    1845             :     }
    1846             :     else if (eDataType == GDT_UInt32)
    1847             :     {
    1848             :         for (int y = 0; y < nBlockYSize; ++y)
    1849             :         {
    1850             :             for (int x = 0; x < nBlockXSize; ++x)
    1851             :             {
    1852             :                 printf("%d ", // ok
    1853             :                        static_cast<uint32_t *>(pImage)[y * nBlockXSize + x]);
    1854             :             }
    1855             :             printf("\n"); // ok
    1856             :         }
    1857             :     }
    1858             :     else if (eDataType == GDT_Int32)
    1859             :     {
    1860             :         for (int y = 0; y < nBlockYSize; ++y)
    1861             :         {
    1862             :             for (int x = 0; x < nBlockXSize; ++x)
    1863             :             {
    1864             :                 printf("%d ", // ok
    1865             :                        static_cast<int32_t *>(pImage)[y * nBlockXSize + x]);
    1866             :             }
    1867             :             printf("\n"); // ok
    1868             :         }
    1869             :     }
    1870             :     else if (eDataType == GDT_Float32)
    1871             :     {
    1872             :         for (int y = 0; y < nBlockYSize; ++y)
    1873             :         {
    1874             :             for (int x = 0; x < nBlockXSize; ++x)
    1875             :             {
    1876             :                 printf("%.8g ", // ok
    1877             :                        static_cast<float *>(pImage)[y * nBlockXSize + x]);
    1878             :             }
    1879             :             printf("\n"); // ok
    1880             :         }
    1881             :     }
    1882             :     else if (eDataType == GDT_Float64)
    1883             :     {
    1884             :         for (int y = 0; y < nBlockYSize; ++y)
    1885             :         {
    1886             :             for (int x = 0; x < nBlockXSize; ++x)
    1887             :             {
    1888             :                 printf("%.17g ", // ok
    1889             :                        static_cast<double *>(pImage)[y * nBlockXSize + x]);
    1890             :             }
    1891             :             printf("\n"); // ok
    1892             :         }
    1893             :     }
    1894             : #endif
    1895             : 
    1896             : #if 0
    1897             :     if (pabyMask)
    1898             :     {
    1899             :         printf("Mask:\n"); // ok
    1900             :         for (int y = 0; y < nBlockYSize; ++y)
    1901             :         {
    1902             :             for (int x = 0; x < nBlockXSize; ++x)
    1903             :             {
    1904             :                 printf("%d ", // ok
    1905             :                        (pabyMask[(y * nBlockXSize + x) / 8] &
    1906             :                         (0x80 >> ((y * nBlockXSize + x) & 7)))
    1907             :                            ? 1
    1908             :                            : 0);
    1909             :             }
    1910             :             printf("\n"); // ok
    1911             :         }
    1912             :     }
    1913             : #endif
    1914             : 
    1915         117 :     return CE_None;
    1916             : }
    1917             : 
    1918             : /************************************************************************/
    1919             : /*                         GetDefaultRAT()                              */
    1920             : /************************************************************************/
    1921             : 
    1922           3 : GDALRasterAttributeTable *GDALOpenFileGDBRasterBand::GetDefaultRAT()
    1923             : {
    1924           3 :     if (m_poRAT)
    1925           1 :         return m_poRAT.get();
    1926           2 :     if (poDS->GetRasterCount() > 1 || m_bIsMask)
    1927           0 :         return nullptr;
    1928           2 :     auto poGDS = cpl::down_cast<OGROpenFileGDBDataSource *>(poDS);
    1929             :     const std::string osVATTableName(
    1930           6 :         std::string("VAT_").append(poGDS->m_osRasterLayerName));
    1931             :     // Instantiate a new dataset, os that the RAT is standalone
    1932           4 :     auto poDSNew = std::make_unique<OGROpenFileGDBDataSource>();
    1933           4 :     GDALOpenInfo oOpenInfo(poGDS->m_osDirName.c_str(), GA_ReadOnly);
    1934           2 :     bool bRetryFileGDBUnused = false;
    1935           2 :     if (!poDSNew->Open(&oOpenInfo, bRetryFileGDBUnused))
    1936           0 :         return nullptr;
    1937           4 :     auto poVatLayer = poDSNew->BuildLayerFromName(osVATTableName.c_str());
    1938           2 :     if (!poVatLayer)
    1939           1 :         return nullptr;
    1940           2 :     m_poRAT = std::make_unique<GDALOpenFileGDBRasterAttributeTable>(
    1941           2 :         std::move(poDSNew), osVATTableName, std::move(poVatLayer));
    1942           1 :     return m_poRAT.get();
    1943             : }

Generated by: LCOV version 1.14