LCOV - code coverage report
Current view: top level - frmts/zarr - zarr_array.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 1278 1467 87.1 %
Date: 2025-01-18 12:42:00 Functions: 42 45 93.3 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  GDAL
       4             :  * Purpose:  Zarr driver
       5             :  * Author:   Even Rouault <even dot rouault at spatialys.com>
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2021, Even Rouault <even dot rouault at spatialys.com>
       9             :  *
      10             :  * SPDX-License-Identifier: MIT
      11             :  ****************************************************************************/
      12             : 
      13             : #include "zarr.h"
      14             : #include "ucs4_utf8.hpp"
      15             : 
      16             : #include "cpl_float.h"
      17             : 
      18             : #include "netcdf_cf_constants.h"  // for CF_UNITS, etc
      19             : 
      20             : #include <algorithm>
      21             : #include <cassert>
      22             : #include <cmath>
      23             : #include <cstdlib>
      24             : #include <limits>
      25             : #include <map>
      26             : #include <set>
      27             : 
      28             : #if defined(__clang__) || defined(_MSC_VER)
      29             : #define COMPILER_WARNS_ABOUT_ABSTRACT_VBASE_INIT
      30             : #endif
      31             : 
      32             : namespace
      33             : {
      34             : 
      35           4 : inline std::vector<GByte> UTF8ToUCS4(const char *pszStr, bool needByteSwap)
      36             : {
      37           4 :     const size_t nLen = strlen(pszStr);
      38             :     // Worst case if that we need 4 more bytes than the UTF-8 one
      39             :     // (when the content is pure ASCII)
      40           4 :     if (nLen > std::numeric_limits<size_t>::max() / sizeof(uint32_t))
      41           0 :         throw std::bad_alloc();
      42           4 :     std::vector<GByte> ret(nLen * sizeof(uint32_t));
      43           4 :     size_t outPos = 0;
      44          27 :     for (size_t i = 0; i < nLen; outPos += sizeof(uint32_t))
      45             :     {
      46          23 :         uint32_t ucs4 = 0;
      47          23 :         int consumed = FcUtf8ToUcs4(
      48          23 :             reinterpret_cast<const uint8_t *>(pszStr + i), &ucs4, nLen - i);
      49          23 :         if (consumed <= 0)
      50             :         {
      51           0 :             ret.resize(outPos);
      52             :         }
      53          23 :         if (needByteSwap)
      54             :         {
      55           1 :             CPL_SWAP32PTR(&ucs4);
      56             :         }
      57          23 :         memcpy(&ret[outPos], &ucs4, sizeof(uint32_t));
      58          23 :         i += consumed;
      59             :     }
      60           4 :     ret.resize(outPos);
      61           4 :     return ret;
      62             : }
      63             : 
      64           6 : inline char *UCS4ToUTF8(const uint8_t *ucs4Ptr, size_t nSize, bool needByteSwap)
      65             : {
      66             :     // A UCS4 char can require up to 6 bytes in UTF8.
      67           6 :     if (nSize > (std::numeric_limits<size_t>::max() - 1) / 6 * 4)
      68           0 :         return nullptr;
      69           6 :     const size_t nOutSize = nSize / 4 * 6 + 1;
      70           6 :     char *ret = static_cast<char *>(VSI_MALLOC_VERBOSE(nOutSize));
      71           6 :     if (ret == nullptr)
      72           0 :         return nullptr;
      73           6 :     size_t outPos = 0;
      74          30 :     for (size_t i = 0; i + sizeof(uint32_t) - 1 < nSize; i += sizeof(uint32_t))
      75             :     {
      76             :         uint32_t ucs4;
      77          24 :         memcpy(&ucs4, ucs4Ptr + i, sizeof(uint32_t));
      78          24 :         if (needByteSwap)
      79             :         {
      80           2 :             CPL_SWAP32PTR(&ucs4);
      81             :         }
      82             :         int written =
      83          24 :             FcUcs4ToUtf8(ucs4, reinterpret_cast<uint8_t *>(ret + outPos));
      84          24 :         outPos += written;
      85             :     }
      86           6 :     ret[outPos] = 0;
      87           6 :     return ret;
      88             : }
      89             : 
      90             : }  // namespace
      91             : 
      92             : /************************************************************************/
      93             : /*                      ZarrArray::ParseChunkSize()                     */
      94             : /************************************************************************/
      95             : 
      96         780 : /* static */ bool ZarrArray::ParseChunkSize(const CPLJSONArray &oChunks,
      97             :                                             const GDALExtendedDataType &oType,
      98             :                                             std::vector<GUInt64> &anBlockSize)
      99             : {
     100         780 :     size_t nBlockSize = oType.GetSize();
     101        1961 :     for (const auto &item : oChunks)
     102             :     {
     103        1184 :         const auto nSize = static_cast<GUInt64>(item.ToLong());
     104        1184 :         if (nSize == 0)
     105             :         {
     106           1 :             CPLError(CE_Failure, CPLE_AppDefined, "Invalid content for chunks");
     107           3 :             return false;
     108             :         }
     109        1183 :         if (nBlockSize > std::numeric_limits<size_t>::max() / nSize)
     110             :         {
     111           2 :             CPLError(CE_Failure, CPLE_AppDefined, "Too large chunks");
     112           2 :             return false;
     113             :         }
     114        1181 :         nBlockSize *= static_cast<size_t>(nSize);
     115        1181 :         anBlockSize.emplace_back(nSize);
     116             :     }
     117             : 
     118         777 :     return true;
     119             : }
     120             : 
     121             : /************************************************************************/
     122             : /*                      ZarrArray::ComputeTileCount()                   */
     123             : /************************************************************************/
     124             : 
     125        1168 : /* static */ uint64_t ZarrArray::ComputeTileCount(
     126             :     const std::string &osName,
     127             :     const std::vector<std::shared_ptr<GDALDimension>> &aoDims,
     128             :     const std::vector<GUInt64> &anBlockSize)
     129             : {
     130        1168 :     uint64_t nTotalTileCount = 1;
     131        2878 :     for (size_t i = 0; i < aoDims.size(); ++i)
     132             :     {
     133             :         uint64_t nTileThisDim =
     134        1712 :             (aoDims[i]->GetSize() / anBlockSize[i]) +
     135        1712 :             (((aoDims[i]->GetSize() % anBlockSize[i]) != 0) ? 1 : 0);
     136        3424 :         if (nTileThisDim != 0 &&
     137             :             nTotalTileCount >
     138        1712 :                 std::numeric_limits<uint64_t>::max() / nTileThisDim)
     139             :         {
     140           2 :             CPLError(
     141             :                 CE_Failure, CPLE_NotSupported,
     142             :                 "Array %s has more than 2^64 tiles. This is not supported.",
     143             :                 osName.c_str());
     144           2 :             return 0;
     145             :         }
     146        1710 :         nTotalTileCount *= nTileThisDim;
     147             :     }
     148        1166 :     return nTotalTileCount;
     149             : }
     150             : 
     151             : /************************************************************************/
     152             : /*                         ZarrArray::ZarrArray()                       */
     153             : /************************************************************************/
     154             : 
     155        1168 : ZarrArray::ZarrArray(
     156             :     const std::shared_ptr<ZarrSharedResource> &poSharedResource,
     157             :     const std::string &osParentName, const std::string &osName,
     158             :     const std::vector<std::shared_ptr<GDALDimension>> &aoDims,
     159             :     const GDALExtendedDataType &oType, const std::vector<DtypeElt> &aoDtypeElts,
     160           0 :     const std::vector<GUInt64> &anBlockSize)
     161             :     :
     162             : #if !defined(COMPILER_WARNS_ABOUT_ABSTRACT_VBASE_INIT)
     163             :       GDALAbstractMDArray(osParentName, osName),
     164             : #endif
     165             :       GDALPamMDArray(osParentName, osName, poSharedResource->GetPAM()),
     166             :       m_poSharedResource(poSharedResource), m_aoDims(aoDims), m_oType(oType),
     167             :       m_aoDtypeElts(aoDtypeElts), m_anBlockSize(anBlockSize),
     168        1168 :       m_oAttrGroup(m_osFullName, /*bContainerIsGroup=*/false)
     169             : {
     170        1168 :     m_nTotalTileCount = ComputeTileCount(osName, aoDims, anBlockSize);
     171        1168 :     if (m_nTotalTileCount == 0)
     172           2 :         return;
     173             : 
     174             :     // Compute individual tile size
     175             :     const size_t nSourceSize =
     176        1166 :         m_aoDtypeElts.back().nativeOffset + m_aoDtypeElts.back().nativeSize;
     177        1166 :     m_nTileSize = nSourceSize;
     178        2872 :     for (const auto &nBlockSize : m_anBlockSize)
     179             :     {
     180        1706 :         m_nTileSize *= static_cast<size_t>(nBlockSize);
     181             :     }
     182             : 
     183        1166 :     m_bUseOptimizedCodePaths = CPLTestBool(
     184             :         CPLGetConfigOption("GDAL_ZARR_USE_OPTIMIZED_CODE_PATHS", "YES"));
     185             : }
     186             : 
     187             : /************************************************************************/
     188             : /*                              ~ZarrArray()                            */
     189             : /************************************************************************/
     190             : 
     191        1168 : ZarrArray::~ZarrArray()
     192             : {
     193        1168 :     if (m_pabyNoData)
     194             :     {
     195         175 :         m_oType.FreeDynamicMemory(&m_pabyNoData[0]);
     196         175 :         CPLFree(m_pabyNoData);
     197             :     }
     198             : 
     199        1168 :     DeallocateDecodedTileData();
     200        1168 : }
     201             : 
     202             : /************************************************************************/
     203             : /*              ZarrArray::SerializeSpecialAttributes()                 */
     204             : /************************************************************************/
     205             : 
     206         405 : CPLJSONObject ZarrArray::SerializeSpecialAttributes()
     207             : {
     208         405 :     m_bSRSModified = false;
     209         405 :     m_oAttrGroup.UnsetModified();
     210             : 
     211         405 :     auto oAttrs = m_oAttrGroup.Serialize();
     212             : 
     213         405 :     if (m_poSRS)
     214             :     {
     215          37 :         CPLJSONObject oCRS;
     216          37 :         const char *const apszOptions[] = {"FORMAT=WKT2_2019", nullptr};
     217          37 :         char *pszWKT = nullptr;
     218          37 :         if (m_poSRS->exportToWkt(&pszWKT, apszOptions) == OGRERR_NONE)
     219             :         {
     220          37 :             oCRS.Add("wkt", pszWKT);
     221             :         }
     222          37 :         CPLFree(pszWKT);
     223             : 
     224             :         {
     225          74 :             CPLErrorStateBackuper oErrorStateBackuper(CPLQuietErrorHandler);
     226          37 :             char *projjson = nullptr;
     227          74 :             if (m_poSRS->exportToPROJJSON(&projjson, nullptr) == OGRERR_NONE &&
     228          37 :                 projjson != nullptr)
     229             :             {
     230          74 :                 CPLJSONDocument oDocProjJSON;
     231          37 :                 if (oDocProjJSON.LoadMemory(std::string(projjson)))
     232             :                 {
     233          37 :                     oCRS.Add("projjson", oDocProjJSON.GetRoot());
     234             :                 }
     235             :             }
     236          37 :             CPLFree(projjson);
     237             :         }
     238             : 
     239          37 :         const char *pszAuthorityCode = m_poSRS->GetAuthorityCode(nullptr);
     240          37 :         const char *pszAuthorityName = m_poSRS->GetAuthorityName(nullptr);
     241          37 :         if (pszAuthorityCode && pszAuthorityName &&
     242           4 :             EQUAL(pszAuthorityName, "EPSG"))
     243             :         {
     244           4 :             oCRS.Add("url",
     245           8 :                      std::string("http://www.opengis.net/def/crs/EPSG/0/") +
     246             :                          pszAuthorityCode);
     247             :         }
     248             : 
     249          37 :         oAttrs.Add(CRS_ATTRIBUTE_NAME, oCRS);
     250             :     }
     251             : 
     252         405 :     if (m_osUnit.empty())
     253             :     {
     254         396 :         if (m_bUnitModified)
     255           0 :             oAttrs.Delete(CF_UNITS);
     256             :     }
     257             :     else
     258             :     {
     259           9 :         oAttrs.Set(CF_UNITS, m_osUnit);
     260             :     }
     261         405 :     m_bUnitModified = false;
     262             : 
     263         405 :     if (!m_bHasOffset)
     264             :     {
     265         402 :         oAttrs.Delete(CF_ADD_OFFSET);
     266             :     }
     267             :     else
     268             :     {
     269           3 :         oAttrs.Set(CF_ADD_OFFSET, m_dfOffset);
     270             :     }
     271         405 :     m_bOffsetModified = false;
     272             : 
     273         405 :     if (!m_bHasScale)
     274             :     {
     275         402 :         oAttrs.Delete(CF_SCALE_FACTOR);
     276             :     }
     277             :     else
     278             :     {
     279           3 :         oAttrs.Set(CF_SCALE_FACTOR, m_dfScale);
     280             :     }
     281         405 :     m_bScaleModified = false;
     282             : 
     283         405 :     return oAttrs;
     284             : }
     285             : 
     286             : /************************************************************************/
     287             : /*                          FillBlockSize()                             */
     288             : /************************************************************************/
     289             : 
     290             : /* static */
     291         429 : bool ZarrArray::FillBlockSize(
     292             :     const std::vector<std::shared_ptr<GDALDimension>> &aoDimensions,
     293             :     const GDALExtendedDataType &oDataType, std::vector<GUInt64> &anBlockSize,
     294             :     CSLConstList papszOptions)
     295             : {
     296         429 :     const auto nDims = aoDimensions.size();
     297         429 :     anBlockSize.resize(nDims);
     298        1020 :     for (size_t i = 0; i < nDims; ++i)
     299         591 :         anBlockSize[i] = 1;
     300         429 :     if (nDims >= 2)
     301             :     {
     302         326 :         anBlockSize[nDims - 2] =
     303         326 :             std::min(std::max<GUInt64>(1, aoDimensions[nDims - 2]->GetSize()),
     304         326 :                      static_cast<GUInt64>(256));
     305         326 :         anBlockSize[nDims - 1] =
     306         326 :             std::min(std::max<GUInt64>(1, aoDimensions[nDims - 1]->GetSize()),
     307         489 :                      static_cast<GUInt64>(256));
     308             :     }
     309         266 :     else if (nDims == 1)
     310             :     {
     311         219 :         anBlockSize[0] = std::max<GUInt64>(1, aoDimensions[0]->GetSize());
     312             :     }
     313             : 
     314         429 :     const char *pszBlockSize = CSLFetchNameValue(papszOptions, "BLOCKSIZE");
     315         429 :     if (pszBlockSize)
     316             :     {
     317             :         const auto aszTokens(
     318          19 :             CPLStringList(CSLTokenizeString2(pszBlockSize, ",", 0)));
     319          19 :         if (static_cast<size_t>(aszTokens.size()) != nDims)
     320             :         {
     321           5 :             CPLError(CE_Failure, CPLE_AppDefined,
     322             :                      "Invalid number of values in BLOCKSIZE");
     323           5 :             return false;
     324             :         }
     325          14 :         size_t nBlockSize = oDataType.GetSize();
     326          43 :         for (size_t i = 0; i < nDims; ++i)
     327             :         {
     328          31 :             anBlockSize[i] = static_cast<GUInt64>(CPLAtoGIntBig(aszTokens[i]));
     329          31 :             if (anBlockSize[i] == 0)
     330             :             {
     331           1 :                 CPLError(CE_Failure, CPLE_AppDefined,
     332             :                          "Values in BLOCKSIZE should be > 0");
     333           1 :                 return false;
     334             :             }
     335          30 :             if (anBlockSize[i] >
     336          30 :                 std::numeric_limits<size_t>::max() / nBlockSize)
     337             :             {
     338           1 :                 CPLError(CE_Failure, CPLE_AppDefined,
     339             :                          "Too large values in BLOCKSIZE");
     340           1 :                 return false;
     341             :             }
     342          29 :             nBlockSize *= static_cast<size_t>(anBlockSize[i]);
     343             :         }
     344             :     }
     345         422 :     return true;
     346             : }
     347             : 
     348             : /************************************************************************/
     349             : /*                      DeallocateDecodedTileData()                     */
     350             : /************************************************************************/
     351             : 
     352        2657 : void ZarrArray::DeallocateDecodedTileData()
     353             : {
     354        2657 :     if (!m_abyDecodedTileData.empty())
     355             :     {
     356         224 :         const size_t nDTSize = m_oType.GetSize();
     357         224 :         GByte *pDst = &m_abyDecodedTileData[0];
     358         224 :         const size_t nValues = m_abyDecodedTileData.size() / nDTSize;
     359         456 :         for (const auto &elt : m_aoDtypeElts)
     360             :         {
     361         232 :             if (elt.nativeType == DtypeElt::NativeType::STRING_ASCII ||
     362         231 :                 elt.nativeType == DtypeElt::NativeType::STRING_UNICODE)
     363             :             {
     364           2 :                 for (size_t i = 0; i < nValues; i++, pDst += nDTSize)
     365             :                 {
     366             :                     char *ptr;
     367           1 :                     char **pptr =
     368           1 :                         reinterpret_cast<char **>(pDst + elt.gdalOffset);
     369           1 :                     memcpy(&ptr, pptr, sizeof(ptr));
     370           1 :                     VSIFree(ptr);
     371             :                 }
     372             :             }
     373             :         }
     374             :     }
     375        2657 : }
     376             : 
     377             : /************************************************************************/
     378             : /*                             EncodeElt()                              */
     379             : /************************************************************************/
     380             : 
     381             : /* Encode from GDAL raw type to Zarr native type */
     382             : /*static*/
     383         121 : void ZarrArray::EncodeElt(const std::vector<DtypeElt> &elts, const GByte *pSrc,
     384             :                           GByte *pDst)
     385             : {
     386         243 :     for (const auto &elt : elts)
     387             :     {
     388         122 :         if (elt.nativeType == DtypeElt::NativeType::STRING_UNICODE)
     389             :         {
     390           0 :             const char *pStr =
     391           0 :                 *reinterpret_cast<const char *const *>(pSrc + elt.gdalOffset);
     392           0 :             if (pStr)
     393             :             {
     394             :                 try
     395             :                 {
     396           0 :                     const auto ucs4 = UTF8ToUCS4(pStr, elt.needByteSwapping);
     397           0 :                     const auto ucs4Len = ucs4.size();
     398           0 :                     memcpy(pDst + elt.nativeOffset, ucs4.data(),
     399           0 :                            std::min(ucs4Len, elt.nativeSize));
     400           0 :                     if (ucs4Len > elt.nativeSize)
     401             :                     {
     402           0 :                         CPLError(CE_Warning, CPLE_AppDefined,
     403             :                                  "Too long string truncated");
     404             :                     }
     405           0 :                     else if (ucs4Len < elt.nativeSize)
     406             :                     {
     407           0 :                         memset(pDst + elt.nativeOffset + ucs4Len, 0,
     408           0 :                                elt.nativeSize - ucs4Len);
     409             :                     }
     410             :                 }
     411           0 :                 catch (const std::exception &)
     412             :                 {
     413           0 :                     memset(pDst + elt.nativeOffset, 0, elt.nativeSize);
     414             :                 }
     415             :             }
     416             :             else
     417             :             {
     418           0 :                 memset(pDst + elt.nativeOffset, 0, elt.nativeSize);
     419             :             }
     420             :         }
     421         122 :         else if (elt.needByteSwapping)
     422             :         {
     423         120 :             if (elt.nativeSize == 2)
     424             :             {
     425          24 :                 if (elt.gdalTypeIsApproxOfNative)
     426             :                 {
     427           0 :                     CPLAssert(elt.nativeType == DtypeElt::NativeType::IEEEFP);
     428           0 :                     CPLAssert(elt.gdalType.GetNumericDataType() == GDT_Float32);
     429           0 :                     const uint32_t uint32Val =
     430           0 :                         *reinterpret_cast<const uint32_t *>(pSrc +
     431           0 :                                                             elt.gdalOffset);
     432           0 :                     bool bHasWarned = false;
     433             :                     uint16_t uint16Val =
     434           0 :                         CPL_SWAP16(CPLFloatToHalf(uint32Val, bHasWarned));
     435           0 :                     memcpy(pDst + elt.nativeOffset, &uint16Val,
     436             :                            sizeof(uint16Val));
     437             :                 }
     438             :                 else
     439             :                 {
     440          24 :                     const uint16_t val =
     441          24 :                         CPL_SWAP16(*reinterpret_cast<const uint16_t *>(
     442             :                             pSrc + elt.gdalOffset));
     443          24 :                     memcpy(pDst + elt.nativeOffset, &val, sizeof(val));
     444             :                 }
     445             :             }
     446          96 :             else if (elt.nativeSize == 4)
     447             :             {
     448          36 :                 const uint32_t val = CPL_SWAP32(
     449             :                     *reinterpret_cast<const uint32_t *>(pSrc + elt.gdalOffset));
     450          36 :                 memcpy(pDst + elt.nativeOffset, &val, sizeof(val));
     451             :             }
     452          60 :             else if (elt.nativeSize == 8)
     453             :             {
     454          48 :                 if (elt.nativeType == DtypeElt::NativeType::COMPLEX_IEEEFP)
     455             :                 {
     456          12 :                     uint32_t val =
     457          12 :                         CPL_SWAP32(*reinterpret_cast<const uint32_t *>(
     458             :                             pSrc + elt.gdalOffset));
     459          12 :                     memcpy(pDst + elt.nativeOffset, &val, sizeof(val));
     460          12 :                     val = CPL_SWAP32(*reinterpret_cast<const uint32_t *>(
     461             :                         pSrc + elt.gdalOffset + 4));
     462          12 :                     memcpy(pDst + elt.nativeOffset + 4, &val, sizeof(val));
     463             :                 }
     464             :                 else
     465             :                 {
     466          36 :                     const uint64_t val =
     467          36 :                         CPL_SWAP64(*reinterpret_cast<const uint64_t *>(
     468             :                             pSrc + elt.gdalOffset));
     469          36 :                     memcpy(pDst + elt.nativeOffset, &val, sizeof(val));
     470             :                 }
     471             :             }
     472          12 :             else if (elt.nativeSize == 16)
     473             :             {
     474          12 :                 uint64_t val = CPL_SWAP64(
     475             :                     *reinterpret_cast<const uint64_t *>(pSrc + elt.gdalOffset));
     476          12 :                 memcpy(pDst + elt.nativeOffset, &val, sizeof(val));
     477          12 :                 val = CPL_SWAP64(*reinterpret_cast<const uint64_t *>(
     478             :                     pSrc + elt.gdalOffset + 8));
     479          12 :                 memcpy(pDst + elt.nativeOffset + 8, &val, sizeof(val));
     480             :             }
     481             :             else
     482             :             {
     483           0 :                 CPLAssert(false);
     484             :             }
     485             :         }
     486           2 :         else if (elt.gdalTypeIsApproxOfNative)
     487             :         {
     488           0 :             if (elt.nativeType == DtypeElt::NativeType::IEEEFP &&
     489           0 :                 elt.nativeSize == 2)
     490             :             {
     491           0 :                 CPLAssert(elt.gdalType.GetNumericDataType() == GDT_Float32);
     492           0 :                 const uint32_t uint32Val =
     493           0 :                     *reinterpret_cast<const uint32_t *>(pSrc + elt.gdalOffset);
     494           0 :                 bool bHasWarned = false;
     495             :                 const uint16_t uint16Val =
     496           0 :                     CPLFloatToHalf(uint32Val, bHasWarned);
     497           0 :                 memcpy(pDst + elt.nativeOffset, &uint16Val, sizeof(uint16Val));
     498             :             }
     499             :             else
     500             :             {
     501           0 :                 CPLAssert(false);
     502             :             }
     503             :         }
     504           2 :         else if (elt.nativeType == DtypeElt::NativeType::STRING_ASCII)
     505             :         {
     506           0 :             const char *pStr =
     507           0 :                 *reinterpret_cast<const char *const *>(pSrc + elt.gdalOffset);
     508           0 :             if (pStr)
     509             :             {
     510           0 :                 const size_t nLen = strlen(pStr);
     511           0 :                 memcpy(pDst + elt.nativeOffset, pStr,
     512           0 :                        std::min(nLen, elt.nativeSize));
     513           0 :                 if (nLen < elt.nativeSize)
     514           0 :                     memset(pDst + elt.nativeOffset + nLen, 0,
     515           0 :                            elt.nativeSize - nLen);
     516             :             }
     517             :             else
     518             :             {
     519           0 :                 memset(pDst + elt.nativeOffset, 0, elt.nativeSize);
     520             :             }
     521             :         }
     522             :         else
     523             :         {
     524           2 :             CPLAssert(elt.nativeSize == elt.gdalSize);
     525           2 :             memcpy(pDst + elt.nativeOffset, pSrc + elt.gdalOffset,
     526           2 :                    elt.nativeSize);
     527             :         }
     528             :     }
     529         121 : }
     530             : 
     531             : /************************************************************************/
     532             : /*                ZarrArray::SerializeNumericNoData()                   */
     533             : /************************************************************************/
     534             : 
     535          16 : void ZarrArray::SerializeNumericNoData(CPLJSONObject &oRoot) const
     536             : {
     537          16 :     if (m_oType.GetNumericDataType() == GDT_Int64)
     538             :     {
     539           0 :         const auto nVal = GetNoDataValueAsInt64();
     540           0 :         oRoot.Add("fill_value", static_cast<GInt64>(nVal));
     541             :     }
     542          16 :     else if (m_oType.GetNumericDataType() == GDT_UInt64)
     543             :     {
     544           0 :         const auto nVal = GetNoDataValueAsUInt64();
     545           0 :         oRoot.Add("fill_value", static_cast<uint64_t>(nVal));
     546             :     }
     547             :     else
     548             :     {
     549          16 :         const double dfVal = GetNoDataValueAsDouble();
     550          16 :         if (std::isnan(dfVal))
     551           2 :             oRoot.Add("fill_value", "NaN");
     552          14 :         else if (dfVal == std::numeric_limits<double>::infinity())
     553           2 :             oRoot.Add("fill_value", "Infinity");
     554          12 :         else if (dfVal == -std::numeric_limits<double>::infinity())
     555           2 :             oRoot.Add("fill_value", "-Infinity");
     556          10 :         else if (GDALDataTypeIsInteger(m_oType.GetNumericDataType()))
     557           8 :             oRoot.Add("fill_value", static_cast<GInt64>(dfVal));
     558             :         else
     559           2 :             oRoot.Add("fill_value", dfVal);
     560             :     }
     561          16 : }
     562             : 
     563             : /************************************************************************/
     564             : /*                    ZarrArray::GetSpatialRef()                        */
     565             : /************************************************************************/
     566             : 
     567          20 : std::shared_ptr<OGRSpatialReference> ZarrArray::GetSpatialRef() const
     568             : {
     569          20 :     if (!CheckValidAndErrorOutIfNot())
     570           0 :         return nullptr;
     571             : 
     572          20 :     if (m_poSRS)
     573          10 :         return m_poSRS;
     574          10 :     return GDALPamMDArray::GetSpatialRef();
     575             : }
     576             : 
     577             : /************************************************************************/
     578             : /*                        SetRawNoDataValue()                           */
     579             : /************************************************************************/
     580             : 
     581          18 : bool ZarrArray::SetRawNoDataValue(const void *pRawNoData)
     582             : {
     583          18 :     if (!CheckValidAndErrorOutIfNot())
     584           0 :         return false;
     585             : 
     586          18 :     if (!m_bUpdatable)
     587             :     {
     588           0 :         CPLError(CE_Failure, CPLE_AppDefined, "Array opened in read-only mode");
     589           0 :         return false;
     590             :     }
     591          18 :     m_bDefinitionModified = true;
     592          18 :     RegisterNoDataValue(pRawNoData);
     593          18 :     return true;
     594             : }
     595             : 
     596             : /************************************************************************/
     597             : /*                        RegisterNoDataValue()                         */
     598             : /************************************************************************/
     599             : 
     600         175 : void ZarrArray::RegisterNoDataValue(const void *pNoData)
     601             : {
     602         175 :     if (m_pabyNoData)
     603             :     {
     604           0 :         m_oType.FreeDynamicMemory(&m_pabyNoData[0]);
     605             :     }
     606             : 
     607         175 :     if (pNoData == nullptr)
     608             :     {
     609           0 :         CPLFree(m_pabyNoData);
     610           0 :         m_pabyNoData = nullptr;
     611             :     }
     612             :     else
     613             :     {
     614         175 :         const auto nSize = m_oType.GetSize();
     615         175 :         if (m_pabyNoData == nullptr)
     616             :         {
     617         175 :             m_pabyNoData = static_cast<GByte *>(CPLMalloc(nSize));
     618             :         }
     619         175 :         memset(m_pabyNoData, 0, nSize);
     620         175 :         GDALExtendedDataType::CopyValue(pNoData, m_oType, m_pabyNoData,
     621         175 :                                         m_oType);
     622             :     }
     623         175 : }
     624             : 
     625             : /************************************************************************/
     626             : /*                      ZarrArray::BlockTranspose()                     */
     627             : /************************************************************************/
     628             : 
     629          66 : void ZarrArray::BlockTranspose(const ZarrByteVectorQuickResize &abySrc,
     630             :                                ZarrByteVectorQuickResize &abyDst,
     631             :                                bool bDecode) const
     632             : {
     633             :     // Perform transposition
     634          66 :     const size_t nDims = m_anBlockSize.size();
     635             :     const size_t nSourceSize =
     636          66 :         m_aoDtypeElts.back().nativeOffset + m_aoDtypeElts.back().nativeSize;
     637             : 
     638             :     struct Stack
     639             :     {
     640             :         size_t nIters = 0;
     641             :         const GByte *src_ptr = nullptr;
     642             :         GByte *dst_ptr = nullptr;
     643             :         size_t src_inc_offset = 0;
     644             :         size_t dst_inc_offset = 0;
     645             :     };
     646             : 
     647         132 :     std::vector<Stack> stack(nDims);
     648             :     stack.emplace_back(
     649          66 :         Stack());  // to make gcc 9.3 -O2 -Wnull-dereference happy
     650             : 
     651          66 :     if (bDecode)
     652             :     {
     653          46 :         stack[0].src_inc_offset = nSourceSize;
     654         118 :         for (size_t i = 1; i < nDims; ++i)
     655             :         {
     656         144 :             stack[i].src_inc_offset = stack[i - 1].src_inc_offset *
     657          72 :                                       static_cast<size_t>(m_anBlockSize[i - 1]);
     658             :         }
     659             : 
     660          46 :         stack[nDims - 1].dst_inc_offset = nSourceSize;
     661         118 :         for (size_t i = nDims - 1; i > 0;)
     662             :         {
     663          72 :             --i;
     664         144 :             stack[i].dst_inc_offset = stack[i + 1].dst_inc_offset *
     665          72 :                                       static_cast<size_t>(m_anBlockSize[i + 1]);
     666             :         }
     667             :     }
     668             :     else
     669             :     {
     670          20 :         stack[0].dst_inc_offset = nSourceSize;
     671          60 :         for (size_t i = 1; i < nDims; ++i)
     672             :         {
     673          80 :             stack[i].dst_inc_offset = stack[i - 1].dst_inc_offset *
     674          40 :                                       static_cast<size_t>(m_anBlockSize[i - 1]);
     675             :         }
     676             : 
     677          20 :         stack[nDims - 1].src_inc_offset = nSourceSize;
     678          60 :         for (size_t i = nDims - 1; i > 0;)
     679             :         {
     680          40 :             --i;
     681          80 :             stack[i].src_inc_offset = stack[i + 1].src_inc_offset *
     682          40 :                                       static_cast<size_t>(m_anBlockSize[i + 1]);
     683             :         }
     684             :     }
     685             : 
     686          66 :     stack[0].src_ptr = abySrc.data();
     687          66 :     stack[0].dst_ptr = &abyDst[0];
     688             : 
     689          66 :     size_t dimIdx = 0;
     690        1058 : lbl_next_depth:
     691        1058 :     if (dimIdx == nDims)
     692             :     {
     693         744 :         void *dst_ptr = stack[nDims].dst_ptr;
     694         744 :         const void *src_ptr = stack[nDims].src_ptr;
     695         744 :         if (nSourceSize == 1)
     696         264 :             *stack[nDims].dst_ptr = *stack[nDims].src_ptr;
     697         480 :         else if (nSourceSize == 2)
     698         120 :             *static_cast<uint16_t *>(dst_ptr) =
     699         120 :                 *static_cast<const uint16_t *>(src_ptr);
     700         360 :         else if (nSourceSize == 4)
     701         168 :             *static_cast<uint32_t *>(dst_ptr) =
     702         168 :                 *static_cast<const uint32_t *>(src_ptr);
     703         192 :         else if (nSourceSize == 8)
     704         168 :             *static_cast<uint64_t *>(dst_ptr) =
     705         168 :                 *static_cast<const uint64_t *>(src_ptr);
     706             :         else
     707          24 :             memcpy(dst_ptr, src_ptr, nSourceSize);
     708             :     }
     709             :     else
     710             :     {
     711         314 :         stack[dimIdx].nIters = static_cast<size_t>(m_anBlockSize[dimIdx]);
     712             :         while (true)
     713             :         {
     714         992 :             dimIdx++;
     715         992 :             stack[dimIdx].src_ptr = stack[dimIdx - 1].src_ptr;
     716         992 :             stack[dimIdx].dst_ptr = stack[dimIdx - 1].dst_ptr;
     717         992 :             goto lbl_next_depth;
     718         992 :         lbl_return_to_caller:
     719         992 :             dimIdx--;
     720         992 :             if ((--stack[dimIdx].nIters) == 0)
     721         314 :                 break;
     722         678 :             stack[dimIdx].src_ptr += stack[dimIdx].src_inc_offset;
     723         678 :             stack[dimIdx].dst_ptr += stack[dimIdx].dst_inc_offset;
     724             :         }
     725             :     }
     726        1058 :     if (dimIdx > 0)
     727         992 :         goto lbl_return_to_caller;
     728          66 : }
     729             : 
     730             : /************************************************************************/
     731             : /*                        DecodeSourceElt()                             */
     732             : /************************************************************************/
     733             : 
     734             : /* static */
     735        1938 : void ZarrArray::DecodeSourceElt(const std::vector<DtypeElt> &elts,
     736             :                                 const GByte *pSrc, GByte *pDst)
     737             : {
     738        3908 :     for (const auto &elt : elts)
     739             :     {
     740        1970 :         if (elt.nativeType == DtypeElt::NativeType::STRING_UNICODE)
     741             :         {
     742             :             char *ptr;
     743           0 :             char **pDstPtr = reinterpret_cast<char **>(pDst + elt.gdalOffset);
     744           0 :             memcpy(&ptr, pDstPtr, sizeof(ptr));
     745           0 :             VSIFree(ptr);
     746             : 
     747           0 :             char *pDstStr = UCS4ToUTF8(pSrc + elt.nativeOffset, elt.nativeSize,
     748           0 :                                        elt.needByteSwapping);
     749           0 :             memcpy(pDstPtr, &pDstStr, sizeof(pDstStr));
     750             :         }
     751        1970 :         else if (elt.needByteSwapping)
     752             :         {
     753        1922 :             if (elt.nativeSize == 2)
     754             :             {
     755             :                 uint16_t val;
     756         386 :                 memcpy(&val, pSrc + elt.nativeOffset, sizeof(val));
     757         386 :                 if (elt.gdalTypeIsApproxOfNative)
     758             :                 {
     759           2 :                     CPLAssert(elt.nativeType == DtypeElt::NativeType::IEEEFP);
     760           2 :                     CPLAssert(elt.gdalType.GetNumericDataType() == GDT_Float32);
     761           2 :                     uint32_t uint32Val = CPLHalfToFloat(CPL_SWAP16(val));
     762           2 :                     memcpy(pDst + elt.gdalOffset, &uint32Val,
     763             :                            sizeof(uint32Val));
     764             :                 }
     765             :                 else
     766             :                 {
     767         384 :                     *reinterpret_cast<uint16_t *>(pDst + elt.gdalOffset) =
     768         384 :                         CPL_SWAP16(val);
     769             :                 }
     770             :             }
     771        1536 :             else if (elt.nativeSize == 4)
     772             :             {
     773             :                 uint32_t val;
     774         576 :                 memcpy(&val, pSrc + elt.nativeOffset, sizeof(val));
     775         576 :                 *reinterpret_cast<uint32_t *>(pDst + elt.gdalOffset) =
     776         576 :                     CPL_SWAP32(val);
     777             :             }
     778         960 :             else if (elt.nativeSize == 8)
     779             :             {
     780         768 :                 if (elt.nativeType == DtypeElt::NativeType::COMPLEX_IEEEFP)
     781             :                 {
     782             :                     uint32_t val;
     783         192 :                     memcpy(&val, pSrc + elt.nativeOffset, sizeof(val));
     784         192 :                     *reinterpret_cast<uint32_t *>(pDst + elt.gdalOffset) =
     785         192 :                         CPL_SWAP32(val);
     786         192 :                     memcpy(&val, pSrc + elt.nativeOffset + 4, sizeof(val));
     787         192 :                     *reinterpret_cast<uint32_t *>(pDst + elt.gdalOffset + 4) =
     788         192 :                         CPL_SWAP32(val);
     789             :                 }
     790             :                 else
     791             :                 {
     792             :                     uint64_t val;
     793         576 :                     memcpy(&val, pSrc + elt.nativeOffset, sizeof(val));
     794         576 :                     *reinterpret_cast<uint64_t *>(pDst + elt.gdalOffset) =
     795         576 :                         CPL_SWAP64(val);
     796             :                 }
     797             :             }
     798         192 :             else if (elt.nativeSize == 16)
     799             :             {
     800             :                 uint64_t val;
     801         192 :                 memcpy(&val, pSrc + elt.nativeOffset, sizeof(val));
     802         192 :                 *reinterpret_cast<uint64_t *>(pDst + elt.gdalOffset) =
     803         192 :                     CPL_SWAP64(val);
     804         192 :                 memcpy(&val, pSrc + elt.nativeOffset + 8, sizeof(val));
     805         192 :                 *reinterpret_cast<uint64_t *>(pDst + elt.gdalOffset + 8) =
     806         192 :                     CPL_SWAP64(val);
     807             :             }
     808             :             else
     809             :             {
     810           0 :                 CPLAssert(false);
     811             :             }
     812             :         }
     813          48 :         else if (elt.gdalTypeIsApproxOfNative)
     814             :         {
     815           2 :             if (elt.nativeType == DtypeElt::NativeType::IEEEFP &&
     816           2 :                 elt.nativeSize == 2)
     817             :             {
     818           2 :                 CPLAssert(elt.gdalType.GetNumericDataType() == GDT_Float32);
     819             :                 uint16_t uint16Val;
     820           2 :                 memcpy(&uint16Val, pSrc + elt.nativeOffset, sizeof(uint16Val));
     821           2 :                 uint32_t uint32Val = CPLHalfToFloat(uint16Val);
     822           2 :                 memcpy(pDst + elt.gdalOffset, &uint32Val, sizeof(uint32Val));
     823             :             }
     824             :             else
     825             :             {
     826           0 :                 CPLAssert(false);
     827             :             }
     828             :         }
     829          46 :         else if (elt.nativeType == DtypeElt::NativeType::STRING_ASCII)
     830             :         {
     831             :             char *ptr;
     832           3 :             char **pDstPtr = reinterpret_cast<char **>(pDst + elt.gdalOffset);
     833           3 :             memcpy(&ptr, pDstPtr, sizeof(ptr));
     834           3 :             VSIFree(ptr);
     835             : 
     836           3 :             char *pDstStr = static_cast<char *>(CPLMalloc(elt.nativeSize + 1));
     837           3 :             memcpy(pDstStr, pSrc + elt.nativeOffset, elt.nativeSize);
     838           3 :             pDstStr[elt.nativeSize] = 0;
     839           3 :             memcpy(pDstPtr, &pDstStr, sizeof(pDstStr));
     840             :         }
     841             :         else
     842             :         {
     843          43 :             CPLAssert(elt.nativeSize == elt.gdalSize);
     844          43 :             memcpy(pDst + elt.gdalOffset, pSrc + elt.nativeOffset,
     845          43 :                    elt.nativeSize);
     846             :         }
     847             :     }
     848        1938 : }
     849             : 
     850             : /************************************************************************/
     851             : /*                  ZarrArray::IAdviseReadCommon()                      */
     852             : /************************************************************************/
     853             : 
     854          12 : bool ZarrArray::IAdviseReadCommon(const GUInt64 *arrayStartIdx,
     855             :                                   const size_t *count,
     856             :                                   CSLConstList papszOptions,
     857             :                                   std::vector<uint64_t> &anIndicesCur,
     858             :                                   int &nThreadsMax,
     859             :                                   std::vector<uint64_t> &anReqTilesIndices,
     860             :                                   size_t &nReqTiles) const
     861             : {
     862          12 :     if (!CheckValidAndErrorOutIfNot())
     863           0 :         return false;
     864             : 
     865          12 :     const size_t nDims = m_aoDims.size();
     866          12 :     anIndicesCur.resize(nDims);
     867          24 :     std::vector<uint64_t> anIndicesMin(nDims);
     868          24 :     std::vector<uint64_t> anIndicesMax(nDims);
     869             : 
     870             :     // Compute min and max tile indices in each dimension, and the total
     871             :     // nomber of tiles this represents.
     872          12 :     nReqTiles = 1;
     873          36 :     for (size_t i = 0; i < nDims; ++i)
     874             :     {
     875          24 :         anIndicesMin[i] = arrayStartIdx[i] / m_anBlockSize[i];
     876          24 :         anIndicesMax[i] = (arrayStartIdx[i] + count[i] - 1) / m_anBlockSize[i];
     877             :         // Overflow on number of tiles already checked in Create()
     878          24 :         nReqTiles *= static_cast<size_t>(anIndicesMax[i] - anIndicesMin[i] + 1);
     879             :     }
     880             : 
     881             :     // Find available cache size
     882          12 :     const size_t nCacheSize = [papszOptions]()
     883             :     {
     884             :         size_t nCacheSizeTmp;
     885             :         const char *pszCacheSize =
     886          12 :             CSLFetchNameValue(papszOptions, "CACHE_SIZE");
     887          12 :         if (pszCacheSize)
     888             :         {
     889           4 :             const auto nCacheSizeBig = CPLAtoGIntBig(pszCacheSize);
     890           8 :             if (nCacheSizeBig < 0 || static_cast<uint64_t>(nCacheSizeBig) >
     891           4 :                                          std::numeric_limits<size_t>::max() / 2)
     892             :             {
     893           0 :                 CPLError(CE_Failure, CPLE_OutOfMemory, "Too big CACHE_SIZE");
     894           0 :                 return std::numeric_limits<size_t>::max();
     895             :             }
     896           4 :             nCacheSizeTmp = static_cast<size_t>(nCacheSizeBig);
     897             :         }
     898             :         else
     899             :         {
     900             :             // Arbitrarily take half of remaining cache size
     901           8 :             nCacheSizeTmp = static_cast<size_t>(std::min(
     902          16 :                 static_cast<uint64_t>(
     903           8 :                     (GDALGetCacheMax64() - GDALGetCacheUsed64()) / 2),
     904          16 :                 static_cast<uint64_t>(std::numeric_limits<size_t>::max() / 2)));
     905           8 :             CPLDebug(ZARR_DEBUG_KEY, "Using implicit CACHE_SIZE=" CPL_FRMT_GUIB,
     906             :                      static_cast<GUIntBig>(nCacheSizeTmp));
     907             :         }
     908          12 :         return nCacheSizeTmp;
     909          12 :     }();
     910          12 :     if (nCacheSize == std::numeric_limits<size_t>::max())
     911           0 :         return false;
     912             : 
     913             :     // Check that cache size is sufficient to hold all needed tiles.
     914             :     // Also check that anReqTilesIndices size computation won't overflow.
     915          12 :     if (nReqTiles > nCacheSize / std::max(m_nTileSize, nDims))
     916             :     {
     917           4 :         CPLError(
     918             :             CE_Failure, CPLE_OutOfMemory,
     919             :             "CACHE_SIZE=" CPL_FRMT_GUIB " is not big enough to cache "
     920             :             "all needed tiles. "
     921             :             "At least " CPL_FRMT_GUIB " bytes would be needed",
     922             :             static_cast<GUIntBig>(nCacheSize),
     923           4 :             static_cast<GUIntBig>(nReqTiles * std::max(m_nTileSize, nDims)));
     924           4 :         return false;
     925             :     }
     926             : 
     927           8 :     const char *pszNumThreads = CSLFetchNameValueDef(
     928             :         papszOptions, "NUM_THREADS",
     929             :         CPLGetConfigOption("GDAL_NUM_THREADS", "ALL_CPUS"));
     930           8 :     if (EQUAL(pszNumThreads, "ALL_CPUS"))
     931           8 :         nThreadsMax = CPLGetNumCPUs();
     932             :     else
     933           0 :         nThreadsMax = std::max(1, atoi(pszNumThreads));
     934           8 :     if (nThreadsMax > 1024)
     935           0 :         nThreadsMax = 1024;
     936           8 :     if (nThreadsMax <= 1)
     937           0 :         return true;
     938           8 :     CPLDebug(ZARR_DEBUG_KEY, "IAdviseRead(): Using up to %d threads",
     939             :              nThreadsMax);
     940             : 
     941           8 :     m_oMapTileIndexToCachedTile.clear();
     942             : 
     943             :     // Overflow checked above
     944             :     try
     945             :     {
     946           8 :         anReqTilesIndices.resize(nDims * nReqTiles);
     947             :     }
     948           0 :     catch (const std::bad_alloc &e)
     949             :     {
     950           0 :         CPLError(CE_Failure, CPLE_OutOfMemory,
     951           0 :                  "Cannot allocate anReqTilesIndices: %s", e.what());
     952           0 :         return false;
     953             :     }
     954             : 
     955           8 :     size_t dimIdx = 0;
     956           8 :     size_t nTileIter = 0;
     957       21608 : lbl_next_depth:
     958       21608 :     if (dimIdx == nDims)
     959             :     {
     960       21344 :         if (nDims == 2)
     961             :         {
     962             :             // optimize in common case
     963       21344 :             memcpy(&anReqTilesIndices[nTileIter * nDims], anIndicesCur.data(),
     964             :                    sizeof(uint64_t) * 2);
     965             :         }
     966           0 :         else if (nDims == 3)
     967             :         {
     968             :             // optimize in common case
     969           0 :             memcpy(&anReqTilesIndices[nTileIter * nDims], anIndicesCur.data(),
     970             :                    sizeof(uint64_t) * 3);
     971             :         }
     972             :         else
     973             :         {
     974           0 :             memcpy(&anReqTilesIndices[nTileIter * nDims], anIndicesCur.data(),
     975           0 :                    sizeof(uint64_t) * nDims);
     976             :         }
     977       21344 :         nTileIter++;
     978             :     }
     979             :     else
     980             :     {
     981             :         // This level of loop loops over blocks
     982         264 :         anIndicesCur[dimIdx] = anIndicesMin[dimIdx];
     983             :         while (true)
     984             :         {
     985       21600 :             dimIdx++;
     986       21600 :             goto lbl_next_depth;
     987       21600 :         lbl_return_to_caller:
     988       21600 :             dimIdx--;
     989       21600 :             if (anIndicesCur[dimIdx] == anIndicesMax[dimIdx])
     990         264 :                 break;
     991       21336 :             ++anIndicesCur[dimIdx];
     992             :         }
     993             :     }
     994       21608 :     if (dimIdx > 0)
     995       21600 :         goto lbl_return_to_caller;
     996           8 :     assert(nTileIter == nReqTiles);
     997             : 
     998           8 :     return true;
     999             : }
    1000             : 
    1001             : /************************************************************************/
    1002             : /*                           ZarrArray::IRead()                         */
    1003             : /************************************************************************/
    1004             : 
    1005        1572 : bool ZarrArray::IRead(const GUInt64 *arrayStartIdx, const size_t *count,
    1006             :                       const GInt64 *arrayStep, const GPtrDiff_t *bufferStride,
    1007             :                       const GDALExtendedDataType &bufferDataType,
    1008             :                       void *pDstBuffer) const
    1009             : {
    1010        1572 :     if (!CheckValidAndErrorOutIfNot())
    1011           0 :         return false;
    1012             : 
    1013        1572 :     if (!AllocateWorkingBuffers())
    1014           3 :         return false;
    1015             : 
    1016             :     // Need to be kept in top-level scope
    1017        3138 :     std::vector<GUInt64> arrayStartIdxMod;
    1018        3138 :     std::vector<GInt64> arrayStepMod;
    1019        3138 :     std::vector<GPtrDiff_t> bufferStrideMod;
    1020             : 
    1021        1569 :     const size_t nDims = m_aoDims.size();
    1022        1569 :     bool negativeStep = false;
    1023        4322 :     for (size_t i = 0; i < nDims; ++i)
    1024             :     {
    1025        2955 :         if (arrayStep[i] < 0)
    1026             :         {
    1027         202 :             negativeStep = true;
    1028         202 :             break;
    1029             :         }
    1030             :     }
    1031             : 
    1032             :     // const auto eBufferDT = bufferDataType.GetNumericDataType();
    1033        1569 :     const auto nBufferDTSize = static_cast<int>(bufferDataType.GetSize());
    1034             : 
    1035             :     // Make sure that arrayStep[i] are positive for sake of simplicity
    1036        1569 :     if (negativeStep)
    1037             :     {
    1038             : #if defined(__GNUC__)
    1039             : #pragma GCC diagnostic push
    1040             : #pragma GCC diagnostic ignored "-Wnull-dereference"
    1041             : #endif
    1042         202 :         arrayStartIdxMod.resize(nDims);
    1043         202 :         arrayStepMod.resize(nDims);
    1044         202 :         bufferStrideMod.resize(nDims);
    1045             : #if defined(__GNUC__)
    1046             : #pragma GCC diagnostic pop
    1047             : #endif
    1048         606 :         for (size_t i = 0; i < nDims; ++i)
    1049             :         {
    1050         404 :             if (arrayStep[i] < 0)
    1051             :             {
    1052         808 :                 arrayStartIdxMod[i] =
    1053         404 :                     arrayStartIdx[i] - (count[i] - 1) * (-arrayStep[i]);
    1054         404 :                 arrayStepMod[i] = -arrayStep[i];
    1055         404 :                 bufferStrideMod[i] = -bufferStride[i];
    1056         404 :                 pDstBuffer =
    1057             :                     static_cast<GByte *>(pDstBuffer) +
    1058         404 :                     bufferStride[i] *
    1059         404 :                         static_cast<GPtrDiff_t>(nBufferDTSize * (count[i] - 1));
    1060             :             }
    1061             :             else
    1062             :             {
    1063           0 :                 arrayStartIdxMod[i] = arrayStartIdx[i];
    1064           0 :                 arrayStepMod[i] = arrayStep[i];
    1065           0 :                 bufferStrideMod[i] = bufferStride[i];
    1066             :             }
    1067             :         }
    1068         202 :         arrayStartIdx = arrayStartIdxMod.data();
    1069         202 :         arrayStep = arrayStepMod.data();
    1070         202 :         bufferStride = bufferStrideMod.data();
    1071             :     }
    1072             : 
    1073        3138 :     std::vector<uint64_t> indicesOuterLoop(nDims + 1);
    1074        3138 :     std::vector<GByte *> dstPtrStackOuterLoop(nDims + 1);
    1075             : 
    1076        3138 :     std::vector<uint64_t> indicesInnerLoop(nDims + 1);
    1077        3138 :     std::vector<GByte *> dstPtrStackInnerLoop(nDims + 1);
    1078             : 
    1079        3138 :     std::vector<GPtrDiff_t> dstBufferStrideBytes;
    1080        4726 :     for (size_t i = 0; i < nDims; ++i)
    1081             :     {
    1082        3157 :         dstBufferStrideBytes.push_back(bufferStride[i] *
    1083        3157 :                                        static_cast<GPtrDiff_t>(nBufferDTSize));
    1084             :     }
    1085        1569 :     dstBufferStrideBytes.push_back(0);
    1086             : 
    1087        1569 :     const auto nDTSize = m_oType.GetSize();
    1088             : 
    1089        3138 :     std::vector<uint64_t> tileIndices(nDims);
    1090             :     const size_t nSourceSize =
    1091        1569 :         m_aoDtypeElts.back().nativeOffset + m_aoDtypeElts.back().nativeSize;
    1092             : 
    1093        3138 :     std::vector<size_t> countInnerLoopInit(nDims + 1, 1);
    1094        3138 :     std::vector<size_t> countInnerLoop(nDims);
    1095             : 
    1096        3121 :     const bool bBothAreNumericDT = m_oType.GetClass() == GEDTC_NUMERIC &&
    1097        1552 :                                    bufferDataType.GetClass() == GEDTC_NUMERIC;
    1098             :     const bool bSameNumericDT =
    1099        3121 :         bBothAreNumericDT &&
    1100        1552 :         m_oType.GetNumericDataType() == bufferDataType.GetNumericDataType();
    1101        1569 :     const auto nSameDTSize = bSameNumericDT ? m_oType.GetSize() : 0;
    1102             :     const bool bSameCompoundAndNoDynamicMem =
    1103        1572 :         m_oType.GetClass() == GEDTC_COMPOUND && m_oType == bufferDataType &&
    1104           3 :         !m_oType.NeedsFreeDynamicMemory();
    1105        3138 :     std::vector<GByte> abyTargetNoData;
    1106        1569 :     bool bNoDataIsZero = false;
    1107             : 
    1108        1569 :     size_t dimIdx = 0;
    1109        1569 :     dstPtrStackOuterLoop[0] = static_cast<GByte *>(pDstBuffer);
    1110       29798 : lbl_next_depth:
    1111       29798 :     if (dimIdx == nDims)
    1112             :     {
    1113       25292 :         size_t dimIdxSubLoop = 0;
    1114       25292 :         dstPtrStackInnerLoop[0] = dstPtrStackOuterLoop[nDims];
    1115       25292 :         bool bEmptyTile = false;
    1116             : 
    1117       25292 :         const GByte *pabySrcTile = m_abyDecodedTileData.empty()
    1118       24300 :                                        ? m_abyRawTileData.data()
    1119       25292 :                                        : m_abyDecodedTileData.data();
    1120       25292 :         bool bMatchFoundInMapTileIndexToCachedTile = false;
    1121             : 
    1122             :         // Use cache built by IAdviseRead() if possible
    1123       25292 :         if (!m_oMapTileIndexToCachedTile.empty())
    1124             :         {
    1125       21352 :             uint64_t nTileIdx = 0;
    1126       64056 :             for (size_t j = 0; j < nDims; ++j)
    1127             :             {
    1128       42704 :                 if (j > 0)
    1129       21352 :                     nTileIdx *= m_aoDims[j - 1]->GetSize();
    1130       42704 :                 nTileIdx += tileIndices[j];
    1131             :             }
    1132       21352 :             const auto oIter = m_oMapTileIndexToCachedTile.find(nTileIdx);
    1133       21352 :             if (oIter != m_oMapTileIndexToCachedTile.end())
    1134             :             {
    1135       21344 :                 bMatchFoundInMapTileIndexToCachedTile = true;
    1136       21344 :                 if (oIter->second.abyDecoded.empty())
    1137             :                 {
    1138           4 :                     bEmptyTile = true;
    1139             :                 }
    1140             :                 else
    1141             :                 {
    1142       21340 :                     pabySrcTile = oIter->second.abyDecoded.data();
    1143             :                 }
    1144             :             }
    1145             :             else
    1146             :             {
    1147           8 :                 CPLDebugOnly(ZARR_DEBUG_KEY,
    1148             :                              "Cache miss for tile " CPL_FRMT_GUIB,
    1149             :                              static_cast<GUIntBig>(nTileIdx));
    1150             :             }
    1151             :         }
    1152             : 
    1153       25292 :         if (!bMatchFoundInMapTileIndexToCachedTile)
    1154             :         {
    1155        3948 :             if (!tileIndices.empty() && tileIndices == m_anCachedTiledIndices)
    1156             :             {
    1157         221 :                 if (!m_bCachedTiledValid)
    1158           0 :                     return false;
    1159         221 :                 bEmptyTile = m_bCachedTiledEmpty;
    1160             :             }
    1161             :             else
    1162             :             {
    1163        3727 :                 if (!FlushDirtyTile())
    1164           0 :                     return false;
    1165             : 
    1166        3727 :                 m_anCachedTiledIndices = tileIndices;
    1167        3727 :                 m_bCachedTiledValid =
    1168        3727 :                     LoadTileData(tileIndices.data(), bEmptyTile);
    1169        3727 :                 if (!m_bCachedTiledValid)
    1170             :                 {
    1171           0 :                     return false;
    1172             :                 }
    1173        3727 :                 m_bCachedTiledEmpty = bEmptyTile;
    1174             :             }
    1175             : 
    1176        7896 :             pabySrcTile = m_abyDecodedTileData.empty()
    1177        2956 :                               ? m_abyRawTileData.data()
    1178         992 :                               : m_abyDecodedTileData.data();
    1179             :         }
    1180             :         const size_t nSrcDTSize =
    1181       25292 :             m_abyDecodedTileData.empty() ? nSourceSize : nDTSize;
    1182             : 
    1183       75998 :         for (size_t i = 0; i < nDims; ++i)
    1184             :         {
    1185       50706 :             countInnerLoopInit[i] = 1;
    1186       50706 :             if (arrayStep[i] != 0)
    1187             :             {
    1188             :                 const auto nextBlockIdx =
    1189       50004 :                     std::min((1 + indicesOuterLoop[i] / m_anBlockSize[i]) *
    1190       50004 :                                  m_anBlockSize[i],
    1191      100008 :                              arrayStartIdx[i] + count[i] * arrayStep[i]);
    1192       50004 :                 countInnerLoopInit[i] = static_cast<size_t>(
    1193       50004 :                     (nextBlockIdx - indicesOuterLoop[i] + arrayStep[i] - 1) /
    1194       50004 :                     arrayStep[i]);
    1195             :             }
    1196             :         }
    1197             : 
    1198       25292 :         if (bEmptyTile && bBothAreNumericDT && abyTargetNoData.empty())
    1199             :         {
    1200         602 :             abyTargetNoData.resize(nBufferDTSize);
    1201         602 :             if (m_pabyNoData)
    1202             :             {
    1203         156 :                 GDALExtendedDataType::CopyValue(
    1204         156 :                     m_pabyNoData, m_oType, &abyTargetNoData[0], bufferDataType);
    1205         156 :                 bNoDataIsZero = true;
    1206        1308 :                 for (size_t i = 0; i < abyTargetNoData.size(); ++i)
    1207             :                 {
    1208        1152 :                     if (abyTargetNoData[i] != 0)
    1209         334 :                         bNoDataIsZero = false;
    1210             :                 }
    1211             :             }
    1212             :             else
    1213             :             {
    1214         446 :                 bNoDataIsZero = true;
    1215         446 :                 GByte zero = 0;
    1216         446 :                 GDALCopyWords(&zero, GDT_Byte, 0, &abyTargetNoData[0],
    1217             :                               bufferDataType.GetNumericDataType(), 0, 1);
    1218             :             }
    1219             :         }
    1220             : 
    1221       24690 :     lbl_next_depth_inner_loop:
    1222      467583 :         if (nDims == 0 || dimIdxSubLoop == nDims - 1)
    1223             :         {
    1224      442167 :             indicesInnerLoop[dimIdxSubLoop] = indicesOuterLoop[dimIdxSubLoop];
    1225      442167 :             void *dst_ptr = dstPtrStackInnerLoop[dimIdxSubLoop];
    1226             : 
    1227      439680 :             if (m_bUseOptimizedCodePaths && bEmptyTile && bBothAreNumericDT &&
    1228      881847 :                 bNoDataIsZero &&
    1229        1134 :                 nBufferDTSize == dstBufferStrideBytes[dimIdxSubLoop])
    1230             :             {
    1231        1068 :                 memset(dst_ptr, 0,
    1232        1068 :                        nBufferDTSize * countInnerLoopInit[dimIdxSubLoop]);
    1233        1068 :                 goto end_inner_loop;
    1234             :             }
    1235      438612 :             else if (m_bUseOptimizedCodePaths && bEmptyTile &&
    1236      880086 :                      !abyTargetNoData.empty() && bBothAreNumericDT &&
    1237         375 :                      dstBufferStrideBytes[dimIdxSubLoop] <
    1238         375 :                          std::numeric_limits<int>::max())
    1239             :             {
    1240         750 :                 GDALCopyWords64(
    1241         375 :                     abyTargetNoData.data(), bufferDataType.GetNumericDataType(),
    1242             :                     0, dst_ptr, bufferDataType.GetNumericDataType(),
    1243         375 :                     static_cast<int>(dstBufferStrideBytes[dimIdxSubLoop]),
    1244         375 :                     static_cast<GPtrDiff_t>(countInnerLoopInit[dimIdxSubLoop]));
    1245         375 :                 goto end_inner_loop;
    1246             :             }
    1247      440724 :             else if (bEmptyTile)
    1248             :             {
    1249        3875 :                 for (size_t i = 0; i < countInnerLoopInit[dimIdxSubLoop];
    1250        2528 :                      ++i, dst_ptr = static_cast<uint8_t *>(dst_ptr) +
    1251        2528 :                                     dstBufferStrideBytes[dimIdxSubLoop])
    1252             :                 {
    1253        2528 :                     if (bNoDataIsZero)
    1254             :                     {
    1255        1930 :                         if (nBufferDTSize == 1)
    1256             :                         {
    1257          36 :                             *static_cast<uint8_t *>(dst_ptr) = 0;
    1258             :                         }
    1259        1894 :                         else if (nBufferDTSize == 2)
    1260             :                         {
    1261          48 :                             *static_cast<uint16_t *>(dst_ptr) = 0;
    1262             :                         }
    1263        1846 :                         else if (nBufferDTSize == 4)
    1264             :                         {
    1265          72 :                             *static_cast<uint32_t *>(dst_ptr) = 0;
    1266             :                         }
    1267        1774 :                         else if (nBufferDTSize == 8)
    1268             :                         {
    1269        1534 :                             *static_cast<uint64_t *>(dst_ptr) = 0;
    1270             :                         }
    1271         240 :                         else if (nBufferDTSize == 16)
    1272             :                         {
    1273         240 :                             static_cast<uint64_t *>(dst_ptr)[0] = 0;
    1274         240 :                             static_cast<uint64_t *>(dst_ptr)[1] = 0;
    1275             :                         }
    1276             :                         else
    1277             :                         {
    1278           0 :                             CPLAssert(false);
    1279             :                         }
    1280             :                     }
    1281         598 :                     else if (m_pabyNoData)
    1282             :                     {
    1283         597 :                         if (bBothAreNumericDT)
    1284             :                         {
    1285         588 :                             const void *src_ptr_v = abyTargetNoData.data();
    1286         588 :                             if (nBufferDTSize == 1)
    1287          24 :                                 *static_cast<uint8_t *>(dst_ptr) =
    1288          24 :                                     *static_cast<const uint8_t *>(src_ptr_v);
    1289         564 :                             else if (nBufferDTSize == 2)
    1290           0 :                                 *static_cast<uint16_t *>(dst_ptr) =
    1291           0 :                                     *static_cast<const uint16_t *>(src_ptr_v);
    1292         564 :                             else if (nBufferDTSize == 4)
    1293          60 :                                 *static_cast<uint32_t *>(dst_ptr) =
    1294          60 :                                     *static_cast<const uint32_t *>(src_ptr_v);
    1295         504 :                             else if (nBufferDTSize == 8)
    1296         504 :                                 *static_cast<uint64_t *>(dst_ptr) =
    1297         504 :                                     *static_cast<const uint64_t *>(src_ptr_v);
    1298           0 :                             else if (nBufferDTSize == 16)
    1299             :                             {
    1300           0 :                                 static_cast<uint64_t *>(dst_ptr)[0] =
    1301           0 :                                     static_cast<const uint64_t *>(src_ptr_v)[0];
    1302           0 :                                 static_cast<uint64_t *>(dst_ptr)[1] =
    1303           0 :                                     static_cast<const uint64_t *>(src_ptr_v)[1];
    1304             :                             }
    1305             :                             else
    1306             :                             {
    1307           0 :                                 CPLAssert(false);
    1308             :                             }
    1309             :                         }
    1310             :                         else
    1311             :                         {
    1312           9 :                             GDALExtendedDataType::CopyValue(
    1313           9 :                                 m_pabyNoData, m_oType, dst_ptr, bufferDataType);
    1314             :                         }
    1315             :                     }
    1316             :                     else
    1317             :                     {
    1318           1 :                         memset(dst_ptr, 0, nBufferDTSize);
    1319             :                     }
    1320             :                 }
    1321             : 
    1322        1347 :                 goto end_inner_loop;
    1323             :             }
    1324             : 
    1325      439377 :             size_t nOffset = 0;
    1326     1330560 :             for (size_t i = 0; i < nDims; i++)
    1327             :             {
    1328      891188 :                 nOffset = static_cast<size_t>(
    1329      891188 :                     nOffset * m_anBlockSize[i] +
    1330     1782380 :                     (indicesInnerLoop[i] - tileIndices[i] * m_anBlockSize[i]));
    1331             :             }
    1332      439377 :             const GByte *src_ptr = pabySrcTile + nOffset * nSrcDTSize;
    1333      439377 :             const auto step = nDims == 0 ? 0 : arrayStep[dimIdxSubLoop];
    1334             : 
    1335      438227 :             if (m_bUseOptimizedCodePaths && bBothAreNumericDT &&
    1336      438203 :                 step <= static_cast<GIntBig>(std::numeric_limits<int>::max() /
    1337      877604 :                                              nDTSize) &&
    1338      438203 :                 dstBufferStrideBytes[dimIdxSubLoop] <=
    1339      438203 :                     std::numeric_limits<int>::max())
    1340             :             {
    1341      438203 :                 GDALCopyWords64(
    1342             :                     src_ptr, m_oType.GetNumericDataType(),
    1343             :                     static_cast<int>(step * nDTSize), dst_ptr,
    1344             :                     bufferDataType.GetNumericDataType(),
    1345      438203 :                     static_cast<int>(dstBufferStrideBytes[dimIdxSubLoop]),
    1346      438203 :                     static_cast<GPtrDiff_t>(countInnerLoopInit[dimIdxSubLoop]));
    1347             : 
    1348      438203 :                 goto end_inner_loop;
    1349             :             }
    1350             : 
    1351        3391 :             for (size_t i = 0; i < countInnerLoopInit[dimIdxSubLoop];
    1352        2217 :                  ++i, src_ptr += step * nSrcDTSize,
    1353        2217 :                         dst_ptr = static_cast<uint8_t *>(dst_ptr) +
    1354        2217 :                                   dstBufferStrideBytes[dimIdxSubLoop])
    1355             :             {
    1356        2217 :                 if (bSameNumericDT)
    1357             :                 {
    1358         814 :                     const void *src_ptr_v = src_ptr;
    1359         814 :                     if (nSameDTSize == 1)
    1360          70 :                         *static_cast<uint8_t *>(dst_ptr) =
    1361          70 :                             *static_cast<const uint8_t *>(src_ptr_v);
    1362         744 :                     else if (nSameDTSize == 2)
    1363             :                     {
    1364          56 :                         *static_cast<uint16_t *>(dst_ptr) =
    1365          56 :                             *static_cast<const uint16_t *>(src_ptr_v);
    1366             :                     }
    1367         688 :                     else if (nSameDTSize == 4)
    1368             :                     {
    1369         154 :                         *static_cast<uint32_t *>(dst_ptr) =
    1370         154 :                             *static_cast<const uint32_t *>(src_ptr_v);
    1371             :                     }
    1372         534 :                     else if (nSameDTSize == 8)
    1373             :                     {
    1374         482 :                         *static_cast<uint64_t *>(dst_ptr) =
    1375         482 :                             *static_cast<const uint64_t *>(src_ptr_v);
    1376             :                     }
    1377          52 :                     else if (nSameDTSize == 16)
    1378             :                     {
    1379          52 :                         static_cast<uint64_t *>(dst_ptr)[0] =
    1380          52 :                             static_cast<const uint64_t *>(src_ptr_v)[0];
    1381          52 :                         static_cast<uint64_t *>(dst_ptr)[1] =
    1382          52 :                             static_cast<const uint64_t *>(src_ptr_v)[1];
    1383             :                     }
    1384             :                     else
    1385             :                     {
    1386           0 :                         CPLAssert(false);
    1387             :                     }
    1388             :                 }
    1389        1403 :                 else if (bSameCompoundAndNoDynamicMem)
    1390             :                 {
    1391           4 :                     memcpy(dst_ptr, src_ptr, nDTSize);
    1392             :                 }
    1393        1399 :                 else if (m_oType.GetClass() == GEDTC_STRING)
    1394             :                 {
    1395          26 :                     if (m_aoDtypeElts.back().nativeType ==
    1396             :                         DtypeElt::NativeType::STRING_UNICODE)
    1397             :                     {
    1398             :                         char *pDstStr =
    1399          12 :                             UCS4ToUTF8(src_ptr, nSourceSize,
    1400           6 :                                        m_aoDtypeElts.back().needByteSwapping);
    1401           6 :                         char **pDstPtr = static_cast<char **>(dst_ptr);
    1402           6 :                         memcpy(pDstPtr, &pDstStr, sizeof(pDstStr));
    1403             :                     }
    1404             :                     else
    1405             :                     {
    1406             :                         char *pDstStr =
    1407          20 :                             static_cast<char *>(CPLMalloc(nSourceSize + 1));
    1408          20 :                         memcpy(pDstStr, src_ptr, nSourceSize);
    1409          20 :                         pDstStr[nSourceSize] = 0;
    1410          20 :                         char **pDstPtr = static_cast<char **>(dst_ptr);
    1411          20 :                         memcpy(pDstPtr, &pDstStr, sizeof(char *));
    1412             :                     }
    1413             :                 }
    1414             :                 else
    1415             :                 {
    1416        1373 :                     GDALExtendedDataType::CopyValue(src_ptr, m_oType, dst_ptr,
    1417             :                                                     bufferDataType);
    1418             :                 }
    1419        1174 :             }
    1420             :         }
    1421             :         else
    1422             :         {
    1423             :             // This level of loop loops over individual samples, within a
    1424             :             // block
    1425       25416 :             indicesInnerLoop[dimIdxSubLoop] = indicesOuterLoop[dimIdxSubLoop];
    1426       25416 :             countInnerLoop[dimIdxSubLoop] = countInnerLoopInit[dimIdxSubLoop];
    1427             :             while (true)
    1428             :             {
    1429      442291 :                 dimIdxSubLoop++;
    1430      442291 :                 dstPtrStackInnerLoop[dimIdxSubLoop] =
    1431      442291 :                     dstPtrStackInnerLoop[dimIdxSubLoop - 1];
    1432      442291 :                 goto lbl_next_depth_inner_loop;
    1433      442291 :             lbl_return_to_caller_inner_loop:
    1434      442291 :                 dimIdxSubLoop--;
    1435      442291 :                 --countInnerLoop[dimIdxSubLoop];
    1436      442291 :                 if (countInnerLoop[dimIdxSubLoop] == 0)
    1437             :                 {
    1438       25416 :                     break;
    1439             :                 }
    1440      416875 :                 indicesInnerLoop[dimIdxSubLoop] += arrayStep[dimIdxSubLoop];
    1441      416875 :                 dstPtrStackInnerLoop[dimIdxSubLoop] +=
    1442      416875 :                     dstBufferStrideBytes[dimIdxSubLoop];
    1443             :             }
    1444             :         }
    1445      467583 :     end_inner_loop:
    1446      467583 :         if (dimIdxSubLoop > 0)
    1447      442291 :             goto lbl_return_to_caller_inner_loop;
    1448             :     }
    1449             :     else
    1450             :     {
    1451             :         // This level of loop loops over blocks
    1452        4506 :         indicesOuterLoop[dimIdx] = arrayStartIdx[dimIdx];
    1453        4506 :         tileIndices[dimIdx] = indicesOuterLoop[dimIdx] / m_anBlockSize[dimIdx];
    1454             :         while (true)
    1455             :         {
    1456       28229 :             dimIdx++;
    1457       28229 :             dstPtrStackOuterLoop[dimIdx] = dstPtrStackOuterLoop[dimIdx - 1];
    1458       28229 :             goto lbl_next_depth;
    1459       28229 :         lbl_return_to_caller:
    1460       28229 :             dimIdx--;
    1461       28229 :             if (count[dimIdx] == 1 || arrayStep[dimIdx] == 0)
    1462             :                 break;
    1463             : 
    1464             :             size_t nIncr;
    1465       27208 :             if (static_cast<GUInt64>(arrayStep[dimIdx]) < m_anBlockSize[dimIdx])
    1466             :             {
    1467             :                 // Compute index at next block boundary
    1468             :                 auto newIdx =
    1469       26800 :                     indicesOuterLoop[dimIdx] +
    1470       26800 :                     (m_anBlockSize[dimIdx] -
    1471       26800 :                      (indicesOuterLoop[dimIdx] % m_anBlockSize[dimIdx]));
    1472             :                 // And round up compared to arrayStartIdx, arrayStep
    1473       26800 :                 nIncr = static_cast<size_t>((newIdx - indicesOuterLoop[dimIdx] +
    1474       26800 :                                              arrayStep[dimIdx] - 1) /
    1475       26800 :                                             arrayStep[dimIdx]);
    1476             :             }
    1477             :             else
    1478             :             {
    1479         408 :                 nIncr = 1;
    1480             :             }
    1481       27208 :             indicesOuterLoop[dimIdx] += nIncr * arrayStep[dimIdx];
    1482       27208 :             if (indicesOuterLoop[dimIdx] >
    1483       27208 :                 arrayStartIdx[dimIdx] + (count[dimIdx] - 1) * arrayStep[dimIdx])
    1484        3485 :                 break;
    1485       23723 :             dstPtrStackOuterLoop[dimIdx] +=
    1486       23723 :                 bufferStride[dimIdx] *
    1487       23723 :                 static_cast<GPtrDiff_t>(nIncr * nBufferDTSize);
    1488       47446 :             tileIndices[dimIdx] =
    1489       23723 :                 indicesOuterLoop[dimIdx] / m_anBlockSize[dimIdx];
    1490       23723 :         }
    1491             :     }
    1492       29798 :     if (dimIdx > 0)
    1493       28229 :         goto lbl_return_to_caller;
    1494             : 
    1495        1569 :     return true;
    1496             : }
    1497             : 
    1498             : /************************************************************************/
    1499             : /*                           ZarrArray::IRead()                         */
    1500             : /************************************************************************/
    1501             : 
    1502         478 : bool ZarrArray::IWrite(const GUInt64 *arrayStartIdx, const size_t *count,
    1503             :                        const GInt64 *arrayStep, const GPtrDiff_t *bufferStride,
    1504             :                        const GDALExtendedDataType &bufferDataType,
    1505             :                        const void *pSrcBuffer)
    1506             : {
    1507         478 :     if (!CheckValidAndErrorOutIfNot())
    1508           0 :         return false;
    1509             : 
    1510         478 :     if (!AllocateWorkingBuffers())
    1511           0 :         return false;
    1512             : 
    1513         478 :     m_oMapTileIndexToCachedTile.clear();
    1514             : 
    1515             :     // Need to be kept in top-level scope
    1516         956 :     std::vector<GUInt64> arrayStartIdxMod;
    1517         956 :     std::vector<GInt64> arrayStepMod;
    1518         956 :     std::vector<GPtrDiff_t> bufferStrideMod;
    1519             : 
    1520         478 :     const size_t nDims = m_aoDims.size();
    1521         478 :     bool negativeStep = false;
    1522         478 :     bool bWriteWholeTileInit = true;
    1523        1296 :     for (size_t i = 0; i < nDims; ++i)
    1524             :     {
    1525         818 :         if (arrayStep[i] < 0)
    1526             :         {
    1527         132 :             negativeStep = true;
    1528         132 :             if (arrayStep[i] != -1 && count[i] > 1)
    1529           0 :                 bWriteWholeTileInit = false;
    1530             :         }
    1531         686 :         else if (arrayStep[i] != 1 && count[i] > 1)
    1532           0 :             bWriteWholeTileInit = false;
    1533             :     }
    1534             : 
    1535         478 :     const auto nBufferDTSize = static_cast<int>(bufferDataType.GetSize());
    1536             : 
    1537             :     // Make sure that arrayStep[i] are positive for sake of simplicity
    1538         478 :     if (negativeStep)
    1539             :     {
    1540             : #if defined(__GNUC__)
    1541             : #pragma GCC diagnostic push
    1542             : #pragma GCC diagnostic ignored "-Wnull-dereference"
    1543             : #endif
    1544          66 :         arrayStartIdxMod.resize(nDims);
    1545          66 :         arrayStepMod.resize(nDims);
    1546          66 :         bufferStrideMod.resize(nDims);
    1547             : #if defined(__GNUC__)
    1548             : #pragma GCC diagnostic pop
    1549             : #endif
    1550         198 :         for (size_t i = 0; i < nDims; ++i)
    1551             :         {
    1552         132 :             if (arrayStep[i] < 0)
    1553             :             {
    1554         264 :                 arrayStartIdxMod[i] =
    1555         132 :                     arrayStartIdx[i] - (count[i] - 1) * (-arrayStep[i]);
    1556         132 :                 arrayStepMod[i] = -arrayStep[i];
    1557         132 :                 bufferStrideMod[i] = -bufferStride[i];
    1558         132 :                 pSrcBuffer =
    1559             :                     static_cast<const GByte *>(pSrcBuffer) +
    1560         132 :                     bufferStride[i] *
    1561         132 :                         static_cast<GPtrDiff_t>(nBufferDTSize * (count[i] - 1));
    1562             :             }
    1563             :             else
    1564             :             {
    1565           0 :                 arrayStartIdxMod[i] = arrayStartIdx[i];
    1566           0 :                 arrayStepMod[i] = arrayStep[i];
    1567           0 :                 bufferStrideMod[i] = bufferStride[i];
    1568             :             }
    1569             :         }
    1570          66 :         arrayStartIdx = arrayStartIdxMod.data();
    1571          66 :         arrayStep = arrayStepMod.data();
    1572          66 :         bufferStride = bufferStrideMod.data();
    1573             :     }
    1574             : 
    1575         956 :     std::vector<uint64_t> indicesOuterLoop(nDims + 1);
    1576         956 :     std::vector<const GByte *> srcPtrStackOuterLoop(nDims + 1);
    1577             : 
    1578         956 :     std::vector<size_t> offsetDstBuffer(nDims + 1);
    1579         956 :     std::vector<const GByte *> srcPtrStackInnerLoop(nDims + 1);
    1580             : 
    1581         956 :     std::vector<GPtrDiff_t> srcBufferStrideBytes;
    1582        1296 :     for (size_t i = 0; i < nDims; ++i)
    1583             :     {
    1584         818 :         srcBufferStrideBytes.push_back(bufferStride[i] *
    1585         818 :                                        static_cast<GPtrDiff_t>(nBufferDTSize));
    1586             :     }
    1587         478 :     srcBufferStrideBytes.push_back(0);
    1588             : 
    1589         478 :     const auto nDTSize = m_oType.GetSize();
    1590             : 
    1591         956 :     std::vector<uint64_t> tileIndices(nDims);
    1592             :     const size_t nNativeSize =
    1593         478 :         m_aoDtypeElts.back().nativeOffset + m_aoDtypeElts.back().nativeSize;
    1594             : 
    1595         956 :     std::vector<size_t> countInnerLoopInit(nDims + 1, 1);
    1596         956 :     std::vector<size_t> countInnerLoop(nDims);
    1597             : 
    1598         952 :     const bool bBothAreNumericDT = m_oType.GetClass() == GEDTC_NUMERIC &&
    1599         474 :                                    bufferDataType.GetClass() == GEDTC_NUMERIC;
    1600             :     const bool bSameNumericDT =
    1601         952 :         bBothAreNumericDT &&
    1602         474 :         m_oType.GetNumericDataType() == bufferDataType.GetNumericDataType();
    1603         478 :     const auto nSameDTSize = bSameNumericDT ? m_oType.GetSize() : 0;
    1604             :     const bool bSameCompoundAndNoDynamicMem =
    1605         478 :         m_oType.GetClass() == GEDTC_COMPOUND && m_oType == bufferDataType &&
    1606           0 :         !m_oType.NeedsFreeDynamicMemory();
    1607             : 
    1608         478 :     size_t dimIdx = 0;
    1609         478 :     size_t dimIdxForCopy = nDims == 0 ? 0 : nDims - 1;
    1610         478 :     if (nDims)
    1611             :     {
    1612         495 :         while (dimIdxForCopy > 0 && count[dimIdxForCopy] == 1)
    1613          17 :             --dimIdxForCopy;
    1614             :     }
    1615             : 
    1616         478 :     srcPtrStackOuterLoop[0] = static_cast<const GByte *>(pSrcBuffer);
    1617       24556 : lbl_next_depth:
    1618       24556 :     if (dimIdx == nDims)
    1619             :     {
    1620       22804 :         bool bWriteWholeTile = bWriteWholeTileInit;
    1621       22804 :         bool bPartialTile = false;
    1622       68535 :         for (size_t i = 0; i < nDims; ++i)
    1623             :         {
    1624       45731 :             countInnerLoopInit[i] = 1;
    1625       45731 :             if (arrayStep[i] != 0)
    1626             :             {
    1627             :                 const auto nextBlockIdx =
    1628       45448 :                     std::min((1 + indicesOuterLoop[i] / m_anBlockSize[i]) *
    1629       45448 :                                  m_anBlockSize[i],
    1630       90896 :                              arrayStartIdx[i] + count[i] * arrayStep[i]);
    1631       45448 :                 countInnerLoopInit[i] = static_cast<size_t>(
    1632       45448 :                     (nextBlockIdx - indicesOuterLoop[i] + arrayStep[i] - 1) /
    1633       45448 :                     arrayStep[i]);
    1634             :             }
    1635       45731 :             if (bWriteWholeTile)
    1636             :             {
    1637             :                 const bool bWholePartialTileThisDim =
    1638       46547 :                     indicesOuterLoop[i] == 0 &&
    1639        1763 :                     countInnerLoopInit[i] == m_aoDims[i]->GetSize();
    1640       44784 :                 bWriteWholeTile = (countInnerLoopInit[i] == m_anBlockSize[i] ||
    1641             :                                    bWholePartialTileThisDim);
    1642       44784 :                 if (bWholePartialTileThisDim)
    1643             :                 {
    1644         517 :                     bPartialTile = true;
    1645             :                 }
    1646             :             }
    1647             :         }
    1648             : 
    1649       22804 :         size_t dimIdxSubLoop = 0;
    1650       22804 :         srcPtrStackInnerLoop[0] = srcPtrStackOuterLoop[nDims];
    1651             :         const size_t nCacheDTSize =
    1652       22804 :             m_abyDecodedTileData.empty() ? nNativeSize : nDTSize;
    1653       22804 :         auto &abyTile = m_abyDecodedTileData.empty() ? m_abyRawTileData
    1654       22804 :                                                      : m_abyDecodedTileData;
    1655             : 
    1656       22804 :         if (!tileIndices.empty() && tileIndices == m_anCachedTiledIndices)
    1657             :         {
    1658           3 :             if (!m_bCachedTiledValid)
    1659           0 :                 return false;
    1660             :         }
    1661             :         else
    1662             :         {
    1663       22801 :             if (!FlushDirtyTile())
    1664           0 :                 return false;
    1665             : 
    1666       22801 :             m_anCachedTiledIndices = tileIndices;
    1667       22801 :             m_bCachedTiledValid = true;
    1668             : 
    1669       22801 :             if (bWriteWholeTile)
    1670             :             {
    1671       21278 :                 if (bPartialTile)
    1672             :                 {
    1673         274 :                     DeallocateDecodedTileData();
    1674         274 :                     memset(&abyTile[0], 0, abyTile.size());
    1675             :                 }
    1676             :             }
    1677             :             else
    1678             :             {
    1679             :                 // If we don't write the whole tile, we need to fetch a
    1680             :                 // potentially existing one.
    1681        1523 :                 bool bEmptyTile = false;
    1682        1523 :                 m_bCachedTiledValid =
    1683        1523 :                     LoadTileData(tileIndices.data(), bEmptyTile);
    1684        1523 :                 if (!m_bCachedTiledValid)
    1685             :                 {
    1686           0 :                     return false;
    1687             :                 }
    1688             : 
    1689        1523 :                 if (bEmptyTile)
    1690             :                 {
    1691        1215 :                     DeallocateDecodedTileData();
    1692             : 
    1693        1215 :                     if (m_pabyNoData == nullptr)
    1694             :                     {
    1695         495 :                         memset(&abyTile[0], 0, abyTile.size());
    1696             :                     }
    1697             :                     else
    1698             :                     {
    1699         720 :                         const size_t nElts = abyTile.size() / nCacheDTSize;
    1700         720 :                         GByte *dstPtr = &abyTile[0];
    1701         720 :                         if (m_oType.GetClass() == GEDTC_NUMERIC)
    1702             :                         {
    1703         720 :                             GDALCopyWords64(
    1704         720 :                                 m_pabyNoData, m_oType.GetNumericDataType(), 0,
    1705             :                                 dstPtr, m_oType.GetNumericDataType(),
    1706         720 :                                 static_cast<int>(m_oType.GetSize()),
    1707             :                                 static_cast<GPtrDiff_t>(nElts));
    1708             :                         }
    1709             :                         else
    1710             :                         {
    1711           0 :                             for (size_t i = 0; i < nElts; ++i)
    1712             :                             {
    1713           0 :                                 GDALExtendedDataType::CopyValue(
    1714           0 :                                     m_pabyNoData, m_oType, dstPtr, m_oType);
    1715           0 :                                 dstPtr += nCacheDTSize;
    1716             :                             }
    1717             :                         }
    1718             :                     }
    1719             :                 }
    1720             :             }
    1721             :         }
    1722       22804 :         m_bDirtyTile = true;
    1723       22804 :         m_bCachedTiledEmpty = false;
    1724       22804 :         if (nDims)
    1725       22804 :             offsetDstBuffer[0] = static_cast<size_t>(
    1726       22804 :                 indicesOuterLoop[0] - tileIndices[0] * m_anBlockSize[0]);
    1727             : 
    1728       22804 :         GByte *pabyTile = &abyTile[0];
    1729             : 
    1730      453210 :     lbl_next_depth_inner_loop:
    1731      453210 :         if (dimIdxSubLoop == dimIdxForCopy)
    1732             :         {
    1733      430457 :             size_t nOffset = offsetDstBuffer[dimIdxSubLoop];
    1734      430457 :             GInt64 step = nDims == 0 ? 0 : arrayStep[dimIdxSubLoop];
    1735      430631 :             for (size_t i = dimIdxSubLoop + 1; i < nDims; ++i)
    1736             :             {
    1737         174 :                 nOffset = static_cast<size_t>(
    1738         174 :                     nOffset * m_anBlockSize[i] +
    1739         348 :                     (indicesOuterLoop[i] - tileIndices[i] * m_anBlockSize[i]));
    1740         174 :                 step *= m_anBlockSize[i];
    1741             :             }
    1742      430457 :             const void *src_ptr = srcPtrStackInnerLoop[dimIdxSubLoop];
    1743      430457 :             GByte *dst_ptr = pabyTile + nOffset * nCacheDTSize;
    1744             : 
    1745      430457 :             if (m_bUseOptimizedCodePaths && bBothAreNumericDT)
    1746             :             {
    1747      429727 :                 if (countInnerLoopInit[dimIdxSubLoop] == 1 && bSameNumericDT)
    1748             :                 {
    1749          84 :                     void *dst_ptr_v = dst_ptr;
    1750          84 :                     if (nSameDTSize == 1)
    1751           6 :                         *static_cast<uint8_t *>(dst_ptr_v) =
    1752           6 :                             *static_cast<const uint8_t *>(src_ptr);
    1753          78 :                     else if (nSameDTSize == 2)
    1754             :                     {
    1755           2 :                         *static_cast<uint16_t *>(dst_ptr_v) =
    1756           2 :                             *static_cast<const uint16_t *>(src_ptr);
    1757             :                     }
    1758          76 :                     else if (nSameDTSize == 4)
    1759             :                     {
    1760           2 :                         *static_cast<uint32_t *>(dst_ptr_v) =
    1761           2 :                             *static_cast<const uint32_t *>(src_ptr);
    1762             :                     }
    1763          74 :                     else if (nSameDTSize == 8)
    1764             :                     {
    1765          52 :                         *static_cast<uint64_t *>(dst_ptr_v) =
    1766          52 :                             *static_cast<const uint64_t *>(src_ptr);
    1767             :                     }
    1768          22 :                     else if (nSameDTSize == 16)
    1769             :                     {
    1770          22 :                         static_cast<uint64_t *>(dst_ptr_v)[0] =
    1771          22 :                             static_cast<const uint64_t *>(src_ptr)[0];
    1772          22 :                         static_cast<uint64_t *>(dst_ptr_v)[1] =
    1773          22 :                             static_cast<const uint64_t *>(src_ptr)[1];
    1774             :                     }
    1775             :                     else
    1776             :                     {
    1777           0 :                         CPLAssert(false);
    1778             :                     }
    1779             :                 }
    1780      429643 :                 else if (step <=
    1781      429643 :                              static_cast<GIntBig>(
    1782      859286 :                                  std::numeric_limits<int>::max() / nDTSize) &&
    1783      429643 :                          srcBufferStrideBytes[dimIdxSubLoop] <=
    1784      429643 :                              std::numeric_limits<int>::max())
    1785             :                 {
    1786      859286 :                     GDALCopyWords64(
    1787             :                         src_ptr, bufferDataType.GetNumericDataType(),
    1788      429643 :                         static_cast<int>(srcBufferStrideBytes[dimIdxSubLoop]),
    1789             :                         dst_ptr, m_oType.GetNumericDataType(),
    1790             :                         static_cast<int>(step * nDTSize),
    1791             :                         static_cast<GPtrDiff_t>(
    1792      429643 :                             countInnerLoopInit[dimIdxSubLoop]));
    1793             :                 }
    1794      429727 :                 goto end_inner_loop;
    1795             :             }
    1796             : 
    1797        2188 :             for (size_t i = 0; i < countInnerLoopInit[dimIdxSubLoop];
    1798        1458 :                  ++i, dst_ptr += step * nCacheDTSize,
    1799        1458 :                         src_ptr = static_cast<const uint8_t *>(src_ptr) +
    1800        1458 :                                   srcBufferStrideBytes[dimIdxSubLoop])
    1801             :             {
    1802        1458 :                 if (bSameNumericDT)
    1803             :                 {
    1804         300 :                     void *dst_ptr_v = dst_ptr;
    1805         300 :                     if (nSameDTSize == 1)
    1806           0 :                         *static_cast<uint8_t *>(dst_ptr_v) =
    1807           0 :                             *static_cast<const uint8_t *>(src_ptr);
    1808         300 :                     else if (nSameDTSize == 2)
    1809             :                     {
    1810           0 :                         *static_cast<uint16_t *>(dst_ptr_v) =
    1811           0 :                             *static_cast<const uint16_t *>(src_ptr);
    1812             :                     }
    1813         300 :                     else if (nSameDTSize == 4)
    1814             :                     {
    1815           0 :                         *static_cast<uint32_t *>(dst_ptr_v) =
    1816           0 :                             *static_cast<const uint32_t *>(src_ptr);
    1817             :                     }
    1818         300 :                     else if (nSameDTSize == 8)
    1819             :                     {
    1820         220 :                         *static_cast<uint64_t *>(dst_ptr_v) =
    1821         220 :                             *static_cast<const uint64_t *>(src_ptr);
    1822             :                     }
    1823          80 :                     else if (nSameDTSize == 16)
    1824             :                     {
    1825          80 :                         static_cast<uint64_t *>(dst_ptr_v)[0] =
    1826          80 :                             static_cast<const uint64_t *>(src_ptr)[0];
    1827          80 :                         static_cast<uint64_t *>(dst_ptr_v)[1] =
    1828          80 :                             static_cast<const uint64_t *>(src_ptr)[1];
    1829             :                     }
    1830             :                     else
    1831             :                     {
    1832           0 :                         CPLAssert(false);
    1833             :                     }
    1834             :                 }
    1835        1158 :                 else if (bSameCompoundAndNoDynamicMem)
    1836             :                 {
    1837           0 :                     memcpy(dst_ptr, src_ptr, nDTSize);
    1838             :                 }
    1839        1158 :                 else if (m_oType.GetClass() == GEDTC_STRING)
    1840             :                 {
    1841           6 :                     const char *pSrcStr =
    1842             :                         *static_cast<const char *const *>(src_ptr);
    1843           6 :                     if (pSrcStr)
    1844             :                     {
    1845           6 :                         const size_t nLen = strlen(pSrcStr);
    1846           6 :                         if (m_aoDtypeElts.back().nativeType ==
    1847             :                             DtypeElt::NativeType::STRING_UNICODE)
    1848             :                         {
    1849             :                             try
    1850             :                             {
    1851             :                                 const auto ucs4 = UTF8ToUCS4(
    1852             :                                     pSrcStr,
    1853           8 :                                     m_aoDtypeElts.back().needByteSwapping);
    1854           4 :                                 const auto ucs4Len = ucs4.size();
    1855           4 :                                 memcpy(dst_ptr, ucs4.data(),
    1856           4 :                                        std::min(ucs4Len, nNativeSize));
    1857           4 :                                 if (ucs4Len > nNativeSize)
    1858             :                                 {
    1859           1 :                                     CPLError(CE_Warning, CPLE_AppDefined,
    1860             :                                              "Too long string truncated");
    1861             :                                 }
    1862           3 :                                 else if (ucs4Len < nNativeSize)
    1863             :                                 {
    1864           1 :                                     memset(dst_ptr + ucs4Len, 0,
    1865           1 :                                            nNativeSize - ucs4Len);
    1866             :                                 }
    1867             :                             }
    1868           0 :                             catch (const std::exception &)
    1869             :                             {
    1870           0 :                                 memset(dst_ptr, 0, nNativeSize);
    1871             :                             }
    1872             :                         }
    1873             :                         else
    1874             :                         {
    1875           2 :                             memcpy(dst_ptr, pSrcStr,
    1876           2 :                                    std::min(nLen, nNativeSize));
    1877           2 :                             if (nLen < nNativeSize)
    1878           1 :                                 memset(dst_ptr + nLen, 0, nNativeSize - nLen);
    1879             :                         }
    1880             :                     }
    1881             :                     else
    1882             :                     {
    1883           0 :                         memset(dst_ptr, 0, nNativeSize);
    1884             :                     }
    1885             :                 }
    1886             :                 else
    1887             :                 {
    1888        1152 :                     if (m_oType.NeedsFreeDynamicMemory())
    1889           0 :                         m_oType.FreeDynamicMemory(dst_ptr);
    1890        1152 :                     GDALExtendedDataType::CopyValue(src_ptr, bufferDataType,
    1891        1152 :                                                     dst_ptr, m_oType);
    1892             :                 }
    1893             :             }
    1894             :         }
    1895             :         else
    1896             :         {
    1897             :             // This level of loop loops over individual samples, within a
    1898             :             // block
    1899       22753 :             countInnerLoop[dimIdxSubLoop] = countInnerLoopInit[dimIdxSubLoop];
    1900             :             while (true)
    1901             :             {
    1902      430406 :                 dimIdxSubLoop++;
    1903      430406 :                 srcPtrStackInnerLoop[dimIdxSubLoop] =
    1904      430406 :                     srcPtrStackInnerLoop[dimIdxSubLoop - 1];
    1905      860812 :                 offsetDstBuffer[dimIdxSubLoop] =
    1906     1291220 :                     static_cast<size_t>(offsetDstBuffer[dimIdxSubLoop - 1] *
    1907      430406 :                                             m_anBlockSize[dimIdxSubLoop] +
    1908      430406 :                                         (indicesOuterLoop[dimIdxSubLoop] -
    1909      860812 :                                          tileIndices[dimIdxSubLoop] *
    1910      430406 :                                              m_anBlockSize[dimIdxSubLoop]));
    1911      430406 :                 goto lbl_next_depth_inner_loop;
    1912      430406 :             lbl_return_to_caller_inner_loop:
    1913      430406 :                 dimIdxSubLoop--;
    1914      430406 :                 --countInnerLoop[dimIdxSubLoop];
    1915      430406 :                 if (countInnerLoop[dimIdxSubLoop] == 0)
    1916             :                 {
    1917       22753 :                     break;
    1918             :                 }
    1919      407653 :                 srcPtrStackInnerLoop[dimIdxSubLoop] +=
    1920      407653 :                     srcBufferStrideBytes[dimIdxSubLoop];
    1921      407653 :                 offsetDstBuffer[dimIdxSubLoop] +=
    1922      407653 :                     static_cast<size_t>(arrayStep[dimIdxSubLoop]);
    1923             :             }
    1924             :         }
    1925      430457 :     end_inner_loop:
    1926      453210 :         if (dimIdxSubLoop > 0)
    1927      430406 :             goto lbl_return_to_caller_inner_loop;
    1928             :     }
    1929             :     else
    1930             :     {
    1931             :         // This level of loop loops over blocks
    1932        1752 :         indicesOuterLoop[dimIdx] = arrayStartIdx[dimIdx];
    1933        1752 :         tileIndices[dimIdx] = indicesOuterLoop[dimIdx] / m_anBlockSize[dimIdx];
    1934             :         while (true)
    1935             :         {
    1936       24078 :             dimIdx++;
    1937       24078 :             srcPtrStackOuterLoop[dimIdx] = srcPtrStackOuterLoop[dimIdx - 1];
    1938       24078 :             goto lbl_next_depth;
    1939       24078 :         lbl_return_to_caller:
    1940       24078 :             dimIdx--;
    1941       24078 :             if (count[dimIdx] == 1 || arrayStep[dimIdx] == 0)
    1942             :                 break;
    1943             : 
    1944             :             size_t nIncr;
    1945       23878 :             if (static_cast<GUInt64>(arrayStep[dimIdx]) < m_anBlockSize[dimIdx])
    1946             :             {
    1947             :                 // Compute index at next block boundary
    1948             :                 auto newIdx =
    1949       23686 :                     indicesOuterLoop[dimIdx] +
    1950       23686 :                     (m_anBlockSize[dimIdx] -
    1951       23686 :                      (indicesOuterLoop[dimIdx] % m_anBlockSize[dimIdx]));
    1952             :                 // And round up compared to arrayStartIdx, arrayStep
    1953       23686 :                 nIncr = static_cast<size_t>((newIdx - indicesOuterLoop[dimIdx] +
    1954       23686 :                                              arrayStep[dimIdx] - 1) /
    1955       23686 :                                             arrayStep[dimIdx]);
    1956             :             }
    1957             :             else
    1958             :             {
    1959         192 :                 nIncr = 1;
    1960             :             }
    1961       23878 :             indicesOuterLoop[dimIdx] += nIncr * arrayStep[dimIdx];
    1962       23878 :             if (indicesOuterLoop[dimIdx] >
    1963       23878 :                 arrayStartIdx[dimIdx] + (count[dimIdx] - 1) * arrayStep[dimIdx])
    1964        1552 :                 break;
    1965       22326 :             srcPtrStackOuterLoop[dimIdx] +=
    1966       22326 :                 bufferStride[dimIdx] *
    1967       22326 :                 static_cast<GPtrDiff_t>(nIncr * nBufferDTSize);
    1968       44652 :             tileIndices[dimIdx] =
    1969       22326 :                 indicesOuterLoop[dimIdx] / m_anBlockSize[dimIdx];
    1970       22326 :         }
    1971             :     }
    1972       24556 :     if (dimIdx > 0)
    1973       24078 :         goto lbl_return_to_caller;
    1974             : 
    1975         478 :     return true;
    1976             : }
    1977             : 
    1978             : /************************************************************************/
    1979             : /*                   ZarrArray::IsEmptyTile()                           */
    1980             : /************************************************************************/
    1981             : 
    1982       22804 : bool ZarrArray::IsEmptyTile(const ZarrByteVectorQuickResize &abyTile) const
    1983             : {
    1984       44441 :     if (m_pabyNoData == nullptr || (m_oType.GetClass() == GEDTC_NUMERIC &&
    1985       21637 :                                     GetNoDataValueAsDouble() == 0.0))
    1986             :     {
    1987       22496 :         const size_t nBytes = abyTile.size();
    1988       22496 :         size_t i = 0;
    1989       25745 :         for (; i + (sizeof(size_t) - 1) < nBytes; i += sizeof(size_t))
    1990             :         {
    1991       25102 :             if (*reinterpret_cast<const size_t *>(abyTile.data() + i) != 0)
    1992             :             {
    1993       21853 :                 return false;
    1994             :             }
    1995             :         }
    1996        1572 :         for (; i < nBytes; ++i)
    1997             :         {
    1998         990 :             if (abyTile[i] != 0)
    1999             :             {
    2000          61 :                 return false;
    2001             :             }
    2002             :         }
    2003         582 :         return true;
    2004             :     }
    2005         616 :     else if (m_oType.GetClass() == GEDTC_NUMERIC &&
    2006         308 :              !GDALDataTypeIsComplex(m_oType.GetNumericDataType()))
    2007             :     {
    2008         308 :         const int nDTSize = static_cast<int>(m_oType.GetSize());
    2009         308 :         const size_t nElts = abyTile.size() / nDTSize;
    2010         308 :         const auto eDT = m_oType.GetNumericDataType();
    2011         308 :         return GDALBufferHasOnlyNoData(abyTile.data(), GetNoDataValueAsDouble(),
    2012             :                                        nElts,        // nWidth
    2013             :                                        1,            // nHeight
    2014             :                                        nElts,        // nLineStride
    2015             :                                        1,            // nComponents
    2016             :                                        nDTSize * 8,  // nBitsPerSample
    2017         308 :                                        GDALDataTypeIsInteger(eDT)
    2018         112 :                                            ? (GDALDataTypeIsSigned(eDT)
    2019         112 :                                                   ? GSF_SIGNED_INT
    2020             :                                                   : GSF_UNSIGNED_INT)
    2021         308 :                                            : GSF_FLOATING_POINT);
    2022             :     }
    2023           0 :     return false;
    2024             : }
    2025             : 
    2026             : /************************************************************************/
    2027             : /*                  ZarrArray::OpenTilePresenceCache()                  */
    2028             : /************************************************************************/
    2029             : 
    2030             : std::shared_ptr<GDALMDArray>
    2031       26598 : ZarrArray::OpenTilePresenceCache(bool bCanCreate) const
    2032             : {
    2033       26598 :     if (m_bHasTriedCacheTilePresenceArray)
    2034       26222 :         return m_poCacheTilePresenceArray;
    2035         376 :     m_bHasTriedCacheTilePresenceArray = true;
    2036             : 
    2037         376 :     if (m_nTotalTileCount == 1)
    2038         170 :         return nullptr;
    2039             : 
    2040         412 :     std::string osCacheFilename;
    2041         412 :     auto poRGCache = GetCacheRootGroup(bCanCreate, osCacheFilename);
    2042         206 :     if (!poRGCache)
    2043         196 :         return nullptr;
    2044             : 
    2045          10 :     const std::string osTilePresenceArrayName(MassageName(GetFullName()) +
    2046          20 :                                               "_tile_presence");
    2047          20 :     auto poTilePresenceArray = poRGCache->OpenMDArray(osTilePresenceArrayName);
    2048          20 :     const auto eByteDT = GDALExtendedDataType::Create(GDT_Byte);
    2049          10 :     if (poTilePresenceArray)
    2050             :     {
    2051           8 :         bool ok = true;
    2052           8 :         const auto &apoDimsCache = poTilePresenceArray->GetDimensions();
    2053          16 :         if (poTilePresenceArray->GetDataType() != eByteDT ||
    2054           8 :             apoDimsCache.size() != m_aoDims.size())
    2055             :         {
    2056           0 :             ok = false;
    2057             :         }
    2058             :         else
    2059             :         {
    2060          24 :             for (size_t i = 0; i < m_aoDims.size(); i++)
    2061             :             {
    2062             :                 const auto nExpectedDimSize =
    2063          16 :                     (m_aoDims[i]->GetSize() + m_anBlockSize[i] - 1) /
    2064          16 :                     m_anBlockSize[i];
    2065          16 :                 if (apoDimsCache[i]->GetSize() != nExpectedDimSize)
    2066             :                 {
    2067           0 :                     ok = false;
    2068           0 :                     break;
    2069             :                 }
    2070             :             }
    2071             :         }
    2072           8 :         if (!ok)
    2073             :         {
    2074           0 :             CPLError(CE_Failure, CPLE_NotSupported,
    2075             :                      "Array %s in %s has not expected characteristics",
    2076             :                      osTilePresenceArrayName.c_str(), osCacheFilename.c_str());
    2077           0 :             return nullptr;
    2078             :         }
    2079             : 
    2080           8 :         if (!poTilePresenceArray->GetAttribute("filling_status") && !bCanCreate)
    2081             :         {
    2082           0 :             CPLDebug(ZARR_DEBUG_KEY,
    2083             :                      "Cache tile presence array for %s found, but filling not "
    2084             :                      "finished",
    2085           0 :                      GetFullName().c_str());
    2086           0 :             return nullptr;
    2087             :         }
    2088             : 
    2089           8 :         CPLDebug(ZARR_DEBUG_KEY, "Using cache tile presence for %s",
    2090           8 :                  GetFullName().c_str());
    2091             :     }
    2092           2 :     else if (bCanCreate)
    2093             :     {
    2094           2 :         int idxDim = 0;
    2095           2 :         std::string osBlockSize;
    2096           2 :         std::vector<std::shared_ptr<GDALDimension>> apoNewDims;
    2097           6 :         for (const auto &poDim : m_aoDims)
    2098             :         {
    2099           4 :             auto poNewDim = poRGCache->CreateDimension(
    2100           8 :                 osTilePresenceArrayName + '_' + std::to_string(idxDim),
    2101           8 :                 std::string(), std::string(),
    2102           4 :                 (poDim->GetSize() + m_anBlockSize[idxDim] - 1) /
    2103          12 :                     m_anBlockSize[idxDim]);
    2104           4 :             if (!poNewDim)
    2105           0 :                 return nullptr;
    2106           4 :             apoNewDims.emplace_back(poNewDim);
    2107             : 
    2108           4 :             if (!osBlockSize.empty())
    2109           2 :                 osBlockSize += ',';
    2110           4 :             constexpr GUInt64 BLOCKSIZE = 256;
    2111             :             osBlockSize +=
    2112           4 :                 std::to_string(std::min(poNewDim->GetSize(), BLOCKSIZE));
    2113             : 
    2114           4 :             idxDim++;
    2115             :         }
    2116             : 
    2117           2 :         CPLStringList aosOptionsTilePresence;
    2118           2 :         aosOptionsTilePresence.SetNameValue("BLOCKSIZE", osBlockSize.c_str());
    2119             :         poTilePresenceArray =
    2120           6 :             poRGCache->CreateMDArray(osTilePresenceArrayName, apoNewDims,
    2121           4 :                                      eByteDT, aosOptionsTilePresence.List());
    2122           2 :         if (!poTilePresenceArray)
    2123             :         {
    2124           0 :             CPLError(CE_Failure, CPLE_NotSupported, "Cannot create %s in %s",
    2125             :                      osTilePresenceArrayName.c_str(), osCacheFilename.c_str());
    2126           0 :             return nullptr;
    2127             :         }
    2128           2 :         poTilePresenceArray->SetNoDataValue(0);
    2129             :     }
    2130             :     else
    2131             :     {
    2132           0 :         return nullptr;
    2133             :     }
    2134             : 
    2135          10 :     m_poCacheTilePresenceArray = poTilePresenceArray;
    2136             : 
    2137          10 :     return poTilePresenceArray;
    2138             : }
    2139             : 
    2140             : /************************************************************************/
    2141             : /*                    ZarrArray::CacheTilePresence()                    */
    2142             : /************************************************************************/
    2143             : 
    2144           4 : bool ZarrArray::CacheTilePresence()
    2145             : {
    2146           4 :     if (m_nTotalTileCount == 1)
    2147           0 :         return true;
    2148             : 
    2149           8 :     const std::string osDirectoryName = GetDataDirectory();
    2150             : 
    2151             :     struct DirCloser
    2152             :     {
    2153             :         DirCloser(const DirCloser &) = delete;
    2154             :         DirCloser &operator=(const DirCloser &) = delete;
    2155             : 
    2156             :         VSIDIR *m_psDir;
    2157             : 
    2158           4 :         explicit DirCloser(VSIDIR *psDir) : m_psDir(psDir)
    2159             :         {
    2160           4 :         }
    2161             : 
    2162           4 :         ~DirCloser()
    2163           4 :         {
    2164           4 :             VSICloseDir(m_psDir);
    2165           4 :         }
    2166             :     };
    2167             : 
    2168           4 :     auto psDir = VSIOpenDir(osDirectoryName.c_str(), -1, nullptr);
    2169           4 :     if (!psDir)
    2170           0 :         return false;
    2171           8 :     DirCloser dirCloser(psDir);
    2172             : 
    2173           8 :     auto poTilePresenceArray = OpenTilePresenceCache(true);
    2174           4 :     if (!poTilePresenceArray)
    2175             :     {
    2176           0 :         return false;
    2177             :     }
    2178             : 
    2179           4 :     if (poTilePresenceArray->GetAttribute("filling_status"))
    2180             :     {
    2181           2 :         CPLDebug(ZARR_DEBUG_KEY,
    2182             :                  "CacheTilePresence(): %s already filled. Nothing to do",
    2183           2 :                  poTilePresenceArray->GetName().c_str());
    2184           2 :         return true;
    2185             :     }
    2186             : 
    2187           4 :     std::vector<GUInt64> anTileIdx(m_aoDims.size());
    2188           4 :     const std::vector<size_t> anCount(m_aoDims.size(), 1);
    2189           4 :     const std::vector<GInt64> anArrayStep(m_aoDims.size(), 0);
    2190           4 :     const std::vector<GPtrDiff_t> anBufferStride(m_aoDims.size(), 0);
    2191           2 :     const auto &apoDimsCache = poTilePresenceArray->GetDimensions();
    2192           4 :     const auto eByteDT = GDALExtendedDataType::Create(GDT_Byte);
    2193             : 
    2194           2 :     CPLDebug(ZARR_DEBUG_KEY,
    2195             :              "CacheTilePresence(): Iterating over %s to find which tiles are "
    2196             :              "present...",
    2197             :              osDirectoryName.c_str());
    2198           2 :     uint64_t nCounter = 0;
    2199             :     const char chSrcFilenameDirSeparator =
    2200           2 :         VSIGetDirectorySeparator(osDirectoryName.c_str())[0];
    2201          14 :     while (const VSIDIREntry *psEntry = VSIGetNextDirEntry(psDir))
    2202             :     {
    2203          12 :         if (!VSI_ISDIR(psEntry->nMode))
    2204             :         {
    2205             :             const CPLStringList aosTokens = GetTileIndicesFromFilename(
    2206           0 :                 CPLString(psEntry->pszName)
    2207           9 :                     .replaceAll(chSrcFilenameDirSeparator, '/')
    2208          18 :                     .c_str());
    2209           9 :             if (aosTokens.size() == static_cast<int>(m_aoDims.size()))
    2210             :             {
    2211             :                 // Get tile indices from filename
    2212           5 :                 bool unexpectedIndex = false;
    2213          15 :                 for (int i = 0; i < aosTokens.size(); ++i)
    2214             :                 {
    2215          10 :                     if (CPLGetValueType(aosTokens[i]) != CPL_VALUE_INTEGER)
    2216             :                     {
    2217           2 :                         unexpectedIndex = true;
    2218             :                     }
    2219          20 :                     anTileIdx[i] =
    2220          10 :                         static_cast<GUInt64>(CPLAtoGIntBig(aosTokens[i]));
    2221          10 :                     if (anTileIdx[i] >= apoDimsCache[i]->GetSize())
    2222             :                     {
    2223           0 :                         unexpectedIndex = true;
    2224             :                     }
    2225             :                 }
    2226           5 :                 if (unexpectedIndex)
    2227             :                 {
    2228           1 :                     continue;
    2229             :                 }
    2230             : 
    2231           4 :                 nCounter++;
    2232           4 :                 if ((nCounter % 1000) == 0)
    2233             :                 {
    2234           0 :                     CPLDebug(ZARR_DEBUG_KEY,
    2235             :                              "CacheTilePresence(): Listing in progress "
    2236             :                              "(last examined %s, at least %.02f %% completed)",
    2237           0 :                              psEntry->pszName,
    2238           0 :                              100.0 * double(nCounter) /
    2239           0 :                                  double(m_nTotalTileCount));
    2240             :                 }
    2241           4 :                 constexpr GByte byOne = 1;
    2242             :                 // CPLDebugOnly(ZARR_DEBUG_KEY, "Marking %s has present",
    2243             :                 // psEntry->pszName);
    2244           8 :                 if (!poTilePresenceArray->Write(
    2245           4 :                         anTileIdx.data(), anCount.data(), anArrayStep.data(),
    2246             :                         anBufferStride.data(), eByteDT, &byOne))
    2247             :                 {
    2248           0 :                     return false;
    2249             :                 }
    2250             :             }
    2251             :         }
    2252          12 :     }
    2253           2 :     CPLDebug(ZARR_DEBUG_KEY, "CacheTilePresence(): finished");
    2254             : 
    2255             :     // Write filling_status attribute
    2256           2 :     auto poAttr = poTilePresenceArray->CreateAttribute(
    2257           4 :         "filling_status", {}, GDALExtendedDataType::CreateString(), nullptr);
    2258           2 :     if (poAttr)
    2259             :     {
    2260           2 :         if (nCounter == 0)
    2261           0 :             poAttr->Write("no_tile_present");
    2262           2 :         else if (nCounter == m_nTotalTileCount)
    2263           0 :             poAttr->Write("all_tiles_present");
    2264             :         else
    2265           2 :             poAttr->Write("some_tiles_missing");
    2266             :     }
    2267             : 
    2268             :     // Force closing
    2269           2 :     m_poCacheTilePresenceArray = nullptr;
    2270           2 :     m_bHasTriedCacheTilePresenceArray = false;
    2271             : 
    2272           2 :     return true;
    2273             : }
    2274             : 
    2275             : /************************************************************************/
    2276             : /*                      ZarrArray::CreateAttribute()                    */
    2277             : /************************************************************************/
    2278             : 
    2279         108 : std::shared_ptr<GDALAttribute> ZarrArray::CreateAttribute(
    2280             :     const std::string &osName, const std::vector<GUInt64> &anDimensions,
    2281             :     const GDALExtendedDataType &oDataType, CSLConstList papszOptions)
    2282             : {
    2283         108 :     if (!CheckValidAndErrorOutIfNot())
    2284           0 :         return nullptr;
    2285             : 
    2286         108 :     if (!m_bUpdatable)
    2287             :     {
    2288           2 :         CPLError(CE_Failure, CPLE_NotSupported,
    2289             :                  "Dataset not open in update mode");
    2290           2 :         return nullptr;
    2291             :     }
    2292         106 :     if (anDimensions.size() >= 2)
    2293             :     {
    2294           2 :         CPLError(CE_Failure, CPLE_NotSupported,
    2295             :                  "Cannot create attributes of dimension >= 2");
    2296           2 :         return nullptr;
    2297             :     }
    2298             :     return m_oAttrGroup.CreateAttribute(osName, anDimensions, oDataType,
    2299         104 :                                         papszOptions);
    2300             : }
    2301             : 
    2302             : /************************************************************************/
    2303             : /*                  ZarrGroupBase::DeleteAttribute()                    */
    2304             : /************************************************************************/
    2305             : 
    2306          18 : bool ZarrArray::DeleteAttribute(const std::string &osName, CSLConstList)
    2307             : {
    2308          18 :     if (!CheckValidAndErrorOutIfNot())
    2309           0 :         return false;
    2310             : 
    2311          18 :     if (!m_bUpdatable)
    2312             :     {
    2313           6 :         CPLError(CE_Failure, CPLE_NotSupported,
    2314             :                  "Dataset not open in update mode");
    2315           6 :         return false;
    2316             :     }
    2317             : 
    2318          12 :     return m_oAttrGroup.DeleteAttribute(osName);
    2319             : }
    2320             : 
    2321             : /************************************************************************/
    2322             : /*                      ZarrArray::SetSpatialRef()                      */
    2323             : /************************************************************************/
    2324             : 
    2325          39 : bool ZarrArray::SetSpatialRef(const OGRSpatialReference *poSRS)
    2326             : {
    2327          39 :     if (!CheckValidAndErrorOutIfNot())
    2328           0 :         return false;
    2329             : 
    2330          39 :     if (!m_bUpdatable)
    2331             :     {
    2332           2 :         return GDALPamMDArray::SetSpatialRef(poSRS);
    2333             :     }
    2334          37 :     m_poSRS.reset();
    2335          37 :     if (poSRS)
    2336          37 :         m_poSRS.reset(poSRS->Clone());
    2337          37 :     m_bSRSModified = true;
    2338          37 :     return true;
    2339             : }
    2340             : 
    2341             : /************************************************************************/
    2342             : /*                         ZarrArray::SetUnit()                         */
    2343             : /************************************************************************/
    2344             : 
    2345           9 : bool ZarrArray::SetUnit(const std::string &osUnit)
    2346             : {
    2347           9 :     if (!CheckValidAndErrorOutIfNot())
    2348           0 :         return false;
    2349             : 
    2350           9 :     if (!m_bUpdatable)
    2351             :     {
    2352           0 :         CPLError(CE_Failure, CPLE_NotSupported,
    2353             :                  "Dataset not open in update mode");
    2354           0 :         return false;
    2355             :     }
    2356           9 :     m_osUnit = osUnit;
    2357           9 :     m_bUnitModified = true;
    2358           9 :     return true;
    2359             : }
    2360             : 
    2361             : /************************************************************************/
    2362             : /*                       ZarrArray::GetOffset()                         */
    2363             : /************************************************************************/
    2364             : 
    2365          79 : double ZarrArray::GetOffset(bool *pbHasOffset,
    2366             :                             GDALDataType *peStorageType) const
    2367             : {
    2368          79 :     if (pbHasOffset)
    2369          79 :         *pbHasOffset = m_bHasOffset;
    2370          79 :     if (peStorageType)
    2371           0 :         *peStorageType = GDT_Unknown;
    2372          79 :     return m_dfOffset;
    2373             : }
    2374             : 
    2375             : /************************************************************************/
    2376             : /*                       ZarrArray::GetScale()                          */
    2377             : /************************************************************************/
    2378             : 
    2379          77 : double ZarrArray::GetScale(bool *pbHasScale, GDALDataType *peStorageType) const
    2380             : {
    2381          77 :     if (pbHasScale)
    2382          77 :         *pbHasScale = m_bHasScale;
    2383          77 :     if (peStorageType)
    2384           0 :         *peStorageType = GDT_Unknown;
    2385          77 :     return m_dfScale;
    2386             : }
    2387             : 
    2388             : /************************************************************************/
    2389             : /*                       ZarrArray::SetOffset()                         */
    2390             : /************************************************************************/
    2391             : 
    2392           3 : bool ZarrArray::SetOffset(double dfOffset, GDALDataType /* eStorageType */)
    2393             : {
    2394           3 :     if (!CheckValidAndErrorOutIfNot())
    2395           0 :         return false;
    2396             : 
    2397           3 :     m_dfOffset = dfOffset;
    2398           3 :     m_bHasOffset = true;
    2399           3 :     m_bOffsetModified = true;
    2400           3 :     return true;
    2401             : }
    2402             : 
    2403             : /************************************************************************/
    2404             : /*                       ZarrArray::SetScale()                          */
    2405             : /************************************************************************/
    2406             : 
    2407           3 : bool ZarrArray::SetScale(double dfScale, GDALDataType /* eStorageType */)
    2408             : {
    2409           3 :     if (!CheckValidAndErrorOutIfNot())
    2410           0 :         return false;
    2411             : 
    2412           3 :     m_dfScale = dfScale;
    2413           3 :     m_bHasScale = true;
    2414           3 :     m_bScaleModified = true;
    2415           3 :     return true;
    2416             : }
    2417             : 
    2418             : /************************************************************************/
    2419             : /*                      GetDimensionTypeDirection()                     */
    2420             : /************************************************************************/
    2421             : 
    2422             : /* static */
    2423         138 : void ZarrArray::GetDimensionTypeDirection(CPLJSONObject &oAttributes,
    2424             :                                           std::string &osType,
    2425             :                                           std::string &osDirection)
    2426             : {
    2427         276 :     std::string osUnit;
    2428         414 :     const auto unit = oAttributes[CF_UNITS];
    2429         138 :     if (unit.GetType() == CPLJSONObject::Type::String)
    2430             :     {
    2431          50 :         osUnit = unit.ToString();
    2432             :     }
    2433             : 
    2434         414 :     const auto oStdName = oAttributes[CF_STD_NAME];
    2435         138 :     if (oStdName.GetType() == CPLJSONObject::Type::String)
    2436             :     {
    2437         150 :         const auto osStdName = oStdName.ToString();
    2438          50 :         if (osStdName == CF_PROJ_X_COORD || osStdName == CF_LONGITUDE_STD_NAME)
    2439             :         {
    2440          25 :             osType = GDAL_DIM_TYPE_HORIZONTAL_X;
    2441          25 :             oAttributes.Delete(CF_STD_NAME);
    2442          25 :             if (osUnit == CF_DEGREES_EAST)
    2443             :             {
    2444          23 :                 osDirection = "EAST";
    2445             :             }
    2446             :         }
    2447          48 :         else if (osStdName == CF_PROJ_Y_COORD ||
    2448          23 :                  osStdName == CF_LATITUDE_STD_NAME)
    2449             :         {
    2450          25 :             osType = GDAL_DIM_TYPE_HORIZONTAL_Y;
    2451          25 :             oAttributes.Delete(CF_STD_NAME);
    2452          25 :             if (osUnit == CF_DEGREES_NORTH)
    2453             :             {
    2454          23 :                 osDirection = "NORTH";
    2455             :             }
    2456             :         }
    2457           0 :         else if (osStdName == "time")
    2458             :         {
    2459           0 :             osType = GDAL_DIM_TYPE_TEMPORAL;
    2460           0 :             oAttributes.Delete(CF_STD_NAME);
    2461             :         }
    2462             :     }
    2463             : 
    2464         414 :     const auto osAxis = oAttributes[CF_AXIS].ToString();
    2465         138 :     if (osAxis == "Z")
    2466             :     {
    2467           0 :         osType = GDAL_DIM_TYPE_VERTICAL;
    2468           0 :         const auto osPositive = oAttributes["positive"].ToString();
    2469           0 :         if (osPositive == "up")
    2470             :         {
    2471           0 :             osDirection = "UP";
    2472           0 :             oAttributes.Delete("positive");
    2473             :         }
    2474           0 :         else if (osPositive == "down")
    2475             :         {
    2476           0 :             osDirection = "DOWN";
    2477           0 :             oAttributes.Delete("positive");
    2478             :         }
    2479           0 :         oAttributes.Delete(CF_AXIS);
    2480             :     }
    2481         138 : }
    2482             : 
    2483             : /************************************************************************/
    2484             : /*                      GetCoordinateVariables()                        */
    2485             : /************************************************************************/
    2486             : 
    2487             : std::vector<std::shared_ptr<GDALMDArray>>
    2488           2 : ZarrArray::GetCoordinateVariables() const
    2489             : {
    2490           2 :     if (!CheckValidAndErrorOutIfNot())
    2491           0 :         return {};
    2492             : 
    2493           4 :     std::vector<std::shared_ptr<GDALMDArray>> ret;
    2494           6 :     const auto poCoordinates = GetAttribute("coordinates");
    2495           1 :     if (poCoordinates &&
    2496           3 :         poCoordinates->GetDataType().GetClass() == GEDTC_STRING &&
    2497           1 :         poCoordinates->GetDimensionCount() == 0)
    2498             :     {
    2499           1 :         const char *pszCoordinates = poCoordinates->ReadAsString();
    2500           1 :         if (pszCoordinates)
    2501             :         {
    2502           2 :             auto poGroup = m_poGroupWeak.lock();
    2503           1 :             if (!poGroup)
    2504             :             {
    2505           0 :                 CPLError(CE_Failure, CPLE_AppDefined,
    2506             :                          "Cannot access coordinate variables of %s has "
    2507             :                          "belonging group has gone out of scope",
    2508           0 :                          GetName().c_str());
    2509             :             }
    2510             :             else
    2511             :             {
    2512             :                 const CPLStringList aosNames(
    2513           2 :                     CSLTokenizeString2(pszCoordinates, " ", 0));
    2514           3 :                 for (int i = 0; i < aosNames.size(); i++)
    2515             :                 {
    2516           6 :                     auto poCoordinateVar = poGroup->OpenMDArray(aosNames[i]);
    2517           2 :                     if (poCoordinateVar)
    2518             :                     {
    2519           2 :                         ret.emplace_back(poCoordinateVar);
    2520             :                     }
    2521             :                     else
    2522             :                     {
    2523           0 :                         CPLError(CE_Warning, CPLE_AppDefined,
    2524             :                                  "Cannot find variable corresponding to "
    2525             :                                  "coordinate %s",
    2526             :                                  aosNames[i]);
    2527             :                     }
    2528             :                 }
    2529             :             }
    2530             :         }
    2531             :     }
    2532             : 
    2533           2 :     return ret;
    2534             : }
    2535             : 
    2536             : /************************************************************************/
    2537             : /*                            Resize()                                  */
    2538             : /************************************************************************/
    2539             : 
    2540          16 : bool ZarrArray::Resize(const std::vector<GUInt64> &anNewDimSizes,
    2541             :                        CSLConstList /* papszOptions */)
    2542             : {
    2543          16 :     if (!CheckValidAndErrorOutIfNot())
    2544           0 :         return false;
    2545             : 
    2546          16 :     if (!IsWritable())
    2547             :     {
    2548           3 :         CPLError(CE_Failure, CPLE_AppDefined,
    2549             :                  "Resize() not supported on read-only file");
    2550           3 :         return false;
    2551             :     }
    2552             : 
    2553          13 :     const auto nDimCount = GetDimensionCount();
    2554          13 :     if (anNewDimSizes.size() != nDimCount)
    2555             :     {
    2556           0 :         CPLError(CE_Failure, CPLE_IllegalArg,
    2557             :                  "Not expected number of values in anNewDimSizes.");
    2558           0 :         return false;
    2559             :     }
    2560             : 
    2561          13 :     auto &dims = GetDimensions();
    2562          26 :     std::vector<size_t> anGrownDimIdx;
    2563          26 :     std::map<GDALDimension *, GUInt64> oMapDimToSize;
    2564          27 :     for (size_t i = 0; i < nDimCount; ++i)
    2565             :     {
    2566          21 :         auto oIter = oMapDimToSize.find(dims[i].get());
    2567          21 :         if (oIter != oMapDimToSize.end() && oIter->second != anNewDimSizes[i])
    2568             :         {
    2569           2 :             CPLError(CE_Failure, CPLE_AppDefined,
    2570             :                      "Cannot resize a dimension referenced several times "
    2571             :                      "to different sizes");
    2572           7 :             return false;
    2573             :         }
    2574          19 :         if (anNewDimSizes[i] != dims[i]->GetSize())
    2575             :         {
    2576          14 :             if (anNewDimSizes[i] < dims[i]->GetSize())
    2577             :             {
    2578           5 :                 CPLError(CE_Failure, CPLE_NotSupported,
    2579             :                          "Resize() does not support shrinking the array.");
    2580           5 :                 return false;
    2581             :             }
    2582             : 
    2583           9 :             oMapDimToSize[dims[i].get()] = anNewDimSizes[i];
    2584           9 :             anGrownDimIdx.push_back(i);
    2585             :         }
    2586             :         else
    2587             :         {
    2588           5 :             oMapDimToSize[dims[i].get()] = dims[i]->GetSize();
    2589             :         }
    2590             :     }
    2591           6 :     if (!anGrownDimIdx.empty())
    2592             :     {
    2593           6 :         m_bDefinitionModified = true;
    2594          13 :         for (size_t dimIdx : anGrownDimIdx)
    2595             :         {
    2596          14 :             auto dim = std::dynamic_pointer_cast<ZarrDimension>(dims[dimIdx]);
    2597           7 :             if (dim)
    2598             :             {
    2599           7 :                 dim->SetSize(anNewDimSizes[dimIdx]);
    2600           7 :                 if (dim->GetName() != dim->GetFullName())
    2601             :                 {
    2602             :                     // This is not a local dimension
    2603           7 :                     m_poSharedResource->UpdateDimensionSize(dim);
    2604             :                 }
    2605             :             }
    2606             :             else
    2607             :             {
    2608           0 :                 CPLAssert(false);
    2609             :             }
    2610             :         }
    2611             :     }
    2612           6 :     return true;
    2613             : }
    2614             : 
    2615             : /************************************************************************/
    2616             : /*                       NotifyChildrenOfRenaming()                     */
    2617             : /************************************************************************/
    2618             : 
    2619          15 : void ZarrArray::NotifyChildrenOfRenaming()
    2620             : {
    2621          15 :     m_oAttrGroup.ParentRenamed(m_osFullName);
    2622          15 : }
    2623             : 
    2624             : /************************************************************************/
    2625             : /*                          ParentRenamed()                             */
    2626             : /************************************************************************/
    2627             : 
    2628           9 : void ZarrArray::ParentRenamed(const std::string &osNewParentFullName)
    2629             : {
    2630           9 :     GDALMDArray::ParentRenamed(osNewParentFullName);
    2631             : 
    2632           9 :     auto poParent = m_poGroupWeak.lock();
    2633             :     // The parent necessarily exist, since it notified us
    2634           9 :     CPLAssert(poParent);
    2635             : 
    2636          27 :     m_osFilename = CPLFormFilenameSafe(
    2637          18 :         CPLFormFilenameSafe(poParent->GetDirectoryName().c_str(),
    2638           9 :                             m_osName.c_str(), nullptr)
    2639             :             .c_str(),
    2640           9 :         CPLGetFilename(m_osFilename.c_str()), nullptr);
    2641           9 : }
    2642             : 
    2643             : /************************************************************************/
    2644             : /*                              Rename()                                */
    2645             : /************************************************************************/
    2646             : 
    2647          21 : bool ZarrArray::Rename(const std::string &osNewName)
    2648             : {
    2649          21 :     if (!CheckValidAndErrorOutIfNot())
    2650           0 :         return false;
    2651             : 
    2652          21 :     if (!m_bUpdatable)
    2653             :     {
    2654           6 :         CPLError(CE_Failure, CPLE_NotSupported,
    2655             :                  "Dataset not open in update mode");
    2656           6 :         return false;
    2657             :     }
    2658          15 :     if (!ZarrGroupBase::IsValidObjectName(osNewName))
    2659             :     {
    2660           3 :         CPLError(CE_Failure, CPLE_NotSupported, "Invalid array name");
    2661           3 :         return false;
    2662             :     }
    2663             : 
    2664          24 :     auto poParent = m_poGroupWeak.lock();
    2665          12 :     if (poParent)
    2666             :     {
    2667          12 :         if (!poParent->CheckArrayOrGroupWithSameNameDoesNotExist(osNewName))
    2668           6 :             return false;
    2669             :     }
    2670             : 
    2671             :     const std::string osRootDirectoryName(
    2672          12 :         CPLGetDirnameSafe(CPLGetDirnameSafe(m_osFilename.c_str()).c_str()));
    2673             :     const std::string osOldDirectoryName = CPLFormFilenameSafe(
    2674          12 :         osRootDirectoryName.c_str(), m_osName.c_str(), nullptr);
    2675             :     const std::string osNewDirectoryName = CPLFormFilenameSafe(
    2676          12 :         osRootDirectoryName.c_str(), osNewName.c_str(), nullptr);
    2677             : 
    2678           6 :     if (VSIRename(osOldDirectoryName.c_str(), osNewDirectoryName.c_str()) != 0)
    2679             :     {
    2680           0 :         CPLError(CE_Failure, CPLE_AppDefined, "Renaming of %s to %s failed",
    2681             :                  osOldDirectoryName.c_str(), osNewDirectoryName.c_str());
    2682           0 :         return false;
    2683             :     }
    2684             : 
    2685           6 :     m_poSharedResource->RenameZMetadataRecursive(osOldDirectoryName,
    2686             :                                                  osNewDirectoryName);
    2687             : 
    2688             :     m_osFilename =
    2689          12 :         CPLFormFilenameSafe(osNewDirectoryName.c_str(),
    2690           6 :                             CPLGetFilename(m_osFilename.c_str()), nullptr);
    2691             : 
    2692           6 :     if (poParent)
    2693             :     {
    2694           6 :         poParent->NotifyArrayRenamed(m_osName, osNewName);
    2695             :     }
    2696             : 
    2697           6 :     BaseRename(osNewName);
    2698             : 
    2699           6 :     return true;
    2700             : }
    2701             : 
    2702             : /************************************************************************/
    2703             : /*                       NotifyChildrenOfDeletion()                     */
    2704             : /************************************************************************/
    2705             : 
    2706           8 : void ZarrArray::NotifyChildrenOfDeletion()
    2707             : {
    2708           8 :     m_oAttrGroup.ParentDeleted();
    2709           8 : }
    2710             : 
    2711             : /************************************************************************/
    2712             : /*                     ParseSpecialAttributes()                         */
    2713             : /************************************************************************/
    2714             : 
    2715         747 : void ZarrArray::ParseSpecialAttributes(
    2716             :     const std::shared_ptr<GDALGroup> &poGroup, CPLJSONObject &oAttributes)
    2717             : {
    2718        2241 :     const auto crs = oAttributes[CRS_ATTRIBUTE_NAME];
    2719         747 :     std::shared_ptr<OGRSpatialReference> poSRS;
    2720         747 :     if (crs.GetType() == CPLJSONObject::Type::Object)
    2721             :     {
    2722          43 :         for (const char *key : {"url", "wkt", "projjson"})
    2723             :         {
    2724          86 :             const auto item = crs[key];
    2725          43 :             if (item.IsValid())
    2726             :             {
    2727          24 :                 poSRS = std::make_shared<OGRSpatialReference>();
    2728          24 :                 poSRS->SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
    2729          48 :                 if (poSRS->SetFromUserInput(
    2730          48 :                         item.ToString().c_str(),
    2731             :                         OGRSpatialReference::
    2732          24 :                             SET_FROM_USER_INPUT_LIMITATIONS_get()) ==
    2733             :                     OGRERR_NONE)
    2734             :                 {
    2735          24 :                     oAttributes.Delete(CRS_ATTRIBUTE_NAME);
    2736          24 :                     break;
    2737             :                 }
    2738           0 :                 poSRS.reset();
    2739             :             }
    2740             :         }
    2741             :     }
    2742             :     else
    2743             :     {
    2744             :         // Check if SRS is using CF-1 conventions
    2745        2169 :         const auto gridMapping = oAttributes["grid_mapping"];
    2746         723 :         if (gridMapping.GetType() == CPLJSONObject::Type::String)
    2747             :         {
    2748             :             const auto gridMappingArray =
    2749           6 :                 poGroup->OpenMDArray(gridMapping.ToString());
    2750           2 :             if (gridMappingArray)
    2751             :             {
    2752           2 :                 poSRS = std::make_shared<OGRSpatialReference>();
    2753           2 :                 poSRS->SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
    2754           4 :                 CPLStringList aosKeyValues;
    2755          22 :                 for (const auto &poAttr : gridMappingArray->GetAttributes())
    2756             :                 {
    2757          20 :                     if (poAttr->GetDataType().GetClass() == GEDTC_STRING)
    2758             :                     {
    2759           4 :                         aosKeyValues.SetNameValue(poAttr->GetName().c_str(),
    2760           8 :                                                   poAttr->ReadAsString());
    2761             :                     }
    2762          16 :                     else if (poAttr->GetDataType().GetClass() == GEDTC_NUMERIC)
    2763             :                     {
    2764          32 :                         std::string osVal;
    2765          32 :                         for (double val : poAttr->ReadAsDoubleArray())
    2766             :                         {
    2767          16 :                             if (!osVal.empty())
    2768           0 :                                 osVal += ',';
    2769          16 :                             osVal += CPLSPrintf("%.17g", val);
    2770             :                         }
    2771          16 :                         aosKeyValues.SetNameValue(poAttr->GetName().c_str(),
    2772          32 :                                                   osVal.c_str());
    2773             :                     }
    2774             :                 }
    2775           2 :                 if (poSRS->importFromCF1(aosKeyValues.List(), nullptr) !=
    2776             :                     OGRERR_NONE)
    2777             :                 {
    2778           0 :                     poSRS.reset();
    2779             :                 }
    2780             :             }
    2781             :         }
    2782             :     }
    2783             : 
    2784         747 :     if (poSRS)
    2785             :     {
    2786          26 :         int iDimX = 0;
    2787          26 :         int iDimY = 0;
    2788          26 :         int iCount = 1;
    2789          78 :         for (const auto &poDim : GetDimensions())
    2790             :         {
    2791          52 :             if (poDim->GetType() == GDAL_DIM_TYPE_HORIZONTAL_X)
    2792           2 :                 iDimX = iCount;
    2793          50 :             else if (poDim->GetType() == GDAL_DIM_TYPE_HORIZONTAL_Y)
    2794           2 :                 iDimY = iCount;
    2795          52 :             iCount++;
    2796             :         }
    2797          26 :         if ((iDimX == 0 || iDimY == 0) && GetDimensionCount() >= 2)
    2798             :         {
    2799          24 :             iDimX = static_cast<int>(GetDimensionCount());
    2800          24 :             iDimY = iDimX - 1;
    2801             :         }
    2802          26 :         if (iDimX > 0 && iDimY > 0)
    2803             :         {
    2804          26 :             const auto &oMapping = poSRS->GetDataAxisToSRSAxisMapping();
    2805          72 :             if (oMapping == std::vector<int>{2, 1} ||
    2806          46 :                 oMapping == std::vector<int>{2, 1, 3})
    2807           6 :                 poSRS->SetDataAxisToSRSAxisMapping({iDimY, iDimX});
    2808          41 :             else if (oMapping == std::vector<int>{1, 2} ||
    2809          21 :                      oMapping == std::vector<int>{1, 2, 3})
    2810          20 :                 poSRS->SetDataAxisToSRSAxisMapping({iDimX, iDimY});
    2811             :         }
    2812             : 
    2813          26 :         SetSRS(poSRS);
    2814             :     }
    2815             : 
    2816        2241 :     const auto unit = oAttributes[CF_UNITS];
    2817         747 :     if (unit.GetType() == CPLJSONObject::Type::String)
    2818             :     {
    2819         159 :         std::string osUnit = unit.ToString();
    2820          53 :         oAttributes.Delete(CF_UNITS);
    2821          53 :         RegisterUnit(osUnit);
    2822             :     }
    2823             : 
    2824        2241 :     const auto offset = oAttributes[CF_ADD_OFFSET];
    2825         747 :     const auto offsetType = offset.GetType();
    2826         747 :     if (offsetType == CPLJSONObject::Type::Integer ||
    2827         747 :         offsetType == CPLJSONObject::Type::Long ||
    2828             :         offsetType == CPLJSONObject::Type::Double)
    2829             :     {
    2830           3 :         double dfOffset = offset.ToDouble();
    2831           3 :         oAttributes.Delete(CF_ADD_OFFSET);
    2832           3 :         RegisterOffset(dfOffset);
    2833             :     }
    2834             : 
    2835        2241 :     const auto scale = oAttributes[CF_SCALE_FACTOR];
    2836         747 :     const auto scaleType = scale.GetType();
    2837         747 :     if (scaleType == CPLJSONObject::Type::Integer ||
    2838         747 :         scaleType == CPLJSONObject::Type::Long ||
    2839             :         scaleType == CPLJSONObject::Type::Double)
    2840             :     {
    2841           3 :         double dfScale = scale.ToDouble();
    2842           3 :         oAttributes.Delete(CF_SCALE_FACTOR);
    2843           3 :         RegisterScale(dfScale);
    2844             :     }
    2845         747 : }
    2846             : 
    2847             : /************************************************************************/
    2848             : /*                           SetStatistics()                            */
    2849             : /************************************************************************/
    2850             : 
    2851           1 : bool ZarrArray::SetStatistics(bool bApproxStats, double dfMin, double dfMax,
    2852             :                               double dfMean, double dfStdDev,
    2853             :                               GUInt64 nValidCount, CSLConstList papszOptions)
    2854             : {
    2855           2 :     if (!bApproxStats && m_bUpdatable &&
    2856           1 :         CPLTestBool(
    2857             :             CSLFetchNameValueDef(papszOptions, "UPDATE_METADATA", "NO")))
    2858             :     {
    2859           3 :         auto poAttr = GetAttribute("actual_range");
    2860           1 :         if (!poAttr)
    2861             :         {
    2862             :             poAttr =
    2863           1 :                 CreateAttribute("actual_range", {2}, GetDataType(), nullptr);
    2864             :         }
    2865           1 :         if (poAttr)
    2866             :         {
    2867           2 :             std::vector<GUInt64> startIdx = {0};
    2868           2 :             std::vector<size_t> count = {2};
    2869           1 :             std::vector<double> values = {dfMin, dfMax};
    2870           2 :             poAttr->Write(startIdx.data(), count.data(), nullptr, nullptr,
    2871           2 :                           GDALExtendedDataType::Create(GDT_Float64),
    2872           1 :                           values.data(), nullptr, 0);
    2873             :         }
    2874             :     }
    2875           1 :     return GDALPamMDArray::SetStatistics(bApproxStats, dfMin, dfMax, dfMean,
    2876           1 :                                          dfStdDev, nValidCount, papszOptions);
    2877             : }

Generated by: LCOV version 1.14