LCOV - code coverage report
Current view: top level - gcore - gdalproxypool.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 575 681 84.4 %
Date: 2025-01-18 12:42:00 Functions: 63 75 84.0 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  GDAL Core
       4             :  * Purpose:  A dataset and raster band classes that differ the opening of the
       5             :  *           underlying dataset in a limited pool of opened datasets.
       6             :  * Author:   Even Rouault <even dot rouault at spatialys.com>
       7             :  *
       8             :  ******************************************************************************
       9             :  * Copyright (c) 2008-2013, Even Rouault <even dot rouault at spatialys.com>
      10             :  *
      11             :  * SPDX-License-Identifier: MIT
      12             :  ****************************************************************************/
      13             : 
      14             : #include "cpl_port.h"
      15             : #include "gdal_proxy.h"
      16             : 
      17             : #include <algorithm>
      18             : #include <cstdio>
      19             : #include <cstdlib>
      20             : #include <cstring>
      21             : 
      22             : #include "cpl_conv.h"
      23             : #include "cpl_error.h"
      24             : #include "cpl_hash_set.h"
      25             : #include "cpl_multiproc.h"
      26             : #include "cpl_string.h"
      27             : #include "gdal.h"
      28             : #include "gdal_priv.h"
      29             : 
      30             : //! @cond Doxygen_Suppress
      31             : 
      32             : /* We *must* share the same mutex as the gdaldataset.cpp file, as we are */
      33             : /* doing GDALOpen() calls that can indirectly call GDALOpenShared() on */
      34             : /* an auxiliary dataset ... */
      35             : /* Then we could get dead-locks in multi-threaded use case */
      36             : 
      37             : /* ******************************************************************** */
      38             : /*                         GDALDatasetPool                              */
      39             : /* ******************************************************************** */
      40             : 
      41             : /* This class is a singleton that maintains a pool of opened datasets */
      42             : /* The cache uses a LRU strategy */
      43             : 
      44             : class GDALDatasetPool;
      45             : static GDALDatasetPool *singleton = nullptr;
      46             : 
      47           0 : void GDALNullifyProxyPoolSingleton()
      48             : {
      49           0 :     singleton = nullptr;
      50           0 : }
      51             : 
      52             : struct _GDALProxyPoolCacheEntry
      53             : {
      54             :     GIntBig responsiblePID;
      55             :     char *pszFileNameAndOpenOptions;
      56             :     char *pszOwner;
      57             :     GDALDataset *poDS;
      58             :     GIntBig nRAMUsage;
      59             : 
      60             :     /* Ref count of the cached dataset */
      61             :     int refCount;
      62             : 
      63             :     GDALProxyPoolCacheEntry *prev;
      64             :     GDALProxyPoolCacheEntry *next;
      65             : };
      66             : 
      67             : // This variable prevents a dataset that is going to be opened in
      68             : // GDALDatasetPool::_RefDataset from increasing refCount if, during its
      69             : // opening, it creates a GDALProxyPoolDataset.
      70             : // We increment it before opening or closing a cached dataset and decrement
      71             : // it afterwards
      72             : // The typical use case is a VRT made of simple sources that are VRT
      73             : // We don't want the "inner" VRT to take a reference on the pool, otherwise
      74             : // there is a high chance that this reference will not be dropped and the pool
      75             : // remain ghost.
      76             : static thread_local int refCountOfDisabledRefCount = 0;
      77             : 
      78             : class GDALDatasetPool
      79             : {
      80             :   private:
      81             :     bool bInDestruction = false;
      82             : 
      83             :     /* Ref count of the pool singleton */
      84             :     /* Taken by "toplevel" GDALProxyPoolDataset in its constructor and released
      85             :      */
      86             :     /* in its destructor. See also refCountOfDisabledRefCount for the difference
      87             :      */
      88             :     /* between toplevel and inner GDALProxyPoolDataset */
      89             :     int refCount = 0;
      90             : 
      91             :     int maxSize = 0;
      92             :     int currentSize = 0;
      93             :     int64_t nMaxRAMUsage = 0;
      94             :     int64_t nRAMUsage = 0;
      95             :     GDALProxyPoolCacheEntry *firstEntry = nullptr;
      96             :     GDALProxyPoolCacheEntry *lastEntry = nullptr;
      97             : 
      98             :     /* Caution : to be sure that we don't run out of entries, size must be at */
      99             :     /* least greater or equal than the maximum number of threads */
     100             :     explicit GDALDatasetPool(int maxSize, int64_t nMaxRAMUsage);
     101             :     ~GDALDatasetPool();
     102             :     GDALProxyPoolCacheEntry *_RefDataset(const char *pszFileName,
     103             :                                          GDALAccess eAccess,
     104             :                                          CSLConstList papszOpenOptions,
     105             :                                          int bShared, bool bForceOpen,
     106             :                                          const char *pszOwner);
     107             :     void _CloseDatasetIfZeroRefCount(const char *pszFileName,
     108             :                                      CSLConstList papszOpenOptions,
     109             :                                      GDALAccess eAccess, const char *pszOwner);
     110             : 
     111             : #ifdef DEBUG_PROXY_POOL
     112             :     // cppcheck-suppress unusedPrivateFunction
     113             :     void ShowContent();
     114             :     void CheckLinks();
     115             : #endif
     116             : 
     117             :     CPL_DISALLOW_COPY_ASSIGN(GDALDatasetPool)
     118             : 
     119             :   public:
     120             :     static void Ref();
     121             :     static void Unref();
     122             :     static GDALProxyPoolCacheEntry *RefDataset(const char *pszFileName,
     123             :                                                GDALAccess eAccess,
     124             :                                                char **papszOpenOptions,
     125             :                                                int bShared, bool bForceOpen,
     126             :                                                const char *pszOwner);
     127             :     static void UnrefDataset(GDALProxyPoolCacheEntry *cacheEntry);
     128             :     static void CloseDatasetIfZeroRefCount(const char *pszFileName,
     129             :                                            CSLConstList papszOpenOptions,
     130             :                                            GDALAccess eAccess,
     131             :                                            const char *pszOwner);
     132             : 
     133             :     static void PreventDestroy();
     134             :     static void ForceDestroy();
     135             : };
     136             : 
     137             : /************************************************************************/
     138             : /*                         GDALDatasetPool()                            */
     139             : /************************************************************************/
     140             : 
     141         358 : GDALDatasetPool::GDALDatasetPool(int maxSizeIn, int64_t nMaxRAMUsageIn)
     142         358 :     : maxSize(maxSizeIn), nMaxRAMUsage(nMaxRAMUsageIn)
     143             : {
     144         358 : }
     145             : 
     146             : /************************************************************************/
     147             : /*                        ~GDALDatasetPool()                            */
     148             : /************************************************************************/
     149             : 
     150         710 : GDALDatasetPool::~GDALDatasetPool()
     151             : {
     152         355 :     bInDestruction = true;
     153         355 :     GDALProxyPoolCacheEntry *cur = firstEntry;
     154         355 :     GIntBig responsiblePID = GDALGetResponsiblePIDForCurrentThread();
     155         699 :     while (cur)
     156             :     {
     157         344 :         GDALProxyPoolCacheEntry *next = cur->next;
     158         344 :         CPLFree(cur->pszFileNameAndOpenOptions);
     159         344 :         CPLFree(cur->pszOwner);
     160         344 :         CPLAssert(cur->refCount == 0);
     161         344 :         if (cur->poDS)
     162             :         {
     163           0 :             GDALSetResponsiblePIDForCurrentThread(cur->responsiblePID);
     164           0 :             GDALClose(cur->poDS);
     165             :         }
     166         344 :         CPLFree(cur);
     167         344 :         cur = next;
     168             :     }
     169         355 :     GDALSetResponsiblePIDForCurrentThread(responsiblePID);
     170         355 : }
     171             : 
     172             : #ifdef DEBUG_PROXY_POOL
     173             : /************************************************************************/
     174             : /*                            ShowContent()                             */
     175             : /************************************************************************/
     176             : 
     177             : void GDALDatasetPool::ShowContent()
     178             : {
     179             :     GDALProxyPoolCacheEntry *cur = firstEntry;
     180             :     int i = 0;
     181             :     while (cur)
     182             :     {
     183             :         printf("[%d] pszFileName=%s, owner=%s, refCount=%d, " /*ok*/
     184             :                "responsiblePID=%d\n",
     185             :                i,
     186             :                cur->pszFileNameAndOpenOptions ? cur->pszFileNameAndOpenOptions
     187             :                                               : "(null)",
     188             :                cur->pszOwner ? cur->pszOwner : "(null)", cur->refCount,
     189             :                (int)cur->responsiblePID);
     190             :         i++;
     191             :         cur = cur->next;
     192             :     }
     193             : }
     194             : 
     195             : /************************************************************************/
     196             : /*                             CheckLinks()                             */
     197             : /************************************************************************/
     198             : 
     199             : void GDALDatasetPool::CheckLinks()
     200             : {
     201             :     GDALProxyPoolCacheEntry *cur = firstEntry;
     202             :     int i = 0;
     203             :     while (cur)
     204             :     {
     205             :         CPLAssert(cur == firstEntry || cur->prev->next == cur);
     206             :         CPLAssert(cur == lastEntry || cur->next->prev == cur);
     207             :         ++i;
     208             :         CPLAssert(cur->next != nullptr || cur == lastEntry);
     209             :         cur = cur->next;
     210             :     }
     211             :     (void)i;
     212             :     CPLAssert(i == currentSize);
     213             : }
     214             : #endif
     215             : 
     216             : /************************************************************************/
     217             : /*                       GetFilenameAndOpenOptions()                    */
     218             : /************************************************************************/
     219             : 
     220       28282 : static std::string GetFilenameAndOpenOptions(const char *pszFileName,
     221             :                                              CSLConstList papszOpenOptions)
     222             : {
     223       28282 :     std::string osFilenameAndOO(pszFileName);
     224       28320 :     for (int i = 0; papszOpenOptions && papszOpenOptions[i]; ++i)
     225             :     {
     226          38 :         osFilenameAndOO += "||";
     227          38 :         osFilenameAndOO += papszOpenOptions[i];
     228             :     }
     229       28282 :     return osFilenameAndOO;
     230             : }
     231             : 
     232             : /************************************************************************/
     233             : /*                            _RefDataset()                             */
     234             : /************************************************************************/
     235             : 
     236             : GDALProxyPoolCacheEntry *
     237       25752 : GDALDatasetPool::_RefDataset(const char *pszFileName, GDALAccess eAccess,
     238             :                              CSLConstList papszOpenOptions, int bShared,
     239             :                              bool bForceOpen, const char *pszOwner)
     240             : {
     241       25752 :     CPLMutex **pMutex = GDALGetphDLMutex();
     242       51503 :     CPLMutexHolderD(pMutex);
     243             : 
     244       25753 :     if (bInDestruction)
     245           0 :         return nullptr;
     246             : 
     247       25753 :     const GIntBig responsiblePID = GDALGetResponsiblePIDForCurrentThread();
     248             : 
     249             :     const auto EvictEntryWithZeroRefCount =
     250        4946 :         [this, responsiblePID](bool evictEntryWithOpenedDataset)
     251             :     {
     252         824 :         GDALProxyPoolCacheEntry *cur = firstEntry;
     253         824 :         GDALProxyPoolCacheEntry *candidate = nullptr;
     254       83030 :         while (cur)
     255             :         {
     256       82206 :             GDALProxyPoolCacheEntry *next = cur->next;
     257             : 
     258       82206 :             if (cur->refCount == 0 &&
     259       79258 :                 (!evictEntryWithOpenedDataset || cur->nRAMUsage > 0))
     260             :             {
     261       79258 :                 candidate = cur;
     262             :             }
     263             : 
     264       82206 :             cur = next;
     265             :         }
     266         824 :         if (candidate == nullptr)
     267           0 :             return false;
     268             : 
     269         824 :         nRAMUsage -= candidate->nRAMUsage;
     270         824 :         candidate->nRAMUsage = 0;
     271             : 
     272         824 :         CPLFree(candidate->pszFileNameAndOpenOptions);
     273         824 :         candidate->pszFileNameAndOpenOptions = nullptr;
     274             : 
     275         824 :         if (candidate->poDS)
     276             :         {
     277             :             /* Close by pretending we are the thread that GDALOpen'ed this */
     278             :             /* dataset */
     279           2 :             GDALSetResponsiblePIDForCurrentThread(candidate->responsiblePID);
     280             : 
     281           2 :             refCountOfDisabledRefCount++;
     282           2 :             GDALClose(candidate->poDS);
     283           2 :             refCountOfDisabledRefCount--;
     284             : 
     285           2 :             candidate->poDS = nullptr;
     286           2 :             GDALSetResponsiblePIDForCurrentThread(responsiblePID);
     287             :         }
     288         824 :         CPLFree(candidate->pszOwner);
     289         824 :         candidate->pszOwner = nullptr;
     290             : 
     291         824 :         if (!evictEntryWithOpenedDataset && candidate != firstEntry)
     292             :         {
     293             :             /* Recycle this entry for the to-be-opened dataset and */
     294             :             /* moves it to the top of the list */
     295         824 :             if (candidate->prev)
     296         824 :                 candidate->prev->next = candidate->next;
     297             : 
     298         824 :             if (candidate->next)
     299           0 :                 candidate->next->prev = candidate->prev;
     300             :             else
     301             :             {
     302         824 :                 CPLAssert(candidate == lastEntry);
     303         824 :                 lastEntry->prev->next = nullptr;
     304         824 :                 lastEntry = lastEntry->prev;
     305             :             }
     306         824 :             candidate->prev = nullptr;
     307         824 :             candidate->next = firstEntry;
     308         824 :             firstEntry->prev = candidate;
     309         824 :             firstEntry = candidate;
     310             : 
     311             : #ifdef DEBUG_PROXY_POOL
     312             :             CheckLinks();
     313             : #endif
     314             :         }
     315             : 
     316         824 :         return true;
     317       25753 :     };
     318             : 
     319       25753 :     GDALProxyPoolCacheEntry *cur = firstEntry;
     320             : 
     321             :     const std::string osFilenameAndOO =
     322       51506 :         GetFilenameAndOpenOptions(pszFileName, papszOpenOptions);
     323             : 
     324      322781 :     while (cur)
     325             :     {
     326      319209 :         GDALProxyPoolCacheEntry *next = cur->next;
     327             : 
     328      397324 :         if (cur->refCount >= 0 && cur->pszFileNameAndOpenOptions &&
     329      738754 :             osFilenameAndOO == cur->pszFileNameAndOpenOptions &&
     330       22122 :             ((bShared && cur->responsiblePID == responsiblePID &&
     331       22119 :               ((cur->pszOwner == nullptr && pszOwner == nullptr) ||
     332       21411 :                (cur->pszOwner != nullptr && pszOwner != nullptr &&
     333       21513 :                 strcmp(cur->pszOwner, pszOwner) == 0))) ||
     334          99 :              (!bShared && cur->refCount == 0)))
     335             :         {
     336       22181 :             if (cur != firstEntry)
     337             :             {
     338             :                 /* Move to begin */
     339        1394 :                 if (cur->next)
     340        1214 :                     cur->next->prev = cur->prev;
     341             :                 else
     342         180 :                     lastEntry = cur->prev;
     343        1394 :                 cur->prev->next = cur->next;
     344        1394 :                 cur->prev = nullptr;
     345        1394 :                 firstEntry->prev = cur;
     346        1394 :                 cur->next = firstEntry;
     347        1394 :                 firstEntry = cur;
     348             : 
     349             : #ifdef DEBUG_PROXY_POOL
     350             :                 CheckLinks();
     351             : #endif
     352             :             }
     353             : 
     354       22181 :             cur->refCount++;
     355       22181 :             return cur;
     356             :         }
     357             : 
     358      297028 :         cur = next;
     359             :     }
     360             : 
     361        3572 :     if (!bForceOpen)
     362        2143 :         return nullptr;
     363             : 
     364        1429 :     if (currentSize == maxSize)
     365             :     {
     366         824 :         if (!EvictEntryWithZeroRefCount(false))
     367             :         {
     368           0 :             CPLError(
     369             :                 CE_Failure, CPLE_AppDefined,
     370             :                 "Too many threads are running for the current value of the "
     371             :                 "dataset pool size (%d).\n"
     372             :                 "or too many proxy datasets are opened in a cascaded way.\n"
     373             :                 "Try increasing GDAL_MAX_DATASET_POOL_SIZE.",
     374             :                 maxSize);
     375           0 :             return nullptr;
     376             :         }
     377             : 
     378         824 :         CPLAssert(firstEntry);
     379         824 :         cur = firstEntry;
     380             :     }
     381             :     else
     382             :     {
     383             :         /* Prepend */
     384             :         cur = static_cast<GDALProxyPoolCacheEntry *>(
     385         605 :             CPLCalloc(1, sizeof(GDALProxyPoolCacheEntry)));
     386         605 :         if (lastEntry == nullptr)
     387         306 :             lastEntry = cur;
     388         605 :         cur->prev = nullptr;
     389         605 :         cur->next = firstEntry;
     390         605 :         if (firstEntry)
     391         299 :             firstEntry->prev = cur;
     392         605 :         firstEntry = cur;
     393         605 :         currentSize++;
     394             : #ifdef DEBUG_PROXY_POOL
     395             :         CheckLinks();
     396             : #endif
     397             :     }
     398             : 
     399        1429 :     cur->pszFileNameAndOpenOptions = CPLStrdup(osFilenameAndOO.c_str());
     400        1429 :     cur->pszOwner = (pszOwner) ? CPLStrdup(pszOwner) : nullptr;
     401        1429 :     cur->responsiblePID = responsiblePID;
     402        1429 :     cur->refCount = -1;  // to mark loading of dataset in progress
     403        1429 :     cur->nRAMUsage = 0;
     404             : 
     405        1429 :     refCountOfDisabledRefCount++;
     406        1429 :     const int nFlag =
     407             :         ((eAccess == GA_Update) ? GDAL_OF_UPDATE : GDAL_OF_READONLY) |
     408        1429 :         GDAL_OF_RASTER | GDAL_OF_VERBOSE_ERROR;
     409        1429 :     CPLConfigOptionSetter oSetter("CPL_ALLOW_VSISTDIN", "NO", true);
     410             : 
     411             :     // Release mutex while opening dataset to avoid lock contention.
     412        1429 :     CPLReleaseMutex(*pMutex);
     413        1429 :     auto poDS = GDALDataset::Open(pszFileName, nFlag, nullptr, papszOpenOptions,
     414             :                                   nullptr);
     415        1429 :     CPLAcquireMutex(*pMutex, 1000.0);
     416             : 
     417        1429 :     cur->poDS = poDS;
     418        1429 :     cur->refCount = 1;
     419             : 
     420        1429 :     refCountOfDisabledRefCount--;
     421             : 
     422        1429 :     if (cur->poDS)
     423             :     {
     424        1219 :         cur->nRAMUsage =
     425        1219 :             std::max<GIntBig>(0, cur->poDS->GetEstimatedRAMUsage());
     426        1219 :         nRAMUsage += cur->nRAMUsage;
     427             :     }
     428             : 
     429        1429 :     if (nMaxRAMUsage > 0 && cur->nRAMUsage > 0)
     430             :     {
     431         100 :         while (nRAMUsage > nMaxRAMUsage && nRAMUsage != cur->nRAMUsage &&
     432           0 :                EvictEntryWithZeroRefCount(true))
     433             :         {
     434             :             // ok
     435             :         }
     436             :     }
     437             : 
     438        1429 :     return cur;
     439             : }
     440             : 
     441             : /************************************************************************/
     442             : /*                   _CloseDatasetIfZeroRefCount()                      */
     443             : /************************************************************************/
     444             : 
     445        2529 : void GDALDatasetPool::_CloseDatasetIfZeroRefCount(const char *pszFileName,
     446             :                                                   CSLConstList papszOpenOptions,
     447             :                                                   GDALAccess /* eAccess */,
     448             :                                                   const char *pszOwner)
     449             : {
     450             :     // May fix https://github.com/OSGeo/gdal/issues/4318
     451        2529 :     if (bInDestruction)
     452           0 :         return;
     453             : 
     454        2529 :     GDALProxyPoolCacheEntry *cur = firstEntry;
     455        2529 :     GIntBig responsiblePID = GDALGetResponsiblePIDForCurrentThread();
     456             : 
     457             :     const std::string osFilenameAndOO =
     458        5058 :         GetFilenameAndOpenOptions(pszFileName, papszOpenOptions);
     459             : 
     460       78654 :     while (cur)
     461             :     {
     462       77342 :         GDALProxyPoolCacheEntry *next = cur->next;
     463             : 
     464       94504 :         if (cur->refCount == 0 && cur->pszFileNameAndOpenOptions &&
     465       18608 :             osFilenameAndOO == cur->pszFileNameAndOpenOptions &&
     466        1442 :             ((pszOwner == nullptr && cur->pszOwner == nullptr) ||
     467        1272 :              (pszOwner != nullptr && cur->pszOwner != nullptr &&
     468      155951 :               strcmp(cur->pszOwner, pszOwner) == 0)) &&
     469        1432 :             cur->poDS != nullptr)
     470             :         {
     471             :             /* Close by pretending we are the thread that GDALOpen'ed this */
     472             :             /* dataset */
     473        1217 :             GDALSetResponsiblePIDForCurrentThread(cur->responsiblePID);
     474             : 
     475        1217 :             GDALDataset *poDS = cur->poDS;
     476             : 
     477        1217 :             nRAMUsage -= cur->nRAMUsage;
     478        1217 :             cur->nRAMUsage = 0;
     479             : 
     480        1217 :             cur->poDS = nullptr;
     481        1217 :             CPLFree(cur->pszFileNameAndOpenOptions);
     482        1217 :             cur->pszFileNameAndOpenOptions = nullptr;
     483        1217 :             CPLFree(cur->pszOwner);
     484        1217 :             cur->pszOwner = nullptr;
     485             : 
     486        1217 :             refCountOfDisabledRefCount++;
     487        1217 :             GDALClose(poDS);
     488        1217 :             refCountOfDisabledRefCount--;
     489             : 
     490        1217 :             GDALSetResponsiblePIDForCurrentThread(responsiblePID);
     491        1217 :             break;
     492             :         }
     493             : 
     494       76125 :         cur = next;
     495             :     }
     496             : }
     497             : 
     498             : /************************************************************************/
     499             : /*                       GDALGetMaxDatasetPoolSize()                    */
     500             : /************************************************************************/
     501             : 
     502             : /** Return the maximum number of datasets simultaneously opened in the
     503             :  * dataset pool.
     504             :  */
     505         395 : int GDALGetMaxDatasetPoolSize()
     506             : {
     507         395 :     int nSize = atoi(CPLGetConfigOption("GDAL_MAX_DATASET_POOL_SIZE", "100"));
     508         395 :     if (nSize < 2)
     509           0 :         nSize = 2;
     510         395 :     else if (nSize > 1000)
     511           0 :         nSize = 1000;
     512         395 :     return nSize;
     513             : }
     514             : 
     515             : /************************************************************************/
     516             : /*                                 Ref()                                */
     517             : /************************************************************************/
     518             : 
     519        2528 : void GDALDatasetPool::Ref()
     520             : {
     521        5057 :     CPLMutexHolderD(GDALGetphDLMutex());
     522        2529 :     if (singleton == nullptr)
     523             :     {
     524             : 
     525             :         // Try to not consume more than 25% of the usable RAM
     526             :         GIntBig l_nMaxRAMUsage =
     527         358 :             (CPLGetUsablePhysicalRAM() - GDALGetCacheMax64()) / 4;
     528             :         const char *pszMaxRAMUsage =
     529         358 :             CPLGetConfigOption("GDAL_MAX_DATASET_POOL_RAM_USAGE", nullptr);
     530         358 :         if (pszMaxRAMUsage)
     531             :         {
     532           0 :             l_nMaxRAMUsage = std::strtoll(pszMaxRAMUsage, nullptr, 10);
     533           0 :             if (strstr(pszMaxRAMUsage, "MB"))
     534           0 :                 l_nMaxRAMUsage *= 1024 * 1024;
     535           0 :             else if (strstr(pszMaxRAMUsage, "GB"))
     536           0 :                 l_nMaxRAMUsage *= 1024 * 1024 * 1024;
     537             :         }
     538             : 
     539         358 :         singleton =
     540         358 :             new GDALDatasetPool(GDALGetMaxDatasetPoolSize(), l_nMaxRAMUsage);
     541             :     }
     542        2529 :     if (refCountOfDisabledRefCount == 0)
     543        2529 :         singleton->refCount++;
     544        2529 : }
     545             : 
     546             : /* keep that in sync with gdaldrivermanager.cpp */
     547         941 : void GDALDatasetPool::PreventDestroy()
     548             : {
     549         941 :     CPLMutexHolderD(GDALGetphDLMutex());
     550         941 :     if (!singleton)
     551         938 :         return;
     552           3 :     refCountOfDisabledRefCount++;
     553             : }
     554             : 
     555             : /* keep that in sync with gdaldrivermanager.cpp */
     556             : extern void GDALDatasetPoolPreventDestroy();
     557             : 
     558         941 : void GDALDatasetPoolPreventDestroy()
     559             : {
     560         941 :     GDALDatasetPool::PreventDestroy();
     561         941 : }
     562             : 
     563             : /************************************************************************/
     564             : /*                               Unref()                                */
     565             : /************************************************************************/
     566             : 
     567        2529 : void GDALDatasetPool::Unref()
     568             : {
     569        2529 :     CPLMutexHolderD(GDALGetphDLMutex());
     570        2529 :     if (!singleton)
     571             :     {
     572           0 :         CPLAssert(false);
     573             :         return;
     574             :     }
     575        2529 :     if (refCountOfDisabledRefCount == 0)
     576             :     {
     577        2499 :         singleton->refCount--;
     578        2499 :         if (singleton->refCount == 0)
     579             :         {
     580         352 :             delete singleton;
     581         352 :             singleton = nullptr;
     582             :         }
     583             :     }
     584             : }
     585             : 
     586             : /* keep that in sync with gdaldrivermanager.cpp */
     587         941 : void GDALDatasetPool::ForceDestroy()
     588             : {
     589         941 :     CPLMutexHolderD(GDALGetphDLMutex());
     590         941 :     if (!singleton)
     591         938 :         return;
     592           3 :     refCountOfDisabledRefCount--;
     593           3 :     CPLAssert(refCountOfDisabledRefCount == 0);
     594           3 :     singleton->refCount = 0;
     595           3 :     delete singleton;
     596           3 :     singleton = nullptr;
     597             : }
     598             : 
     599             : /* keep that in sync with gdaldrivermanager.cpp */
     600             : extern void GDALDatasetPoolForceDestroy();
     601             : 
     602         941 : void GDALDatasetPoolForceDestroy()
     603             : {
     604         941 :     GDALDatasetPool::ForceDestroy();
     605         941 : }
     606             : 
     607             : /************************************************************************/
     608             : /*                           RefDataset()                               */
     609             : /************************************************************************/
     610             : 
     611             : GDALProxyPoolCacheEntry *
     612       25753 : GDALDatasetPool::RefDataset(const char *pszFileName, GDALAccess eAccess,
     613             :                             char **papszOpenOptions, int bShared,
     614             :                             bool bForceOpen, const char *pszOwner)
     615             : {
     616       25753 :     return singleton->_RefDataset(pszFileName, eAccess, papszOpenOptions,
     617       25753 :                                   bShared, bForceOpen, pszOwner);
     618             : }
     619             : 
     620             : /************************************************************************/
     621             : /*                       UnrefDataset()                                 */
     622             : /************************************************************************/
     623             : 
     624       23609 : void GDALDatasetPool::UnrefDataset(GDALProxyPoolCacheEntry *cacheEntry)
     625             : {
     626       23609 :     CPLMutexHolderD(GDALGetphDLMutex());
     627       23610 :     cacheEntry->refCount--;
     628       23610 : }
     629             : 
     630             : /************************************************************************/
     631             : /*                   CloseDatasetIfZeroRefCount()                       */
     632             : /************************************************************************/
     633             : 
     634        2529 : void GDALDatasetPool::CloseDatasetIfZeroRefCount(const char *pszFileName,
     635             :                                                  CSLConstList papszOpenOptions,
     636             :                                                  GDALAccess eAccess,
     637             :                                                  const char *pszOwner)
     638             : {
     639        5058 :     CPLMutexHolderD(GDALGetphDLMutex());
     640        2529 :     singleton->_CloseDatasetIfZeroRefCount(pszFileName, papszOpenOptions,
     641             :                                            eAccess, pszOwner);
     642        2529 : }
     643             : 
     644             : struct GetMetadataElt
     645             : {
     646             :     char *pszDomain;
     647             :     char **papszMetadata;
     648             : };
     649             : 
     650          42 : static unsigned long hash_func_get_metadata(const void *_elt)
     651             : {
     652          42 :     const GetMetadataElt *elt = static_cast<const GetMetadataElt *>(_elt);
     653          42 :     return CPLHashSetHashStr(elt->pszDomain);
     654             : }
     655             : 
     656           8 : static int equal_func_get_metadata(const void *_elt1, const void *_elt2)
     657             : {
     658           8 :     const GetMetadataElt *elt1 = static_cast<const GetMetadataElt *>(_elt1);
     659           8 :     const GetMetadataElt *elt2 = static_cast<const GetMetadataElt *>(_elt2);
     660           8 :     return CPLHashSetEqualStr(elt1->pszDomain, elt2->pszDomain);
     661             : }
     662             : 
     663          24 : static void free_func_get_metadata(void *_elt)
     664             : {
     665          24 :     GetMetadataElt *elt = static_cast<GetMetadataElt *>(_elt);
     666          24 :     CPLFree(elt->pszDomain);
     667          24 :     CSLDestroy(elt->papszMetadata);
     668          24 :     CPLFree(elt);
     669          24 : }
     670             : 
     671             : struct GetMetadataItemElt
     672             : {
     673             :     char *pszName;
     674             :     char *pszDomain;
     675             :     char *pszMetadataItem;
     676             : };
     677             : 
     678          68 : static unsigned long hash_func_get_metadata_item(const void *_elt)
     679             : {
     680          68 :     const GetMetadataItemElt *elt =
     681             :         static_cast<const GetMetadataItemElt *>(_elt);
     682          68 :     return CPLHashSetHashStr(elt->pszName) ^ CPLHashSetHashStr(elt->pszDomain);
     683             : }
     684             : 
     685          18 : static int equal_func_get_metadata_item(const void *_elt1, const void *_elt2)
     686             : {
     687          18 :     const GetMetadataItemElt *elt1 =
     688             :         static_cast<const GetMetadataItemElt *>(_elt1);
     689          18 :     const GetMetadataItemElt *elt2 =
     690             :         static_cast<const GetMetadataItemElt *>(_elt2);
     691          36 :     return CPLHashSetEqualStr(elt1->pszName, elt2->pszName) &&
     692          36 :            CPLHashSetEqualStr(elt1->pszDomain, elt2->pszDomain);
     693             : }
     694             : 
     695          43 : static void free_func_get_metadata_item(void *_elt)
     696             : {
     697          43 :     GetMetadataItemElt *elt = static_cast<GetMetadataItemElt *>(_elt);
     698          43 :     CPLFree(elt->pszName);
     699          43 :     CPLFree(elt->pszDomain);
     700          43 :     CPLFree(elt->pszMetadataItem);
     701          43 :     CPLFree(elt);
     702          43 : }
     703             : 
     704             : /* ******************************************************************** */
     705             : /*                     GDALProxyPoolDataset                             */
     706             : /* ******************************************************************** */
     707             : 
     708             : /* Note : the bShared parameter must be used with caution. You can */
     709             : /* set it to TRUE  for being used as a VRT source : in that case, */
     710             : /* VRTSimpleSource will take care of destroying it when there are no */
     711             : /* reference to it (in VRTSimpleSource::~VRTSimpleSource()) */
     712             : /* However this will not be registered as a genuine shared dataset, like it */
     713             : /* would have been with MarkAsShared(). But MarkAsShared() is not usable for */
     714             : /* GDALProxyPoolDataset objects, as they share the same description as their */
     715             : /* underlying dataset. So *NEVER* call MarkAsShared() on a GDALProxyPoolDataset
     716             :  */
     717             : /* object */
     718             : 
     719             : /* pszOwner is only honoured in the bShared case, and restrict the scope */
     720             : /* of the sharing. Only calls to _RefDataset() with the same value of */
     721             : /* pszOwner can effectively use the same dataset. The use case is */
     722             : /* to avoid 2 VRTs (potentially the same one) opened by a single thread,
     723             :  * pointing to */
     724             : /* the same source datasets. In that case, they would use the same dataset */
     725             : /* So even if the VRT handles themselves are used from different threads, since
     726             :  */
     727             : /* the underlying sources are shared, that might cause crashes (#6939). */
     728             : /* But we want to allow a same VRT referencing the same source dataset,*/
     729             : /* for example if it has multiple bands. So in practice the value of pszOwner */
     730             : /* is the serialized value (%p formatting) of the VRT dataset handle. */
     731             : 
     732        1247 : GDALProxyPoolDataset::GDALProxyPoolDataset(
     733             :     const char *pszSourceDatasetDescription, int nRasterXSizeIn,
     734             :     int nRasterYSizeIn, GDALAccess eAccessIn, int bSharedIn,
     735             :     const char *pszProjectionRefIn, double *padfGeoTransform,
     736        1247 :     const char *pszOwner)
     737        2494 :     : responsiblePID(GDALGetResponsiblePIDForCurrentThread()),
     738        1247 :       pszProjectionRef(pszProjectionRefIn ? CPLStrdup(pszProjectionRefIn)
     739        1247 :                                           : nullptr)
     740             : {
     741        1247 :     GDALDatasetPool::Ref();
     742             : 
     743        1247 :     SetDescription(pszSourceDatasetDescription);
     744             : 
     745        1247 :     nRasterXSize = nRasterXSizeIn;
     746        1247 :     nRasterYSize = nRasterYSizeIn;
     747        1247 :     eAccess = eAccessIn;
     748             : 
     749        1247 :     bShared = CPL_TO_BOOL(bSharedIn);
     750        1247 :     m_pszOwner = pszOwner ? CPLStrdup(pszOwner) : nullptr;
     751             : 
     752        1247 :     if (padfGeoTransform)
     753             :     {
     754        1198 :         memcpy(adfGeoTransform, padfGeoTransform, 6 * sizeof(double));
     755        1198 :         bHasSrcGeoTransform = true;
     756             :     }
     757             :     else
     758             :     {
     759          49 :         adfGeoTransform[0] = 0;
     760          49 :         adfGeoTransform[1] = 1;
     761          49 :         adfGeoTransform[2] = 0;
     762          49 :         adfGeoTransform[3] = 0;
     763          49 :         adfGeoTransform[4] = 0;
     764          49 :         adfGeoTransform[5] = 1;
     765          49 :         bHasSrcGeoTransform = false;
     766             :     }
     767             : 
     768        1247 :     if (pszProjectionRefIn)
     769             :     {
     770        1208 :         m_poSRS = new OGRSpatialReference();
     771        1208 :         m_poSRS->importFromWkt(pszProjectionRefIn);
     772        1208 :         m_bHasSrcSRS = true;
     773             :     }
     774        1247 : }
     775             : 
     776             : /* Constructor where the parameters (raster size, etc.) are obtained
     777             :  * by opening the underlying dataset.
     778             :  */
     779        1282 : GDALProxyPoolDataset::GDALProxyPoolDataset(
     780             :     const char *pszSourceDatasetDescription, GDALAccess eAccessIn,
     781        1282 :     int bSharedIn, const char *pszOwner)
     782        1282 :     : responsiblePID(GDALGetResponsiblePIDForCurrentThread())
     783             : {
     784        1281 :     GDALDatasetPool::Ref();
     785             : 
     786        1282 :     SetDescription(pszSourceDatasetDescription);
     787             : 
     788        1282 :     eAccess = eAccessIn;
     789             : 
     790        1282 :     bShared = CPL_TO_BOOL(bSharedIn);
     791        1282 :     m_pszOwner = pszOwner ? CPLStrdup(pszOwner) : nullptr;
     792        1282 : }
     793             : 
     794             : /************************************************************************/
     795             : /*                              Create()                                */
     796             : /************************************************************************/
     797             : 
     798             : /* Instantiate a GDALProxyPoolDataset where the parameters (raster size, etc.)
     799             :  * are obtained by opening the underlying dataset.
     800             :  * Its bands are also instantiated.
     801             :  */
     802        1282 : GDALProxyPoolDataset *GDALProxyPoolDataset::Create(
     803             :     const char *pszSourceDatasetDescription, CSLConstList papszOpenOptionsIn,
     804             :     GDALAccess eAccessIn, int bSharedIn, const char *pszOwner)
     805             : {
     806             :     std::unique_ptr<GDALProxyPoolDataset> poSelf(new GDALProxyPoolDataset(
     807        2564 :         pszSourceDatasetDescription, eAccessIn, bSharedIn, pszOwner));
     808        1282 :     poSelf->SetOpenOptions(papszOpenOptionsIn);
     809        1282 :     GDALDataset *poUnderlyingDS = poSelf->RefUnderlyingDataset();
     810        1282 :     if (!poUnderlyingDS)
     811         215 :         return nullptr;
     812        1067 :     poSelf->nRasterXSize = poUnderlyingDS->GetRasterXSize();
     813        1067 :     poSelf->nRasterYSize = poUnderlyingDS->GetRasterYSize();
     814        1067 :     if (poUnderlyingDS->GetGeoTransform(poSelf->adfGeoTransform) == CE_None)
     815         880 :         poSelf->bHasSrcGeoTransform = true;
     816        1067 :     const auto poSRS = poUnderlyingDS->GetSpatialRef();
     817        1067 :     if (poSRS)
     818             :     {
     819         796 :         poSelf->m_poSRS = poSRS->Clone();
     820         794 :         poSelf->m_bHasSrcSRS = true;
     821             :     }
     822        3205 :     for (int i = 1; i <= poUnderlyingDS->GetRasterCount(); ++i)
     823             :     {
     824        2138 :         auto poSrcBand = poUnderlyingDS->GetRasterBand(i);
     825        2137 :         if (!poSrcBand)
     826             :         {
     827           0 :             poSelf->UnrefUnderlyingDataset(poUnderlyingDS);
     828           0 :             return nullptr;
     829             :         }
     830             :         int nSrcBlockXSize, nSrcBlockYSize;
     831        2137 :         poSrcBand->GetBlockSize(&nSrcBlockXSize, &nSrcBlockYSize);
     832        2137 :         poSelf->AddSrcBandDescription(poSrcBand->GetRasterDataType(),
     833             :                                       nSrcBlockXSize, nSrcBlockYSize);
     834             :     }
     835        1068 :     poSelf->UnrefUnderlyingDataset(poUnderlyingDS);
     836        1067 :     return poSelf.release();
     837             : }
     838             : 
     839             : /************************************************************************/
     840             : /*                    ~GDALProxyPoolDataset()                           */
     841             : /************************************************************************/
     842             : 
     843        5048 : GDALProxyPoolDataset::~GDALProxyPoolDataset()
     844             : {
     845        2529 :     GDALDatasetPool::CloseDatasetIfZeroRefCount(
     846        2529 :         GetDescription(), papszOpenOptions, eAccess, m_pszOwner);
     847             : 
     848             :     /* See comment in constructor */
     849             :     /* It is not really a genuine shared dataset, so we don't */
     850             :     /* want ~GDALDataset() to try to release it from its */
     851             :     /* shared dataset hashset. This will save a */
     852             :     /* "Should not happen. Cannot find %s, this=%p in phSharedDatasetSet" debug
     853             :      * message */
     854        2529 :     bShared = false;
     855             : 
     856        2529 :     CPLFree(pszProjectionRef);
     857        2529 :     CPLFree(pszGCPProjection);
     858        2529 :     if (nGCPCount)
     859             :     {
     860           0 :         GDALDeinitGCPs(nGCPCount, pasGCPList);
     861           0 :         CPLFree(pasGCPList);
     862             :     }
     863        2529 :     if (metadataSet)
     864           4 :         CPLHashSetDestroy(metadataSet);
     865        2529 :     if (metadataItemSet)
     866           2 :         CPLHashSetDestroy(metadataItemSet);
     867        2529 :     CPLFree(m_pszOwner);
     868        2529 :     if (m_poSRS)
     869        2004 :         m_poSRS->Release();
     870        2529 :     if (m_poGCPSRS)
     871           0 :         m_poGCPSRS->Release();
     872             : 
     873        2529 :     GDALDatasetPool::Unref();
     874        5048 : }
     875             : 
     876             : /************************************************************************/
     877             : /*                        SetOpenOptions()                              */
     878             : /************************************************************************/
     879             : 
     880        2480 : void GDALProxyPoolDataset::SetOpenOptions(CSLConstList papszOpenOptionsIn)
     881             : {
     882        2480 :     CPLAssert(papszOpenOptions == nullptr);
     883        2480 :     papszOpenOptions = CSLDuplicate(papszOpenOptionsIn);
     884        2480 : }
     885             : 
     886             : /************************************************************************/
     887             : /*                    AddSrcBandDescription()                           */
     888             : /************************************************************************/
     889             : 
     890        4467 : void GDALProxyPoolDataset::AddSrcBandDescription(GDALDataType eDataType,
     891             :                                                  int nBlockXSize,
     892             :                                                  int nBlockYSize)
     893             : {
     894        4466 :     SetBand(nBands + 1, new GDALProxyPoolRasterBand(this, nBands + 1, eDataType,
     895        4467 :                                                     nBlockXSize, nBlockYSize));
     896        4467 : }
     897             : 
     898             : /************************************************************************/
     899             : /*                    AddSrcBand()                                      */
     900             : /************************************************************************/
     901             : 
     902           0 : void GDALProxyPoolDataset::AddSrcBand(int nBand, GDALDataType eDataType,
     903             :                                       int nBlockXSize, int nBlockYSize)
     904             : {
     905           0 :     SetBand(nBand, new GDALProxyPoolRasterBand(this, nBand, eDataType,
     906           0 :                                                nBlockXSize, nBlockYSize));
     907           0 : }
     908             : 
     909             : /************************************************************************/
     910             : /*                    RefUnderlyingDataset()                            */
     911             : /************************************************************************/
     912             : 
     913        2532 : GDALDataset *GDALProxyPoolDataset::RefUnderlyingDataset() const
     914             : {
     915        2532 :     return RefUnderlyingDataset(true);
     916             : }
     917             : 
     918       25753 : GDALDataset *GDALProxyPoolDataset::RefUnderlyingDataset(bool bForceOpen) const
     919             : {
     920             :     /* We pretend that the current thread is responsiblePID, that is */
     921             :     /* to say the thread that created that GDALProxyPoolDataset object. */
     922             :     /* This is for the case when a GDALProxyPoolDataset is created by a */
     923             :     /* thread and used by other threads. These other threads, when doing actual
     924             :      */
     925             :     /* IO, will come there and potentially open the underlying dataset. */
     926             :     /* By doing this, they can indirectly call GDALOpenShared() on .aux file */
     927             :     /* for example. So this call to GDALOpenShared() must occur as if it */
     928             :     /* was done by the creating thread, otherwise it will not be correctly
     929             :      * closed afterwards... */
     930             :     /* To make a long story short : this is necessary when warping with
     931             :      * ChunkAndWarpMulti */
     932             :     /* a VRT of GeoTIFFs that have associated .aux files */
     933       25753 :     GIntBig curResponsiblePID = GDALGetResponsiblePIDForCurrentThread();
     934       25752 :     GDALSetResponsiblePIDForCurrentThread(responsiblePID);
     935       25753 :     cacheEntry =
     936       25753 :         GDALDatasetPool::RefDataset(GetDescription(), eAccess, papszOpenOptions,
     937       25752 :                                     GetShared(), bForceOpen, m_pszOwner);
     938       25753 :     GDALSetResponsiblePIDForCurrentThread(curResponsiblePID);
     939       25753 :     if (cacheEntry != nullptr)
     940             :     {
     941       23610 :         if (cacheEntry->poDS != nullptr)
     942       23395 :             return cacheEntry->poDS;
     943             :         else
     944         215 :             GDALDatasetPool::UnrefDataset(cacheEntry);
     945             :     }
     946        2358 :     return nullptr;
     947             : }
     948             : 
     949             : /************************************************************************/
     950             : /*                    UnrefUnderlyingDataset()                        */
     951             : /************************************************************************/
     952             : 
     953       23395 : void GDALProxyPoolDataset::UnrefUnderlyingDataset(
     954             :     CPL_UNUSED GDALDataset *poUnderlyingDataset) const
     955             : {
     956       23395 :     if (cacheEntry != nullptr)
     957             :     {
     958       23395 :         CPLAssert(cacheEntry->poDS == poUnderlyingDataset);
     959       23395 :         if (cacheEntry->poDS != nullptr)
     960       23395 :             GDALDatasetPool::UnrefDataset(cacheEntry);
     961             :     }
     962       23395 : }
     963             : 
     964             : /************************************************************************/
     965             : /*                         FlushCache()                                 */
     966             : /************************************************************************/
     967             : 
     968           0 : CPLErr GDALProxyPoolDataset::FlushCache(bool bAtClosing)
     969             : {
     970           0 :     CPLErr eErr = CE_None;
     971           0 :     GDALDataset *poUnderlyingDataset = RefUnderlyingDataset(false);
     972           0 :     if (poUnderlyingDataset)
     973             :     {
     974           0 :         eErr = poUnderlyingDataset->FlushCache(bAtClosing);
     975           0 :         UnrefUnderlyingDataset(poUnderlyingDataset);
     976             :     }
     977           0 :     return eErr;
     978             : }
     979             : 
     980             : /************************************************************************/
     981             : /*                        SetSpatialRef()                               */
     982             : /************************************************************************/
     983             : 
     984           0 : CPLErr GDALProxyPoolDataset::SetSpatialRef(const OGRSpatialReference *poSRS)
     985             : {
     986           0 :     m_bHasSrcSRS = false;
     987           0 :     return GDALProxyDataset::SetSpatialRef(poSRS);
     988             : }
     989             : 
     990             : /************************************************************************/
     991             : /*                        GetSpatialRef()                               */
     992             : /************************************************************************/
     993             : 
     994         200 : const OGRSpatialReference *GDALProxyPoolDataset::GetSpatialRef() const
     995             : {
     996         200 :     if (m_bHasSrcSRS)
     997         198 :         return m_poSRS;
     998             :     else
     999             :     {
    1000           2 :         if (m_poSRS)
    1001           0 :             m_poSRS->Release();
    1002           2 :         m_poSRS = nullptr;
    1003           2 :         auto poSRS = GDALProxyDataset::GetSpatialRef();
    1004           2 :         if (poSRS)
    1005           0 :             m_poSRS = poSRS->Clone();
    1006           2 :         return m_poSRS;
    1007             :     }
    1008             : }
    1009             : 
    1010             : /************************************************************************/
    1011             : /*                        SetGeoTransform()                             */
    1012             : /************************************************************************/
    1013             : 
    1014           0 : CPLErr GDALProxyPoolDataset::SetGeoTransform(double *padfGeoTransform)
    1015             : {
    1016           0 :     bHasSrcGeoTransform = false;
    1017           0 :     return GDALProxyDataset::SetGeoTransform(padfGeoTransform);
    1018             : }
    1019             : 
    1020             : /************************************************************************/
    1021             : /*                        GetGeoTransform()                             */
    1022             : /************************************************************************/
    1023             : 
    1024         565 : CPLErr GDALProxyPoolDataset::GetGeoTransform(double *padfGeoTransform)
    1025             : {
    1026         565 :     if (bHasSrcGeoTransform)
    1027             :     {
    1028         565 :         memcpy(padfGeoTransform, adfGeoTransform, 6 * sizeof(double));
    1029         565 :         return CE_None;
    1030             :     }
    1031             :     else
    1032             :     {
    1033           0 :         return GDALProxyDataset::GetGeoTransform(padfGeoTransform);
    1034             :     }
    1035             : }
    1036             : 
    1037             : /************************************************************************/
    1038             : /*                            GetMetadata()                             */
    1039             : /************************************************************************/
    1040             : 
    1041          22 : char **GDALProxyPoolDataset::GetMetadata(const char *pszDomain)
    1042             : {
    1043          22 :     if (metadataSet == nullptr)
    1044           4 :         metadataSet =
    1045           4 :             CPLHashSetNew(hash_func_get_metadata, equal_func_get_metadata,
    1046             :                           free_func_get_metadata);
    1047             : 
    1048          22 :     GDALDataset *poUnderlyingDataset = RefUnderlyingDataset();
    1049          22 :     if (poUnderlyingDataset == nullptr)
    1050           0 :         return nullptr;
    1051             : 
    1052             :     char **papszUnderlyingMetadata =
    1053          22 :         poUnderlyingDataset->GetMetadata(pszDomain);
    1054             : 
    1055             :     GetMetadataElt *pElt =
    1056          22 :         static_cast<GetMetadataElt *>(CPLMalloc(sizeof(GetMetadataElt)));
    1057          22 :     pElt->pszDomain = (pszDomain) ? CPLStrdup(pszDomain) : nullptr;
    1058          22 :     pElt->papszMetadata = CSLDuplicate(papszUnderlyingMetadata);
    1059          22 :     CPLHashSetInsert(metadataSet, pElt);
    1060             : 
    1061          22 :     UnrefUnderlyingDataset(poUnderlyingDataset);
    1062             : 
    1063          22 :     return pElt->papszMetadata;
    1064             : }
    1065             : 
    1066             : /************************************************************************/
    1067             : /*                        GetMetadataItem()                             */
    1068             : /************************************************************************/
    1069             : 
    1070           4 : const char *GDALProxyPoolDataset::GetMetadataItem(const char *pszName,
    1071             :                                                   const char *pszDomain)
    1072             : {
    1073           4 :     if (metadataItemSet == nullptr)
    1074           2 :         metadataItemSet = CPLHashSetNew(hash_func_get_metadata_item,
    1075             :                                         equal_func_get_metadata_item,
    1076             :                                         free_func_get_metadata_item);
    1077             : 
    1078           4 :     GDALDataset *poUnderlyingDataset = RefUnderlyingDataset();
    1079           4 :     if (poUnderlyingDataset == nullptr)
    1080           0 :         return nullptr;
    1081             : 
    1082             :     const char *pszUnderlyingMetadataItem =
    1083           4 :         poUnderlyingDataset->GetMetadataItem(pszName, pszDomain);
    1084             : 
    1085             :     GetMetadataItemElt *pElt = static_cast<GetMetadataItemElt *>(
    1086           4 :         CPLMalloc(sizeof(GetMetadataItemElt)));
    1087           4 :     pElt->pszName = (pszName) ? CPLStrdup(pszName) : nullptr;
    1088           4 :     pElt->pszDomain = (pszDomain) ? CPLStrdup(pszDomain) : nullptr;
    1089           4 :     pElt->pszMetadataItem = (pszUnderlyingMetadataItem)
    1090           4 :                                 ? CPLStrdup(pszUnderlyingMetadataItem)
    1091             :                                 : nullptr;
    1092           4 :     CPLHashSetInsert(metadataItemSet, pElt);
    1093             : 
    1094           4 :     UnrefUnderlyingDataset(poUnderlyingDataset);
    1095             : 
    1096           4 :     return pElt->pszMetadataItem;
    1097             : }
    1098             : 
    1099             : /************************************************************************/
    1100             : /*                      GetInternalHandle()                             */
    1101             : /************************************************************************/
    1102             : 
    1103           0 : void *GDALProxyPoolDataset::GetInternalHandle(const char *pszRequest)
    1104             : {
    1105           0 :     CPLError(
    1106             :         CE_Warning, CPLE_AppDefined,
    1107             :         "GetInternalHandle() cannot be safely called on a proxy pool dataset\n"
    1108             :         "as the returned value may be invalidated at any time.\n");
    1109           0 :     return GDALProxyDataset::GetInternalHandle(pszRequest);
    1110             : }
    1111             : 
    1112             : /************************************************************************/
    1113             : /*                     GetGCPSpatialRef()                               */
    1114             : /************************************************************************/
    1115             : 
    1116           0 : const OGRSpatialReference *GDALProxyPoolDataset::GetGCPSpatialRef() const
    1117             : {
    1118           0 :     GDALDataset *poUnderlyingDataset = RefUnderlyingDataset();
    1119           0 :     if (poUnderlyingDataset == nullptr)
    1120           0 :         return nullptr;
    1121             : 
    1122           0 :     m_poGCPSRS->Release();
    1123           0 :     m_poGCPSRS = nullptr;
    1124             : 
    1125           0 :     const auto poUnderlyingGCPSRS = poUnderlyingDataset->GetGCPSpatialRef();
    1126           0 :     if (poUnderlyingGCPSRS)
    1127           0 :         m_poGCPSRS = poUnderlyingGCPSRS->Clone();
    1128             : 
    1129           0 :     UnrefUnderlyingDataset(poUnderlyingDataset);
    1130             : 
    1131           0 :     return m_poGCPSRS;
    1132             : }
    1133             : 
    1134             : /************************************************************************/
    1135             : /*                            GetGCPs()                                 */
    1136             : /************************************************************************/
    1137             : 
    1138           0 : const GDAL_GCP *GDALProxyPoolDataset::GetGCPs()
    1139             : {
    1140           0 :     GDALDataset *poUnderlyingDataset = RefUnderlyingDataset();
    1141           0 :     if (poUnderlyingDataset == nullptr)
    1142           0 :         return nullptr;
    1143             : 
    1144           0 :     if (nGCPCount)
    1145             :     {
    1146           0 :         GDALDeinitGCPs(nGCPCount, pasGCPList);
    1147           0 :         CPLFree(pasGCPList);
    1148           0 :         pasGCPList = nullptr;
    1149             :     }
    1150             : 
    1151           0 :     const GDAL_GCP *pasUnderlyingGCPList = poUnderlyingDataset->GetGCPs();
    1152           0 :     nGCPCount = poUnderlyingDataset->GetGCPCount();
    1153           0 :     if (nGCPCount)
    1154           0 :         pasGCPList = GDALDuplicateGCPs(nGCPCount, pasUnderlyingGCPList);
    1155             : 
    1156           0 :     UnrefUnderlyingDataset(poUnderlyingDataset);
    1157             : 
    1158           0 :     return pasGCPList;
    1159             : }
    1160             : 
    1161             : /************************************************************************/
    1162             : /*                     GDALProxyPoolDatasetCreate()                     */
    1163             : /************************************************************************/
    1164             : 
    1165        1198 : GDALProxyPoolDatasetH GDALProxyPoolDatasetCreate(
    1166             :     const char *pszSourceDatasetDescription, int nRasterXSize, int nRasterYSize,
    1167             :     GDALAccess eAccess, int bShared, const char *pszProjectionRef,
    1168             :     double *padfGeoTransform)
    1169             : {
    1170             :     return reinterpret_cast<GDALProxyPoolDatasetH>(new GDALProxyPoolDataset(
    1171             :         pszSourceDatasetDescription, nRasterXSize, nRasterYSize, eAccess,
    1172        1198 :         bShared, pszProjectionRef, padfGeoTransform));
    1173             : }
    1174             : 
    1175             : /************************************************************************/
    1176             : /*                       GDALProxyPoolDatasetDelete()                   */
    1177             : /************************************************************************/
    1178             : 
    1179           0 : void GDALProxyPoolDatasetDelete(GDALProxyPoolDatasetH hProxyPoolDataset)
    1180             : {
    1181           0 :     delete reinterpret_cast<GDALProxyPoolDataset *>(hProxyPoolDataset);
    1182           0 : }
    1183             : 
    1184             : /************************************************************************/
    1185             : /*              GDALProxyPoolDatasetAddSrcBandDescription()             */
    1186             : /************************************************************************/
    1187             : 
    1188        2289 : void GDALProxyPoolDatasetAddSrcBandDescription(
    1189             :     GDALProxyPoolDatasetH hProxyPoolDataset, GDALDataType eDataType,
    1190             :     int nBlockXSize, int nBlockYSize)
    1191             : {
    1192             :     reinterpret_cast<GDALProxyPoolDataset *>(hProxyPoolDataset)
    1193        2289 :         ->AddSrcBandDescription(eDataType, nBlockXSize, nBlockYSize);
    1194        2289 : }
    1195             : 
    1196             : /* ******************************************************************** */
    1197             : /*                    GDALProxyPoolRasterBand()                         */
    1198             : /* ******************************************************************** */
    1199             : 
    1200        4486 : GDALProxyPoolRasterBand::GDALProxyPoolRasterBand(GDALProxyPoolDataset *poDSIn,
    1201             :                                                  int nBandIn,
    1202             :                                                  GDALDataType eDataTypeIn,
    1203             :                                                  int nBlockXSizeIn,
    1204        4486 :                                                  int nBlockYSizeIn)
    1205             : {
    1206        4487 :     poDS = poDSIn;
    1207        4487 :     nBand = nBandIn;
    1208        4487 :     eDataType = eDataTypeIn;
    1209        4487 :     nRasterXSize = poDSIn->GetRasterXSize();
    1210        4486 :     nRasterYSize = poDSIn->GetRasterYSize();
    1211        4488 :     nBlockXSize = nBlockXSizeIn;
    1212        4488 :     nBlockYSize = nBlockYSizeIn;
    1213        4488 : }
    1214             : 
    1215             : /* ******************************************************************** */
    1216             : /*                    GDALProxyPoolRasterBand()                         */
    1217             : /* ******************************************************************** */
    1218             : 
    1219          60 : GDALProxyPoolRasterBand::GDALProxyPoolRasterBand(
    1220          60 :     GDALProxyPoolDataset *poDSIn, GDALRasterBand *poUnderlyingRasterBand)
    1221             : {
    1222          60 :     poDS = poDSIn;
    1223          60 :     nBand = poUnderlyingRasterBand->GetBand();
    1224          60 :     eDataType = poUnderlyingRasterBand->GetRasterDataType();
    1225          60 :     nRasterXSize = poUnderlyingRasterBand->GetXSize();
    1226          60 :     nRasterYSize = poUnderlyingRasterBand->GetYSize();
    1227          60 :     poUnderlyingRasterBand->GetBlockSize(&nBlockXSize, &nBlockYSize);
    1228          60 : }
    1229             : 
    1230             : /* ******************************************************************** */
    1231             : /*                   ~GDALProxyPoolRasterBand()                         */
    1232             : /* ******************************************************************** */
    1233        9015 : GDALProxyPoolRasterBand::~GDALProxyPoolRasterBand()
    1234             : {
    1235        4548 :     if (metadataSet)
    1236           2 :         CPLHashSetDestroy(metadataSet);
    1237        4548 :     if (metadataItemSet)
    1238          19 :         CPLHashSetDestroy(metadataItemSet);
    1239        4548 :     CPLFree(pszUnitType);
    1240        4548 :     CSLDestroy(papszCategoryNames);
    1241        4548 :     if (poColorTable)
    1242           6 :         delete poColorTable;
    1243             : 
    1244        4572 :     for (int i = 0; i < nSizeProxyOverviewRasterBand; i++)
    1245             :     {
    1246          24 :         if (papoProxyOverviewRasterBand[i])
    1247          24 :             delete papoProxyOverviewRasterBand[i];
    1248             :     }
    1249        4548 :     CPLFree(papoProxyOverviewRasterBand);
    1250        4548 :     if (poProxyMaskBand)
    1251          57 :         delete poProxyMaskBand;
    1252        9015 : }
    1253             : 
    1254             : /************************************************************************/
    1255             : /*                AddSrcMaskBandDescriptionFromUnderlying()             */
    1256             : /************************************************************************/
    1257             : 
    1258          15 : void GDALProxyPoolRasterBand::AddSrcMaskBandDescriptionFromUnderlying()
    1259             : {
    1260          15 :     if (poProxyMaskBand != nullptr)
    1261           6 :         return;
    1262           9 :     GDALRasterBand *poUnderlyingBand = RefUnderlyingRasterBand();
    1263           9 :     if (poUnderlyingBand == nullptr)
    1264           0 :         return;
    1265           9 :     auto poSrcMaskBand = poUnderlyingBand->GetMaskBand();
    1266             :     int nSrcBlockXSize, nSrcBlockYSize;
    1267           9 :     poSrcMaskBand->GetBlockSize(&nSrcBlockXSize, &nSrcBlockYSize);
    1268           9 :     poProxyMaskBand = new GDALProxyPoolMaskBand(
    1269           9 :         cpl::down_cast<GDALProxyPoolDataset *>(poDS), this,
    1270           9 :         poSrcMaskBand->GetRasterDataType(), nSrcBlockXSize, nSrcBlockYSize);
    1271           9 :     UnrefUnderlyingRasterBand(poUnderlyingBand);
    1272             : }
    1273             : 
    1274             : /************************************************************************/
    1275             : /*                 AddSrcMaskBandDescription()                          */
    1276             : /************************************************************************/
    1277             : 
    1278          12 : void GDALProxyPoolRasterBand::AddSrcMaskBandDescription(
    1279             :     GDALDataType eDataTypeIn, int nBlockXSizeIn, int nBlockYSizeIn)
    1280             : {
    1281          12 :     CPLAssert(poProxyMaskBand == nullptr);
    1282          12 :     poProxyMaskBand = new GDALProxyPoolMaskBand(
    1283          12 :         cpl::down_cast<GDALProxyPoolDataset *>(poDS), this, eDataTypeIn,
    1284          12 :         nBlockXSizeIn, nBlockYSizeIn);
    1285          12 : }
    1286             : 
    1287             : /************************************************************************/
    1288             : /*                  RefUnderlyingRasterBand()                           */
    1289             : /************************************************************************/
    1290             : 
    1291             : GDALRasterBand *
    1292       23220 : GDALProxyPoolRasterBand::RefUnderlyingRasterBand(bool bForceOpen) const
    1293             : {
    1294             :     GDALDataset *poUnderlyingDataset =
    1295       23220 :         (cpl::down_cast<GDALProxyPoolDataset *>(poDS))
    1296       23220 :             ->RefUnderlyingDataset(bForceOpen);
    1297       23220 :     if (poUnderlyingDataset == nullptr)
    1298        2143 :         return nullptr;
    1299             : 
    1300       21077 :     GDALRasterBand *poBand = poUnderlyingDataset->GetRasterBand(nBand);
    1301       21077 :     if (poBand == nullptr)
    1302             :     {
    1303           0 :         (cpl::down_cast<GDALProxyPoolDataset *>(poDS))
    1304           0 :             ->UnrefUnderlyingDataset(poUnderlyingDataset);
    1305             :     }
    1306       21077 :     else if (nBlockXSize <= 0 || nBlockYSize <= 0)
    1307             :     {
    1308             :         // Here we try to load nBlockXSize&nBlockYSize from underlying band
    1309             :         // but we must guarantee that we will not access directly to
    1310             :         // nBlockXSize/nBlockYSize before RefUnderlyingRasterBand() is called
    1311             :         int nSrcBlockXSize, nSrcBlockYSize;
    1312          10 :         poBand->GetBlockSize(&nSrcBlockXSize, &nSrcBlockYSize);
    1313          10 :         const_cast<GDALProxyPoolRasterBand *>(this)->nBlockXSize =
    1314             :             nSrcBlockXSize;
    1315          10 :         const_cast<GDALProxyPoolRasterBand *>(this)->nBlockYSize =
    1316             :             nSrcBlockYSize;
    1317             :     }
    1318             : 
    1319       21077 :     return poBand;
    1320             : }
    1321             : 
    1322             : /************************************************************************/
    1323             : /*                  UnrefUnderlyingRasterBand()                       */
    1324             : /************************************************************************/
    1325             : 
    1326       21077 : void GDALProxyPoolRasterBand::UnrefUnderlyingRasterBand(
    1327             :     GDALRasterBand *poUnderlyingRasterBand) const
    1328             : {
    1329       21077 :     if (poUnderlyingRasterBand)
    1330       21077 :         (cpl::down_cast<GDALProxyPoolDataset *>(poDS))
    1331       21076 :             ->UnrefUnderlyingDataset(poUnderlyingRasterBand->GetDataset());
    1332       21077 : }
    1333             : 
    1334             : /************************************************************************/
    1335             : /*                             FlushCache()                             */
    1336             : /************************************************************************/
    1337             : 
    1338        3441 : CPLErr GDALProxyPoolRasterBand::FlushCache(bool bAtClosing)
    1339             : {
    1340        3441 :     GDALRasterBand *poUnderlyingRasterBand = RefUnderlyingRasterBand(false);
    1341        3441 :     if (poUnderlyingRasterBand)
    1342             :     {
    1343        1298 :         CPLErr eErr = poUnderlyingRasterBand->FlushCache(bAtClosing);
    1344        1298 :         UnrefUnderlyingRasterBand(poUnderlyingRasterBand);
    1345        1298 :         return eErr;
    1346             :     }
    1347        2143 :     return CE_None;
    1348             : }
    1349             : 
    1350             : /************************************************************************/
    1351             : /*                            GetMetadata()                             */
    1352             : /************************************************************************/
    1353             : 
    1354           2 : char **GDALProxyPoolRasterBand::GetMetadata(const char *pszDomain)
    1355             : {
    1356           2 :     if (metadataSet == nullptr)
    1357           2 :         metadataSet =
    1358           2 :             CPLHashSetNew(hash_func_get_metadata, equal_func_get_metadata,
    1359             :                           free_func_get_metadata);
    1360             : 
    1361           2 :     GDALRasterBand *poUnderlyingRasterBand = RefUnderlyingRasterBand();
    1362           2 :     if (poUnderlyingRasterBand == nullptr)
    1363           0 :         return nullptr;
    1364             : 
    1365             :     char **papszUnderlyingMetadata =
    1366           2 :         poUnderlyingRasterBand->GetMetadata(pszDomain);
    1367             : 
    1368             :     GetMetadataElt *pElt =
    1369           2 :         static_cast<GetMetadataElt *>(CPLMalloc(sizeof(GetMetadataElt)));
    1370           2 :     pElt->pszDomain = (pszDomain) ? CPLStrdup(pszDomain) : nullptr;
    1371           2 :     pElt->papszMetadata = CSLDuplicate(papszUnderlyingMetadata);
    1372           2 :     CPLHashSetInsert(metadataSet, pElt);
    1373             : 
    1374           2 :     UnrefUnderlyingRasterBand(poUnderlyingRasterBand);
    1375             : 
    1376           2 :     return pElt->papszMetadata;
    1377             : }
    1378             : 
    1379             : /************************************************************************/
    1380             : /*                        GetMetadataItem()                             */
    1381             : /************************************************************************/
    1382             : 
    1383          39 : const char *GDALProxyPoolRasterBand::GetMetadataItem(const char *pszName,
    1384             :                                                      const char *pszDomain)
    1385             : {
    1386          39 :     if (metadataItemSet == nullptr)
    1387          19 :         metadataItemSet = CPLHashSetNew(hash_func_get_metadata_item,
    1388             :                                         equal_func_get_metadata_item,
    1389             :                                         free_func_get_metadata_item);
    1390             : 
    1391          39 :     GDALRasterBand *poUnderlyingRasterBand = RefUnderlyingRasterBand();
    1392          39 :     if (poUnderlyingRasterBand == nullptr)
    1393           0 :         return nullptr;
    1394             : 
    1395             :     const char *pszUnderlyingMetadataItem =
    1396          39 :         poUnderlyingRasterBand->GetMetadataItem(pszName, pszDomain);
    1397             : 
    1398             :     GetMetadataItemElt *pElt = static_cast<GetMetadataItemElt *>(
    1399          39 :         CPLMalloc(sizeof(GetMetadataItemElt)));
    1400          39 :     pElt->pszName = (pszName) ? CPLStrdup(pszName) : nullptr;
    1401          39 :     pElt->pszDomain = (pszDomain) ? CPLStrdup(pszDomain) : nullptr;
    1402          39 :     pElt->pszMetadataItem = (pszUnderlyingMetadataItem)
    1403          39 :                                 ? CPLStrdup(pszUnderlyingMetadataItem)
    1404             :                                 : nullptr;
    1405          39 :     CPLHashSetInsert(metadataItemSet, pElt);
    1406             : 
    1407          39 :     UnrefUnderlyingRasterBand(poUnderlyingRasterBand);
    1408             : 
    1409          39 :     return pElt->pszMetadataItem;
    1410             : }
    1411             : 
    1412             : /* ******************************************************************** */
    1413             : /*                       GetCategoryNames()                             */
    1414             : /* ******************************************************************** */
    1415             : 
    1416           2 : char **GDALProxyPoolRasterBand::GetCategoryNames()
    1417             : {
    1418           2 :     GDALRasterBand *poUnderlyingRasterBand = RefUnderlyingRasterBand();
    1419           2 :     if (poUnderlyingRasterBand == nullptr)
    1420           0 :         return nullptr;
    1421             : 
    1422           2 :     CSLDestroy(papszCategoryNames);
    1423           2 :     papszCategoryNames = nullptr;
    1424             : 
    1425             :     char **papszUnderlyingCategoryNames =
    1426           2 :         poUnderlyingRasterBand->GetCategoryNames();
    1427           2 :     if (papszUnderlyingCategoryNames)
    1428           0 :         papszCategoryNames = CSLDuplicate(papszUnderlyingCategoryNames);
    1429             : 
    1430           2 :     UnrefUnderlyingRasterBand(poUnderlyingRasterBand);
    1431             : 
    1432           2 :     return papszCategoryNames;
    1433             : }
    1434             : 
    1435             : /* ******************************************************************** */
    1436             : /*                           GetUnitType()                              */
    1437             : /* ******************************************************************** */
    1438             : 
    1439           2 : const char *GDALProxyPoolRasterBand::GetUnitType()
    1440             : {
    1441           2 :     GDALRasterBand *poUnderlyingRasterBand = RefUnderlyingRasterBand();
    1442           2 :     if (poUnderlyingRasterBand == nullptr)
    1443           0 :         return nullptr;
    1444             : 
    1445           2 :     CPLFree(pszUnitType);
    1446           2 :     pszUnitType = nullptr;
    1447             : 
    1448           2 :     const char *pszUnderlyingUnitType = poUnderlyingRasterBand->GetUnitType();
    1449           2 :     if (pszUnderlyingUnitType)
    1450           2 :         pszUnitType = CPLStrdup(pszUnderlyingUnitType);
    1451             : 
    1452           2 :     UnrefUnderlyingRasterBand(poUnderlyingRasterBand);
    1453             : 
    1454           2 :     return pszUnitType;
    1455             : }
    1456             : 
    1457             : /* ******************************************************************** */
    1458             : /*                          GetColorTable()                             */
    1459             : /* ******************************************************************** */
    1460             : 
    1461         432 : GDALColorTable *GDALProxyPoolRasterBand::GetColorTable()
    1462             : {
    1463         432 :     GDALRasterBand *poUnderlyingRasterBand = RefUnderlyingRasterBand();
    1464         432 :     if (poUnderlyingRasterBand == nullptr)
    1465           0 :         return nullptr;
    1466             : 
    1467         432 :     if (poColorTable)
    1468          31 :         delete poColorTable;
    1469         432 :     poColorTable = nullptr;
    1470             : 
    1471             :     GDALColorTable *poUnderlyingColorTable =
    1472         432 :         poUnderlyingRasterBand->GetColorTable();
    1473         432 :     if (poUnderlyingColorTable)
    1474          37 :         poColorTable = poUnderlyingColorTable->Clone();
    1475             : 
    1476         432 :     UnrefUnderlyingRasterBand(poUnderlyingRasterBand);
    1477             : 
    1478         432 :     return poColorTable;
    1479             : }
    1480             : 
    1481             : /* ******************************************************************** */
    1482             : /*                           GetOverview()                              */
    1483             : /* ******************************************************************** */
    1484             : 
    1485          36 : GDALRasterBand *GDALProxyPoolRasterBand::GetOverview(int nOverviewBand)
    1486             : {
    1487          36 :     if (nOverviewBand >= 0 && nOverviewBand < nSizeProxyOverviewRasterBand)
    1488             :     {
    1489          12 :         if (papoProxyOverviewRasterBand[nOverviewBand])
    1490          12 :             return papoProxyOverviewRasterBand[nOverviewBand];
    1491             :     }
    1492             : 
    1493          24 :     GDALRasterBand *poUnderlyingRasterBand = RefUnderlyingRasterBand();
    1494          24 :     if (poUnderlyingRasterBand == nullptr)
    1495           0 :         return nullptr;
    1496             : 
    1497             :     GDALRasterBand *poOverviewRasterBand =
    1498          24 :         poUnderlyingRasterBand->GetOverview(nOverviewBand);
    1499          24 :     if (poOverviewRasterBand == nullptr)
    1500             :     {
    1501           0 :         UnrefUnderlyingRasterBand(poUnderlyingRasterBand);
    1502           0 :         return nullptr;
    1503             :     }
    1504             : 
    1505          24 :     if (nOverviewBand >= nSizeProxyOverviewRasterBand)
    1506             :     {
    1507          24 :         papoProxyOverviewRasterBand =
    1508             :             static_cast<GDALProxyPoolOverviewRasterBand **>(
    1509          48 :                 CPLRealloc(papoProxyOverviewRasterBand,
    1510             :                            sizeof(GDALProxyPoolOverviewRasterBand *) *
    1511          24 :                                (nOverviewBand + 1)));
    1512          48 :         for (int i = nSizeProxyOverviewRasterBand; i < nOverviewBand + 1; i++)
    1513          24 :             papoProxyOverviewRasterBand[i] = nullptr;
    1514          24 :         nSizeProxyOverviewRasterBand = nOverviewBand + 1;
    1515             :     }
    1516             : 
    1517          24 :     papoProxyOverviewRasterBand[nOverviewBand] =
    1518             :         new GDALProxyPoolOverviewRasterBand(
    1519          24 :             cpl::down_cast<GDALProxyPoolDataset *>(poDS), poOverviewRasterBand,
    1520          24 :             this, nOverviewBand);
    1521             : 
    1522          24 :     UnrefUnderlyingRasterBand(poUnderlyingRasterBand);
    1523             : 
    1524          24 :     return papoProxyOverviewRasterBand[nOverviewBand];
    1525             : }
    1526             : 
    1527             : /* ******************************************************************** */
    1528             : /*                     GetRasterSampleOverview()                        */
    1529             : /* ******************************************************************** */
    1530             : 
    1531             : GDALRasterBand *
    1532           0 : GDALProxyPoolRasterBand::GetRasterSampleOverview(GUIntBig /* nDesiredSamples */)
    1533             : {
    1534           0 :     CPLError(CE_Failure, CPLE_AppDefined,
    1535             :              "GDALProxyPoolRasterBand::GetRasterSampleOverview : not "
    1536             :              "implemented yet");
    1537           0 :     return nullptr;
    1538             : }
    1539             : 
    1540             : /* ******************************************************************** */
    1541             : /*                           GetMaskBand()                              */
    1542             : /* ******************************************************************** */
    1543             : 
    1544          85 : GDALRasterBand *GDALProxyPoolRasterBand::GetMaskBand()
    1545             : {
    1546          85 :     if (poProxyMaskBand)
    1547          49 :         return poProxyMaskBand;
    1548             : 
    1549          36 :     GDALRasterBand *poUnderlyingRasterBand = RefUnderlyingRasterBand();
    1550          36 :     if (poUnderlyingRasterBand == nullptr)
    1551           0 :         return nullptr;
    1552             : 
    1553          36 :     GDALRasterBand *poMaskBand = poUnderlyingRasterBand->GetMaskBand();
    1554             : 
    1555          36 :     poProxyMaskBand = new GDALProxyPoolMaskBand(
    1556          36 :         cpl::down_cast<GDALProxyPoolDataset *>(poDS), poMaskBand, this);
    1557             : 
    1558          36 :     UnrefUnderlyingRasterBand(poUnderlyingRasterBand);
    1559             : 
    1560          36 :     return poProxyMaskBand;
    1561             : }
    1562             : 
    1563             : /* ******************************************************************** */
    1564             : /*             GDALProxyPoolOverviewRasterBand()                        */
    1565             : /* ******************************************************************** */
    1566             : 
    1567          24 : GDALProxyPoolOverviewRasterBand::GDALProxyPoolOverviewRasterBand(
    1568             :     GDALProxyPoolDataset *poDSIn, GDALRasterBand *poUnderlyingOverviewBand,
    1569          24 :     GDALProxyPoolRasterBand *poMainBandIn, int nOverviewBandIn)
    1570             :     : GDALProxyPoolRasterBand(poDSIn, poUnderlyingOverviewBand),
    1571          24 :       poMainBand(poMainBandIn), nOverviewBand(nOverviewBandIn)
    1572             : {
    1573          24 : }
    1574             : 
    1575             : /* ******************************************************************** */
    1576             : /*                  ~GDALProxyPoolOverviewRasterBand()                  */
    1577             : /* ******************************************************************** */
    1578             : 
    1579          48 : GDALProxyPoolOverviewRasterBand::~GDALProxyPoolOverviewRasterBand()
    1580             : {
    1581          24 :     CPLAssert(nRefCountUnderlyingMainRasterBand == 0);
    1582          48 : }
    1583             : 
    1584             : /* ******************************************************************** */
    1585             : /*                    RefUnderlyingRasterBand()                         */
    1586             : /* ******************************************************************** */
    1587             : 
    1588             : GDALRasterBand *
    1589           0 : GDALProxyPoolOverviewRasterBand::RefUnderlyingRasterBand(bool bForceOpen) const
    1590             : {
    1591           0 :     poUnderlyingMainRasterBand =
    1592           0 :         poMainBand->RefUnderlyingRasterBand(bForceOpen);
    1593           0 :     if (poUnderlyingMainRasterBand == nullptr)
    1594           0 :         return nullptr;
    1595             : 
    1596           0 :     nRefCountUnderlyingMainRasterBand++;
    1597           0 :     return poUnderlyingMainRasterBand->GetOverview(nOverviewBand);
    1598             : }
    1599             : 
    1600             : /* ******************************************************************** */
    1601             : /*                  UnrefUnderlyingRasterBand()                         */
    1602             : /* ******************************************************************** */
    1603             : 
    1604           0 : void GDALProxyPoolOverviewRasterBand::UnrefUnderlyingRasterBand(
    1605             :     GDALRasterBand * /* poUnderlyingRasterBand */) const
    1606             : {
    1607           0 :     poMainBand->UnrefUnderlyingRasterBand(poUnderlyingMainRasterBand);
    1608           0 :     nRefCountUnderlyingMainRasterBand--;
    1609           0 : }
    1610             : 
    1611             : /* ******************************************************************** */
    1612             : /*                     GDALProxyPoolMaskBand()                          */
    1613             : /* ******************************************************************** */
    1614             : 
    1615          36 : GDALProxyPoolMaskBand::GDALProxyPoolMaskBand(
    1616             :     GDALProxyPoolDataset *poDSIn, GDALRasterBand *poUnderlyingMaskBand,
    1617          36 :     GDALProxyPoolRasterBand *poMainBandIn)
    1618          36 :     : GDALProxyPoolRasterBand(poDSIn, poUnderlyingMaskBand)
    1619             : {
    1620          36 :     poMainBand = poMainBandIn;
    1621             : 
    1622          36 :     poUnderlyingMainRasterBand = nullptr;
    1623          36 :     nRefCountUnderlyingMainRasterBand = 0;
    1624          36 : }
    1625             : 
    1626             : /* ******************************************************************** */
    1627             : /*                     GDALProxyPoolMaskBand()                          */
    1628             : /* ******************************************************************** */
    1629             : 
    1630          21 : GDALProxyPoolMaskBand::GDALProxyPoolMaskBand(
    1631             :     GDALProxyPoolDataset *poDSIn, GDALProxyPoolRasterBand *poMainBandIn,
    1632          21 :     GDALDataType eDataTypeIn, int nBlockXSizeIn, int nBlockYSizeIn)
    1633             :     : GDALProxyPoolRasterBand(poDSIn, 1, eDataTypeIn, nBlockXSizeIn,
    1634             :                               nBlockYSizeIn),
    1635          21 :       poMainBand(poMainBandIn)
    1636             : {
    1637          21 : }
    1638             : 
    1639             : /* ******************************************************************** */
    1640             : /*                          ~GDALProxyPoolMaskBand()                    */
    1641             : /* ******************************************************************** */
    1642             : 
    1643         114 : GDALProxyPoolMaskBand::~GDALProxyPoolMaskBand()
    1644             : {
    1645          57 :     CPLAssert(nRefCountUnderlyingMainRasterBand == 0);
    1646         114 : }
    1647             : 
    1648             : /* ******************************************************************** */
    1649             : /*                    RefUnderlyingRasterBand()                         */
    1650             : /* ******************************************************************** */
    1651             : 
    1652             : GDALRasterBand *
    1653         243 : GDALProxyPoolMaskBand::RefUnderlyingRasterBand(bool bForceOpen) const
    1654             : {
    1655         243 :     poUnderlyingMainRasterBand =
    1656         243 :         poMainBand->RefUnderlyingRasterBand(bForceOpen);
    1657         243 :     if (poUnderlyingMainRasterBand == nullptr)
    1658           0 :         return nullptr;
    1659             : 
    1660         243 :     nRefCountUnderlyingMainRasterBand++;
    1661         243 :     return poUnderlyingMainRasterBand->GetMaskBand();
    1662             : }
    1663             : 
    1664             : /* ******************************************************************** */
    1665             : /*                  UnrefUnderlyingRasterBand()                         */
    1666             : /* ******************************************************************** */
    1667             : 
    1668         243 : void GDALProxyPoolMaskBand::UnrefUnderlyingRasterBand(
    1669             :     GDALRasterBand * /* poUnderlyingRasterBand */) const
    1670             : {
    1671         243 :     poMainBand->UnrefUnderlyingRasterBand(poUnderlyingMainRasterBand);
    1672         243 :     nRefCountUnderlyingMainRasterBand--;
    1673         243 : }
    1674             : 
    1675             : //! @endcond

Generated by: LCOV version 1.14