LCOV - code coverage report
Current view: top level - frmts/daas - daasdataset.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 957 1135 84.3 %
Date: 2025-01-18 12:42:00 Functions: 40 41 97.6 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  DAAS driver
       4             :  * Purpose:  DAAS driver
       5             :  * Author:   Even Rouault, <even.rouault at spatialys.com>
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2018-2019, Airbus DS Intelligence
       9             :  *
      10             :  * SPDX-License-Identifier: MIT
      11             :  ****************************************************************************/
      12             : 
      13             : #include "cpl_http.h"
      14             : #include "gdal_frmts.h"
      15             : #include "gdal_alg.h"
      16             : #include "gdal_priv.h"
      17             : #include "ogr_spatialref.h"
      18             : #include "gdal_mdreader.h"
      19             : #include "../mem/memdataset.h"
      20             : 
      21             : #include "cpl_json.h"
      22             : 
      23             : #include <algorithm>
      24             : #include <array>
      25             : #include <memory>
      26             : 
      27             : constexpr int knMIN_BLOCKSIZE = 64;
      28             : constexpr int knDEFAULT_BLOCKSIZE = 512;
      29             : constexpr int knMAX_BLOCKSIZE = 8192;
      30             : 
      31             : constexpr GUInt32 RETRY_PER_BAND = 1;
      32             : constexpr GUInt32 RETRY_SPATIAL_SPLIT = 2;
      33             : 
      34             : // Let's limit to 100 MB uncompressed per request
      35             : constexpr int knDEFAULT_SERVER_BYTE_LIMIT = 100 * 1024 * 1024;
      36             : 
      37             : constexpr int MAIN_MASK_BAND_NUMBER = 0;
      38             : 
      39             : /************************************************************************/
      40             : /*                          GDALDAASBandDesc                            */
      41             : /************************************************************************/
      42             : 
      43             : class GDALDAASBandDesc
      44             : {
      45             :   public:
      46             :     int nIndex = 0;
      47             :     GDALDataType eDT =
      48             :         GDT_Unknown;  // as declared in the GetMetadata response bands[]
      49             :     CPLString osName;
      50             :     CPLString osDescription;
      51             :     CPLString osColorInterp;
      52             :     bool bIsMask = false;
      53             : };
      54             : 
      55             : /************************************************************************/
      56             : /*                          GDALDAASDataset                             */
      57             : /************************************************************************/
      58             : 
      59             : class GDALDAASRasterBand;
      60             : 
      61             : class GDALDAASDataset final : public GDALDataset
      62             : {
      63             :   public:
      64             :     enum class Format
      65             :     {
      66             :         RAW,
      67             :         PNG,
      68             :         JPEG,
      69             :         JPEG2000,
      70             :     };
      71             : 
      72             :   private:
      73             :     friend class GDALDAASRasterBand;
      74             : 
      75             :     CPLString m_osGetMetadataURL;
      76             : 
      77             :     CPLString m_osAuthURL;
      78             :     CPLString m_osAccessToken;
      79             :     time_t m_nExpirationTime = 0;
      80             :     CPLString m_osXForwardUser;
      81             : 
      82             :     GDALDAASDataset *m_poParentDS = nullptr;
      83             :     // int         m_iOvrLevel = 0;
      84             : 
      85             :     OGRSpatialReference m_oSRS{};
      86             :     CPLString m_osSRSType;
      87             :     CPLString m_osSRSValue;
      88             :     bool m_bGotGeoTransform = false;
      89             :     std::array<double, 6> m_adfGeoTransform{{0.0, 1.0, 0.0, 0.0, 0.0, 1.0}};
      90             :     bool m_bRequestInGeoreferencedCoordinates = false;
      91             :     GDALDataType m_eDT = GDT_Unknown;
      92             :     int m_nActualBitDepth = 0;
      93             :     bool m_bHasNoData = false;
      94             :     double m_dfNoDataValue = 0.0;
      95             :     CPLString m_osGetBufferURL;
      96             :     int m_nBlockSize = knDEFAULT_BLOCKSIZE;
      97             :     Format m_eFormat = Format::RAW;
      98             :     GIntBig m_nServerByteLimit = knDEFAULT_SERVER_BYTE_LIMIT;
      99             :     GDALRIOResampleAlg m_eCurrentResampleAlg = GRIORA_NearestNeighbour;
     100             : 
     101             :     int m_nMainMaskBandIndex = 0;
     102             :     CPLString m_osMainMaskName;
     103             :     GDALDAASRasterBand *m_poMaskBand = nullptr;
     104             :     std::vector<GDALDAASBandDesc> m_aoBandDesc;
     105             : 
     106             :     int m_nXOffAdvise = 0;
     107             :     int m_nYOffAdvise = 0;
     108             :     int m_nXSizeAdvise = 0;
     109             :     int m_nYSizeAdvise = 0;
     110             : 
     111             :     int m_nXOffFetched = 0;
     112             :     int m_nYOffFetched = 0;
     113             :     int m_nXSizeFetched = 0;
     114             :     int m_nYSizeFetched = 0;
     115             : 
     116             :     std::vector<std::unique_ptr<GDALDAASDataset>> m_apoOverviewDS;
     117             : 
     118             :     char **m_papszOpenOptions = nullptr;
     119             : 
     120             :     // Methods
     121             :     bool Open(GDALOpenInfo *poOpenInfo);
     122             :     bool GetAuthorization();
     123             :     bool GetImageMetadata();
     124             :     char **GetHTTPOptions();
     125             :     void ReadSRS(const CPLJSONObject &oProperties);
     126             :     void ReadRPCs(const CPLJSONObject &oProperties);
     127             :     bool SetupServerSideReprojection(const char *pszTargetSRS);
     128             :     void InstantiateBands();
     129             : 
     130             :   public:
     131             :     GDALDAASDataset();
     132             :     GDALDAASDataset(GDALDAASDataset *poParentDS, int iOvrLevel);
     133             :     ~GDALDAASDataset();
     134             : 
     135             :     static int Identify(GDALOpenInfo *poOpenInfo);
     136             :     static GDALDataset *OpenStatic(GDALOpenInfo *poOpenInfo);
     137             : 
     138             :     CPLErr GetGeoTransform(double *padfTransform) override;
     139             :     const OGRSpatialReference *GetSpatialRef() const override;
     140             :     CPLErr IRasterIO(GDALRWFlag eRWFlag, int nXOff, int nYOff, int nXSize,
     141             :                      int nYSize, void *pData, int nBufXSize, int nBufYSize,
     142             :                      GDALDataType eBufType, int nBandCount,
     143             :                      BANDMAP_TYPE panBands, GSpacing nPixelSpace,
     144             :                      GSpacing nLineSpace, GSpacing nBandSpace,
     145             :                      GDALRasterIOExtraArg *psExtraArg) override;
     146             :     CPLErr AdviseRead(int nXOff, int nYOff, int nXSize, int nYSize,
     147             :                       int /* nBufXSize */, int /* nBufYSize */,
     148             :                       GDALDataType /* eBufType */, int /*nBands*/,
     149             :                       int * /*panBands*/, char ** /* papszOptions */) override;
     150             :     CPLErr FlushCache(bool bAtClosing) override;
     151             : };
     152             : 
     153             : /************************************************************************/
     154             : /*                         GDALDAASRasterBand                            */
     155             : /************************************************************************/
     156             : 
     157             : class GDALDAASRasterBand final : public GDALRasterBand
     158             : {
     159             :     friend class GDALDAASDataset;
     160             : 
     161             :     int m_nSrcIndex = 0;
     162             :     GDALColorInterp m_eColorInterp = GCI_Undefined;
     163             : 
     164             :     CPLErr GetBlocks(int nBlockXOff, int nBlockYOff, int nXBlocks, int nYBlocks,
     165             :                      const std::vector<int> &anRequestedBands, void *pBuffer);
     166             : 
     167             :     GUInt32 PrefetchBlocks(int nXOff, int nYOff, int nXSize, int nYSize,
     168             :                            const std::vector<int> &anRequestedBands);
     169             : 
     170             :   public:
     171             :     GDALDAASRasterBand(GDALDAASDataset *poDS, int nBand,
     172             :                        const GDALDAASBandDesc &oBandDesc);
     173             : 
     174             :     CPLErr IReadBlock(int nBlockXOff, int nBlockYOff, void *pData) override;
     175             :     CPLErr IRasterIO(GDALRWFlag eRWFlag, int nXOff, int nYOff, int nXSize,
     176             :                      int nYSize, void *pData, int nBufXSize, int nBufYSize,
     177             :                      GDALDataType eBufType, GSpacing nPixelSpace,
     178             :                      GSpacing nLineSpace,
     179             :                      GDALRasterIOExtraArg *psExtraArg) override;
     180             :     CPLErr AdviseRead(int nXOff, int nYOff, int nXSize, int nYSize,
     181             :                       int /* nBufXSize */, int /* nBufYSize */,
     182             :                       GDALDataType /* eBufType */,
     183             :                       char ** /* papszOptions */) override;
     184             :     double GetNoDataValue(int *pbHasNoData) override;
     185             :     GDALColorInterp GetColorInterpretation() override;
     186             :     GDALRasterBand *GetMaskBand() override;
     187             :     int GetMaskFlags() override;
     188             :     int GetOverviewCount() override;
     189             :     GDALRasterBand *GetOverview(int) override;
     190             : };
     191             : 
     192             : /************************************************************************/
     193             : /*                          GDALDAASDataset()                           */
     194             : /************************************************************************/
     195             : 
     196          45 : GDALDAASDataset::GDALDAASDataset()
     197             :     : m_osAuthURL(CPLGetConfigOption(
     198             :           "GDAL_DAAS_AUTH_URL", "https://authenticate.geoapi-airbusds.com/auth/"
     199          45 :                                 "realms/IDP/protocol/openid-connect/token"))
     200             : {
     201          45 :     m_oSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
     202          45 : }
     203             : 
     204             : /************************************************************************/
     205             : /*                          GDALDAASDataset()                           */
     206             : /************************************************************************/
     207             : 
     208           6 : GDALDAASDataset::GDALDAASDataset(GDALDAASDataset *poParentDS, int iOvrLevel)
     209           6 :     : m_osGetMetadataURL(poParentDS->m_osGetMetadataURL),
     210           6 :       m_osAuthURL(poParentDS->m_osAuthURL),
     211             :       m_osAccessToken(CPLString()),   // only used by parent
     212             :       m_nExpirationTime(0),           // only used by parent
     213             :       m_osXForwardUser(CPLString()),  // only used by parent
     214             :       m_poParentDS(poParentDS),
     215             :       // m_iOvrLevel(iOvrLevel),
     216          12 :       m_oSRS(poParentDS->m_oSRS), m_osSRSType(poParentDS->m_osSRSType),
     217           6 :       m_osSRSValue(poParentDS->m_osSRSValue),
     218           6 :       m_bGotGeoTransform(poParentDS->m_bGotGeoTransform),
     219             :       m_bRequestInGeoreferencedCoordinates(
     220           6 :           poParentDS->m_bRequestInGeoreferencedCoordinates),
     221           6 :       m_eDT(poParentDS->m_eDT),
     222           6 :       m_nActualBitDepth(poParentDS->m_nActualBitDepth),
     223           6 :       m_bHasNoData(poParentDS->m_bHasNoData),
     224           6 :       m_dfNoDataValue(poParentDS->m_dfNoDataValue),
     225           6 :       m_osGetBufferURL(poParentDS->m_osGetBufferURL),
     226           6 :       m_eFormat(poParentDS->m_eFormat),
     227           6 :       m_nServerByteLimit(poParentDS->m_nServerByteLimit),
     228           6 :       m_nMainMaskBandIndex(poParentDS->m_nMainMaskBandIndex),
     229           6 :       m_osMainMaskName(poParentDS->m_osMainMaskName), m_poMaskBand(nullptr),
     230           6 :       m_aoBandDesc(poParentDS->m_aoBandDesc)
     231             : {
     232           6 :     nRasterXSize = m_poParentDS->nRasterXSize >> iOvrLevel;
     233           6 :     nRasterYSize = m_poParentDS->nRasterYSize >> iOvrLevel;
     234           6 :     m_adfGeoTransform[0] = m_poParentDS->m_adfGeoTransform[0];
     235           6 :     m_adfGeoTransform[1] = m_poParentDS->m_adfGeoTransform[1] *
     236           6 :                            m_poParentDS->nRasterXSize / nRasterXSize;
     237           6 :     m_adfGeoTransform[2] = m_poParentDS->m_adfGeoTransform[2];
     238           6 :     m_adfGeoTransform[3] = m_poParentDS->m_adfGeoTransform[3];
     239           6 :     m_adfGeoTransform[4] = m_poParentDS->m_adfGeoTransform[4];
     240           6 :     m_adfGeoTransform[5] = m_poParentDS->m_adfGeoTransform[5] *
     241           6 :                            m_poParentDS->nRasterYSize / nRasterYSize;
     242             : 
     243           6 :     InstantiateBands();
     244             : 
     245           6 :     SetMetadata(m_poParentDS->GetMetadata());
     246           6 :     SetMetadata(m_poParentDS->GetMetadata("RPC"), "RPC");
     247           6 : }
     248             : 
     249             : /************************************************************************/
     250             : /*                         ~GDALDAASDataset()                            */
     251             : /************************************************************************/
     252             : 
     253         102 : GDALDAASDataset::~GDALDAASDataset()
     254             : {
     255          51 :     if (m_poParentDS == nullptr)
     256             :     {
     257          45 :         char **papszOptions = nullptr;
     258          45 :         papszOptions = CSLSetNameValue(papszOptions, "CLOSE_PERSISTENT",
     259             :                                        CPLSPrintf("%p", this));
     260          45 :         CPLHTTPDestroyResult(CPLHTTPFetch("", papszOptions));
     261          45 :         CSLDestroy(papszOptions);
     262             :     }
     263             : 
     264          51 :     delete m_poMaskBand;
     265          51 :     CSLDestroy(m_papszOpenOptions);
     266         102 : }
     267             : 
     268             : /************************************************************************/
     269             : /*                          InstantiateBands()                          */
     270             : /************************************************************************/
     271             : 
     272          25 : void GDALDAASDataset::InstantiateBands()
     273             : {
     274          64 :     for (int i = 0; i < static_cast<int>(m_aoBandDesc.size()); i++)
     275             :     {
     276             :         GDALRasterBand *poBand =
     277          39 :             new GDALDAASRasterBand(this, i + 1, m_aoBandDesc[i]);
     278          39 :         SetBand(i + 1, poBand);
     279             :     }
     280             : 
     281          25 :     if (!m_osMainMaskName.empty())
     282             :     {
     283           1 :         GDALDAASBandDesc oDesc;
     284           1 :         oDesc.nIndex = m_nMainMaskBandIndex;
     285           1 :         oDesc.osName = m_osMainMaskName;
     286           1 :         m_poMaskBand = new GDALDAASRasterBand(this, 0, oDesc);
     287             :     }
     288             : 
     289          25 :     if (nBands > 1)
     290             :     {
     291             :         // Hint for users of the driver
     292           7 :         GDALDataset::SetMetadataItem("INTERLEAVE", "PIXEL", "IMAGE_STRUCTURE");
     293             :     }
     294          25 : }
     295             : 
     296             : /************************************************************************/
     297             : /*                            Identify()                                */
     298             : /************************************************************************/
     299             : 
     300       51229 : int GDALDAASDataset::Identify(GDALOpenInfo *poOpenInfo)
     301             : {
     302       51229 :     return STARTS_WITH_CI(poOpenInfo->pszFilename, "DAAS:");
     303             : }
     304             : 
     305             : /************************************************************************/
     306             : /*                        GetGeoTransform()                             */
     307             : /************************************************************************/
     308             : 
     309           1 : CPLErr GDALDAASDataset::GetGeoTransform(double *padfTransform)
     310             : {
     311           1 :     std::copy_n(m_adfGeoTransform.begin(), m_adfGeoTransform.size(),
     312             :                 padfTransform);
     313           1 :     return (m_bGotGeoTransform) ? CE_None : CE_Failure;
     314             : }
     315             : 
     316             : /************************************************************************/
     317             : /*                          GetSpatialRef()                            */
     318             : /************************************************************************/
     319             : 
     320           1 : const OGRSpatialReference *GDALDAASDataset::GetSpatialRef() const
     321             : {
     322           1 :     return m_oSRS.IsEmpty() ? nullptr : &m_oSRS;
     323             : }
     324             : 
     325             : /********************-****************************************************/
     326             : /*                             URLEscape()                              */
     327             : /************************************************************************/
     328             : 
     329          14 : static CPLString URLEscape(const CPLString &osStr)
     330             : {
     331          14 :     char *pszEscaped = CPLEscapeString(osStr.c_str(), -1, CPLES_URL);
     332          14 :     CPLString osRet(pszEscaped);
     333          14 :     CPLFree(pszEscaped);
     334          14 :     return osRet;
     335             : }
     336             : 
     337             : /************************************************************************/
     338             : /*                             GetHTTPOptions()                         */
     339             : /************************************************************************/
     340             : 
     341          72 : char **GDALDAASDataset::GetHTTPOptions()
     342             : {
     343          72 :     if (m_poParentDS)
     344           1 :         return m_poParentDS->GetHTTPOptions();
     345             : 
     346          71 :     char **papszOptions = nullptr;
     347          71 :     CPLString osHeaders;
     348          71 :     if (!m_osAccessToken.empty())
     349             :     {
     350             :         // Renew token if needed
     351           4 :         if (m_nExpirationTime != 0 && time(nullptr) >= m_nExpirationTime)
     352             :         {
     353           1 :             GetAuthorization();
     354             :         }
     355           4 :         osHeaders += "Authorization: Bearer " + m_osAccessToken;
     356             :     }
     357             :     else
     358             :     {
     359             :         const char *pszAuthorization =
     360          67 :             CPLGetConfigOption("GDAL_DAAS_AUTHORIZATION", nullptr);
     361          67 :         if (pszAuthorization)
     362           0 :             osHeaders += pszAuthorization;
     363             :     }
     364          71 :     if (!m_osXForwardUser.empty())
     365             :     {
     366           1 :         if (!osHeaders.empty())
     367           1 :             osHeaders += "\r\n";
     368           1 :         osHeaders += "X-Forwarded-User: " + m_osXForwardUser;
     369             :     }
     370          71 :     if (!osHeaders.empty())
     371             :     {
     372             :         papszOptions =
     373           4 :             CSLSetNameValue(papszOptions, "HEADERS", osHeaders.c_str());
     374             :     }
     375             :     papszOptions =
     376          71 :         CSLSetNameValue(papszOptions, "PERSISTENT", CPLSPrintf("%p", this));
     377             :     // 30 minutes
     378          71 :     papszOptions = CSLSetNameValue(papszOptions, "TIMEOUT", "1800");
     379          71 :     return papszOptions;
     380             : }
     381             : 
     382             : /************************************************************************/
     383             : /*                          DAASBackoffFactor()                         */
     384             : /************************************************************************/
     385             : 
     386             : /* Add a small amount of random jitter to avoid cyclic server stampedes */
     387           8 : static double DAASBackoffFactor(double base)
     388             : {
     389             :     // We don't need cryptographic quality randomness...
     390             :     // coverity[dont_call]
     391           8 :     return base + rand() * 0.5 / RAND_MAX;
     392             : }
     393             : 
     394             : /************************************************************************/
     395             : /*                          DAAS_CPLHTTPFetch()                         */
     396             : /************************************************************************/
     397             : 
     398          78 : static CPLHTTPResult *DAAS_CPLHTTPFetch(const char *pszURL, char **papszOptions)
     399             : {
     400             :     CPLHTTPResult *psResult;
     401          78 :     const int RETRY_COUNT = 4;
     402             :     // coverity[tainted_data]
     403             :     double dfRetryDelay =
     404          78 :         CPLAtof(CPLGetConfigOption("GDAL_DAAS_INITIAL_RETRY_DELAY", "1.0"));
     405          86 :     for (int i = 0; i <= RETRY_COUNT; i++)
     406             :     {
     407          86 :         psResult = CPLHTTPFetch(pszURL, papszOptions);
     408          86 :         if (psResult == nullptr)
     409           0 :             break;
     410             : 
     411          86 :         if (psResult->nDataLen != 0 && psResult->nStatus == 0 &&
     412          60 :             psResult->pszErrBuf == nullptr)
     413             :         {
     414             :             /* got a valid response */
     415          57 :             CPLErrorReset();
     416          57 :             break;
     417             :         }
     418             :         else
     419             :         {
     420          29 :             const char *pszErrorText =
     421          29 :                 psResult->pszErrBuf ? psResult->pszErrBuf : "(null)";
     422             : 
     423             :             /* Get HTTP status code */
     424          29 :             int nHTTPStatus = -1;
     425          29 :             if (psResult->pszErrBuf != nullptr &&
     426          28 :                 EQUALN(psResult->pszErrBuf,
     427             :                        "HTTP error code : ", strlen("HTTP error code : ")))
     428             :             {
     429             :                 nHTTPStatus =
     430          28 :                     atoi(psResult->pszErrBuf + strlen("HTTP error code : "));
     431          28 :                 if (psResult->pabyData)
     432           3 :                     pszErrorText = (const char *)psResult->pabyData;
     433             :             }
     434             : 
     435          29 :             if ((nHTTPStatus == 500 ||
     436           9 :                  (nHTTPStatus >= 502 && nHTTPStatus <= 504)) &&
     437             :                 i < RETRY_COUNT)
     438             :             {
     439           8 :                 CPLError(CE_Warning, CPLE_FileIO,
     440             :                          "Error when downloading %s,"
     441             :                          "HTTP status=%d, retrying in %.2fs : %s",
     442             :                          pszURL, nHTTPStatus, dfRetryDelay, pszErrorText);
     443           8 :                 CPLHTTPDestroyResult(psResult);
     444           8 :                 psResult = nullptr;
     445             : 
     446           8 :                 CPLSleep(dfRetryDelay);
     447           8 :                 dfRetryDelay *= DAASBackoffFactor(4);
     448             :             }
     449             :             else
     450             :             {
     451             :                 break;
     452             :             }
     453             :         }
     454             :     }
     455             : 
     456          78 :     return psResult;
     457             : }
     458             : 
     459             : /************************************************************************/
     460             : /*                           GetAuthorization()                         */
     461             : /************************************************************************/
     462             : 
     463          10 : bool GDALDAASDataset::GetAuthorization()
     464             : {
     465             :     const CPLString osClientId =
     466          10 :         CSLFetchNameValueDef(m_papszOpenOptions, "CLIENT_ID",
     467          20 :                              CPLGetConfigOption("GDAL_DAAS_CLIENT_ID", ""));
     468             :     const CPLString osAPIKey =
     469          10 :         CSLFetchNameValueDef(m_papszOpenOptions, "API_KEY",
     470          20 :                              CPLGetConfigOption("GDAL_DAAS_API_KEY", ""));
     471             :     const CPLString osAuthorization =
     472          10 :         CSLFetchNameValueDef(m_papszOpenOptions, "ACCESS_TOKEN",
     473          20 :                              CPLGetConfigOption("GDAL_DAAS_ACCESS_TOKEN", ""));
     474             :     m_osXForwardUser = CSLFetchNameValueDef(
     475          10 :         m_papszOpenOptions, "X_FORWARDED_USER",
     476          10 :         CPLGetConfigOption("GDAL_DAAS_X_FORWARDED_USER", ""));
     477             : 
     478          10 :     if (!osAuthorization.empty())
     479             :     {
     480           1 :         if (!osClientId.empty() && !osAPIKey.empty())
     481             :         {
     482           0 :             CPLError(
     483             :                 CE_Warning, CPLE_AppDefined,
     484             :                 "GDAL_DAAS_CLIENT_ID + GDAL_DAAS_API_KEY and "
     485             :                 "GDAL_DAAS_ACCESS_TOKEN defined. Only the later taken into "
     486             :                 "account");
     487             :         }
     488           1 :         m_osAccessToken = osAuthorization;
     489           1 :         return true;
     490             :     }
     491             : 
     492           9 :     if (osClientId.empty() && osAPIKey.empty())
     493             :     {
     494           0 :         CPLDebug("DAAS",
     495             :                  "Neither GDAL_DAAS_CLIENT_ID, GDAL_DAAS_API_KEY "
     496             :                  "nor GDAL_DAAS_ACCESS_TOKEN is defined. Trying without "
     497             :                  "authorization");
     498           0 :         return true;
     499             :     }
     500             : 
     501           9 :     if (osClientId.empty())
     502             :     {
     503           1 :         CPLError(CE_Failure, CPLE_AppDefined,
     504             :                  "GDAL_DAAS_API_KEY defined, but GDAL_DAAS_CLIENT_ID missing.");
     505           1 :         return false;
     506             :     }
     507             : 
     508           8 :     if (osAPIKey.empty())
     509             :     {
     510           1 :         CPLError(CE_Failure, CPLE_AppDefined,
     511             :                  "GDAL_DAAS_CLIENT_ID defined, but GDAL_DAAS_API_KEY missing.");
     512           1 :         return false;
     513             :     }
     514             : 
     515          14 :     CPLString osPostContent;
     516           7 :     osPostContent += "client_id=" + URLEscape(osClientId);
     517           7 :     osPostContent += "&apikey=" + URLEscape(osAPIKey);
     518           7 :     osPostContent += "&grant_type=api_key";
     519             : 
     520           7 :     char **papszOptions = nullptr;
     521             :     papszOptions =
     522           7 :         CSLSetNameValue(papszOptions, "POSTFIELDS", osPostContent.c_str());
     523          14 :     CPLString osHeaders("Content-Type: application/x-www-form-urlencoded");
     524           7 :     papszOptions = CSLSetNameValue(papszOptions, "HEADERS", osHeaders.c_str());
     525             :     // FIXME for server side: make sure certificates are valid
     526           7 :     papszOptions = CSLSetNameValue(papszOptions, "UNSAFESSL", "YES");
     527           7 :     CPLHTTPResult *psResult = DAAS_CPLHTTPFetch(m_osAuthURL, papszOptions);
     528           7 :     CSLDestroy(papszOptions);
     529             : 
     530           7 :     if (psResult == nullptr)
     531             :     {
     532           0 :         return false;
     533             :     }
     534             : 
     535           7 :     if (psResult->pszErrBuf != nullptr)
     536             :     {
     537           1 :         CPLError(CE_Failure, CPLE_AppDefined, "Get request %s failed: %s",
     538             :                  m_osAuthURL.c_str(),
     539           2 :                  psResult->pabyData ? CPLSPrintf("%s: %s", psResult->pszErrBuf,
     540             :                                                  reinterpret_cast<const char *>(
     541           1 :                                                      psResult->pabyData))
     542             :                                     : psResult->pszErrBuf);
     543           1 :         CPLHTTPDestroyResult(psResult);
     544           1 :         return false;
     545             :     }
     546             : 
     547           6 :     if (psResult->pabyData == nullptr)
     548             :     {
     549           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     550             :                  "Authorization request failed: "
     551             :                  "Empty content returned by server");
     552           0 :         CPLHTTPDestroyResult(psResult);
     553           0 :         return false;
     554             :     }
     555             : 
     556             :     CPLString osAuthorizationResponse(
     557          12 :         reinterpret_cast<char *>(psResult->pabyData));
     558           6 :     CPLHTTPDestroyResult(psResult);
     559             : 
     560          12 :     CPLJSONDocument oDoc;
     561           6 :     if (!oDoc.LoadMemory(osAuthorizationResponse))
     562             :     {
     563           1 :         CPLError(CE_Failure, CPLE_AppDefined,
     564             :                  "Cannont parse GetAuthorization response");
     565           1 :         return false;
     566             :     }
     567             : 
     568           5 :     m_osAccessToken = oDoc.GetRoot().GetString("access_token");
     569           5 :     if (m_osAccessToken.empty())
     570             :     {
     571           1 :         CPLError(CE_Failure, CPLE_AppDefined, "Cannot retrieve access_token");
     572           1 :         return false;
     573             :     }
     574             : 
     575           4 :     int nExpiresIn = oDoc.GetRoot().GetInteger("expires_in");
     576           4 :     if (nExpiresIn > 0)
     577             :     {
     578           2 :         m_nExpirationTime = time(nullptr) + nExpiresIn - 60;
     579             :     }
     580             : 
     581           4 :     return true;
     582             : }
     583             : 
     584             : /************************************************************************/
     585             : /*                           GetObject()                                */
     586             : /************************************************************************/
     587             : 
     588         359 : static CPLJSONObject GetObject(const CPLJSONObject &oContainer,
     589             :                                const char *pszPath,
     590             :                                CPLJSONObject::Type eExpectedType,
     591             :                                const char *pszExpectedType, bool bVerboseError,
     592             :                                bool &bError)
     593             : {
     594         718 :     CPLJSONObject oObj = oContainer.GetObj(pszPath);
     595         359 :     if (!oObj.IsValid())
     596             :     {
     597         156 :         if (bVerboseError)
     598             :         {
     599           3 :             CPLError(CE_Failure, CPLE_AppDefined, "%s missing", pszPath);
     600             :         }
     601         156 :         bError = true;
     602         156 :         oObj.Deinit();
     603         156 :         return oObj;
     604             :     }
     605         203 :     if (oObj.GetType() != eExpectedType)
     606             :     {
     607           1 :         CPLError(CE_Failure, CPLE_AppDefined, "%s not %s", pszPath,
     608             :                  pszExpectedType);
     609           1 :         bError = true;
     610           1 :         oObj.Deinit();
     611           1 :         return oObj;
     612             :     }
     613         202 :     return oObj;
     614             : }
     615             : 
     616             : /************************************************************************/
     617             : /*                          GetInteger()                                */
     618             : /************************************************************************/
     619             : 
     620          84 : static int GetInteger(const CPLJSONObject &oContainer, const char *pszPath,
     621             :                       bool bVerboseError, bool &bError)
     622             : {
     623             :     CPLJSONObject oObj =
     624             :         GetObject(oContainer, pszPath, CPLJSONObject::Type::Integer,
     625         168 :                   "an integer", bVerboseError, bError);
     626          84 :     if (!oObj.IsValid())
     627             :     {
     628          29 :         return 0;
     629             :     }
     630          55 :     return oObj.ToInteger();
     631             : }
     632             : 
     633             : /************************************************************************/
     634             : /*                          GetDouble()                                */
     635             : /************************************************************************/
     636             : 
     637          92 : static double GetDouble(const CPLJSONObject &oContainer, const char *pszPath,
     638             :                         bool bVerboseError, bool &bError)
     639             : {
     640         276 :     CPLJSONObject oObj = oContainer.GetObj(pszPath);
     641          92 :     if (!oObj.IsValid())
     642             :     {
     643          66 :         if (bVerboseError)
     644             :         {
     645          10 :             CPLError(CE_Failure, CPLE_AppDefined, "%s missing", pszPath);
     646             :         }
     647          66 :         bError = true;
     648          66 :         return 0.0;
     649             :     }
     650          28 :     if (oObj.GetType() != CPLJSONObject::Type::Integer &&
     651           2 :         oObj.GetType() != CPLJSONObject::Type::Double)
     652             :     {
     653           0 :         CPLError(CE_Failure, CPLE_AppDefined, "%s not a double", pszPath);
     654           0 :         bError = true;
     655           0 :         return 0.0;
     656             :     }
     657          26 :     return oObj.ToDouble();
     658             : }
     659             : 
     660             : /************************************************************************/
     661             : /*                          GetString()                                 */
     662             : /************************************************************************/
     663             : 
     664         275 : static CPLString GetString(const CPLJSONObject &oContainer, const char *pszPath,
     665             :                            bool bVerboseError, bool &bError)
     666             : {
     667             :     CPLJSONObject oObj =
     668             :         GetObject(oContainer, pszPath, CPLJSONObject::Type::String, "a string",
     669         550 :                   bVerboseError, bError);
     670         275 :     if (!oObj.IsValid())
     671             :     {
     672         128 :         return CPLString();
     673             :     }
     674         294 :     return oObj.ToString();
     675             : }
     676             : 
     677             : /************************************************************************/
     678             : /*                     GetGDALDataTypeFromDAASPixelType()               */
     679             : /************************************************************************/
     680             : 
     681             : static GDALDataType
     682          46 : GetGDALDataTypeFromDAASPixelType(const CPLString &osPixelType)
     683             : {
     684             :     const struct
     685             :     {
     686             :         const char *pszName;
     687             :         GDALDataType eDT;
     688          46 :     } asDataTypes[] = {
     689             :         {"Byte", GDT_Byte},       {"UInt16", GDT_UInt16},
     690             :         {"Int16", GDT_Int16},     {"UInt32", GDT_UInt32},
     691             :         {"Int32", GDT_Int32},     {"Float32", GDT_Float32},
     692             :         {"Float64", GDT_Float64},
     693             :     };
     694             : 
     695          71 :     for (size_t i = 0; i < CPL_ARRAYSIZE(asDataTypes); ++i)
     696             :     {
     697          69 :         if (osPixelType == asDataTypes[i].pszName)
     698             :         {
     699          44 :             return asDataTypes[i].eDT;
     700             :         }
     701             :     }
     702           2 :     return GDT_Unknown;
     703             : }
     704             : 
     705             : /************************************************************************/
     706             : /*                         GetImageMetadata()                           */
     707             : /************************************************************************/
     708             : 
     709          39 : bool GDALDAASDataset::GetImageMetadata()
     710             : {
     711          39 :     char **papszOptions = GetHTTPOptions();
     712             :     CPLHTTPResult *psResult =
     713          39 :         DAAS_CPLHTTPFetch(m_osGetMetadataURL, papszOptions);
     714          39 :     CSLDestroy(papszOptions);
     715          39 :     if (psResult == nullptr)
     716           0 :         return false;
     717             : 
     718          39 :     if (psResult->pszErrBuf != nullptr)
     719             :     {
     720           8 :         CPLError(CE_Failure, CPLE_AppDefined, "Get request %s failed: %s",
     721             :                  m_osGetMetadataURL.c_str(),
     722           9 :                  psResult->pabyData ? CPLSPrintf("%s: %s", psResult->pszErrBuf,
     723             :                                                  reinterpret_cast<const char *>(
     724           1 :                                                      psResult->pabyData))
     725             :                                     : psResult->pszErrBuf);
     726           8 :         CPLHTTPDestroyResult(psResult);
     727           8 :         return false;
     728             :     }
     729             : 
     730          31 :     if (psResult->pabyData == nullptr)
     731             :     {
     732           1 :         CPLError(CE_Failure, CPLE_AppDefined,
     733             :                  "Get request %s failed: "
     734             :                  "Empty content returned by server",
     735             :                  m_osGetMetadataURL.c_str());
     736           1 :         CPLHTTPDestroyResult(psResult);
     737           1 :         return false;
     738             :     }
     739             : 
     740          60 :     CPLString osResponse(reinterpret_cast<char *>(psResult->pabyData));
     741          30 :     CPLHTTPDestroyResult(psResult);
     742             : 
     743          60 :     CPLJSONDocument oDoc;
     744          30 :     CPLDebug("DAAS", "%s", osResponse.c_str());
     745          30 :     if (!oDoc.LoadMemory(osResponse))
     746             :     {
     747           1 :         CPLError(CE_Failure, CPLE_AppDefined,
     748             :                  "Cannont parse GetImageMetadata response");
     749           1 :         return false;
     750             :     }
     751             : 
     752          29 :     CPLJSONObject oProperties = oDoc.GetRoot().GetObj(
     753          87 :         "response/payload/payload/imageMetadata/properties");
     754          29 :     if (!oProperties.IsValid())
     755             :     {
     756          11 :         oProperties = oDoc.GetRoot().GetObj("properties");
     757          11 :         if (!oProperties.IsValid())
     758             :         {
     759           1 :             CPLError(CE_Failure, CPLE_AppDefined,
     760             :                      "Cannont find response/payload/payload/imageMetadata/"
     761             :                      "properties nor properties in GetImageMetadata response");
     762           1 :             return false;
     763             :         }
     764             :     }
     765             : 
     766          28 :     bool bError = false;
     767          28 :     nRasterXSize = GetInteger(oProperties, "width", true, bError);
     768          28 :     nRasterYSize = GetInteger(oProperties, "height", true, bError);
     769          28 :     if (!bError && !GDALCheckDatasetDimensions(nRasterXSize, nRasterYSize))
     770             :     {
     771           1 :         bError = true;
     772             :     }
     773             : 
     774          28 :     bool bIgnoredError = false;
     775             : 
     776          28 :     m_nActualBitDepth =
     777          28 :         GetInteger(oProperties, "actualBitDepth", false, bIgnoredError);
     778             : 
     779          28 :     bool bNoDataError = false;
     780          28 :     m_dfNoDataValue =
     781          28 :         GetDouble(oProperties, "noDataValue", false, bNoDataError);
     782          28 :     m_bHasNoData = !bNoDataError;
     783             : 
     784          84 :     CPLJSONObject oGetBufferObj = oProperties.GetObj("_links/getBuffer");
     785          28 :     if (!oGetBufferObj.IsValid())
     786             :     {
     787           1 :         CPLError(CE_Failure, CPLE_AppDefined, "%s missing", "_links/getBuffer");
     788           1 :         bError = true;
     789             :     }
     790          56 :     CPLJSONObject oGetBufferDict;
     791          28 :     oGetBufferDict.Deinit();
     792          28 :     if (oGetBufferObj.GetType() == CPLJSONObject::Type::Array)
     793             :     {
     794           0 :         auto array = oGetBufferObj.ToArray();
     795           0 :         if (array.Size() > 0)
     796             :         {
     797           0 :             oGetBufferDict = array[0];
     798             :         }
     799             :     }
     800          28 :     else if (oGetBufferObj.GetType() == CPLJSONObject::Type::Object)
     801             :     {
     802          27 :         oGetBufferDict = oGetBufferObj;
     803             :     }
     804          28 :     CPL_IGNORE_RET_VAL(oGetBufferObj);
     805          28 :     if (!oGetBufferDict.IsValid())
     806             :     {
     807           1 :         CPLError(CE_Failure, CPLE_AppDefined, "%s missing",
     808             :                  "_links/getBuffer/href");
     809           1 :         bError = true;
     810             :     }
     811             :     else
     812             :     {
     813          27 :         m_osGetBufferURL = GetString(oGetBufferDict, "href", true, bError);
     814             :     }
     815             : 
     816             : #ifndef REMOVE_THAT_LEGACY_CODE
     817          28 :     if (!STARTS_WITH_CI(m_osGetMetadataURL, "https://192.168.") &&
     818          56 :         !STARTS_WITH_CI(m_osGetMetadataURL, "http://192.168.") &&
     819          28 :         STARTS_WITH_CI(m_osGetBufferURL, "http://192.168."))
     820             :     {
     821           0 :         size_t nPosDaas = m_osGetMetadataURL.find("/daas/");
     822           0 :         size_t nPosImages = m_osGetMetadataURL.find("/images/");
     823           0 :         if (nPosDaas != std::string::npos && nPosImages != std::string::npos)
     824             :         {
     825             :             m_osGetBufferURL =
     826           0 :                 m_osGetMetadataURL.substr(0, nPosDaas) + "/daas/images/" +
     827           0 :                 m_osGetMetadataURL.substr(nPosImages + strlen("/images/")) +
     828           0 :                 "/buffer";
     829             :         }
     830             :     }
     831             : #endif
     832             : 
     833          84 :     CPLJSONArray oGTArray = oProperties.GetArray("geotransform");
     834          28 :     if (oGTArray.IsValid() && oGTArray.Size() == 6)
     835             :     {
     836           3 :         m_bGotGeoTransform = true;
     837          21 :         for (int i = 0; i < 6; i++)
     838             :         {
     839          18 :             m_adfGeoTransform[i] = oGTArray[i].ToDouble();
     840             :         }
     841             :     }
     842             : 
     843          84 :     CPLJSONArray oBandArray = oProperties.GetArray("bands");
     844          28 :     if (!oBandArray.IsValid() || oBandArray.Size() == 0)
     845             :     {
     846           1 :         CPLError(CE_Failure, CPLE_AppDefined, "Missing or empty bands array");
     847           1 :         bError = true;
     848             :     }
     849             :     else
     850             :     {
     851          69 :         for (int i = 0; i < oBandArray.Size(); ++i)
     852             :         {
     853          42 :             CPLJSONObject oBandObj = oBandArray[i];
     854          42 :             if (oBandObj.GetType() == CPLJSONObject::Type::Object)
     855             :             {
     856          42 :                 GDALDAASBandDesc oDesc;
     857          42 :                 oDesc.nIndex = i + 1;
     858          42 :                 oDesc.osName = GetString(oBandObj, "name", true, bError);
     859             :                 oDesc.osDescription =
     860          42 :                     GetString(oBandObj, "description", false, bIgnoredError);
     861          84 :                 oDesc.osColorInterp = GetString(oBandObj, "colorInterpretation",
     862          42 :                                                 false, bIgnoredError);
     863          42 :                 oDesc.bIsMask = oBandObj.GetBool("isMask");
     864             : 
     865             :                 const CPLString osPixelType(
     866          42 :                     GetString(oBandObj, "pixelType", true, bError));
     867          42 :                 oDesc.eDT = GetGDALDataTypeFromDAASPixelType(osPixelType);
     868          42 :                 if (oDesc.eDT == GDT_Unknown)
     869             :                 {
     870           2 :                     CPLError(CE_Failure, CPLE_NotSupported,
     871             :                              "Unsupported value pixelType = '%s'",
     872             :                              osPixelType.c_str());
     873           2 :                     bError = true;
     874             :                 }
     875          42 :                 if (i == 0)
     876             :                 {
     877          27 :                     m_eDT = oDesc.eDT;
     878             :                 }
     879             : 
     880          42 :                 if (!CPLFetchBool(m_papszOpenOptions, "MASKS", true) &&
     881           0 :                     oDesc.bIsMask)
     882             :                 {
     883           0 :                     continue;
     884             :                 }
     885          43 :                 if (oDesc.osColorInterp == "MAIN_MASK" &&
     886           1 :                     m_osMainMaskName.empty())
     887             :                 {
     888           1 :                     m_nMainMaskBandIndex = i + 1;
     889           1 :                     m_osMainMaskName = oDesc.osName;
     890             :                 }
     891             :                 else
     892             :                 {
     893          41 :                     m_aoBandDesc.push_back(oDesc);
     894             :                 }
     895             :             }
     896             :             else
     897             :             {
     898           0 :                 CPLError(CE_Failure, CPLE_AppDefined,
     899             :                          "Invalid bands[] element");
     900           0 :                 bError = true;
     901             :             }
     902             :         }
     903             :     }
     904             : 
     905          28 :     ReadSRS(oProperties);
     906             : 
     907          28 :     ReadRPCs(oProperties);
     908             : 
     909             :     // Collect other metadata
     910         183 :     for (const auto &oObj : oProperties.GetChildren())
     911             :     {
     912         310 :         const CPLString &osName(oObj.GetName());
     913         155 :         const auto &oType(oObj.GetType());
     914         465 :         if (osName != "aoiFactor" && osName != "crsCode" &&
     915         465 :             osName != "nbBands" && osName != "nbBits" && osName != "nBits" &&
     916         434 :             osName != "actualBitDepth" && osName != "width" &&
     917         322 :             osName != "height" && osName != "noDataValue" && osName != "step" &&
     918         194 :             osName != "pixelType" && oObj.IsValid() &&
     919          97 :             oType != CPLJSONObject::Type::Null &&
     920         377 :             oType != CPLJSONObject::Type::Array &&
     921          67 :             oType != CPLJSONObject::Type::Object)
     922             :         {
     923           8 :             SetMetadataItem(osName.c_str(), oObj.ToString().c_str());
     924             :         }
     925             :     }
     926             : 
     927             :     // Metadata for IMAGERY domain
     928             :     CPLString osAcquisitionDate(
     929          56 :         GetString(oProperties, "acquisitionDate", false, bIgnoredError));
     930          28 :     if (!osAcquisitionDate.empty())
     931             :     {
     932           2 :         int iYear = 0;
     933           2 :         int iMonth = 0;
     934           2 :         int iDay = 0;
     935           2 :         int iHours = 0;
     936           2 :         int iMin = 0;
     937           2 :         int iSec = 0;
     938             :         const int r =
     939           2 :             sscanf(osAcquisitionDate.c_str(), "%d-%d-%dT%d:%d:%d.%*dZ", &iYear,
     940             :                    &iMonth, &iDay, &iHours, &iMin, &iSec);
     941           2 :         if (r == 6)
     942             :         {
     943           2 :             SetMetadataItem(MD_NAME_ACQDATETIME,
     944             :                             CPLSPrintf("%04d-%02d-%02d %02d:%02d:%02d", iYear,
     945             :                                        iMonth, iDay, iHours, iMin, iSec),
     946           2 :                             MD_DOMAIN_IMAGERY);
     947             :         }
     948             :     }
     949             : 
     950          28 :     bIgnoredError = false;
     951             :     double dfCloudCover =
     952          28 :         GetDouble(oProperties, "cloudCover", false, bIgnoredError);
     953          28 :     if (!bIgnoredError)
     954             :     {
     955           2 :         SetMetadataItem(MD_NAME_CLOUDCOVER, CPLSPrintf("%.2f", dfCloudCover),
     956           2 :                         MD_DOMAIN_IMAGERY);
     957             :     }
     958             : 
     959             :     CPLString osSatellite(
     960          28 :         GetString(oProperties, "satellite", false, bIgnoredError));
     961          28 :     if (!osSatellite.empty())
     962             :     {
     963           2 :         SetMetadataItem(MD_NAME_SATELLITE, osSatellite.c_str(),
     964           2 :                         MD_DOMAIN_IMAGERY);
     965             :     }
     966             : 
     967          28 :     return !bError;
     968             : }
     969             : 
     970             : /************************************************************************/
     971             : /*                            ReadSRS()                                 */
     972             : /************************************************************************/
     973             : 
     974          28 : void GDALDAASDataset::ReadSRS(const CPLJSONObject &oProperties)
     975             : {
     976          84 :     CPLJSONArray oSRSArray = oProperties.GetArray("srsExpression/names");
     977          28 :     if (oSRSArray.IsValid())
     978             :     {
     979          12 :         for (int i = 0; i < oSRSArray.Size(); ++i)
     980             :         {
     981          20 :             CPLJSONObject oSRSObj = oSRSArray[i];
     982          10 :             if (oSRSObj.GetType() == CPLJSONObject::Type::Object)
     983             :             {
     984          10 :                 bool bError = false;
     985             :                 const std::string osType(
     986          20 :                     GetString(oSRSObj, "type", true, bError));
     987             :                 const std::string osValue(
     988          20 :                     GetString(oSRSObj, "value", true, bError));
     989             :                 // Use urn in priority
     990          10 :                 if (osType == "urn" && !osValue.empty())
     991             :                 {
     992           2 :                     m_osSRSType = osType;
     993           2 :                     m_osSRSValue = osValue;
     994             :                 }
     995             :                 // Use proj4 if urn not already set
     996          12 :                 else if (osType == "proj4" && !osValue.empty() &&
     997           4 :                          m_osSRSType != "urn")
     998             :                 {
     999           2 :                     m_osSRSType = osType;
    1000           2 :                     m_osSRSValue = osValue;
    1001             :                 }
    1002             :                 // If no SRS set, take the first one
    1003           8 :                 else if (m_osSRSValue.empty() && !osType.empty() &&
    1004           2 :                          !osValue.empty())
    1005             :                 {
    1006           2 :                     m_osSRSType = osType;
    1007           2 :                     m_osSRSValue = osValue;
    1008             :                 }
    1009             :             }
    1010             :         }
    1011             :     }
    1012             :     else
    1013             :     {
    1014          78 :         auto osCrsCode = oProperties.GetString("crsCode");
    1015          26 :         if (!osCrsCode.empty())
    1016             :         {
    1017           0 :             m_osSRSType = "urn";
    1018           0 :             m_osSRSValue = osCrsCode;
    1019             :         }
    1020             :     }
    1021             : 
    1022          28 :     if (m_osSRSType == "urn" || m_osSRSType == "proj4")
    1023             :     {
    1024           2 :         m_oSRS.SetFromUserInput(
    1025             :             m_osSRSValue,
    1026             :             OGRSpatialReference::SET_FROM_USER_INPUT_LIMITATIONS_get());
    1027             :     }
    1028          28 : }
    1029             : 
    1030             : /************************************************************************/
    1031             : /*                            ReadRPCs()                                */
    1032             : /************************************************************************/
    1033             : 
    1034          28 : void GDALDAASDataset::ReadRPCs(const CPLJSONObject &oProperties)
    1035             : {
    1036          84 :     CPLJSONObject oRPC = oProperties.GetObj("rpc");
    1037          28 :     if (oRPC.IsValid())
    1038             :     {
    1039           3 :         bool bRPCError = false;
    1040           6 :         CPLStringList aoRPC;
    1041             : 
    1042             :         const struct
    1043             :         {
    1044             :             const char *pszJsonName;
    1045             :             const char *pszGDALName;
    1046           3 :         } asRPCSingleValues[] = {
    1047             :             {"errBias", RPC_ERR_BIAS},     {"errRand", RPC_ERR_RAND},
    1048             :             {"sampOff", RPC_SAMP_OFF},     {"lineOff", RPC_LINE_OFF},
    1049             :             {"latOff", RPC_LAT_OFF},       {"longOff", RPC_LONG_OFF},
    1050             :             {"heightOff", RPC_HEIGHT_OFF}, {"lineScale", RPC_LINE_SCALE},
    1051             :             {"sampScale", RPC_SAMP_SCALE}, {"latScale", RPC_LAT_SCALE},
    1052             :             {"longScale", RPC_LONG_SCALE}, {"heightScale", RPC_HEIGHT_SCALE},
    1053             :         };
    1054             : 
    1055          39 :         for (size_t i = 0; i < CPL_ARRAYSIZE(asRPCSingleValues); ++i)
    1056             :         {
    1057          36 :             bool bRPCErrorTmp = false;
    1058          36 :             const bool bVerboseError =
    1059          69 :                 !(strcmp(asRPCSingleValues[i].pszGDALName, RPC_ERR_BIAS) == 0 ||
    1060          33 :                   strcmp(asRPCSingleValues[i].pszGDALName, RPC_ERR_RAND) == 0);
    1061          36 :             double dfRPCVal = GetDouble(oRPC, asRPCSingleValues[i].pszJsonName,
    1062             :                                         bVerboseError, bRPCErrorTmp);
    1063          36 :             if (bRPCErrorTmp)
    1064             :             {
    1065          14 :                 if (bVerboseError)
    1066             :                 {
    1067          10 :                     bRPCError = true;
    1068             :                 }
    1069          14 :                 continue;
    1070             :             }
    1071          22 :             aoRPC.SetNameValue(asRPCSingleValues[i].pszGDALName,
    1072          22 :                                CPLSPrintf("%.17g", dfRPCVal));
    1073             :         }
    1074             : 
    1075             :         const struct
    1076             :         {
    1077             :             const char *pszJsonName;
    1078             :             const char *pszGDALName;
    1079           3 :         } asRPCArrayValues[] = {
    1080             :             {"lineNumCoeff", RPC_LINE_NUM_COEFF},
    1081             :             {"lineDenCoeff", RPC_LINE_DEN_COEFF},
    1082             :             {"sampNumCoeff", RPC_SAMP_NUM_COEFF},
    1083             :             {"sampDenCoeff", RPC_SAMP_DEN_COEFF},
    1084             :         };
    1085             : 
    1086          15 :         for (size_t i = 0; i < CPL_ARRAYSIZE(asRPCArrayValues); ++i)
    1087             :         {
    1088             :             CPLJSONArray oRPCArray =
    1089          36 :                 oRPC.GetArray(asRPCArrayValues[i].pszJsonName);
    1090          12 :             if (oRPCArray.IsValid() && oRPCArray.Size() == 20)
    1091             :             {
    1092          16 :                 CPLString osVal;
    1093         168 :                 for (int j = 0; j < 20; j++)
    1094             :                 {
    1095         160 :                     if (j > 0)
    1096         152 :                         osVal += " ";
    1097         160 :                     osVal += CPLSPrintf("%.17g", oRPCArray[j].ToDouble());
    1098             :                 }
    1099           8 :                 aoRPC.SetNameValue(asRPCArrayValues[i].pszGDALName,
    1100           8 :                                    osVal.c_str());
    1101             :             }
    1102             :             else
    1103             :             {
    1104           4 :                 CPLError(CE_Failure, CPLE_AppDefined, "Cannot find %s",
    1105           4 :                          asRPCArrayValues[i].pszJsonName);
    1106             :             }
    1107             :         }
    1108           3 :         if (!bRPCError)
    1109             :         {
    1110           2 :             SetMetadata(aoRPC.List(), "RPC");
    1111             :         }
    1112             :     }
    1113          28 : }
    1114             : 
    1115             : /************************************************************************/
    1116             : /*                      SetupServerSideReprojection()                   */
    1117             : /************************************************************************/
    1118             : 
    1119           0 : bool GDALDAASDataset::SetupServerSideReprojection(const char *pszTargetSRS)
    1120             : {
    1121           0 :     if (m_oSRS.IsEmpty() || !m_bGotGeoTransform)
    1122             :     {
    1123           0 :         CPLError(CE_Failure, CPLE_AppDefined,
    1124             :                  "TARGET_SRS is specified, but projection and/or "
    1125             :                  "geotransform are missing in image metadata");
    1126           0 :         return false;
    1127             :     }
    1128             : 
    1129           0 :     OGRSpatialReference oSRS;
    1130           0 :     if (oSRS.SetFromUserInput(
    1131             :             pszTargetSRS,
    1132           0 :             OGRSpatialReference::SET_FROM_USER_INPUT_LIMITATIONS_get()) !=
    1133             :         OGRERR_NONE)
    1134             :     {
    1135           0 :         CPLError(CE_Failure, CPLE_AppDefined, "Invalid TARGET_SRS value");
    1136           0 :         return false;
    1137             :     }
    1138             : 
    1139             :     // Check that we can find the EPSG code as we will need to
    1140             :     // provide as a urn to getBuffer
    1141           0 :     const char *pszAuthorityCode = oSRS.GetAuthorityCode(nullptr);
    1142           0 :     const char *pszAuthorityName = oSRS.GetAuthorityName(nullptr);
    1143           0 :     if (pszAuthorityName == nullptr || !EQUAL(pszAuthorityName, "EPSG") ||
    1144             :         pszAuthorityCode == nullptr)
    1145             :     {
    1146           0 :         CPLError(CE_Failure, CPLE_AppDefined,
    1147             :                  "TARGET_SRS cannot be identified to a EPSG code");
    1148           0 :         return false;
    1149             :     }
    1150             : 
    1151           0 :     CPLString osTargetEPSGCode = CPLString("epsg:") + pszAuthorityCode;
    1152             : 
    1153           0 :     char *pszWKT = nullptr;
    1154           0 :     oSRS.exportToWkt(&pszWKT);
    1155           0 :     char **papszTO = CSLSetNameValue(nullptr, "DST_SRS", pszWKT);
    1156           0 :     CPLFree(pszWKT);
    1157             : 
    1158             :     void *hTransformArg =
    1159           0 :         GDALCreateGenImgProjTransformer2(this, nullptr, papszTO);
    1160           0 :     if (hTransformArg == nullptr)
    1161             :     {
    1162           0 :         CSLDestroy(papszTO);
    1163           0 :         return false;
    1164             :     }
    1165             : 
    1166           0 :     GDALTransformerInfo *psInfo = (GDALTransformerInfo *)hTransformArg;
    1167             :     double adfGeoTransform[6];
    1168             :     double adfExtent[4];
    1169             :     int nXSize, nYSize;
    1170             : 
    1171           0 :     if (GDALSuggestedWarpOutput2(this, psInfo->pfnTransform, hTransformArg,
    1172             :                                  adfGeoTransform, &nXSize, &nYSize, adfExtent,
    1173           0 :                                  0) != CE_None)
    1174             :     {
    1175           0 :         CPLError(CE_Failure, CPLE_AppDefined,
    1176             :                  "Cannot find extent in specified TARGET_SRS");
    1177           0 :         CSLDestroy(papszTO);
    1178           0 :         GDALDestroyGenImgProjTransformer(hTransformArg);
    1179           0 :         return false;
    1180             :     }
    1181             : 
    1182           0 :     GDALDestroyGenImgProjTransformer(hTransformArg);
    1183             : 
    1184           0 :     std::copy_n(adfGeoTransform, 6, m_adfGeoTransform.begin());
    1185           0 :     m_bRequestInGeoreferencedCoordinates = true;
    1186           0 :     m_osSRSType = "epsg";
    1187           0 :     m_osSRSValue = std::move(osTargetEPSGCode);
    1188           0 :     m_oSRS = std::move(oSRS);
    1189           0 :     nRasterXSize = nXSize;
    1190           0 :     nRasterYSize = nYSize;
    1191           0 :     return true;
    1192             : }
    1193             : 
    1194             : /************************************************************************/
    1195             : /*                              Open()                                  */
    1196             : /************************************************************************/
    1197             : 
    1198          45 : bool GDALDAASDataset::Open(GDALOpenInfo *poOpenInfo)
    1199             : {
    1200          45 :     m_papszOpenOptions = CSLDuplicate(poOpenInfo->papszOpenOptions);
    1201             :     m_osGetMetadataURL =
    1202          45 :         CSLFetchNameValueDef(poOpenInfo->papszOpenOptions, "GET_METADATA_URL",
    1203          45 :                              poOpenInfo->pszFilename + strlen("DAAS:"));
    1204          45 :     if (m_osGetMetadataURL.empty())
    1205             :     {
    1206           1 :         CPLError(CE_Failure, CPLE_AppDefined, "GET_METADATA_URL is missing");
    1207           1 :         return false;
    1208             :     }
    1209          44 :     m_nBlockSize =
    1210          44 :         std::max(knMIN_BLOCKSIZE,
    1211             :                  std::min(knMAX_BLOCKSIZE,
    1212          44 :                           atoi(CSLFetchNameValueDef(
    1213          44 :                               poOpenInfo->papszOpenOptions, "BLOCK_SIZE",
    1214          44 :                               CPLSPrintf("%d", m_nBlockSize)))));
    1215          44 :     m_nServerByteLimit =
    1216          44 :         atoi(CPLGetConfigOption("GDAL_DAAS_SERVER_BYTE_LIMIT",
    1217             :                                 CPLSPrintf("%d", knDEFAULT_SERVER_BYTE_LIMIT)));
    1218             : 
    1219          53 :     if (CPLTestBool(CPLGetConfigOption("GDAL_DAAS_PERFORM_AUTH", "YES")) &&
    1220           9 :         !GetAuthorization())
    1221           5 :         return false;
    1222          39 :     if (!GetImageMetadata())
    1223          19 :         return false;
    1224             : 
    1225          20 :     const char *pszFormat = CSLFetchNameValueDef(poOpenInfo->papszOpenOptions,
    1226             :                                                  "PIXEL_ENCODING", "AUTO");
    1227          20 :     if (EQUAL(pszFormat, "AUTO"))
    1228             :     {
    1229           9 :         if ((m_aoBandDesc.size() == 1 || m_aoBandDesc.size() == 3 ||
    1230          14 :              m_aoBandDesc.size() == 4) &&
    1231           7 :             m_eDT == GDT_Byte)
    1232             :         {
    1233           7 :             m_eFormat = Format::PNG;
    1234             :         }
    1235             :         else
    1236             :         {
    1237           0 :             m_eFormat = Format::RAW;
    1238             :         }
    1239             :     }
    1240          13 :     else if (EQUAL(pszFormat, "RAW"))
    1241             :     {
    1242           6 :         m_eFormat = Format::RAW;
    1243             :     }
    1244           7 :     else if (EQUAL(pszFormat, "PNG"))
    1245             :     {
    1246           3 :         if ((m_aoBandDesc.size() == 1 || m_aoBandDesc.size() == 3 ||
    1247           4 :              m_aoBandDesc.size() == 4) &&
    1248           2 :             m_eDT == GDT_Byte)
    1249             :         {
    1250           1 :             m_eFormat = Format::PNG;
    1251             :         }
    1252             :         else
    1253             :         {
    1254           1 :             CPLError(CE_Warning, CPLE_AppDefined,
    1255             :                      "PNG only supported for 1, 3 or 4-band Byte dataset. "
    1256             :                      "Falling back to RAW");
    1257           1 :             m_eFormat = Format::RAW;
    1258             :         }
    1259             :     }
    1260           5 :     else if (EQUAL(pszFormat, "JPEG"))
    1261             :     {
    1262           4 :         if ((m_aoBandDesc.size() == 1 || m_aoBandDesc.size() == 3) &&
    1263           2 :             m_eDT == GDT_Byte)
    1264             :         {
    1265           1 :             m_eFormat = Format::JPEG;
    1266             :         }
    1267             :         else
    1268             :         {
    1269           1 :             CPLError(CE_Warning, CPLE_AppDefined,
    1270             :                      "JPEG only supported for 1 or 3-band Byte dataset. "
    1271             :                      "Falling back to RAW");
    1272           1 :             m_eFormat = Format::RAW;
    1273             :         }
    1274             :     }
    1275           3 :     else if (EQUAL(pszFormat, "JPEG2000"))
    1276             :     {
    1277           2 :         if (m_eDT != GDT_Float32 && m_eDT != GDT_Float64)
    1278             :         {
    1279           1 :             m_eFormat = Format::JPEG2000;
    1280             :         }
    1281             :         else
    1282             :         {
    1283           1 :             CPLError(CE_Warning, CPLE_AppDefined,
    1284             :                      "JPEG2000 only supported for integer datatype dataset. "
    1285             :                      "Falling back to RAW");
    1286           1 :             m_eFormat = Format::RAW;
    1287             :         }
    1288             :     }
    1289             :     else
    1290             :     {
    1291           1 :         CPLError(CE_Failure, CPLE_NotSupported, "Unsupported PIXEL_ENCODING=%s",
    1292             :                  pszFormat);
    1293           1 :         return false;
    1294             :     }
    1295             : 
    1296             :     const char *pszTargetSRS =
    1297          19 :         CSLFetchNameValue(poOpenInfo->papszOpenOptions, "TARGET_SRS");
    1298          19 :     if (pszTargetSRS)
    1299             :     {
    1300           0 :         if (!SetupServerSideReprojection(pszTargetSRS))
    1301             :         {
    1302           0 :             return false;
    1303             :         }
    1304             :     }
    1305             : 
    1306          19 :     InstantiateBands();
    1307             : 
    1308             :     // Instantiate overviews
    1309          19 :     int iOvr = 0;
    1310          25 :     while ((nRasterXSize >> iOvr) > 256 || (nRasterYSize >> iOvr) > 256)
    1311             :     {
    1312           6 :         iOvr++;
    1313           6 :         if ((nRasterXSize >> iOvr) == 0 || (nRasterYSize >> iOvr) == 0)
    1314             :         {
    1315             :             break;
    1316             :         }
    1317           6 :         m_apoOverviewDS.push_back(
    1318          12 :             std::make_unique<GDALDAASDataset>(this, iOvr));
    1319             :     }
    1320             : 
    1321          19 :     return true;
    1322             : }
    1323             : 
    1324          45 : GDALDataset *GDALDAASDataset::OpenStatic(GDALOpenInfo *poOpenInfo)
    1325             : {
    1326          45 :     if (!Identify(poOpenInfo))
    1327           0 :         return nullptr;
    1328             : 
    1329          90 :     auto poDS = std::make_unique<GDALDAASDataset>();
    1330          45 :     if (!poDS->Open(poOpenInfo))
    1331          26 :         return nullptr;
    1332          19 :     return poDS.release();
    1333             : }
    1334             : 
    1335             : /************************************************************************/
    1336             : /*                       GDALDAASRasterBand()                          */
    1337             : /************************************************************************/
    1338             : 
    1339          40 : GDALDAASRasterBand::GDALDAASRasterBand(GDALDAASDataset *poDSIn, int nBandIn,
    1340          40 :                                        const GDALDAASBandDesc &oBandDesc)
    1341             : {
    1342          40 :     poDS = poDSIn;
    1343          40 :     nBand = nBandIn;
    1344          40 :     eDataType = poDSIn->m_eDT;
    1345          40 :     nRasterXSize = poDSIn->GetRasterXSize();
    1346          40 :     nRasterYSize = poDSIn->GetRasterYSize();
    1347          40 :     nBlockXSize = poDSIn->m_nBlockSize;
    1348          40 :     nBlockYSize = poDSIn->m_nBlockSize;
    1349          40 :     m_nSrcIndex = oBandDesc.nIndex;
    1350             : 
    1351          40 :     SetDescription(oBandDesc.osName);
    1352          40 :     if (!oBandDesc.osDescription.empty())
    1353             :     {
    1354           4 :         SetMetadataItem("DESCRIPTION", oBandDesc.osDescription);
    1355             :     }
    1356             : 
    1357             :     const struct
    1358             :     {
    1359             :         const char *pszName;
    1360             :         GDALColorInterp eColorInterp;
    1361          40 :     } asColorInterpretations[] = {
    1362             :         {"RED", GCI_RedBand},     {"GREEN", GCI_GreenBand},
    1363             :         {"BLUE", GCI_BlueBand},   {"GRAY", GCI_GrayIndex},
    1364             :         {"ALPHA", GCI_AlphaBand}, {"UNDEFINED", GCI_Undefined},
    1365             :     };
    1366             : 
    1367         268 :     for (size_t i = 0; i < CPL_ARRAYSIZE(asColorInterpretations); ++i)
    1368             :     {
    1369         232 :         if (EQUAL(oBandDesc.osColorInterp, asColorInterpretations[i].pszName))
    1370             :         {
    1371           4 :             m_eColorInterp = asColorInterpretations[i].eColorInterp;
    1372           4 :             break;
    1373             :         }
    1374             :     }
    1375          40 :     if (!oBandDesc.osColorInterp.empty() &&
    1376          44 :         !EQUAL(oBandDesc.osColorInterp, "UNDEFINED") &&
    1377           4 :         m_eColorInterp != GCI_Undefined)
    1378             :     {
    1379           4 :         SetMetadataItem("COLOR_INTERPRETATION", oBandDesc.osColorInterp);
    1380             :     }
    1381             : 
    1382          40 :     if (poDSIn->m_nActualBitDepth != 0 && poDSIn->m_nActualBitDepth != 8 &&
    1383           2 :         poDSIn->m_nActualBitDepth != 16 && poDSIn->m_nActualBitDepth != 32 &&
    1384           2 :         poDSIn->m_nActualBitDepth != 64)
    1385             :     {
    1386           2 :         SetMetadataItem("NBITS", CPLSPrintf("%d", poDSIn->m_nActualBitDepth),
    1387             :                         "IMAGE_STRUCTURE");
    1388             :     }
    1389          40 : }
    1390             : 
    1391             : /************************************************************************/
    1392             : /*                         GetNoDataValue()                             */
    1393             : /************************************************************************/
    1394             : 
    1395           2 : double GDALDAASRasterBand::GetNoDataValue(int *pbHasNoData)
    1396             : {
    1397           2 :     GDALDAASDataset *poGDS = reinterpret_cast<GDALDAASDataset *>(poDS);
    1398           2 :     if (poGDS->m_bHasNoData)
    1399             :     {
    1400           1 :         if (pbHasNoData)
    1401           1 :             *pbHasNoData = true;
    1402           1 :         return poGDS->m_dfNoDataValue;
    1403             :     }
    1404           1 :     if (pbHasNoData)
    1405           1 :         *pbHasNoData = false;
    1406           1 :     return 0.0;
    1407             : }
    1408             : 
    1409             : /************************************************************************/
    1410             : /*                       GetColorInterpretation()                       */
    1411             : /************************************************************************/
    1412             : 
    1413           1 : GDALColorInterp GDALDAASRasterBand::GetColorInterpretation()
    1414             : {
    1415           1 :     return m_eColorInterp;
    1416             : }
    1417             : 
    1418             : /************************************************************************/
    1419             : /*                            GetMaskBand()                             */
    1420             : /************************************************************************/
    1421             : 
    1422           3 : GDALRasterBand *GDALDAASRasterBand::GetMaskBand()
    1423             : {
    1424           3 :     GDALDAASDataset *poGDS = reinterpret_cast<GDALDAASDataset *>(poDS);
    1425           3 :     if (poGDS->m_poMaskBand)
    1426           3 :         return poGDS->m_poMaskBand;
    1427           0 :     return GDALRasterBand::GetMaskBand();
    1428             : }
    1429             : 
    1430             : /************************************************************************/
    1431             : /*                           GetMaskFlags()                             */
    1432             : /************************************************************************/
    1433             : 
    1434           1 : int GDALDAASRasterBand::GetMaskFlags()
    1435             : {
    1436           1 :     GDALDAASDataset *poGDS = reinterpret_cast<GDALDAASDataset *>(poDS);
    1437           1 :     if (poGDS->m_poMaskBand)
    1438           1 :         return GMF_PER_DATASET;
    1439           0 :     return GDALRasterBand::GetMaskFlags();
    1440             : }
    1441             : 
    1442             : /************************************************************************/
    1443             : /*                      CanSpatiallySplit()                             */
    1444             : /************************************************************************/
    1445             : 
    1446          32 : static bool CanSpatiallySplit(GUInt32 nRetryFlags, int nXOff, int nYOff,
    1447             :                               int nXSize, int nYSize, int nBufXSize,
    1448             :                               int nBufYSize, int nBlockXSize, int nBlockYSize,
    1449             :                               GSpacing nPixelSpace, GSpacing nLineSpace,
    1450             :                               int &nXOff1, int &nYOff1, int &nXSize1,
    1451             :                               int &nYSize1, int &nXOff2, int &nYOff2,
    1452             :                               int &nXSize2, int &nYSize2, GSpacing &nDataShift2)
    1453             : {
    1454          32 :     if ((nRetryFlags & RETRY_SPATIAL_SPLIT) && nXSize == nBufXSize &&
    1455           1 :         nYSize == nBufYSize && nYSize > nBlockYSize)
    1456             :     {
    1457             :         int nHalf =
    1458           0 :             std::max(nBlockYSize, ((nYSize / 2) / nBlockYSize) * nBlockYSize);
    1459           0 :         nXOff1 = nXOff;
    1460           0 :         nYOff1 = nYOff;
    1461           0 :         nXSize1 = nXSize;
    1462           0 :         nYSize1 = nHalf;
    1463           0 :         nXOff2 = nXOff;
    1464           0 :         nYOff2 = nYOff + nHalf;
    1465           0 :         nXSize2 = nXSize;
    1466           0 :         nYSize2 = nYSize - nHalf;
    1467           0 :         nDataShift2 = nHalf * nLineSpace;
    1468           0 :         return true;
    1469             :     }
    1470          32 :     else if ((nRetryFlags & RETRY_SPATIAL_SPLIT) && nXSize == nBufXSize &&
    1471           1 :              nYSize == nBufYSize && nXSize > nBlockXSize)
    1472             :     {
    1473             :         int nHalf =
    1474           1 :             std::max(nBlockXSize, ((nXSize / 2) / nBlockXSize) * nBlockXSize);
    1475           1 :         nXOff1 = nXOff;
    1476           1 :         nYOff1 = nYOff;
    1477           1 :         nXSize1 = nHalf;
    1478           1 :         nYSize1 = nYSize;
    1479           1 :         nXOff2 = nXOff + nHalf;
    1480           1 :         nYOff2 = nYOff;
    1481           1 :         nXSize2 = nXSize - nHalf;
    1482           1 :         nYSize2 = nYSize;
    1483           1 :         nDataShift2 = nHalf * nPixelSpace;
    1484           1 :         return true;
    1485             :     }
    1486          31 :     return false;
    1487             : }
    1488             : 
    1489             : /************************************************************************/
    1490             : /*                           IRasterIO()                                */
    1491             : /************************************************************************/
    1492             : 
    1493           4 : CPLErr GDALDAASDataset::IRasterIO(GDALRWFlag eRWFlag, int nXOff, int nYOff,
    1494             :                                   int nXSize, int nYSize, void *pData,
    1495             :                                   int nBufXSize, int nBufYSize,
    1496             :                                   GDALDataType eBufType, int nBandCount,
    1497             :                                   BANDMAP_TYPE panBandMap, GSpacing nPixelSpace,
    1498             :                                   GSpacing nLineSpace, GSpacing nBandSpace,
    1499             :                                   GDALRasterIOExtraArg *psExtraArg)
    1500             : {
    1501           4 :     m_eCurrentResampleAlg = psExtraArg->eResampleAlg;
    1502             : 
    1503             :     /* ==================================================================== */
    1504             :     /*      Do we have overviews that would be appropriate to satisfy       */
    1505             :     /*      this request?                                                   */
    1506             :     /* ==================================================================== */
    1507           4 :     if ((nBufXSize < nXSize || nBufYSize < nYSize) &&
    1508           8 :         GetRasterBand(1)->GetOverviewCount() > 0 && eRWFlag == GF_Read)
    1509             :     {
    1510             :         GDALRasterIOExtraArg sExtraArg;
    1511           0 :         GDALCopyRasterIOExtraArg(&sExtraArg, psExtraArg);
    1512             : 
    1513           0 :         const int nOverview = GDALBandGetBestOverviewLevel2(
    1514             :             GetRasterBand(1), nXOff, nYOff, nXSize, nYSize, nBufXSize,
    1515             :             nBufYSize, &sExtraArg);
    1516           0 :         if (nOverview >= 0)
    1517             :         {
    1518             :             GDALRasterBand *poOverviewBand =
    1519           0 :                 GetRasterBand(1)->GetOverview(nOverview);
    1520           0 :             if (poOverviewBand == nullptr ||
    1521           0 :                 poOverviewBand->GetDataset() == nullptr)
    1522             :             {
    1523           0 :                 return CE_Failure;
    1524             :             }
    1525             : 
    1526           0 :             return poOverviewBand->GetDataset()->RasterIO(
    1527             :                 eRWFlag, nXOff, nYOff, nXSize, nYSize, pData, nBufXSize,
    1528             :                 nBufYSize, eBufType, nBandCount, panBandMap, nPixelSpace,
    1529           0 :                 nLineSpace, nBandSpace, &sExtraArg);
    1530             :         }
    1531             :     }
    1532             : 
    1533             :     GDALDAASRasterBand *poBand =
    1534           4 :         cpl::down_cast<GDALDAASRasterBand *>(GetRasterBand(1));
    1535             : 
    1536           8 :     std::vector<int> anRequestedBands;
    1537           4 :     if (m_poMaskBand)
    1538           1 :         anRequestedBands.push_back(0);
    1539           8 :     for (int i = 1; i <= GetRasterCount(); i++)
    1540           4 :         anRequestedBands.push_back(i);
    1541             :     GUInt32 nRetryFlags =
    1542           4 :         poBand->PrefetchBlocks(nXOff, nYOff, nXSize, nYSize, anRequestedBands);
    1543             :     int nBlockXSize, nBlockYSize;
    1544           4 :     poBand->GetBlockSize(&nBlockXSize, &nBlockYSize);
    1545           4 :     int nXOff1 = 0;
    1546           4 :     int nYOff1 = 0;
    1547           4 :     int nXSize1 = 0;
    1548           4 :     int nYSize1 = 0;
    1549           4 :     int nXOff2 = 0;
    1550           4 :     int nYOff2 = 0;
    1551           4 :     int nXSize2 = 0;
    1552           4 :     int nYSize2 = 0;
    1553           4 :     GSpacing nDataShift2 = 0;
    1554           4 :     if (CanSpatiallySplit(nRetryFlags, nXOff, nYOff, nXSize, nYSize, nBufXSize,
    1555             :                           nBufYSize, nBlockXSize, nBlockYSize, nPixelSpace,
    1556             :                           nLineSpace, nXOff1, nYOff1, nXSize1, nYSize1, nXOff2,
    1557             :                           nYOff2, nXSize2, nYSize2, nDataShift2))
    1558             :     {
    1559             :         GDALRasterIOExtraArg sExtraArg;
    1560           0 :         INIT_RASTERIO_EXTRA_ARG(sExtraArg);
    1561             : 
    1562             :         CPLErr eErr =
    1563           0 :             IRasterIO(eRWFlag, nXOff1, nYOff1, nXSize1, nYSize1, pData, nXSize1,
    1564             :                       nYSize1, eBufType, nBandCount, panBandMap, nPixelSpace,
    1565             :                       nLineSpace, nBandSpace, &sExtraArg);
    1566           0 :         if (eErr == CE_None)
    1567             :         {
    1568           0 :             eErr = IRasterIO(eRWFlag, nXOff2, nYOff2, nXSize2, nYSize2,
    1569           0 :                              static_cast<GByte *>(pData) + nDataShift2, nXSize2,
    1570             :                              nYSize2, eBufType, nBandCount, panBandMap,
    1571             :                              nPixelSpace, nLineSpace, nBandSpace, &sExtraArg);
    1572             :         }
    1573           0 :         return eErr;
    1574             :     }
    1575           4 :     else if ((nRetryFlags & RETRY_PER_BAND) && nBands > 1)
    1576             :     {
    1577           0 :         for (int iBand = 1; iBand <= nBands; iBand++)
    1578             :         {
    1579           0 :             poBand = cpl::down_cast<GDALDAASRasterBand *>(GetRasterBand(iBand));
    1580           0 :             CPL_IGNORE_RET_VAL(poBand->PrefetchBlocks(
    1581           0 :                 nXOff, nYOff, nXSize, nYSize, std::vector<int>{iBand}));
    1582             :         }
    1583             :     }
    1584             : 
    1585           4 :     return GDALDataset::IRasterIO(eRWFlag, nXOff, nYOff, nXSize, nYSize, pData,
    1586             :                                   nBufXSize, nBufYSize, eBufType, nBandCount,
    1587             :                                   panBandMap, nPixelSpace, nLineSpace,
    1588           4 :                                   nBandSpace, psExtraArg);
    1589             : }
    1590             : 
    1591             : /************************************************************************/
    1592             : /*                          AdviseRead()                                */
    1593             : /************************************************************************/
    1594             : 
    1595           1 : CPLErr GDALDAASDataset::AdviseRead(int nXOff, int nYOff, int nXSize, int nYSize,
    1596             :                                    int nBufXSize, int nBufYSize,
    1597             :                                    GDALDataType /* eBufType */, int /*nBands*/,
    1598             :                                    int * /*panBands*/,
    1599             :                                    char ** /* papszOptions */)
    1600             : {
    1601           1 :     if (nXSize == nBufXSize && nYSize == nBufYSize)
    1602             :     {
    1603           1 :         m_nXOffAdvise = nXOff;
    1604           1 :         m_nYOffAdvise = nYOff;
    1605           1 :         m_nXSizeAdvise = nXSize;
    1606           1 :         m_nYSizeAdvise = nYSize;
    1607             :     }
    1608           1 :     return CE_None;
    1609             : }
    1610             : 
    1611             : /************************************************************************/
    1612             : /*                          FlushCache()                                */
    1613             : /************************************************************************/
    1614             : 
    1615          10 : CPLErr GDALDAASDataset::FlushCache(bool bAtClosing)
    1616             : {
    1617          10 :     CPLErr eErr = GDALDataset::FlushCache(bAtClosing);
    1618          10 :     m_nXOffFetched = 0;
    1619          10 :     m_nYOffFetched = 0;
    1620          10 :     m_nXSizeFetched = 0;
    1621          10 :     m_nYSizeFetched = 0;
    1622          10 :     return eErr;
    1623             : }
    1624             : 
    1625             : /************************************************************************/
    1626             : /*                           GetOverviewCount()                         */
    1627             : /************************************************************************/
    1628             : 
    1629           4 : int GDALDAASRasterBand::GetOverviewCount()
    1630             : {
    1631           4 :     GDALDAASDataset *poGDS = reinterpret_cast<GDALDAASDataset *>(poDS);
    1632           4 :     return static_cast<int>(poGDS->m_apoOverviewDS.size());
    1633             : }
    1634             : 
    1635             : /************************************************************************/
    1636             : /*                              GetOverview()                           */
    1637             : /************************************************************************/
    1638             : 
    1639           5 : GDALRasterBand *GDALDAASRasterBand::GetOverview(int iIndex)
    1640             : {
    1641           5 :     GDALDAASDataset *poGDS = reinterpret_cast<GDALDAASDataset *>(poDS);
    1642           5 :     if (iIndex >= 0 && iIndex < static_cast<int>(poGDS->m_apoOverviewDS.size()))
    1643             :     {
    1644           3 :         return poGDS->m_apoOverviewDS[iIndex]->GetRasterBand(nBand);
    1645             :     }
    1646           2 :     return nullptr;
    1647             : }
    1648             : 
    1649             : /************************************************************************/
    1650             : /*                          IReadBlock()                                */
    1651             : /************************************************************************/
    1652             : 
    1653          11 : CPLErr GDALDAASRasterBand::IReadBlock(int nBlockXOff, int nBlockYOff,
    1654             :                                       void *pImage)
    1655             : {
    1656          22 :     return GetBlocks(nBlockXOff, nBlockYOff, 1, 1, std::vector<int>{nBand},
    1657          22 :                      pImage);
    1658             : }
    1659             : 
    1660             : /************************************************************************/
    1661             : /*                           IRasterIO()                                */
    1662             : /************************************************************************/
    1663             : 
    1664          29 : CPLErr GDALDAASRasterBand::IRasterIO(GDALRWFlag eRWFlag, int nXOff, int nYOff,
    1665             :                                      int nXSize, int nYSize, void *pData,
    1666             :                                      int nBufXSize, int nBufYSize,
    1667             :                                      GDALDataType eBufType,
    1668             :                                      GSpacing nPixelSpace, GSpacing nLineSpace,
    1669             :                                      GDALRasterIOExtraArg *psExtraArg)
    1670             : {
    1671          29 :     GDALDAASDataset *poGDS = reinterpret_cast<GDALDAASDataset *>(poDS);
    1672             : 
    1673          29 :     poGDS->m_eCurrentResampleAlg = psExtraArg->eResampleAlg;
    1674             : 
    1675             :     /* ==================================================================== */
    1676             :     /*      Do we have overviews that would be appropriate to satisfy       */
    1677             :     /*      this request?                                                   */
    1678             :     /* ==================================================================== */
    1679          29 :     if ((nBufXSize < nXSize || nBufYSize < nYSize) && GetOverviewCount() > 0 &&
    1680             :         eRWFlag == GF_Read)
    1681             :     {
    1682             :         GDALRasterIOExtraArg sExtraArg;
    1683           1 :         GDALCopyRasterIOExtraArg(&sExtraArg, psExtraArg);
    1684             : 
    1685             :         const int nOverview =
    1686           1 :             GDALBandGetBestOverviewLevel2(this, nXOff, nYOff, nXSize, nYSize,
    1687             :                                           nBufXSize, nBufYSize, &sExtraArg);
    1688           1 :         if (nOverview >= 0)
    1689             :         {
    1690           1 :             GDALRasterBand *poOverviewBand = GetOverview(nOverview);
    1691           1 :             if (poOverviewBand == nullptr)
    1692           1 :                 return CE_Failure;
    1693             : 
    1694           1 :             return poOverviewBand->RasterIO(
    1695             :                 eRWFlag, nXOff, nYOff, nXSize, nYSize, pData, nBufXSize,
    1696           1 :                 nBufYSize, eBufType, nPixelSpace, nLineSpace, &sExtraArg);
    1697             :         }
    1698             :     }
    1699             : 
    1700          56 :     std::vector<int> anRequestedBands;
    1701          28 :     if (poGDS->m_poMaskBand)
    1702           3 :         anRequestedBands.push_back(0);
    1703          88 :     for (int i = 1; i <= poGDS->GetRasterCount(); i++)
    1704          60 :         anRequestedBands.push_back(i);
    1705             :     GUInt32 nRetryFlags =
    1706          28 :         PrefetchBlocks(nXOff, nYOff, nXSize, nYSize, anRequestedBands);
    1707          28 :     int nXOff1 = 0;
    1708          28 :     int nYOff1 = 0;
    1709          28 :     int nXSize1 = 0;
    1710          28 :     int nYSize1 = 0;
    1711          28 :     int nXOff2 = 0;
    1712          28 :     int nYOff2 = 0;
    1713          28 :     int nXSize2 = 0;
    1714          28 :     int nYSize2 = 0;
    1715          28 :     GSpacing nDataShift2 = 0;
    1716          28 :     if (CanSpatiallySplit(nRetryFlags, nXOff, nYOff, nXSize, nYSize, nBufXSize,
    1717             :                           nBufYSize, nBlockXSize, nBlockYSize, nPixelSpace,
    1718             :                           nLineSpace, nXOff1, nYOff1, nXSize1, nYSize1, nXOff2,
    1719             :                           nYOff2, nXSize2, nYSize2, nDataShift2))
    1720             :     {
    1721             :         GDALRasterIOExtraArg sExtraArg;
    1722           1 :         INIT_RASTERIO_EXTRA_ARG(sExtraArg);
    1723             : 
    1724             :         CPLErr eErr =
    1725           1 :             IRasterIO(eRWFlag, nXOff1, nYOff1, nXSize1, nYSize1, pData, nXSize1,
    1726             :                       nYSize1, eBufType, nPixelSpace, nLineSpace, &sExtraArg);
    1727           1 :         if (eErr == CE_None)
    1728             :         {
    1729           1 :             eErr = IRasterIO(eRWFlag, nXOff2, nYOff2, nXSize2, nYSize2,
    1730           1 :                              static_cast<GByte *>(pData) + nDataShift2, nXSize2,
    1731             :                              nYSize2, eBufType, nPixelSpace, nLineSpace,
    1732             :                              &sExtraArg);
    1733             :         }
    1734           1 :         return eErr;
    1735             :     }
    1736          27 :     else if ((nRetryFlags & RETRY_PER_BAND) && poGDS->nBands > 1)
    1737             :     {
    1738           0 :         CPL_IGNORE_RET_VAL(PrefetchBlocks(nXOff, nYOff, nXSize, nYSize,
    1739           0 :                                           std::vector<int>{nBand}));
    1740             :     }
    1741             : 
    1742          27 :     return GDALRasterBand::IRasterIO(eRWFlag, nXOff, nYOff, nXSize, nYSize,
    1743             :                                      pData, nBufXSize, nBufYSize, eBufType,
    1744          27 :                                      nPixelSpace, nLineSpace, psExtraArg);
    1745             : }
    1746             : 
    1747             : /************************************************************************/
    1748             : /*                          AdviseRead()                                */
    1749             : /************************************************************************/
    1750             : 
    1751           2 : CPLErr GDALDAASRasterBand::AdviseRead(int nXOff, int nYOff, int nXSize,
    1752             :                                       int nYSize, int nBufXSize, int nBufYSize,
    1753             :                                       GDALDataType /* eBufType */,
    1754             :                                       char ** /* papszOptions */)
    1755             : {
    1756           2 :     GDALDAASDataset *poGDS = reinterpret_cast<GDALDAASDataset *>(poDS);
    1757           2 :     if (nXSize == nBufXSize && nYSize == nBufYSize)
    1758             :     {
    1759           2 :         poGDS->m_nXOffAdvise = nXOff;
    1760           2 :         poGDS->m_nYOffAdvise = nYOff;
    1761           2 :         poGDS->m_nXSizeAdvise = nXSize;
    1762           2 :         poGDS->m_nYSizeAdvise = nYSize;
    1763             :     }
    1764           2 :     return CE_None;
    1765             : }
    1766             : 
    1767             : /************************************************************************/
    1768             : /*                          PrefetchBlocks()                            */
    1769             : /************************************************************************/
    1770             : 
    1771             : // Return or'ed flags among 0, RETRY_PER_BAND, RETRY_SPATIAL_SPLIT if the user
    1772             : // should try to split the request in smaller chunks
    1773             : 
    1774             : GUInt32
    1775          32 : GDALDAASRasterBand::PrefetchBlocks(int nXOff, int nYOff, int nXSize, int nYSize,
    1776             :                                    const std::vector<int> &anRequestedBands)
    1777             : {
    1778          32 :     GDALDAASDataset *poGDS = reinterpret_cast<GDALDAASDataset *>(poDS);
    1779             : 
    1780          32 :     if (anRequestedBands.size() > 1)
    1781             :     {
    1782          20 :         if (poGDS->m_nXOffFetched == nXOff && poGDS->m_nYOffFetched == nYOff &&
    1783          20 :             poGDS->m_nXSizeFetched == nXSize &&
    1784           5 :             poGDS->m_nYSizeFetched == nYSize)
    1785             :         {
    1786           5 :             return 0;
    1787             :         }
    1788          15 :         poGDS->m_nXOffFetched = nXOff;
    1789          15 :         poGDS->m_nYOffFetched = nYOff;
    1790          15 :         poGDS->m_nXSizeFetched = nXSize;
    1791          15 :         poGDS->m_nYSizeFetched = nYSize;
    1792             :     }
    1793             : 
    1794          27 :     int nBlockXOff = nXOff / nBlockXSize;
    1795          27 :     int nBlockYOff = nYOff / nBlockYSize;
    1796          27 :     int nXBlocks = (nXOff + nXSize - 1) / nBlockXSize - nBlockXOff + 1;
    1797          27 :     int nYBlocks = (nYOff + nYSize - 1) / nBlockYSize - nBlockYOff + 1;
    1798             : 
    1799          27 :     int nTotalDataTypeSize = 0;
    1800          27 :     const int nQueriedBands = static_cast<int>(anRequestedBands.size());
    1801          82 :     for (int i = 0; i < nQueriedBands; i++)
    1802             :     {
    1803          55 :         const int iBand = anRequestedBands[i];
    1804          55 :         if (iBand > 0 && iBand <= poGDS->GetRasterCount())
    1805             :         {
    1806          53 :             nTotalDataTypeSize += GDALGetDataTypeSizeBytes(
    1807             :                 poGDS->GetRasterBand(iBand)->GetRasterDataType());
    1808             :         }
    1809             :         else
    1810             :         {
    1811           2 :             nTotalDataTypeSize += GDALGetDataTypeSizeBytes(
    1812           2 :                 poGDS->m_poMaskBand->GetRasterDataType());
    1813             :         }
    1814             :     }
    1815             : 
    1816             :     // If AdviseRead() was called before, and the current requested area is
    1817             :     // in it, check if we can prefetch the whole advised area
    1818          27 :     const GIntBig nCacheMax = GDALGetCacheMax64() / 2;
    1819          27 :     if (poGDS->m_nXSizeAdvise > 0 && nXOff >= poGDS->m_nXOffAdvise &&
    1820          10 :         nYOff >= poGDS->m_nYOffAdvise &&
    1821          10 :         nXOff + nXSize <= poGDS->m_nXOffAdvise + poGDS->m_nXSizeAdvise &&
    1822          10 :         nYOff + nYSize <= poGDS->m_nYOffAdvise + poGDS->m_nYSizeAdvise)
    1823             :     {
    1824          10 :         int nBlockXOffAdvise = poGDS->m_nXOffAdvise / nBlockXSize;
    1825          10 :         int nBlockYOffAdvise = poGDS->m_nYOffAdvise / nBlockYSize;
    1826          10 :         int nXBlocksAdvise =
    1827          10 :             (poGDS->m_nXOffAdvise + poGDS->m_nXSizeAdvise - 1) / nBlockXSize -
    1828             :             nBlockXOffAdvise + 1;
    1829          10 :         int nYBlocksAdvise =
    1830          10 :             (poGDS->m_nYOffAdvise + poGDS->m_nYSizeAdvise - 1) / nBlockYSize -
    1831             :             nBlockYOffAdvise + 1;
    1832          10 :         const GIntBig nUncompressedSize = static_cast<GIntBig>(nXBlocksAdvise) *
    1833          10 :                                           nYBlocksAdvise * nBlockXSize *
    1834          10 :                                           nBlockYSize * nTotalDataTypeSize;
    1835          10 :         if (nUncompressedSize <= nCacheMax &&
    1836          10 :             nUncompressedSize <= poGDS->m_nServerByteLimit)
    1837             :         {
    1838           7 :             CPLDebug("DAAS", "Using advise read");
    1839           7 :             nBlockXOff = nBlockXOffAdvise;
    1840           7 :             nBlockYOff = nBlockYOffAdvise;
    1841           7 :             nXBlocks = nXBlocksAdvise;
    1842           7 :             nYBlocks = nYBlocksAdvise;
    1843           7 :             if (anRequestedBands.size() > 1)
    1844             :             {
    1845           0 :                 poGDS->m_nXOffAdvise = 0;
    1846           0 :                 poGDS->m_nYOffAdvise = 0;
    1847           0 :                 poGDS->m_nXSizeAdvise = 0;
    1848           0 :                 poGDS->m_nYSizeAdvise = 0;
    1849             :             }
    1850             :         }
    1851             :     }
    1852             : 
    1853             :     // Check the number of already cached blocks, and remove fully
    1854             :     // cached lines at the top of the area of interest from the queried
    1855             :     // blocks
    1856          27 :     int nBlocksCached = 0;
    1857          27 :     int nBlocksCachedForThisBand = 0;
    1858          27 :     bool bAllLineCached = true;
    1859          54 :     for (int iYBlock = 0; iYBlock < nYBlocks;)
    1860             :     {
    1861          62 :         for (int iXBlock = 0; iXBlock < nXBlocks; iXBlock++)
    1862             :         {
    1863          98 :             for (int i = 0; i < nQueriedBands; i++)
    1864             :             {
    1865          63 :                 const int iBand = anRequestedBands[i];
    1866          63 :                 GDALRasterBlock *poBlock = nullptr;
    1867             :                 GDALDAASRasterBand *poIterBand;
    1868          63 :                 if (iBand > 0 && iBand <= poGDS->GetRasterCount())
    1869             :                     poIterBand = reinterpret_cast<GDALDAASRasterBand *>(
    1870          61 :                         poGDS->GetRasterBand(iBand));
    1871             :                 else
    1872           2 :                     poIterBand = poGDS->m_poMaskBand;
    1873             : 
    1874         126 :                 poBlock = poIterBand->TryGetLockedBlockRef(
    1875          63 :                     nBlockXOff + iXBlock, nBlockYOff + iYBlock);
    1876          63 :                 if (poBlock != nullptr)
    1877             :                 {
    1878          11 :                     nBlocksCached++;
    1879          11 :                     if (iBand == nBand)
    1880          10 :                         nBlocksCachedForThisBand++;
    1881          11 :                     poBlock->DropLock();
    1882          11 :                     continue;
    1883             :                 }
    1884             :                 else
    1885             :                 {
    1886          52 :                     bAllLineCached = false;
    1887             :                 }
    1888             :             }
    1889             :         }
    1890             : 
    1891          27 :         if (bAllLineCached)
    1892             :         {
    1893           5 :             nBlocksCached -= nXBlocks * nQueriedBands;
    1894           5 :             nBlocksCachedForThisBand -= nXBlocks;
    1895           5 :             nBlockYOff++;
    1896           5 :             nYBlocks--;
    1897             :         }
    1898             :         else
    1899             :         {
    1900          22 :             iYBlock++;
    1901             :         }
    1902             :     }
    1903             : 
    1904          27 :     if (nXBlocks > 0 && nYBlocks > 0)
    1905             :     {
    1906          22 :         bool bMustReturn = false;
    1907          22 :         GUInt32 nRetryFlags = 0;
    1908             : 
    1909             :         // Get the blocks if the number of already cached blocks is lesser
    1910             :         // than 25% of the to be queried blocks
    1911          22 :         if (nBlocksCached > (nQueriedBands * nXBlocks * nYBlocks) / 4)
    1912             :         {
    1913           1 :             if (nBlocksCachedForThisBand <= (nXBlocks * nYBlocks) / 4)
    1914             :             {
    1915           1 :                 nRetryFlags |= RETRY_PER_BAND;
    1916             :             }
    1917             :             else
    1918             :             {
    1919           0 :                 bMustReturn = true;
    1920             :             }
    1921             :         }
    1922             : 
    1923             :         // Make sure that we have enough cache (with a margin of 50%)
    1924             :         // and the number of queried pixels isn't too big w.r.t server
    1925             :         // limit
    1926          22 :         const GIntBig nUncompressedSize = static_cast<GIntBig>(nXBlocks) *
    1927          22 :                                           nYBlocks * nBlockXSize * nBlockYSize *
    1928          22 :                                           nTotalDataTypeSize;
    1929          22 :         if (nUncompressedSize > nCacheMax ||
    1930          22 :             nUncompressedSize > poGDS->m_nServerByteLimit)
    1931             :         {
    1932           3 :             if (anRequestedBands.size() > 1 && poGDS->GetRasterCount() > 1)
    1933             :             {
    1934           0 :                 const int nThisDTSize = GDALGetDataTypeSizeBytes(eDataType);
    1935           0 :                 const GIntBig nUncompressedSizeThisBand =
    1936           0 :                     static_cast<GIntBig>(nXBlocks) * nYBlocks * nBlockXSize *
    1937           0 :                     nBlockYSize * nThisDTSize;
    1938           0 :                 if (nUncompressedSizeThisBand <= poGDS->m_nServerByteLimit &&
    1939             :                     nUncompressedSizeThisBand <= nCacheMax)
    1940             :                 {
    1941           0 :                     nRetryFlags |= RETRY_PER_BAND;
    1942             :                 }
    1943             :             }
    1944           3 :             if (nXBlocks > 1 || nYBlocks > 1)
    1945             :             {
    1946           1 :                 nRetryFlags |= RETRY_SPATIAL_SPLIT;
    1947             :             }
    1948           3 :             return nRetryFlags;
    1949             :         }
    1950          19 :         if (bMustReturn)
    1951           0 :             return nRetryFlags;
    1952             : 
    1953          19 :         GetBlocks(nBlockXOff, nBlockYOff, nXBlocks, nYBlocks, anRequestedBands,
    1954             :                   nullptr);
    1955             :     }
    1956             : 
    1957          24 :     return 0;
    1958             : }
    1959             : 
    1960             : /************************************************************************/
    1961             : /*                           GetBlocks()                                */
    1962             : /************************************************************************/
    1963             : 
    1964          34 : CPLErr GDALDAASRasterBand::GetBlocks(int nBlockXOff, int nBlockYOff,
    1965             :                                      int nXBlocks, int nYBlocks,
    1966             :                                      const std::vector<int> &anRequestedBands,
    1967             :                                      void *pDstBuffer)
    1968             : {
    1969          34 :     GDALDAASDataset *poGDS = reinterpret_cast<GDALDAASDataset *>(poDS);
    1970             : 
    1971          34 :     CPLAssert(!anRequestedBands.empty());
    1972          34 :     if (pDstBuffer)
    1973             :     {
    1974          11 :         CPLAssert(nXBlocks == 1 && nYBlocks == 1 &&
    1975             :                   anRequestedBands.size() == 1);
    1976             :     }
    1977             : 
    1978             :     // Detect if there is a mix of non-mask and mask bands
    1979          34 :     if (anRequestedBands.size() > 1)
    1980             :     {
    1981          15 :         std::vector<int> anNonMasks;
    1982          15 :         std::vector<int> anMasks;
    1983          58 :         for (auto &iBand : anRequestedBands)
    1984             :         {
    1985          84 :             if (iBand == MAIN_MASK_BAND_NUMBER ||
    1986          41 :                 poGDS->m_aoBandDesc[iBand - 1].bIsMask)
    1987           2 :                 anMasks.push_back(iBand);
    1988             :             else
    1989          41 :                 anNonMasks.push_back(iBand);
    1990             :         }
    1991          15 :         if (!anNonMasks.empty() && !anMasks.empty())
    1992             :         {
    1993           2 :             return GetBlocks(nBlockXOff, nBlockYOff, nXBlocks, nYBlocks,
    1994           2 :                              anNonMasks, nullptr) == CE_None &&
    1995           2 :                            GetBlocks(nBlockXOff, nBlockYOff, nXBlocks, nYBlocks,
    1996             :                                      anMasks, nullptr) == CE_None
    1997           4 :                        ? CE_None
    1998           2 :                        : CE_Failure;
    1999             :         }
    2000             :     }
    2001             : 
    2002          32 :     char **papszOptions = poGDS->GetHTTPOptions();
    2003             : 
    2004          64 :     CPLString osHeaders = CSLFetchNameValueDef(papszOptions, "HEADERS", "");
    2005          32 :     if (!osHeaders.empty())
    2006           0 :         osHeaders += "\r\n";
    2007          32 :     osHeaders += "Content-Type: application/json";
    2008          32 :     osHeaders += "\r\n";
    2009          64 :     CPLString osDataContentType("application/octet-stream");
    2010          32 :     GDALDAASDataset::Format eRequestFormat(GDALDAASDataset::Format::RAW);
    2011          41 :     if (poGDS->m_eFormat == GDALDAASDataset::Format::PNG &&
    2012           9 :         (anRequestedBands.size() == 1 || anRequestedBands.size() == 3 ||
    2013           0 :          anRequestedBands.size() == 4))
    2014             :     {
    2015           6 :         eRequestFormat = poGDS->m_eFormat;
    2016           6 :         osDataContentType = "image/png";
    2017             :     }
    2018          28 :     else if (poGDS->m_eFormat == GDALDAASDataset::Format::JPEG &&
    2019           2 :              (anRequestedBands.size() == 1 || anRequestedBands.size() == 3))
    2020             :     {
    2021           1 :         eRequestFormat = poGDS->m_eFormat;
    2022           1 :         osDataContentType = "image/jpeg";
    2023             :     }
    2024          25 :     else if (poGDS->m_eFormat == GDALDAASDataset::Format::JPEG2000)
    2025             :     {
    2026           1 :         eRequestFormat = poGDS->m_eFormat;
    2027           1 :         osDataContentType = "image/jp2";
    2028             :     }
    2029          32 :     osHeaders += "Accept: " + osDataContentType;
    2030          32 :     papszOptions = CSLSetNameValue(papszOptions, "HEADERS", osHeaders);
    2031             : 
    2032             :     // Build request JSon document
    2033          64 :     CPLJSONDocument oDoc;
    2034          64 :     CPLJSONObject oBBox;
    2035             : 
    2036          32 :     if (poGDS->m_bRequestInGeoreferencedCoordinates)
    2037             :     {
    2038           0 :         CPLJSONObject oSRS;
    2039           0 :         oSRS.Add("type", poGDS->m_osSRSType);
    2040           0 :         oSRS.Add("value", poGDS->m_osSRSValue);
    2041           0 :         oBBox.Add("srs", oSRS);
    2042             :     }
    2043             :     else
    2044             :     {
    2045          32 :         CPLJSONObject oSRS;
    2046          32 :         oSRS.Add("type", "image");
    2047          32 :         oBBox.Add("srs", oSRS);
    2048             :     }
    2049             : 
    2050          32 :     const int nMainXSize = poGDS->m_poParentDS
    2051          32 :                                ? poGDS->m_poParentDS->GetRasterXSize()
    2052          32 :                                : nRasterXSize;
    2053          32 :     const int nMainYSize = poGDS->m_poParentDS
    2054          32 :                                ? poGDS->m_poParentDS->GetRasterYSize()
    2055          32 :                                : nRasterYSize;
    2056          32 :     const int nULX = nBlockXOff * nBlockXSize;
    2057          32 :     const int nULY = nBlockYOff * nBlockYSize;
    2058             :     const int nLRX =
    2059          32 :         std::min(nRasterXSize, (nBlockXOff + nXBlocks) * nBlockXSize);
    2060             :     const int nLRY =
    2061          32 :         std::min(nRasterYSize, (nBlockYOff + nYBlocks) * nBlockYSize);
    2062             : 
    2063          64 :     CPLJSONObject oUL;
    2064          64 :     CPLJSONObject oLR;
    2065          32 :     if (poGDS->m_bRequestInGeoreferencedCoordinates)
    2066             :     {
    2067             :         double dfULX, dfULY;
    2068           0 :         GDALApplyGeoTransform(poGDS->m_adfGeoTransform.data(), nULX, nULY,
    2069             :                               &dfULX, &dfULY);
    2070           0 :         oUL.Add("x", dfULX);
    2071           0 :         oUL.Add("y", dfULY);
    2072             : 
    2073             :         double dfLRX, dfLRY;
    2074           0 :         GDALApplyGeoTransform(poGDS->m_adfGeoTransform.data(), nLRX, nLRY,
    2075             :                               &dfLRX, &dfLRY);
    2076           0 :         oLR.Add("x", dfLRX);
    2077           0 :         oLR.Add("y", dfLRY);
    2078             :     }
    2079             :     else
    2080             :     {
    2081          32 :         oUL.Add("x",
    2082          32 :                 static_cast<int>((static_cast<GIntBig>(nULX) * nMainXSize) /
    2083          32 :                                  nRasterXSize));
    2084          32 :         oUL.Add("y",
    2085          32 :                 static_cast<int>((static_cast<GIntBig>(nULY) * nMainYSize) /
    2086          32 :                                  nRasterYSize));
    2087             : 
    2088          33 :         oLR.Add("x", (nLRX == nRasterXSize)
    2089             :                          ? nMainXSize
    2090             :                          : static_cast<int>(
    2091           1 :                                (static_cast<GIntBig>(nLRX) * nMainXSize) /
    2092           1 :                                nRasterXSize));
    2093          32 :         oLR.Add("y", (nLRY == nRasterYSize)
    2094             :                          ? nMainYSize
    2095             :                          : static_cast<int>(
    2096           0 :                                (static_cast<GIntBig>(nLRY) * nMainYSize) /
    2097           0 :                                nRasterYSize));
    2098             :     }
    2099          32 :     oBBox.Add("ul", oUL);
    2100          32 :     oBBox.Add("lr", oLR);
    2101          32 :     oDoc.GetRoot().Add("bbox", oBBox);
    2102             : 
    2103          64 :     CPLJSONObject oTargetModel;
    2104             : 
    2105          64 :     CPLJSONObject oStepTargetModel;
    2106          32 :     if (poGDS->m_bRequestInGeoreferencedCoordinates)
    2107             :     {
    2108           0 :         oStepTargetModel.Add("x", poGDS->m_adfGeoTransform[1]);
    2109           0 :         oStepTargetModel.Add("y", fabs(poGDS->m_adfGeoTransform[5]));
    2110             :     }
    2111             :     else
    2112             :     {
    2113          32 :         oStepTargetModel.Add("x", 0);
    2114          32 :         oStepTargetModel.Add("y", 0);
    2115             :     }
    2116          32 :     oTargetModel.Add("step", oStepTargetModel);
    2117             : 
    2118          64 :     CPLJSONObject oSize;
    2119          32 :     int nRequestWidth = nLRX - nULX;
    2120          32 :     int nRequestHeight = nLRY - nULY;
    2121          32 :     oSize.Add("columns", nRequestWidth);
    2122          32 :     oSize.Add("lines", nRequestHeight);
    2123          32 :     oTargetModel.Add("size", oSize);
    2124             : 
    2125          32 :     if (poGDS->m_eCurrentResampleAlg == GRIORA_NearestNeighbour)
    2126             :     {
    2127          32 :         oTargetModel.Add("sampling-algo", "NEAREST");
    2128             :     }
    2129           0 :     else if (poGDS->m_eCurrentResampleAlg == GRIORA_Bilinear)
    2130             :     {
    2131           0 :         oTargetModel.Add("sampling-algo", "BILINEAR");
    2132             :     }
    2133           0 :     else if (poGDS->m_eCurrentResampleAlg == GRIORA_Cubic)
    2134             :     {
    2135           0 :         oTargetModel.Add("sampling-algo", "BICUBIC");
    2136             :     }
    2137           0 :     else if (poGDS->m_eCurrentResampleAlg == GRIORA_Average)
    2138             :     {
    2139           0 :         oTargetModel.Add("sampling-algo", "AVERAGE");
    2140             :     }
    2141             :     else
    2142             :     {
    2143             :         // Defaults to BILINEAR for other GDAL methods not supported by
    2144             :         // server
    2145           0 :         oTargetModel.Add("sampling-algo", "BILINEAR");
    2146             :     }
    2147             : 
    2148          32 :     oTargetModel.Add("strictOutputSize", true);
    2149             : 
    2150          32 :     if (!poGDS->m_bRequestInGeoreferencedCoordinates)
    2151             :     {
    2152          32 :         CPLJSONObject oSRS;
    2153          32 :         oSRS.Add("type", "image");
    2154          32 :         oTargetModel.Add("srs", oSRS);
    2155             :     }
    2156             : 
    2157          32 :     oDoc.GetRoot().Add("target-model", oTargetModel);
    2158             : 
    2159          64 :     CPLJSONArray oBands;
    2160          32 :     bool bOK = true;
    2161          90 :     for (const int iBand : anRequestedBands)
    2162             :     {
    2163             :         auto desc = (iBand == MAIN_MASK_BAND_NUMBER)
    2164          58 :                         ? poGDS->m_poMaskBand->GetDescription()
    2165          56 :                         : poGDS->GetRasterBand(iBand)->GetDescription();
    2166          58 :         if (EQUAL(desc, ""))
    2167           0 :             bOK = false;
    2168             :         else
    2169          58 :             oBands.Add(desc);
    2170             :     }
    2171          32 :     if (bOK)
    2172             :     {
    2173          32 :         oDoc.GetRoot().Add("bands", oBands);
    2174             :     }
    2175             : 
    2176          32 :     papszOptions = CSLSetNameValue(
    2177             :         papszOptions, "POSTFIELDS",
    2178          64 :         oDoc.GetRoot().Format(CPLJSONObject::PrettyFormat::Pretty).c_str());
    2179             : 
    2180             :     CPLString osURL(CPLGetConfigOption("GDAL_DAAS_GET_BUFFER_URL",
    2181          64 :                                        poGDS->m_osGetBufferURL.c_str()));
    2182          32 :     CPLHTTPResult *psResult = DAAS_CPLHTTPFetch(osURL, papszOptions);
    2183          32 :     CSLDestroy(papszOptions);
    2184          32 :     if (psResult == nullptr)
    2185           0 :         return CE_Failure;
    2186             : 
    2187          32 :     if (psResult->pszErrBuf != nullptr)
    2188             :     {
    2189          11 :         CPLError(CE_Failure, CPLE_AppDefined, "Get request %s failed: %s",
    2190             :                  osURL.c_str(),
    2191          12 :                  psResult->pabyData ? CPLSPrintf("%s: %s", psResult->pszErrBuf,
    2192             :                                                  reinterpret_cast<const char *>(
    2193           1 :                                                      psResult->pabyData))
    2194             :                                     : psResult->pszErrBuf);
    2195          11 :         CPLHTTPDestroyResult(psResult);
    2196          11 :         return CE_Failure;
    2197             :     }
    2198             : 
    2199          21 :     if (psResult->nDataLen == 0)
    2200             :     {
    2201             :         // Presumably HTTP 204 empty
    2202           0 :         CPLHTTPDestroyResult(psResult);
    2203             : 
    2204           0 :         for (int iYBlock = 0; iYBlock < nYBlocks; iYBlock++)
    2205             :         {
    2206           0 :             for (int iXBlock = 0; iXBlock < nXBlocks; iXBlock++)
    2207             :             {
    2208           0 :                 for (const int iBand : anRequestedBands)
    2209             :                 {
    2210           0 :                     GByte *pabyDstBuffer = nullptr;
    2211             :                     GDALDAASRasterBand *poIterBand;
    2212           0 :                     if (iBand == MAIN_MASK_BAND_NUMBER)
    2213             :                     {
    2214           0 :                         poIterBand = poGDS->m_poMaskBand;
    2215             :                     }
    2216             :                     else
    2217             :                     {
    2218             :                         poIterBand = reinterpret_cast<GDALDAASRasterBand *>(
    2219           0 :                             poGDS->GetRasterBand(iBand));
    2220             :                     }
    2221             : 
    2222           0 :                     GDALRasterBlock *poBlock = nullptr;
    2223           0 :                     if (pDstBuffer != nullptr)
    2224             :                     {
    2225           0 :                         pabyDstBuffer = static_cast<GByte *>(pDstBuffer);
    2226             :                     }
    2227             :                     else
    2228             :                     {
    2229             :                         // Check if the same block in other bands is already in
    2230             :                         // the GDAL block cache
    2231           0 :                         poBlock = poIterBand->TryGetLockedBlockRef(
    2232           0 :                             nBlockXOff + iXBlock, nBlockYOff + iYBlock);
    2233           0 :                         if (poBlock != nullptr)
    2234             :                         {
    2235             :                             // Yes, no need to do further work
    2236           0 :                             poBlock->DropLock();
    2237           0 :                             continue;
    2238             :                         }
    2239             :                         // Instantiate the block
    2240           0 :                         poBlock = poIterBand->GetLockedBlockRef(
    2241           0 :                             nBlockXOff + iXBlock, nBlockYOff + iYBlock, TRUE);
    2242           0 :                         if (poBlock == nullptr)
    2243             :                         {
    2244           0 :                             continue;
    2245             :                         }
    2246             :                         pabyDstBuffer =
    2247           0 :                             static_cast<GByte *>(poBlock->GetDataRef());
    2248             :                     }
    2249             : 
    2250           0 :                     const int nDTSize = GDALGetDataTypeSizeBytes(
    2251             :                         poIterBand->GetRasterDataType());
    2252           0 :                     double dfNoDataValue = poIterBand->GetNoDataValue(nullptr);
    2253           0 :                     GDALCopyWords(&dfNoDataValue, GDT_Float64, 0, pabyDstBuffer,
    2254             :                                   poIterBand->GetRasterDataType(), nDTSize,
    2255           0 :                                   nBlockXSize * nBlockYSize);
    2256           0 :                     if (poBlock)
    2257           0 :                         poBlock->DropLock();
    2258             :                 }
    2259             :             }
    2260             :         }
    2261             : 
    2262           0 :         return CE_None;
    2263             :     }
    2264             : 
    2265             : #ifdef DEBUG_VERBOSE
    2266             :     CPLDebug("DAAS", "Response = '%s'",
    2267             :              reinterpret_cast<const char *>(psResult->pabyData));
    2268             : #endif
    2269          21 :     if (!CPLHTTPParseMultipartMime(psResult))
    2270             :     {
    2271           1 :         CPLError(CE_Failure, CPLE_AppDefined,
    2272             :                  "Get request %s failed: "
    2273             :                  "Invalid content returned by server",
    2274             :                  osURL.c_str());
    2275           1 :         CPLHTTPDestroyResult(psResult);
    2276           1 :         return CE_Failure;
    2277             :     }
    2278          20 :     int iMetadataPart = -1;
    2279          20 :     int iDataPart = -1;
    2280             :     // Identify metadata and data parts
    2281          58 :     for (int i = 0; i < psResult->nMimePartCount; i++)
    2282             :     {
    2283          76 :         const char *pszContentType = CSLFetchNameValue(
    2284          38 :             psResult->pasMimePart[i].papszHeaders, "Content-Type");
    2285          76 :         const char *pszContentDisposition = CSLFetchNameValue(
    2286          38 :             psResult->pasMimePart[i].papszHeaders, "Content-Disposition");
    2287          38 :         if (pszContentType)
    2288             :         {
    2289          38 :             if (EQUAL(pszContentType, "application/json"))
    2290             :             {
    2291          19 :                 iMetadataPart = i;
    2292             :             }
    2293          19 :             else if (EQUAL(pszContentType, osDataContentType))
    2294             :             {
    2295          19 :                 iDataPart = i;
    2296             :             }
    2297             :         }
    2298          38 :         if (pszContentDisposition)
    2299             :         {
    2300          38 :             if (EQUAL(pszContentDisposition, "form-data; name=\"Data\";"))
    2301             :             {
    2302          19 :                 iDataPart = i;
    2303             :             }
    2304             :         }
    2305             :     }
    2306          20 :     if (iDataPart < 0)
    2307             :     {
    2308           1 :         CPLError(CE_Failure, CPLE_AppDefined,
    2309             :                  "Cannot find part with Content-Type: %s in GetBuffer response",
    2310             :                  osDataContentType.c_str());
    2311           1 :         CPLHTTPDestroyResult(psResult);
    2312           1 :         return CE_Failure;
    2313             :     }
    2314          19 :     if (iMetadataPart < 0)
    2315             :     {
    2316           1 :         CPLError(CE_Failure, CPLE_AppDefined,
    2317             :                  "Cannot find part with Content-Type: %s in GetBuffer response",
    2318             :                  "application/json");
    2319           1 :         CPLHTTPDestroyResult(psResult);
    2320           1 :         return CE_Failure;
    2321             :     }
    2322             : 
    2323          36 :     CPLString osJson;
    2324             :     osJson.assign(reinterpret_cast<const char *>(
    2325          18 :                       psResult->pasMimePart[iMetadataPart].pabyData),
    2326          18 :                   psResult->pasMimePart[iMetadataPart].nDataLen);
    2327          18 :     CPLDebug("DAAS", "GetBuffer metadata response: %s", osJson.c_str());
    2328          18 :     if (!oDoc.LoadMemory(osJson))
    2329             :     {
    2330           0 :         CPLHTTPDestroyResult(psResult);
    2331           0 :         return CE_Failure;
    2332             :     }
    2333          36 :     auto oDocRoot = oDoc.GetRoot();
    2334          18 :     int nGotHeight = oDocRoot.GetInteger("properties/height");
    2335          18 :     int nGotWidth = oDocRoot.GetInteger("properties/width");
    2336          18 :     if (nGotHeight != nRequestHeight || nGotWidth != nRequestWidth)
    2337             :     {
    2338           1 :         CPLError(CE_Failure, CPLE_AppDefined,
    2339             :                  "Got buffer of size %dx%d, whereas %dx%d was expected",
    2340             :                  nGotWidth, nGotHeight, nRequestWidth, nRequestHeight);
    2341           1 :         CPLHTTPDestroyResult(psResult);
    2342           1 :         return CE_Failure;
    2343             :     }
    2344             : 
    2345             :     // Get the actual data type of the buffer response
    2346             :     GDALDataType eBufferDataType =
    2347          17 :         anRequestedBands[0] == MAIN_MASK_BAND_NUMBER
    2348          32 :             ? GDT_Byte
    2349          15 :             : poGDS->m_aoBandDesc[anRequestedBands[0] - 1].eDT;
    2350          51 :     auto oBandArray = oDocRoot.GetArray("properties/bands");
    2351          17 :     if (oBandArray.IsValid() && oBandArray.Size() >= 1)
    2352             :     {
    2353             :         bool bIgnored;
    2354           4 :         auto oBandProperties = oBandArray[0];
    2355             :         auto osPixelType =
    2356           4 :             GetString(oBandProperties, "pixelType", false, bIgnored);
    2357           4 :         if (!osPixelType.empty())
    2358             :         {
    2359           4 :             eBufferDataType = GetGDALDataTypeFromDAASPixelType(osPixelType);
    2360           4 :             if (eBufferDataType == GDT_Unknown)
    2361             :             {
    2362           0 :                 CPLError(CE_Failure, CPLE_AppDefined, "Invalid pixelType: %s",
    2363             :                          osPixelType.c_str());
    2364           0 :                 CPLHTTPDestroyResult(psResult);
    2365           0 :                 return CE_Failure;
    2366             :             }
    2367             :         }
    2368             :     }
    2369             : 
    2370          17 :     const int nBufferDTSize = GDALGetDataTypeSizeBytes(eBufferDataType);
    2371          17 :     std::shared_ptr<GDALDataset> poTileDS;
    2372          17 :     if (eRequestFormat == GDALDAASDataset::Format::RAW)
    2373             :     {
    2374          22 :         int nExpectedBytes = nGotHeight * nGotWidth * nBufferDTSize *
    2375          11 :                              static_cast<int>(anRequestedBands.size());
    2376          11 :         if (psResult->pasMimePart[iDataPart].nDataLen != nExpectedBytes)
    2377             :         {
    2378           1 :             CPLError(CE_Failure, CPLE_AppDefined,
    2379             :                      "Got buffer of %d bytes, whereas %d were expected",
    2380           1 :                      psResult->pasMimePart[iDataPart].nDataLen, nExpectedBytes);
    2381           1 :             CPLHTTPDestroyResult(psResult);
    2382           1 :             return CE_Failure;
    2383             :         }
    2384             : 
    2385          10 :         GByte *pabySrcData = psResult->pasMimePart[iDataPart].pabyData;
    2386             : #ifdef CPL_MSB
    2387             :         GDALSwapWords(pabySrcData, nBufferDTSize,
    2388             :                       nGotHeight * nGotWidth *
    2389             :                           static_cast<int>(anRequestedBands.size()),
    2390             :                       nBufferDTSize);
    2391             : #endif
    2392             : 
    2393          10 :         auto poMEMDS = MEMDataset::Create("", nRequestWidth, nRequestHeight, 0,
    2394             :                                           eBufferDataType, nullptr);
    2395          10 :         poTileDS.reset(poMEMDS);
    2396          22 :         for (int i = 0; i < static_cast<int>(anRequestedBands.size()); i++)
    2397             :         {
    2398          24 :             auto hBand = MEMCreateRasterBandEx(
    2399             :                 poMEMDS, i + 1,
    2400          12 :                 pabySrcData + i * nGotHeight * nGotWidth * nBufferDTSize,
    2401             :                 eBufferDataType, 0, 0, false);
    2402          12 :             poMEMDS->AddMEMBand(hBand);
    2403             :         }
    2404             :     }
    2405             :     else
    2406             :     {
    2407           6 :         const CPLString osTmpMemFile = VSIMemGenerateHiddenFilename("daas");
    2408           6 :         VSIFCloseL(VSIFileFromMemBuffer(
    2409           6 :             osTmpMemFile, psResult->pasMimePart[iDataPart].pabyData,
    2410           6 :             psResult->pasMimePart[iDataPart].nDataLen, false));
    2411           6 :         poTileDS.reset(GDALDataset::Open(osTmpMemFile,
    2412             :                                          GDAL_OF_RASTER | GDAL_OF_INTERNAL,
    2413             :                                          nullptr, nullptr, nullptr));
    2414           6 :         if (!poTileDS)
    2415             :         {
    2416           1 :             CPLError(CE_Failure, CPLE_AppDefined, "Cannot decode image");
    2417           1 :             VSIUnlink(osTmpMemFile);
    2418           1 :             CPLHTTPDestroyResult(psResult);
    2419           1 :             return CE_Failure;
    2420             :         }
    2421             :     }
    2422             : 
    2423          15 :     CPLErr eErr = CE_None;
    2424          15 :     poTileDS->MarkSuppressOnClose();
    2425             : 
    2426             :     bool bExpectedImageCharacteristics =
    2427          29 :         (poTileDS->GetRasterXSize() == nRequestWidth &&
    2428          14 :          poTileDS->GetRasterYSize() == nRequestHeight);
    2429          15 :     if (bExpectedImageCharacteristics)
    2430             :     {
    2431          28 :         if (poTileDS->GetRasterCount() ==
    2432          14 :             static_cast<int>(anRequestedBands.size()))
    2433             :         {
    2434             :             // ok
    2435             :         }
    2436           1 :         else if (eRequestFormat == GDALDAASDataset::Format::PNG &&
    2437           2 :                  anRequestedBands.size() == 1 &&
    2438           1 :                  poTileDS->GetRasterCount() == 4)
    2439             :         {
    2440             :             // ok
    2441             :         }
    2442             :         else
    2443             :         {
    2444           0 :             bExpectedImageCharacteristics = false;
    2445             :         }
    2446             :     }
    2447             : 
    2448          15 :     if (!bExpectedImageCharacteristics)
    2449             :     {
    2450           1 :         CPLError(CE_Failure, CPLE_AppDefined,
    2451             :                  "Got tile of size %dx%dx%d, whereas %dx%dx%d was expected",
    2452             :                  poTileDS->GetRasterXSize(), poTileDS->GetRasterYSize(),
    2453             :                  poTileDS->GetRasterCount(), nRequestWidth, nRequestHeight,
    2454           1 :                  static_cast<int>(anRequestedBands.size()));
    2455           1 :         CPLHTTPDestroyResult(psResult);
    2456           1 :         return CE_Failure;
    2457             :     }
    2458             : 
    2459          28 :     for (int iYBlock = 0; eErr == CE_None && iYBlock < nYBlocks; iYBlock++)
    2460             :     {
    2461             :         int nBlockActualYSize = std::min(
    2462          14 :             nBlockYSize, nRasterYSize - (iYBlock + nBlockYOff) * nBlockYSize);
    2463          30 :         for (int iXBlock = 0; eErr == CE_None && iXBlock < nXBlocks; iXBlock++)
    2464             :         {
    2465             :             int nBlockActualXSize =
    2466          16 :                 std::min(nBlockXSize,
    2467          16 :                          nRasterXSize - (iXBlock + nBlockXOff) * nBlockXSize);
    2468             : 
    2469          40 :             for (int i = 0; i < static_cast<int>(anRequestedBands.size()); i++)
    2470             :             {
    2471          24 :                 const int iBand = anRequestedBands[i];
    2472          24 :                 GByte *pabyDstBuffer = nullptr;
    2473             :                 GDALDAASRasterBand *poIterBand;
    2474          24 :                 if (iBand == MAIN_MASK_BAND_NUMBER)
    2475             :                 {
    2476           2 :                     poIterBand = poGDS->m_poMaskBand;
    2477             :                 }
    2478             :                 else
    2479             :                 {
    2480             :                     poIterBand = reinterpret_cast<GDALDAASRasterBand *>(
    2481          22 :                         poGDS->GetRasterBand(iBand));
    2482             :                 }
    2483             : 
    2484          24 :                 GDALRasterBlock *poBlock = nullptr;
    2485          24 :                 if (pDstBuffer != nullptr)
    2486           2 :                     pabyDstBuffer = static_cast<GByte *>(pDstBuffer);
    2487             :                 else
    2488             :                 {
    2489             :                     // Check if the same block in other bands is already in
    2490             :                     // the GDAL block cache
    2491          44 :                     poBlock = poIterBand->TryGetLockedBlockRef(
    2492          22 :                         nBlockXOff + iXBlock, nBlockYOff + iYBlock);
    2493          22 :                     if (poBlock != nullptr)
    2494             :                     {
    2495             :                         // Yes, no need to do further work
    2496           1 :                         poBlock->DropLock();
    2497           1 :                         continue;
    2498             :                     }
    2499             :                     // Instantiate the block
    2500          42 :                     poBlock = poIterBand->GetLockedBlockRef(
    2501          21 :                         nBlockXOff + iXBlock, nBlockYOff + iYBlock, TRUE);
    2502          21 :                     if (poBlock == nullptr)
    2503             :                     {
    2504           0 :                         continue;
    2505             :                     }
    2506          21 :                     pabyDstBuffer = static_cast<GByte *>(poBlock->GetDataRef());
    2507             :                 }
    2508             : 
    2509          23 :                 GDALRasterBand *poTileBand = poTileDS->GetRasterBand(i + 1);
    2510          23 :                 const auto eIterBandDT = poIterBand->GetRasterDataType();
    2511          23 :                 const int nDTSize = GDALGetDataTypeSizeBytes(eIterBandDT);
    2512          46 :                 eErr = poTileBand->RasterIO(
    2513          23 :                     GF_Read, iXBlock * nBlockXSize, iYBlock * nBlockYSize,
    2514             :                     nBlockActualXSize, nBlockActualYSize, pabyDstBuffer,
    2515             :                     nBlockActualXSize, nBlockActualYSize, eIterBandDT, nDTSize,
    2516          23 :                     static_cast<GSpacing>(nDTSize) * nBlockXSize, nullptr);
    2517             : 
    2518          23 :                 if (poBlock)
    2519          21 :                     poBlock->DropLock();
    2520          23 :                 if (eErr != CE_None)
    2521           0 :                     break;
    2522             :             }
    2523             :         }
    2524             :     }
    2525             : 
    2526          14 :     CPLHTTPDestroyResult(psResult);
    2527          14 :     return eErr;
    2528             : }
    2529             : 
    2530             : /************************************************************************/
    2531             : /*                       GDALRegister_DAAS()                            */
    2532             : /************************************************************************/
    2533             : 
    2534        1682 : void GDALRegister_DAAS()
    2535             : 
    2536             : {
    2537        1682 :     if (GDALGetDriverByName("DAAS") != nullptr)
    2538         301 :         return;
    2539             : 
    2540        1381 :     GDALDriver *poDriver = new GDALDriver();
    2541             : 
    2542        1381 :     poDriver->SetDescription("DAAS");
    2543        1381 :     poDriver->SetMetadataItem(GDAL_DCAP_RASTER, "YES");
    2544        1381 :     poDriver->SetMetadataItem(GDAL_DMD_LONGNAME, "Airbus DS Intelligence "
    2545        1381 :                                                  "Data As A Service driver");
    2546        1381 :     poDriver->SetMetadataItem(GDAL_DMD_HELPTOPIC, "drivers/raster/daas.html");
    2547             : 
    2548        1381 :     poDriver->SetMetadataItem(
    2549             :         GDAL_DMD_OPENOPTIONLIST,
    2550             :         "<OpenOptionList>"
    2551             :         "  <Option name='GET_METADATA_URL' type='string' "
    2552             :         "description='URL to GetImageMetadata' "
    2553             :         "required='true'/>"
    2554             :         "  <Option name='API_KEY' alt_config_option='GDAL_DAAS_API_KEY' "
    2555             :         "type='string' "
    2556             :         "description='API key'/>"
    2557             :         "  <Option name='CLIENT_ID' alt_config_option='GDAL_DAAS_CLIENT_ID' "
    2558             :         "type='string' description='Client id'/>"
    2559             :         "  <Option name='ACCESS_TOKEN' "
    2560             :         "alt_config_option='GDAL_DAAS_ACCESS_TOKEN' "
    2561             :         "type='string' description='Authorization access token'/>"
    2562             :         "  <Option name='X_FORWARDED_USER' "
    2563             :         "alt_config_option='GDAL_DAAS_X_FORWARDED_USER' type='string' "
    2564             :         "description='User from which the request originates from'/>"
    2565             :         "  <Option name='BLOCK_SIZE' type='integer' "
    2566             :         "description='Size of a block' default='512'/>"
    2567             :         "  <Option name='PIXEL_ENCODING' type='string-select' "
    2568             :         "description='Format in which pixels are queried'>"
    2569             :         "       <Value>AUTO</Value>"
    2570             :         "       <Value>RAW</Value>"
    2571             :         "       <Value>PNG</Value>"
    2572             :         "       <Value>JPEG</Value>"
    2573             :         "       <Value>JPEG2000</Value>"
    2574             :         "   </Option>"
    2575             :         "  <Option name='TARGET_SRS' type='string' description="
    2576             :         "'SRS name for server-side reprojection.'/>"
    2577             :         "  <Option name='MASKS' type='boolean' "
    2578             :         "description='Whether to expose mask bands' default='YES'/>"
    2579        1381 :         "</OpenOptionList>");
    2580             : 
    2581        1381 :     poDriver->SetMetadataItem(GDAL_DMD_CONNECTION_PREFIX, "DAAS:");
    2582             : 
    2583        1381 :     poDriver->pfnIdentify = GDALDAASDataset::Identify;
    2584        1381 :     poDriver->pfnOpen = GDALDAASDataset::OpenStatic;
    2585             : 
    2586        1381 :     GetGDALDriverManager()->RegisterDriver(poDriver);
    2587             : }

Generated by: LCOV version 1.14