LCOV - code coverage report
Current view: top level - frmts/eeda - eedaidataset.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 491 651 75.4 %
Date: 2025-01-18 12:42:00 Functions: 24 26 92.3 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  Earth Engine Data API Images driver
       4             :  * Purpose:  Earth Engine Data API Images driver
       5             :  * Author:   Even Rouault, even dot rouault at spatialys.com
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2017-2018, Planet Labs
       9             :  *
      10             :  * SPDX-License-Identifier: MIT
      11             :  ****************************************************************************/
      12             : 
      13             : #include "gdal_priv.h"
      14             : #include "cpl_http.h"
      15             : #include "cpl_conv.h"
      16             : #include "ogrlibjsonutils.h"
      17             : #include "eeda.h"
      18             : 
      19             : #include <algorithm>
      20             : #include <vector>
      21             : #include <map>
      22             : #include <limits>
      23             : 
      24             : extern "C" void GDALRegister_EEDAI();
      25             : 
      26             : static const int DEFAULT_BLOCK_SIZE = 256;
      27             : 
      28             : const GUInt32 RETRY_PER_BAND = 1;
      29             : const GUInt32 RETRY_SPATIAL_SPLIT = 2;
      30             : 
      31             : // Eart engine server only allows up to 16 MB per request
      32             : const int SERVER_BYTE_LIMIT = 16 * 1024 * 1024;
      33             : const int SERVER_SIMUTANEOUS_BAND_LIMIT = 100;
      34             : const int SERVER_DIMENSION_LIMIT = 10000;
      35             : 
      36             : /************************************************************************/
      37             : /*                          GDALEEDAIDataset                            */
      38             : /************************************************************************/
      39             : 
      40             : class GDALEEDAIDataset final : public GDALEEDABaseDataset
      41             : {
      42             :     CPL_DISALLOW_COPY_ASSIGN(GDALEEDAIDataset)
      43             : 
      44             :     friend class GDALEEDAIRasterBand;
      45             : 
      46             :     int m_nBlockSize;
      47             :     CPLString m_osAsset{};
      48             :     CPLString m_osAssetName{};
      49             :     GDALEEDAIDataset *m_poParentDS;
      50             : #ifdef DEBUG_VERBOSE
      51             :     int m_iOvrLevel;
      52             : #endif
      53             :     CPLString m_osPixelEncoding{};
      54             :     bool m_bQueryMultipleBands;
      55             :     OGRSpatialReference m_oSRS{};
      56             :     double m_adfGeoTransform[6];
      57             :     std::vector<GDALEEDAIDataset *> m_apoOverviewDS{};
      58             : 
      59             :     GDALEEDAIDataset(GDALEEDAIDataset *poParentDS, int iOvrLevel);
      60             : 
      61             :     void
      62             :     SetMetadataFromProperties(json_object *poProperties,
      63             :                               const std::map<CPLString, int> &aoMapBandNames);
      64             : 
      65             :   public:
      66             :     GDALEEDAIDataset();
      67             :     virtual ~GDALEEDAIDataset();
      68             : 
      69             :     const OGRSpatialReference *GetSpatialRef() const override;
      70             :     virtual CPLErr GetGeoTransform(double *) override;
      71             : 
      72             :     virtual CPLErr IRasterIO(GDALRWFlag eRWFlag, int nXOff, int nYOff,
      73             :                              int nXSize, int nYSize, void *pData, int nBufXSize,
      74             :                              int nBufYSize, GDALDataType eBufType,
      75             :                              int nBandCount, BANDMAP_TYPE panBandMap,
      76             :                              GSpacing nPixelSpace, GSpacing nLineSpace,
      77             :                              GSpacing nBandSpace,
      78             :                              GDALRasterIOExtraArg *psExtraArg) override;
      79             : 
      80             :     bool ComputeQueryStrategy();
      81             : 
      82             :     bool Open(GDALOpenInfo *poOpenInfo);
      83             : };
      84             : 
      85             : /************************************************************************/
      86             : /*                        GDALEEDAIRasterBand                           */
      87             : /************************************************************************/
      88             : 
      89             : class GDALEEDAIRasterBand final : public GDALRasterBand
      90             : {
      91             :     CPL_DISALLOW_COPY_ASSIGN(GDALEEDAIRasterBand)
      92             : 
      93             :     friend class GDALEEDAIDataset;
      94             : 
      95             :     GDALColorInterp m_eInterp;
      96             : 
      97             :     bool DecodeNPYArray(const GByte *pabyData, int nDataLen,
      98             :                         bool bQueryAllBands, void *pDstBuffer, int nBlockXOff,
      99             :                         int nBlockYOff, int nXBlocks, int nYBlocks,
     100             :                         int nReqXSize, int nReqYSize) const;
     101             :     bool DecodeGDALDataset(const GByte *pabyData, int nDataLen,
     102             :                            bool bQueryAllBands, void *pDstBuffer,
     103             :                            int nBlockXOff, int nBlockYOff, int nXBlocks,
     104             :                            int nYBlocks, int nReqXSize, int nReqYSize);
     105             : 
     106             :     CPLErr GetBlocks(int nBlockXOff, int nBlockYOff, int nXBlocks, int nYBlocks,
     107             :                      bool bQueryAllBands, void *pBuffer);
     108             :     GUInt32 PrefetchBlocks(int nXOff, int nYOff, int nXSize, int nYSize,
     109             :                            int nBufXSize, int nBufYSize, bool bQueryAllBands);
     110             : 
     111             :   public:
     112             :     GDALEEDAIRasterBand(GDALEEDAIDataset *poDSIn, GDALDataType eDT);
     113             :     virtual ~GDALEEDAIRasterBand();
     114             : 
     115             :     virtual CPLErr IRasterIO(GDALRWFlag eRWFlag, int nXOff, int nYOff,
     116             :                              int nXSize, int nYSize, void *pData, int nBufXSize,
     117             :                              int nBufYSize, GDALDataType eBufType,
     118             :                              GSpacing nPixelSpace, GSpacing nLineSpace,
     119             :                              GDALRasterIOExtraArg *psExtraArg) CPL_OVERRIDE;
     120             : 
     121             :     virtual CPLErr IReadBlock(int, int, void *) CPL_OVERRIDE;
     122             :     virtual int GetOverviewCount() CPL_OVERRIDE;
     123             :     virtual GDALRasterBand *GetOverview(int) CPL_OVERRIDE;
     124             : 
     125           0 :     virtual CPLErr SetColorInterpretation(GDALColorInterp eInterp) CPL_OVERRIDE
     126             :     {
     127           0 :         m_eInterp = eInterp;
     128           0 :         return CE_None;
     129             :     }
     130             : 
     131           7 :     virtual GDALColorInterp GetColorInterpretation() CPL_OVERRIDE
     132             :     {
     133           7 :         return m_eInterp;
     134             :     }
     135             : };
     136             : 
     137             : /************************************************************************/
     138             : /*                         GDALEEDAIDataset()                           */
     139             : /************************************************************************/
     140             : 
     141           9 : GDALEEDAIDataset::GDALEEDAIDataset()
     142             :     : m_nBlockSize(DEFAULT_BLOCK_SIZE), m_poParentDS(nullptr),
     143             : #ifdef DEBUG_VERBOSE
     144             :       m_iOvrLevel(0),
     145             : #endif
     146           9 :       m_bQueryMultipleBands(false)
     147             : {
     148           9 :     m_oSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
     149           9 :     m_adfGeoTransform[0] = 0.0;
     150           9 :     m_adfGeoTransform[1] = 1.0;
     151           9 :     m_adfGeoTransform[2] = 0.0;
     152           9 :     m_adfGeoTransform[3] = 0.0;
     153           9 :     m_adfGeoTransform[4] = 0.0;
     154           9 :     m_adfGeoTransform[5] = 1.0;
     155           9 : }
     156             : 
     157             : /************************************************************************/
     158             : /*                         GDALEEDAIDataset()                           */
     159             : /************************************************************************/
     160             : 
     161          27 : GDALEEDAIDataset::GDALEEDAIDataset(GDALEEDAIDataset *poParentDS, int iOvrLevel)
     162          27 :     : m_nBlockSize(poParentDS->m_nBlockSize), m_osAsset(poParentDS->m_osAsset),
     163          27 :       m_osAssetName(poParentDS->m_osAssetName), m_poParentDS(poParentDS),
     164             : #ifdef DEBUG_VERBOSE
     165             :       m_iOvrLevel(iOvrLevel),
     166             : #endif
     167          27 :       m_osPixelEncoding(poParentDS->m_osPixelEncoding),
     168          27 :       m_bQueryMultipleBands(poParentDS->m_bQueryMultipleBands),
     169          27 :       m_oSRS(poParentDS->m_oSRS)
     170             : {
     171          27 :     m_osBaseURL = poParentDS->m_osBaseURL;
     172          27 :     nRasterXSize = m_poParentDS->nRasterXSize >> iOvrLevel;
     173          27 :     nRasterYSize = m_poParentDS->nRasterYSize >> iOvrLevel;
     174          27 :     m_adfGeoTransform[0] = m_poParentDS->m_adfGeoTransform[0];
     175          27 :     m_adfGeoTransform[1] = m_poParentDS->m_adfGeoTransform[1] *
     176          27 :                            m_poParentDS->nRasterXSize / nRasterXSize;
     177          27 :     m_adfGeoTransform[2] = m_poParentDS->m_adfGeoTransform[2];
     178          27 :     m_adfGeoTransform[3] = m_poParentDS->m_adfGeoTransform[3];
     179          27 :     m_adfGeoTransform[4] = m_poParentDS->m_adfGeoTransform[4];
     180          27 :     m_adfGeoTransform[5] = m_poParentDS->m_adfGeoTransform[5] *
     181          27 :                            m_poParentDS->nRasterYSize / nRasterYSize;
     182          27 : }
     183             : 
     184             : /************************************************************************/
     185             : /*                        ~GDALEEDAIDataset()                           */
     186             : /************************************************************************/
     187             : 
     188          72 : GDALEEDAIDataset::~GDALEEDAIDataset()
     189             : {
     190          63 :     for (size_t i = 0; i < m_apoOverviewDS.size(); i++)
     191             :     {
     192          27 :         delete m_apoOverviewDS[i];
     193             :     }
     194          72 : }
     195             : 
     196             : /************************************************************************/
     197             : /*                        GDALEEDAIRasterBand()                         */
     198             : /************************************************************************/
     199             : 
     200          63 : GDALEEDAIRasterBand::GDALEEDAIRasterBand(GDALEEDAIDataset *poDSIn,
     201          63 :                                          GDALDataType eDT)
     202          63 :     : m_eInterp(GCI_Undefined)
     203             : {
     204          63 :     eDataType = eDT;
     205          63 :     nBlockXSize = poDSIn->m_nBlockSize;
     206          63 :     nBlockYSize = poDSIn->m_nBlockSize;
     207          63 : }
     208             : 
     209             : /************************************************************************/
     210             : /*                       ~GDALEEDAIRasterBand()                         */
     211             : /************************************************************************/
     212             : 
     213         126 : GDALEEDAIRasterBand::~GDALEEDAIRasterBand()
     214             : {
     215         126 : }
     216             : 
     217             : /************************************************************************/
     218             : /*                           GetOverviewCount()                         */
     219             : /************************************************************************/
     220             : 
     221          14 : int GDALEEDAIRasterBand::GetOverviewCount()
     222             : {
     223          14 :     GDALEEDAIDataset *poGDS = reinterpret_cast<GDALEEDAIDataset *>(poDS);
     224          14 :     return static_cast<int>(poGDS->m_apoOverviewDS.size());
     225             : }
     226             : 
     227             : /************************************************************************/
     228             : /*                              GetOverview()                           */
     229             : /************************************************************************/
     230             : 
     231          12 : GDALRasterBand *GDALEEDAIRasterBand::GetOverview(int iIndex)
     232             : {
     233          12 :     GDALEEDAIDataset *poGDS = reinterpret_cast<GDALEEDAIDataset *>(poDS);
     234          12 :     if (iIndex >= 0 && iIndex < static_cast<int>(poGDS->m_apoOverviewDS.size()))
     235             :     {
     236          10 :         return poGDS->m_apoOverviewDS[iIndex]->GetRasterBand(nBand);
     237             :     }
     238           2 :     return nullptr;
     239             : }
     240             : 
     241             : /************************************************************************/
     242             : /*                            DecodeNPYArray()                          */
     243             : /************************************************************************/
     244             : 
     245           1 : bool GDALEEDAIRasterBand::DecodeNPYArray(const GByte *pabyData, int nDataLen,
     246             :                                          bool bQueryAllBands, void *pDstBuffer,
     247             :                                          int nBlockXOff, int nBlockYOff,
     248             :                                          int nXBlocks, int nYBlocks,
     249             :                                          int nReqXSize, int nReqYSize) const
     250             : {
     251           1 :     GDALEEDAIDataset *poGDS = reinterpret_cast<GDALEEDAIDataset *>(poDS);
     252             : 
     253             :     // See https://docs.scipy.org/doc/numpy-1.13.0/neps/npy-format.html
     254             :     // for description of NPY array serialization format
     255           1 :     if (nDataLen < 10)
     256             :     {
     257           0 :         CPLError(CE_Failure, CPLE_AppDefined, "Non NPY array returned");
     258           0 :         return false;
     259             :     }
     260             : 
     261           1 :     if (memcmp(pabyData, "\x93NUMPY", 6) != 0)
     262             :     {
     263           0 :         CPLError(CE_Failure, CPLE_AppDefined, "Non NPY array returned");
     264           0 :         return false;
     265             :     }
     266           1 :     const int nVersionMajor = pabyData[6];
     267           1 :     if (nVersionMajor != 1)
     268             :     {
     269           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     270             :                  "Only version 1 of NPY array supported. Here found %d",
     271             :                  nVersionMajor);
     272           0 :         return false;
     273             :     }
     274             :     // Ignore version minor
     275           1 :     const int nHeaderLen = pabyData[8] | (pabyData[9] << 8);
     276           1 :     if (nDataLen < 10 + nHeaderLen)
     277             :     {
     278           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     279             :                  "Corrupted NPY array returned: not enough bytes for header");
     280           0 :         return false;
     281             :     }
     282             : 
     283             : #ifdef DEBUG
     284           2 :     CPLString osDescr;
     285           1 :     osDescr.assign(reinterpret_cast<const char *>(pabyData) + 10, nHeaderLen);
     286             :     // Should be something like
     287             :     // {'descr': [('B2', '<u2'), ('B3', '<u2'), ('B4', '<u2'), ('B8', '<u2'),
     288             :     // ('QA10', '<u2')], 'fortran_order': False, 'shape': (256, 256), }
     289           1 :     CPLDebug("EEDAI", "NPY descr: %s", osDescr.c_str());
     290             :     // TODO: validate that the descr is the one expected
     291             : #endif
     292             : 
     293           1 :     int nTotalDataTypeSize = 0;
     294           3 :     for (int i = 1; i <= poGDS->GetRasterCount(); i++)
     295             :     {
     296           2 :         if (bQueryAllBands || i == nBand)
     297             :         {
     298           2 :             nTotalDataTypeSize += GDALGetDataTypeSizeBytes(
     299             :                 poGDS->GetRasterBand(i)->GetRasterDataType());
     300             :         }
     301             :     }
     302           1 :     int nDataSize = nTotalDataTypeSize * nReqXSize * nReqYSize;
     303           1 :     if (nDataLen < 10 + nHeaderLen + nDataSize)
     304             :     {
     305           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     306             :                  "Corrupted NPY array returned: not enough bytes for payload. "
     307             :                  "%d needed, only %d found",
     308           0 :                  10 + nHeaderLen + nDataSize, nDataLen);
     309           0 :         return false;
     310             :     }
     311           1 :     else if (nDataLen > 10 + nHeaderLen + nDataSize)
     312             :     {
     313           0 :         CPLError(CE_Warning, CPLE_AppDefined,
     314             :                  "Possibly corrupted NPY array returned: "
     315             :                  "expected bytes for payload. "
     316             :                  "%d needed, got %d found",
     317           0 :                  10 + nHeaderLen + nDataSize, nDataLen);
     318             :     }
     319             : 
     320           2 :     for (int iYBlock = 0; iYBlock < nYBlocks; iYBlock++)
     321             :     {
     322           1 :         int nBlockActualYSize = nBlockYSize;
     323           1 :         if ((iYBlock + nBlockYOff + 1) * nBlockYSize > nRasterYSize)
     324             :         {
     325           1 :             nBlockActualYSize =
     326           1 :                 nRasterYSize - (iYBlock + nBlockYOff) * nBlockYSize;
     327             :         }
     328             : 
     329           2 :         for (int iXBlock = 0; iXBlock < nXBlocks; iXBlock++)
     330             :         {
     331           1 :             int nBlockActualXSize = nBlockXSize;
     332           1 :             if ((iXBlock + nBlockXOff + 1) * nBlockXSize > nRasterXSize)
     333             :             {
     334           1 :                 nBlockActualXSize =
     335           1 :                     nRasterXSize - (iXBlock + nBlockXOff) * nBlockXSize;
     336             :             }
     337             : 
     338           1 :             int nOffsetBand =
     339           1 :                 10 + nHeaderLen +
     340           1 :                 (iYBlock * nBlockYSize * nReqXSize + iXBlock * nBlockXSize) *
     341             :                     nTotalDataTypeSize;
     342             : 
     343           3 :             for (int i = 1; i <= poGDS->GetRasterCount(); i++)
     344             :             {
     345           2 :                 GDALRasterBlock *poBlock = nullptr;
     346             :                 GByte *pabyDstBuffer;
     347           2 :                 if (i == nBand && pDstBuffer != nullptr)
     348           0 :                     pabyDstBuffer = reinterpret_cast<GByte *>(pDstBuffer);
     349           2 :                 else if (bQueryAllBands ||
     350           0 :                          (i == nBand && pDstBuffer == nullptr))
     351             :                 {
     352             :                     GDALEEDAIRasterBand *poOtherBand =
     353             :                         reinterpret_cast<GDALEEDAIRasterBand *>(
     354           2 :                             poGDS->GetRasterBand(i));
     355           4 :                     poBlock = poOtherBand->TryGetLockedBlockRef(
     356           2 :                         nBlockXOff + iXBlock, nBlockYOff + iYBlock);
     357           2 :                     if (poBlock != nullptr)
     358             :                     {
     359           0 :                         poBlock->DropLock();
     360           0 :                         continue;
     361             :                     }
     362           4 :                     poBlock = poOtherBand->GetLockedBlockRef(
     363           2 :                         nBlockXOff + iXBlock, nBlockYOff + iYBlock, TRUE);
     364           2 :                     if (poBlock == nullptr)
     365             :                     {
     366           0 :                         continue;
     367             :                     }
     368             :                     pabyDstBuffer =
     369           2 :                         reinterpret_cast<GByte *>(poBlock->GetDataRef());
     370             :                 }
     371             :                 else
     372             :                 {
     373           0 :                     continue;
     374             :                 }
     375             : 
     376           2 :                 GDALDataType eDT = poGDS->GetRasterBand(i)->GetRasterDataType();
     377           2 :                 const int nDTSize = GDALGetDataTypeSizeBytes(eDT);
     378             : 
     379          80 :                 for (int iLine = 0; iLine < nBlockActualYSize; iLine++)
     380             :                 {
     381          78 :                     GByte *pabyLineDest =
     382          78 :                         pabyDstBuffer + iLine * nDTSize * nBlockXSize;
     383          78 :                     GDALCopyWords(const_cast<GByte *>(pabyData) + nOffsetBand +
     384          78 :                                       iLine * nTotalDataTypeSize * nReqXSize,
     385             :                                   eDT, nTotalDataTypeSize, pabyLineDest, eDT,
     386             :                                   nDTSize, nBlockActualXSize);
     387             : #ifdef CPL_MSB
     388             :                     if (nDTSize > 1)
     389             :                     {
     390             :                         GDALSwapWords(pabyLineDest, nDTSize, nBlockActualXSize,
     391             :                                       nDTSize);
     392             :                     }
     393             : #endif
     394             :                 }
     395             : 
     396           2 :                 nOffsetBand += nDTSize;
     397             : 
     398           2 :                 if (poBlock)
     399           2 :                     poBlock->DropLock();
     400             :             }
     401             :         }
     402             :     }
     403           1 :     return true;
     404             : }
     405             : 
     406             : /************************************************************************/
     407             : /*                            DecodeGDALDataset()                         */
     408             : /************************************************************************/
     409             : 
     410           4 : bool GDALEEDAIRasterBand::DecodeGDALDataset(const GByte *pabyData, int nDataLen,
     411             :                                             bool bQueryAllBands,
     412             :                                             void *pDstBuffer, int nBlockXOff,
     413             :                                             int nBlockYOff, int nXBlocks,
     414             :                                             int nYBlocks, int nReqXSize,
     415             :                                             int nReqYSize)
     416             : {
     417           4 :     GDALEEDAIDataset *poGDS = reinterpret_cast<GDALEEDAIDataset *>(poDS);
     418             : 
     419           8 :     const CPLString osTmpFilename(VSIMemGenerateHiddenFilename("eedai"));
     420           4 :     VSIFCloseL(VSIFileFromMemBuffer(
     421             :         osTmpFilename, const_cast<GByte *>(pabyData), nDataLen, false));
     422           4 :     const char *const apszDrivers[] = {"PNG", "JPEG", "GTIFF", nullptr};
     423           4 :     GDALDataset *poTileDS = GDALDataset::FromHandle(GDALOpenEx(
     424             :         osTmpFilename, GDAL_OF_RASTER, apszDrivers, nullptr, nullptr));
     425           4 :     if (poTileDS == nullptr)
     426             :     {
     427           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     428             :                  "Cannot decode buffer returned by the "
     429             :                  "server as a PNG, JPEG or GeoTIFF image");
     430           0 :         VSIUnlink(osTmpFilename);
     431           0 :         return false;
     432             :     }
     433           4 :     if (poTileDS->GetRasterXSize() != nReqXSize ||
     434           8 :         poTileDS->GetRasterYSize() != nReqYSize ||
     435             :         // The server might return a RGBA image even if only 3 bands are
     436             :         // requested
     437           4 :         poTileDS->GetRasterCount() <
     438           4 :             (bQueryAllBands ? poGDS->GetRasterCount() : 1))
     439             :     {
     440           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     441             :                  "Bad dimensions/band count for image returned "
     442             :                  "by server: %dx%dx%d",
     443             :                  poTileDS->GetRasterXSize(), poTileDS->GetRasterYSize(),
     444             :                  poTileDS->GetRasterCount());
     445           0 :         delete poTileDS;
     446           0 :         VSIUnlink(osTmpFilename);
     447           0 :         return false;
     448             :     }
     449             : 
     450           8 :     for (int iYBlock = 0; iYBlock < nYBlocks; iYBlock++)
     451             :     {
     452           4 :         int nBlockActualYSize = nBlockYSize;
     453           4 :         if ((iYBlock + nBlockYOff + 1) * nBlockYSize > nRasterYSize)
     454             :         {
     455           0 :             nBlockActualYSize =
     456           0 :                 nRasterYSize - (iYBlock + nBlockYOff) * nBlockYSize;
     457             :         }
     458             : 
     459           8 :         for (int iXBlock = 0; iXBlock < nXBlocks; iXBlock++)
     460             :         {
     461           4 :             int nBlockActualXSize = nBlockXSize;
     462           4 :             if ((iXBlock + nBlockXOff + 1) * nBlockXSize > nRasterXSize)
     463             :             {
     464           0 :                 nBlockActualXSize =
     465           0 :                     nRasterXSize - (iXBlock + nBlockXOff) * nBlockXSize;
     466             :             }
     467             : 
     468          14 :             for (int i = 1; i <= poGDS->GetRasterCount(); i++)
     469             :             {
     470          10 :                 GDALRasterBlock *poBlock = nullptr;
     471             :                 GByte *pabyDstBuffer;
     472          10 :                 if (i == nBand && pDstBuffer != nullptr)
     473           0 :                     pabyDstBuffer = reinterpret_cast<GByte *>(pDstBuffer);
     474          10 :                 else if (bQueryAllBands ||
     475           0 :                          (i == nBand && pDstBuffer == nullptr))
     476             :                 {
     477             :                     GDALEEDAIRasterBand *poOtherBand =
     478             :                         reinterpret_cast<GDALEEDAIRasterBand *>(
     479          10 :                             poGDS->GetRasterBand(i));
     480          20 :                     poBlock = poOtherBand->TryGetLockedBlockRef(
     481          10 :                         nBlockXOff + iXBlock, nBlockYOff + iYBlock);
     482          10 :                     if (poBlock != nullptr)
     483             :                     {
     484           0 :                         poBlock->DropLock();
     485           0 :                         continue;
     486             :                     }
     487          20 :                     poBlock = poOtherBand->GetLockedBlockRef(
     488          10 :                         nBlockXOff + iXBlock, nBlockYOff + iYBlock, TRUE);
     489          10 :                     if (poBlock == nullptr)
     490             :                     {
     491           0 :                         continue;
     492             :                     }
     493             :                     pabyDstBuffer =
     494          10 :                         reinterpret_cast<GByte *>(poBlock->GetDataRef());
     495             :                 }
     496             :                 else
     497             :                 {
     498           0 :                     continue;
     499             :                 }
     500             : 
     501          10 :                 GDALDataType eDT = poGDS->GetRasterBand(i)->GetRasterDataType();
     502          10 :                 const int nDTSize = GDALGetDataTypeSizeBytes(eDT);
     503          10 :                 const int nTileBand = bQueryAllBands ? i : 1;
     504          10 :                 CPLErr eErr = poTileDS->GetRasterBand(nTileBand)->RasterIO(
     505          10 :                     GF_Read, iXBlock * nBlockXSize, iYBlock * nBlockYSize,
     506             :                     nBlockActualXSize, nBlockActualYSize, pabyDstBuffer,
     507             :                     nBlockActualXSize, nBlockActualYSize, eDT, nDTSize,
     508          10 :                     static_cast<GSpacing>(nDTSize) * nBlockXSize, nullptr);
     509             : 
     510          10 :                 if (poBlock)
     511          10 :                     poBlock->DropLock();
     512          10 :                 if (eErr != CE_None)
     513             :                 {
     514           0 :                     delete poTileDS;
     515           0 :                     VSIUnlink(osTmpFilename);
     516           0 :                     return false;
     517             :                 }
     518             :             }
     519             :         }
     520             :     }
     521             : 
     522           4 :     delete poTileDS;
     523           4 :     VSIUnlink(osTmpFilename);
     524           4 :     return true;
     525             : }
     526             : 
     527           5 : CPLErr GDALEEDAIRasterBand::GetBlocks(int nBlockXOff, int nBlockYOff,
     528             :                                       int nXBlocks, int nYBlocks,
     529             :                                       bool bQueryAllBands, void *pBuffer)
     530             : {
     531           5 :     GDALEEDAIDataset *poGDS = reinterpret_cast<GDALEEDAIDataset *>(poDS);
     532             : 
     533             :     // Build request content
     534           5 :     json_object *poReq = json_object_new_object();
     535           5 :     json_object_object_add(poReq, "fileFormat",
     536             :                            json_object_new_string(poGDS->m_osPixelEncoding));
     537           5 :     json_object *poBands = json_object_new_array();
     538          17 :     for (int i = 1; i <= poGDS->GetRasterCount(); i++)
     539             :     {
     540          12 :         if (bQueryAllBands || i == nBand)
     541             :         {
     542          12 :             json_object_array_add(
     543             :                 poBands, json_object_new_string(
     544          12 :                              poGDS->GetRasterBand(i)->GetDescription()));
     545             :         }
     546             :     }
     547           5 :     json_object_object_add(poReq, "bandIds", poBands);
     548             : 
     549           5 :     int nReqXSize = nBlockXSize * nXBlocks;
     550           5 :     if ((nBlockXOff + nXBlocks) * nBlockXSize > nRasterXSize)
     551           1 :         nReqXSize = nRasterXSize - nBlockXOff * nBlockXSize;
     552           5 :     int nReqYSize = nBlockYSize * nYBlocks;
     553           5 :     if ((nBlockYOff + nYBlocks) * nBlockYSize > nRasterYSize)
     554           1 :         nReqYSize = nRasterYSize - nBlockYOff * nBlockYSize;
     555           5 :     const double dfX0 = poGDS->m_adfGeoTransform[0] +
     556           5 :                         nBlockXOff * nBlockXSize * poGDS->m_adfGeoTransform[1];
     557           5 :     const double dfY0 = poGDS->m_adfGeoTransform[3] +
     558           5 :                         nBlockYOff * nBlockYSize * poGDS->m_adfGeoTransform[5];
     559             : #ifdef DEBUG_VERBOSE
     560             :     CPLDebug("EEDAI",
     561             :              "nBlockYOff=%d nBlockYOff=%d "
     562             :              "nXBlocks=%d nYBlocks=%d nReqXSize=%d nReqYSize=%d",
     563             :              nBlockYOff, nBlockYOff, nXBlocks, nYBlocks, nReqXSize, nReqYSize);
     564             : #endif
     565             : 
     566           5 :     json_object *poPixelGrid = json_object_new_object();
     567             : 
     568           5 :     json_object *poAffineTransform = json_object_new_object();
     569           5 :     json_object_object_add(
     570             :         poAffineTransform, "translateX",
     571             :         json_object_new_double_with_significant_figures(dfX0, 18));
     572           5 :     json_object_object_add(
     573             :         poAffineTransform, "translateY",
     574             :         json_object_new_double_with_significant_figures(dfY0, 18));
     575           5 :     json_object_object_add(poAffineTransform, "scaleX",
     576             :                            json_object_new_double_with_significant_figures(
     577             :                                poGDS->m_adfGeoTransform[1], 18));
     578           5 :     json_object_object_add(poAffineTransform, "scaleY",
     579             :                            json_object_new_double_with_significant_figures(
     580             :                                poGDS->m_adfGeoTransform[5], 18));
     581           5 :     json_object_object_add(
     582             :         poAffineTransform, "shearX",
     583             :         json_object_new_double_with_significant_figures(0.0, 18));
     584           5 :     json_object_object_add(
     585             :         poAffineTransform, "shearY",
     586             :         json_object_new_double_with_significant_figures(0.0, 18));
     587           5 :     json_object_object_add(poPixelGrid, "affineTransform", poAffineTransform);
     588             : 
     589           5 :     json_object *poDimensions = json_object_new_object();
     590           5 :     json_object_object_add(poDimensions, "width",
     591             :                            json_object_new_int(nReqXSize));
     592           5 :     json_object_object_add(poDimensions, "height",
     593             :                            json_object_new_int(nReqYSize));
     594           5 :     json_object_object_add(poPixelGrid, "dimensions", poDimensions);
     595           5 :     json_object_object_add(poReq, "grid", poPixelGrid);
     596             : 
     597          10 :     CPLString osPostContent = json_object_get_string(poReq);
     598           5 :     json_object_put(poReq);
     599             : 
     600             :     // Issue request
     601           5 :     char **papszOptions = (poGDS->m_poParentDS)
     602           5 :                               ? poGDS->m_poParentDS->GetBaseHTTPOptions()
     603           4 :                               : poGDS->GetBaseHTTPOptions();
     604           5 :     papszOptions = CSLSetNameValue(papszOptions, "CUSTOMREQUEST", "POST");
     605          10 :     CPLString osHeaders = CSLFetchNameValueDef(papszOptions, "HEADERS", "");
     606           5 :     if (!osHeaders.empty())
     607           5 :         osHeaders += "\r\n";
     608           5 :     osHeaders += "Content-Type: application/json";
     609           5 :     papszOptions = CSLSetNameValue(papszOptions, "HEADERS", osHeaders);
     610           5 :     papszOptions = CSLSetNameValue(papszOptions, "POSTFIELDS", osPostContent);
     611           5 :     CPLHTTPResult *psResult = EEDAHTTPFetch(
     612          10 :         (poGDS->m_osBaseURL + poGDS->m_osAssetName + ":getPixels").c_str(),
     613             :         papszOptions);
     614           5 :     CSLDestroy(papszOptions);
     615           5 :     if (psResult == nullptr)
     616           0 :         return CE_Failure;
     617             : 
     618           5 :     if (psResult->pszErrBuf != nullptr)
     619             :     {
     620           0 :         if (psResult->pabyData)
     621             :         {
     622           0 :             CPLError(CE_Failure, CPLE_AppDefined, "%s: %s", psResult->pszErrBuf,
     623           0 :                      reinterpret_cast<const char *>(psResult->pabyData));
     624             :         }
     625             :         else
     626             :         {
     627           0 :             CPLError(CE_Failure, CPLE_AppDefined, "%s", psResult->pszErrBuf);
     628             :         }
     629           0 :         CPLHTTPDestroyResult(psResult);
     630           0 :         return CE_Failure;
     631             :     }
     632             : 
     633           5 :     if (psResult->pabyData == nullptr)
     634             :     {
     635           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     636             :                  "Empty content returned by server");
     637           0 :         CPLHTTPDestroyResult(psResult);
     638           0 :         return CE_Failure;
     639             :     }
     640             : #ifdef DEBUG_VERBOSE
     641             :     CPLDebug("EEADI", "Result: %s (%d bytes)",
     642             :              reinterpret_cast<const char *>(psResult->pabyData),
     643             :              psResult->nDataLen);
     644             : #endif
     645             : 
     646           5 :     if (EQUAL(poGDS->m_osPixelEncoding, "NPY"))
     647             :     {
     648           1 :         if (!DecodeNPYArray(psResult->pabyData, psResult->nDataLen,
     649             :                             bQueryAllBands, pBuffer, nBlockXOff, nBlockYOff,
     650             :                             nXBlocks, nYBlocks, nReqXSize, nReqYSize))
     651             :         {
     652           0 :             CPLHTTPDestroyResult(psResult);
     653           0 :             return CE_Failure;
     654             :         }
     655             :     }
     656             :     else
     657             :     {
     658           4 :         if (!DecodeGDALDataset(psResult->pabyData, psResult->nDataLen,
     659             :                                bQueryAllBands, pBuffer, nBlockXOff, nBlockYOff,
     660             :                                nXBlocks, nYBlocks, nReqXSize, nReqYSize))
     661             :         {
     662           0 :             CPLHTTPDestroyResult(psResult);
     663           0 :             return CE_Failure;
     664             :         }
     665             :     }
     666             : 
     667           5 :     CPLHTTPDestroyResult(psResult);
     668             : 
     669           5 :     return CE_None;
     670             : }
     671             : 
     672             : /************************************************************************/
     673             : /*                             IReadBlock()                             */
     674             : /************************************************************************/
     675             : 
     676           0 : CPLErr GDALEEDAIRasterBand::IReadBlock(int nBlockXOff, int nBlockYOff,
     677             :                                        void *pBuffer)
     678             : {
     679           0 :     GDALEEDAIDataset *poGDS = reinterpret_cast<GDALEEDAIDataset *>(poDS);
     680             : #ifdef DEBUG_VERBOSE
     681             :     CPLDebug("EEDAI", "ReadBlock x=%d y=%d band=%d level=%d", nBlockXOff,
     682             :              nBlockYOff, nBand, poGDS->m_iOvrLevel);
     683             : #endif
     684             : 
     685           0 :     return GetBlocks(nBlockXOff, nBlockYOff, 1, 1, poGDS->m_bQueryMultipleBands,
     686           0 :                      pBuffer);
     687             : }
     688             : 
     689             : /************************************************************************/
     690             : /*                          PrefetchBlocks()                            */
     691             : /************************************************************************/
     692             : 
     693             : // Return or'ed flags among 0, RETRY_PER_BAND, RETRY_SPATIAL_SPLIT if the user
     694             : // should try to split the request in smaller chunks
     695             : 
     696          17 : GUInt32 GDALEEDAIRasterBand::PrefetchBlocks(int nXOff, int nYOff, int nXSize,
     697             :                                             int nYSize, int nBufXSize,
     698             :                                             int nBufYSize, bool bQueryAllBands)
     699             : {
     700          17 :     CPL_IGNORE_RET_VAL(nBufXSize);
     701          17 :     CPL_IGNORE_RET_VAL(nBufYSize);
     702             : 
     703          17 :     GDALEEDAIDataset *poGDS = reinterpret_cast<GDALEEDAIDataset *>(poDS);
     704          17 :     int nBlockXOff = nXOff / nBlockXSize;
     705          17 :     int nBlockYOff = nYOff / nBlockYSize;
     706          17 :     int nXBlocks = (nXOff + nXSize - 1) / nBlockXSize - nBlockXOff + 1;
     707          17 :     int nYBlocks = (nYOff + nYSize - 1) / nBlockYSize - nBlockYOff + 1;
     708             : 
     709          17 :     const int nThisDTSize = GDALGetDataTypeSizeBytes(GetRasterDataType());
     710          17 :     int nTotalDataTypeSize = 0;
     711          17 :     int nQueriedBands = 0;
     712          64 :     for (int i = 1; i <= poGDS->GetRasterCount(); i++)
     713             :     {
     714          47 :         if (bQueryAllBands || i == nBand)
     715             :         {
     716          47 :             nQueriedBands++;
     717          47 :             nTotalDataTypeSize += GDALGetDataTypeSizeBytes(
     718             :                 poGDS->GetRasterBand(i)->GetRasterDataType());
     719             :         }
     720             :     }
     721             : 
     722             :     // Check the number of already cached blocks, and remove fully
     723             :     // cached lines at the top of the area of interest from the queried
     724             :     // blocks
     725          17 :     int nBlocksCached = 0;
     726          17 :     int nBlocksCachedForThisBand = 0;
     727          17 :     bool bAllLineCached = true;
     728          34 :     for (int iYBlock = 0; iYBlock < nYBlocks;)
     729             :     {
     730          34 :         for (int iXBlock = 0; iXBlock < nXBlocks; iXBlock++)
     731             :         {
     732          64 :             for (int i = 1; i <= poGDS->GetRasterCount(); i++)
     733             :             {
     734          47 :                 GDALRasterBlock *poBlock = nullptr;
     735          47 :                 if (bQueryAllBands || i == nBand)
     736             :                 {
     737             :                     GDALEEDAIRasterBand *poOtherBand =
     738             :                         reinterpret_cast<GDALEEDAIRasterBand *>(
     739          47 :                             poGDS->GetRasterBand(i));
     740          94 :                     poBlock = poOtherBand->TryGetLockedBlockRef(
     741          47 :                         nBlockXOff + iXBlock, nBlockYOff + iYBlock);
     742          47 :                     if (poBlock != nullptr)
     743             :                     {
     744          35 :                         nBlocksCached++;
     745          35 :                         if (i == nBand)
     746          12 :                             nBlocksCachedForThisBand++;
     747          35 :                         poBlock->DropLock();
     748          35 :                         continue;
     749             :                     }
     750             :                     else
     751             :                     {
     752          12 :                         bAllLineCached = false;
     753             :                     }
     754             :                 }
     755             :             }
     756             :         }
     757             : 
     758          17 :         if (bAllLineCached)
     759             :         {
     760          12 :             nBlocksCached -= nXBlocks * nQueriedBands;
     761          12 :             nBlocksCachedForThisBand -= nXBlocks;
     762          12 :             nBlockYOff++;
     763          12 :             nYBlocks--;
     764             :         }
     765             :         else
     766             :         {
     767           5 :             iYBlock++;
     768             :         }
     769             :     }
     770             : 
     771          17 :     if (nXBlocks > 0 && nYBlocks > 0)
     772             :     {
     773           5 :         bool bMustReturn = false;
     774           5 :         GUInt32 nRetryFlags = 0;
     775             : 
     776             :         // Get the blocks if the number of already cached blocks is lesser
     777             :         // than 25% of the to be queried blocks
     778           5 :         if (nBlocksCached > (nQueriedBands * nXBlocks * nYBlocks) / 4)
     779             :         {
     780           0 :             if (nBlocksCachedForThisBand <= (nXBlocks * nYBlocks) / 4)
     781             :             {
     782           0 :                 nRetryFlags |= RETRY_PER_BAND;
     783             :             }
     784             :             else
     785             :             {
     786           0 :                 bMustReturn = true;
     787             :             }
     788             :         }
     789             : 
     790             :         // Don't request too many pixels in one dimension
     791           5 :         if (nXBlocks * nBlockXSize > SERVER_DIMENSION_LIMIT ||
     792           5 :             nYBlocks * nBlockYSize > SERVER_DIMENSION_LIMIT)
     793             :         {
     794           0 :             bMustReturn = true;
     795           0 :             nRetryFlags |= RETRY_SPATIAL_SPLIT;
     796             :         }
     797             : 
     798             :         // Make sure that we have enough cache (with a margin of 50%)
     799             :         // and the number of queried pixels isn't too big w.r.t server
     800             :         // limit
     801           5 :         const GIntBig nUncompressedSize = static_cast<GIntBig>(nXBlocks) *
     802           5 :                                           nYBlocks * nBlockXSize * nBlockYSize *
     803           5 :                                           nTotalDataTypeSize;
     804           5 :         const GIntBig nCacheMax = GDALGetCacheMax64() / 2;
     805           5 :         if (nUncompressedSize > nCacheMax ||
     806             :             nUncompressedSize > SERVER_BYTE_LIMIT)
     807             :         {
     808           0 :             if (bQueryAllBands && poGDS->GetRasterCount() > 1)
     809             :             {
     810           0 :                 const GIntBig nUncompressedSizeThisBand =
     811           0 :                     static_cast<GIntBig>(nXBlocks) * nYBlocks * nBlockXSize *
     812           0 :                     nBlockYSize * nThisDTSize;
     813           0 :                 if (nUncompressedSizeThisBand <= SERVER_BYTE_LIMIT &&
     814             :                     nUncompressedSizeThisBand <= nCacheMax)
     815             :                 {
     816           0 :                     nRetryFlags |= RETRY_PER_BAND;
     817             :                 }
     818             :             }
     819           0 :             if (nXBlocks > 1 || nYBlocks > 1)
     820             :             {
     821           0 :                 nRetryFlags |= RETRY_SPATIAL_SPLIT;
     822             :             }
     823           0 :             return nRetryFlags;
     824             :         }
     825           5 :         if (bMustReturn)
     826           0 :             return nRetryFlags;
     827             : 
     828           5 :         GetBlocks(nBlockXOff, nBlockYOff, nXBlocks, nYBlocks, bQueryAllBands,
     829             :                   nullptr);
     830             :     }
     831             : 
     832          17 :     return 0;
     833             : }
     834             : 
     835             : /************************************************************************/
     836             : /*                              IRasterIO()                             */
     837             : /************************************************************************/
     838             : 
     839          15 : CPLErr GDALEEDAIRasterBand::IRasterIO(GDALRWFlag eRWFlag, int nXOff, int nYOff,
     840             :                                       int nXSize, int nYSize, void *pData,
     841             :                                       int nBufXSize, int nBufYSize,
     842             :                                       GDALDataType eBufType,
     843             :                                       GSpacing nPixelSpace, GSpacing nLineSpace,
     844             :                                       GDALRasterIOExtraArg *psExtraArg)
     845             : 
     846             : {
     847             : 
     848             :     /* ==================================================================== */
     849             :     /*      Do we have overviews that would be appropriate to satisfy       */
     850             :     /*      this request?                                                   */
     851             :     /* ==================================================================== */
     852          15 :     if ((nBufXSize < nXSize || nBufYSize < nYSize) && GetOverviewCount() > 0 &&
     853             :         eRWFlag == GF_Read)
     854             :     {
     855             :         GDALRasterIOExtraArg sExtraArg;
     856           1 :         GDALCopyRasterIOExtraArg(&sExtraArg, psExtraArg);
     857             : 
     858             :         const int nOverview =
     859           1 :             GDALBandGetBestOverviewLevel2(this, nXOff, nYOff, nXSize, nYSize,
     860             :                                           nBufXSize, nBufYSize, &sExtraArg);
     861           1 :         if (nOverview >= 0)
     862             :         {
     863           1 :             GDALRasterBand *poOverviewBand = GetOverview(nOverview);
     864           1 :             if (poOverviewBand == nullptr)
     865           1 :                 return CE_Failure;
     866             : 
     867           1 :             return poOverviewBand->RasterIO(
     868             :                 eRWFlag, nXOff, nYOff, nXSize, nYSize, pData, nBufXSize,
     869           1 :                 nBufYSize, eBufType, nPixelSpace, nLineSpace, &sExtraArg);
     870             :         }
     871             :     }
     872             : 
     873          14 :     GDALEEDAIDataset *poGDS = reinterpret_cast<GDALEEDAIDataset *>(poDS);
     874             :     GUInt32 nRetryFlags =
     875          28 :         PrefetchBlocks(nXOff, nYOff, nXSize, nYSize, nBufXSize, nBufYSize,
     876          14 :                        poGDS->m_bQueryMultipleBands);
     877          14 :     if ((nRetryFlags & RETRY_SPATIAL_SPLIT) && nXSize == nBufXSize &&
     878           0 :         nYSize == nBufYSize && nYSize > nBlockYSize)
     879             :     {
     880             :         GDALRasterIOExtraArg sExtraArg;
     881           0 :         INIT_RASTERIO_EXTRA_ARG(sExtraArg);
     882             : 
     883             :         int nHalf =
     884           0 :             std::max(nBlockYSize, ((nYSize / 2) / nBlockYSize) * nBlockYSize);
     885             :         CPLErr eErr =
     886           0 :             IRasterIO(eRWFlag, nXOff, nYOff, nXSize, nHalf, pData, nXSize,
     887             :                       nHalf, eBufType, nPixelSpace, nLineSpace, &sExtraArg);
     888           0 :         if (eErr == CE_None)
     889             :         {
     890           0 :             eErr = IRasterIO(
     891             :                 eRWFlag, nXOff, nYOff + nHalf, nXSize, nYSize - nHalf,
     892           0 :                 static_cast<GByte *>(pData) + nHalf * nLineSpace, nXSize,
     893             :                 nYSize - nHalf, eBufType, nPixelSpace, nLineSpace, &sExtraArg);
     894             :         }
     895           0 :         return eErr;
     896             :     }
     897          14 :     else if ((nRetryFlags & RETRY_SPATIAL_SPLIT) && nXSize == nBufXSize &&
     898           0 :              nYSize == nBufYSize && nXSize > nBlockXSize)
     899             :     {
     900             :         GDALRasterIOExtraArg sExtraArg;
     901           0 :         INIT_RASTERIO_EXTRA_ARG(sExtraArg);
     902             : 
     903             :         int nHalf =
     904           0 :             std::max(nBlockXSize, ((nXSize / 2) / nBlockXSize) * nBlockXSize);
     905             :         CPLErr eErr =
     906           0 :             IRasterIO(eRWFlag, nXOff, nYOff, nHalf, nYSize, pData, nHalf,
     907             :                       nYSize, eBufType, nPixelSpace, nLineSpace, &sExtraArg);
     908           0 :         if (eErr == CE_None)
     909             :         {
     910             :             eErr =
     911           0 :                 IRasterIO(eRWFlag, nXOff + nHalf, nYOff, nXSize - nHalf, nYSize,
     912           0 :                           static_cast<GByte *>(pData) + nHalf * nPixelSpace,
     913             :                           nXSize - nHalf, nYSize, eBufType, nPixelSpace,
     914             :                           nLineSpace, &sExtraArg);
     915             :         }
     916           0 :         return eErr;
     917             :     }
     918          14 :     else if ((nRetryFlags & RETRY_PER_BAND) && poGDS->m_bQueryMultipleBands &&
     919           0 :              poGDS->nBands > 1)
     920             :     {
     921           0 :         CPL_IGNORE_RET_VAL(PrefetchBlocks(nXOff, nYOff, nXSize, nYSize,
     922             :                                           nBufXSize, nBufYSize, false));
     923             :     }
     924             : 
     925          14 :     return GDALRasterBand::IRasterIO(eRWFlag, nXOff, nYOff, nXSize, nYSize,
     926             :                                      pData, nBufXSize, nBufYSize, eBufType,
     927          14 :                                      nPixelSpace, nLineSpace, psExtraArg);
     928             : }
     929             : 
     930             : /************************************************************************/
     931             : /*                              IRasterIO()                             */
     932             : /************************************************************************/
     933             : 
     934           4 : CPLErr GDALEEDAIDataset::IRasterIO(
     935             :     GDALRWFlag eRWFlag, int nXOff, int nYOff, int nXSize, int nYSize,
     936             :     void *pData, int nBufXSize, int nBufYSize, GDALDataType eBufType,
     937             :     int nBandCount, BANDMAP_TYPE panBandMap, GSpacing nPixelSpace,
     938             :     GSpacing nLineSpace, GSpacing nBandSpace, GDALRasterIOExtraArg *psExtraArg)
     939             : {
     940             : 
     941             :     /* ==================================================================== */
     942             :     /*      Do we have overviews that would be appropriate to satisfy       */
     943             :     /*      this request?                                                   */
     944             :     /* ==================================================================== */
     945           3 :     if ((nBufXSize < nXSize || nBufYSize < nYSize) &&
     946           7 :         GetRasterBand(1)->GetOverviewCount() > 0 && eRWFlag == GF_Read)
     947             :     {
     948             :         GDALRasterIOExtraArg sExtraArg;
     949           1 :         GDALCopyRasterIOExtraArg(&sExtraArg, psExtraArg);
     950             : 
     951           1 :         const int nOverview = GDALBandGetBestOverviewLevel2(
     952             :             GetRasterBand(1), nXOff, nYOff, nXSize, nYSize, nBufXSize,
     953             :             nBufYSize, &sExtraArg);
     954           1 :         if (nOverview >= 0)
     955             :         {
     956             :             GDALRasterBand *poOverviewBand =
     957           1 :                 GetRasterBand(1)->GetOverview(nOverview);
     958           2 :             if (poOverviewBand == nullptr ||
     959           1 :                 poOverviewBand->GetDataset() == nullptr)
     960             :             {
     961           1 :                 return CE_Failure;
     962             :             }
     963             : 
     964           1 :             return poOverviewBand->GetDataset()->RasterIO(
     965             :                 eRWFlag, nXOff, nYOff, nXSize, nYSize, pData, nBufXSize,
     966             :                 nBufYSize, eBufType, nBandCount, panBandMap, nPixelSpace,
     967           1 :                 nLineSpace, nBandSpace, &sExtraArg);
     968             :         }
     969             :     }
     970             : 
     971             :     GDALEEDAIRasterBand *poBand =
     972           3 :         cpl::down_cast<GDALEEDAIRasterBand *>(GetRasterBand(1));
     973             : 
     974             :     GUInt32 nRetryFlags =
     975           6 :         poBand->PrefetchBlocks(nXOff, nYOff, nXSize, nYSize, nBufXSize,
     976           3 :                                nBufYSize, m_bQueryMultipleBands);
     977             :     int nBlockXSize, nBlockYSize;
     978           3 :     poBand->GetBlockSize(&nBlockXSize, &nBlockYSize);
     979           3 :     if ((nRetryFlags & RETRY_SPATIAL_SPLIT) && nXSize == nBufXSize &&
     980           0 :         nYSize == nBufYSize && nYSize > nBlockYSize)
     981             :     {
     982             :         GDALRasterIOExtraArg sExtraArg;
     983           0 :         INIT_RASTERIO_EXTRA_ARG(sExtraArg);
     984             : 
     985             :         int nHalf =
     986           0 :             std::max(nBlockYSize, ((nYSize / 2) / nBlockYSize) * nBlockYSize);
     987             :         CPLErr eErr =
     988           0 :             IRasterIO(eRWFlag, nXOff, nYOff, nXSize, nHalf, pData, nXSize,
     989             :                       nHalf, eBufType, nBandCount, panBandMap, nPixelSpace,
     990             :                       nLineSpace, nBandSpace, &sExtraArg);
     991           0 :         if (eErr == CE_None)
     992             :         {
     993           0 :             eErr = IRasterIO(
     994             :                 eRWFlag, nXOff, nYOff + nHalf, nXSize, nYSize - nHalf,
     995           0 :                 static_cast<GByte *>(pData) + nHalf * nLineSpace, nXSize,
     996             :                 nYSize - nHalf, eBufType, nBandCount, panBandMap, nPixelSpace,
     997             :                 nLineSpace, nBandSpace, &sExtraArg);
     998             :         }
     999           0 :         return eErr;
    1000             :     }
    1001           3 :     else if ((nRetryFlags & RETRY_SPATIAL_SPLIT) && nXSize == nBufXSize &&
    1002           0 :              nYSize == nBufYSize && nXSize > nBlockXSize)
    1003             :     {
    1004             :         GDALRasterIOExtraArg sExtraArg;
    1005           0 :         INIT_RASTERIO_EXTRA_ARG(sExtraArg);
    1006             : 
    1007             :         int nHalf =
    1008           0 :             std::max(nBlockXSize, ((nXSize / 2) / nBlockXSize) * nBlockXSize);
    1009             :         CPLErr eErr =
    1010           0 :             IRasterIO(eRWFlag, nXOff, nYOff, nHalf, nYSize, pData, nHalf,
    1011             :                       nYSize, eBufType, nBandCount, panBandMap, nPixelSpace,
    1012             :                       nLineSpace, nBandSpace, &sExtraArg);
    1013           0 :         if (eErr == CE_None)
    1014             :         {
    1015           0 :             eErr = IRasterIO(
    1016             :                 eRWFlag, nXOff + nHalf, nYOff, nXSize - nHalf, nYSize,
    1017           0 :                 static_cast<GByte *>(pData) + nHalf * nPixelSpace,
    1018             :                 nXSize - nHalf, nYSize, eBufType, nBandCount, panBandMap,
    1019             :                 nPixelSpace, nLineSpace, nBandSpace, &sExtraArg);
    1020             :         }
    1021           0 :         return eErr;
    1022             :     }
    1023           3 :     else if ((nRetryFlags & RETRY_PER_BAND) && m_bQueryMultipleBands &&
    1024           0 :              nBands > 1)
    1025             :     {
    1026           0 :         for (int iBand = 1; iBand <= nBands; iBand++)
    1027             :         {
    1028             :             poBand =
    1029           0 :                 cpl::down_cast<GDALEEDAIRasterBand *>(GetRasterBand(iBand));
    1030           0 :             CPL_IGNORE_RET_VAL(poBand->PrefetchBlocks(
    1031             :                 nXOff, nYOff, nXSize, nYSize, nBufXSize, nBufYSize, false));
    1032             :         }
    1033             :     }
    1034             : 
    1035           3 :     return GDALDataset::IRasterIO(eRWFlag, nXOff, nYOff, nXSize, nYSize, pData,
    1036             :                                   nBufXSize, nBufYSize, eBufType, nBandCount,
    1037             :                                   panBandMap, nPixelSpace, nLineSpace,
    1038           3 :                                   nBandSpace, psExtraArg);
    1039             : }
    1040             : 
    1041             : /************************************************************************/
    1042             : /*                        ComputeQueryStrategy()                        */
    1043             : /************************************************************************/
    1044             : 
    1045          35 : bool GDALEEDAIDataset::ComputeQueryStrategy()
    1046             : {
    1047          35 :     m_bQueryMultipleBands = true;
    1048          35 :     m_osPixelEncoding.toupper();
    1049             : 
    1050          35 :     bool bHeterogeneousDataTypes = false;
    1051          35 :     if (nBands >= 2)
    1052             :     {
    1053          24 :         GDALDataType eDTFirstBand = GetRasterBand(1)->GetRasterDataType();
    1054          52 :         for (int i = 2; i <= nBands; i++)
    1055             :         {
    1056          28 :             if (GetRasterBand(i)->GetRasterDataType() != eDTFirstBand)
    1057             :             {
    1058           0 :                 bHeterogeneousDataTypes = true;
    1059           0 :                 break;
    1060             :             }
    1061             :         }
    1062             :     }
    1063             : 
    1064          35 :     if (EQUAL(m_osPixelEncoding, "AUTO"))
    1065             :     {
    1066          31 :         if (bHeterogeneousDataTypes)
    1067             :         {
    1068           0 :             m_osPixelEncoding = "NPY";
    1069             :         }
    1070             :         else
    1071             :         {
    1072          31 :             m_osPixelEncoding = "PNG";
    1073          86 :             for (int i = 1; i <= nBands; i++)
    1074             :             {
    1075          55 :                 if (GetRasterBand(i)->GetRasterDataType() != GDT_Byte)
    1076             :                 {
    1077          43 :                     m_osPixelEncoding = "GEO_TIFF";
    1078             :                 }
    1079             :             }
    1080             :         }
    1081             :     }
    1082             : 
    1083          66 :     if (EQUAL(m_osPixelEncoding, "PNG") || EQUAL(m_osPixelEncoding, "JPEG") ||
    1084          31 :         EQUAL(m_osPixelEncoding, "AUTO_JPEG_PNG"))
    1085             :     {
    1086           4 :         if (nBands != 1 && nBands != 3)
    1087             :         {
    1088           0 :             m_bQueryMultipleBands = false;
    1089             :         }
    1090          16 :         for (int i = 1; i <= nBands; i++)
    1091             :         {
    1092          12 :             if (GetRasterBand(i)->GetRasterDataType() != GDT_Byte)
    1093             :             {
    1094           0 :                 CPLError(
    1095             :                     CE_Failure, CPLE_NotSupported,
    1096             :                     "This dataset has non-Byte bands, which is incompatible "
    1097             :                     "with PIXEL_ENCODING=%s",
    1098             :                     m_osPixelEncoding.c_str());
    1099           0 :                 return false;
    1100             :             }
    1101             :         }
    1102             :     }
    1103             : 
    1104          35 :     if (nBands > SERVER_SIMUTANEOUS_BAND_LIMIT)
    1105             :     {
    1106           0 :         m_bQueryMultipleBands = false;
    1107             :     }
    1108             : 
    1109          35 :     if (m_bQueryMultipleBands && m_osPixelEncoding != "NPY" &&
    1110             :         bHeterogeneousDataTypes)
    1111             :     {
    1112           0 :         CPLDebug("EEDAI",
    1113             :                  "%s PIXEL_ENCODING does not support heterogeneous data types. "
    1114             :                  "Falling back to querying band per band",
    1115             :                  m_osPixelEncoding.c_str());
    1116           0 :         m_bQueryMultipleBands = false;
    1117             :     }
    1118             : 
    1119          35 :     return true;
    1120             : }
    1121             : 
    1122             : /************************************************************************/
    1123             : /*                          GetSpatialRef()                             */
    1124             : /************************************************************************/
    1125             : 
    1126           3 : const OGRSpatialReference *GDALEEDAIDataset::GetSpatialRef() const
    1127             : {
    1128           3 :     return m_oSRS.IsEmpty() ? nullptr : &m_oSRS;
    1129             : }
    1130             : 
    1131             : /************************************************************************/
    1132             : /*                          GetGeoTransform()                           */
    1133             : /************************************************************************/
    1134             : 
    1135           8 : CPLErr GDALEEDAIDataset::GetGeoTransform(double *adfGeoTransform)
    1136             : {
    1137           8 :     memcpy(adfGeoTransform, m_adfGeoTransform, 6 * sizeof(double));
    1138           8 :     return CE_None;
    1139             : }
    1140             : 
    1141             : /************************************************************************/
    1142             : /*                               Open()                                 */
    1143             : /************************************************************************/
    1144             : 
    1145           9 : bool GDALEEDAIDataset::Open(GDALOpenInfo *poOpenInfo)
    1146             : {
    1147             :     m_osBaseURL = CPLGetConfigOption(
    1148           9 :         "EEDA_URL", "https://earthengine-highvolume.googleapis.com/v1alpha/");
    1149             : 
    1150           9 :     m_osAsset = CSLFetchNameValueDef(poOpenInfo->papszOpenOptions, "ASSET", "");
    1151             :     CPLString osBandList(
    1152          18 :         CSLFetchNameValueDef(poOpenInfo->papszOpenOptions, "BANDS", ""));
    1153           9 :     if (m_osAsset.empty())
    1154             :     {
    1155             :         char **papszTokens =
    1156           9 :             CSLTokenizeString2(poOpenInfo->pszFilename, ":", 0);
    1157           9 :         if (CSLCount(papszTokens) < 2)
    1158             :         {
    1159           0 :             CPLError(CE_Failure, CPLE_AppDefined,
    1160             :                      "No asset specified in connection string or "
    1161             :                      "ASSET open option");
    1162           0 :             CSLDestroy(papszTokens);
    1163           0 :             return false;
    1164             :         }
    1165           9 :         if (CSLCount(papszTokens) == 3)
    1166             :         {
    1167           2 :             osBandList = papszTokens[2];
    1168             :         }
    1169             : 
    1170           9 :         m_osAsset = papszTokens[1];
    1171           9 :         CSLDestroy(papszTokens);
    1172             :     }
    1173           9 :     m_osAssetName = ConvertPathToName(m_osAsset);
    1174             : 
    1175           9 :     m_osPixelEncoding = CSLFetchNameValueDef(poOpenInfo->papszOpenOptions,
    1176           9 :                                              "PIXEL_ENCODING", "AUTO");
    1177           9 :     m_nBlockSize =
    1178           9 :         atoi(CSLFetchNameValueDef(poOpenInfo->papszOpenOptions, "BLOCK_SIZE",
    1179             :                                   CPLSPrintf("%d", DEFAULT_BLOCK_SIZE)));
    1180           9 :     if (m_nBlockSize < 128 &&
    1181           0 :         !CPLTestBool(CPLGetConfigOption("EEDA_FORCE_BLOCK_SIZE", "FALSE")))
    1182             :     {
    1183           0 :         CPLError(CE_Failure, CPLE_NotSupported, "Invalid BLOCK_SIZE");
    1184           0 :         return false;
    1185             :     }
    1186             : 
    1187          18 :     std::set<CPLString> oSetUserBandNames;
    1188             :     {
    1189           9 :         char **papszTokens = CSLTokenizeString2(osBandList, ",", 0);
    1190          12 :         for (int i = 0; papszTokens && papszTokens[i]; i++)
    1191           3 :             oSetUserBandNames.insert(papszTokens[i]);
    1192           9 :         CSLDestroy(papszTokens);
    1193             :     }
    1194             : 
    1195             :     // Issue request to get image metadata
    1196           9 :     char **papszOptions = GetBaseHTTPOptions();
    1197           9 :     if (papszOptions == nullptr)
    1198           1 :         return false;
    1199             :     CPLHTTPResult *psResult =
    1200           8 :         EEDAHTTPFetch((m_osBaseURL + m_osAssetName).c_str(), papszOptions);
    1201           8 :     CSLDestroy(papszOptions);
    1202           8 :     if (psResult == nullptr)
    1203           0 :         return false;
    1204           8 :     if (psResult->pszErrBuf != nullptr)
    1205             :     {
    1206           0 :         if (psResult->pabyData)
    1207             :         {
    1208           0 :             CPLError(CE_Failure, CPLE_AppDefined, "%s: %s", psResult->pszErrBuf,
    1209           0 :                      reinterpret_cast<const char *>(psResult->pabyData));
    1210             :         }
    1211             :         else
    1212             :         {
    1213           0 :             CPLError(CE_Failure, CPLE_AppDefined, "%s", psResult->pszErrBuf);
    1214             :         }
    1215           0 :         CPLHTTPDestroyResult(psResult);
    1216           0 :         return false;
    1217             :     }
    1218             : 
    1219           8 :     if (psResult->pabyData == nullptr)
    1220             :     {
    1221           0 :         CPLError(CE_Failure, CPLE_AppDefined,
    1222             :                  "Empty content returned by server");
    1223           0 :         CPLHTTPDestroyResult(psResult);
    1224           0 :         return false;
    1225             :     }
    1226             : 
    1227           8 :     const char *pszText = reinterpret_cast<const char *>(psResult->pabyData);
    1228             : #ifdef DEBUG_VERBOSE
    1229             :     CPLDebug("EEDAI", "%s", pszText);
    1230             : #endif
    1231             : 
    1232           8 :     json_object *poObj = nullptr;
    1233           8 :     if (!OGRJSonParse(pszText, &poObj, true))
    1234             :     {
    1235           0 :         CPLHTTPDestroyResult(psResult);
    1236           0 :         return false;
    1237             :     }
    1238             : 
    1239           8 :     CPLHTTPDestroyResult(psResult);
    1240             : 
    1241           8 :     if (json_object_get_type(poObj) != json_type_object)
    1242             :     {
    1243           0 :         CPLError(CE_Failure, CPLE_AppDefined,
    1244             :                  "Return is not a JSON dictionary");
    1245           0 :         json_object_put(poObj);
    1246           0 :         return false;
    1247             :     }
    1248             : 
    1249           8 :     json_object *poType = CPL_json_object_object_get(poObj, "type");
    1250           8 :     const char *pszType = json_object_get_string(poType);
    1251           8 :     if (pszType == nullptr || !EQUAL(pszType, "IMAGE"))
    1252             :     {
    1253           0 :         CPLError(CE_Failure, CPLE_AppDefined, "Asset is not an image, but %s",
    1254             :                  pszType ? pszType : "(null)");
    1255           0 :         json_object_put(poObj);
    1256           0 :         return false;
    1257             :     }
    1258             : 
    1259           8 :     json_object *poBands = CPL_json_object_object_get(poObj, "bands");
    1260           8 :     if (poBands == nullptr || json_object_get_type(poBands) != json_type_array)
    1261             :     {
    1262           0 :         CPLError(CE_Failure, CPLE_AppDefined, "No band found");
    1263           0 :         json_object_put(poObj);
    1264           0 :         return false;
    1265             :     }
    1266             : 
    1267          16 :     std::map<CPLString, CPLString> oMapCodeToWKT;
    1268             :     std::vector<EEDAIBandDesc> aoBandDesc =
    1269          16 :         BuildBandDescArray(poBands, oMapCodeToWKT);
    1270          16 :     std::map<CPLString, int> aoMapBandNames;
    1271             : 
    1272           8 :     if (aoBandDesc.empty())
    1273             :     {
    1274           0 :         CPLError(CE_Failure, CPLE_AppDefined, "No band found");
    1275           0 :         json_object_put(poObj);
    1276           0 :         return false;
    1277             :     }
    1278             : 
    1279             :     // Indices are aoBandDesc indices
    1280          16 :     std::map<int, std::vector<int>> oMapSimilarBands;
    1281             : 
    1282           8 :     size_t iIdxFirstBand = 0;
    1283          30 :     for (size_t i = 0; i < aoBandDesc.size(); i++)
    1284             :     {
    1285             :         // Instantiate bands if they are compatible between them, and
    1286             :         // if they are requested by the user (when user explicitly requested
    1287             :         // them)
    1288          28 :         if ((oSetUserBandNames.empty() ||
    1289           6 :              oSetUserBandNames.find(aoBandDesc[i].osName) !=
    1290          44 :                  oSetUserBandNames.end()) &&
    1291          19 :             (nBands == 0 || aoBandDesc[i].IsSimilar(aoBandDesc[iIdxFirstBand])))
    1292             :         {
    1293          15 :             if (nBands == 0)
    1294             :             {
    1295           8 :                 iIdxFirstBand = i;
    1296           8 :                 nRasterXSize = aoBandDesc[i].nWidth;
    1297           8 :                 nRasterYSize = aoBandDesc[i].nHeight;
    1298           8 :                 memcpy(m_adfGeoTransform, aoBandDesc[i].adfGeoTransform.data(),
    1299             :                        6 * sizeof(double));
    1300           8 :                 m_oSRS.importFromWkt(aoBandDesc[i].osWKT);
    1301           8 :                 int iOvr = 0;
    1302          35 :                 while ((nRasterXSize >> iOvr) > 256 ||
    1303           8 :                        (nRasterYSize >> iOvr) > 256)
    1304             :                 {
    1305          27 :                     iOvr++;
    1306          27 :                     m_apoOverviewDS.push_back(new GDALEEDAIDataset(this, iOvr));
    1307             :                 }
    1308             :             }
    1309             : 
    1310             :             GDALRasterBand *poBand =
    1311          15 :                 new GDALEEDAIRasterBand(this, aoBandDesc[i].eDT);
    1312          15 :             const int iBand = nBands + 1;
    1313          15 :             SetBand(iBand, poBand);
    1314          15 :             poBand->SetDescription(aoBandDesc[i].osName);
    1315             : 
    1316             :             // as images in USDA/NAIP/DOQQ catalog
    1317          15 :             if (EQUAL(aoBandDesc[i].osName, "R"))
    1318           0 :                 poBand->SetColorInterpretation(GCI_RedBand);
    1319          15 :             else if (EQUAL(aoBandDesc[i].osName, "G"))
    1320           0 :                 poBand->SetColorInterpretation(GCI_GreenBand);
    1321          15 :             else if (EQUAL(aoBandDesc[i].osName, "B"))
    1322           0 :                 poBand->SetColorInterpretation(GCI_BlueBand);
    1323             : 
    1324          63 :             for (size_t iOvr = 0; iOvr < m_apoOverviewDS.size(); iOvr++)
    1325             :             {
    1326             :                 GDALRasterBand *poOvrBand = new GDALEEDAIRasterBand(
    1327          48 :                     m_apoOverviewDS[iOvr], aoBandDesc[i].eDT);
    1328          48 :                 m_apoOverviewDS[iOvr]->SetBand(iBand, poOvrBand);
    1329          48 :                 poOvrBand->SetDescription(aoBandDesc[i].osName);
    1330             :             }
    1331             : 
    1332          15 :             aoMapBandNames[aoBandDesc[i].osName] = iBand;
    1333             :         }
    1334             :         else
    1335             :         {
    1336           7 :             if (oSetUserBandNames.find(aoBandDesc[i].osName) !=
    1337          14 :                 oSetUserBandNames.end())
    1338             :             {
    1339           0 :                 CPLError(CE_Warning, CPLE_AppDefined,
    1340             :                          "Band %s is not compatible of other bands",
    1341           0 :                          aoBandDesc[i].osName.c_str());
    1342             :             }
    1343           7 :             aoMapBandNames[aoBandDesc[i].osName] = -1;
    1344             :         }
    1345             : 
    1346             :         // Group similar bands to be able to build subdataset list
    1347             :         std::map<int, std::vector<int>>::iterator oIter =
    1348          22 :             oMapSimilarBands.begin();
    1349          28 :         for (; oIter != oMapSimilarBands.end(); ++oIter)
    1350             :         {
    1351          14 :             if (aoBandDesc[i].IsSimilar(aoBandDesc[oIter->first]))
    1352             :             {
    1353           8 :                 oIter->second.push_back(static_cast<int>(i));
    1354           8 :                 break;
    1355             :             }
    1356             :         }
    1357          22 :         if (oIter == oMapSimilarBands.end())
    1358             :         {
    1359          14 :             oMapSimilarBands[static_cast<int>(i)].push_back(
    1360          14 :                 static_cast<int>(i));
    1361             :         }
    1362             :     }
    1363             : 
    1364           8 :     if (!ComputeQueryStrategy())
    1365             :     {
    1366           0 :         json_object_put(poObj);
    1367           0 :         return false;
    1368             :     }
    1369          35 :     for (size_t i = 0; i < m_apoOverviewDS.size(); i++)
    1370             :     {
    1371          27 :         m_apoOverviewDS[i]->ComputeQueryStrategy();
    1372             :     }
    1373             : 
    1374           8 :     if (nBands > 1)
    1375             :     {
    1376           6 :         SetMetadataItem("INTERLEAVE", m_bQueryMultipleBands ? "PIXEL" : "BAND",
    1377           6 :                         "IMAGE_STRUCTURE");
    1378             :     }
    1379             : 
    1380             :     // Build subdataset list
    1381           8 :     if (oSetUserBandNames.empty() && oMapSimilarBands.size() > 1)
    1382             :     {
    1383           8 :         CPLStringList aoSubDSList;
    1384             :         std::map<int, std::vector<int>>::iterator oIter =
    1385           4 :             oMapSimilarBands.begin();
    1386          12 :         for (; oIter != oMapSimilarBands.end(); ++oIter)
    1387             :         {
    1388          16 :             CPLString osSubDSSuffix;
    1389          20 :             for (size_t i = 0; i < oIter->second.size(); ++i)
    1390             :             {
    1391          12 :                 if (!osSubDSSuffix.empty())
    1392           4 :                     osSubDSSuffix += ",";
    1393          12 :                 osSubDSSuffix += aoBandDesc[oIter->second[i]].osName;
    1394             :             }
    1395             :             aoSubDSList.AddNameValue(
    1396          16 :                 CPLSPrintf("SUBDATASET_%d_NAME", aoSubDSList.size() / 2 + 1),
    1397             :                 CPLSPrintf("EEDAI:%s:%s", m_osAsset.c_str(),
    1398           8 :                            osSubDSSuffix.c_str()));
    1399             :             aoSubDSList.AddNameValue(
    1400          16 :                 CPLSPrintf("SUBDATASET_%d_DESC", aoSubDSList.size() / 2 + 1),
    1401             :                 CPLSPrintf("Band%s %s of %s",
    1402           8 :                            oIter->second.size() > 1 ? "s" : "",
    1403          16 :                            osSubDSSuffix.c_str(), m_osAsset.c_str()));
    1404             :         }
    1405           4 :         SetMetadata(aoSubDSList.List(), "SUBDATASETS");
    1406             :     }
    1407             : 
    1408             :     // Attach metadata to dataset or bands
    1409           8 :     json_object *poProperties = CPL_json_object_object_get(poObj, "properties");
    1410           8 :     if (poProperties && json_object_get_type(poProperties) == json_type_object)
    1411             :     {
    1412           6 :         SetMetadataFromProperties(poProperties, aoMapBandNames);
    1413             :     }
    1414           8 :     json_object_put(poObj);
    1415             : 
    1416           8 :     SetDescription(poOpenInfo->pszFilename);
    1417             : 
    1418           8 :     return true;
    1419             : }
    1420             : 
    1421             : /************************************************************************/
    1422             : /*                       SetMetadataFromProperties()                    */
    1423             : /************************************************************************/
    1424             : 
    1425           6 : void GDALEEDAIDataset::SetMetadataFromProperties(
    1426             :     json_object *poProperties, const std::map<CPLString, int> &aoMapBandNames)
    1427             : {
    1428             :     json_object_iter it;
    1429           6 :     it.key = nullptr;
    1430           6 :     it.val = nullptr;
    1431           6 :     it.entry = nullptr;
    1432          24 :     json_object_object_foreachC(poProperties, it)
    1433             :     {
    1434          18 :         if (it.val)
    1435             :         {
    1436          36 :             CPLString osKey(it.key);
    1437          18 :             int nBandForMD = 0;
    1438             :             std::map<CPLString, int>::const_iterator oIter =
    1439          18 :                 aoMapBandNames.begin();
    1440          54 :             for (; oIter != aoMapBandNames.end(); ++oIter)
    1441             :             {
    1442          48 :                 CPLString osBandName(oIter->first);
    1443          48 :                 CPLString osNeedle("_" + osBandName);
    1444          48 :                 size_t nPos = osKey.find(osNeedle);
    1445          54 :                 if (nPos != std::string::npos &&
    1446           6 :                     nPos + osNeedle.size() == osKey.size())
    1447             :                 {
    1448           6 :                     nBandForMD = oIter->second;
    1449           6 :                     osKey.resize(nPos);
    1450           6 :                     break;
    1451             :                 }
    1452             : 
    1453             :                 // Landsat bands are named Bxxx, must their metadata
    1454             :                 // are _BAND_xxxx ...
    1455          84 :                 if (osBandName.size() > 1 && osBandName[0] == 'B' &&
    1456          42 :                     atoi(osBandName.c_str() + 1) > 0)
    1457             :                 {
    1458          42 :                     osNeedle = "_BAND_" + osBandName.substr(1);
    1459          42 :                     nPos = osKey.find(osNeedle);
    1460          48 :                     if (nPos != std::string::npos &&
    1461           6 :                         nPos + osNeedle.size() == osKey.size())
    1462             :                     {
    1463           6 :                         nBandForMD = oIter->second;
    1464           6 :                         osKey.resize(nPos);
    1465           6 :                         break;
    1466             :                     }
    1467             :                 }
    1468             :             }
    1469             : 
    1470          18 :             if (nBandForMD > 0)
    1471             :             {
    1472           6 :                 GetRasterBand(nBandForMD)
    1473           6 :                     ->SetMetadataItem(osKey, json_object_get_string(it.val));
    1474             :             }
    1475          12 :             else if (nBandForMD == 0)
    1476             :             {
    1477           6 :                 SetMetadataItem(osKey, json_object_get_string(it.val));
    1478             :             }
    1479             :         }
    1480             :     }
    1481           6 : }
    1482             : 
    1483             : /************************************************************************/
    1484             : /*                          GDALEEDAIIdentify()                         */
    1485             : /************************************************************************/
    1486             : 
    1487       51199 : static int GDALEEDAIIdentify(GDALOpenInfo *poOpenInfo)
    1488             : {
    1489       51199 :     return STARTS_WITH_CI(poOpenInfo->pszFilename, "EEDAI:");
    1490             : }
    1491             : 
    1492             : /************************************************************************/
    1493             : /*                            GDALEEDAIOpen()                           */
    1494             : /************************************************************************/
    1495             : 
    1496           9 : static GDALDataset *GDALEEDAIOpen(GDALOpenInfo *poOpenInfo)
    1497             : {
    1498           9 :     if (!GDALEEDAIIdentify(poOpenInfo))
    1499           0 :         return nullptr;
    1500             : 
    1501           9 :     GDALEEDAIDataset *poDS = new GDALEEDAIDataset();
    1502           9 :     if (!poDS->Open(poOpenInfo))
    1503             :     {
    1504           1 :         delete poDS;
    1505           1 :         return nullptr;
    1506             :     }
    1507           8 :     return poDS;
    1508             : }
    1509             : 
    1510             : /************************************************************************/
    1511             : /*                         GDALRegister_EEDAI()                         */
    1512             : /************************************************************************/
    1513             : 
    1514        1682 : void GDALRegister_EEDAI()
    1515             : 
    1516             : {
    1517        1682 :     if (GDALGetDriverByName("EEDAI") != nullptr)
    1518         301 :         return;
    1519             : 
    1520        1381 :     GDALDriver *poDriver = new GDALDriver();
    1521             : 
    1522        1381 :     poDriver->SetDescription("EEDAI");
    1523        1381 :     poDriver->SetMetadataItem(GDAL_DCAP_RASTER, "YES");
    1524        1381 :     poDriver->SetMetadataItem(GDAL_DMD_LONGNAME, "Earth Engine Data API Image");
    1525        1381 :     poDriver->SetMetadataItem(GDAL_DMD_HELPTOPIC, "drivers/raster/eedai.html");
    1526        1381 :     poDriver->SetMetadataItem(GDAL_DMD_CONNECTION_PREFIX, "EEDAI:");
    1527        1381 :     poDriver->SetMetadataItem(
    1528             :         GDAL_DMD_OPENOPTIONLIST,
    1529             :         "<OpenOptionList>"
    1530             :         "  <Option name='ASSET' type='string' description='Asset name'/>"
    1531             :         "  <Option name='BANDS' type='string' "
    1532             :         "description='Comma separated list of band names'/>"
    1533             :         "  <Option name='PIXEL_ENCODING' type='string-select' "
    1534             :         "description='Format in which pixls are queried'>"
    1535             :         "       <Value>AUTO</Value>"
    1536             :         "       <Value>PNG</Value>"
    1537             :         "       <Value>JPEG</Value>"
    1538             :         "       <Value>GEO_TIFF</Value>"
    1539             :         "       <Value>AUTO_JPEG_PNG</Value>"
    1540             :         "       <Value>NPY</Value>"
    1541             :         "   </Option>"
    1542             :         "  <Option name='BLOCK_SIZE' type='integer' "
    1543             :         "description='Size of a block' default='256'/>"
    1544             :         "  <Option name='VSI_PATH_FOR_AUTH' type='string' "
    1545             :         "description='/vsigs/... path onto which a "
    1546             :         "GOOGLE_APPLICATION_CREDENTIALS path specific "
    1547             :         "option is set'/>"
    1548        1381 :         "</OpenOptionList>");
    1549        1381 :     poDriver->SetMetadataItem(GDAL_DMD_SUBDATASETS, "YES");
    1550             : 
    1551        1381 :     poDriver->pfnOpen = GDALEEDAIOpen;
    1552        1381 :     poDriver->pfnIdentify = GDALEEDAIIdentify;
    1553             : 
    1554        1381 :     GetGDALDriverManager()->RegisterDriver(poDriver);
    1555             : }

Generated by: LCOV version 1.14