LCOV - code coverage report
Current view: top level - frmts/opjlike - jp2opjlikedataset.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 1580 1822 86.7 %
Date: 2025-01-18 12:42:00 Functions: 38 39 97.4 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  JPEG2000 driver based on OpenJPEG or Grok library
       4             :  * Purpose:  JPEG2000 driver based on OpenJPEG or Grok library
       5             :  * Authors:  Even Rouault, <even dot rouault at spatialys dot com>
       6             :  *           Aaron Boxer, <boxerab at protonmail dot com>
       7             :  *
       8             :  ******************************************************************************
       9             :  * Copyright (c) 2010-2014, Even Rouault <even dot rouault at spatialys dot com>
      10             :  * Copyright (c) 2015, European Union (European Environment Agency)
      11             :  * Copyright (c) 2023, Grok Image Compression Inc.
      12             :  *
      13             :  * SPDX-License-Identifier: MIT
      14             :  ****************************************************************************/
      15             : 
      16             : #include <cassert>
      17             : #include <vector>
      18             : 
      19             : #include "cpl_atomic_ops.h"
      20             : #include "cpl_multiproc.h"
      21             : #include "cpl_string.h"
      22             : #include "cpl_worker_thread_pool.h"
      23             : #include "gdal_frmts.h"
      24             : #include "gdaljp2abstractdataset.h"
      25             : #include "gdaljp2metadata.h"
      26             : #include "vrt/vrtdataset.h"
      27             : 
      28             : #include <algorithm>
      29             : 
      30             : #include "jp2opjlikedataset.h"
      31             : 
      32             : /************************************************************************/
      33             : /*                        JP2OPJLikeRasterBand()                        */
      34             : /************************************************************************/
      35             : 
      36             : template <typename CODEC, typename BASE>
      37        1146 : JP2OPJLikeRasterBand<CODEC, BASE>::JP2OPJLikeRasterBand(
      38             :     JP2OPJLikeDataset<CODEC, BASE> *poDSIn, int nBandIn,
      39             :     GDALDataType eDataTypeIn, int nBits, int bPromoteTo8BitIn,
      40        1146 :     int nBlockXSizeIn, int nBlockYSizeIn)
      41             : 
      42             : {
      43        1146 :     this->eDataType = eDataTypeIn;
      44        1146 :     this->nBlockXSize = nBlockXSizeIn;
      45        1146 :     this->nBlockYSize = nBlockYSizeIn;
      46        1146 :     this->bPromoteTo8Bit = bPromoteTo8BitIn;
      47        1146 :     poCT = nullptr;
      48             : 
      49        1146 :     if ((nBits % 8) != 0)
      50          41 :         GDALRasterBand::SetMetadataItem(
      51          82 :             "NBITS", CPLString().Printf("%d", nBits), "IMAGE_STRUCTURE");
      52        1146 :     GDALRasterBand::SetMetadataItem("COMPRESSION", "JPEG2000",
      53             :                                     "IMAGE_STRUCTURE");
      54        1146 :     this->poDS = poDSIn;
      55        1146 :     this->nBand = nBandIn;
      56        1146 : }
      57             : 
      58             : template <typename CODEC, typename BASE>
      59          50 : GDALColorTable *JP2OPJLikeRasterBand<CODEC, BASE>::GetColorTable()
      60             : {
      61          50 :     return poCT;
      62             : }
      63             : 
      64             : template <typename CODEC, typename BASE>
      65           0 : int JP2OPJLikeRasterBand<CODEC, BASE>::HasArbitraryOverviews()
      66             : {
      67           0 :     return poCT == nullptr;
      68             : }
      69             : 
      70             : /************************************************************************/
      71             : /*                      ~JP2OPJLikeRasterBand()                         */
      72             : /************************************************************************/
      73             : 
      74             : template <typename CODEC, typename BASE>
      75        2292 : JP2OPJLikeRasterBand<CODEC, BASE>::~JP2OPJLikeRasterBand()
      76             : {
      77          24 :     delete poCT;
      78        2316 : }
      79             : 
      80             : /************************************************************************/
      81             : /*                            CLAMP_0_255()                             */
      82             : /************************************************************************/
      83             : 
      84      134000 : static CPL_INLINE GByte CLAMP_0_255(int val)
      85             : {
      86      134000 :     if (val < 0)
      87        4039 :         return 0;
      88      129961 :     else if (val > 255)
      89         128 :         return 255;
      90             :     else
      91      129833 :         return (GByte)val;
      92             : }
      93             : 
      94             : /************************************************************************/
      95             : /*                             IReadBlock()                             */
      96             : /************************************************************************/
      97             : 
      98             : template <typename CODEC, typename BASE>
      99         254 : CPLErr JP2OPJLikeRasterBand<CODEC, BASE>::IReadBlock(int nBlockXOff,
     100             :                                                      int nBlockYOff,
     101             :                                                      void *pImage)
     102             : {
     103         254 :     auto poGDS = (JP2OPJLikeDataset<CODEC, BASE> *)poDS;
     104             : 
     105             : #ifdef DEBUG_VERBOSE
     106             :     int nXOff = nBlockXOff * nBlockXSize;
     107             :     int nYOff = nBlockYOff * nBlockYSize;
     108             :     int nXSize = std::min(nBlockXSize, nRasterXSize - nXOff);
     109             :     int nYSize = std::min(nBlockYSize, nRasterYSize - nYOff);
     110             :     if (poGDS->iLevel == 0)
     111             :     {
     112             :         CPLDebug(CODEC::debugId(),
     113             :                  "ds.GetRasterBand(%d).ReadRaster(%d,%d,%d,%d)", nBand, nXOff,
     114             :                  nYOff, nXSize, nYSize);
     115             :     }
     116             :     else
     117             :     {
     118             :         CPLDebug(CODEC::debugId(),
     119             :                  "ds.GetRasterBand(%d).GetOverview(%d).ReadRaster(%d,%d,%d,%d)",
     120             :                  nBand, poGDS->iLevel - 1, nXOff, nYOff, nXSize, nYSize);
     121             :     }
     122             : #endif
     123             : 
     124         254 :     if (poGDS->bEnoughMemoryToLoadOtherBands)
     125             :         return poGDS->ReadBlock(nBand, poGDS->fp_, nBlockXOff, nBlockYOff,
     126         254 :                                 pImage, poGDS->nBands, nullptr);
     127             :     else
     128             :         return poGDS->ReadBlock(nBand, poGDS->fp_, nBlockXOff, nBlockYOff,
     129           0 :                                 pImage, 1, &nBand);
     130             : }
     131             : 
     132             : /************************************************************************/
     133             : /*                             IRasterIO()                              */
     134             : /************************************************************************/
     135             : 
     136             : template <typename CODEC, typename BASE>
     137         227 : CPLErr JP2OPJLikeRasterBand<CODEC, BASE>::IRasterIO(
     138             :     GDALRWFlag eRWFlag, int nXOff, int nYOff, int nXSize, int nYSize,
     139             :     void *pData, int nBufXSize, int nBufYSize, GDALDataType eBufType,
     140             :     GSpacing nPixelSpace, GSpacing nLineSpace, GDALRasterIOExtraArg *psExtraArg)
     141             : {
     142         227 :     auto poGDS = (JP2OPJLikeDataset<CODEC, BASE> *)poDS;
     143             : 
     144         227 :     if (eRWFlag != GF_Read)
     145           0 :         return CE_Failure;
     146             : 
     147             :     /* ==================================================================== */
     148             :     /*      Do we have overviews that would be appropriate to satisfy       */
     149             :     /*      this request?                                                   */
     150             :     /* ==================================================================== */
     151         227 :     if ((nBufXSize < nXSize || nBufYSize < nYSize) && GetOverviewCount() > 0)
     152             :     {
     153             :         int bTried;
     154           1 :         CPLErr eErr = TryOverviewRasterIO(
     155             :             eRWFlag, nXOff, nYOff, nXSize, nYSize, pData, nBufXSize, nBufYSize,
     156             :             eBufType, nPixelSpace, nLineSpace, psExtraArg, &bTried);
     157           1 :         if (bTried)
     158           1 :             return eErr;
     159             :     }
     160             : 
     161         226 :     int nRet =
     162             :         poGDS->PreloadBlocks(this, nXOff, nYOff, nXSize, nYSize, 0, nullptr);
     163         226 :     if (nRet < 0)
     164           0 :         return CE_Failure;
     165         226 :     poGDS->bEnoughMemoryToLoadOtherBands = nRet;
     166             : 
     167         226 :     CPLErr eErr = GDALPamRasterBand::IRasterIO(
     168             :         eRWFlag, nXOff, nYOff, nXSize, nYSize, pData, nBufXSize, nBufYSize,
     169             :         eBufType, nPixelSpace, nLineSpace, psExtraArg);
     170             : 
     171             :     // cppcheck-suppress redundantAssignment
     172         226 :     poGDS->bEnoughMemoryToLoadOtherBands = TRUE;
     173         226 :     return eErr;
     174             : }
     175             : 
     176             : template <typename CODEC, typename BASE> class JP2JobStruct
     177             : {
     178             :   public:
     179             :     JP2OPJLikeDataset<CODEC, BASE> *poGDS_;
     180             :     int nBand;
     181             :     std::vector<std::pair<int, int>> oPairs;
     182             :     volatile int nCurPair;
     183             :     int nBandCount;
     184             :     const int *panBandMap;
     185             :     volatile bool bSuccess;
     186             : };
     187             : 
     188             : /************************************************************************/
     189             : /*                   GetFileHandle()                                    */
     190             : /************************************************************************/
     191             : 
     192             : template <typename CODEC, typename BASE>
     193           8 : VSILFILE *JP2OPJLikeDataset<CODEC, BASE>::GetFileHandle()
     194             : {
     195           8 :     return this->fp_;
     196             : }
     197             : 
     198             : /************************************************************************/
     199             : /*                   ReadBlockInThread()                                */
     200             : /************************************************************************/
     201             : 
     202             : template <typename CODEC, typename BASE>
     203          17 : void JP2OPJLikeDataset<CODEC, BASE>::ReadBlockInThread(void *userdata)
     204             : {
     205             :     int nPair;
     206          17 :     auto poJob = (JP2JobStruct<CODEC, BASE> *)userdata;
     207             : 
     208          17 :     JP2OPJLikeDataset *poGDS = poJob->poGDS_;
     209          17 :     int nBand = poJob->nBand;
     210          17 :     int nPairs = (int)poJob->oPairs.size();
     211          17 :     int nBandCount = poJob->nBandCount;
     212          17 :     const int *panBandMap = poJob->panBandMap;
     213          17 :     VSILFILE *fp = VSIFOpenL(poGDS->m_osFilename.c_str(), "rb");
     214          17 :     if (fp == nullptr)
     215             :     {
     216           0 :         CPLDebug(CODEC::debugId(), "Cannot open %s",
     217             :                  poGDS->m_osFilename.c_str());
     218           0 :         poJob->bSuccess = false;
     219             :         // VSIFree(pDummy);
     220           0 :         return;
     221             :     }
     222             : 
     223         290 :     while ((nPair = CPLAtomicInc(&(poJob->nCurPair))) < nPairs &&
     224         137 :            poJob->bSuccess)
     225             :     {
     226         137 :         int nBlockXOff = poJob->oPairs[nPair].first;
     227         137 :         int nBlockYOff = poJob->oPairs[nPair].second;
     228         137 :         poGDS->AcquireMutex();
     229             :         GDALRasterBlock *poBlock =
     230         137 :             poGDS->GetRasterBand(nBand)->GetLockedBlockRef(nBlockXOff,
     231             :                                                            nBlockYOff, TRUE);
     232         137 :         poGDS->ReleaseMutex();
     233         137 :         if (poBlock == nullptr)
     234             :         {
     235           0 :             poJob->bSuccess = false;
     236           0 :             break;
     237             :         }
     238             : 
     239         137 :         void *pDstBuffer = poBlock->GetDataRef();
     240         137 :         if (poGDS->ReadBlock(nBand, fp, nBlockXOff, nBlockYOff, pDstBuffer,
     241         137 :                              nBandCount, panBandMap) != CE_None)
     242             :         {
     243           0 :             poJob->bSuccess = false;
     244             :         }
     245             : 
     246         137 :         poBlock->DropLock();
     247             :     }
     248             : 
     249          16 :     VSIFCloseL(fp);
     250             :     // VSIFree(pDummy);
     251             : }
     252             : 
     253             : /************************************************************************/
     254             : /*                           PreloadBlocks()                            */
     255             : /************************************************************************/
     256             : 
     257             : template <typename CODEC, typename BASE>
     258         240 : int JP2OPJLikeDataset<CODEC, BASE>::PreloadBlocks(
     259             :     JP2OPJLikeRasterBand<CODEC, BASE> *poBand, int nXOff, int nYOff, int nXSize,
     260             :     int nYSize, int nBandCount, const int *panBandMap)
     261             : {
     262         240 :     int bRet = TRUE;
     263         240 :     int nXStart = nXOff / poBand->nBlockXSize;
     264         240 :     int nXEnd = (nXOff + nXSize - 1) / poBand->nBlockXSize;
     265         240 :     int nYStart = nYOff / poBand->nBlockYSize;
     266         240 :     int nYEnd = (nYOff + nYSize - 1) / poBand->nBlockYSize;
     267         480 :     GIntBig nReqMem = (GIntBig)(nXEnd - nXStart + 1) * (nYEnd - nYStart + 1) *
     268         240 :                       poBand->nBlockXSize * poBand->nBlockYSize *
     269         240 :                       (GDALGetDataTypeSize(poBand->eDataType) / 8);
     270             : 
     271         240 :     int nMaxThreads = this->GetNumThreads();
     272         240 :     if (!this->bUseSetDecodeArea && nMaxThreads > 1)
     273             :     {
     274          96 :         if (nReqMem > GDALGetCacheMax64() / (nBandCount == 0 ? 1 : nBandCount))
     275           0 :             return FALSE;
     276             : 
     277          96 :         JP2JobStruct<CODEC, BASE> oJob;
     278          96 :         this->m_nBlocksToLoad = 0;
     279             :         try
     280             :         {
     281         226 :             for (int nBlockXOff = nXStart; nBlockXOff <= nXEnd; ++nBlockXOff)
     282             :             {
     283         484 :                 for (int nBlockYOff = nYStart; nBlockYOff <= nYEnd;
     284             :                      ++nBlockYOff)
     285             :                 {
     286         354 :                     GDALRasterBlock *poBlock =
     287             :                         poBand->TryGetLockedBlockRef(nBlockXOff, nBlockYOff);
     288         354 :                     if (poBlock != nullptr)
     289             :                     {
     290         156 :                         poBlock->DropLock();
     291         156 :                         continue;
     292             :                     }
     293         198 :                     oJob.oPairs.push_back(
     294             :                         std::pair<int, int>(nBlockXOff, nBlockYOff));
     295         198 :                     this->m_nBlocksToLoad++;
     296             :                 }
     297             :             }
     298             :         }
     299           0 :         catch (const std::bad_alloc &)
     300             :         {
     301           0 :             CPLError(CE_Failure, CPLE_OutOfMemory, "Out of memory error");
     302           0 :             this->m_nBlocksToLoad = 0;
     303           0 :             return -1;
     304             :         }
     305             : 
     306          96 :         if (this->m_nBlocksToLoad > 1)
     307             :         {
     308           5 :             const int l_nThreads = std::min(this->m_nBlocksToLoad, nMaxThreads);
     309             :             CPLJoinableThread **pahThreads =
     310           5 :                 (CPLJoinableThread **)VSI_CALLOC_VERBOSE(
     311             :                     sizeof(CPLJoinableThread *), l_nThreads);
     312           5 :             if (pahThreads == nullptr)
     313             :             {
     314           0 :                 this->m_nBlocksToLoad = 0;
     315           0 :                 return -1;
     316             :             }
     317             :             int i;
     318             : 
     319           5 :             CPLDebug(CODEC::debugId(), "%d blocks to load (%d threads)",
     320             :                      this->m_nBlocksToLoad, l_nThreads);
     321             : 
     322           5 :             oJob.poGDS_ = this;
     323           5 :             oJob.nBand = poBand->GetBand();
     324           5 :             oJob.nCurPair = -1;
     325           5 :             if (nBandCount > 0)
     326             :             {
     327           2 :                 oJob.nBandCount = nBandCount;
     328           2 :                 oJob.panBandMap = panBandMap;
     329             :             }
     330             :             else
     331             :             {
     332           3 :                 if (nReqMem <= GDALGetCacheMax64() / nBands)
     333             :                 {
     334           3 :                     oJob.nBandCount = nBands;
     335           3 :                     oJob.panBandMap = nullptr;
     336             :                 }
     337             :                 else
     338             :                 {
     339           0 :                     bRet = FALSE;
     340           0 :                     oJob.nBandCount = 1;
     341           0 :                     oJob.panBandMap = &oJob.nBand;
     342             :                 }
     343             :             }
     344           5 :             oJob.bSuccess = true;
     345             : 
     346             :             /* Flushes all dirty blocks from cache to disk to avoid them */
     347             :             /* to be flushed randomly, and simultaneously, from our worker
     348             :              * threads, */
     349             :             /* which might cause races in the output driver. */
     350             :             /* This is a workaround to a design defect of the block cache */
     351           5 :             GDALRasterBlock::FlushDirtyBlocks();
     352             : 
     353          22 :             for (i = 0; i < l_nThreads; i++)
     354             :             {
     355          34 :                 pahThreads[i] =
     356          17 :                     CPLCreateJoinableThread(ReadBlockInThread, &oJob);
     357          17 :                 if (pahThreads[i] == nullptr)
     358           0 :                     oJob.bSuccess = false;
     359             :             }
     360           5 :             TemporarilyDropReadWriteLock();
     361          22 :             for (i = 0; i < l_nThreads; i++)
     362          17 :                 CPLJoinThread(pahThreads[i]);
     363           5 :             ReacquireReadWriteLock();
     364           5 :             CPLFree(pahThreads);
     365           5 :             if (!oJob.bSuccess)
     366             :             {
     367           0 :                 this->m_nBlocksToLoad = 0;
     368           0 :                 return -1;
     369             :             }
     370           5 :             this->m_nBlocksToLoad = 0;
     371             :         }
     372             :     }
     373             : 
     374         240 :     return bRet;
     375             : }
     376             : 
     377             : /************************************************************************/
     378             : /*                      GetEstimatedRAMUsage()                          */
     379             : /************************************************************************/
     380             : 
     381             : template <typename CODEC, typename BASE>
     382         100 : GIntBig JP2OPJLikeDataset<CODEC, BASE>::GetEstimatedRAMUsage()
     383             : {
     384             :     // libopenjp2 holds the code block values in a uint32_t array.
     385         100 :     GIntBig nVal = static_cast<GIntBig>(this->m_nTileWidth) *
     386         100 :                    this->m_nTileHeight * this->nBands * sizeof(uint32_t);
     387         100 :     if (this->bSingleTiled)
     388             :     {
     389             :         // libopenjp2 ingests the codestream for a whole tile. So for a
     390             :         // single-tiled image, this is roughly the size of the file.
     391         100 :         const auto nCurPos = VSIFTellL(this->fp_);
     392         100 :         VSIFSeekL(this->fp_, 0, SEEK_END);
     393         100 :         nVal += VSIFTellL(this->fp_);
     394         100 :         VSIFSeekL(this->fp_, nCurPos, SEEK_SET);
     395             :     }
     396         100 :     CPLDebug(CODEC::debugId(), "Estimated RAM usage for %s: %.2f GB",
     397         100 :              GetDescription(), static_cast<double>(nVal * 1e-9));
     398         100 :     return nVal;
     399             : }
     400             : 
     401             : /************************************************************************/
     402             : /*                             IRasterIO()                              */
     403             : /************************************************************************/
     404             : 
     405             : template <typename CODEC, typename BASE>
     406          15 : CPLErr JP2OPJLikeDataset<CODEC, BASE>::IRasterIO(
     407             :     GDALRWFlag eRWFlag, int nXOff, int nYOff, int nXSize, int nYSize,
     408             :     void *pData, int nBufXSize, int nBufYSize, GDALDataType eBufType,
     409             :     int nBandCount, BANDMAP_TYPE panBandMap, GSpacing nPixelSpace,
     410             :     GSpacing nLineSpace, GSpacing nBandSpace, GDALRasterIOExtraArg *psExtraArg)
     411             : {
     412          15 :     if (eRWFlag != GF_Read)
     413           0 :         return CE_Failure;
     414             : 
     415          15 :     if (nBandCount < 1)
     416           0 :         return CE_Failure;
     417             : 
     418             :     auto poBand =
     419          15 :         (JP2OPJLikeRasterBand<CODEC, BASE> *)GetRasterBand(panBandMap[0]);
     420             : 
     421             :     /* ==================================================================== */
     422             :     /*      Do we have overviews that would be appropriate to satisfy       */
     423             :     /*      this request?                                                   */
     424             :     /* ==================================================================== */
     425             : 
     426          16 :     if ((nBufXSize < nXSize || nBufYSize < nYSize) &&
     427           1 :         poBand->GetOverviewCount() > 0)
     428             :     {
     429             :         int bTried;
     430           1 :         CPLErr eErr = TryOverviewRasterIO(
     431             :             eRWFlag, nXOff, nYOff, nXSize, nYSize, pData, nBufXSize, nBufYSize,
     432             :             eBufType, nBandCount, panBandMap, nPixelSpace, nLineSpace,
     433             :             nBandSpace, psExtraArg, &bTried);
     434           1 :         if (bTried)
     435           1 :             return eErr;
     436             :     }
     437             : 
     438          14 :     this->bEnoughMemoryToLoadOtherBands = PreloadBlocks(
     439             :         poBand, nXOff, nYOff, nXSize, nYSize, nBandCount, panBandMap);
     440             : 
     441          14 :     CPLErr eErr = GDALPamDataset::IRasterIO(
     442             :         eRWFlag, nXOff, nYOff, nXSize, nYSize, pData, nBufXSize, nBufYSize,
     443             :         eBufType, nBandCount, panBandMap, nPixelSpace, nLineSpace, nBandSpace,
     444             :         psExtraArg);
     445             : 
     446          14 :     this->bEnoughMemoryToLoadOtherBands = TRUE;
     447          14 :     return eErr;
     448             : }
     449             : 
     450             : /************************************************************************/
     451             : /*                          IBuildOverviews()                           */
     452             : /************************************************************************/
     453             : 
     454             : template <typename CODEC, typename BASE>
     455           3 : CPLErr JP2OPJLikeDataset<CODEC, BASE>::IBuildOverviews(
     456             :     const char *pszResampling, int nOverviews, const int *panOverviewList,
     457             :     int nListBands, const int *panBandList, GDALProgressFunc pfnProgress,
     458             :     void *pProgressData, CSLConstList papszOptions)
     459             : 
     460             : {
     461             :     // In order for building external overviews to work properly, we
     462             :     // discard any concept of internal overviews when the user
     463             :     // first requests to build external overviews.
     464           5 :     for (int i = 0; i < this->nOverviewCount; i++)
     465             :     {
     466           2 :         delete papoOverviewDS[i];
     467             :     }
     468           3 :     CPLFree(papoOverviewDS);
     469           3 :     papoOverviewDS = nullptr;
     470           3 :     this->nOverviewCount = 0;
     471             : 
     472           3 :     return GDALPamDataset::IBuildOverviews(
     473             :         pszResampling, nOverviews, panOverviewList, nListBands, panBandList,
     474           3 :         pfnProgress, pProgressData, papszOptions);
     475             : }
     476             : 
     477             : /************************************************************************/
     478             : /*                             ReadBlock()                              */
     479             : /************************************************************************/
     480             : 
     481             : template <typename CODEC, typename BASE>
     482         391 : CPLErr JP2OPJLikeDataset<CODEC, BASE>::ReadBlock(int nBand, VSILFILE *fpIn,
     483             :                                                  int nBlockXOff, int nBlockYOff,
     484             :                                                  void *pImage, int nBandCount,
     485             :                                                  const int *panBandMap)
     486             : {
     487         391 :     CPLErr eErr = CE_None;
     488         391 :     CODEC localctx;
     489             : 
     490         391 :     auto poBand = (JP2OPJLikeRasterBand<CODEC, BASE> *)GetRasterBand(nBand);
     491         391 :     int nBlockXSize = poBand->nBlockXSize;
     492         391 :     int nBlockYSize = poBand->nBlockYSize;
     493         391 :     GDALDataType eDataType = poBand->eDataType;
     494             : 
     495         391 :     int nDataTypeSize = (GDALGetDataTypeSize(eDataType) / 8);
     496             : 
     497         391 :     int nTileNumber = nBlockXOff + nBlockYOff * poBand->nBlocksPerRow;
     498         391 :     const int nWidthToRead =
     499         391 :         std::min(nBlockXSize, nRasterXSize - nBlockXOff * nBlockXSize);
     500         391 :     const int nHeightToRead =
     501         391 :         std::min(nBlockYSize, nRasterYSize - nBlockYOff * nBlockYSize);
     502             : 
     503         391 :     eErr = this->readBlockInit(fpIn, &localctx, nBlockXOff, nBlockYOff,
     504             :                                this->nRasterXSize, this->nRasterYSize,
     505             :                                nBlockXSize, nBlockYSize, nTileNumber);
     506         391 :     if (eErr != CE_None)
     507           0 :         goto end;
     508             : 
     509         841 :     for (unsigned int iBand = 0; iBand < localctx.psImage->numcomps; iBand++)
     510             :     {
     511         450 :         if (localctx.psImage->comps[iBand].data == nullptr)
     512             :         {
     513           0 :             CPLError(CE_Failure, CPLE_AppDefined,
     514             :                      "localctx.psImage->comps[%d].data == nullptr", iBand);
     515           0 :             eErr = CE_Failure;
     516           0 :             goto end;
     517             :         }
     518             :     }
     519             : 
     520         841 :     for (int xBand = 0; xBand < nBandCount; xBand++)
     521             :     {
     522         450 :         GDALRasterBlock *poBlock = nullptr;
     523         450 :         int iBand = (panBandMap) ? panBandMap[xBand] : xBand + 1;
     524         450 :         int bPromoteTo8Bit =
     525         450 :             ((JP2OPJLikeRasterBand<CODEC, BASE> *)GetRasterBand(iBand))
     526             :                 ->bPromoteTo8Bit;
     527             : 
     528         450 :         void *pDstBuffer = nullptr;
     529         450 :         if (iBand == nBand)
     530         391 :             pDstBuffer = pImage;
     531             :         else
     532             :         {
     533          59 :             AcquireMutex();
     534         118 :             poBlock =
     535          59 :                 ((JP2OPJLikeRasterBand<CODEC, BASE> *)GetRasterBand(iBand))
     536             :                     ->TryGetLockedBlockRef(nBlockXOff, nBlockYOff);
     537          59 :             if (poBlock != nullptr)
     538             :             {
     539           0 :                 ReleaseMutex();
     540           0 :                 poBlock->DropLock();
     541           0 :                 continue;
     542             :             }
     543             : 
     544          59 :             poBlock = GetRasterBand(iBand)->GetLockedBlockRef(nBlockXOff,
     545             :                                                               nBlockYOff, TRUE);
     546          59 :             ReleaseMutex();
     547          59 :             if (poBlock == nullptr)
     548             :             {
     549           0 :                 continue;
     550             :             }
     551             : 
     552          59 :             pDstBuffer = poBlock->GetDataRef();
     553             :         }
     554             : 
     555         450 :         if (this->bIs420)
     556             :         {
     557           7 :             if ((int)localctx.psImage->comps[0].w < nWidthToRead ||
     558           7 :                 (int)localctx.psImage->comps[0].h < nHeightToRead ||
     559           7 :                 localctx.psImage->comps[1].w !=
     560           7 :                     (localctx.psImage->comps[0].w + 1) / 2 ||
     561           7 :                 localctx.psImage->comps[1].h !=
     562           7 :                     (localctx.psImage->comps[0].h + 1) / 2 ||
     563           7 :                 localctx.psImage->comps[2].w !=
     564           7 :                     (localctx.psImage->comps[0].w + 1) / 2 ||
     565           7 :                 localctx.psImage->comps[2].h !=
     566           7 :                     (localctx.psImage->comps[0].h + 1) / 2 ||
     567           7 :                 (nBands == 4 &&
     568           4 :                  ((int)localctx.psImage->comps[3].w < nWidthToRead ||
     569           4 :                   (int)localctx.psImage->comps[3].h < nHeightToRead)))
     570             :             {
     571           0 :                 CPLError(CE_Failure, CPLE_AssertionFailed,
     572             :                          "Assertion at line %d of %s failed", __LINE__,
     573             :                          __FILE__);
     574           0 :                 if (poBlock != nullptr)
     575           0 :                     poBlock->DropLock();
     576           0 :                 eErr = CE_Failure;
     577           0 :                 goto end;
     578             :             }
     579             : 
     580           7 :             GByte *pDst = (GByte *)pDstBuffer;
     581           7 :             if (iBand == 4)
     582             :             {
     583           1 :                 const auto pSrcA = localctx.psImage->comps[3].data;
     584         151 :                 for (GPtrDiff_t j = 0; j < nHeightToRead; j++)
     585             :                 {
     586         300 :                     memcpy(pDst + j * nBlockXSize,
     587         150 :                            pSrcA + j * localctx.stride(localctx.psImage->comps),
     588             :                            nWidthToRead);
     589             :                 }
     590             :             }
     591             :             else
     592             :             {
     593           6 :                 const auto pSrcY = localctx.psImage->comps[0].data;
     594           6 :                 const auto pSrcCb = localctx.psImage->comps[1].data;
     595           6 :                 const auto pSrcCr = localctx.psImage->comps[2].data;
     596         606 :                 for (GPtrDiff_t j = 0; j < nHeightToRead; j++)
     597             :                 {
     598       81000 :                     for (int i = 0; i < nWidthToRead; i++)
     599             :                     {
     600       80400 :                         int Y =
     601       80400 :                             pSrcY[j * localctx.stride(localctx.psImage->comps) +
     602       80400 :                                   i];
     603       80400 :                         int Cb =
     604      160800 :                             pSrcCb[(j / 2) * localctx.stride(
     605       80400 :                                                  localctx.psImage->comps + 1) +
     606       80400 :                                    (i / 2)];
     607       80400 :                         int Cr =
     608      160800 :                             pSrcCr[(j / 2) * localctx.stride(
     609       80400 :                                                  localctx.psImage->comps + 2) +
     610       80400 :                                    (i / 2)];
     611       80400 :                         if (iBand == 1)
     612       26800 :                             pDst[j * nBlockXSize + i] =
     613       26800 :                                 CLAMP_0_255((int)(Y + 1.402 * (Cr - 128)));
     614       53600 :                         else if (iBand == 2)
     615       26800 :                             pDst[j * nBlockXSize + i] =
     616       26800 :                                 CLAMP_0_255((int)(Y - 0.34414 * (Cb - 128) -
     617       26800 :                                                   0.71414 * (Cr - 128)));
     618       26800 :                         else if (iBand == 3)
     619       26800 :                             pDst[j * nBlockXSize + i] =
     620       26800 :                                 CLAMP_0_255((int)(Y + 1.772 * (Cb - 128)));
     621             :                     }
     622             :                 }
     623             :             }
     624             : 
     625           7 :             if (bPromoteTo8Bit)
     626             :             {
     627           0 :                 for (GPtrDiff_t j = 0; j < nHeightToRead; j++)
     628             :                 {
     629           0 :                     for (int i = 0; i < nWidthToRead; i++)
     630             :                     {
     631           0 :                         pDst[j * nBlockXSize + i] *= 255;
     632             :                     }
     633             :                 }
     634             :             }
     635             :         }
     636             :         else
     637             :         {
     638         443 :             if ((int)localctx.psImage->comps[iBand - 1].w < nWidthToRead ||
     639         443 :                 (int)localctx.psImage->comps[iBand - 1].h < nHeightToRead)
     640             :             {
     641           0 :                 CPLError(CE_Failure, CPLE_AssertionFailed,
     642             :                          "Assertion at line %d of %s failed", __LINE__,
     643             :                          __FILE__);
     644           0 :                 if (poBlock != nullptr)
     645           0 :                     poBlock->DropLock();
     646           0 :                 eErr = CE_Failure;
     647           0 :                 goto end;
     648             :             }
     649             : 
     650         443 :             if (bPromoteTo8Bit)
     651             :             {
     652         529 :                 for (GPtrDiff_t j = 0; j < nHeightToRead; j++)
     653             :                 {
     654       79500 :                     for (int i = 0; i < nWidthToRead; i++)
     655             :                     {
     656      157950 :                         localctx.psImage->comps[iBand - 1]
     657      157950 :                             .data[j * localctx.stride(localctx.psImage->comps +
     658       78975 :                                                       iBand - 1) +
     659       78975 :                                   i] *= 255;
     660             :                     }
     661             :                 }
     662             :             }
     663             : 
     664         443 :             if ((int)localctx.stride(localctx.psImage->comps + iBand - 1) ==
     665         857 :                     nBlockXSize &&
     666         414 :                 (int)localctx.psImage->comps[iBand - 1].h == nBlockYSize)
     667             :             {
     668         393 :                 GDALCopyWords64(
     669         393 :                     localctx.psImage->comps[iBand - 1].data, GDT_Int32, 4,
     670             :                     pDstBuffer, eDataType, nDataTypeSize,
     671         393 :                     static_cast<GPtrDiff_t>(nBlockXSize) * nBlockYSize);
     672             :             }
     673             :             else
     674             :             {
     675       12249 :                 for (GPtrDiff_t j = 0; j < nHeightToRead; j++)
     676             :                 {
     677       24398 :                     GDALCopyWords(
     678       24398 :                         localctx.psImage->comps[iBand - 1].data +
     679       12199 :                             j * localctx.stride(localctx.psImage->comps +
     680       12199 :                                                 iBand - 1),
     681             :                         GDT_Int32, 4,
     682       12199 :                         (GByte *)pDstBuffer + j * nBlockXSize * nDataTypeSize,
     683             :                         eDataType, nDataTypeSize, nWidthToRead);
     684             :                 }
     685             :             }
     686             :         }
     687             : 
     688         450 :         if (poBlock != nullptr)
     689          59 :             poBlock->DropLock();
     690             :     }
     691             : 
     692         391 : end:
     693         391 :     this->cache(&localctx);
     694             : 
     695         781 :     return eErr;
     696             : }
     697             : 
     698             : /************************************************************************/
     699             : /*                         GetOverviewCount()                           */
     700             : /************************************************************************/
     701             : 
     702             : template <typename CODEC, typename BASE>
     703         119 : int JP2OPJLikeRasterBand<CODEC, BASE>::GetOverviewCount()
     704             : {
     705         119 :     auto poGDS = cpl::down_cast<JP2OPJLikeDataset<CODEC, BASE> *>(poDS);
     706         119 :     if (!poGDS->AreOverviewsEnabled())
     707           0 :         return 0;
     708             : 
     709         119 :     if (GDALPamRasterBand::GetOverviewCount() > 0)
     710           4 :         return GDALPamRasterBand::GetOverviewCount();
     711             : 
     712         115 :     return poGDS->nOverviewCount;
     713             : }
     714             : 
     715             : /************************************************************************/
     716             : /*                            GetOverview()                             */
     717             : /************************************************************************/
     718             : 
     719             : template <typename CODEC, typename BASE>
     720          22 : GDALRasterBand *JP2OPJLikeRasterBand<CODEC, BASE>::GetOverview(int iOvrLevel)
     721             : {
     722          22 :     if (GDALPamRasterBand::GetOverviewCount() > 0)
     723           6 :         return GDALPamRasterBand::GetOverview(iOvrLevel);
     724             : 
     725          16 :     auto poGDS = (JP2OPJLikeDataset<CODEC, BASE> *)poDS;
     726          16 :     if (iOvrLevel < 0 || iOvrLevel >= poGDS->nOverviewCount)
     727           0 :         return nullptr;
     728             : 
     729          16 :     return poGDS->papoOverviewDS[iOvrLevel]->GetRasterBand(nBand);
     730             : }
     731             : 
     732             : /************************************************************************/
     733             : /*                       GetColorInterpretation()                       */
     734             : /************************************************************************/
     735             : 
     736             : template <typename CODEC, typename BASE>
     737         570 : GDALColorInterp JP2OPJLikeRasterBand<CODEC, BASE>::GetColorInterpretation()
     738             : {
     739         570 :     auto poGDS = (JP2OPJLikeDataset<CODEC, BASE> *)poDS;
     740             : 
     741         570 :     if (poCT)
     742           6 :         return GCI_PaletteIndex;
     743             : 
     744         564 :     if (nBand == poGDS->nAlphaIndex + 1)
     745          22 :         return GCI_AlphaBand;
     746             : 
     747         949 :     if (poGDS->nBands <= 2 &&
     748         407 :         poGDS->eColorSpace == CODEC::cvtenum(JP2_CLRSPC_GRAY))
     749         386 :         return GCI_GrayIndex;
     750         247 :     else if (poGDS->eColorSpace == CODEC::cvtenum(JP2_CLRSPC_SRGB) ||
     751          91 :              poGDS->eColorSpace == CODEC::cvtenum(JP2_CLRSPC_SYCC))
     752             :     {
     753          77 :         if (nBand == poGDS->nRedIndex + 1)
     754          26 :             return GCI_RedBand;
     755          51 :         if (nBand == poGDS->nGreenIndex + 1)
     756          25 :             return GCI_GreenBand;
     757          26 :         if (nBand == poGDS->nBlueIndex + 1)
     758          25 :             return GCI_BlueBand;
     759             :     }
     760             : 
     761          80 :     return GCI_Undefined;
     762             : }
     763             : 
     764             : /************************************************************************/
     765             : /* ==================================================================== */
     766             : /*                           JP2OPJLikeDataset                          */
     767             : /* ==================================================================== */
     768             : /************************************************************************/
     769             : 
     770             : /************************************************************************/
     771             : /*                        JP2OPJLikeDataset()                           */
     772             : /************************************************************************/
     773             : 
     774             : template <typename CODEC, typename BASE>
     775        1902 : JP2OPJLikeDataset<CODEC, BASE>::JP2OPJLikeDataset()
     776             : {
     777        1902 :     this->init();
     778        1902 : }
     779             : 
     780             : /************************************************************************/
     781             : /*                         ~JP2OPJLikeDataset()                         */
     782             : /************************************************************************/
     783             : 
     784             : template <typename CODEC, typename BASE>
     785        2778 : JP2OPJLikeDataset<CODEC, BASE>::~JP2OPJLikeDataset()
     786             : 
     787             : {
     788        1902 :     JP2OPJLikeDataset::Close();
     789        1902 :     this->deinit();
     790        4680 : }
     791             : 
     792             : /************************************************************************/
     793             : /*                              Close()                                 */
     794             : /************************************************************************/
     795             : 
     796             : template <typename CODEC, typename BASE>
     797        2594 : CPLErr JP2OPJLikeDataset<CODEC, BASE>::Close()
     798             : {
     799        2594 :     CPLErr eErr = CE_None;
     800        2594 :     if (nOpenFlags != OPEN_FLAGS_CLOSED)
     801             :     {
     802        1902 :         if (JP2OPJLikeDataset::FlushCache(true) != CE_None)
     803           0 :             eErr = CE_Failure;
     804             : 
     805        1902 :         this->closeJP2();
     806        1902 :         if (this->iLevel == 0 && this->fp_ != nullptr)
     807             :         {
     808         759 :             if (this->bRewrite)
     809             :             {
     810          18 :                 GDALJP2Box oBox(this->fp_);
     811           9 :                 vsi_l_offset nOffsetJP2C = 0;
     812           9 :                 vsi_l_offset nLengthJP2C = 0;
     813           9 :                 vsi_l_offset nOffsetXML = 0;
     814           9 :                 vsi_l_offset nOffsetASOC = 0;
     815           9 :                 vsi_l_offset nOffsetUUID = 0;
     816           9 :                 vsi_l_offset nOffsetIHDR = 0;
     817           9 :                 vsi_l_offset nLengthIHDR = 0;
     818           9 :                 int bMSIBox = FALSE;
     819           9 :                 int bGMLData = FALSE;
     820           9 :                 int bUnsupportedConfiguration = FALSE;
     821           9 :                 if (oBox.ReadFirst())
     822             :                 {
     823          47 :                     while (strlen(oBox.GetType()) > 0)
     824             :                     {
     825          47 :                         if (EQUAL(oBox.GetType(), "jp2c"))
     826             :                         {
     827           9 :                             if (nOffsetJP2C == 0)
     828             :                             {
     829           9 :                                 nOffsetJP2C = VSIFTellL(this->fp_);
     830           9 :                                 nLengthJP2C = oBox.GetDataLength();
     831             :                             }
     832             :                             else
     833           0 :                                 bUnsupportedConfiguration = TRUE;
     834             :                         }
     835          38 :                         else if (EQUAL(oBox.GetType(), "jp2h"))
     836             :                         {
     837          18 :                             GDALJP2Box oSubBox(this->fp_);
     838          18 :                             if (oSubBox.ReadFirstChild(&oBox) &&
     839           9 :                                 EQUAL(oSubBox.GetType(), "ihdr"))
     840             :                             {
     841           9 :                                 nOffsetIHDR = VSIFTellL(this->fp_);
     842           9 :                                 nLengthIHDR = oSubBox.GetDataLength();
     843             :                             }
     844             :                         }
     845          29 :                         else if (EQUAL(oBox.GetType(), "xml "))
     846             :                         {
     847           2 :                             if (nOffsetXML == 0)
     848           2 :                                 nOffsetXML = VSIFTellL(this->fp_);
     849             :                         }
     850          27 :                         else if (EQUAL(oBox.GetType(), "asoc"))
     851             :                         {
     852           3 :                             if (nOffsetASOC == 0)
     853           3 :                                 nOffsetASOC = VSIFTellL(this->fp_);
     854             : 
     855           6 :                             GDALJP2Box oSubBox(this->fp_);
     856           6 :                             if (oSubBox.ReadFirstChild(&oBox) &&
     857           3 :                                 EQUAL(oSubBox.GetType(), "lbl "))
     858             :                             {
     859           3 :                                 char *pszLabel = (char *)oSubBox.ReadBoxData();
     860           3 :                                 if (pszLabel != nullptr &&
     861           3 :                                     EQUAL(pszLabel, "gml.data"))
     862             :                                 {
     863           3 :                                     bGMLData = TRUE;
     864             :                                 }
     865             :                                 else
     866           0 :                                     bUnsupportedConfiguration = TRUE;
     867           3 :                                 CPLFree(pszLabel);
     868             :                             }
     869             :                             else
     870           0 :                                 bUnsupportedConfiguration = TRUE;
     871             :                         }
     872          24 :                         else if (EQUAL(oBox.GetType(), "uuid"))
     873             :                         {
     874           4 :                             if (nOffsetUUID == 0)
     875           4 :                                 nOffsetUUID = VSIFTellL(this->fp_);
     876           4 :                             if (GDALJP2Metadata::IsUUID_MSI(oBox.GetUUID()))
     877           4 :                                 bMSIBox = TRUE;
     878           0 :                             else if (!GDALJP2Metadata::IsUUID_XMP(
     879             :                                          oBox.GetUUID()))
     880           0 :                                 bUnsupportedConfiguration = TRUE;
     881             :                         }
     882          20 :                         else if (!EQUAL(oBox.GetType(), "jP  ") &&
     883          11 :                                  !EQUAL(oBox.GetType(), "ftyp") &&
     884           2 :                                  !EQUAL(oBox.GetType(), "rreq") &&
     885          31 :                                  !EQUAL(oBox.GetType(), "jp2h") &&
     886           0 :                                  !EQUAL(oBox.GetType(), "jp2i"))
     887             :                         {
     888           0 :                             bUnsupportedConfiguration = TRUE;
     889             :                         }
     890             : 
     891          47 :                         if (bUnsupportedConfiguration || !oBox.ReadNext())
     892           9 :                             break;
     893             :                     }
     894             :                 }
     895             : 
     896             :                 const char *pszGMLJP2;
     897           9 :                 int bGeoreferencingCompatOfGMLJP2 =
     898           9 :                     (!m_oSRS.IsEmpty() && bGeoTransformValid && nGCPCount == 0);
     899           9 :                 if (bGeoreferencingCompatOfGMLJP2 &&
     900           3 :                     ((this->bHasGeoreferencingAtOpening && bGMLData) ||
     901           2 :                      (!this->bHasGeoreferencingAtOpening)))
     902           2 :                     pszGMLJP2 = "GMLJP2=YES";
     903             :                 else
     904           7 :                     pszGMLJP2 = "GMLJP2=NO";
     905             : 
     906             :                 const char *pszGeoJP2;
     907           9 :                 int bGeoreferencingCompatOfGeoJP2 =
     908           9 :                     (!m_oSRS.IsEmpty() || nGCPCount != 0 || bGeoTransformValid);
     909           9 :                 if (bGeoreferencingCompatOfGeoJP2 &&
     910           6 :                     ((this->bHasGeoreferencingAtOpening && bMSIBox) ||
     911           4 :                      (!this->bHasGeoreferencingAtOpening) ||
     912           2 :                      this->nGCPCount > 0))
     913           5 :                     pszGeoJP2 = "GeoJP2=YES";
     914             :                 else
     915           4 :                     pszGeoJP2 = "GeoJP2=NO";
     916             : 
     917             :                 /* Test that the length of the JP2C box is not 0 */
     918           9 :                 int bJP2CBoxOKForRewriteInPlace = TRUE;
     919           9 :                 if (nOffsetJP2C > 16 && !bUnsupportedConfiguration)
     920             :                 {
     921           9 :                     VSIFSeekL(this->fp_, nOffsetJP2C - 8, SEEK_SET);
     922             :                     GByte abyBuffer[8];
     923           9 :                     VSIFReadL(abyBuffer, 1, 8, this->fp_);
     924           9 :                     if (STARTS_WITH_CI((const char *)abyBuffer + 4, "jp2c") &&
     925           9 :                         abyBuffer[0] == 0 && abyBuffer[1] == 0 &&
     926           9 :                         abyBuffer[2] == 0 && abyBuffer[3] == 0)
     927             :                     {
     928           1 :                         if ((vsi_l_offset)(GUInt32)(nLengthJP2C + 8) ==
     929           1 :                             (nLengthJP2C + 8))
     930             :                         {
     931           1 :                             CPLDebug(
     932             :                                 CODEC::debugId(),
     933             :                                 "Patching length of JP2C box with real length");
     934           1 :                             VSIFSeekL(this->fp_, nOffsetJP2C - 8, SEEK_SET);
     935           1 :                             GUInt32 nLength = (GUInt32)nLengthJP2C + 8;
     936           1 :                             CPL_MSBPTR32(&nLength);
     937           1 :                             if (VSIFWriteL(&nLength, 1, 4, this->fp_) != 1)
     938           1 :                                 eErr = CE_Failure;
     939             :                         }
     940             :                         else
     941           0 :                             bJP2CBoxOKForRewriteInPlace = FALSE;
     942             :                     }
     943             :                 }
     944             : 
     945           9 :                 if (nOffsetJP2C == 0 || bUnsupportedConfiguration)
     946             :                 {
     947           0 :                     eErr = CE_Failure;
     948           0 :                     CPLError(CE_Failure, CPLE_AppDefined,
     949             :                              "Cannot rewrite file due to unsupported JP2 box "
     950             :                              "configuration");
     951           0 :                     VSIFCloseL(this->fp_);
     952             :                 }
     953           9 :                 else if (bJP2CBoxOKForRewriteInPlace &&
     954           9 :                          (nOffsetXML == 0 || nOffsetXML > nOffsetJP2C) &&
     955           9 :                          (nOffsetASOC == 0 || nOffsetASOC > nOffsetJP2C) &&
     956           4 :                          (nOffsetUUID == 0 || nOffsetUUID > nOffsetJP2C))
     957             :                 {
     958           5 :                     CPLDebug(CODEC::debugId(),
     959             :                              "Rewriting boxes after codestream");
     960             : 
     961             :                     /* Update IPR flag */
     962           5 :                     if (nLengthIHDR == 14)
     963             :                     {
     964           5 :                         VSIFSeekL(this->fp_, nOffsetIHDR + nLengthIHDR - 1,
     965             :                                   SEEK_SET);
     966           5 :                         GByte bIPR = GetMetadata("xml:IPR") != nullptr;
     967           5 :                         if (VSIFWriteL(&bIPR, 1, 1, this->fp_) != 1)
     968           0 :                             eErr = CE_Failure;
     969             :                     }
     970             : 
     971           5 :                     VSIFSeekL(this->fp_, nOffsetJP2C + nLengthJP2C, SEEK_SET);
     972             : 
     973          10 :                     GDALJP2Metadata oJP2MD;
     974           5 :                     if (GetGCPCount() > 0)
     975             :                     {
     976           1 :                         oJP2MD.SetGCPs(GetGCPCount(), GetGCPs());
     977           1 :                         oJP2MD.SetSpatialRef(GetGCPSpatialRef());
     978             :                     }
     979             :                     else
     980             :                     {
     981           4 :                         const OGRSpatialReference *poSRS = GetSpatialRef();
     982           4 :                         if (poSRS != nullptr)
     983             :                         {
     984           1 :                             oJP2MD.SetSpatialRef(poSRS);
     985             :                         }
     986           4 :                         if (bGeoTransformValid)
     987             :                         {
     988           1 :                             oJP2MD.SetGeoTransform(adfGeoTransform);
     989             :                         }
     990             :                     }
     991             : 
     992             :                     const char *pszAreaOrPoint =
     993           5 :                         GetMetadataItem(GDALMD_AREA_OR_POINT);
     994           5 :                     oJP2MD.bPixelIsPoint =
     995           5 :                         pszAreaOrPoint != nullptr &&
     996           0 :                         EQUAL(pszAreaOrPoint, GDALMD_AOP_POINT);
     997             : 
     998           5 :                     if (!WriteIPRBox(this->fp_, this))
     999           0 :                         eErr = CE_Failure;
    1000             : 
    1001           5 :                     if (bGeoreferencingCompatOfGMLJP2 &&
    1002           1 :                         EQUAL(pszGMLJP2, "GMLJP2=YES"))
    1003             :                     {
    1004             :                         GDALJP2Box *poBox =
    1005           1 :                             oJP2MD.CreateGMLJP2(nRasterXSize, nRasterYSize);
    1006           1 :                         if (!WriteBox(this->fp_, poBox))
    1007           0 :                             eErr = CE_Failure;
    1008           1 :                         delete poBox;
    1009             :                     }
    1010             : 
    1011          10 :                     if (!WriteXMLBoxes(this->fp_, this) ||
    1012           5 :                         !WriteGDALMetadataBox(this->fp_, this, nullptr))
    1013           0 :                         eErr = CE_Failure;
    1014             : 
    1015           5 :                     if (bGeoreferencingCompatOfGeoJP2 &&
    1016           2 :                         EQUAL(pszGeoJP2, "GeoJP2=YES"))
    1017             :                     {
    1018           2 :                         GDALJP2Box *poBox = oJP2MD.CreateJP2GeoTIFF();
    1019           2 :                         if (!WriteBox(this->fp_, poBox))
    1020           0 :                             eErr = CE_Failure;
    1021           2 :                         delete poBox;
    1022             :                     }
    1023             : 
    1024           5 :                     if (!WriteXMPBox(this->fp_, this))
    1025           0 :                         eErr = CE_Failure;
    1026             : 
    1027           5 :                     if (VSIFTruncateL(this->fp_, VSIFTellL(this->fp_)) != 0)
    1028           0 :                         eErr = CE_Failure;
    1029             : 
    1030           5 :                     if (VSIFCloseL(this->fp_) != 0)
    1031           5 :                         eErr = CE_Failure;
    1032             :                 }
    1033             :                 else
    1034             :                 {
    1035           4 :                     VSIFCloseL(this->fp_);
    1036             : 
    1037           4 :                     CPLDebug(CODEC::debugId(), "Rewriting whole file");
    1038             : 
    1039           4 :                     const char *const apszOptions[] = {"USE_SRC_CODESTREAM=YES",
    1040             :                                                        "CODEC=JP2",
    1041             :                                                        "WRITE_METADATA=YES",
    1042             :                                                        pszGMLJP2,
    1043             :                                                        pszGeoJP2,
    1044             :                                                        nullptr};
    1045           8 :                     CPLString osTmpFilename(
    1046           4 :                         CPLSPrintf("%s.tmp", GetDescription()));
    1047           4 :                     GDALDataset *poOutDS = CreateCopy(
    1048             :                         osTmpFilename, this, FALSE, (char **)apszOptions,
    1049             :                         GDALDummyProgress, nullptr);
    1050           4 :                     if (poOutDS)
    1051             :                     {
    1052           4 :                         if (GDALClose(poOutDS) != CE_None)
    1053           0 :                             eErr = CE_Failure;
    1054           4 :                         if (VSIRename(osTmpFilename, GetDescription()) != 0)
    1055           0 :                             eErr = CE_Failure;
    1056             :                     }
    1057             :                     else
    1058             :                     {
    1059           0 :                         eErr = CE_Failure;
    1060           0 :                         VSIUnlink(osTmpFilename);
    1061             :                     }
    1062           4 :                     VSIUnlink(CPLSPrintf("%s.tmp.aux.xml", GetDescription()));
    1063             :                 }
    1064             :             }
    1065             :             else
    1066         750 :                 VSIFCloseL(this->fp_);
    1067             :         }
    1068             : 
    1069        1902 :         JP2OPJLikeDataset::CloseDependentDatasets();
    1070             : 
    1071        1902 :         if (GDALPamDataset::Close() != CE_None)
    1072           0 :             eErr = CE_Failure;
    1073             :     }
    1074        2594 :     return eErr;
    1075             : }
    1076             : 
    1077             : /************************************************************************/
    1078             : /*                      CloseDependentDatasets()                        */
    1079             : /************************************************************************/
    1080             : 
    1081             : template <typename CODEC, typename BASE>
    1082        1914 : int JP2OPJLikeDataset<CODEC, BASE>::CloseDependentDatasets()
    1083             : {
    1084        1914 :     int bRet = GDALJP2AbstractDataset::CloseDependentDatasets();
    1085        1914 :     if (papoOverviewDS)
    1086             :     {
    1087         178 :         for (int i = 0; i < this->nOverviewCount; i++)
    1088         115 :             delete papoOverviewDS[i];
    1089          63 :         CPLFree(papoOverviewDS);
    1090          63 :         papoOverviewDS = nullptr;
    1091          63 :         bRet = TRUE;
    1092             :     }
    1093        1914 :     return bRet;
    1094             : }
    1095             : 
    1096             : /************************************************************************/
    1097             : /*                           SetSpatialRef()                            */
    1098             : /************************************************************************/
    1099             : 
    1100             : template <typename CODEC, typename BASE>
    1101             : CPLErr
    1102          18 : JP2OPJLikeDataset<CODEC, BASE>::SetSpatialRef(const OGRSpatialReference *poSRS)
    1103             : {
    1104          18 :     if (eAccess == GA_Update)
    1105             :     {
    1106           2 :         this->bRewrite = TRUE;
    1107           2 :         m_oSRS.Clear();
    1108           2 :         if (poSRS)
    1109           1 :             m_oSRS = *poSRS;
    1110           2 :         return CE_None;
    1111             :     }
    1112             :     else
    1113          16 :         return GDALJP2AbstractDataset::SetSpatialRef(poSRS);
    1114             : }
    1115             : 
    1116             : /************************************************************************/
    1117             : /*                           SetGeoTransform()                          */
    1118             : /************************************************************************/
    1119             : 
    1120             : template <typename CODEC, typename BASE>
    1121          24 : CPLErr JP2OPJLikeDataset<CODEC, BASE>::SetGeoTransform(double *padfGeoTransform)
    1122             : {
    1123          24 :     if (eAccess == GA_Update)
    1124             :     {
    1125           4 :         this->bRewrite = TRUE;
    1126           4 :         memcpy(adfGeoTransform, padfGeoTransform, 6 * sizeof(double));
    1127           4 :         bGeoTransformValid =
    1128           6 :             !(adfGeoTransform[0] == 0.0 && adfGeoTransform[1] == 1.0 &&
    1129           2 :               adfGeoTransform[2] == 0.0 && adfGeoTransform[3] == 0.0 &&
    1130           1 :               adfGeoTransform[4] == 0.0 && adfGeoTransform[5] == 1.0);
    1131           4 :         return CE_None;
    1132             :     }
    1133             :     else
    1134          20 :         return GDALJP2AbstractDataset::SetGeoTransform(padfGeoTransform);
    1135             : }
    1136             : 
    1137             : /************************************************************************/
    1138             : /*                           SetGCPs()                                  */
    1139             : /************************************************************************/
    1140             : 
    1141             : template <typename CODEC, typename BASE>
    1142           4 : CPLErr JP2OPJLikeDataset<CODEC, BASE>::SetGCPs(int nGCPCountIn,
    1143             :                                                const GDAL_GCP *pasGCPListIn,
    1144             :                                                const OGRSpatialReference *poSRS)
    1145             : {
    1146           4 :     if (eAccess == GA_Update)
    1147             :     {
    1148           3 :         this->bRewrite = TRUE;
    1149           3 :         if (nGCPCount > 0)
    1150             :         {
    1151           1 :             GDALDeinitGCPs(nGCPCount, pasGCPList);
    1152           1 :             CPLFree(pasGCPList);
    1153             :         }
    1154             : 
    1155           3 :         m_oSRS.Clear();
    1156           3 :         if (poSRS)
    1157           2 :             m_oSRS = *poSRS;
    1158             : 
    1159           3 :         nGCPCount = nGCPCountIn;
    1160           3 :         pasGCPList = GDALDuplicateGCPs(nGCPCount, pasGCPListIn);
    1161             : 
    1162           3 :         return CE_None;
    1163             :     }
    1164             :     else
    1165           1 :         return GDALJP2AbstractDataset::SetGCPs(nGCPCountIn, pasGCPListIn,
    1166           1 :                                                poSRS);
    1167             : }
    1168             : 
    1169             : /************************************************************************/
    1170             : /*                            SetMetadata()                             */
    1171             : /************************************************************************/
    1172             : 
    1173             : template <typename CODEC, typename BASE>
    1174          11 : CPLErr JP2OPJLikeDataset<CODEC, BASE>::SetMetadata(char **papszMetadata,
    1175             :                                                    const char *pszDomain)
    1176             : {
    1177          11 :     if (eAccess == GA_Update)
    1178             :     {
    1179           2 :         this->bRewrite = TRUE;
    1180           2 :         if (pszDomain == nullptr || EQUAL(pszDomain, ""))
    1181             :         {
    1182           1 :             CSLDestroy(m_papszMainMD);
    1183           1 :             m_papszMainMD = CSLDuplicate(papszMetadata);
    1184             :         }
    1185           2 :         return GDALDataset::SetMetadata(papszMetadata, pszDomain);
    1186             :     }
    1187           9 :     return GDALJP2AbstractDataset::SetMetadata(papszMetadata, pszDomain);
    1188             : }
    1189             : 
    1190             : /************************************************************************/
    1191             : /*                            SetMetadata()                             */
    1192             : /************************************************************************/
    1193             : 
    1194             : template <typename CODEC, typename BASE>
    1195           3 : CPLErr JP2OPJLikeDataset<CODEC, BASE>::SetMetadataItem(const char *pszName,
    1196             :                                                        const char *pszValue,
    1197             :                                                        const char *pszDomain)
    1198             : {
    1199           3 :     if (eAccess == GA_Update)
    1200             :     {
    1201           1 :         this->bRewrite = TRUE;
    1202           1 :         if (pszDomain == nullptr || EQUAL(pszDomain, ""))
    1203             :         {
    1204           1 :             m_papszMainMD = CSLSetNameValue(GetMetadata(), pszName, pszValue);
    1205             :         }
    1206           1 :         return GDALDataset::SetMetadataItem(pszName, pszValue, pszDomain);
    1207             :     }
    1208           2 :     return GDALJP2AbstractDataset::SetMetadataItem(pszName, pszValue,
    1209           2 :                                                    pszDomain);
    1210             : }
    1211             : 
    1212             : /************************************************************************/
    1213             : /*                            Identify()                                */
    1214             : /************************************************************************/
    1215             : 
    1216             : #ifndef jpc_header_defined
    1217             : #define jpc_header_defined
    1218             : static const unsigned char jpc_header[] = {0xff, 0x4f, 0xff,
    1219             :                                            0x51};  // SOC + RSIZ markers
    1220             : static const unsigned char jp2_box_jp[] = {0x6a, 0x50, 0x20, 0x20}; /* 'jP  ' */
    1221             : #endif
    1222             : 
    1223             : template <typename CODEC, typename BASE>
    1224         766 : int JP2OPJLikeDataset<CODEC, BASE>::Identify(GDALOpenInfo *poOpenInfo)
    1225             : 
    1226             : {
    1227         766 :     if (poOpenInfo->nHeaderBytes >= 16 &&
    1228         766 :         (memcmp(poOpenInfo->pabyHeader, jpc_header, sizeof(jpc_header)) == 0 ||
    1229         683 :          memcmp(poOpenInfo->pabyHeader + 4, jp2_box_jp, sizeof(jp2_box_jp)) ==
    1230             :              0))
    1231         766 :         return TRUE;
    1232             : 
    1233             :     else
    1234           0 :         return FALSE;
    1235             : }
    1236             : 
    1237             : /************************************************************************/
    1238             : /*                        JP2FindCodeStream()                           */
    1239             : /************************************************************************/
    1240             : 
    1241         773 : static vsi_l_offset JP2FindCodeStream(VSILFILE *fp, vsi_l_offset *pnLength)
    1242             : {
    1243         773 :     vsi_l_offset nCodeStreamStart = 0;
    1244         773 :     vsi_l_offset nCodeStreamLength = 0;
    1245             : 
    1246         773 :     VSIFSeekL(fp, 0, SEEK_SET);
    1247             :     GByte abyHeader[16];
    1248         773 :     VSIFReadL(abyHeader, 1, 16, fp);
    1249             : 
    1250         773 :     if (memcmp(abyHeader, jpc_header, sizeof(jpc_header)) == 0)
    1251             :     {
    1252          83 :         VSIFSeekL(fp, 0, SEEK_END);
    1253          83 :         nCodeStreamLength = VSIFTellL(fp);
    1254             :     }
    1255         690 :     else if (memcmp(abyHeader + 4, jp2_box_jp, sizeof(jp2_box_jp)) == 0)
    1256             :     {
    1257             :         /* Find offset of first jp2c box */
    1258        1378 :         GDALJP2Box oBox(fp);
    1259         689 :         if (oBox.ReadFirst())
    1260             :         {
    1261        3585 :             while (strlen(oBox.GetType()) > 0)
    1262             :             {
    1263        3585 :                 if (EQUAL(oBox.GetType(), "jp2c"))
    1264             :                 {
    1265         688 :                     nCodeStreamStart = VSIFTellL(fp);
    1266         688 :                     nCodeStreamLength = oBox.GetDataLength();
    1267         688 :                     break;
    1268             :                 }
    1269             : 
    1270        2897 :                 if (!oBox.ReadNext())
    1271           1 :                     break;
    1272             :             }
    1273             :         }
    1274             :     }
    1275         773 :     *pnLength = nCodeStreamLength;
    1276         773 :     return nCodeStreamStart;
    1277             : }
    1278             : 
    1279             : /************************************************************************/
    1280             : /*                                Open()                                */
    1281             : /************************************************************************/
    1282             : 
    1283             : template <typename CODEC, typename BASE>
    1284         766 : GDALDataset *JP2OPJLikeDataset<CODEC, BASE>::Open(GDALOpenInfo *poOpenInfo)
    1285             : 
    1286             : {
    1287         766 :     if (!Identify(poOpenInfo) || poOpenInfo->fpL == nullptr)
    1288           0 :         return nullptr;
    1289             : 
    1290             :     /* Detect which codec to use : J2K or JP2 ? */
    1291         766 :     vsi_l_offset nCodeStreamLength = 0;
    1292             :     vsi_l_offset nCodeStreamStart =
    1293         766 :         JP2FindCodeStream(poOpenInfo->fpL, &nCodeStreamLength);
    1294             : 
    1295         766 :     if (nCodeStreamStart == 0 && nCodeStreamLength == 0)
    1296             :     {
    1297           1 :         CPLError(CE_Failure, CPLE_AppDefined, "No code-stream in JP2 file");
    1298           1 :         return nullptr;
    1299             :     }
    1300        1530 :     JP2OPJLikeDataset oTmpDS;
    1301         765 :     int numThreads = oTmpDS.GetNumThreads();
    1302         765 :     auto eCodecFormat = (nCodeStreamStart == 0) ? CODEC::cvtenum(JP2_CODEC_J2K)
    1303         682 :                                                 : CODEC::cvtenum(JP2_CODEC_JP2);
    1304             : 
    1305         765 :     uint32_t nTileW = 0, nTileH = 0;
    1306         765 :     int numResolutions = 0;
    1307        1530 :     CODEC localctx;
    1308         765 :     localctx.open(poOpenInfo->fpL, nCodeStreamStart);
    1309         765 :     if (!localctx.setUpDecompress(numThreads, nCodeStreamLength, &nTileW,
    1310             :                                   &nTileH, &numResolutions))
    1311           6 :         return nullptr;
    1312             : 
    1313         759 :     GDALDataType eDataType = GDT_Byte;
    1314         759 :     if (localctx.psImage->comps[0].prec > 16)
    1315             :     {
    1316           0 :         if (localctx.psImage->comps[0].sgnd)
    1317           0 :             eDataType = GDT_Int32;
    1318             :         else
    1319           0 :             eDataType = GDT_UInt32;
    1320             :     }
    1321         759 :     else if (localctx.psImage->comps[0].prec > 8)
    1322             :     {
    1323          26 :         if (localctx.psImage->comps[0].sgnd)
    1324           7 :             eDataType = GDT_Int16;
    1325             :         else
    1326          19 :             eDataType = GDT_UInt16;
    1327             :     }
    1328             : 
    1329         759 :     int bIs420 =
    1330        1518 :         (localctx.psImage->color_space != CODEC::cvtenum(JP2_CLRSPC_SRGB) &&
    1331         733 :          eDataType == GDT_Byte &&
    1332         733 :          (localctx.psImage->numcomps == 3 || localctx.psImage->numcomps == 4) &&
    1333          58 :          localctx.psImage->comps[1].w == localctx.psImage->comps[0].w / 2 &&
    1334           3 :          localctx.psImage->comps[1].h == localctx.psImage->comps[0].h / 2 &&
    1335           3 :          localctx.psImage->comps[2].w == localctx.psImage->comps[0].w / 2 &&
    1336        1521 :          localctx.psImage->comps[2].h == localctx.psImage->comps[0].h / 2) &&
    1337           3 :         (localctx.psImage->numcomps == 3 ||
    1338           2 :          (localctx.psImage->numcomps == 4 &&
    1339           2 :           localctx.psImage->comps[3].w == localctx.psImage->comps[0].w &&
    1340           2 :           localctx.psImage->comps[3].h == localctx.psImage->comps[0].h));
    1341             : 
    1342         759 :     if (bIs420)
    1343             :     {
    1344           3 :         CPLDebug(CODEC::debugId(), "420 format");
    1345             :     }
    1346             :     else
    1347             :     {
    1348             :         int iBand;
    1349         917 :         for (iBand = 2; iBand <= (int)localctx.psImage->numcomps; iBand++)
    1350             :         {
    1351         161 :             if (localctx.psImage->comps[iBand - 1].w !=
    1352         161 :                     localctx.psImage->comps[0].w ||
    1353         161 :                 localctx.psImage->comps[iBand - 1].h !=
    1354         161 :                     localctx.psImage->comps[0].h)
    1355             :             {
    1356           0 :                 CPLDebug(CODEC::debugId(), "Unable to handle that image (2)");
    1357           0 :                 localctx.free();
    1358           0 :                 return nullptr;
    1359             :             }
    1360             :         }
    1361             :     }
    1362             : 
    1363             :     /* -------------------------------------------------------------------- */
    1364             :     /*      Create a corresponding GDALDataset.                             */
    1365             :     /* -------------------------------------------------------------------- */
    1366             :     JP2OPJLikeDataset *poDS;
    1367             :     int iBand;
    1368             : 
    1369         759 :     poDS = new JP2OPJLikeDataset();
    1370         759 :     poDS->m_osFilename = poOpenInfo->pszFilename;
    1371         759 :     if (eCodecFormat == CODEC::cvtenum(JP2_CODEC_JP2))
    1372         678 :         poDS->eAccess = poOpenInfo->eAccess;
    1373         759 :     poDS->eColorSpace = localctx.psImage->color_space;
    1374         759 :     poDS->nRasterXSize = localctx.psImage->x1 - localctx.psImage->x0;
    1375         759 :     poDS->nRasterYSize = localctx.psImage->y1 - localctx.psImage->y0;
    1376         759 :     poDS->nBands = localctx.psImage->numcomps;
    1377         759 :     poDS->fp_ = poOpenInfo->fpL;
    1378         759 :     poOpenInfo->fpL = nullptr;
    1379         759 :     poDS->nCodeStreamStart = nCodeStreamStart;
    1380         759 :     poDS->nCodeStreamLength = nCodeStreamLength;
    1381         759 :     poDS->bIs420 = bIs420;
    1382        1468 :     poDS->bSingleTiled = (poDS->nRasterXSize == (int)nTileW &&
    1383         709 :                           poDS->nRasterYSize == (int)nTileH);
    1384         759 :     poDS->m_nX0 = localctx.psImage->x0;
    1385         759 :     poDS->m_nY0 = localctx.psImage->y0;
    1386         759 :     poDS->m_nTileWidth = nTileW;
    1387         759 :     poDS->m_nTileHeight = nTileH;
    1388             : 
    1389         759 :     int nBlockXSize = (int)nTileW;
    1390         759 :     int nBlockYSize = (int)nTileH;
    1391             : 
    1392         759 :     if (CPLFetchBool(poOpenInfo->papszOpenOptions, "USE_TILE_AS_BLOCK", false))
    1393             :     {
    1394           0 :         poDS->bUseSetDecodeArea = false;
    1395             :     }
    1396             : 
    1397         759 :     poDS->m_bStrict = CPLTestBool(
    1398         759 :         CSLFetchNameValueDef(poOpenInfo->papszOpenOptions, "STRICT", "YES"));
    1399         759 :     localctx.updateStrict(poDS->m_bStrict);
    1400             : 
    1401         759 :     if (localctx.preferPerBlockDeCompress())
    1402             :     {
    1403             :         /* Some Sentinel2 preview datasets are 343x343 large, but with 8x8 blocks */
    1404             :         /* Using the tile API for that is super slow, so expose a single block */
    1405         759 :         if (poDS->nRasterXSize <= 1024 && poDS->nRasterYSize <= 1024 &&
    1406         743 :             nTileW < 32 && nTileH < 32)
    1407             :         {
    1408         574 :             poDS->bUseSetDecodeArea = true;
    1409         574 :             nBlockXSize = poDS->nRasterXSize;
    1410         574 :             nBlockYSize = poDS->nRasterYSize;
    1411             :         }
    1412             :         else
    1413             :         {
    1414         185 :             poDS->bUseSetDecodeArea =
    1415         322 :                 poDS->bSingleTiled &&
    1416         137 :                 (poDS->nRasterXSize > 1024 || poDS->nRasterYSize > 1024);
    1417             : 
    1418             :             /* Other Sentinel2 preview datasets are 343x343 and 60m are 1830x1830,
    1419             :              * but they */
    1420             :             /* are tiled with tile dimensions 2048x2048. It would be a waste of */
    1421             :             /* memory to allocate such big blocks */
    1422         185 :             if (poDS->nRasterXSize < (int)nTileW &&
    1423          18 :                 poDS->nRasterYSize < (int)nTileH)
    1424             :             {
    1425          18 :                 poDS->bUseSetDecodeArea = TRUE;
    1426          18 :                 nBlockXSize = poDS->nRasterXSize;
    1427          18 :                 nBlockYSize = poDS->nRasterYSize;
    1428          18 :                 if (nBlockXSize > 2048)
    1429           0 :                     nBlockXSize = 2048;
    1430          18 :                 if (nBlockYSize > 2048)
    1431           0 :                     nBlockYSize = 2048;
    1432             :             }
    1433         167 :             else if (poDS->bUseSetDecodeArea)
    1434             :             {
    1435             :                 // Arbitrary threshold... ~4 million at least needed for the GRIB2
    1436             :                 // images mentioned below.
    1437           7 :                 if (nTileH == 1 && nTileW < 20 * 1024 * 1024)
    1438             :                 {
    1439             :                     // Some GRIB2 JPEG2000 compressed images are a 2D image
    1440             :                     // organized as a single line image...
    1441             :                 }
    1442             :                 else
    1443             :                 {
    1444           7 :                     if (nBlockXSize > 1024)
    1445           7 :                         nBlockXSize = 1024;
    1446           7 :                     if (nBlockYSize > 1024)
    1447           7 :                         nBlockYSize = 1024;
    1448             :                 }
    1449             :             }
    1450             :         }
    1451             :     }
    1452             : 
    1453         759 :     GDALColorTable *poCT = nullptr;
    1454             : 
    1455             :     /* -------------------------------------------------------------------- */
    1456             :     /*      Look for color table or cdef box                                */
    1457             :     /* -------------------------------------------------------------------- */
    1458         759 :     if (eCodecFormat == CODEC::cvtenum(JP2_CODEC_JP2))
    1459             :     {
    1460         678 :         vsi_l_offset nCurOffset = VSIFTellL(poDS->fp_);
    1461             : 
    1462        1356 :         GDALJP2Box oBox(poDS->fp_);
    1463         678 :         if (oBox.ReadFirst())
    1464             :         {
    1465        3625 :             while (strlen(oBox.GetType()) > 0)
    1466             :             {
    1467        3625 :                 if (EQUAL(oBox.GetType(), "jp2h"))
    1468             :                 {
    1469        1356 :                     GDALJP2Box oSubBox(poDS->fp_);
    1470             : 
    1471        2146 :                     for (oSubBox.ReadFirstChild(&oBox);
    1472        2146 :                          strlen(oSubBox.GetType()) > 0;
    1473        1468 :                          oSubBox.ReadNextChild(&oBox))
    1474             :                     {
    1475        1468 :                         GIntBig nDataLength = oSubBox.GetDataLength();
    1476        2911 :                         if (poCT == nullptr &&
    1477        1443 :                             EQUAL(oSubBox.GetType(), "pclr") &&
    1478        2911 :                             nDataLength >= 3 &&
    1479             :                             nDataLength <= 2 + 1 + 4 + 4 * 256)
    1480             :                         {
    1481          24 :                             GByte *pabyCT = oSubBox.ReadBoxData();
    1482          24 :                             if (pabyCT != nullptr)
    1483             :                             {
    1484          24 :                                 int nEntries = (pabyCT[0] << 8) | pabyCT[1];
    1485          24 :                                 int nComponents = pabyCT[2];
    1486             :                                 /* CPLDebug(CODEC::debugId(), "Color table found"); */
    1487          24 :                                 if (nEntries <= 256 && nComponents == 3)
    1488             :                                 {
    1489             :                                     /*CPLDebug(CODEC::debugId(), "resol[0] = %d",
    1490             :                                     pabyCT[3]); CPLDebug(CODEC::debugId(), "resol[1] =
    1491             :                                     %d", pabyCT[4]); CPLDebug(CODEC::debugId(),
    1492             :                                     "resol[2] = %d", pabyCT[5]);*/
    1493          19 :                                     if (pabyCT[3] == 7 && pabyCT[4] == 7 &&
    1494          19 :                                         pabyCT[5] == 7 &&
    1495          19 :                                         nDataLength == 2 + 1 + 3 + 3 * nEntries)
    1496             :                                     {
    1497          19 :                                         poCT = new GDALColorTable();
    1498        1103 :                                         for (int i = 0; i < nEntries; i++)
    1499             :                                         {
    1500             :                                             GDALColorEntry sEntry;
    1501        1084 :                                             sEntry.c1 = pabyCT[6 + 3 * i];
    1502        1084 :                                             sEntry.c2 = pabyCT[6 + 3 * i + 1];
    1503        1084 :                                             sEntry.c3 = pabyCT[6 + 3 * i + 2];
    1504        1084 :                                             sEntry.c4 = 255;
    1505        1084 :                                             poCT->SetColorEntry(i, &sEntry);
    1506             :                                         }
    1507          19 :                                     }
    1508             :                                 }
    1509           5 :                                 else if (nEntries <= 256 && nComponents == 4)
    1510             :                                 {
    1511           5 :                                     if (pabyCT[3] == 7 && pabyCT[4] == 7 &&
    1512           5 :                                         pabyCT[5] == 7 && pabyCT[6] == 7 &&
    1513           5 :                                         nDataLength == 2 + 1 + 4 + 4 * nEntries)
    1514             :                                     {
    1515           5 :                                         poCT = new GDALColorTable();
    1516          17 :                                         for (int i = 0; i < nEntries; i++)
    1517             :                                         {
    1518             :                                             GDALColorEntry sEntry;
    1519          12 :                                             sEntry.c1 = pabyCT[7 + 4 * i];
    1520          12 :                                             sEntry.c2 = pabyCT[7 + 4 * i + 1];
    1521          12 :                                             sEntry.c3 = pabyCT[7 + 4 * i + 2];
    1522          12 :                                             sEntry.c4 = pabyCT[7 + 4 * i + 3];
    1523          12 :                                             poCT->SetColorEntry(i, &sEntry);
    1524             :                                         }
    1525             :                                     }
    1526             :                                 }
    1527          24 :                                 CPLFree(pabyCT);
    1528             :                             }
    1529             :                         }
    1530             :                         /* There's a bug/misfeature in openjpeg: the color_space
    1531             :                            only gets set at read tile time */
    1532        1444 :                         else if (EQUAL(oSubBox.GetType(), "colr") &&
    1533             :                                  nDataLength == 7)
    1534             :                         {
    1535         678 :                             GByte *pabyContent = oSubBox.ReadBoxData();
    1536         678 :                             if (pabyContent != nullptr)
    1537             :                             {
    1538         678 :                                 if (pabyContent[0] ==
    1539             :                                     1 /* enumerated colourspace */)
    1540             :                                 {
    1541         678 :                                     GUInt32 enumcs = (pabyContent[3] << 24) |
    1542         678 :                                                      (pabyContent[4] << 16) |
    1543         678 :                                                      (pabyContent[5] << 8) |
    1544             :                                                      (pabyContent[6]);
    1545         678 :                                     if (enumcs == 16)
    1546             :                                     {
    1547          53 :                                         poDS->eColorSpace =
    1548          53 :                                             CODEC::cvtenum(JP2_CLRSPC_SRGB);
    1549          53 :                                         CPLDebug(CODEC::debugId(),
    1550             :                                                  "SRGB color space");
    1551             :                                     }
    1552         625 :                                     else if (enumcs == 17)
    1553             :                                     {
    1554         618 :                                         poDS->eColorSpace =
    1555         618 :                                             CODEC::cvtenum(JP2_CLRSPC_GRAY);
    1556         618 :                                         CPLDebug(CODEC::debugId(),
    1557             :                                                  "Grayscale color space");
    1558             :                                     }
    1559           7 :                                     else if (enumcs == 18)
    1560             :                                     {
    1561           3 :                                         poDS->eColorSpace =
    1562           3 :                                             CODEC::cvtenum(JP2_CLRSPC_SYCC);
    1563           3 :                                         CPLDebug(CODEC::debugId(),
    1564             :                                                  "SYCC color space");
    1565             :                                     }
    1566           4 :                                     else if (enumcs == 20)
    1567             :                                     {
    1568             :                                         /* Used by
    1569             :                                          * J2KP4files/testfiles_jp2/file7.jp2 */
    1570           0 :                                         poDS->eColorSpace =
    1571           0 :                                             CODEC::cvtenum(JP2_CLRSPC_SRGB);
    1572           0 :                                         CPLDebug(CODEC::debugId(),
    1573             :                                                  "e-sRGB color space");
    1574             :                                     }
    1575           4 :                                     else if (enumcs == 21)
    1576             :                                     {
    1577             :                                         /* Used by
    1578             :                                          * J2KP4files/testfiles_jp2/file5.jp2 */
    1579           0 :                                         poDS->eColorSpace =
    1580           0 :                                             CODEC::cvtenum(JP2_CLRSPC_SRGB);
    1581           0 :                                         CPLDebug(CODEC::debugId(),
    1582             :                                                  "ROMM-RGB color space");
    1583             :                                     }
    1584             :                                     else
    1585             :                                     {
    1586           4 :                                         poDS->eColorSpace =
    1587           4 :                                             CODEC::cvtenum(JP2_CLRSPC_UNKNOWN);
    1588           4 :                                         CPLDebug(CODEC::debugId(),
    1589             :                                                  "Unknown color space");
    1590             :                                     }
    1591             :                                 }
    1592         678 :                                 CPLFree(pabyContent);
    1593             :                             }
    1594             :                         }
    1595             :                         /* Check if there's an alpha channel or odd channel
    1596             :                          * attribution */
    1597         796 :                         else if (EQUAL(oSubBox.GetType(), "cdef") &&
    1598          30 :                                  nDataLength == 2 + poDS->nBands * 6)
    1599             :                         {
    1600          29 :                             GByte *pabyContent = oSubBox.ReadBoxData();
    1601          29 :                             if (pabyContent != nullptr)
    1602             :                             {
    1603          29 :                                 int nEntries =
    1604          29 :                                     (pabyContent[0] << 8) | pabyContent[1];
    1605          29 :                                 if (nEntries == poDS->nBands)
    1606             :                                 {
    1607          29 :                                     poDS->nRedIndex = -1;
    1608          29 :                                     poDS->nGreenIndex = -1;
    1609          29 :                                     poDS->nBlueIndex = -1;
    1610         130 :                                     for (int i = 0; i < poDS->nBands; i++)
    1611             :                                     {
    1612         101 :                                         int CNi =
    1613         101 :                                             (pabyContent[2 + 6 * i] << 8) |
    1614         101 :                                             pabyContent[2 + 6 * i + 1];
    1615         101 :                                         int Typi =
    1616         101 :                                             (pabyContent[2 + 6 * i + 2] << 8) |
    1617         101 :                                             pabyContent[2 + 6 * i + 3];
    1618         101 :                                         int Asoci =
    1619         101 :                                             (pabyContent[2 + 6 * i + 4] << 8) |
    1620         101 :                                             pabyContent[2 + 6 * i + 5];
    1621         101 :                                         if (CNi < 0 || CNi >= poDS->nBands)
    1622             :                                         {
    1623           0 :                                             CPLError(CE_Failure,
    1624             :                                                      CPLE_AppDefined,
    1625             :                                                      "Wrong value of CN%d=%d",
    1626             :                                                      i, CNi);
    1627           0 :                                             break;
    1628             :                                         }
    1629         101 :                                         if (Typi == 0)
    1630             :                                         {
    1631          75 :                                             if (Asoci == 1)
    1632          24 :                                                 poDS->nRedIndex = CNi;
    1633          51 :                                             else if (Asoci == 2)
    1634          18 :                                                 poDS->nGreenIndex = CNi;
    1635          33 :                                             else if (Asoci == 3)
    1636          18 :                                                 poDS->nBlueIndex = CNi;
    1637          15 :                                             else if (Asoci < 0 ||
    1638          15 :                                                      (Asoci > poDS->nBands &&
    1639             :                                                       Asoci != 65535))
    1640             :                                             {
    1641           0 :                                                 CPLError(
    1642             :                                                     CE_Failure, CPLE_AppDefined,
    1643             :                                                     "Wrong value of Asoc%d=%d",
    1644             :                                                     i, Asoci);
    1645           0 :                                                 break;
    1646             :                                             }
    1647             :                                         }
    1648          26 :                                         else if (Typi == 1)
    1649             :                                         {
    1650          26 :                                             poDS->nAlphaIndex = CNi;
    1651             :                                         }
    1652             :                                     }
    1653             :                                 }
    1654             :                                 else
    1655             :                                 {
    1656           0 :                                     CPLDebug(CODEC::debugId(),
    1657             :                                              "Unsupported cdef content");
    1658             :                                 }
    1659          29 :                                 CPLFree(pabyContent);
    1660             :                             }
    1661             :                         }
    1662             :                     }
    1663             :                 }
    1664             : 
    1665        3625 :                 if (!oBox.ReadNext())
    1666         678 :                     break;
    1667             :             }
    1668             :         }
    1669             : 
    1670         678 :         VSIFSeekL(poDS->fp_, nCurOffset, SEEK_SET);
    1671             : 
    1672         678 :         if (poDS->eColorSpace == CODEC::cvtenum(JP2_CLRSPC_GRAY) &&
    1673         618 :             poDS->nBands == 4 && poDS->nRedIndex == 0 &&
    1674        1301 :             poDS->nGreenIndex == 1 && poDS->nBlueIndex == 2 &&
    1675           5 :             poDS->m_osFilename.find("dop10rgbi") != std::string::npos)
    1676             :         {
    1677           0 :             CPLDebug(CODEC::debugId(),
    1678             :                      "Autofix wrong colorspace from Greyscale to sRGB");
    1679             :             // Workaround https://github.com/uclouvain/openjpeg/issues/1464
    1680             :             // dop10rgbi products from https://www.opengeodata.nrw.de/produkte/geobasis/lusat/dop/dop_jp2_f10/
    1681             :             // have a wrong color space.
    1682           0 :             poDS->eColorSpace = CODEC::cvtenum(JP2_CLRSPC_SRGB);
    1683             :         }
    1684             :     }
    1685             : 
    1686             :     /* -------------------------------------------------------------------- */
    1687             :     /*      Create band information objects.                                */
    1688             :     /* -------------------------------------------------------------------- */
    1689        1687 :     for (iBand = 1; iBand <= poDS->nBands; iBand++)
    1690             :     {
    1691         928 :         const bool bPromoteTo8Bit =
    1692         954 :             iBand == poDS->nAlphaIndex + 1 &&
    1693          26 :             localctx.psImage
    1694          26 :                     ->comps[(poDS->nAlphaIndex == 0 && poDS->nBands > 1) ? 1
    1695             :                                                                          : 0]
    1696          26 :                     .prec == 8 &&
    1697         968 :             localctx.psImage->comps[poDS->nAlphaIndex].prec == 1 &&
    1698          14 :             CPLFetchBool(poOpenInfo->papszOpenOptions, "1BIT_ALPHA_PROMOTION",
    1699          14 :                          CPLTestBool(CPLGetConfigOption(
    1700             :                              "JP2OPENJPEG_PROMOTE_1BIT_ALPHA_AS_8BIT", "YES")));
    1701         928 :         if (bPromoteTo8Bit)
    1702          10 :             CPLDebug(CODEC::debugId(),
    1703             :                      "Alpha band is promoted from 1 bit to 8 bit");
    1704             : 
    1705        1846 :         auto poBand = new JP2OPJLikeRasterBand<CODEC, BASE>(
    1706             :             poDS, iBand, eDataType,
    1707         918 :             bPromoteTo8Bit ? 8 : localctx.psImage->comps[iBand - 1].prec,
    1708             :             bPromoteTo8Bit, nBlockXSize, nBlockYSize);
    1709         928 :         if (iBand == 1 && poCT != nullptr)
    1710          24 :             poBand->poCT = poCT;
    1711         928 :         poDS->SetBand(iBand, poBand);
    1712             :     }
    1713             : 
    1714             :     /* -------------------------------------------------------------------- */
    1715             :     /*      Create overview datasets.                                       */
    1716             :     /* -------------------------------------------------------------------- */
    1717         759 :     int nW = poDS->nRasterXSize;
    1718         759 :     int nH = poDS->nRasterYSize;
    1719         759 :     poDS->nParentXSize = poDS->nRasterXSize;
    1720         759 :     poDS->nParentYSize = poDS->nRasterYSize;
    1721             : 
    1722             :     /* Lower resolutions are not compatible with a color-table */
    1723         759 :     if (poCT != nullptr)
    1724          24 :         numResolutions = 0;
    1725             : 
    1726         759 :     if (poDS->bSingleTiled && poDS->bUseSetDecodeArea)
    1727             :     {
    1728         579 :         poDS->cacheNew(&localctx);
    1729             :     }
    1730         759 :     poDS->m_pnLastLevel = new int(-1);
    1731             : 
    1732         117 :     while (
    1733         876 :         poDS->nOverviewCount + 1 < numResolutions && (nW > 128 || nH > 128) &&
    1734         120 :         (poDS->bUseSetDecodeArea || ((nTileW % 2) == 0 && (nTileH % 2) == 0)))
    1735             :     {
    1736             :         // This must be this exact formula per the JPEG2000 standard
    1737         117 :         nW = (nW + 1) / 2;
    1738         117 :         nH = (nH + 1) / 2;
    1739             : 
    1740         234 :         poDS->papoOverviewDS = (JP2OPJLikeDataset<CODEC, BASE> **)CPLRealloc(
    1741         117 :             poDS->papoOverviewDS, (poDS->nOverviewCount + 1) *
    1742             :                                       sizeof(JP2OPJLikeDataset<CODEC, BASE> *));
    1743         117 :         JP2OPJLikeDataset *poODS = new JP2OPJLikeDataset();
    1744         117 :         poODS->m_osFilename = poDS->m_osFilename;
    1745         117 :         poODS->nParentXSize = poDS->nRasterXSize;
    1746         117 :         poODS->nParentYSize = poDS->nRasterYSize;
    1747         117 :         poODS->SetDescription(poOpenInfo->pszFilename);
    1748         117 :         poODS->iLevel = poDS->nOverviewCount + 1;
    1749         117 :         poODS->bSingleTiled = poDS->bSingleTiled;
    1750         117 :         poODS->bUseSetDecodeArea = poDS->bUseSetDecodeArea;
    1751         117 :         poODS->nRedIndex = poDS->nRedIndex;
    1752         117 :         poODS->nGreenIndex = poDS->nGreenIndex;
    1753         117 :         poODS->nBlueIndex = poDS->nBlueIndex;
    1754         117 :         poODS->nAlphaIndex = poDS->nAlphaIndex;
    1755         117 :         if (!poDS->bUseSetDecodeArea)
    1756             :         {
    1757          83 :             nTileW /= 2;
    1758          83 :             nTileH /= 2;
    1759          83 :             nBlockXSize = (int)nTileW;
    1760          83 :             nBlockYSize = (int)nTileH;
    1761             :         }
    1762             :         else
    1763             :         {
    1764          34 :             nBlockXSize = std::min(nW, (int)nTileW);
    1765          34 :             nBlockYSize = std::min(nH, (int)nTileH);
    1766             :         }
    1767             : 
    1768         117 :         poODS->eColorSpace = poDS->eColorSpace;
    1769         117 :         poODS->nRasterXSize = nW;
    1770         117 :         poODS->nRasterYSize = nH;
    1771         117 :         poODS->nBands = poDS->nBands;
    1772         117 :         poODS->fp_ = poDS->fp_;
    1773         117 :         poODS->nCodeStreamStart = nCodeStreamStart;
    1774         117 :         poODS->nCodeStreamLength = nCodeStreamLength;
    1775         117 :         poODS->bIs420 = bIs420;
    1776             : 
    1777         117 :         if (poODS->bSingleTiled && poODS->bUseSetDecodeArea)
    1778             :         {
    1779          32 :             poODS->cache(poDS);
    1780             :         }
    1781         117 :         poODS->m_pnLastLevel = poDS->m_pnLastLevel;
    1782         117 :         poODS->m_bStrict = poDS->m_bStrict;
    1783             : 
    1784         117 :         poODS->m_nX0 = poDS->m_nX0;
    1785         117 :         poODS->m_nY0 = poDS->m_nY0;
    1786             : 
    1787         335 :         for (iBand = 1; iBand <= poDS->nBands; iBand++)
    1788             :         {
    1789         218 :             const bool bPromoteTo8Bit =
    1790         237 :                 iBand == poDS->nAlphaIndex + 1 &&
    1791          19 :                 localctx.psImage
    1792          19 :                         ->comps[(poDS->nAlphaIndex == 0 && poDS->nBands > 1)
    1793             :                                     ? 1
    1794             :                                     : 0]
    1795          19 :                         .prec == 8 &&
    1796         246 :                 localctx.psImage->comps[poDS->nAlphaIndex].prec == 1 &&
    1797           9 :                 CPLFetchBool(
    1798           9 :                     poOpenInfo->papszOpenOptions, "1BIT_ALPHA_PROMOTION",
    1799           9 :                     CPLTestBool(CPLGetConfigOption(
    1800             :                         "JP2OPENJPEG_PROMOTE_1BIT_ALPHA_AS_8BIT", "YES")));
    1801             : 
    1802         218 :             poODS->SetBand(iBand,
    1803         430 :                            new JP2OPJLikeRasterBand<CODEC, BASE>(
    1804             :                                poODS, iBand, eDataType,
    1805             :                                bPromoteTo8Bit
    1806             :                                    ? 8
    1807         212 :                                    : localctx.psImage->comps[iBand - 1].prec,
    1808             :                                bPromoteTo8Bit, nBlockXSize, nBlockYSize));
    1809             :         }
    1810             : 
    1811         117 :         poDS->papoOverviewDS[poDS->nOverviewCount++] = poODS;
    1812             :     }
    1813             : 
    1814         759 :     poDS->openCompleteJP2(&localctx);
    1815             : 
    1816             :     /* -------------------------------------------------------------------- */
    1817             :     /*      More metadata.                                                  */
    1818             :     /* -------------------------------------------------------------------- */
    1819         759 :     if (poDS->nBands > 1)
    1820             :     {
    1821          72 :         poDS->GDALDataset::SetMetadataItem("INTERLEAVE", "PIXEL",
    1822             :                                            "IMAGE_STRUCTURE");
    1823             :     }
    1824             : 
    1825         759 :     poOpenInfo->fpL = poDS->fp_;
    1826         759 :     vsi_l_offset nCurOffset = VSIFTellL(poDS->fp_);
    1827         759 :     poDS->LoadJP2Metadata(poOpenInfo);
    1828         759 :     VSIFSeekL(poDS->fp_, nCurOffset, SEEK_SET);
    1829         759 :     poOpenInfo->fpL = nullptr;
    1830             : 
    1831         759 :     poDS->bHasGeoreferencingAtOpening =
    1832         994 :         (!poDS->m_oSRS.IsEmpty() || poDS->nGCPCount != 0 ||
    1833         235 :          poDS->bGeoTransformValid);
    1834             : 
    1835             :     /* -------------------------------------------------------------------- */
    1836             :     /*      Vector layers                                                   */
    1837             :     /* -------------------------------------------------------------------- */
    1838         759 :     if (poOpenInfo->nOpenFlags & GDAL_OF_VECTOR)
    1839             :     {
    1840          66 :         poDS->LoadVectorLayers(CPLFetchBool(poOpenInfo->papszOpenOptions,
    1841             :                                             "OPEN_REMOTE_GML", false));
    1842             : 
    1843             :         // If file opened in vector-only mode and there's no vector,
    1844             :         // return
    1845          70 :         if ((poOpenInfo->nOpenFlags & GDAL_OF_RASTER) == 0 &&
    1846           4 :             poDS->GetLayerCount() == 0)
    1847             :         {
    1848           2 :             delete poDS;
    1849           2 :             return nullptr;
    1850             :         }
    1851             :     }
    1852             : 
    1853             :     /* -------------------------------------------------------------------- */
    1854             :     /*      Initialize any PAM information.                                 */
    1855             :     /* -------------------------------------------------------------------- */
    1856         757 :     poDS->SetDescription(poOpenInfo->pszFilename);
    1857         757 :     poDS->TryLoadXML(poOpenInfo->GetSiblingFiles());
    1858             : 
    1859             :     /* -------------------------------------------------------------------- */
    1860             :     /*      Check for overviews.                                            */
    1861             :     /* -------------------------------------------------------------------- */
    1862         757 :     poDS->oOvManager.Initialize(poDS, poOpenInfo);
    1863             : 
    1864         757 :     return poDS;
    1865             : }
    1866             : 
    1867             : /************************************************************************/
    1868             : /*                           WriteBox()                                 */
    1869             : /************************************************************************/
    1870             : 
    1871             : template <typename CODEC, typename BASE>
    1872         939 : bool JP2OPJLikeDataset<CODEC, BASE>::WriteBox(VSILFILE *fp, GDALJP2Box *poBox)
    1873             : {
    1874             :     GUInt32 nLBox;
    1875             :     GUInt32 nTBox;
    1876             : 
    1877         939 :     if (poBox == nullptr)
    1878           0 :         return true;
    1879             : 
    1880         939 :     nLBox = (int)poBox->GetDataLength() + 8;
    1881         939 :     nLBox = CPL_MSBWORD32(nLBox);
    1882             : 
    1883         939 :     memcpy(&nTBox, poBox->GetType(), 4);
    1884             : 
    1885         939 :     return VSIFWriteL(&nLBox, 4, 1, fp) == 1 &&
    1886        1878 :            VSIFWriteL(&nTBox, 4, 1, fp) == 1 &&
    1887         939 :            VSIFWriteL(poBox->GetWritableData(), (int)poBox->GetDataLength(), 1,
    1888         939 :                       fp) == 1;
    1889             : }
    1890             : 
    1891             : /************************************************************************/
    1892             : /*                         WriteGDALMetadataBox()                       */
    1893             : /************************************************************************/
    1894             : 
    1895             : template <typename CODEC, typename BASE>
    1896          19 : bool JP2OPJLikeDataset<CODEC, BASE>::WriteGDALMetadataBox(VSILFILE *fp,
    1897             :                                                           GDALDataset *poSrcDS,
    1898             :                                                           char **papszOptions)
    1899             : {
    1900          19 :     bool bRet = true;
    1901          38 :     GDALJP2Box *poBox = GDALJP2Metadata::CreateGDALMultiDomainMetadataXMLBox(
    1902          19 :         poSrcDS, CPLFetchBool(papszOptions, "MAIN_MD_DOMAIN_ONLY", false));
    1903          19 :     if (poBox)
    1904           6 :         bRet = WriteBox(fp, poBox);
    1905          19 :     delete poBox;
    1906          19 :     return bRet;
    1907             : }
    1908             : 
    1909             : /************************************************************************/
    1910             : /*                         WriteXMLBoxes()                              */
    1911             : /************************************************************************/
    1912             : 
    1913             : template <typename CODEC, typename BASE>
    1914          19 : bool JP2OPJLikeDataset<CODEC, BASE>::WriteXMLBoxes(VSILFILE *fp,
    1915             :                                                    GDALDataset *poSrcDS)
    1916             : {
    1917          19 :     bool bRet = true;
    1918          19 :     int nBoxes = 0;
    1919          19 :     GDALJP2Box **papoBoxes = GDALJP2Metadata::CreateXMLBoxes(poSrcDS, &nBoxes);
    1920          21 :     for (int i = 0; i < nBoxes; i++)
    1921             :     {
    1922           2 :         if (!WriteBox(fp, papoBoxes[i]))
    1923           0 :             bRet = false;
    1924           2 :         delete papoBoxes[i];
    1925             :     }
    1926          19 :     CPLFree(papoBoxes);
    1927          19 :     return bRet;
    1928             : }
    1929             : 
    1930             : /************************************************************************/
    1931             : /*                           WriteXMPBox()                              */
    1932             : /************************************************************************/
    1933             : 
    1934             : template <typename CODEC, typename BASE>
    1935          19 : bool JP2OPJLikeDataset<CODEC, BASE>::WriteXMPBox(VSILFILE *fp,
    1936             :                                                  GDALDataset *poSrcDS)
    1937             : {
    1938          19 :     bool bRet = true;
    1939          19 :     GDALJP2Box *poBox = GDALJP2Metadata::CreateXMPBox(poSrcDS);
    1940          19 :     if (poBox)
    1941           2 :         bRet = WriteBox(fp, poBox);
    1942          19 :     delete poBox;
    1943          19 :     return bRet;
    1944             : }
    1945             : 
    1946             : /************************************************************************/
    1947             : /*                           WriteIPRBox()                              */
    1948             : /************************************************************************/
    1949             : 
    1950             : template <typename CODEC, typename BASE>
    1951          19 : bool JP2OPJLikeDataset<CODEC, BASE>::WriteIPRBox(VSILFILE *fp,
    1952             :                                                  GDALDataset *poSrcDS)
    1953             : {
    1954          19 :     bool bRet = true;
    1955          19 :     GDALJP2Box *poBox = GDALJP2Metadata::CreateIPRBox(poSrcDS);
    1956          19 :     if (poBox)
    1957           2 :         bRet = WriteBox(fp, poBox);
    1958          19 :     delete poBox;
    1959          19 :     return bRet;
    1960             : }
    1961             : 
    1962             : /************************************************************************/
    1963             : /*                         FloorPowerOfTwo()                            */
    1964             : /************************************************************************/
    1965             : 
    1966         542 : static int FloorPowerOfTwo(int nVal)
    1967             : {
    1968         542 :     int nBits = 0;
    1969        3791 :     while (nVal > 1)
    1970             :     {
    1971        3249 :         nBits++;
    1972        3249 :         nVal >>= 1;
    1973             :     }
    1974         542 :     return 1 << nBits;
    1975             : }
    1976             : 
    1977             : /************************************************************************/
    1978             : /*                          CreateCopy()                                */
    1979             : /************************************************************************/
    1980             : 
    1981             : template <typename CODEC, typename BASE>
    1982         281 : GDALDataset *JP2OPJLikeDataset<CODEC, BASE>::CreateCopy(
    1983             :     const char *pszFilename, GDALDataset *poSrcDS, CPL_UNUSED int bStrict,
    1984             :     char **papszOptions, GDALProgressFunc pfnProgress, void *pProgressData)
    1985             : 
    1986             : {
    1987         281 :     int nBands = poSrcDS->GetRasterCount();
    1988         281 :     int nXSize = poSrcDS->GetRasterXSize();
    1989         281 :     int nYSize = poSrcDS->GetRasterYSize();
    1990             : 
    1991         281 :     if (nBands == 0 || nBands > 16384)
    1992             :     {
    1993           2 :         CPLError(
    1994             :             CE_Failure, CPLE_NotSupported,
    1995             :             "Unable to export files with %d bands. Must be >= 1 and <= 16384",
    1996             :             nBands);
    1997           2 :         return nullptr;
    1998             :     }
    1999             : 
    2000         279 :     GDALColorTable *poCT = poSrcDS->GetRasterBand(1)->GetColorTable();
    2001         279 :     if (poCT != nullptr && nBands != 1)
    2002             :     {
    2003           1 :         CPLError(CE_Failure, CPLE_NotSupported,
    2004             :                  "JP2 driver only supports a color table for a "
    2005             :                  "single-band dataset");
    2006           1 :         return nullptr;
    2007             :     }
    2008             : 
    2009         278 :     GDALDataType eDataType = poSrcDS->GetRasterBand(1)->GetRasterDataType();
    2010         278 :     int nDataTypeSize = (GDALGetDataTypeSize(eDataType) / 8);
    2011         278 :     if (eDataType != GDT_Byte && eDataType != GDT_Int16 &&
    2012           8 :         eDataType != GDT_UInt16 && eDataType != GDT_Int32 &&
    2013             :         eDataType != GDT_UInt32)
    2014             :     {
    2015           6 :         CPLError(CE_Failure, CPLE_NotSupported,
    2016             :                  "JP2 driver only supports creating Byte, GDT_Int16, "
    2017             :                  "GDT_UInt16, GDT_Int32, GDT_UInt32");
    2018           6 :         return nullptr;
    2019             :     }
    2020             : 
    2021         272 :     const bool bInspireTG = CPLFetchBool(papszOptions, "INSPIRE_TG", false);
    2022             : 
    2023             :     /* -------------------------------------------------------------------- */
    2024             :     /*      Analyze creation options.                                       */
    2025             :     /* -------------------------------------------------------------------- */
    2026         272 :     auto eCodecFormat = CODEC::cvtenum(JP2_CODEC_J2K);
    2027         272 :     const char *pszCodec = CSLFetchNameValueDef(papszOptions, "CODEC", nullptr);
    2028         272 :     if (pszCodec)
    2029             :     {
    2030          14 :         if (EQUAL(pszCodec, "JP2"))
    2031           5 :             eCodecFormat = CODEC::cvtenum(JP2_CODEC_JP2);
    2032           9 :         else if (EQUAL(pszCodec, "J2K"))
    2033           9 :             eCodecFormat = CODEC::cvtenum(JP2_CODEC_J2K);
    2034             :         else
    2035             :         {
    2036           0 :             CPLError(CE_Warning, CPLE_NotSupported,
    2037             :                      "Unsupported value for CODEC : %s. Defaulting to J2K",
    2038             :                      pszCodec);
    2039             :         }
    2040             :     }
    2041             :     else
    2042             :     {
    2043         258 :         if (strlen(pszFilename) > 4 &&
    2044         258 :             EQUAL(pszFilename + strlen(pszFilename) - 4, ".JP2"))
    2045             :         {
    2046         223 :             eCodecFormat = CODEC::cvtenum(JP2_CODEC_JP2);
    2047             :         }
    2048             :     }
    2049         272 :     if (eCodecFormat != CODEC::cvtenum(JP2_CODEC_JP2) && bInspireTG)
    2050             :     {
    2051           1 :         CPLError(CE_Warning, CPLE_NotSupported,
    2052             :                  "INSPIRE_TG=YES mandates CODEC=JP2 (TG requirement 21)");
    2053           1 :         return nullptr;
    2054             :     }
    2055             : 
    2056             :     // NOTE: if changing the default block size, the logic in nitfdataset.cpp
    2057             :     // CreateCopy() will have to be changed as well.
    2058         271 :     int nBlockXSize =
    2059         271 :         atoi(CSLFetchNameValueDef(papszOptions, "BLOCKXSIZE", "1024"));
    2060         271 :     int nBlockYSize =
    2061         271 :         atoi(CSLFetchNameValueDef(papszOptions, "BLOCKYSIZE", "1024"));
    2062         271 :     if (nBlockXSize <= 0 || nBlockYSize <= 0)
    2063             :     {
    2064           0 :         CPLError(CE_Failure, CPLE_NotSupported, "Invalid block size");
    2065           0 :         return nullptr;
    2066             :     }
    2067             : 
    2068             :     // By default do not generate tile sizes larger than the dataset
    2069             :     // dimensions
    2070         542 :     if (!CPLFetchBool(papszOptions, "BLOCKSIZE_STRICT", false) &&
    2071         271 :         !CPLFetchBool(papszOptions, "@BLOCKSIZE_STRICT", false))
    2072             :     {
    2073         267 :         if (nBlockXSize < 32 || nBlockYSize < 32)
    2074             :         {
    2075           0 :             CPLError(CE_Failure, CPLE_NotSupported, "Invalid block size");
    2076           0 :             return nullptr;
    2077             :         }
    2078             : 
    2079         267 :         if (nXSize < nBlockXSize)
    2080             :         {
    2081         242 :             CPLDebug(CODEC::debugId(), "Adjusting block width from %d to %d",
    2082             :                      nBlockXSize, nXSize);
    2083         242 :             nBlockXSize = nXSize;
    2084             :         }
    2085         267 :         if (nYSize < nBlockYSize)
    2086             :         {
    2087         243 :             CPLDebug(CODEC::debugId(), "Adjusting block width from %d to %d",
    2088             :                      nBlockYSize, nYSize);
    2089         243 :             nBlockYSize = nYSize;
    2090             :         }
    2091             :     }
    2092             : 
    2093         271 :     JP2_PROG_ORDER eProgOrder = JP2_LRCP;
    2094             :     const char *pszPROGORDER =
    2095         271 :         CSLFetchNameValueDef(papszOptions, "PROGRESSION", "LRCP");
    2096         271 :     if (EQUAL(pszPROGORDER, "LRCP"))
    2097         271 :         eProgOrder = JP2_LRCP;
    2098           0 :     else if (EQUAL(pszPROGORDER, "RLCP"))
    2099           0 :         eProgOrder = JP2_RLCP;
    2100           0 :     else if (EQUAL(pszPROGORDER, "RPCL"))
    2101           0 :         eProgOrder = JP2_RPCL;
    2102           0 :     else if (EQUAL(pszPROGORDER, "PCRL"))
    2103           0 :         eProgOrder = JP2_PCRL;
    2104           0 :     else if (EQUAL(pszPROGORDER, "CPRL"))
    2105           0 :         eProgOrder = JP2_CPRL;
    2106             :     else
    2107             :     {
    2108           0 :         CPLError(CE_Warning, CPLE_NotSupported,
    2109             :                  "Unsupported value for PROGRESSION : %s. Defaulting to LRCP",
    2110             :                  pszPROGORDER);
    2111             :     }
    2112             : 
    2113         271 :     const bool bIsIrreversible =
    2114         271 :         !CPLFetchBool(papszOptions, "REVERSIBLE", poCT != nullptr);
    2115             : 
    2116         542 :     std::vector<double> adfRates;
    2117             :     const char *pszQuality =
    2118         271 :         CSLFetchNameValueDef(papszOptions, "QUALITY", nullptr);
    2119         271 :     double dfDefaultQuality = (poCT != nullptr) ? 100.0 : 25.0;
    2120         271 :     if (pszQuality)
    2121             :     {
    2122             :         char **papszTokens =
    2123          41 :             CSLTokenizeStringComplex(pszQuality, ",", FALSE, FALSE);
    2124         158 :         for (int i = 0; papszTokens[i] != nullptr; i++)
    2125             :         {
    2126         117 :             double dfQuality = CPLAtof(papszTokens[i]);
    2127         117 :             if (dfQuality > 0 && dfQuality <= 100)
    2128             :             {
    2129         117 :                 double dfRate = 100 / dfQuality;
    2130         117 :                 adfRates.push_back(dfRate);
    2131             :             }
    2132             :             else
    2133             :             {
    2134           0 :                 CPLError(CE_Warning, CPLE_NotSupported,
    2135             :                          "Unsupported value for QUALITY: %s. Defaulting to "
    2136             :                          "single-layer, with quality=%.0f",
    2137           0 :                          papszTokens[i], dfDefaultQuality);
    2138           0 :                 adfRates.resize(0);
    2139           0 :                 break;
    2140             :             }
    2141             :         }
    2142          41 :         if (papszTokens[0] == nullptr)
    2143             :         {
    2144           0 :             CPLError(CE_Warning, CPLE_NotSupported,
    2145             :                      "Unsupported value for QUALITY: %s. Defaulting to "
    2146             :                      "single-layer, with quality=%.0f",
    2147             :                      pszQuality, dfDefaultQuality);
    2148             :         }
    2149          41 :         CSLDestroy(papszTokens);
    2150             :     }
    2151         271 :     if (adfRates.empty())
    2152             :     {
    2153         230 :         adfRates.push_back(100. / dfDefaultQuality);
    2154         230 :         assert(!adfRates.empty());
    2155             :     }
    2156             : 
    2157         271 :     if (poCT != nullptr && (bIsIrreversible || adfRates.back() != 1.0))
    2158             :     {
    2159           2 :         CPLError(CE_Warning, CPLE_AppDefined,
    2160             :                  "Encoding a dataset with a color table with REVERSIBLE != YES "
    2161             :                  "or QUALITY != 100 will likely lead to bad visual results");
    2162             :     }
    2163             : 
    2164         271 :     const int nMaxTileDim = std::max(nBlockXSize, nBlockYSize);
    2165         271 :     const int nMinTileDim = std::min(nBlockXSize, nBlockYSize);
    2166         271 :     int nNumResolutions = 1;
    2167             :     /* Pickup a reasonable value compatible with PROFILE_1 requirements */
    2168         362 :     while ((nMaxTileDim >> (nNumResolutions - 1)) > 128 &&
    2169          92 :            (nMinTileDim >> nNumResolutions) > 0)
    2170          91 :         nNumResolutions++;
    2171         271 :     int nMinProfile1Resolutions = nNumResolutions;
    2172             :     const char *pszResolutions =
    2173         271 :         CSLFetchNameValueDef(papszOptions, "RESOLUTIONS", nullptr);
    2174         271 :     if (pszResolutions)
    2175             :     {
    2176          10 :         nNumResolutions = atoi(pszResolutions);
    2177          10 :         if (nNumResolutions <= 0 || nNumResolutions >= 32 ||
    2178           9 :             (nMinTileDim >> nNumResolutions) == 0 ||
    2179           9 :             (nMaxTileDim >> nNumResolutions) == 0)
    2180             :         {
    2181           1 :             CPLError(CE_Warning, CPLE_NotSupported,
    2182             :                      "Unsupported value for RESOLUTIONS : %s. Defaulting to %d",
    2183             :                      pszResolutions, nMinProfile1Resolutions);
    2184           1 :             nNumResolutions = nMinProfile1Resolutions;
    2185             :         }
    2186             :     }
    2187         271 :     int nRedBandIndex = -1;
    2188         271 :     int nGreenBandIndex = -1;
    2189         271 :     int nBlueBandIndex = -1;
    2190         271 :     int nAlphaBandIndex = -1;
    2191         606 :     for (int i = 0; i < nBands; i++)
    2192             :     {
    2193             :         GDALColorInterp eInterp =
    2194         335 :             poSrcDS->GetRasterBand(i + 1)->GetColorInterpretation();
    2195         335 :         if (eInterp == GCI_RedBand)
    2196          15 :             nRedBandIndex = i;
    2197         320 :         else if (eInterp == GCI_GreenBand)
    2198          15 :             nGreenBandIndex = i;
    2199         305 :         else if (eInterp == GCI_BlueBand)
    2200          15 :             nBlueBandIndex = i;
    2201         290 :         else if (eInterp == GCI_AlphaBand)
    2202           7 :             nAlphaBandIndex = i;
    2203             :     }
    2204         271 :     const char *pszAlpha = CSLFetchNameValue(papszOptions, "ALPHA");
    2205         273 :     if (nAlphaBandIndex < 0 && nBands > 1 && pszAlpha != nullptr &&
    2206           2 :         CPLTestBool(pszAlpha))
    2207             :     {
    2208           2 :         nAlphaBandIndex = nBands - 1;
    2209             :     }
    2210             : 
    2211         271 :     const char *pszYCBCR420 = CSLFetchNameValue(papszOptions, "YCBCR420");
    2212         271 :     int bYCBCR420 = FALSE;
    2213         271 :     if (pszYCBCR420 && CPLTestBool(pszYCBCR420))
    2214             :     {
    2215           2 :         if ((nBands == 3 || nBands == 4) && eDataType == GDT_Byte &&
    2216           2 :             nRedBandIndex == 0 && nGreenBandIndex == 1 && nBlueBandIndex == 2)
    2217             :         {
    2218           2 :             if (((nXSize % 2) == 0 && (nYSize % 2) == 0 &&
    2219           2 :                  (nBlockXSize % 2) == 0 && (nBlockYSize % 2) == 0))
    2220             :             {
    2221           2 :                 bYCBCR420 = TRUE;
    2222             :             }
    2223             :             else
    2224             :             {
    2225           0 :                 CPLError(CE_Warning, CPLE_NotSupported,
    2226             :                          "YCBCR420 unsupported when image size and/or tile "
    2227             :                          "size are not multiple of 2");
    2228             :             }
    2229             :         }
    2230             :         else
    2231             :         {
    2232           0 :             CPLError(CE_Warning, CPLE_NotSupported,
    2233             :                      "YCBCR420 unsupported with this image band count and/or "
    2234             :                      "data byte");
    2235             :         }
    2236             :     }
    2237             : 
    2238         271 :     const char *pszYCC = CSLFetchNameValue(papszOptions, "YCC");
    2239         292 :     int bYCC = ((nBands == 3 || nBands == 4) &&
    2240          21 :                 CPLTestBool(CSLFetchNameValueDef(papszOptions, "YCC", "TRUE")));
    2241             : 
    2242         271 :     if (bYCBCR420 && bYCC)
    2243             :     {
    2244           2 :         if (pszYCC != nullptr)
    2245             :         {
    2246           0 :             CPLError(CE_Warning, CPLE_NotSupported,
    2247             :                      "YCC unsupported when YCbCr requesting");
    2248             :         }
    2249           2 :         bYCC = FALSE;
    2250             :     }
    2251             : 
    2252             :     /* -------------------------------------------------------------------- */
    2253             :     /*      Deal with codeblocks size                                       */
    2254             :     /* -------------------------------------------------------------------- */
    2255             : 
    2256             :     int nCblockW =
    2257         271 :         atoi(CSLFetchNameValueDef(papszOptions, "CODEBLOCK_WIDTH", "64"));
    2258             :     int nCblockH =
    2259         271 :         atoi(CSLFetchNameValueDef(papszOptions, "CODEBLOCK_HEIGHT", "64"));
    2260         271 :     if (nCblockW < 4 || nCblockW > 1024 || nCblockH < 4 || nCblockH > 1024)
    2261             :     {
    2262           4 :         CPLError(CE_Warning, CPLE_NotSupported,
    2263             :                  "Invalid values for codeblock size. Defaulting to 64x64");
    2264           4 :         nCblockW = 64;
    2265           4 :         nCblockH = 64;
    2266             :     }
    2267         267 :     else if (nCblockW * nCblockH > 4096)
    2268             :     {
    2269           1 :         CPLError(CE_Warning, CPLE_NotSupported,
    2270             :                  "Invalid values for codeblock size. "
    2271             :                  "CODEBLOCK_WIDTH * CODEBLOCK_HEIGHT should be <= 4096. "
    2272             :                  "Defaulting to 64x64");
    2273           1 :         nCblockW = 64;
    2274           1 :         nCblockH = 64;
    2275             :     }
    2276         271 :     int nCblockW_po2 = FloorPowerOfTwo(nCblockW);
    2277         271 :     int nCblockH_po2 = FloorPowerOfTwo(nCblockH);
    2278         271 :     if (nCblockW_po2 != nCblockW || nCblockH_po2 != nCblockH)
    2279             :     {
    2280           1 :         CPLError(CE_Warning, CPLE_NotSupported,
    2281             :                  "Non power of two values used for codeblock size. "
    2282             :                  "Using to %dx%d",
    2283             :                  nCblockW_po2, nCblockH_po2);
    2284             :     }
    2285         271 :     nCblockW = nCblockW_po2;
    2286         271 :     nCblockH = nCblockH_po2;
    2287             : 
    2288             :     /* -------------------------------------------------------------------- */
    2289             :     /*      Deal with codestream PROFILE                                    */
    2290             :     /* -------------------------------------------------------------------- */
    2291             :     const char *pszProfile =
    2292         271 :         CSLFetchNameValueDef(papszOptions, "PROFILE", "AUTO");
    2293         271 :     int bProfile1 = FALSE;
    2294         271 :     if (EQUAL(pszProfile, "UNRESTRICTED"))
    2295             :     {
    2296           1 :         bProfile1 = FALSE;
    2297           1 :         if (bInspireTG)
    2298             :         {
    2299           1 :             CPLError(CE_Failure, CPLE_NotSupported,
    2300             :                      "INSPIRE_TG=YES mandates PROFILE=PROFILE_1 (TG "
    2301             :                      "requirement 21)");
    2302           1 :             return nullptr;
    2303             :         }
    2304             :     }
    2305         270 :     else if (EQUAL(pszProfile, "UNRESTRICTED_FORCED"))
    2306             :     {
    2307           0 :         bProfile1 = FALSE;
    2308             :     }
    2309         270 :     else if (EQUAL(pszProfile,
    2310             :                    "PROFILE_1_FORCED")) /* For debug only: can produce
    2311             :                                            inconsistent codestream */
    2312             :     {
    2313           0 :         bProfile1 = TRUE;
    2314             :     }
    2315             :     else
    2316             :     {
    2317         270 :         if (!(EQUAL(pszProfile, "PROFILE_1") || EQUAL(pszProfile, "AUTO")))
    2318             :         {
    2319           0 :             CPLError(CE_Warning, CPLE_NotSupported,
    2320             :                      "Unsupported value for PROFILE : %s. Defaulting to AUTO",
    2321             :                      pszProfile);
    2322           0 :             pszProfile = "AUTO";
    2323             :         }
    2324             : 
    2325         270 :         bProfile1 = TRUE;
    2326         270 :         const char *pszReq21OrEmpty = bInspireTG ? " (TG requirement 21)" : "";
    2327         270 :         if ((nBlockXSize != nXSize || nBlockYSize != nYSize) &&
    2328          24 :             (nBlockXSize != nBlockYSize || nBlockXSize > 1024 ||
    2329          19 :              nBlockYSize > 1024))
    2330             :         {
    2331           5 :             bProfile1 = FALSE;
    2332           5 :             if (bInspireTG || EQUAL(pszProfile, "PROFILE_1"))
    2333             :             {
    2334           2 :                 CPLError(
    2335             :                     CE_Failure, CPLE_NotSupported,
    2336             :                     "Tile dimensions incompatible with PROFILE_1%s. "
    2337             :                     "Should be whole image or square with dimension <= 1024.",
    2338             :                     pszReq21OrEmpty);
    2339           2 :                 return nullptr;
    2340             :             }
    2341             :         }
    2342         268 :         if ((nMaxTileDim >> (nNumResolutions - 1)) > 128)
    2343             :         {
    2344           4 :             bProfile1 = FALSE;
    2345           4 :             if (bInspireTG || EQUAL(pszProfile, "PROFILE_1"))
    2346             :             {
    2347           1 :                 CPLError(CE_Failure, CPLE_NotSupported,
    2348             :                          "Number of resolutions incompatible with PROFILE_1%s. "
    2349             :                          "Should be at least %d.",
    2350             :                          pszReq21OrEmpty, nMinProfile1Resolutions);
    2351           1 :                 return nullptr;
    2352             :             }
    2353             :         }
    2354         267 :         if (nCblockW > 64 || nCblockH > 64)
    2355             :         {
    2356           2 :             bProfile1 = FALSE;
    2357           2 :             if (bInspireTG || EQUAL(pszProfile, "PROFILE_1"))
    2358             :             {
    2359           2 :                 CPLError(CE_Failure, CPLE_NotSupported,
    2360             :                          "Codeblock width incompatible with PROFILE_1%s. "
    2361             :                          "Codeblock width or height should be <= 64.",
    2362             :                          pszReq21OrEmpty);
    2363           2 :                 return nullptr;
    2364             :             }
    2365             :         }
    2366             :     }
    2367             : 
    2368             :     /* -------------------------------------------------------------------- */
    2369             :     /*      Work out the precision.                                         */
    2370             :     /* -------------------------------------------------------------------- */
    2371             :     int nBits;
    2372         265 :     if (CSLFetchNameValue(papszOptions, "NBITS") != nullptr)
    2373             :     {
    2374          22 :         nBits = atoi(CSLFetchNameValue(papszOptions, "NBITS"));
    2375          22 :         if (bInspireTG &&
    2376           1 :             !(nBits == 1 || nBits == 8 || nBits == 16 || nBits == 32))
    2377             :         {
    2378           1 :             CPLError(CE_Failure, CPLE_NotSupported,
    2379             :                      "INSPIRE_TG=YES mandates NBITS=1,8,16 or 32 (TG "
    2380             :                      "requirement 24)");
    2381           1 :             return nullptr;
    2382             :         }
    2383             :     }
    2384         243 :     else if (poSrcDS->GetRasterBand(1)->GetMetadataItem(
    2385         243 :                  "NBITS", "IMAGE_STRUCTURE") != nullptr)
    2386             :     {
    2387           3 :         nBits = atoi(poSrcDS->GetRasterBand(1)->GetMetadataItem(
    2388             :             "NBITS", "IMAGE_STRUCTURE"));
    2389           3 :         if (bInspireTG &&
    2390           1 :             !(nBits == 1 || nBits == 8 || nBits == 16 || nBits == 32))
    2391             :         {
    2392             :             /* Implements "NOTE If the original data do not satisfy this "
    2393             :                "requirement, they will be converted in a representation using "
    2394             :                "the next higher power of 2" */
    2395           1 :             nBits = GDALGetDataTypeSize(eDataType);
    2396             :         }
    2397             :     }
    2398             :     else
    2399             :     {
    2400         240 :         nBits = GDALGetDataTypeSize(eDataType);
    2401             :     }
    2402             : 
    2403         516 :     if ((GDALGetDataTypeSize(eDataType) == 8 && nBits > 8) ||
    2404         780 :         (GDALGetDataTypeSize(eDataType) == 16 && (nBits <= 8 || nBits > 16)) ||
    2405         264 :         (GDALGetDataTypeSize(eDataType) == 32 && (nBits <= 16 || nBits > 32)))
    2406             :     {
    2407           0 :         CPLError(CE_Warning, CPLE_NotSupported,
    2408             :                  "Inconsistent NBITS value with data type. Using %d",
    2409             :                  GDALGetDataTypeSize(eDataType));
    2410             :     }
    2411             : 
    2412             :     /* -------------------------------------------------------------------- */
    2413             :     /*      Georeferencing options                                          */
    2414             :     /* -------------------------------------------------------------------- */
    2415             : 
    2416         264 :     bool bGMLJP2Option = CPLFetchBool(papszOptions, "GMLJP2", true);
    2417         264 :     int nGMLJP2Version = 1;
    2418             :     const char *pszGMLJP2V2Def =
    2419         264 :         CSLFetchNameValue(papszOptions, "GMLJP2V2_DEF");
    2420         264 :     if (pszGMLJP2V2Def != nullptr)
    2421             :     {
    2422          28 :         bGMLJP2Option = true;
    2423          28 :         nGMLJP2Version = 2;
    2424          28 :         if (bInspireTG)
    2425             :         {
    2426           0 :             CPLError(CE_Warning, CPLE_NotSupported,
    2427             :                      "INSPIRE_TG=YES is only compatible with GMLJP2 v1");
    2428           0 :             return nullptr;
    2429             :         }
    2430             :     }
    2431         264 :     const bool bGeoJP2Option = CPLFetchBool(papszOptions, "GeoJP2", true);
    2432             : 
    2433         528 :     GDALJP2Metadata oJP2MD;
    2434             : 
    2435         264 :     int bGeoreferencingCompatOfGeoJP2 = FALSE;
    2436         264 :     int bGeoreferencingCompatOfGMLJP2 = FALSE;
    2437         270 :     if (eCodecFormat == CODEC::cvtenum(JP2_CODEC_JP2) &&
    2438           6 :         (bGMLJP2Option || bGeoJP2Option))
    2439             :     {
    2440         219 :         if (poSrcDS->GetGCPCount() > 0)
    2441             :         {
    2442           3 :             bGeoreferencingCompatOfGeoJP2 = TRUE;
    2443           3 :             oJP2MD.SetGCPs(poSrcDS->GetGCPCount(), poSrcDS->GetGCPs());
    2444           3 :             oJP2MD.SetSpatialRef(poSrcDS->GetGCPSpatialRef());
    2445             :         }
    2446             :         else
    2447             :         {
    2448         216 :             const OGRSpatialReference *poSRS = poSrcDS->GetSpatialRef();
    2449         216 :             if (poSRS)
    2450             :             {
    2451          57 :                 bGeoreferencingCompatOfGeoJP2 = TRUE;
    2452          57 :                 oJP2MD.SetSpatialRef(poSRS);
    2453             :             }
    2454             :             double adfGeoTransform[6];
    2455         216 :             if (poSrcDS->GetGeoTransform(adfGeoTransform) == CE_None)
    2456             :             {
    2457         164 :                 bGeoreferencingCompatOfGeoJP2 = TRUE;
    2458         164 :                 oJP2MD.SetGeoTransform(adfGeoTransform);
    2459         164 :                 if (poSRS && !poSRS->IsEmpty())
    2460             :                 {
    2461          57 :                     bGeoreferencingCompatOfGMLJP2 =
    2462          57 :                         GDALJP2Metadata::IsSRSCompatible(poSRS);
    2463          57 :                     if (!bGeoreferencingCompatOfGMLJP2)
    2464             :                     {
    2465           1 :                         CPLDebug(
    2466             :                             CODEC::debugId(),
    2467             :                             "Cannot write GMLJP2 box due to unsupported SRS");
    2468             :                     }
    2469             :                 }
    2470             :             }
    2471             :         }
    2472         219 :         if (poSrcDS->GetMetadata("RPC") != nullptr)
    2473             :         {
    2474           1 :             oJP2MD.SetRPCMD(poSrcDS->GetMetadata("RPC"));
    2475           1 :             bGeoreferencingCompatOfGeoJP2 = TRUE;
    2476             :         }
    2477             : 
    2478             :         const char *pszAreaOrPoint =
    2479         219 :             poSrcDS->GetMetadataItem(GDALMD_AREA_OR_POINT);
    2480         260 :         oJP2MD.bPixelIsPoint = pszAreaOrPoint != nullptr &&
    2481          41 :                                EQUAL(pszAreaOrPoint, GDALMD_AOP_POINT);
    2482             : 
    2483         434 :         if (bGMLJP2Option &&
    2484         215 :             CPLGetConfigOption("GMLJP2OVERRIDE", nullptr) != nullptr)
    2485             :         {
    2486             :             // Force V1 since this is the branch in which the hack is
    2487             :             // implemented
    2488           7 :             nGMLJP2Version = 1;
    2489           7 :             bGeoreferencingCompatOfGMLJP2 = TRUE;
    2490             :         }
    2491             :     }
    2492             : 
    2493         264 :     if (CSLFetchNameValue(papszOptions, "GMLJP2") != nullptr && bGMLJP2Option &&
    2494             :         !bGeoreferencingCompatOfGMLJP2)
    2495             :     {
    2496           0 :         CPLError(CE_Warning, CPLE_AppDefined,
    2497             :                  "GMLJP2 box was explicitly required but cannot be written due "
    2498             :                  "to lack of georeferencing and/or unsupported georeferencing "
    2499             :                  "for GMLJP2");
    2500             :     }
    2501             : 
    2502         264 :     if (CSLFetchNameValue(papszOptions, "GeoJP2") != nullptr && bGeoJP2Option &&
    2503             :         !bGeoreferencingCompatOfGeoJP2)
    2504             :     {
    2505           0 :         CPLError(CE_Warning, CPLE_AppDefined,
    2506             :                  "GeoJP2 box was explicitly required but cannot be written due "
    2507             :                  "to lack of georeferencing");
    2508             :     }
    2509             :     const bool bGeoBoxesAfter =
    2510         264 :         CPLFetchBool(papszOptions, "GEOBOXES_AFTER_JP2C", bInspireTG);
    2511         264 :     GDALJP2Box *poGMLJP2Box = nullptr;
    2512         264 :     if (eCodecFormat == CODEC::cvtenum(JP2_CODEC_JP2) && bGMLJP2Option &&
    2513             :         bGeoreferencingCompatOfGMLJP2)
    2514             :     {
    2515          61 :         if (nGMLJP2Version == 1)
    2516          33 :             poGMLJP2Box = oJP2MD.CreateGMLJP2(nXSize, nYSize);
    2517             :         else
    2518             :             poGMLJP2Box =
    2519          28 :                 oJP2MD.CreateGMLJP2V2(nXSize, nYSize, pszGMLJP2V2Def, poSrcDS);
    2520          61 :         if (poGMLJP2Box == nullptr)
    2521           3 :             return nullptr;
    2522             :     }
    2523             : 
    2524             :     /* ---------------------------------------------------------------- */
    2525             :     /* If the input driver is identifed as "GEORASTER" the following    */
    2526             :     /* section will try to dump a ORACLE GeoRaster JP2 BLOB into a file */
    2527             :     /* ---------------------------------------------------------------- */
    2528             : 
    2529         261 :     if (EQUAL(poSrcDS->GetDriverName(), "GEORASTER"))
    2530             :     {
    2531             :         const char *pszGEOR_compress =
    2532           0 :             poSrcDS->GetMetadataItem("COMPRESSION", "IMAGE_STRUCTURE");
    2533             : 
    2534           0 :         if (pszGEOR_compress == nullptr)
    2535             :         {
    2536           0 :             pszGEOR_compress = "NONE";
    2537             :         }
    2538             : 
    2539             :         /* Check if the JP2 BLOB needs re-shaping */
    2540             : 
    2541           0 :         bool bGEOR_reshape = false;
    2542             : 
    2543           0 :         const char *apszIgnoredOptions[] = {"BLOCKXSIZE",
    2544             :                                             "BLOCKYSIZE",
    2545             :                                             "QUALITY",
    2546             :                                             "REVERSIBLE",
    2547             :                                             "RESOLUTIONS",
    2548             :                                             "PROGRESSION",
    2549             :                                             "SOP",
    2550             :                                             "EPH",
    2551             :                                             "YCBCR420",
    2552             :                                             "YCC",
    2553             :                                             "NBITS",
    2554             :                                             "1BIT_ALPHA",
    2555             :                                             "PRECINCTS",
    2556             :                                             "TILEPARTS",
    2557             :                                             "CODEBLOCK_WIDTH",
    2558             :                                             "CODEBLOCK_HEIGHT",
    2559             :                                             "PLT",
    2560             :                                             "TLM",
    2561             :                                             nullptr};
    2562             : 
    2563           0 :         for (int i = 0; apszIgnoredOptions[i]; i++)
    2564             :         {
    2565           0 :             if (CSLFetchNameValue(papszOptions, apszIgnoredOptions[i]))
    2566             :             {
    2567           0 :                 bGEOR_reshape = true;
    2568             :             }
    2569             :         }
    2570             : 
    2571           0 :         if (CSLFetchNameValue(papszOptions, "USE_SRC_CODESTREAM"))
    2572             :         {
    2573           0 :             bGEOR_reshape = false;
    2574             :         }
    2575             : 
    2576           0 :         char **papszGEOR_files = poSrcDS->GetFileList();
    2577             : 
    2578           0 :         if (EQUAL(pszGEOR_compress, "JP2-F") && CSLCount(papszGEOR_files) > 0 &&
    2579           0 :             bGEOR_reshape == false)
    2580             :         {
    2581             : 
    2582           0 :             const char *pszVsiOciLob = papszGEOR_files[0];
    2583             : 
    2584           0 :             VSILFILE *fpBlob = VSIFOpenL(pszVsiOciLob, "r");
    2585           0 :             if (fpBlob == nullptr)
    2586             :             {
    2587           0 :                 CPLError(CE_Failure, CPLE_AppDefined, "Cannot open %s",
    2588             :                          pszVsiOciLob);
    2589           0 :                 delete poGMLJP2Box;
    2590           0 :                 return nullptr;
    2591             :             }
    2592           0 :             VSILFILE *fp = VSIFOpenL(pszFilename, "w+b");
    2593           0 :             if (fp == nullptr)
    2594             :             {
    2595           0 :                 CPLError(CE_Failure, CPLE_AppDefined, "Cannot create %s",
    2596             :                          pszFilename);
    2597           0 :                 delete poGMLJP2Box;
    2598           0 :                 VSIFCloseL(fpBlob);
    2599           0 :                 return nullptr;
    2600             :             }
    2601             : 
    2602           0 :             VSIFSeekL(fpBlob, 0, SEEK_END);
    2603             : 
    2604           0 :             size_t nBlobSize = static_cast<size_t>(VSIFTellL(fpBlob));
    2605           0 :             size_t nChunk = (size_t)(GDALGetCacheMax() * 0.25);
    2606           0 :             size_t nSize = 0;
    2607           0 :             size_t nCount = 0;
    2608             : 
    2609           0 :             void *pBuffer = (GByte *)VSI_MALLOC_VERBOSE(nChunk);
    2610           0 :             if (pBuffer == nullptr)
    2611             :             {
    2612           0 :                 delete poGMLJP2Box;
    2613           0 :                 VSIFCloseL(fpBlob);
    2614           0 :                 VSIFCloseL(fp);
    2615           0 :                 return nullptr;
    2616             :             }
    2617             : 
    2618           0 :             VSIFSeekL(fpBlob, 0, SEEK_SET);
    2619             : 
    2620           0 :             while ((nSize = VSIFReadL(pBuffer, 1, nChunk, fpBlob)) > 0)
    2621             :             {
    2622           0 :                 VSIFWriteL(pBuffer, 1, nSize, fp);
    2623           0 :                 nCount += nSize;
    2624           0 :                 pfnProgress((float)nCount / (float)nBlobSize, nullptr,
    2625             :                             pProgressData);
    2626             :             }
    2627             : 
    2628           0 :             CPLFree(pBuffer);
    2629           0 :             VSIFCloseL(fpBlob);
    2630             : 
    2631           0 :             VSIFCloseL(fp);
    2632             : 
    2633             :             /* Return the GDALDaset object */
    2634             : 
    2635           0 :             GDALOpenInfo oOpenInfo(pszFilename, GA_Update);
    2636           0 :             GDALDataset *poDS = JP2OPJLikeDataset::Open(&oOpenInfo);
    2637             : 
    2638             :             /* Copy essential metadata */
    2639             : 
    2640             :             double adfGeoTransform[6];
    2641             : 
    2642           0 :             if (poSrcDS->GetGeoTransform(adfGeoTransform) == CE_None)
    2643             :             {
    2644           0 :                 poDS->SetGeoTransform(adfGeoTransform);
    2645             :             }
    2646             : 
    2647           0 :             const OGRSpatialReference *poSRS = poSrcDS->GetSpatialRef();
    2648           0 :             if (poSRS)
    2649             :             {
    2650           0 :                 poDS->SetSpatialRef(poSRS);
    2651             :             }
    2652             : 
    2653           0 :             delete poGMLJP2Box;
    2654           0 :             return poDS;
    2655             :         }
    2656             :     }
    2657             : 
    2658             :     /* -------------------------------------------------------------------- */
    2659             :     /*      Setup encoder                                                   */
    2660             :     /* -------------------------------------------------------------------- */
    2661             : 
    2662         522 :     JP2OPJLikeDataset oTmpDS;
    2663         261 :     int numThreads = oTmpDS.GetNumThreads();
    2664             : 
    2665         522 :     CODEC localctx;
    2666         261 :     localctx.allocComponentParams(nBands);
    2667             :     int iBand;
    2668         261 :     int bSamePrecision = TRUE;
    2669         261 :     int b1BitAlpha = FALSE;
    2670         586 :     for (iBand = 0; iBand < nBands; iBand++)
    2671             :     {
    2672         325 :         localctx.pasBandParams[iBand].x0 = 0;
    2673         325 :         localctx.pasBandParams[iBand].y0 = 0;
    2674         325 :         if (bYCBCR420 && (iBand == 1 || iBand == 2))
    2675             :         {
    2676           4 :             localctx.pasBandParams[iBand].dx = 2;
    2677           4 :             localctx.pasBandParams[iBand].dy = 2;
    2678           4 :             localctx.pasBandParams[iBand].w = nXSize / 2;
    2679           4 :             localctx.pasBandParams[iBand].h = nYSize / 2;
    2680             :         }
    2681             :         else
    2682             :         {
    2683         321 :             localctx.pasBandParams[iBand].dx = 1;
    2684         321 :             localctx.pasBandParams[iBand].dy = 1;
    2685         321 :             localctx.pasBandParams[iBand].w = nXSize;
    2686         321 :             localctx.pasBandParams[iBand].h = nYSize;
    2687             :         }
    2688             : 
    2689         325 :         localctx.pasBandParams[iBand].sgnd =
    2690         325 :             (eDataType == GDT_Int16 || eDataType == GDT_Int32);
    2691         325 :         localctx.pasBandParams[iBand].prec = nBits;
    2692             : 
    2693             :         const char *pszNBits =
    2694         325 :             poSrcDS->GetRasterBand(iBand + 1)->GetMetadataItem(
    2695             :                 "NBITS", "IMAGE_STRUCTURE");
    2696             :         /* Recommendation 38 In the case of an opacity channel, the bit depth
    2697             :          * should be 1-bit. */
    2698         334 :         if (iBand == nAlphaBandIndex &&
    2699           0 :             ((pszNBits != nullptr && EQUAL(pszNBits, "1")) ||
    2700           9 :              CPLFetchBool(papszOptions, "1BIT_ALPHA", bInspireTG)))
    2701             :         {
    2702           3 :             if (iBand != nBands - 1 && nBits != 1)
    2703             :             {
    2704             :                 /* Might be a bug in openjpeg, but it seems that if the alpha */
    2705             :                 /* band is the first one, it would select 1-bit for all
    2706             :                  * channels... */
    2707           0 :                 CPLError(CE_Warning, CPLE_NotSupported,
    2708             :                          "Cannot output 1-bit alpha channel if it is not the "
    2709             :                          "last one");
    2710             :             }
    2711             :             else
    2712             :             {
    2713           3 :                 CPLDebug(CODEC::debugId(), "Using 1-bit alpha channel");
    2714           3 :                 localctx.pasBandParams[iBand].sgnd = 0;
    2715           3 :                 localctx.pasBandParams[iBand].prec = 1;
    2716           3 :                 bSamePrecision = FALSE;
    2717           3 :                 b1BitAlpha = TRUE;
    2718             :             }
    2719             :         }
    2720             :     }
    2721             : 
    2722         261 :     if (bInspireTG && nAlphaBandIndex >= 0 && !b1BitAlpha)
    2723             :     {
    2724           0 :         CPLError(
    2725             :             CE_Warning, CPLE_NotSupported,
    2726             :             "INSPIRE_TG=YES recommends 1BIT_ALPHA=YES (Recommendation 38)");
    2727             :     }
    2728         261 :     auto eColorSpace = CODEC::cvtenum(JP2_CLRSPC_GRAY);
    2729             : 
    2730         261 :     if (bYCBCR420)
    2731             :     {
    2732           2 :         eColorSpace = CODEC::cvtenum(JP2_CLRSPC_SYCC);
    2733             :     }
    2734         259 :     else if ((nBands == 3 || nBands == 4) && nRedBandIndex >= 0 &&
    2735          13 :              nGreenBandIndex >= 0 && nBlueBandIndex >= 0)
    2736             :     {
    2737          13 :         eColorSpace = CODEC::cvtenum(JP2_CLRSPC_SRGB);
    2738             :     }
    2739         246 :     else if (poCT != nullptr)
    2740             :     {
    2741           6 :         eColorSpace = CODEC::cvtenum(JP2_CLRSPC_SRGB);
    2742             :     }
    2743             : 
    2744             :     /* -------------------------------------------------------------------- */
    2745             :     /*      Create the dataset.                                             */
    2746             :     /* -------------------------------------------------------------------- */
    2747             : 
    2748         261 :     const char *pszAccess =
    2749         261 :         STARTS_WITH_CI(pszFilename, "/vsisubfile/") ? "r+b" : "w+b";
    2750         261 :     VSILFILE *fp = VSIFOpenL(pszFilename, pszAccess);
    2751         261 :     if (fp == nullptr)
    2752             :     {
    2753           3 :         CPLError(CE_Failure, CPLE_AppDefined, "Cannot create file");
    2754           3 :         CPLFree(localctx.pasBandParams);
    2755           3 :         localctx.pasBandParams = nullptr;
    2756           3 :         delete poGMLJP2Box;
    2757           3 :         return nullptr;
    2758             :     }
    2759             : 
    2760             :     /* -------------------------------------------------------------------- */
    2761             :     /*      Add JP2 boxes.                                                  */
    2762             :     /* -------------------------------------------------------------------- */
    2763         258 :     vsi_l_offset nStartJP2C = 0;
    2764         258 :     int bUseXLBoxes = FALSE;
    2765             : 
    2766         258 :     if (eCodecFormat == CODEC::cvtenum(JP2_CODEC_JP2))
    2767             :     {
    2768         436 :         GDALJP2Box jPBox(fp);
    2769         218 :         jPBox.SetType("jP  ");
    2770         218 :         jPBox.AppendWritableData(4, "\x0D\x0A\x87\x0A");
    2771         218 :         WriteBox(fp, &jPBox);
    2772             : 
    2773         436 :         GDALJP2Box ftypBox(fp);
    2774         218 :         ftypBox.SetType("ftyp");
    2775             :         // http://docs.opengeospatial.org/is/08-085r5/08-085r5.html Req 19
    2776         218 :         const bool bJPXOption = CPLFetchBool(papszOptions, "JPX", true);
    2777         218 :         if (nGMLJP2Version == 2 && bJPXOption)
    2778          25 :             ftypBox.AppendWritableData(4, "jpx "); /* Branding */
    2779             :         else
    2780         193 :             ftypBox.AppendWritableData(4, "jp2 "); /* Branding */
    2781         218 :         ftypBox.AppendUInt32(0);                   /* minimum version */
    2782         218 :         ftypBox.AppendWritableData(
    2783             :             4, "jp2 "); /* Compatibility list: first value */
    2784             : 
    2785         218 :         if (bInspireTG && poGMLJP2Box != nullptr && !bJPXOption)
    2786             :         {
    2787           1 :             CPLError(
    2788             :                 CE_Warning, CPLE_AppDefined,
    2789             :                 "INSPIRE_TG=YES implies following GMLJP2 specification which "
    2790             :                 "recommends advertise reader requirement 67 feature, and thus "
    2791             :                 "JPX capability");
    2792             :         }
    2793         217 :         else if (poGMLJP2Box != nullptr && bJPXOption)
    2794             :         {
    2795             :             /* GMLJP2 uses lbl and asoc boxes, which are JPEG2000 Part II spec
    2796             :              */
    2797             :             /* advertizing jpx is required per 8.1 of 05-047r3 GMLJP2 */
    2798          57 :             ftypBox.AppendWritableData(
    2799             :                 4, "jpx "); /* Compatibility list: second value */
    2800             :         }
    2801         218 :         WriteBox(fp, &ftypBox);
    2802             : 
    2803         220 :         const bool bIPR = poSrcDS->GetMetadata("xml:IPR") != nullptr &&
    2804           2 :                           CPLFetchBool(papszOptions, "WRITE_METADATA", false);
    2805             : 
    2806             :         /* Reader requirement box */
    2807         218 :         if (poGMLJP2Box != nullptr && bJPXOption)
    2808             :         {
    2809         114 :             GDALJP2Box rreqBox(fp);
    2810          57 :             rreqBox.SetType("rreq");
    2811          57 :             rreqBox.AppendUInt8(1); /* ML = 1 byte for mask length */
    2812             : 
    2813          57 :             rreqBox.AppendUInt8(0x80 | 0x40 | (bIPR ? 0x20 : 0)); /* FUAM */
    2814          57 :             rreqBox.AppendUInt8(0x80);                            /* DCM */
    2815             : 
    2816          57 :             rreqBox.AppendUInt16(
    2817             :                 2 + (bIPR ? 1 : 0)); /* NSF: Number of standard features */
    2818             : 
    2819          57 :             rreqBox.AppendUInt16(
    2820             :                 (bProfile1) ? 4 : 5);  /* SF0 : PROFILE 1 or PROFILE 2 */
    2821          57 :             rreqBox.AppendUInt8(0x80); /* SM0 */
    2822             : 
    2823          57 :             rreqBox.AppendUInt16(67);  /* SF1 : GMLJP2 box */
    2824          57 :             rreqBox.AppendUInt8(0x40); /* SM1 */
    2825             : 
    2826          57 :             if (bIPR)
    2827             :             {
    2828           0 :                 rreqBox.AppendUInt16(35);  /* SF2 : IPR metadata */
    2829           0 :                 rreqBox.AppendUInt8(0x20); /* SM2 */
    2830             :             }
    2831          57 :             rreqBox.AppendUInt16(0); /* NVF */
    2832          57 :             WriteBox(fp, &rreqBox);
    2833             :         }
    2834             : 
    2835         436 :         GDALJP2Box ihdrBox(fp);
    2836         218 :         ihdrBox.SetType("ihdr");
    2837         218 :         ihdrBox.AppendUInt32(nYSize);
    2838         218 :         ihdrBox.AppendUInt32(nXSize);
    2839         218 :         ihdrBox.AppendUInt16(static_cast<GUInt16>(nBands));
    2840             :         GByte BPC;
    2841         218 :         if (bSamePrecision)
    2842         215 :             BPC = static_cast<GByte>((localctx.pasBandParams[0].prec - 1) |
    2843         215 :                                      (localctx.pasBandParams[0].sgnd << 7));
    2844             :         else
    2845           3 :             BPC = 255;
    2846         218 :         ihdrBox.AppendUInt8(BPC);
    2847         218 :         ihdrBox.AppendUInt8(7); /* C=Compression type: fixed value */
    2848         218 :         ihdrBox.AppendUInt8(0); /* UnkC: 0= colourspace of the image is known */
    2849             :         /*and correctly specified in the Colourspace Specification boxes within
    2850             :          * the file */
    2851         218 :         ihdrBox.AppendUInt8(
    2852             :             bIPR ? 1 : 0); /* IPR: 0=no intellectual property, 1=IPR box */
    2853             : 
    2854         436 :         GDALJP2Box bpccBox(fp);
    2855         218 :         if (!bSamePrecision)
    2856             :         {
    2857           3 :             bpccBox.SetType("bpcc");
    2858          13 :             for (int i = 0; i < nBands; i++)
    2859          10 :                 bpccBox.AppendUInt8(
    2860          10 :                     static_cast<GByte>((localctx.pasBandParams[i].prec - 1) |
    2861          10 :                                        (localctx.pasBandParams[i].sgnd << 7)));
    2862             :         }
    2863             : 
    2864         436 :         GDALJP2Box colrBox(fp);
    2865         218 :         colrBox.SetType("colr");
    2866         218 :         colrBox.AppendUInt8(1); /* METHOD: 1=Enumerated Colourspace */
    2867         218 :         colrBox.AppendUInt8(
    2868             :             0); /* PREC: Precedence. 0=(field reserved for ISO use) */
    2869         218 :         colrBox.AppendUInt8(0); /* APPROX: Colourspace approximation. */
    2870         218 :         GUInt32 enumcs = 16;
    2871         218 :         if (eColorSpace == CODEC::cvtenum(JP2_CLRSPC_SRGB))
    2872          16 :             enumcs = 16;
    2873         202 :         else if (eColorSpace == CODEC::cvtenum(JP2_CLRSPC_GRAY))
    2874         200 :             enumcs = 17;
    2875           2 :         else if (eColorSpace == CODEC::cvtenum(JP2_CLRSPC_SYCC))
    2876           2 :             enumcs = 18;
    2877         218 :         colrBox.AppendUInt32(enumcs); /* EnumCS: Enumerated colourspace */
    2878             : 
    2879         436 :         GDALJP2Box pclrBox(fp);
    2880         436 :         GDALJP2Box cmapBox(fp);
    2881         218 :         int nCTComponentCount = 0;
    2882         218 :         if (poCT != nullptr)
    2883             :         {
    2884           6 :             pclrBox.SetType("pclr");
    2885           6 :             const int nEntries = std::min(256, poCT->GetColorEntryCount());
    2886             :             nCTComponentCount =
    2887           6 :                 atoi(CSLFetchNameValueDef(papszOptions, "CT_COMPONENTS", "0"));
    2888           6 :             if (bInspireTG)
    2889             :             {
    2890           0 :                 if (nCTComponentCount != 0 && nCTComponentCount != 3)
    2891           0 :                     CPLError(
    2892             :                         CE_Warning, CPLE_AppDefined,
    2893             :                         "Inspire TG mandates 3 components for color table");
    2894             :                 else
    2895           0 :                     nCTComponentCount = 3;
    2896             :             }
    2897           6 :             else if (nCTComponentCount != 3 && nCTComponentCount != 4)
    2898             :             {
    2899           5 :                 nCTComponentCount = 3;
    2900          21 :                 for (int i = 0; i < nEntries; i++)
    2901             :                 {
    2902          17 :                     const GDALColorEntry *psEntry = poCT->GetColorEntry(i);
    2903          17 :                     if (psEntry->c4 != 255)
    2904             :                     {
    2905           1 :                         CPLDebug(
    2906             :                             CODEC::debugId(),
    2907             :                             "Color table has at least one non-opaque value. "
    2908             :                             "This may cause compatibility problems with some "
    2909             :                             "readers. "
    2910             :                             "In which case use CT_COMPONENTS=3 creation "
    2911             :                             "option");
    2912           1 :                         nCTComponentCount = 4;
    2913           1 :                         break;
    2914             :                     }
    2915             :                 }
    2916             :             }
    2917           6 :             nRedBandIndex = 0;
    2918           6 :             nGreenBandIndex = 1;
    2919           6 :             nBlueBandIndex = 2;
    2920           6 :             nAlphaBandIndex = (nCTComponentCount == 4) ? 3 : -1;
    2921             : 
    2922           6 :             pclrBox.AppendUInt16(static_cast<GUInt16>(nEntries));
    2923           6 :             pclrBox.AppendUInt8(static_cast<GByte>(
    2924             :                 nCTComponentCount)); /* NPC: Number of components */
    2925          25 :             for (int i = 0; i < nCTComponentCount; i++)
    2926             :             {
    2927          19 :                 pclrBox.AppendUInt8(7); /* Bi: unsigned 8 bits */
    2928             :             }
    2929          30 :             for (int i = 0; i < nEntries; i++)
    2930             :             {
    2931          24 :                 const GDALColorEntry *psEntry = poCT->GetColorEntry(i);
    2932          24 :                 pclrBox.AppendUInt8((GByte)psEntry->c1);
    2933          24 :                 pclrBox.AppendUInt8((GByte)psEntry->c2);
    2934          24 :                 pclrBox.AppendUInt8((GByte)psEntry->c3);
    2935          24 :                 if (nCTComponentCount == 4)
    2936           4 :                     pclrBox.AppendUInt8((GByte)psEntry->c4);
    2937             :             }
    2938             : 
    2939           6 :             cmapBox.SetType("cmap");
    2940          25 :             for (int i = 0; i < nCTComponentCount; i++)
    2941             :             {
    2942          19 :                 cmapBox.AppendUInt16(0); /* CMPi: code stream component index */
    2943          19 :                 cmapBox.AppendUInt8(1);  /* MYTPi: 1=palette mapping */
    2944          19 :                 cmapBox.AppendUInt8(static_cast<GByte>(
    2945             :                     i)); /* PCOLi: index component from the map */
    2946             :             }
    2947             :         }
    2948             : 
    2949         436 :         GDALJP2Box cdefBox(fp);
    2950         228 :         if (((nBands == 3 || nBands == 4) &&
    2951          22 :              (eColorSpace == CODEC::cvtenum(JP2_CLRSPC_SRGB) ||
    2952          18 :               eColorSpace == CODEC::cvtenum(JP2_CLRSPC_SYCC)) &&
    2953          10 :              (nRedBandIndex != 0 || nGreenBandIndex != 1 ||
    2954         436 :               nBlueBandIndex != 2)) ||
    2955             :             nAlphaBandIndex >= 0)
    2956             :         {
    2957          11 :             cdefBox.SetType("cdef");
    2958          11 :             int nComponents = (nCTComponentCount == 4) ? 4 : nBands;
    2959          11 :             cdefBox.AppendUInt16(static_cast<GUInt16>(nComponents));
    2960          50 :             for (int i = 0; i < nComponents; i++)
    2961             :             {
    2962          39 :                 cdefBox.AppendUInt16(
    2963             :                     static_cast<GUInt16>(i)); /* Component number */
    2964          39 :                 if (i != nAlphaBandIndex)
    2965             :                 {
    2966          29 :                     cdefBox.AppendUInt16(
    2967             :                         0); /* Signification: This channel is the colour image
    2968             :                                data for the associated colour */
    2969          29 :                     if (eColorSpace == CODEC::cvtenum(JP2_CLRSPC_GRAY) &&
    2970             :                         nComponents == 2)
    2971           2 :                         cdefBox.AppendUInt16(
    2972             :                             1); /* Colour of the component: associated with a
    2973             :                                    particular colour */
    2974          33 :                     else if ((eColorSpace == CODEC::cvtenum(JP2_CLRSPC_SRGB) ||
    2975          54 :                               eColorSpace == CODEC::cvtenum(JP2_CLRSPC_SYCC)) &&
    2976          21 :                              (nComponents == 3 || nComponents == 4))
    2977             :                     {
    2978          24 :                         if (i == nRedBandIndex)
    2979           8 :                             cdefBox.AppendUInt16(1);
    2980          16 :                         else if (i == nGreenBandIndex)
    2981           8 :                             cdefBox.AppendUInt16(2);
    2982           8 :                         else if (i == nBlueBandIndex)
    2983           8 :                             cdefBox.AppendUInt16(3);
    2984             :                         else
    2985             :                         {
    2986           0 :                             CPLError(CE_Warning, CPLE_AppDefined,
    2987             :                                      "Could not associate band %d with a "
    2988             :                                      "red/green/blue channel",
    2989             :                                      i + 1);
    2990           0 :                             cdefBox.AppendUInt16(65535);
    2991             :                         }
    2992             :                     }
    2993             :                     else
    2994           3 :                         cdefBox.AppendUInt16(
    2995             :                             65535); /* Colour of the component: not associated
    2996             :                                        with any particular colour */
    2997             :                 }
    2998             :                 else
    2999             :                 {
    3000          10 :                     cdefBox.AppendUInt16(
    3001             :                         1); /* Signification: Non pre-multiplied alpha */
    3002          10 :                     cdefBox.AppendUInt16(
    3003             :                         0); /* Colour of the component: This channel is
    3004             :                                associated as the image as a whole */
    3005             :                 }
    3006             :             }
    3007             :         }
    3008             : 
    3009             :         // Add res box if needed
    3010         218 :         GDALJP2Box *poRes = nullptr;
    3011         218 :         if (poSrcDS->GetMetadataItem("TIFFTAG_XRESOLUTION") != nullptr &&
    3012         223 :             poSrcDS->GetMetadataItem("TIFFTAG_YRESOLUTION") != nullptr &&
    3013           5 :             poSrcDS->GetMetadataItem("TIFFTAG_RESOLUTIONUNIT") != nullptr)
    3014             :         {
    3015             :             double dfXRes =
    3016           5 :                 CPLAtof(poSrcDS->GetMetadataItem("TIFFTAG_XRESOLUTION"));
    3017             :             double dfYRes =
    3018           5 :                 CPLAtof(poSrcDS->GetMetadataItem("TIFFTAG_YRESOLUTION"));
    3019             :             int nResUnit =
    3020           5 :                 atoi(poSrcDS->GetMetadataItem("TIFFTAG_RESOLUTIONUNIT"));
    3021             : #define PIXELS_PER_INCH 2
    3022             : #define PIXELS_PER_CM 3
    3023             : 
    3024           5 :             if (nResUnit == PIXELS_PER_INCH)
    3025             :             {
    3026             :                 // convert pixels per inch to pixels per cm.
    3027           2 :                 dfXRes = dfXRes * 39.37 / 100.0;
    3028           2 :                 dfYRes = dfYRes * 39.37 / 100.0;
    3029           2 :                 nResUnit = PIXELS_PER_CM;
    3030             :             }
    3031             : 
    3032           5 :             if (nResUnit == PIXELS_PER_CM && dfXRes > 0 && dfYRes > 0 &&
    3033           5 :                 dfXRes < 65535 && dfYRes < 65535)
    3034             :             {
    3035             :                 /* Format a resd box and embed it inside a res box */
    3036          10 :                 GDALJP2Box oResd;
    3037           5 :                 oResd.SetType("resd");
    3038             : 
    3039           5 :                 int nYDenom = 1;
    3040          58 :                 while (nYDenom < 32767 && dfYRes < 32767)
    3041             :                 {
    3042          53 :                     dfYRes *= 2;
    3043          53 :                     nYDenom *= 2;
    3044             :                 }
    3045           5 :                 int nXDenom = 1;
    3046          56 :                 while (nXDenom < 32767 && dfXRes < 32767)
    3047             :                 {
    3048          51 :                     dfXRes *= 2;
    3049          51 :                     nXDenom *= 2;
    3050             :                 }
    3051             : 
    3052           5 :                 oResd.AppendUInt16((GUInt16)dfYRes);
    3053           5 :                 oResd.AppendUInt16((GUInt16)nYDenom);
    3054           5 :                 oResd.AppendUInt16((GUInt16)dfXRes);
    3055           5 :                 oResd.AppendUInt16((GUInt16)nXDenom);
    3056           5 :                 oResd.AppendUInt8(2); /* vertical exponent */
    3057           5 :                 oResd.AppendUInt8(2); /* horizontal exponent */
    3058             : 
    3059           5 :                 GDALJP2Box *poResd = &oResd;
    3060           5 :                 poRes = GDALJP2Box::CreateAsocBox(1, &poResd);
    3061           5 :                 poRes->SetType("res ");
    3062             :             }
    3063             :         }
    3064             : 
    3065             :         /* Build and write jp2h super box now */
    3066             :         GDALJP2Box *apoBoxes[7];
    3067         218 :         int nBoxes = 1;
    3068         218 :         apoBoxes[0] = &ihdrBox;
    3069         218 :         if (bpccBox.GetDataLength())
    3070           3 :             apoBoxes[nBoxes++] = &bpccBox;
    3071         218 :         apoBoxes[nBoxes++] = &colrBox;
    3072         218 :         if (pclrBox.GetDataLength())
    3073           6 :             apoBoxes[nBoxes++] = &pclrBox;
    3074         218 :         if (cmapBox.GetDataLength())
    3075           6 :             apoBoxes[nBoxes++] = &cmapBox;
    3076         218 :         if (cdefBox.GetDataLength())
    3077          11 :             apoBoxes[nBoxes++] = &cdefBox;
    3078         218 :         if (poRes)
    3079           5 :             apoBoxes[nBoxes++] = poRes;
    3080             :         GDALJP2Box *psJP2HBox =
    3081         218 :             GDALJP2Box::CreateSuperBox("jp2h", nBoxes, apoBoxes);
    3082         218 :         WriteBox(fp, psJP2HBox);
    3083         218 :         delete psJP2HBox;
    3084         218 :         delete poRes;
    3085             : 
    3086         218 :         if (!bGeoBoxesAfter)
    3087             :         {
    3088         207 :             if (bGeoJP2Option && bGeoreferencingCompatOfGeoJP2)
    3089             :             {
    3090         150 :                 GDALJP2Box *poBox = oJP2MD.CreateJP2GeoTIFF();
    3091         150 :                 WriteBox(fp, poBox);
    3092         150 :                 delete poBox;
    3093             :             }
    3094             : 
    3095         218 :             if (CPLFetchBool(papszOptions, "WRITE_METADATA", false) &&
    3096          11 :                 !CPLFetchBool(papszOptions, "MAIN_MD_DOMAIN_ONLY", false))
    3097             :             {
    3098          11 :                 WriteXMPBox(fp, poSrcDS);
    3099             :             }
    3100             : 
    3101         207 :             if (CPLFetchBool(papszOptions, "WRITE_METADATA", false))
    3102             :             {
    3103          11 :                 if (!CPLFetchBool(papszOptions, "MAIN_MD_DOMAIN_ONLY", false))
    3104          11 :                     WriteXMLBoxes(fp, poSrcDS);
    3105          11 :                 WriteGDALMetadataBox(fp, poSrcDS, papszOptions);
    3106             :             }
    3107             : 
    3108         207 :             if (poGMLJP2Box != nullptr)
    3109             :             {
    3110          53 :                 WriteBox(fp, poGMLJP2Box);
    3111             :             }
    3112             :         }
    3113             :     }
    3114             : 
    3115             :     /* -------------------------------------------------------------------- */
    3116             :     /*      Try lossless reuse of an existing JPEG2000 codestream           */
    3117             :     /* -------------------------------------------------------------------- */
    3118         258 :     vsi_l_offset nCodeStreamLength = 0;
    3119         258 :     vsi_l_offset nCodeStreamStart = 0;
    3120         258 :     VSILFILE *fpSrc = nullptr;
    3121         258 :     if (CPLFetchBool(papszOptions, "USE_SRC_CODESTREAM", false))
    3122             :     {
    3123          14 :         CPLString osSrcFilename(poSrcDS->GetDescription());
    3124          14 :         if (poSrcDS->GetDriver() != nullptr &&
    3125           7 :             poSrcDS->GetDriver() == GDALGetDriverByName("VRT"))
    3126             :         {
    3127           0 :             VRTDataset *poVRTDS = (VRTDataset *)poSrcDS;
    3128           0 :             GDALDataset *poSimpleSourceDS = poVRTDS->GetSingleSimpleSource();
    3129           0 :             if (poSimpleSourceDS)
    3130           0 :                 osSrcFilename = poSimpleSourceDS->GetDescription();
    3131             :         }
    3132             : 
    3133           7 :         fpSrc = VSIFOpenL(osSrcFilename, "rb");
    3134           7 :         if (fpSrc)
    3135             :         {
    3136           7 :             nCodeStreamStart = JP2FindCodeStream(fpSrc, &nCodeStreamLength);
    3137             :         }
    3138           7 :         if (nCodeStreamLength == 0)
    3139             :         {
    3140           1 :             CPLError(
    3141             :                 CE_Warning, CPLE_AppDefined,
    3142             :                 "USE_SRC_CODESTREAM=YES specified, but no codestream found");
    3143             :         }
    3144             :     }
    3145             : 
    3146         258 :     if (eCodecFormat == CODEC::cvtenum(JP2_CODEC_JP2))
    3147             :     {
    3148             :         // Start codestream box
    3149         218 :         nStartJP2C = VSIFTellL(fp);
    3150         218 :         if (nCodeStreamLength)
    3151           6 :             bUseXLBoxes =
    3152           6 :                 ((vsi_l_offset)(GUInt32)nCodeStreamLength != nCodeStreamLength);
    3153             :         else
    3154         424 :             bUseXLBoxes = CPLFetchBool(papszOptions, "JP2C_XLBOX",
    3155         423 :                                        false) || /* For debugging */
    3156         211 :                           (GIntBig)nXSize * nYSize * nBands * nDataTypeSize /
    3157         211 :                                   adfRates.back() >
    3158             :                               4e9;
    3159         218 :         GUInt32 nLBox = (bUseXLBoxes) ? 1 : 0;
    3160         218 :         CPL_MSBPTR32(&nLBox);
    3161         218 :         VSIFWriteL(&nLBox, 1, 4, fp);
    3162         218 :         VSIFWriteL("jp2c", 1, 4, fp);
    3163         218 :         if (bUseXLBoxes)
    3164             :         {
    3165           1 :             GUIntBig nXLBox = 0;
    3166           1 :             VSIFWriteL(&nXLBox, 1, 8, fp);
    3167             :         }
    3168             :     }
    3169             : 
    3170             :     /* -------------------------------------------------------------------- */
    3171             :     /*      Do lossless reuse of an existing JPEG2000 codestream            */
    3172             :     /* -------------------------------------------------------------------- */
    3173         258 :     if (fpSrc)
    3174             :     {
    3175           7 :         const char *apszIgnoredOptions[] = {"BLOCKXSIZE",
    3176             :                                             "BLOCKYSIZE",
    3177             :                                             "QUALITY",
    3178             :                                             "REVERSIBLE",
    3179             :                                             "RESOLUTIONS",
    3180             :                                             "PROGRESSION",
    3181             :                                             "SOP",
    3182             :                                             "EPH",
    3183             :                                             "YCBCR420",
    3184             :                                             "YCC",
    3185             :                                             "NBITS",
    3186             :                                             "1BIT_ALPHA",
    3187             :                                             "PRECINCTS",
    3188             :                                             "TILEPARTS",
    3189             :                                             "CODEBLOCK_WIDTH",
    3190             :                                             "CODEBLOCK_HEIGHT",
    3191             :                                             "PLT",
    3192             :                                             nullptr};
    3193         126 :         for (int i = 0; apszIgnoredOptions[i]; i++)
    3194             :         {
    3195         119 :             if (CSLFetchNameValue(papszOptions, apszIgnoredOptions[i]))
    3196             :             {
    3197           1 :                 CPLError(CE_Warning, CPLE_NotSupported,
    3198             :                          "Option %s ignored when USE_SRC_CODESTREAM=YES",
    3199             :                          apszIgnoredOptions[i]);
    3200             :             }
    3201             :         }
    3202             :         GByte abyBuffer[4096];
    3203           7 :         VSIFSeekL(fpSrc, nCodeStreamStart, SEEK_SET);
    3204           7 :         vsi_l_offset nRead = 0;
    3205             :         /* coverity[tainted_data] */
    3206          17 :         while (nRead < nCodeStreamLength)
    3207             :         {
    3208          10 :             int nToRead = (nCodeStreamLength - nRead > 4096)
    3209             :                               ? 4096
    3210           6 :                               : (int)(nCodeStreamLength - nRead);
    3211          10 :             if ((int)VSIFReadL(abyBuffer, 1, nToRead, fpSrc) != nToRead)
    3212             :             {
    3213           0 :                 VSIFCloseL(fp);
    3214           0 :                 VSIFCloseL(fpSrc);
    3215           0 :                 delete poGMLJP2Box;
    3216           0 :                 return nullptr;
    3217             :             }
    3218          10 :             if (nRead == 0 && (pszProfile || bInspireTG) &&
    3219           6 :                 abyBuffer[2] == 0xFF && abyBuffer[3] == 0x51)
    3220             :             {
    3221           6 :                 if (EQUAL(pszProfile, "UNRESTRICTED"))
    3222             :                 {
    3223           0 :                     abyBuffer[6] = 0;
    3224           0 :                     abyBuffer[7] = 0;
    3225             :                 }
    3226           6 :                 else if (EQUAL(pszProfile, "PROFILE_1") || bInspireTG)
    3227             :                 {
    3228             :                     // TODO: ultimately we should check that we can really set
    3229             :                     // Profile 1
    3230           1 :                     abyBuffer[6] = 0;
    3231           1 :                     abyBuffer[7] = 2;
    3232             :                 }
    3233             :             }
    3234          20 :             if ((int)VSIFWriteL(abyBuffer, 1, nToRead, fp) != nToRead ||
    3235          10 :                 !pfnProgress((nRead + nToRead) * 1.0 / nCodeStreamLength,
    3236             :                              nullptr, pProgressData))
    3237             :             {
    3238           0 :                 VSIFCloseL(fp);
    3239           0 :                 VSIFCloseL(fpSrc);
    3240           0 :                 delete poGMLJP2Box;
    3241           0 :                 return nullptr;
    3242             :             }
    3243          10 :             nRead += nToRead;
    3244             :         }
    3245             : 
    3246           7 :         VSIFCloseL(fpSrc);
    3247             :     }
    3248             :     else
    3249             :     {
    3250         251 :         localctx.open(fp);
    3251         251 :         if (!localctx.initCompress(papszOptions, adfRates, nBlockXSize,
    3252             :                                    nBlockYSize, bIsIrreversible,
    3253             :                                    nNumResolutions, eProgOrder, bYCC, nCblockW,
    3254             :                                    nCblockH, bYCBCR420, bProfile1, nBands,
    3255             :                                    nXSize, nYSize, eColorSpace, numThreads))
    3256             :         {
    3257           0 :             CPLError(CE_Failure, CPLE_AppDefined, "init compress failed");
    3258           0 :             localctx.free();
    3259           0 :             VSIFCloseL(fp);
    3260           0 :             delete poGMLJP2Box;
    3261          11 :             return nullptr;
    3262             :         }
    3263         251 :         const int nTilesX = DIV_ROUND_UP(nXSize, nBlockXSize);
    3264         251 :         const int nTilesY = DIV_ROUND_UP(nYSize, nBlockYSize);
    3265             : 
    3266         251 :         const GUIntBig nTileSize =
    3267         251 :             (GUIntBig)nBlockXSize * nBlockYSize * nBands * nDataTypeSize;
    3268         251 :         GByte *pTempBuffer = nullptr;
    3269             : 
    3270         251 :         const bool bUseIOThread =
    3271         502 :             CODEC::preferPerTileCompress() && (nTilesX > 1 || nTilesY > 1) &&
    3272          11 :             nTileSize < 10 * 1024 * 1024 &&
    3273         513 :             strcmp(CPLGetThreadingModel(), "stub") != 0 &&
    3274          11 :             CPLTestBool(
    3275             :                 CPLGetConfigOption("JP2OPENJPEG_USE_THREADED_IO", "YES"));
    3276             : 
    3277         251 :         if (nTileSize > UINT_MAX)
    3278             :         {
    3279           1 :             CPLError(CE_Failure, CPLE_NotSupported, "Tile size exceeds 4GB");
    3280           1 :             pTempBuffer = nullptr;
    3281             :         }
    3282             :         else
    3283             :         {
    3284             :             // Double memory buffer when using threaded I/O
    3285         250 :             const size_t nBufferSize =
    3286             :                 static_cast<size_t>(bUseIOThread ? nTileSize * 2 : nTileSize);
    3287         250 :             pTempBuffer = (GByte *)VSIMalloc(nBufferSize);
    3288             :         }
    3289         251 :         if (pTempBuffer == nullptr)
    3290             :         {
    3291           1 :             localctx.free();
    3292           1 :             VSIFCloseL(fp);
    3293           1 :             delete poGMLJP2Box;
    3294           1 :             return nullptr;
    3295             :         }
    3296             : 
    3297         250 :         GByte *pYUV420Buffer = nullptr;
    3298         250 :         if (bYCBCR420)
    3299             :         {
    3300           4 :             pYUV420Buffer = (GByte *)VSIMalloc(
    3301           3 :                 3 * nBlockXSize * nBlockYSize / 2 +
    3302           1 :                 ((nBands == 4) ? nBlockXSize * nBlockYSize : 0));
    3303           2 :             if (pYUV420Buffer == nullptr)
    3304             :             {
    3305           0 :                 localctx.free();
    3306           0 :                 CPLFree(pTempBuffer);
    3307           0 :                 VSIFCloseL(fp);
    3308           0 :                 delete poGMLJP2Box;
    3309           0 :                 return nullptr;
    3310             :             }
    3311             :         }
    3312             : 
    3313             :         /* --------------------------------------------------------------------
    3314             :          */
    3315             :         /*      Iterate over the tiles */
    3316             :         /* --------------------------------------------------------------------
    3317             :          */
    3318         250 :         pfnProgress(0.0, nullptr, pProgressData);
    3319             : 
    3320             :         struct ReadRasterJob
    3321             :         {
    3322             :             GDALDataset *poSrcDS;
    3323             :             int nXOff;
    3324             :             int nYOff;
    3325             :             int nWidthToRead;
    3326             :             int nHeightToRead;
    3327             :             GDALDataType eDataType;
    3328             :             GByte *pBuffer;
    3329             :             int nBands;
    3330             :             CPLErr eErr;
    3331             :         };
    3332             : 
    3333         812 :         const auto ReadRasterFunction = [](void *threadData)
    3334             :         {
    3335         406 :             ReadRasterJob *job = static_cast<ReadRasterJob *>(threadData);
    3336         812 :             job->eErr = job->poSrcDS->RasterIO(
    3337             :                 GF_Read, job->nXOff, job->nYOff, job->nWidthToRead,
    3338         406 :                 job->nHeightToRead, job->pBuffer, job->nWidthToRead,
    3339             :                 job->nHeightToRead, job->eDataType, job->nBands, nullptr, 0, 0,
    3340             :                 0, nullptr);
    3341             :         };
    3342             : 
    3343         250 :         CPLWorkerThreadPool oPool;
    3344         250 :         if (bUseIOThread)
    3345             :         {
    3346          10 :             oPool.Setup(1, nullptr, nullptr);
    3347             :         }
    3348             : 
    3349         250 :         GByte *pabyActiveBuffer = pTempBuffer;
    3350         250 :         GByte *pabyBackgroundBuffer =
    3351         250 :             pTempBuffer + static_cast<size_t>(nTileSize);
    3352             : 
    3353         250 :         CPLErr eErr = CE_None;
    3354         250 :         int iTile = 0;
    3355             : 
    3356             :         ReadRasterJob job;
    3357         250 :         job.eDataType = eDataType;
    3358         250 :         job.pBuffer = pabyActiveBuffer;
    3359         250 :         job.nBands = nBands;
    3360         250 :         job.eErr = CE_Failure;
    3361         250 :         job.poSrcDS = poSrcDS;
    3362             : 
    3363         250 :         if (bUseIOThread)
    3364             :         {
    3365          10 :             job.nXOff = 0;
    3366          10 :             job.nYOff = 0;
    3367          10 :             job.nWidthToRead = std::min(nBlockXSize, nXSize);
    3368          10 :             job.nHeightToRead = std::min(nBlockYSize, nYSize);
    3369          10 :             job.pBuffer = pabyBackgroundBuffer;
    3370          10 :             ReadRasterFunction(&job);
    3371          10 :             eErr = job.eErr;
    3372             :         }
    3373             : 
    3374         524 :         for (int nBlockYOff = 0; eErr == CE_None && nBlockYOff < nTilesY;
    3375             :              nBlockYOff++)
    3376             :         {
    3377         680 :             for (int nBlockXOff = 0; eErr == CE_None && nBlockXOff < nTilesX;
    3378             :                  nBlockXOff++)
    3379             :             {
    3380         406 :                 const int nWidthToRead =
    3381         406 :                     std::min(nBlockXSize, nXSize - nBlockXOff * nBlockXSize);
    3382         406 :                 const int nHeightToRead =
    3383         406 :                     std::min(nBlockYSize, nYSize - nBlockYOff * nBlockYSize);
    3384             : 
    3385         406 :                 if (bUseIOThread)
    3386             :                 {
    3387             :                     // Wait for previous background I/O task to be finished
    3388         100 :                     oPool.WaitCompletion();
    3389         100 :                     eErr = job.eErr;
    3390             : 
    3391             :                     // Swap buffers
    3392         100 :                     std::swap(pabyBackgroundBuffer, pabyActiveBuffer);
    3393             : 
    3394             :                     // Prepare for next I/O task
    3395         100 :                     int nNextBlockXOff = nBlockXOff + 1;
    3396         100 :                     int nNextBlockYOff = nBlockYOff;
    3397         100 :                     if (nNextBlockXOff == nTilesX)
    3398             :                     {
    3399          26 :                         nNextBlockXOff = 0;
    3400          26 :                         nNextBlockYOff++;
    3401             :                     }
    3402         100 :                     if (nNextBlockYOff != nTilesY)
    3403             :                     {
    3404          90 :                         job.nXOff = nNextBlockXOff * nBlockXSize;
    3405          90 :                         job.nYOff = nNextBlockYOff * nBlockYSize;
    3406          90 :                         job.nWidthToRead =
    3407          90 :                             std::min(nBlockXSize, nXSize - job.nXOff);
    3408          90 :                         job.nHeightToRead =
    3409          90 :                             std::min(nBlockYSize, nYSize - job.nYOff);
    3410          90 :                         job.pBuffer = pabyBackgroundBuffer;
    3411             : 
    3412             :                         // Submit next job
    3413          90 :                         oPool.SubmitJob(ReadRasterFunction, &job);
    3414             :                     }
    3415             :                 }
    3416             :                 else
    3417             :                 {
    3418         306 :                     job.nXOff = nBlockXOff * nBlockXSize;
    3419         306 :                     job.nYOff = nBlockYOff * nBlockYSize;
    3420         306 :                     job.nWidthToRead = nWidthToRead;
    3421         306 :                     job.nHeightToRead = nHeightToRead;
    3422         306 :                     ReadRasterFunction(&job);
    3423         306 :                     eErr = job.eErr;
    3424             :                 }
    3425             : 
    3426         406 :                 if (b1BitAlpha)
    3427             :                 {
    3428       64987 :                     for (int i = 0; i < nWidthToRead * nHeightToRead; i++)
    3429             :                     {
    3430       64984 :                         if (pabyActiveBuffer[nAlphaBandIndex * nWidthToRead *
    3431       64984 :                                                  nHeightToRead +
    3432             :                                              i])
    3433       25040 :                             pabyActiveBuffer[nAlphaBandIndex * nWidthToRead *
    3434       25040 :                                                  nHeightToRead +
    3435       25040 :                                              i] = 1;
    3436             :                         else
    3437       39944 :                             pabyActiveBuffer[nAlphaBandIndex * nWidthToRead *
    3438       39944 :                                                  nHeightToRead +
    3439       39944 :                                              i] = 0;
    3440             :                     }
    3441             :                 }
    3442         406 :                 if (eErr == CE_None)
    3443             :                 {
    3444         406 :                     if (bYCBCR420)
    3445             :                     {
    3446             :                         int j, i;
    3447         202 :                         for (j = 0; j < nHeightToRead; j++)
    3448             :                         {
    3449       27000 :                             for (i = 0; i < nWidthToRead; i++)
    3450             :                             {
    3451       26800 :                                 int R = pabyActiveBuffer[j * nWidthToRead + i];
    3452       26800 :                                 int G = pabyActiveBuffer[nHeightToRead *
    3453       26800 :                                                              nWidthToRead +
    3454       26800 :                                                          j * nWidthToRead + i];
    3455       26800 :                                 int B = pabyActiveBuffer[2 * nHeightToRead *
    3456       26800 :                                                              nWidthToRead +
    3457       26800 :                                                          j * nWidthToRead + i];
    3458       26800 :                                 int Y =
    3459       26800 :                                     (int)(0.299 * R + 0.587 * G + 0.114 * B);
    3460       26800 :                                 int Cb =
    3461       26800 :                                     CLAMP_0_255((int)(-0.1687 * R - 0.3313 * G +
    3462       26800 :                                                       0.5 * B + 128));
    3463       26800 :                                 int Cr =
    3464       26800 :                                     CLAMP_0_255((int)(0.5 * R - 0.4187 * G -
    3465       26800 :                                                       0.0813 * B + 128));
    3466       26800 :                                 pYUV420Buffer[j * nWidthToRead + i] = (GByte)Y;
    3467       26800 :                                 pYUV420Buffer[nHeightToRead * nWidthToRead +
    3468       26800 :                                               ((j / 2) * ((nWidthToRead) / 2) +
    3469       26800 :                                                i / 2)] = (GByte)Cb;
    3470       26800 :                                 pYUV420Buffer[5 * nHeightToRead * nWidthToRead /
    3471       26800 :                                                   4 +
    3472       26800 :                                               ((j / 2) * ((nWidthToRead) / 2) +
    3473       26800 :                                                i / 2)] = (GByte)Cr;
    3474       26800 :                                 if (nBands == 4)
    3475             :                                 {
    3476       24300 :                                     pYUV420Buffer[3 * nHeightToRead *
    3477       24300 :                                                       nWidthToRead / 2 +
    3478       24300 :                                                   j * nWidthToRead + i] =
    3479       24300 :                                         (GByte)pabyActiveBuffer
    3480       24300 :                                             [3 * nHeightToRead * nWidthToRead +
    3481       24300 :                                              j * nWidthToRead + i];
    3482             :                                 }
    3483             :                             }
    3484             :                         }
    3485             : 
    3486           2 :                         int nBytesToWrite =
    3487           2 :                             3 * nWidthToRead * nHeightToRead / 2;
    3488           2 :                         if (nBands == 4)
    3489           1 :                             nBytesToWrite += nBlockXSize * nBlockYSize;
    3490             : 
    3491           2 :                         if (!localctx.compressTile(iTile, pYUV420Buffer,
    3492             :                                                    nBytesToWrite))
    3493             :                         {
    3494           0 :                             CPLError(CE_Failure, CPLE_AppDefined,
    3495             :                                      "compress tile failed");
    3496           0 :                             eErr = CE_Failure;
    3497             :                         }
    3498             :                     }
    3499             :                     else
    3500             :                     {
    3501         404 :                         if (!localctx.compressTile(iTile, pabyActiveBuffer,
    3502         404 :                                                    nWidthToRead *
    3503         404 :                                                        nHeightToRead * nBands *
    3504             :                                                        nDataTypeSize))
    3505             :                         {
    3506           0 :                             CPLError(CE_Failure, CPLE_AppDefined,
    3507             :                                      "compress tile failed");
    3508           0 :                             eErr = CE_Failure;
    3509             :                         }
    3510             :                     }
    3511             :                 }
    3512             : 
    3513         406 :                 if (!pfnProgress((iTile + 1) * 1.0 / (nTilesX * nTilesY),
    3514             :                                  nullptr, pProgressData))
    3515           0 :                     eErr = CE_Failure;
    3516             : 
    3517         406 :                 iTile++;
    3518             :             }
    3519             :         }
    3520             : 
    3521         250 :         if (bUseIOThread && eErr == CE_Failure)
    3522             :         {
    3523             :             // Wait for previous background I/O task to be finished
    3524             :             // before freeing buffers (pTempBuffer, etc.)
    3525           0 :             oPool.WaitCompletion();
    3526             :         }
    3527             : 
    3528         250 :         VSIFree(pTempBuffer);
    3529         250 :         VSIFree(pYUV420Buffer);
    3530             : 
    3531         250 :         if (eErr != CE_None)
    3532             :         {
    3533           0 :             localctx.free();
    3534           0 :             VSIFCloseL(fp);
    3535           0 :             delete poGMLJP2Box;
    3536           0 :             return nullptr;
    3537             :         }
    3538             : 
    3539         250 :         if (!localctx.finishCompress())
    3540             :         {
    3541          10 :             localctx.free();
    3542          10 :             VSIFCloseL(fp);
    3543          10 :             delete poGMLJP2Box;
    3544          10 :             return nullptr;
    3545             :         }
    3546         240 :         localctx.free();
    3547             :     }
    3548             : 
    3549             :     /* -------------------------------------------------------------------- */
    3550             :     /*      Patch JP2C box length and add trailing JP2 boxes                */
    3551             :     /* -------------------------------------------------------------------- */
    3552         247 :     bool bRet = true;
    3553         464 :     if (eCodecFormat == CODEC::cvtenum(JP2_CODEC_JP2) &&
    3554         217 :         !CPLFetchBool(papszOptions, "JP2C_LENGTH_ZERO",
    3555             :                       false) /* debug option */)
    3556             :     {
    3557         216 :         vsi_l_offset nEndJP2C = VSIFTellL(fp);
    3558         216 :         GUIntBig nBoxSize = nEndJP2C - nStartJP2C;
    3559         216 :         if (bUseXLBoxes)
    3560             :         {
    3561           1 :             VSIFSeekL(fp, nStartJP2C + 8, SEEK_SET);
    3562           1 :             CPL_MSBPTR64(&nBoxSize);
    3563           1 :             if (VSIFWriteL(&nBoxSize, 8, 1, fp) != 1)
    3564           0 :                 bRet = false;
    3565             :         }
    3566             :         else
    3567             :         {
    3568         215 :             GUInt32 nBoxSize32 = (GUInt32)nBoxSize;
    3569         215 :             if ((vsi_l_offset)nBoxSize32 != nBoxSize)
    3570             :             {
    3571             :                 /*  Should not happen hopefully */
    3572           0 :                 if ((bGeoreferencingCompatOfGeoJP2 || poGMLJP2Box) &&
    3573             :                     bGeoBoxesAfter)
    3574             :                 {
    3575           0 :                     CPLError(CE_Warning, CPLE_AppDefined,
    3576             :                              "Cannot write GMLJP2/GeoJP2 boxes as codestream "
    3577             :                              "is unexpectedly > 4GB");
    3578           0 :                     bGeoreferencingCompatOfGeoJP2 = FALSE;
    3579           0 :                     delete poGMLJP2Box;
    3580           0 :                     poGMLJP2Box = nullptr;
    3581             :                 }
    3582             :             }
    3583             :             else
    3584             :             {
    3585         215 :                 VSIFSeekL(fp, nStartJP2C, SEEK_SET);
    3586         215 :                 CPL_MSBPTR32(&nBoxSize32);
    3587         215 :                 if (VSIFWriteL(&nBoxSize32, 4, 1, fp) != 1)
    3588           0 :                     bRet = false;
    3589             :             }
    3590             :         }
    3591         216 :         VSIFSeekL(fp, 0, SEEK_END);
    3592             : 
    3593         216 :         if (CPLFetchBool(papszOptions, "WRITE_METADATA", false))
    3594             :         {
    3595          14 :             if (!WriteIPRBox(fp, poSrcDS))
    3596           0 :                 bRet = false;
    3597             :         }
    3598             : 
    3599         216 :         if (bGeoBoxesAfter)
    3600             :         {
    3601          11 :             if (poGMLJP2Box != nullptr)
    3602             :             {
    3603           5 :                 if (!WriteBox(fp, poGMLJP2Box))
    3604           0 :                     bRet = false;
    3605             :             }
    3606             : 
    3607          11 :             if (CPLFetchBool(papszOptions, "WRITE_METADATA", false))
    3608             :             {
    3609           3 :                 if (!CPLFetchBool(papszOptions, "MAIN_MD_DOMAIN_ONLY", false))
    3610             :                 {
    3611           3 :                     if (!WriteXMLBoxes(fp, poSrcDS))
    3612           0 :                         bRet = false;
    3613             :                 }
    3614           3 :                 if (!WriteGDALMetadataBox(fp, poSrcDS, papszOptions))
    3615           0 :                     bRet = false;
    3616             :             }
    3617             : 
    3618          11 :             if (bGeoJP2Option && bGeoreferencingCompatOfGeoJP2)
    3619             :             {
    3620           5 :                 GDALJP2Box *poBox = oJP2MD.CreateJP2GeoTIFF();
    3621           5 :                 if (!WriteBox(fp, poBox))
    3622           0 :                     bRet = false;
    3623           5 :                 delete poBox;
    3624             :             }
    3625             : 
    3626          14 :             if (CPLFetchBool(papszOptions, "WRITE_METADATA", false) &&
    3627           3 :                 !CPLFetchBool(papszOptions, "MAIN_MD_DOMAIN_ONLY", false))
    3628             :             {
    3629           3 :                 if (!WriteXMPBox(fp, poSrcDS))
    3630           0 :                     bRet = false;
    3631             :             }
    3632             :         }
    3633             :     }
    3634             : 
    3635         247 :     if (VSIFCloseL(fp) != 0)
    3636           0 :         bRet = false;
    3637         247 :     delete poGMLJP2Box;
    3638         247 :     if (!bRet)
    3639           0 :         return nullptr;
    3640             : 
    3641             :     /* -------------------------------------------------------------------- */
    3642             :     /*      Re-open dataset, and copy any auxiliary pam information.        */
    3643             :     /* -------------------------------------------------------------------- */
    3644             : 
    3645         247 :     GDALOpenInfo oOpenInfo(pszFilename, GA_ReadOnly);
    3646         247 :     auto poDS = (JP2OPJLikeDataset *)JP2OPJLikeDataset::Open(&oOpenInfo);
    3647             : 
    3648         247 :     if (poDS)
    3649             :     {
    3650         244 :         poDS->CloneInfo(poSrcDS, GCIF_PAM_DEFAULT & (~GCIF_METADATA));
    3651             : 
    3652             :         /* Only write relevant metadata to PAM, and if needed */
    3653         244 :         if (!CPLFetchBool(papszOptions, "WRITE_METADATA", false))
    3654             :         {
    3655         230 :             char **papszSrcMD = CSLDuplicate(poSrcDS->GetMetadata());
    3656             :             papszSrcMD =
    3657         230 :                 CSLSetNameValue(papszSrcMD, GDALMD_AREA_OR_POINT, nullptr);
    3658         230 :             papszSrcMD = CSLSetNameValue(papszSrcMD, "Corder", nullptr);
    3659         321 :             for (char **papszSrcMDIter = papszSrcMD;
    3660         321 :                  papszSrcMDIter && *papszSrcMDIter;)
    3661             :             {
    3662             :                 /* Remove entries like KEY= (without value) */
    3663          91 :                 if ((*papszSrcMDIter)[0] &&
    3664          91 :                     (*papszSrcMDIter)[strlen((*papszSrcMDIter)) - 1] == '=')
    3665             :                 {
    3666          37 :                     CPLFree(*papszSrcMDIter);
    3667          37 :                     memmove(papszSrcMDIter, papszSrcMDIter + 1,
    3668             :                             sizeof(char *) *
    3669          37 :                                 (CSLCount(papszSrcMDIter + 1) + 1));
    3670             :                 }
    3671             :                 else
    3672          54 :                     ++papszSrcMDIter;
    3673             :             }
    3674         230 :             char **papszMD = CSLDuplicate(poDS->GetMetadata());
    3675         230 :             papszMD = CSLSetNameValue(papszMD, GDALMD_AREA_OR_POINT, nullptr);
    3676         245 :             if (papszSrcMD && papszSrcMD[0] != nullptr &&
    3677          15 :                 CSLCount(papszSrcMD) != CSLCount(papszMD))
    3678             :             {
    3679           9 :                 poDS->SetMetadata(papszSrcMD);
    3680             :             }
    3681         230 :             CSLDestroy(papszSrcMD);
    3682         230 :             CSLDestroy(papszMD);
    3683             :         }
    3684             :     }
    3685             : 
    3686         247 :     return poDS;
    3687             : }
    3688             : 
    3689             : #ifdef unused
    3690             : template <typename CODEC, typename BASE>
    3691             : void GDALRegisterJP2(const std::string &libraryName,
    3692             :                      const std::string &driverName)
    3693             : 
    3694             : {
    3695             :     if (!GDAL_CHECK_VERSION((driverName + " driver").c_str()))
    3696             :         return;
    3697             : 
    3698             :     if (GDALGetDriverByName(driverName.c_str()) != nullptr)
    3699             :         return;
    3700             : 
    3701             :     GDALDriver *poDriver = new GDALDriver();
    3702             :     poDriver->SetDescription(driverName.c_str());
    3703             :     poDriver->SetMetadataItem(GDAL_DCAP_RASTER, "YES");
    3704             :     poDriver->SetMetadataItem(GDAL_DCAP_VECTOR, "YES");
    3705             :     poDriver->SetMetadataItem(
    3706             :         GDAL_DMD_LONGNAME,
    3707             :         ("JPEG-2000 driver based on " + libraryName + " library").c_str());
    3708             : 
    3709             :     poDriver->SetMetadataItem(
    3710             :         GDAL_DMD_HELPTOPIC,
    3711             :         ("drivers/raster/jp2" + CPLString(libraryName).tolower() + ".html")
    3712             :             .c_str());
    3713             :     poDriver->SetMetadataItem(GDAL_DMD_MIMETYPE, "image/jp2");
    3714             :     poDriver->SetMetadataItem(GDAL_DMD_EXTENSION, "jp2");
    3715             :     poDriver->SetMetadataItem(GDAL_DMD_EXTENSIONS, "jp2 j2k");
    3716             :     poDriver->SetMetadataItem(GDAL_DMD_CREATIONDATATYPES,
    3717             :                               "Byte Int16 UInt16 Int32 UInt32");
    3718             : 
    3719             :     poDriver->SetMetadataItem(GDAL_DCAP_VIRTUALIO, "YES");
    3720             :     BASE::setMetaData(poDriver);
    3721             : 
    3722             :     poDriver->pfnIdentify = JP2OPJLikeDataset<CODEC, BASE>::Identify;
    3723             :     poDriver->pfnOpen = JP2OPJLikeDataset<CODEC, BASE>::Open;
    3724             :     poDriver->pfnCreateCopy = JP2OPJLikeDataset<CODEC, BASE>::CreateCopy;
    3725             : 
    3726             :     GetGDALDriverManager()->RegisterDriver(poDriver);
    3727             : }
    3728             : #endif

Generated by: LCOV version 1.14