LCOV - code coverage report
Current view: top level - gcore - gdalarraybandblockcache.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 153 163 93.9 %
Date: 2025-01-18 12:42:00 Functions: 12 12 100.0 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  GDAL Core
       4             :  * Purpose:  Store cached blocks in a array or a two-level array
       5             :  * Author:   Even Rouault, <even dot rouault at spatialys dot org>
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 1998, Frank Warmerdam
       9             :  * Copyright (c) 2007-2015, Even Rouault <even dot rouault at spatialys dot org>
      10             :  *
      11             :  * SPDX-License-Identifier: MIT
      12             :  ****************************************************************************/
      13             : 
      14             : #include "cpl_port.h"
      15             : #include "gdal_priv.h"
      16             : 
      17             : #include <cassert>
      18             : #include <climits>
      19             : #include <cstddef>
      20             : #include <new>
      21             : 
      22             : #include "cpl_conv.h"
      23             : #include "cpl_error.h"
      24             : #include "cpl_multiproc.h"
      25             : #include "cpl_vsi.h"
      26             : 
      27             : //! @cond Doxygen_Suppress
      28             : 
      29             : constexpr int SUBBLOCK_SIZE = 64;
      30             : #define TO_SUBBLOCK(x) ((x) >> 6)
      31             : #define WITHIN_SUBBLOCK(x) ((x)&0x3f)
      32             : 
      33             : /* ******************************************************************** */
      34             : /*                        GDALArrayBandBlockCache                       */
      35             : /* ******************************************************************** */
      36             : 
      37             : class GDALArrayBandBlockCache final : public GDALAbstractBandBlockCache
      38             : {
      39             :     bool bSubBlockingActive = false;
      40             :     int nSubBlocksPerRow = 0;
      41             :     int nSubBlocksPerColumn = 0;
      42             : 
      43             :     union u
      44             :     {
      45             :         GDALRasterBlock **papoBlocks;
      46             :         GDALRasterBlock ***papapoBlocks;
      47             : 
      48       33693 :         u() : papoBlocks(nullptr)
      49             :         {
      50       33693 :         }
      51             :     } u{};
      52             : 
      53             :     CPL_DISALLOW_COPY_ASSIGN(GDALArrayBandBlockCache)
      54             : 
      55             :   public:
      56             :     explicit GDALArrayBandBlockCache(GDALRasterBand *poBand);
      57             :     ~GDALArrayBandBlockCache() override;
      58             : 
      59             :     bool Init() override;
      60             :     bool IsInitOK() override;
      61             :     CPLErr FlushCache() override;
      62             :     CPLErr AdoptBlock(GDALRasterBlock *) override;
      63             :     GDALRasterBlock *TryGetLockedBlockRef(int nXBlockOff,
      64             :                                           int nYBlockYOff) override;
      65             :     CPLErr UnreferenceBlock(GDALRasterBlock *poBlock) override;
      66             :     CPLErr FlushBlock(int nXBlockOff, int nYBlockOff,
      67             :                       int bWriteDirtyBlock) override;
      68             : };
      69             : 
      70             : /************************************************************************/
      71             : /*                     GDALArrayBandBlockCacheCreate()                 */
      72             : /************************************************************************/
      73             : 
      74             : GDALAbstractBandBlockCache *
      75       33694 : GDALArrayBandBlockCacheCreate(GDALRasterBand *poBand)
      76             : {
      77       33694 :     return new (std::nothrow) GDALArrayBandBlockCache(poBand);
      78             : }
      79             : 
      80             : /************************************************************************/
      81             : /*                       GDALArrayBandBlockCache()                      */
      82             : /************************************************************************/
      83             : 
      84       33693 : GDALArrayBandBlockCache::GDALArrayBandBlockCache(GDALRasterBand *poBandIn)
      85       33693 :     : GDALAbstractBandBlockCache(poBandIn)
      86             : {
      87       33693 : }
      88             : 
      89             : /************************************************************************/
      90             : /*                      ~GDALArrayBandBlockCache()                     */
      91             : /************************************************************************/
      92             : 
      93       67385 : GDALArrayBandBlockCache::~GDALArrayBandBlockCache()
      94             : {
      95       33690 :     GDALArrayBandBlockCache::FlushCache();
      96             : 
      97       33693 :     if (!bSubBlockingActive)
      98       33568 :         CPLFree(u.papoBlocks);
      99             :     else
     100         125 :         CPLFree(u.papapoBlocks);
     101       67386 : }
     102             : 
     103             : /************************************************************************/
     104             : /*                                  Init()                              */
     105             : /************************************************************************/
     106             : 
     107       33690 : bool GDALArrayBandBlockCache::Init()
     108             : {
     109       33690 :     if (poBand->nBlocksPerRow < SUBBLOCK_SIZE / 2)
     110             :     {
     111       33564 :         bSubBlockingActive = false;
     112             : 
     113       33564 :         if (poBand->nBlocksPerRow < INT_MAX / poBand->nBlocksPerColumn)
     114             :         {
     115       33557 :             u.papoBlocks = static_cast<GDALRasterBlock **>(VSICalloc(
     116       33561 :                 sizeof(void *), cpl::fits_on<int>(poBand->nBlocksPerRow *
     117       33561 :                                                   poBand->nBlocksPerColumn)));
     118       33559 :             if (u.papoBlocks == nullptr)
     119             :             {
     120           0 :                 poBand->ReportError(CE_Failure, CPLE_OutOfMemory,
     121             :                                     "Out of memory in InitBlockInfo().");
     122           0 :                 return false;
     123             :             }
     124             :         }
     125             :         else
     126             :         {
     127           3 :             poBand->ReportError(
     128             :                 CE_Failure, CPLE_NotSupported, "Too many blocks : %d x %d",
     129           3 :                 poBand->nBlocksPerRow, poBand->nBlocksPerColumn);
     130           0 :             return false;
     131             :         }
     132             :     }
     133             :     else
     134             :     {
     135         126 :         bSubBlockingActive = true;
     136             : 
     137         126 :         nSubBlocksPerRow = DIV_ROUND_UP(poBand->nBlocksPerRow, SUBBLOCK_SIZE);
     138         126 :         nSubBlocksPerColumn =
     139         126 :             DIV_ROUND_UP(poBand->nBlocksPerColumn, SUBBLOCK_SIZE);
     140             : 
     141         126 :         if (nSubBlocksPerRow < INT_MAX / nSubBlocksPerColumn)
     142             :         {
     143         125 :             u.papapoBlocks = static_cast<GDALRasterBlock ***>(VSICalloc(
     144             :                 sizeof(void *),
     145         125 :                 cpl::fits_on<int>(nSubBlocksPerRow * nSubBlocksPerColumn)));
     146         126 :             if (u.papapoBlocks == nullptr)
     147             :             {
     148           0 :                 poBand->ReportError(CE_Failure, CPLE_OutOfMemory,
     149             :                                     "Out of memory in InitBlockInfo().");
     150           0 :                 return false;
     151             :             }
     152             :         }
     153             :         else
     154             :         {
     155           1 :             poBand->ReportError(CE_Failure, CPLE_NotSupported,
     156             :                                 "Too many subblocks : %d x %d",
     157             :                                 nSubBlocksPerRow, nSubBlocksPerColumn);
     158           0 :             return false;
     159             :         }
     160             :     }
     161             : 
     162       33685 :     return true;
     163             : }
     164             : 
     165             : /************************************************************************/
     166             : /*                             IsInitOK()                               */
     167             : /************************************************************************/
     168             : 
     169     8425050 : bool GDALArrayBandBlockCache::IsInitOK()
     170             : {
     171     8425050 :     return (!bSubBlockingActive) ? u.papoBlocks != nullptr
     172     8425050 :                                  : u.papapoBlocks != nullptr;
     173             : }
     174             : 
     175             : /************************************************************************/
     176             : /*                            AdoptBlock()                              */
     177             : /************************************************************************/
     178             : 
     179     1132400 : CPLErr GDALArrayBandBlockCache::AdoptBlock(GDALRasterBlock *poBlock)
     180             : 
     181             : {
     182     1132400 :     const int nXBlockOff = poBlock->GetXOff();
     183     1132370 :     const int nYBlockOff = poBlock->GetYOff();
     184             : 
     185     1132380 :     FreeDanglingBlocks();
     186             : 
     187             :     /* -------------------------------------------------------------------- */
     188             :     /*      Simple case without subblocking.                                */
     189             :     /* -------------------------------------------------------------------- */
     190             : 
     191     1132370 :     if (!bSubBlockingActive)
     192             :     {
     193     1074630 :         const int nBlockIndex = nXBlockOff + nYBlockOff * poBand->nBlocksPerRow;
     194             : 
     195     1074630 :         CPLAssert(u.papoBlocks[nBlockIndex] == nullptr);
     196     1074630 :         u.papoBlocks[nBlockIndex] = poBlock;
     197             :     }
     198             :     else
     199             :     {
     200             :         /* --------------------------------------------------------------------
     201             :          */
     202             :         /*      Identify the subblock in which our target occurs, and create */
     203             :         /*      it if necessary. */
     204             :         /* --------------------------------------------------------------------
     205             :          */
     206       57743 :         const int nSubBlock = TO_SUBBLOCK(nXBlockOff) +
     207       57743 :                               TO_SUBBLOCK(nYBlockOff) * nSubBlocksPerRow;
     208             : 
     209       57743 :         if (u.papapoBlocks[nSubBlock] == nullptr)
     210             :         {
     211         169 :             const int nSubGridSize =
     212             :                 sizeof(GDALRasterBlock *) * SUBBLOCK_SIZE * SUBBLOCK_SIZE;
     213             : 
     214         338 :             u.papapoBlocks[nSubBlock] =
     215         169 :                 static_cast<GDALRasterBlock **>(VSICalloc(1, nSubGridSize));
     216         169 :             if (u.papapoBlocks[nSubBlock] == nullptr)
     217             :             {
     218           0 :                 poBand->ReportError(CE_Failure, CPLE_OutOfMemory,
     219             :                                     "Out of memory in AdoptBlock().");
     220           0 :                 return CE_Failure;
     221             :             }
     222             :         }
     223             : 
     224             :         /* --------------------------------------------------------------------
     225             :          */
     226             :         /*      Check within subblock. */
     227             :         /* --------------------------------------------------------------------
     228             :          */
     229       57743 :         GDALRasterBlock **papoSubBlockGrid = u.papapoBlocks[nSubBlock];
     230             : 
     231       57743 :         const int nBlockInSubBlock =
     232       57743 :             WITHIN_SUBBLOCK(nXBlockOff) +
     233       57743 :             WITHIN_SUBBLOCK(nYBlockOff) * SUBBLOCK_SIZE;
     234             : 
     235       57743 :         CPLAssert(papoSubBlockGrid[nBlockInSubBlock] == nullptr);
     236       57743 :         papoSubBlockGrid[nBlockInSubBlock] = poBlock;
     237             :     }
     238             : 
     239     1132370 :     return CE_None;
     240             : }
     241             : 
     242             : /************************************************************************/
     243             : /*                            FlushCache()                              */
     244             : /************************************************************************/
     245             : 
     246      182277 : CPLErr GDALArrayBandBlockCache::FlushCache()
     247             : {
     248      182277 :     FreeDanglingBlocks();
     249             : 
     250      182275 :     CPLErr eGlobalErr = poBand->eFlushBlockErr;
     251             : 
     252      182275 :     StartDirtyBlockFlushingLog();
     253             : 
     254             :     /* -------------------------------------------------------------------- */
     255             :     /*      Flush all blocks in memory ... this case is without subblocking.*/
     256             :     /* -------------------------------------------------------------------- */
     257      182275 :     if (!bSubBlockingActive && u.papoBlocks != nullptr)
     258             :     {
     259      181853 :         const int nBlocksPerColumn = poBand->nBlocksPerColumn;
     260      181853 :         const int nBlocksPerRow = poBand->nBlocksPerRow;
     261    87944900 :         for (int iY = 0; iY < nBlocksPerColumn; iY++)
     262             :         {
     263   175848000 :             for (int iX = 0; iX < nBlocksPerRow; iX++)
     264             :             {
     265    88084800 :                 if (u.papoBlocks[iX + iY * nBlocksPerRow] != nullptr)
     266             :                 {
     267     1045200 :                     CPLErr eErr = FlushBlock(iX, iY, eGlobalErr == CE_None);
     268             : 
     269     1045200 :                     if (eErr != CE_None)
     270           5 :                         eGlobalErr = eErr;
     271             :                 }
     272             :             }
     273      181855 :         }
     274             :     }
     275             : 
     276             :     /* -------------------------------------------------------------------- */
     277             :     /*      With subblocking.  We can short circuit missing subblocks.      */
     278             :     /* -------------------------------------------------------------------- */
     279         422 :     else if (u.papapoBlocks != nullptr)
     280             :     {
     281         906 :         for (int iSBY = 0; iSBY < nSubBlocksPerColumn; iSBY++)
     282             :         {
     283        1522 :             for (int iSBX = 0; iSBX < nSubBlocksPerRow; iSBX++)
     284             :             {
     285        1039 :                 const int nSubBlock = iSBX + iSBY * nSubBlocksPerRow;
     286             : 
     287        1039 :                 GDALRasterBlock **papoSubBlockGrid = u.papapoBlocks[nSubBlock];
     288             : 
     289        1039 :                 if (papoSubBlockGrid == nullptr)
     290         870 :                     continue;
     291             : 
     292       10985 :                 for (int iY = 0; iY < SUBBLOCK_SIZE; iY++)
     293             :                 {
     294      703040 :                     for (int iX = 0; iX < SUBBLOCK_SIZE; iX++)
     295             :                     {
     296      692224 :                         if (papoSubBlockGrid[iX + iY * SUBBLOCK_SIZE] !=
     297             :                             nullptr)
     298             :                         {
     299      112116 :                             CPLErr eErr = FlushBlock(iX + iSBX * SUBBLOCK_SIZE,
     300       56058 :                                                      iY + iSBY * SUBBLOCK_SIZE,
     301             :                                                      eGlobalErr == CE_None);
     302       56058 :                             if (eErr != CE_None)
     303           2 :                                 eGlobalErr = eErr;
     304             :                         }
     305             :                     }
     306             :                 }
     307             : 
     308             :                 // We might as well get rid of this grid chunk since we know
     309             :                 // it is now empty.
     310         169 :                 u.papapoBlocks[nSubBlock] = nullptr;
     311         169 :                 CPLFree(papoSubBlockGrid);
     312             :             }
     313             :         }
     314             :     }
     315             : 
     316      182277 :     EndDirtyBlockFlushingLog();
     317             : 
     318      182279 :     WaitCompletionPendingTasks();
     319             : 
     320      182277 :     return (eGlobalErr);
     321             : }
     322             : 
     323             : /************************************************************************/
     324             : /*                        UnreferenceBlock()                            */
     325             : /************************************************************************/
     326             : 
     327       29595 : CPLErr GDALArrayBandBlockCache::UnreferenceBlock(GDALRasterBlock *poBlock)
     328             : {
     329       29595 :     const int nXBlockOff = poBlock->GetXOff();
     330       29595 :     const int nYBlockOff = poBlock->GetYOff();
     331             : 
     332       29595 :     UnreferenceBlockBase();
     333             : 
     334             :     /* -------------------------------------------------------------------- */
     335             :     /*      Simple case for single level caches.                            */
     336             :     /* -------------------------------------------------------------------- */
     337       29595 :     if (!bSubBlockingActive)
     338             :     {
     339       27912 :         const int nBlockIndex = nXBlockOff + nYBlockOff * poBand->nBlocksPerRow;
     340             : 
     341       27912 :         u.papoBlocks[nBlockIndex] = nullptr;
     342             :     }
     343             : 
     344             :     /* -------------------------------------------------------------------- */
     345             :     /*      Identify our subblock.                                          */
     346             :     /* -------------------------------------------------------------------- */
     347             :     else
     348             :     {
     349        1683 :         const int nSubBlock = TO_SUBBLOCK(nXBlockOff) +
     350        1683 :                               TO_SUBBLOCK(nYBlockOff) * nSubBlocksPerRow;
     351             : 
     352             :         /* --------------------------------------------------------------------
     353             :          */
     354             :         /*      Check within subblock. */
     355             :         /* --------------------------------------------------------------------
     356             :          */
     357        1683 :         GDALRasterBlock **papoSubBlockGrid = u.papapoBlocks[nSubBlock];
     358        1683 :         if (papoSubBlockGrid == nullptr)
     359           0 :             return CE_None;
     360             : 
     361        1683 :         const int nBlockInSubBlock =
     362        1683 :             WITHIN_SUBBLOCK(nXBlockOff) +
     363        1683 :             WITHIN_SUBBLOCK(nYBlockOff) * SUBBLOCK_SIZE;
     364             : 
     365        1683 :         papoSubBlockGrid[nBlockInSubBlock] = nullptr;
     366             :     }
     367             : 
     368       29595 :     return CE_None;
     369             : }
     370             : 
     371             : /************************************************************************/
     372             : /*                            FlushBlock()                              */
     373             : /************************************************************************/
     374             : 
     375     1103560 : CPLErr GDALArrayBandBlockCache::FlushBlock(int nXBlockOff, int nYBlockOff,
     376             :                                            int bWriteDirtyBlock)
     377             : 
     378             : {
     379     1103560 :     GDALRasterBlock *poBlock = nullptr;
     380             : 
     381             :     /* -------------------------------------------------------------------- */
     382             :     /*      Simple case for single level caches.                            */
     383             :     /* -------------------------------------------------------------------- */
     384     1103560 :     if (!bSubBlockingActive)
     385             :     {
     386     1047500 :         const int nBlockIndex = nXBlockOff + nYBlockOff * poBand->nBlocksPerRow;
     387             : 
     388     1047500 :         assert(u.papoBlocks);
     389     1047500 :         poBlock = u.papoBlocks[nBlockIndex];
     390     1047500 :         u.papoBlocks[nBlockIndex] = nullptr;
     391             :     }
     392             : 
     393             :     /* -------------------------------------------------------------------- */
     394             :     /*      Identify our subblock.                                          */
     395             :     /* -------------------------------------------------------------------- */
     396             :     else
     397             :     {
     398       56060 :         const int nSubBlock = TO_SUBBLOCK(nXBlockOff) +
     399       56060 :                               TO_SUBBLOCK(nYBlockOff) * nSubBlocksPerRow;
     400             : 
     401             :         /* --------------------------------------------------------------------
     402             :          */
     403             :         /*      Check within subblock. */
     404             :         /* --------------------------------------------------------------------
     405             :          */
     406       56060 :         GDALRasterBlock **papoSubBlockGrid = u.papapoBlocks[nSubBlock];
     407       56060 :         if (papoSubBlockGrid == nullptr)
     408           0 :             return CE_None;
     409             : 
     410       56060 :         const int nBlockInSubBlock =
     411       56060 :             WITHIN_SUBBLOCK(nXBlockOff) +
     412       56060 :             WITHIN_SUBBLOCK(nYBlockOff) * SUBBLOCK_SIZE;
     413             : 
     414       56060 :         poBlock = papoSubBlockGrid[nBlockInSubBlock];
     415       56060 :         papoSubBlockGrid[nBlockInSubBlock] = nullptr;
     416             :     }
     417             : 
     418     1103560 :     if (poBlock == nullptr)
     419         701 :         return CE_None;
     420             : 
     421     1102860 :     if (!poBlock->DropLockForRemovalFromStorage())
     422           2 :         return CE_None;
     423             : 
     424             :     /* -------------------------------------------------------------------- */
     425             :     /*      Is the target block dirty?  If so we need to write it.          */
     426             :     /* -------------------------------------------------------------------- */
     427     1102850 :     poBlock->Detach();
     428             : 
     429     1102860 :     CPLErr eErr = CE_None;
     430             : 
     431     1102860 :     if (!m_nWriteDirtyBlocksDisabled && bWriteDirtyBlock && poBlock->GetDirty())
     432             :     {
     433      175139 :         UpdateDirtyBlockFlushingLog();
     434             : 
     435      175139 :         eErr = poBlock->Write();
     436             :     }
     437             : 
     438             :     /* -------------------------------------------------------------------- */
     439             :     /*      Deallocate the block;                                           */
     440             :     /* -------------------------------------------------------------------- */
     441     1102850 :     delete poBlock;
     442             : 
     443     1102850 :     return eErr;
     444             : }
     445             : 
     446             : /************************************************************************/
     447             : /*                        TryGetLockedBlockRef()                        */
     448             : /************************************************************************/
     449             : 
     450     6940030 : GDALRasterBlock *GDALArrayBandBlockCache::TryGetLockedBlockRef(int nXBlockOff,
     451             :                                                                int nYBlockOff)
     452             : 
     453             : {
     454             :     /* -------------------------------------------------------------------- */
     455             :     /*      Simple case for single level caches.                            */
     456             :     /* -------------------------------------------------------------------- */
     457     6940030 :     if (!bSubBlockingActive)
     458             :     {
     459     6157320 :         const int nBlockIndex = nXBlockOff + nYBlockOff * poBand->nBlocksPerRow;
     460             : 
     461     6157320 :         GDALRasterBlock *poBlock = u.papoBlocks[nBlockIndex];
     462     6157320 :         if (poBlock == nullptr || !poBlock->TakeLock())
     463     1125510 :             return nullptr;
     464     5032040 :         return poBlock;
     465             :     }
     466             :     else
     467             :     {
     468             :         /* --------------------------------------------------------------------
     469             :          */
     470             :         /*      Identify our subblock. */
     471             :         /* --------------------------------------------------------------------
     472             :          */
     473      782711 :         const int nSubBlock = TO_SUBBLOCK(nXBlockOff) +
     474      782711 :                               TO_SUBBLOCK(nYBlockOff) * nSubBlocksPerRow;
     475             : 
     476             :         /* --------------------------------------------------------------------
     477             :          */
     478             :         /*      Check within subblock. */
     479             :         /* --------------------------------------------------------------------
     480             :          */
     481      782711 :         GDALRasterBlock **papoSubBlockGrid = u.papapoBlocks[nSubBlock];
     482      782711 :         if (papoSubBlockGrid == nullptr)
     483          52 :             return nullptr;
     484             : 
     485      782659 :         const int nBlockInSubBlock =
     486      782659 :             WITHIN_SUBBLOCK(nXBlockOff) +
     487      782659 :             WITHIN_SUBBLOCK(nYBlockOff) * SUBBLOCK_SIZE;
     488             : 
     489      782659 :         GDALRasterBlock *poBlock = papoSubBlockGrid[nBlockInSubBlock];
     490      782659 :         if (poBlock == nullptr || !poBlock->TakeLock())
     491       57581 :             return nullptr;
     492      725078 :         return poBlock;
     493             :     }
     494             : }
     495             : 
     496             : //! @endcond

Generated by: LCOV version 1.14