LCOV - code coverage report
Current view: top level - frmts/webp - webpdataset.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 341 470 72.6 %
Date: 2025-01-18 12:42:00 Functions: 17 18 94.4 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  GDAL WEBP Driver
       4             :  * Purpose:  Implement GDAL WEBP Support based on libwebp
       5             :  * Author:   Even Rouault, <even dot rouault at spatialys.com>
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2011-2013, Even Rouault <even dot rouault at spatialys.com>
       9             :  *
      10             :  * SPDX-License-Identifier: MIT
      11             :  ****************************************************************************/
      12             : 
      13             : #include "cpl_string.h"
      14             : #include "gdal_frmts.h"
      15             : #include "gdal_pam.h"
      16             : 
      17             : #include "webp_headers.h"
      18             : #include "webpdrivercore.h"
      19             : 
      20             : #include <limits>
      21             : 
      22             : /************************************************************************/
      23             : /* ==================================================================== */
      24             : /*                               WEBPDataset                            */
      25             : /* ==================================================================== */
      26             : /************************************************************************/
      27             : 
      28             : class WEBPRasterBand;
      29             : 
      30             : class WEBPDataset final : public GDALPamDataset
      31             : {
      32             :     friend class WEBPRasterBand;
      33             : 
      34             :     VSILFILE *fpImage;
      35             :     GByte *pabyUncompressed;
      36             :     int bHasBeenUncompressed;
      37             :     CPLErr eUncompressErrRet;
      38             :     CPLErr Uncompress();
      39             : 
      40             :     int bHasReadXMPMetadata;
      41             : 
      42             :   public:
      43             :     WEBPDataset();
      44             :     virtual ~WEBPDataset();
      45             : 
      46             :     virtual CPLErr IRasterIO(GDALRWFlag, int, int, int, int, void *, int, int,
      47             :                              GDALDataType, int, BANDMAP_TYPE,
      48             :                              GSpacing nPixelSpace, GSpacing nLineSpace,
      49             :                              GSpacing nBandSpace,
      50             :                              GDALRasterIOExtraArg *psExtraArg) override;
      51             : 
      52             :     virtual char **GetMetadataDomainList() override;
      53             :     virtual char **GetMetadata(const char *pszDomain = "") override;
      54             : 
      55             :     CPLStringList GetCompressionFormats(int nXOff, int nYOff, int nXSize,
      56             :                                         int nYSize, int nBandCount,
      57             :                                         const int *panBandList) override;
      58             :     CPLErr ReadCompressedData(const char *pszFormat, int nXOff, int nYOff,
      59             :                               int nXSize, int nYSize, int nBandCount,
      60             :                               const int *panBandList, void **ppBuffer,
      61             :                               size_t *pnBufferSize,
      62             :                               char **ppszDetailedFormat) override;
      63             : 
      64             :     static GDALPamDataset *OpenPAM(GDALOpenInfo *poOpenInfo);
      65             :     static GDALDataset *Open(GDALOpenInfo *);
      66             :     static GDALDataset *CreateCopy(const char *pszFilename,
      67             :                                    GDALDataset *poSrcDS, int bStrict,
      68             :                                    char **papszOptions,
      69             :                                    GDALProgressFunc pfnProgress,
      70             :                                    void *pProgressData);
      71             : };
      72             : 
      73             : /************************************************************************/
      74             : /* ==================================================================== */
      75             : /*                            WEBPRasterBand                            */
      76             : /* ==================================================================== */
      77             : /************************************************************************/
      78             : 
      79             : class WEBPRasterBand final : public GDALPamRasterBand
      80             : {
      81             :     friend class WEBPDataset;
      82             : 
      83             :   public:
      84             :     WEBPRasterBand(WEBPDataset *, int);
      85             : 
      86             :     virtual CPLErr IReadBlock(int, int, void *) override;
      87             :     virtual GDALColorInterp GetColorInterpretation() override;
      88             : };
      89             : 
      90             : /************************************************************************/
      91             : /*                          WEBPRasterBand()                            */
      92             : /************************************************************************/
      93             : 
      94        1697 : WEBPRasterBand::WEBPRasterBand(WEBPDataset *poDSIn, int)
      95             : {
      96        1697 :     poDS = poDSIn;
      97             : 
      98        1697 :     eDataType = GDT_Byte;
      99             : 
     100        1697 :     nBlockXSize = poDSIn->nRasterXSize;
     101        1697 :     nBlockYSize = 1;
     102        1697 : }
     103             : 
     104             : /************************************************************************/
     105             : /*                             IReadBlock()                             */
     106             : /************************************************************************/
     107             : 
     108       11626 : CPLErr WEBPRasterBand::IReadBlock(CPL_UNUSED int nBlockXOff, int nBlockYOff,
     109             :                                   void *pImage)
     110             : {
     111       11626 :     WEBPDataset *poGDS = reinterpret_cast<WEBPDataset *>(poDS);
     112             : 
     113       11626 :     if (poGDS->Uncompress() != CE_None)
     114           0 :         return CE_Failure;
     115             : 
     116       11626 :     GByte *pabyUncompressed =
     117       11626 :         &poGDS->pabyUncompressed[nBlockYOff * nRasterXSize * poGDS->nBands +
     118       11626 :                                  nBand - 1];
     119     2882680 :     for (int i = 0; i < nRasterXSize; i++)
     120     2871060 :         reinterpret_cast<GByte *>(pImage)[i] =
     121     2871060 :             pabyUncompressed[poGDS->nBands * i];
     122             : 
     123       11626 :     return CE_None;
     124             : }
     125             : 
     126             : /************************************************************************/
     127             : /*                       GetColorInterpretation()                       */
     128             : /************************************************************************/
     129             : 
     130          78 : GDALColorInterp WEBPRasterBand::GetColorInterpretation()
     131             : 
     132             : {
     133          78 :     if (nBand == 1)
     134          25 :         return GCI_RedBand;
     135             : 
     136          53 :     else if (nBand == 2)
     137          25 :         return GCI_GreenBand;
     138             : 
     139          28 :     else if (nBand == 3)
     140          25 :         return GCI_BlueBand;
     141             : 
     142           3 :     return GCI_AlphaBand;
     143             : }
     144             : 
     145             : /************************************************************************/
     146             : /* ==================================================================== */
     147             : /*                             WEBPDataset                               */
     148             : /* ==================================================================== */
     149             : /************************************************************************/
     150             : 
     151             : /************************************************************************/
     152             : /*                            WEBPDataset()                              */
     153             : /************************************************************************/
     154             : 
     155         467 : WEBPDataset::WEBPDataset()
     156             :     : fpImage(nullptr), pabyUncompressed(nullptr), bHasBeenUncompressed(FALSE),
     157         467 :       eUncompressErrRet(CE_None), bHasReadXMPMetadata(FALSE)
     158             : {
     159         467 : }
     160             : 
     161             : /************************************************************************/
     162             : /*                           ~WEBPDataset()                             */
     163             : /************************************************************************/
     164             : 
     165         934 : WEBPDataset::~WEBPDataset()
     166             : 
     167             : {
     168         467 :     FlushCache(true);
     169         467 :     if (fpImage)
     170         467 :         VSIFCloseL(fpImage);
     171         467 :     VSIFree(pabyUncompressed);
     172         934 : }
     173             : 
     174             : /************************************************************************/
     175             : /*                      GetMetadataDomainList()                         */
     176             : /************************************************************************/
     177             : 
     178           1 : char **WEBPDataset::GetMetadataDomainList()
     179             : {
     180           1 :     return BuildMetadataDomainList(GDALPamDataset::GetMetadataDomainList(),
     181           1 :                                    TRUE, "xml:XMP", nullptr);
     182             : }
     183             : 
     184             : /************************************************************************/
     185             : /*                           GetMetadata()                              */
     186             : /************************************************************************/
     187             : 
     188          52 : char **WEBPDataset::GetMetadata(const char *pszDomain)
     189             : {
     190          52 :     if ((pszDomain != nullptr && EQUAL(pszDomain, "xml:XMP")) &&
     191           4 :         !bHasReadXMPMetadata)
     192             :     {
     193           4 :         bHasReadXMPMetadata = TRUE;
     194             : 
     195           4 :         VSIFSeekL(fpImage, 12, SEEK_SET);
     196             : 
     197           4 :         bool bFirst = true;
     198             :         while (true)
     199             :         {
     200             :             char szHeader[5];
     201             :             GUInt32 nChunkSize;
     202             : 
     203          16 :             if (VSIFReadL(szHeader, 1, 4, fpImage) != 4 ||
     204           8 :                 VSIFReadL(&nChunkSize, 1, 4, fpImage) != 4)
     205           4 :                 break;
     206             : 
     207           8 :             szHeader[4] = '\0';
     208           8 :             CPL_LSBPTR32(&nChunkSize);
     209             : 
     210           8 :             if (bFirst)
     211             :             {
     212           4 :                 if (strcmp(szHeader, "VP8X") != 0 || nChunkSize < 10)
     213             :                     break;
     214             : 
     215             :                 int l_nFlags;
     216           2 :                 if (VSIFReadL(&l_nFlags, 1, 4, fpImage) != 4)
     217           0 :                     break;
     218           2 :                 CPL_LSBPTR32(&l_nFlags);
     219           2 :                 if ((l_nFlags & 8) == 0)
     220           0 :                     break;
     221             : 
     222           2 :                 VSIFSeekL(fpImage, nChunkSize - 4, SEEK_CUR);
     223             : 
     224           2 :                 bFirst = false;
     225             :             }
     226           4 :             else if (strcmp(szHeader, "META") == 0)
     227             :             {
     228           2 :                 if (nChunkSize > 1024 * 1024)
     229           0 :                     break;
     230             : 
     231             :                 char *pszXMP =
     232           2 :                     reinterpret_cast<char *>(VSIMalloc(nChunkSize + 1));
     233           2 :                 if (pszXMP == nullptr)
     234           0 :                     break;
     235             : 
     236           2 :                 if (static_cast<GUInt32>(VSIFReadL(pszXMP, 1, nChunkSize,
     237           2 :                                                    fpImage)) != nChunkSize)
     238             :                 {
     239           0 :                     VSIFree(pszXMP);
     240           0 :                     break;
     241             :                 }
     242           2 :                 pszXMP[nChunkSize] = '\0';
     243             : 
     244             :                 /* Avoid setting the PAM dirty bit just for that */
     245           2 :                 const int nOldPamFlags = nPamFlags;
     246             : 
     247           2 :                 char *apszMDList[2] = {pszXMP, nullptr};
     248           2 :                 SetMetadata(apszMDList, "xml:XMP");
     249             : 
     250             :                 // cppcheck-suppress redundantAssignment
     251           2 :                 nPamFlags = nOldPamFlags;
     252             : 
     253           2 :                 VSIFree(pszXMP);
     254           2 :                 break;
     255             :             }
     256             :             else
     257           2 :                 VSIFSeekL(fpImage, nChunkSize, SEEK_CUR);
     258           4 :         }
     259             :     }
     260             : 
     261          52 :     return GDALPamDataset::GetMetadata(pszDomain);
     262             : }
     263             : 
     264             : /************************************************************************/
     265             : /*                            Uncompress()                              */
     266             : /************************************************************************/
     267             : 
     268       11943 : CPLErr WEBPDataset::Uncompress()
     269             : {
     270       11943 :     if (bHasBeenUncompressed)
     271       11605 :         return eUncompressErrRet;
     272             : 
     273         338 :     bHasBeenUncompressed = TRUE;
     274         338 :     eUncompressErrRet = CE_Failure;
     275             : 
     276             :     // To avoid excessive memory allocation attempts
     277             :     // Normally WebP images are no larger than 16383x16383*4 ~= 1 GB
     278         338 :     if (nRasterXSize > INT_MAX / (nRasterYSize * nBands))
     279             :     {
     280           0 :         CPLError(CE_Failure, CPLE_NotSupported, "Too large image");
     281           0 :         return CE_Failure;
     282             :     }
     283             : 
     284         338 :     pabyUncompressed = reinterpret_cast<GByte *>(
     285         338 :         VSIMalloc3(nRasterXSize, nRasterYSize, nBands));
     286         338 :     if (pabyUncompressed == nullptr)
     287           0 :         return CE_Failure;
     288             : 
     289         338 :     VSIFSeekL(fpImage, 0, SEEK_END);
     290         338 :     vsi_l_offset nSizeLarge = VSIFTellL(fpImage);
     291         338 :     if (nSizeLarge !=
     292         338 :         static_cast<vsi_l_offset>(static_cast<uint32_t>(nSizeLarge)))
     293           0 :         return CE_Failure;
     294         338 :     VSIFSeekL(fpImage, 0, SEEK_SET);
     295         338 :     uint32_t nSize = static_cast<uint32_t>(nSizeLarge);
     296         338 :     uint8_t *pabyCompressed = reinterpret_cast<uint8_t *>(VSIMalloc(nSize));
     297         338 :     if (pabyCompressed == nullptr)
     298           0 :         return CE_Failure;
     299         338 :     VSIFReadL(pabyCompressed, 1, nSize, fpImage);
     300             :     uint8_t *pRet;
     301             : 
     302         338 :     if (nBands == 4)
     303         216 :         pRet = WebPDecodeRGBAInto(pabyCompressed, static_cast<uint32_t>(nSize),
     304         216 :                                   static_cast<uint8_t *>(pabyUncompressed),
     305         216 :                                   static_cast<size_t>(nRasterXSize) *
     306         216 :                                       nRasterYSize * nBands,
     307         216 :                                   nRasterXSize * nBands);
     308             :     else
     309         122 :         pRet = WebPDecodeRGBInto(pabyCompressed, static_cast<uint32_t>(nSize),
     310         122 :                                  static_cast<uint8_t *>(pabyUncompressed),
     311         122 :                                  static_cast<size_t>(nRasterXSize) *
     312         122 :                                      nRasterYSize * nBands,
     313         122 :                                  nRasterXSize * nBands);
     314             : 
     315         338 :     VSIFree(pabyCompressed);
     316         338 :     if (pRet == nullptr)
     317             :     {
     318           0 :         CPLError(CE_Failure, CPLE_AppDefined, "WebPDecodeRGBInto() failed");
     319           0 :         return CE_Failure;
     320             :     }
     321         338 :     eUncompressErrRet = CE_None;
     322             : 
     323         338 :     return CE_None;
     324             : }
     325             : 
     326             : /************************************************************************/
     327             : /*                             IRasterIO()                              */
     328             : /************************************************************************/
     329             : 
     330         321 : CPLErr WEBPDataset::IRasterIO(GDALRWFlag eRWFlag, int nXOff, int nYOff,
     331             :                               int nXSize, int nYSize, void *pData,
     332             :                               int nBufXSize, int nBufYSize,
     333             :                               GDALDataType eBufType, int nBandCount,
     334             :                               BANDMAP_TYPE panBandMap, GSpacing nPixelSpace,
     335             :                               GSpacing nLineSpace, GSpacing nBandSpace,
     336             :                               GDALRasterIOExtraArg *psExtraArg)
     337             : 
     338             : {
     339         321 :     if ((eRWFlag == GF_Read) && (nBandCount == nBands) && (nXOff == 0) &&
     340         321 :         (nYOff == 0) && (nXSize == nBufXSize) && (nXSize == nRasterXSize) &&
     341         317 :         (nYSize == nBufYSize) && (nYSize == nRasterYSize) &&
     342         317 :         (eBufType == GDT_Byte) && (pData != nullptr) && (panBandMap[0] == 1) &&
     343         317 :         (panBandMap[1] == 2) && (panBandMap[2] == 3) &&
     344         317 :         (nBands == 3 || panBandMap[3] == 4))
     345             :     {
     346         317 :         if (Uncompress() != CE_None)
     347           0 :             return CE_Failure;
     348         317 :         if (nPixelSpace == nBands && nLineSpace == (nPixelSpace * nXSize) &&
     349             :             nBandSpace == 1)
     350             :         {
     351          77 :             memcpy(pData, pabyUncompressed,
     352          77 :                    static_cast<size_t>(nBands) * nXSize * nYSize);
     353             :         }
     354             :         else
     355             :         {
     356        8024 :             for (int y = 0; y < nYSize; ++y)
     357             :             {
     358        7784 :                 GByte *pabyScanline = pabyUncompressed + y * nBands * nXSize;
     359     1101290 :                 for (int x = 0; x < nXSize; ++x)
     360             :                 {
     361     5276670 :                     for (int iBand = 0; iBand < nBands; iBand++)
     362             :                         reinterpret_cast<GByte *>(
     363     4183170 :                             pData)[(y * nLineSpace) + (x * nPixelSpace) +
     364     4183170 :                                    iBand * nBandSpace] =
     365     4183170 :                             pabyScanline[x * nBands + iBand];
     366             :                 }
     367             :             }
     368             :         }
     369             : 
     370         317 :         return CE_None;
     371             :     }
     372             : 
     373           4 :     return GDALPamDataset::IRasterIO(eRWFlag, nXOff, nYOff, nXSize, nYSize,
     374             :                                      pData, nBufXSize, nBufYSize, eBufType,
     375             :                                      nBandCount, panBandMap, nPixelSpace,
     376           4 :                                      nLineSpace, nBandSpace, psExtraArg);
     377             : }
     378             : 
     379             : /************************************************************************/
     380             : /*                       GetCompressionFormats()                        */
     381             : /************************************************************************/
     382             : 
     383           0 : CPLStringList WEBPDataset::GetCompressionFormats(int nXOff, int nYOff,
     384             :                                                  int nXSize, int nYSize,
     385             :                                                  int nBandCount,
     386             :                                                  const int *panBandList)
     387             : {
     388           0 :     CPLStringList aosRet;
     389           0 :     if (nXOff == 0 && nYOff == 0 && nXSize == nRasterXSize &&
     390           0 :         nYSize == nRasterYSize && IsAllBands(nBandCount, panBandList))
     391             :     {
     392           0 :         aosRet.AddString("WEBP");
     393             :     }
     394           0 :     return aosRet;
     395             : }
     396             : 
     397             : /************************************************************************/
     398             : /*                       ReadCompressedData()                           */
     399             : /************************************************************************/
     400             : 
     401           2 : CPLErr WEBPDataset::ReadCompressedData(const char *pszFormat, int nXOff,
     402             :                                        int nYOff, int nXSize, int nYSize,
     403             :                                        int nBandCount, const int *panBandList,
     404             :                                        void **ppBuffer, size_t *pnBufferSize,
     405             :                                        char **ppszDetailedFormat)
     406             : {
     407           2 :     if (nXOff == 0 && nYOff == 0 && nXSize == nRasterXSize &&
     408           4 :         nYSize == nRasterYSize && IsAllBands(nBandCount, panBandList))
     409             :     {
     410           2 :         const CPLStringList aosTokens(CSLTokenizeString2(pszFormat, ";", 0));
     411           2 :         if (aosTokens.size() != 1)
     412           0 :             return CE_Failure;
     413             : 
     414           2 :         if (EQUAL(aosTokens[0], "WEBP"))
     415             :         {
     416           2 :             if (ppszDetailedFormat)
     417           0 :                 *ppszDetailedFormat = VSIStrdup("WEBP");
     418           2 :             VSIFSeekL(fpImage, 0, SEEK_END);
     419           2 :             const auto nFileSize = VSIFTellL(fpImage);
     420           2 :             if (nFileSize > std::numeric_limits<uint32_t>::max())
     421           0 :                 return CE_Failure;
     422           2 :             auto nSize = static_cast<uint32_t>(nFileSize);
     423           2 :             if (ppBuffer)
     424             :             {
     425           2 :                 if (!pnBufferSize)
     426           0 :                     return CE_Failure;
     427           2 :                 bool bFreeOnError = false;
     428           2 :                 if (*ppBuffer)
     429             :                 {
     430           0 :                     if (*pnBufferSize < nSize)
     431           0 :                         return CE_Failure;
     432             :                 }
     433             :                 else
     434             :                 {
     435           2 :                     *ppBuffer = VSI_MALLOC_VERBOSE(nSize);
     436           2 :                     if (*ppBuffer == nullptr)
     437           0 :                         return CE_Failure;
     438           2 :                     bFreeOnError = true;
     439             :                 }
     440           2 :                 VSIFSeekL(fpImage, 0, SEEK_SET);
     441           2 :                 if (VSIFReadL(*ppBuffer, nSize, 1, fpImage) != 1)
     442             :                 {
     443           0 :                     if (bFreeOnError)
     444             :                     {
     445           0 :                         VSIFree(*ppBuffer);
     446           0 :                         *ppBuffer = nullptr;
     447             :                     }
     448           0 :                     return CE_Failure;
     449             :                 }
     450             : 
     451             :                 // Remove META box
     452           2 :                 if (nSize > 12 && memcmp(*ppBuffer, "RIFF", 4) == 0)
     453             :                 {
     454           2 :                     size_t nPos = 12;
     455           2 :                     GByte *pabyData = static_cast<GByte *>(*ppBuffer);
     456           6 :                     while (nPos <= nSize - 8)
     457             :                     {
     458           4 :                         char szBoxName[5] = {0, 0, 0, 0, 0};
     459           4 :                         memcpy(szBoxName, pabyData + nPos, 4);
     460             :                         uint32_t nChunkSize;
     461           4 :                         memcpy(&nChunkSize, pabyData + nPos + 4, 4);
     462           4 :                         CPL_LSBPTR32(&nChunkSize);
     463           4 :                         if (nChunkSize % 2)  // Payload padding if needed
     464           1 :                             nChunkSize++;
     465           4 :                         if (nChunkSize > nSize - (nPos + 8))
     466           0 :                             break;
     467           4 :                         if (memcmp(szBoxName, "META", 4) == 0)
     468             :                         {
     469           1 :                             CPLDebug("WEBP",
     470             :                                      "Remove existing %s box from "
     471             :                                      "source compressed data",
     472             :                                      szBoxName);
     473           1 :                             if (nPos + 8 + nChunkSize < nSize)
     474             :                             {
     475           0 :                                 memmove(pabyData + nPos,
     476           0 :                                         pabyData + nPos + 8 + nChunkSize,
     477           0 :                                         nSize - (nPos + 8 + nChunkSize));
     478             :                             }
     479           1 :                             nSize -= 8 + nChunkSize;
     480             :                         }
     481             :                         else
     482             :                         {
     483           3 :                             nPos += 8 + nChunkSize;
     484             :                         }
     485             :                     }
     486             : 
     487             :                     // Patch size of RIFF
     488           2 :                     uint32_t nSize32 = nSize - 8;
     489           2 :                     CPL_LSBPTR32(&nSize32);
     490           2 :                     memcpy(pabyData + 4, &nSize32, 4);
     491             :                 }
     492             :             }
     493           2 :             if (pnBufferSize)
     494           2 :                 *pnBufferSize = nSize;
     495           2 :             return CE_None;
     496             :         }
     497             :     }
     498           0 :     return CE_Failure;
     499             : }
     500             : 
     501             : /************************************************************************/
     502             : /*                          OpenPAM()                                   */
     503             : /************************************************************************/
     504             : 
     505         467 : GDALPamDataset *WEBPDataset::OpenPAM(GDALOpenInfo *poOpenInfo)
     506             : 
     507             : {
     508         467 :     if (!WEBPDriverIdentify(poOpenInfo) || poOpenInfo->fpL == nullptr)
     509           0 :         return nullptr;
     510             : 
     511             :     int nWidth, nHeight;
     512         467 :     if (!WebPGetInfo(reinterpret_cast<const uint8_t *>(poOpenInfo->pabyHeader),
     513         467 :                      static_cast<uint32_t>(poOpenInfo->nHeaderBytes), &nWidth,
     514             :                      &nHeight))
     515           0 :         return nullptr;
     516             : 
     517         467 :     int nBands = 3;
     518             : 
     519         934 :     auto poDS = std::make_unique<WEBPDataset>();
     520             : 
     521             : #if WEBP_DECODER_ABI_VERSION >= 0x0002
     522             :     WebPDecoderConfig config;
     523         467 :     if (!WebPInitDecoderConfig(&config))
     524           0 :         return nullptr;
     525             : 
     526             :     const bool bOK =
     527         467 :         WebPGetFeatures(poOpenInfo->pabyHeader, poOpenInfo->nHeaderBytes,
     528         467 :                         &config.input) == VP8_STATUS_OK;
     529             : 
     530             :     // Cf commit https://github.com/webmproject/libwebp/commit/86c0031eb2c24f78d4dcfc5dab752ebc9f511607#diff-859d219dccb3163cc11cd538effed461ff0145135070abfe70bd263f16408023
     531             :     // Added in webp 0.4.0
     532             : #if WEBP_DECODER_ABI_VERSION >= 0x0202
     533         934 :     poDS->GDALDataset::SetMetadataItem(
     534             :         "COMPRESSION_REVERSIBILITY",
     535         467 :         config.input.format == 2 ? "LOSSLESS" : "LOSSY", "IMAGE_STRUCTURE");
     536             : #endif
     537             : 
     538         467 :     if (config.input.has_alpha)
     539         296 :         nBands = 4;
     540             : 
     541         467 :     WebPFreeDecBuffer(&config.output);
     542             : 
     543         467 :     if (!bOK)
     544           0 :         return nullptr;
     545             : 
     546             : #endif
     547             : 
     548         467 :     if (poOpenInfo->eAccess == GA_Update)
     549             :     {
     550           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     551             :                  "The WEBP driver does not support update access to existing"
     552             :                  " datasets.\n");
     553           0 :         return nullptr;
     554             :     }
     555             : 
     556             :     /* -------------------------------------------------------------------- */
     557             :     /*      Create a corresponding GDALDataset.                             */
     558             :     /* -------------------------------------------------------------------- */
     559         467 :     poDS->nRasterXSize = nWidth;
     560         467 :     poDS->nRasterYSize = nHeight;
     561         467 :     poDS->fpImage = poOpenInfo->fpL;
     562         467 :     poOpenInfo->fpL = nullptr;
     563             : 
     564             :     /* -------------------------------------------------------------------- */
     565             :     /*      Create band information objects.                                */
     566             :     /* -------------------------------------------------------------------- */
     567        2164 :     for (int iBand = 0; iBand < nBands; iBand++)
     568        1697 :         poDS->SetBand(iBand + 1, new WEBPRasterBand(poDS.get(), iBand + 1));
     569             : 
     570             :     /* -------------------------------------------------------------------- */
     571             :     /*      Initialize any PAM information.                                 */
     572             :     /* -------------------------------------------------------------------- */
     573         467 :     poDS->SetDescription(poOpenInfo->pszFilename);
     574             : 
     575         467 :     poDS->TryLoadXML(poOpenInfo->GetSiblingFiles());
     576             : 
     577             :     /* -------------------------------------------------------------------- */
     578             :     /*      Open overviews.                                                 */
     579             :     /* -------------------------------------------------------------------- */
     580         934 :     poDS->oOvManager.Initialize(poDS.get(), poOpenInfo->pszFilename,
     581         467 :                                 poOpenInfo->GetSiblingFiles());
     582             : 
     583         467 :     return poDS.release();
     584             : }
     585             : 
     586             : /************************************************************************/
     587             : /*                             Open()                                   */
     588             : /************************************************************************/
     589             : 
     590         362 : GDALDataset *WEBPDataset::Open(GDALOpenInfo *poOpenInfo)
     591             : 
     592             : {
     593         362 :     return OpenPAM(poOpenInfo);
     594             : }
     595             : 
     596             : /************************************************************************/
     597             : /*                              WebPUserData                            */
     598             : /************************************************************************/
     599             : 
     600             : typedef struct
     601             : {
     602             :     VSILFILE *fp;
     603             :     GDALProgressFunc pfnProgress;
     604             :     void *pProgressData;
     605             : } WebPUserData;
     606             : 
     607             : /************************************************************************/
     608             : /*                         WEBPDatasetWriter()                          */
     609             : /************************************************************************/
     610             : 
     611         843 : static int WEBPDatasetWriter(const uint8_t *data, size_t data_size,
     612             :                              const WebPPicture *const picture)
     613             : {
     614         843 :     WebPUserData *pUserData =
     615             :         reinterpret_cast<WebPUserData *>(picture->custom_ptr);
     616         843 :     return VSIFWriteL(data, 1, data_size, pUserData->fp) == data_size;
     617             : }
     618             : 
     619             : /************************************************************************/
     620             : /*                        WEBPDatasetProgressHook()                     */
     621             : /************************************************************************/
     622             : 
     623             : #if WEBP_ENCODER_ABI_VERSION >= 0x0100
     624         972 : static int WEBPDatasetProgressHook(int percent,
     625             :                                    const WebPPicture *const picture)
     626             : {
     627         972 :     WebPUserData *pUserData =
     628             :         reinterpret_cast<WebPUserData *>(picture->custom_ptr);
     629         972 :     return pUserData->pfnProgress(percent / 100.0, nullptr,
     630         972 :                                   pUserData->pProgressData);
     631             : }
     632             : #endif
     633             : 
     634             : /************************************************************************/
     635             : /*                              CreateCopy()                            */
     636             : /************************************************************************/
     637             : 
     638         132 : GDALDataset *WEBPDataset::CreateCopy(const char *pszFilename,
     639             :                                      GDALDataset *poSrcDS, int bStrict,
     640             :                                      char **papszOptions,
     641             :                                      GDALProgressFunc pfnProgress,
     642             :                                      void *pProgressData)
     643             : 
     644             : {
     645             :     const char *pszLossLessCopy =
     646         132 :         CSLFetchNameValueDef(papszOptions, "LOSSLESS_COPY", "AUTO");
     647         132 :     if (EQUAL(pszLossLessCopy, "AUTO") || CPLTestBool(pszLossLessCopy))
     648             :     {
     649         132 :         void *pWEBPContent = nullptr;
     650         132 :         size_t nWEBPContent = 0;
     651         132 :         if (poSrcDS->ReadCompressedData(
     652             :                 "WEBP", 0, 0, poSrcDS->GetRasterXSize(),
     653             :                 poSrcDS->GetRasterYSize(), poSrcDS->GetRasterCount(), nullptr,
     654         264 :                 &pWEBPContent, &nWEBPContent, nullptr) == CE_None)
     655             :         {
     656           2 :             CPLDebug("WEBP", "Lossless copy from source dataset");
     657           2 :             std::vector<GByte> abyData;
     658             :             try
     659             :             {
     660           2 :                 abyData.assign(static_cast<const GByte *>(pWEBPContent),
     661           2 :                                static_cast<const GByte *>(pWEBPContent) +
     662           2 :                                    nWEBPContent);
     663             : 
     664           2 :                 char **papszXMP = poSrcDS->GetMetadata("xml:XMP");
     665           2 :                 if (papszXMP && papszXMP[0])
     666             :                 {
     667             :                     GByte abyChunkHeader[8];
     668           1 :                     memcpy(abyChunkHeader, "META", 4);
     669           1 :                     const size_t nXMPSize = strlen(papszXMP[0]);
     670           1 :                     uint32_t nChunkSize = static_cast<uint32_t>(nXMPSize);
     671           1 :                     CPL_LSBPTR32(&nChunkSize);
     672           1 :                     memcpy(abyChunkHeader + 4, &nChunkSize, 4);
     673           0 :                     abyData.insert(abyData.end(), abyChunkHeader,
     674           1 :                                    abyChunkHeader + sizeof(abyChunkHeader));
     675             :                     abyData.insert(
     676           0 :                         abyData.end(), reinterpret_cast<GByte *>(papszXMP[0]),
     677           1 :                         reinterpret_cast<GByte *>(papszXMP[0]) + nXMPSize);
     678           1 :                     if ((abyData.size() % 2) != 0)  // Payload padding if needed
     679           1 :                         abyData.push_back(0);
     680             : 
     681             :                     // Patch size of RIFF
     682             :                     uint32_t nSize32 =
     683           1 :                         static_cast<uint32_t>(abyData.size()) - 8;
     684           1 :                     CPL_LSBPTR32(&nSize32);
     685           1 :                     memcpy(abyData.data() + 4, &nSize32, 4);
     686             :                 }
     687             :             }
     688           0 :             catch (const std::exception &e)
     689             :             {
     690           0 :                 CPLError(CE_Failure, CPLE_AppDefined, "Exception occurred: %s",
     691           0 :                          e.what());
     692           0 :                 abyData.clear();
     693             :             }
     694           2 :             VSIFree(pWEBPContent);
     695             : 
     696           2 :             if (!abyData.empty())
     697             :             {
     698           2 :                 VSILFILE *fpImage = VSIFOpenL(pszFilename, "wb");
     699           2 :                 if (fpImage == nullptr)
     700             :                 {
     701           0 :                     CPLError(CE_Failure, CPLE_OpenFailed,
     702             :                              "Unable to create jpeg file %s.", pszFilename);
     703             : 
     704           0 :                     return nullptr;
     705             :                 }
     706           4 :                 if (VSIFWriteL(abyData.data(), 1, abyData.size(), fpImage) !=
     707           2 :                     abyData.size())
     708             :                 {
     709           0 :                     CPLError(CE_Failure, CPLE_FileIO,
     710           0 :                              "Failure writing data: %s", VSIStrerror(errno));
     711           0 :                     VSIFCloseL(fpImage);
     712           0 :                     return nullptr;
     713             :                 }
     714           2 :                 if (VSIFCloseL(fpImage) != 0)
     715             :                 {
     716           0 :                     CPLError(CE_Failure, CPLE_FileIO,
     717           0 :                              "Failure writing data: %s", VSIStrerror(errno));
     718           0 :                     return nullptr;
     719             :                 }
     720             : 
     721           2 :                 pfnProgress(1.0, nullptr, pProgressData);
     722             : 
     723             :                 // Re-open file and clone missing info to PAM
     724           2 :                 GDALOpenInfo oOpenInfo(pszFilename, GA_ReadOnly);
     725           2 :                 auto poDS = OpenPAM(&oOpenInfo);
     726           2 :                 if (poDS)
     727             :                 {
     728           2 :                     poDS->CloneInfo(poSrcDS, GCIF_PAM_DEFAULT);
     729             :                 }
     730             : 
     731           2 :                 return poDS;
     732             :             }
     733             :         }
     734             :     }
     735             : 
     736         130 :     const bool bLossless = CPLFetchBool(papszOptions, "LOSSLESS", false);
     737         251 :     if (!bLossless &&
     738         121 :         (!EQUAL(pszLossLessCopy, "AUTO") && CPLTestBool(pszLossLessCopy)))
     739             :     {
     740           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     741             :                  "LOSSLESS_COPY=YES requested but not possible");
     742           0 :         return nullptr;
     743             :     }
     744             : 
     745             :     /* -------------------------------------------------------------------- */
     746             :     /*      WEBP library initialization                                     */
     747             :     /* -------------------------------------------------------------------- */
     748             : 
     749             :     WebPPicture sPicture;
     750         130 :     if (!WebPPictureInit(&sPicture))
     751             :     {
     752           0 :         CPLError(CE_Failure, CPLE_AppDefined, "WebPPictureInit() failed");
     753           0 :         return nullptr;
     754             :     }
     755             : 
     756             :     /* -------------------------------------------------------------------- */
     757             :     /*      Some some rudimentary checks                                    */
     758             :     /* -------------------------------------------------------------------- */
     759             : 
     760         130 :     const int nXSize = poSrcDS->GetRasterXSize();
     761         130 :     const int nYSize = poSrcDS->GetRasterYSize();
     762         130 :     if (nXSize > 16383 || nYSize > 16383)
     763             :     {
     764           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     765             :                  "WEBP maximum image dimensions are 16383 x 16383.");
     766             : 
     767           0 :         return nullptr;
     768             :     }
     769             : 
     770         130 :     const int nBands = poSrcDS->GetRasterCount();
     771         130 :     if (nBands != 3
     772             : #if WEBP_ENCODER_ABI_VERSION >= 0x0100
     773          96 :         && nBands != 4
     774             : #endif
     775             :     )
     776             :     {
     777          14 :         CPLError(CE_Failure, CPLE_NotSupported,
     778             :                  "WEBP driver doesn't support %d bands. Must be 3 (RGB) "
     779             : #if WEBP_ENCODER_ABI_VERSION >= 0x0100
     780             :                  "or 4 (RGBA) "
     781             : #endif
     782             :                  "bands.",
     783             :                  nBands);
     784             : 
     785          14 :         return nullptr;
     786             :     }
     787             : 
     788         116 :     const GDALDataType eDT = poSrcDS->GetRasterBand(1)->GetRasterDataType();
     789             : 
     790         116 :     if (eDT != GDT_Byte)
     791             :     {
     792           0 :         CPLError((bStrict) ? CE_Failure : CE_Warning, CPLE_NotSupported,
     793             :                  "WEBP driver doesn't support data type %s. "
     794             :                  "Only eight bit byte bands supported.",
     795             :                  GDALGetDataTypeName(
     796             :                      poSrcDS->GetRasterBand(1)->GetRasterDataType()));
     797             : 
     798           0 :         if (bStrict)
     799           0 :             return nullptr;
     800             :     }
     801             : 
     802             :     /* -------------------------------------------------------------------- */
     803             :     /*      What options has the user selected?                             */
     804             :     /* -------------------------------------------------------------------- */
     805         116 :     float fQuality = 75.0f;
     806         116 :     const char *pszQUALITY = CSLFetchNameValue(papszOptions, "QUALITY");
     807         116 :     if (pszQUALITY != nullptr)
     808             :     {
     809          91 :         fQuality = static_cast<float>(CPLAtof(pszQUALITY));
     810          91 :         if (fQuality < 0.0f || fQuality > 100.0f)
     811             :         {
     812           0 :             CPLError(CE_Failure, CPLE_IllegalArg, "%s=%s is not a legal value.",
     813             :                      "QUALITY", pszQUALITY);
     814           0 :             return nullptr;
     815             :         }
     816             :     }
     817             : 
     818         116 :     WebPPreset nPreset = WEBP_PRESET_DEFAULT;
     819             :     const char *pszPRESET =
     820         116 :         CSLFetchNameValueDef(papszOptions, "PRESET", "DEFAULT");
     821         116 :     if (EQUAL(pszPRESET, "DEFAULT"))
     822         116 :         nPreset = WEBP_PRESET_DEFAULT;
     823           0 :     else if (EQUAL(pszPRESET, "PICTURE"))
     824           0 :         nPreset = WEBP_PRESET_PICTURE;
     825           0 :     else if (EQUAL(pszPRESET, "PHOTO"))
     826           0 :         nPreset = WEBP_PRESET_PHOTO;
     827           0 :     else if (EQUAL(pszPRESET, "PICTURE"))
     828           0 :         nPreset = WEBP_PRESET_PICTURE;
     829           0 :     else if (EQUAL(pszPRESET, "DRAWING"))
     830           0 :         nPreset = WEBP_PRESET_DRAWING;
     831           0 :     else if (EQUAL(pszPRESET, "ICON"))
     832           0 :         nPreset = WEBP_PRESET_ICON;
     833           0 :     else if (EQUAL(pszPRESET, "TEXT"))
     834           0 :         nPreset = WEBP_PRESET_TEXT;
     835             :     else
     836             :     {
     837           0 :         CPLError(CE_Failure, CPLE_IllegalArg, "%s=%s is not a legal value.",
     838             :                  "PRESET", pszPRESET);
     839           0 :         return nullptr;
     840             :     }
     841             : 
     842             :     WebPConfig sConfig;
     843         116 :     if (!WebPConfigInitInternal(&sConfig, nPreset, fQuality,
     844             :                                 WEBP_ENCODER_ABI_VERSION))
     845             :     {
     846           0 :         CPLError(CE_Failure, CPLE_AppDefined, "WebPConfigInit() failed");
     847           0 :         return nullptr;
     848             :     }
     849             : 
     850             :     // TODO: Get rid of this macro in a reasonable way.
     851             : #define FETCH_AND_SET_OPTION_INT(name, fieldname, minval, maxval)              \
     852             :     {                                                                          \
     853             :         const char *pszVal = CSLFetchNameValue(papszOptions, name);            \
     854             :         if (pszVal != nullptr)                                                 \
     855             :         {                                                                      \
     856             :             sConfig.fieldname = atoi(pszVal);                                  \
     857             :             if (sConfig.fieldname < minval || sConfig.fieldname > maxval)      \
     858             :             {                                                                  \
     859             :                 CPLError(CE_Failure, CPLE_IllegalArg,                          \
     860             :                          "%s=%s is not a legal value.", name, pszVal);         \
     861             :                 return nullptr;                                                \
     862             :             }                                                                  \
     863             :         }                                                                      \
     864             :     }
     865             : 
     866         116 :     FETCH_AND_SET_OPTION_INT("TARGETSIZE", target_size, 0, INT_MAX - 1);
     867             : 
     868         116 :     const char *pszPSNR = CSLFetchNameValue(papszOptions, "PSNR");
     869         116 :     if (pszPSNR)
     870             :     {
     871           0 :         sConfig.target_PSNR = static_cast<float>(CPLAtof(pszPSNR));
     872           0 :         if (sConfig.target_PSNR < 0)
     873             :         {
     874           0 :             CPLError(CE_Failure, CPLE_IllegalArg,
     875             :                      "PSNR=%s is not a legal value.", pszPSNR);
     876           0 :             return nullptr;
     877             :         }
     878             :     }
     879             : 
     880         116 :     FETCH_AND_SET_OPTION_INT("METHOD", method, 0, 6);
     881         116 :     FETCH_AND_SET_OPTION_INT("SEGMENTS", segments, 1, 4);
     882         116 :     FETCH_AND_SET_OPTION_INT("SNS_STRENGTH", sns_strength, 0, 100);
     883         116 :     FETCH_AND_SET_OPTION_INT("FILTER_STRENGTH", filter_strength, 0, 100);
     884         116 :     FETCH_AND_SET_OPTION_INT("FILTER_SHARPNESS", filter_sharpness, 0, 7);
     885         116 :     FETCH_AND_SET_OPTION_INT("FILTER_TYPE", filter_type, 0, 1);
     886         116 :     FETCH_AND_SET_OPTION_INT("AUTOFILTER", autofilter, 0, 1);
     887         116 :     FETCH_AND_SET_OPTION_INT("PASS", pass, 1, 10);
     888         116 :     FETCH_AND_SET_OPTION_INT("PREPROCESSING", preprocessing, 0, 1);
     889         116 :     FETCH_AND_SET_OPTION_INT("PARTITIONS", partitions, 0, 3);
     890             : #if WEBP_ENCODER_ABI_VERSION >= 0x0002
     891         116 :     FETCH_AND_SET_OPTION_INT("PARTITION_LIMIT", partition_limit, 0, 100);
     892             : #endif
     893             : #if WEBP_ENCODER_ABI_VERSION >= 0x0100
     894         116 :     sConfig.lossless = bLossless;
     895         116 :     if (sConfig.lossless)
     896           9 :         sPicture.use_argb = 1;
     897             : #endif
     898             : #if WEBP_ENCODER_ABI_VERSION >= 0x0209
     899         116 :     FETCH_AND_SET_OPTION_INT("EXACT", exact, 0, 1);
     900             : #endif
     901             : 
     902         116 :     if (!WebPValidateConfig(&sConfig))
     903             :     {
     904           0 :         CPLError(CE_Failure, CPLE_AppDefined, "WebPValidateConfig() failed");
     905           0 :         return nullptr;
     906             :     }
     907             : 
     908             :     /* -------------------------------------------------------------------- */
     909             :     /*      Allocate memory                                                 */
     910             :     /* -------------------------------------------------------------------- */
     911             :     GByte *pabyBuffer =
     912         116 :         reinterpret_cast<GByte *>(VSI_MALLOC3_VERBOSE(nBands, nXSize, nYSize));
     913         116 :     if (pabyBuffer == nullptr)
     914             :     {
     915           0 :         return nullptr;
     916             :     }
     917             : 
     918             :     /* -------------------------------------------------------------------- */
     919             :     /*      Create the dataset.                                             */
     920             :     /* -------------------------------------------------------------------- */
     921         116 :     VSILFILE *fpImage = VSIFOpenL(pszFilename, "wb");
     922         116 :     if (fpImage == nullptr)
     923             :     {
     924           3 :         CPLError(CE_Failure, CPLE_OpenFailed,
     925             :                  "Unable to create WEBP file %s.\n", pszFilename);
     926           3 :         VSIFree(pabyBuffer);
     927           3 :         return nullptr;
     928             :     }
     929             : 
     930             :     WebPUserData sUserData;
     931         113 :     sUserData.fp = fpImage;
     932         113 :     sUserData.pfnProgress = pfnProgress ? pfnProgress : GDALDummyProgress;
     933         113 :     sUserData.pProgressData = pProgressData;
     934             : 
     935             :     /* -------------------------------------------------------------------- */
     936             :     /*      WEBP library settings                                           */
     937             :     /* -------------------------------------------------------------------- */
     938             : 
     939         113 :     sPicture.width = nXSize;
     940         113 :     sPicture.height = nYSize;
     941         113 :     sPicture.writer = WEBPDatasetWriter;
     942         113 :     sPicture.custom_ptr = &sUserData;
     943             : #if WEBP_ENCODER_ABI_VERSION >= 0x0100
     944         113 :     sPicture.progress_hook = WEBPDatasetProgressHook;
     945             : #endif
     946         113 :     if (!WebPPictureAlloc(&sPicture))
     947             :     {
     948           0 :         CPLError(CE_Failure, CPLE_AppDefined, "WebPPictureAlloc() failed");
     949           0 :         VSIFree(pabyBuffer);
     950           0 :         VSIFCloseL(fpImage);
     951           0 :         return nullptr;
     952             :     }
     953             : 
     954             :     /* -------------------------------------------------------------------- */
     955             :     /*      Acquire source imagery.                                         */
     956             :     /* -------------------------------------------------------------------- */
     957             :     CPLErr eErr =
     958         226 :         poSrcDS->RasterIO(GF_Read, 0, 0, nXSize, nYSize, pabyBuffer, nXSize,
     959             :                           nYSize, GDT_Byte, nBands, nullptr, nBands,
     960         113 :                           static_cast<GSpacing>(nBands) * nXSize, 1, nullptr);
     961             : 
     962             : /* -------------------------------------------------------------------- */
     963             : /*      Import and write to file                                        */
     964             : /* -------------------------------------------------------------------- */
     965             : #if WEBP_ENCODER_ABI_VERSION >= 0x0100
     966         113 :     if (eErr == CE_None && nBands == 4)
     967             :     {
     968          82 :         if (!WebPPictureImportRGBA(&sPicture, pabyBuffer, nBands * nXSize))
     969             :         {
     970           0 :             CPLError(CE_Failure, CPLE_AppDefined,
     971             :                      "WebPPictureImportRGBA() failed");
     972           0 :             eErr = CE_Failure;
     973             :         }
     974             :     }
     975             :     else
     976             : #endif
     977          62 :         if (eErr == CE_None &&
     978          31 :             !WebPPictureImportRGB(&sPicture, pabyBuffer, nBands * nXSize))
     979             :     {
     980           0 :         CPLError(CE_Failure, CPLE_AppDefined, "WebPPictureImportRGB() failed");
     981           0 :         eErr = CE_Failure;
     982             :     }
     983             : 
     984         113 :     if (eErr == CE_None && !WebPEncode(&sConfig, &sPicture))
     985             :     {
     986             : #if WEBP_ENCODER_ABI_VERSION >= 0x0100
     987          10 :         const char *pszErrorMsg = nullptr;
     988          10 :         switch (sPicture.error_code)
     989             :         {
     990           0 :             case VP8_ENC_ERROR_OUT_OF_MEMORY:
     991           0 :                 pszErrorMsg = "Out of memory";
     992           0 :                 break;
     993           0 :             case VP8_ENC_ERROR_BITSTREAM_OUT_OF_MEMORY:
     994           0 :                 pszErrorMsg = "Out of memory while flushing bits";
     995           0 :                 break;
     996           0 :             case VP8_ENC_ERROR_NULL_PARAMETER:
     997           0 :                 pszErrorMsg = "A pointer parameter is NULL";
     998           0 :                 break;
     999           0 :             case VP8_ENC_ERROR_INVALID_CONFIGURATION:
    1000           0 :                 pszErrorMsg = "Configuration is invalid";
    1001           0 :                 break;
    1002           0 :             case VP8_ENC_ERROR_BAD_DIMENSION:
    1003           0 :                 pszErrorMsg = "Picture has invalid width/height";
    1004           0 :                 break;
    1005           0 :             case VP8_ENC_ERROR_PARTITION0_OVERFLOW:
    1006           0 :                 pszErrorMsg = "Partition is bigger than 512k. Try using less "
    1007             :                               "SEGMENTS, or increase PARTITION_LIMIT value";
    1008           0 :                 break;
    1009           0 :             case VP8_ENC_ERROR_PARTITION_OVERFLOW:
    1010           0 :                 pszErrorMsg = "Partition is bigger than 16M";
    1011           0 :                 break;
    1012           5 :             case VP8_ENC_ERROR_BAD_WRITE:
    1013           5 :                 pszErrorMsg = "Error while flushing bytes";
    1014           5 :                 break;
    1015           0 :             case VP8_ENC_ERROR_FILE_TOO_BIG:
    1016           0 :                 pszErrorMsg = "File is bigger than 4G";
    1017           0 :                 break;
    1018           0 :             case VP8_ENC_ERROR_USER_ABORT:
    1019           0 :                 pszErrorMsg = "User interrupted";
    1020           0 :                 break;
    1021           5 :             default:
    1022           5 :                 CPLError(CE_Failure, CPLE_AppDefined,
    1023             :                          "WebPEncode returned an unknown error code: %d",
    1024           5 :                          sPicture.error_code);
    1025           5 :                 pszErrorMsg = "Unknown WebP error type.";
    1026           5 :                 break;
    1027             :         }
    1028          10 :         CPLError(CE_Failure, CPLE_AppDefined, "WebPEncode() failed : %s",
    1029             :                  pszErrorMsg);
    1030             : #else
    1031             :         CPLError(CE_Failure, CPLE_AppDefined, "WebPEncode() failed");
    1032             : #endif
    1033          10 :         eErr = CE_Failure;
    1034             :     }
    1035             : 
    1036             :     /* -------------------------------------------------------------------- */
    1037             :     /*      Cleanup and close.                                              */
    1038             :     /* -------------------------------------------------------------------- */
    1039         113 :     CPLFree(pabyBuffer);
    1040             : 
    1041         113 :     WebPPictureFree(&sPicture);
    1042             : 
    1043         113 :     VSIFCloseL(fpImage);
    1044             : 
    1045         113 :     if (pfnProgress)
    1046         113 :         pfnProgress(1.0, "", pProgressData);
    1047             : 
    1048         113 :     if (eErr != CE_None)
    1049             :     {
    1050          10 :         VSIUnlink(pszFilename);
    1051          10 :         return nullptr;
    1052             :     }
    1053             : 
    1054             :     /* -------------------------------------------------------------------- */
    1055             :     /*      Re-open dataset, and copy any auxiliary pam information.        */
    1056             :     /* -------------------------------------------------------------------- */
    1057         206 :     GDALOpenInfo oOpenInfo(pszFilename, GA_ReadOnly);
    1058             : 
    1059             :     /* If writing to stdout, we can't reopen it, so return */
    1060             :     /* a fake dataset to make the caller happy */
    1061         103 :     CPLPushErrorHandler(CPLQuietErrorHandler);
    1062         103 :     auto poDS = WEBPDataset::OpenPAM(&oOpenInfo);
    1063         103 :     CPLPopErrorHandler();
    1064         103 :     if (poDS)
    1065             :     {
    1066         103 :         poDS->CloneInfo(poSrcDS, GCIF_PAM_DEFAULT);
    1067         103 :         return poDS;
    1068             :     }
    1069             : 
    1070           0 :     return nullptr;
    1071             : }
    1072             : 
    1073             : /************************************************************************/
    1074             : /*                         GDALRegister_WEBP()                          */
    1075             : /************************************************************************/
    1076             : 
    1077          11 : void GDALRegister_WEBP()
    1078             : 
    1079             : {
    1080          11 :     if (GDALGetDriverByName(DRIVER_NAME) != nullptr)
    1081           0 :         return;
    1082             : 
    1083          11 :     GDALDriver *poDriver = new GDALDriver();
    1084          11 :     WEBPDriverSetCommonMetadata(poDriver);
    1085             : 
    1086          11 :     poDriver->pfnOpen = WEBPDataset::Open;
    1087          11 :     poDriver->pfnCreateCopy = WEBPDataset::CreateCopy;
    1088             : 
    1089          11 :     GetGDALDriverManager()->RegisterDriver(poDriver);
    1090             : }

Generated by: LCOV version 1.14