LCOV - code coverage report
Current view: top level - frmts/gtiff - tif_jxl.c (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 487 638 76.3 %
Date: 2025-01-18 12:42:00 Functions: 15 15 100.0 %

          Line data    Source code
       1             : /*
       2             :  * Copyright (c) 2021, Airbus DS Intelligence
       3             :  * Author: <even.rouault at spatialys.com>
       4             :  *
       5             :  * Permission to use, copy, modify, distribute, and sell this software and
       6             :  * its documentation for any purpose is hereby granted without fee, provided
       7             :  * that (i) the above copyright notices and this permission notice appear in
       8             :  * all copies of the software and related documentation, and (ii) the names of
       9             :  * Sam Leffler and Silicon Graphics may not be used in any advertising or
      10             :  * publicity relating to the software without the specific, prior written
      11             :  * permission of Sam Leffler and Silicon Graphics.
      12             :  *
      13             :  * THE SOFTWARE IS PROVIDED "AS-IS" AND WITHOUT WARRANTY OF ANY KIND,
      14             :  * EXPRESS, IMPLIED OR OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY
      15             :  * WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.
      16             :  *
      17             :  * IN NO EVENT SHALL SAM LEFFLER OR SILICON GRAPHICS BE LIABLE FOR
      18             :  * ANY SPECIAL, INCIDENTAL, INDIRECT OR CONSEQUENTIAL DAMAGES OF ANY KIND,
      19             :  * OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
      20             :  * WHETHER OR NOT ADVISED OF THE POSSIBILITY OF DAMAGE, AND ON ANY THEORY OF
      21             :  * LIABILITY, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
      22             :  * OF THIS SOFTWARE.
      23             :  */
      24             : 
      25             : #include "tiffiop.h"
      26             : #include "tif_jxl.h"
      27             : 
      28             : #include <jxl/decode.h>
      29             : #include <jxl/encode.h>
      30             : 
      31             : #include <stdint.h>
      32             : 
      33             : #include <assert.h>
      34             : 
      35             : #define LSTATE_INIT_DECODE 0x01
      36             : #define LSTATE_INIT_ENCODE 0x02
      37             : 
      38             : /*
      39             :  * State block for each open TIFF file using JXL compression/decompression.
      40             :  */
      41             : typedef struct
      42             : {
      43             :     int state; /* state flags */
      44             : 
      45             :     int lossless;         /* TRUE (default) or FALSE */
      46             :     int effort;           /* 3 to 9. default: 7 */
      47             :     float distance;       /* 0 to 15. default: 1.0 */
      48             :     float alpha_distance; /* 0 to 15. default: -1.0 (same as distance) */
      49             : 
      50             :     uint32_t segment_width;
      51             :     uint32_t segment_height;
      52             : 
      53             :     unsigned int uncompressed_size;
      54             :     unsigned int uncompressed_alloc;
      55             :     uint8_t *uncompressed_buffer;
      56             :     unsigned int uncompressed_offset;
      57             : 
      58             :     JxlDecoder *decoder;
      59             : 
      60             :     TIFFVGetMethod vgetparent; /* super-class method */
      61             :     TIFFVSetMethod vsetparent; /* super-class method */
      62             : } JXLState;
      63             : 
      64             : #define LState(tif) ((JXLState *)(tif)->tif_data)
      65             : #define DecoderState(tif) LState(tif)
      66             : #define EncoderState(tif) LState(tif)
      67             : 
      68             : static int JXLEncode(TIFF *tif, uint8_t *bp, tmsize_t cc, uint16_t s);
      69             : static int JXLDecode(TIFF *tif, uint8_t *op, tmsize_t occ, uint16_t s);
      70             : 
      71         927 : static int GetJXLDataType(TIFF *tif)
      72             : {
      73         927 :     TIFFDirectory *td = &tif->tif_dir;
      74             :     static const char module[] = "GetJXLDataType";
      75             : 
      76         927 :     if (td->td_sampleformat == SAMPLEFORMAT_UINT && td->td_bitspersample == 8)
      77             :     {
      78         523 :         return JXL_TYPE_UINT8;
      79             :     }
      80             : 
      81         404 :     if (td->td_sampleformat == SAMPLEFORMAT_UINT && td->td_bitspersample == 16)
      82             :     {
      83         203 :         return JXL_TYPE_UINT16;
      84             :     }
      85             : 
      86             :     /* 20210903: Not supported yet by libjxl*/
      87             :     /*
      88             :     if( td->td_sampleformat == SAMPLEFORMAT_INT &&
      89             :             td->td_bitspersample == 32 )
      90             :     {
      91             :         return JXL_TYPE_UINT32;
      92             :     }
      93             :     */
      94             : 
      95         201 :     if (td->td_sampleformat == SAMPLEFORMAT_IEEEFP &&
      96         201 :         td->td_bitspersample == 32)
      97             :     {
      98         196 :         return JXL_TYPE_FLOAT;
      99             :     }
     100             : 
     101           5 :     if (td->td_sampleformat == SAMPLEFORMAT_IEEEFP &&
     102           5 :         td->td_bitspersample == 16)
     103             :     {
     104           5 :         return JXL_TYPE_FLOAT16;
     105             :     }
     106             : 
     107           0 :     TIFFErrorExtR(
     108             :         tif, module,
     109             :         "Unsupported combination of SampleFormat(=%d) and BitsPerSample(=%d)",
     110           0 :         td->td_sampleformat, td->td_bitspersample);
     111           0 :     return -1;
     112             : }
     113             : 
     114         816 : static int GetJXLDataTypeSize(JxlDataType dtype)
     115             : {
     116         816 :     switch (dtype)
     117             :     {
     118         454 :         case JXL_TYPE_UINT8:
     119         454 :             return 1;
     120         182 :         case JXL_TYPE_UINT16:
     121         182 :             return 2;
     122         176 :         case JXL_TYPE_FLOAT:
     123         176 :             return 4;
     124           4 :         case JXL_TYPE_FLOAT16:
     125           4 :             return 2;
     126           0 :         default:
     127           0 :             return 0;
     128             :     }
     129             : }
     130             : 
     131         385 : static int JXLFixupTags(TIFF *tif)
     132             : {
     133             :     (void)tif;
     134         385 :     return 1;
     135             : }
     136             : 
     137         117 : static int JXLSetupDecode(TIFF *tif)
     138             : {
     139         117 :     JXLState *sp = DecoderState(tif);
     140             : 
     141         117 :     assert(sp != NULL);
     142             : 
     143             :     /* if we were last encoding, terminate this mode */
     144         117 :     if (sp->state & LSTATE_INIT_ENCODE)
     145             :     {
     146           2 :         sp->state = 0;
     147             :     }
     148             : 
     149         117 :     sp->state |= LSTATE_INIT_DECODE;
     150         117 :     return 1;
     151             : }
     152             : 
     153         408 : static int SetupUncompressedBuffer(TIFF *tif, JXLState *sp, const char *module)
     154             : {
     155         408 :     TIFFDirectory *td = &tif->tif_dir;
     156             :     uint64_t new_size_64;
     157             :     uint64_t new_alloc_64;
     158             :     unsigned int new_size;
     159             :     unsigned int new_alloc;
     160             : 
     161         408 :     sp->uncompressed_offset = 0;
     162             : 
     163         408 :     if (isTiled(tif))
     164             :     {
     165         288 :         sp->segment_width = td->td_tilewidth;
     166         288 :         sp->segment_height = td->td_tilelength;
     167             :     }
     168             :     else
     169             :     {
     170         120 :         sp->segment_width = td->td_imagewidth;
     171         120 :         sp->segment_height = td->td_imagelength - tif->tif_row;
     172         120 :         if (sp->segment_height > td->td_rowsperstrip)
     173          29 :             sp->segment_height = td->td_rowsperstrip;
     174             :     }
     175             : 
     176         408 :     JxlDataType dtype = GetJXLDataType(tif);
     177             :     if (dtype < 0)
     178             :     {
     179             :         _TIFFfreeExt(tif, sp->uncompressed_buffer);
     180             :         sp->uncompressed_buffer = 0;
     181             :         sp->uncompressed_alloc = 0;
     182             :         return 0;
     183             :     }
     184         408 :     int nBytesPerSample = GetJXLDataTypeSize(dtype);
     185         408 :     new_size_64 =
     186         408 :         (uint64_t)sp->segment_width * sp->segment_height * nBytesPerSample;
     187         408 :     if (td->td_planarconfig == PLANARCONFIG_CONTIG)
     188             :     {
     189         204 :         new_size_64 *= td->td_samplesperpixel;
     190             :     }
     191             : 
     192         408 :     new_size = (unsigned int)new_size_64;
     193         408 :     sp->uncompressed_size = new_size;
     194             : 
     195             :     /* add some margin */
     196         408 :     new_alloc_64 = 100 + new_size_64 + new_size_64 / 3;
     197         408 :     new_alloc = (unsigned int)new_alloc_64;
     198         408 :     if (new_alloc != new_alloc_64)
     199             :     {
     200           0 :         TIFFErrorExtR(tif, module, "Too large uncompressed strip/tile");
     201           0 :         _TIFFfreeExt(tif, sp->uncompressed_buffer);
     202           0 :         sp->uncompressed_buffer = 0;
     203           0 :         sp->uncompressed_alloc = 0;
     204           0 :         return 0;
     205             :     }
     206             : 
     207         408 :     if (sp->uncompressed_alloc < new_alloc)
     208             :     {
     209         226 :         _TIFFfreeExt(tif, sp->uncompressed_buffer);
     210         226 :         sp->uncompressed_buffer = _TIFFmallocExt(tif, new_alloc);
     211         226 :         if (!sp->uncompressed_buffer)
     212             :         {
     213           0 :             TIFFErrorExtR(tif, module, "Cannot allocate buffer");
     214           0 :             _TIFFfreeExt(tif, sp->uncompressed_buffer);
     215           0 :             sp->uncompressed_buffer = 0;
     216           0 :             sp->uncompressed_alloc = 0;
     217           0 :             return 0;
     218             :         }
     219         226 :         sp->uncompressed_alloc = new_alloc;
     220             :     }
     221             : 
     222         408 :     return 1;
     223             : }
     224             : 
     225             : /*
     226             :  * Setup state for decoding a strip.
     227             :  */
     228         206 : static int JXLPreDecode(TIFF *tif, uint16_t s)
     229             : {
     230             :     static const char module[] = "JXLPreDecode";
     231         206 :     JXLState *sp = DecoderState(tif);
     232         206 :     TIFFDirectory *td = &tif->tif_dir;
     233             : 
     234             :     (void)s;
     235         206 :     assert(sp != NULL);
     236         206 :     if (sp->state != LSTATE_INIT_DECODE)
     237           2 :         tif->tif_setupdecode(tif);
     238             : 
     239         206 :     const int jxlDataType = GetJXLDataType(tif);
     240         206 :     if (jxlDataType < 0)
     241           0 :         return 0;
     242             : 
     243         206 :     if (!SetupUncompressedBuffer(tif, sp, module))
     244           0 :         return 0;
     245             : 
     246         206 :     if (sp->decoder == NULL)
     247             :     {
     248         117 :         sp->decoder = JxlDecoderCreate(NULL);
     249         117 :         if (sp->decoder == NULL)
     250             :         {
     251           0 :             TIFFErrorExtR(tif, module, "JxlDecoderCreate() failed");
     252           0 :             return 0;
     253             :         }
     254             :     }
     255             :     else
     256             :     {
     257          89 :         JxlDecoderReset(sp->decoder);
     258             :     }
     259             : 
     260             :     JxlDecoderStatus status;
     261         206 :     status = JxlDecoderSubscribeEvents(sp->decoder,
     262             :                                        JXL_DEC_BASIC_INFO | JXL_DEC_FULL_IMAGE);
     263         206 :     if (status != JXL_DEC_SUCCESS)
     264             :     {
     265           0 :         TIFFErrorExtR(tif, module, "JxlDecoderSubscribeEvents() failed");
     266           0 :         return 0;
     267             :     }
     268             : 
     269         206 :     status = JxlDecoderSetInput(sp->decoder, (const uint8_t *)tif->tif_rawcp,
     270         206 :                                 (size_t)tif->tif_rawcc);
     271         206 :     if (status != JXL_DEC_SUCCESS)
     272             :     {
     273           0 :         TIFFErrorExtR(tif, module, "JxlDecoderSetInput() failed with %d",
     274             :                       status);
     275           0 :         return 0;
     276             :     }
     277             : 
     278         206 :     status = JxlDecoderProcessInput(sp->decoder);
     279         206 :     if (status != JXL_DEC_BASIC_INFO)
     280             :     {
     281           0 :         TIFFErrorExtR(tif, module, "JxlDecoderProcessInput() failed with %d",
     282             :                       status);
     283           0 :         JxlDecoderReleaseInput(sp->decoder);
     284           0 :         return 0;
     285             :     }
     286             : 
     287             :     JxlBasicInfo info;
     288         206 :     status = JxlDecoderGetBasicInfo(sp->decoder, &info);
     289         206 :     if (status != JXL_DEC_SUCCESS)
     290             :     {
     291           0 :         TIFFErrorExtR(tif, module, "JxlDecoderGetBasicInfo() failed with %d",
     292             :                       status);
     293           0 :         JxlDecoderReleaseInput(sp->decoder);
     294           0 :         return 0;
     295             :     }
     296             : 
     297         206 :     if (sp->segment_width != info.xsize)
     298             :     {
     299           0 :         TIFFErrorExtR(tif, module,
     300             :                       "JXL basic info xsize = %d, whereas %u was expected",
     301             :                       info.xsize, sp->segment_width);
     302           0 :         JxlDecoderReleaseInput(sp->decoder);
     303           0 :         return 0;
     304             :     }
     305             : 
     306         206 :     if (sp->segment_height != info.ysize)
     307             :     {
     308           0 :         TIFFErrorExtR(tif, module,
     309             :                       "JXL basic info ysize = %d, whereas %u was expected",
     310             :                       info.ysize, sp->segment_height);
     311           0 :         JxlDecoderReleaseInput(sp->decoder);
     312           0 :         return 0;
     313             :     }
     314             : 
     315         206 :     if (td->td_bitspersample != info.bits_per_sample)
     316             :     {
     317           0 :         TIFFErrorExtR(
     318             :             tif, module,
     319             :             "JXL basic info bits_per_sample = %d, whereas %d was expected",
     320           0 :             info.bits_per_sample, td->td_bitspersample);
     321           0 :         JxlDecoderReleaseInput(sp->decoder);
     322           0 :         return 0;
     323             :     }
     324             : 
     325         206 :     if (td->td_planarconfig == PLANARCONFIG_CONTIG)
     326             :     {
     327         104 :         if (info.num_color_channels + info.num_extra_channels !=
     328         104 :             td->td_samplesperpixel)
     329             :         {
     330           0 :             TIFFErrorExtR(tif, module,
     331             :                           "JXL basic info invalid number of channels");
     332           0 :             JxlDecoderReleaseInput(sp->decoder);
     333           0 :             return 0;
     334             :         }
     335             :     }
     336             :     else
     337             :     {
     338         102 :         if (info.num_color_channels != 1 || info.alpha_bits > 0 ||
     339         102 :             info.num_extra_channels > 0)
     340             :         {
     341           0 :             TIFFErrorExtR(tif, module,
     342             :                           "JXL basic info invalid number of channels");
     343           0 :             JxlDecoderReleaseInput(sp->decoder);
     344           0 :             return 0;
     345             :         }
     346             :     }
     347             : 
     348         206 :     JxlPixelFormat format = {0};
     349         206 :     format.data_type = jxlDataType;
     350         206 :     format.endianness = JXL_NATIVE_ENDIAN;
     351         206 :     format.align = 0;
     352             :     // alpha_bits is set even for a gray, gray, Alpha, gray, gray
     353             :     // or for R, G, B, undefined, Alpha
     354             :     // Probably a defect of libjxl: https://github.com/libjxl/libjxl/issues/1773
     355             :     // So for num_color_channels==3, num_color_channels > 1 and
     356             :     // alpha_bits != 0, get information of the first extra channel to
     357             :     // check if it is alpha, to detect R, G, B, Alpha, undefined.
     358             :     // Note: there's no difference in the codestream if writing RGBAU
     359             :     // as num_channels == 3 with 2 extra channels the first one being
     360             :     // explicitly set to alpha, or with num_channels == 4.
     361         206 :     int bAlphaEmbedded = 0;
     362         206 :     if (info.alpha_bits != 0)
     363             :     {
     364          37 :         if ((info.num_color_channels == 3 || info.num_color_channels == 1) &&
     365          37 :             (info.num_extra_channels == 1))
     366             :         {
     367          28 :             bAlphaEmbedded = 1;
     368             :         }
     369           9 :         else if (info.num_color_channels == 3 && info.num_extra_channels > 1)
     370             :         {
     371             :             JxlExtraChannelInfo extra_channel_info;
     372           6 :             memset(&extra_channel_info, 0, sizeof(extra_channel_info));
     373           6 :             if (JxlDecoderGetExtraChannelInfo(
     374           6 :                     sp->decoder, 0, &extra_channel_info) == JXL_DEC_SUCCESS &&
     375           6 :                 extra_channel_info.type == JXL_CHANNEL_ALPHA)
     376             :             {
     377           3 :                 bAlphaEmbedded = 1;
     378             :             }
     379             :         }
     380             :     }
     381         206 :     uint32_t nFirstExtraChannel = (bAlphaEmbedded) ? 1 : 0;
     382         206 :     unsigned int main_buffer_size = sp->uncompressed_size;
     383         206 :     unsigned int channel_size = main_buffer_size / td->td_samplesperpixel;
     384         206 :     uint8_t *extra_channel_buffer = NULL;
     385             : 
     386         206 :     int nBytesPerSample = GetJXLDataTypeSize(format.data_type);
     387             : 
     388         206 :     if (nFirstExtraChannel < info.num_extra_channels)
     389             :     {
     390          17 :         int nExtraChannelsToExtract =
     391          17 :             info.num_extra_channels - nFirstExtraChannel;
     392          17 :         format.num_channels = 1;
     393          17 :         main_buffer_size =
     394          17 :             channel_size * (info.num_color_channels + (bAlphaEmbedded ? 1 : 0));
     395             :         extra_channel_buffer =
     396          17 :             _TIFFmallocExt(tif, channel_size * nExtraChannelsToExtract);
     397          17 :         if (extra_channel_buffer == NULL)
     398           0 :             return 0;
     399          55 :         for (int i = 0; i < nExtraChannelsToExtract; ++i)
     400             :         {
     401             :             size_t buffer_size;
     402          38 :             const int iCorrectedIdx = i + nFirstExtraChannel;
     403             : 
     404          38 :             if (JxlDecoderExtraChannelBufferSize(sp->decoder, &format,
     405             :                                                  &buffer_size, iCorrectedIdx) !=
     406             :                 JXL_DEC_SUCCESS)
     407             :             {
     408           0 :                 TIFFErrorExtR(tif, module,
     409             :                               "JxlDecoderExtraChannelBufferSize failed()");
     410           0 :                 _TIFFfreeExt(tif, extra_channel_buffer);
     411           0 :                 return 0;
     412             :             }
     413          38 :             if (buffer_size != channel_size)
     414             :             {
     415           0 :                 TIFFErrorExtR(tif, module,
     416             :                               "JxlDecoderExtraChannelBufferSize returned "
     417             :                               "%" TIFF_SIZE_FORMAT ", expecting %u",
     418             :                               buffer_size, channel_size);
     419           0 :                 _TIFFfreeExt(tif, extra_channel_buffer);
     420           0 :                 return 0;
     421             :             }
     422             : 
     423             : #if 0
     424             :                 // Check consistency of JXL codestream header regarding
     425             :                 // extra alpha channels and TIFF ExtraSamples tag
     426             :                 JxlExtraChannelInfo extra_channel_info;
     427             :                 memset(&extra_channel_info, 0, sizeof(extra_channel_info));
     428             :                 if( JxlDecoderGetExtraChannelInfo(sp->decoder, iCorrectedIdx, &extra_channel_info) == JXL_DEC_SUCCESS )
     429             :                 {
     430             :                     if( extra_channel_info.type == JXL_CHANNEL_ALPHA &&
     431             :                         !extra_channel_info.alpha_premultiplied )
     432             :                     {
     433             :                         if( iCorrectedIdx < td->td_extrasamples &&
     434             :                             td->td_sampleinfo[iCorrectedIdx] == EXTRASAMPLE_UNASSALPHA )
     435             :                         {
     436             :                             // ok
     437             :                         }
     438             :                         else
     439             :                         {
     440             :                             TIFFWarningExtR(tif, module,
     441             :                                            "Unpremultiplied alpha channel expected from JXL codestream "
     442             :                                            "in extra channel %d, but other value found in ExtraSamples tag", iCorrectedIdx);
     443             :                         }
     444             :                     }
     445             :                     else if( extra_channel_info.type == JXL_CHANNEL_ALPHA &&
     446             :                              extra_channel_info.alpha_premultiplied )
     447             :                     {
     448             :                         if( iCorrectedIdx < td->td_extrasamples &&
     449             :                             td->td_sampleinfo[iCorrectedIdx] == EXTRASAMPLE_ASSOCALPHA )
     450             :                         {
     451             :                             // ok
     452             :                         }
     453             :                         else
     454             :                         {
     455             :                             TIFFWarningExtR(tif, module,
     456             :                                            "Premultiplied alpha channel expected from JXL codestream "
     457             :                                            "in extra channel %d, but other value found in ExtraSamples tag", iCorrectedIdx);
     458             :                         }
     459             :                     }
     460             :                     else if( iCorrectedIdx < td->td_extrasamples &&
     461             :                              td->td_sampleinfo[iCorrectedIdx] == EXTRASAMPLE_UNASSALPHA )
     462             :                     {
     463             :                         TIFFWarningExtR(tif, module,
     464             :                                        "Unpremultiplied alpha channel expected from ExtraSamples tag "
     465             :                                        "in extra channel %d, but other value found in JXL codestream", iCorrectedIdx);
     466             :                     }
     467             :                     else if( iCorrectedIdx < td->td_extrasamples &&
     468             :                              td->td_sampleinfo[iCorrectedIdx] == EXTRASAMPLE_ASSOCALPHA )
     469             :                     {
     470             :                         TIFFWarningExtR(tif, module,
     471             :                                        "Premultiplied alpha channel expected from ExtraSamples tag "
     472             :                                        "in extra channel %d, but other value found in JXL codestream", iCorrectedIdx);
     473             :                     }
     474             :                 }
     475             : #endif
     476          38 :             if (JxlDecoderSetExtraChannelBuffer(
     477             :                     sp->decoder, &format,
     478          38 :                     extra_channel_buffer + i * channel_size, channel_size,
     479             :                     i + nFirstExtraChannel) != JXL_DEC_SUCCESS)
     480             :             {
     481           0 :                 TIFFErrorExtR(tif, module,
     482             :                               "JxlDecoderSetExtraChannelBuffer failed()");
     483           0 :                 _TIFFfreeExt(tif, extra_channel_buffer);
     484           0 :                 return 0;
     485             :             }
     486             :         }
     487             :     }
     488             : 
     489         206 :     format.num_channels = info.num_color_channels;
     490         206 :     if (bAlphaEmbedded)
     491          31 :         format.num_channels++;
     492             : 
     493         206 :     status = JxlDecoderProcessInput(sp->decoder);
     494         206 :     if (status != JXL_DEC_NEED_IMAGE_OUT_BUFFER)
     495             :     {
     496           0 :         TIFFErrorExtR(tif, module,
     497             :                       "JxlDecoderProcessInput() (second call) failed with %d",
     498             :                       status);
     499           0 :         JxlDecoderReleaseInput(sp->decoder);
     500           0 :         _TIFFfreeExt(tif, extra_channel_buffer);
     501           0 :         return 0;
     502             :     }
     503             : 
     504         206 :     status = JxlDecoderSetImageOutBuffer(
     505         206 :         sp->decoder, &format, sp->uncompressed_buffer, main_buffer_size);
     506         206 :     if (status != JXL_DEC_SUCCESS)
     507             :     {
     508           0 :         TIFFErrorExtR(tif, module,
     509             :                       "JxlDecoderSetImageOutBuffer() failed with %d", status);
     510           0 :         JxlDecoderReleaseInput(sp->decoder);
     511           0 :         _TIFFfreeExt(tif, extra_channel_buffer);
     512           0 :         return 0;
     513             :     }
     514             : 
     515         206 :     status = JxlDecoderProcessInput(sp->decoder);
     516         206 :     if (status != JXL_DEC_FULL_IMAGE)
     517             :     {
     518           0 :         TIFFErrorExtR(tif, module,
     519             :                       "JxlDecoderProcessInput() (third call) failed with %d",
     520             :                       status);
     521           0 :         JxlDecoderReleaseInput(sp->decoder);
     522           0 :         _TIFFfreeExt(tif, extra_channel_buffer);
     523           0 :         return 0;
     524             :     }
     525         206 :     if (nFirstExtraChannel < info.num_extra_channels)
     526             :     {
     527             :         // first reorder the main buffer
     528          20 :         const int nMainChannels = bAlphaEmbedded ? info.num_color_channels + 1
     529          17 :                                                  : info.num_color_channels;
     530          17 :         const unsigned int mainPixSize = nMainChannels * nBytesPerSample;
     531          17 :         const unsigned int fullPixSize =
     532          17 :             td->td_samplesperpixel * nBytesPerSample;
     533          17 :         assert(fullPixSize > mainPixSize);
     534             : 
     535             :         /* Find min value of k such that k * fullPixSize >= (k + 1) * mainPixSize:
     536             :          * ==> k = ceil(mainPixSize / (fullPixSize - mainPixSize))
     537             :          * ==> k = (mainPixSize + (fullPixSize - mainPixSize) - 1) / (fullPixSize - mainPixSize)
     538             :          * ==> k = (fullPixSize - 1) / (fullPixSize - mainPixSize)
     539             :          */
     540          17 :         const unsigned int nNumPixels = info.xsize * info.ysize;
     541          17 :         unsigned int outOff = sp->uncompressed_size - fullPixSize;
     542          17 :         unsigned int inOff = main_buffer_size - mainPixSize;
     543          17 :         const unsigned int kThreshold =
     544          17 :             (fullPixSize - 1) / (fullPixSize - mainPixSize);
     545          17 :         if (mainPixSize == 1)
     546             :         {
     547      131872 :             for (unsigned int k = kThreshold; k < nNumPixels; ++k)
     548             :             {
     549      131868 :                 memcpy(sp->uncompressed_buffer + outOff,
     550      131868 :                        sp->uncompressed_buffer + inOff, /*mainPixSize=*/1);
     551      131868 :                 inOff -= /*mainPixSize=*/1;
     552      131868 :                 outOff -= fullPixSize;
     553             :             }
     554             :         }
     555          13 :         else if (mainPixSize == 2)
     556             :         {
     557      131072 :             for (unsigned int k = kThreshold; k < nNumPixels; ++k)
     558             :             {
     559      131070 :                 memcpy(sp->uncompressed_buffer + outOff,
     560      131070 :                        sp->uncompressed_buffer + inOff, /*mainPixSize=*/2);
     561      131070 :                 inOff -= /*mainPixSize=*/2;
     562      131070 :                 outOff -= fullPixSize;
     563             :             }
     564             :         }
     565          11 :         else if (mainPixSize == 3)
     566             :         {
     567      131070 :             for (unsigned int k = kThreshold; k < nNumPixels; ++k)
     568             :             {
     569      131068 :                 memcpy(sp->uncompressed_buffer + outOff,
     570      131068 :                        sp->uncompressed_buffer + inOff, /*mainPixSize=*/3);
     571      131068 :                 inOff -= /*mainPixSize=*/3;
     572      131068 :                 outOff -= fullPixSize;
     573             :             }
     574             :         }
     575           9 :         else if (mainPixSize == 4)
     576             :         {
     577      196605 :             for (unsigned int k = kThreshold; k < nNumPixels; ++k)
     578             :             {
     579      196602 :                 memcpy(sp->uncompressed_buffer + outOff,
     580      196602 :                        sp->uncompressed_buffer + inOff, /*mainPixSize=*/4);
     581      196602 :                 inOff -= /*mainPixSize=*/4;
     582      196602 :                 outOff -= fullPixSize;
     583             :             }
     584             :         }
     585           6 :         else if (mainPixSize == 3 * 2)
     586             :         {
     587      131070 :             for (unsigned int k = kThreshold; k < nNumPixels; ++k)
     588             :             {
     589      131068 :                 memcpy(sp->uncompressed_buffer + outOff,
     590      131068 :                        sp->uncompressed_buffer + inOff, /*mainPixSize=*/3 * 2);
     591      131068 :                 inOff -= /*mainPixSize=*/3 * 2;
     592      131068 :                 outOff -= fullPixSize;
     593             :             }
     594             :         }
     595           4 :         else if (mainPixSize == 4 * 2)
     596             :         {
     597       65533 :             for (unsigned int k = kThreshold; k < nNumPixels; ++k)
     598             :             {
     599       65532 :                 memcpy(sp->uncompressed_buffer + outOff,
     600       65532 :                        sp->uncompressed_buffer + inOff, /*mainPixSize=*/4 * 2);
     601       65532 :                 inOff -= /*mainPixSize=*/4 * 2;
     602       65532 :                 outOff -= fullPixSize;
     603             :             }
     604             :         }
     605             :         else
     606             :         {
     607      196603 :             for (unsigned int k = kThreshold; k < nNumPixels; ++k)
     608             :             {
     609      196600 :                 memcpy(sp->uncompressed_buffer + outOff,
     610      196600 :                        sp->uncompressed_buffer + inOff, mainPixSize);
     611      196600 :                 inOff -= mainPixSize;
     612      196600 :                 outOff -= fullPixSize;
     613             :             }
     614             :         }
     615             :         /* Last iterations need memmove() because of overlapping between */
     616             :         /* source and target regions. */
     617          32 :         for (unsigned int k = kThreshold; k > 1;)
     618             :         {
     619          15 :             --k;
     620          15 :             memmove(sp->uncompressed_buffer + outOff,
     621          15 :                     sp->uncompressed_buffer + inOff, mainPixSize);
     622          15 :             inOff -= mainPixSize;
     623          15 :             outOff -= fullPixSize;
     624             :         }
     625             :         // then copy over the data from the extra_channel_buffer
     626          17 :         const int nExtraChannelsToExtract =
     627          17 :             info.num_extra_channels - nFirstExtraChannel;
     628          55 :         for (int i = 0; i < nExtraChannelsToExtract; ++i)
     629             :         {
     630          38 :             outOff = (i + nMainChannels) * nBytesPerSample;
     631          38 :             uint8_t *channel_buffer = extra_channel_buffer + i * channel_size;
     632          38 :             if (nBytesPerSample == 1)
     633             :             {
     634      658578 :                 for (; outOff < sp->uncompressed_size;
     635      658560 :                      outOff += fullPixSize,
     636      658560 :                      channel_buffer += /*nBytesPerSample=*/1)
     637             :                 {
     638      658560 :                     memcpy(sp->uncompressed_buffer + outOff, channel_buffer,
     639             :                            /*nBytesPerSample=*/1);
     640             :                 }
     641             :             }
     642          20 :             else if (nBytesPerSample == 2)
     643             :             {
     644      655370 :                 for (; outOff < sp->uncompressed_size;
     645      655360 :                      outOff += fullPixSize,
     646      655360 :                      channel_buffer += /*nBytesPerSample=*/2)
     647             :                 {
     648      655360 :                     memcpy(sp->uncompressed_buffer + outOff, channel_buffer,
     649             :                            /*nBytesPerSample=*/2);
     650             :                 }
     651             :             }
     652             :             else
     653             :             {
     654          10 :                 assert(nBytesPerSample == 4);
     655      655370 :                 for (; outOff < sp->uncompressed_size;
     656      655360 :                      outOff += fullPixSize, channel_buffer += nBytesPerSample)
     657             :                 {
     658      655360 :                     memcpy(sp->uncompressed_buffer + outOff, channel_buffer,
     659             :                            nBytesPerSample);
     660             :                 }
     661             :             }
     662             :         }
     663          17 :         _TIFFfreeExt(tif, extra_channel_buffer);
     664             :     }
     665             : 
     666         206 :     /*const size_t nRemaining = */ JxlDecoderReleaseInput(sp->decoder);
     667             :     /*if( nRemaining != 0 )
     668             :     {
     669             :         TIFFErrorExtR(tif, module,
     670             :                      "JxlDecoderReleaseInput(): %u input bytes remaining",
     671             :                      (unsigned)nRemaining);
     672             :     }*/
     673             : 
     674         206 :     return 1;
     675             : }
     676             : 
     677             : /*
     678             :  * Decode a strip, tile or scanline.
     679             :  */
     680         206 : static int JXLDecode(TIFF *tif, uint8_t *op, tmsize_t occ, uint16_t s)
     681             : {
     682             :     static const char module[] = "JXLDecode";
     683         206 :     JXLState *sp = DecoderState(tif);
     684             : 
     685             :     (void)s;
     686         206 :     assert(sp != NULL);
     687         206 :     assert(sp->state == LSTATE_INIT_DECODE);
     688             : 
     689         206 :     if (sp->uncompressed_buffer == 0)
     690             :     {
     691           0 :         TIFFErrorExtR(tif, module, "Uncompressed buffer not allocated");
     692           0 :         return 0;
     693             :     }
     694             : 
     695         206 :     if ((uint64_t)sp->uncompressed_offset + (uint64_t)occ >
     696         206 :         sp->uncompressed_size)
     697             :     {
     698           0 :         TIFFErrorExtR(tif, module, "Too many bytes read");
     699           0 :         return 0;
     700             :     }
     701             : 
     702         206 :     memcpy(op, sp->uncompressed_buffer + sp->uncompressed_offset, occ);
     703         206 :     sp->uncompressed_offset += (unsigned)occ;
     704             : 
     705         206 :     return 1;
     706             : }
     707             : 
     708         111 : static int JXLSetupEncode(TIFF *tif)
     709             : {
     710         111 :     JXLState *sp = EncoderState(tif);
     711             : 
     712         111 :     assert(sp != NULL);
     713         111 :     if (sp->state & LSTATE_INIT_DECODE)
     714             :     {
     715           0 :         sp->state = 0;
     716             :     }
     717             : 
     718         111 :     if (GetJXLDataType(tif) < 0)
     719           0 :         return 0;
     720             : 
     721         111 :     sp->state |= LSTATE_INIT_ENCODE;
     722             : 
     723         111 :     return 1;
     724             : }
     725             : 
     726             : /*
     727             :  * Reset encoding state at the start of a strip.
     728             :  */
     729         202 : static int JXLPreEncode(TIFF *tif, uint16_t s)
     730             : {
     731             :     static const char module[] = "JXLPreEncode";
     732         202 :     JXLState *sp = EncoderState(tif);
     733             : 
     734             :     (void)s;
     735         202 :     assert(sp != NULL);
     736         202 :     if (sp->state != LSTATE_INIT_ENCODE)
     737           0 :         tif->tif_setupencode(tif);
     738             : 
     739         202 :     if (!SetupUncompressedBuffer(tif, sp, module))
     740           0 :         return 0;
     741             : 
     742         202 :     return 1;
     743             : }
     744             : 
     745             : /*
     746             :  * Encode a chunk of pixels.
     747             :  */
     748         202 : static int JXLEncode(TIFF *tif, uint8_t *bp, tmsize_t cc, uint16_t s)
     749             : {
     750             :     static const char module[] = "JXLEncode";
     751         202 :     JXLState *sp = EncoderState(tif);
     752             : 
     753             :     (void)s;
     754         202 :     assert(sp != NULL);
     755         202 :     assert(sp->state == LSTATE_INIT_ENCODE);
     756             : 
     757         202 :     if ((uint64_t)sp->uncompressed_offset + (uint64_t)cc >
     758         202 :         sp->uncompressed_size)
     759             :     {
     760           0 :         TIFFErrorExtR(tif, module, "Too many bytes written");
     761           0 :         return 0;
     762             :     }
     763             : 
     764         202 :     memcpy(sp->uncompressed_buffer + sp->uncompressed_offset, bp, cc);
     765         202 :     sp->uncompressed_offset += (unsigned)cc;
     766             : 
     767         202 :     return 1;
     768             : }
     769             : 
     770             : /*
     771             :  * Finish off an encoded strip by flushing it.
     772             :  */
     773         202 : static int JXLPostEncode(TIFF *tif)
     774             : {
     775             :     static const char module[] = "JXLPostEncode";
     776         202 :     JXLState *sp = EncoderState(tif);
     777         202 :     TIFFDirectory *td = &tif->tif_dir;
     778             : 
     779         202 :     if (sp->uncompressed_offset != sp->uncompressed_size)
     780             :     {
     781           0 :         TIFFErrorExtR(tif, module, "Unexpected number of bytes in the buffer");
     782           0 :         return 0;
     783             :     }
     784             : 
     785         202 :     JxlEncoder *enc = JxlEncoderCreate(NULL);
     786         202 :     if (enc == NULL)
     787             :     {
     788           0 :         TIFFErrorExtR(tif, module, "JxlEncoderCreate() failed");
     789           0 :         return 0;
     790             :     }
     791         202 :     JxlEncoderUseContainer(enc, JXL_FALSE);
     792             : 
     793             : #ifdef HAVE_JxlEncoderFrameSettingsCreate
     794         202 :     JxlEncoderFrameSettings *opts = JxlEncoderFrameSettingsCreate(enc, NULL);
     795             : #else
     796             :     JxlEncoderOptions *opts = JxlEncoderOptionsCreate(enc, NULL);
     797             : #endif
     798         202 :     if (opts == NULL)
     799             :     {
     800           0 :         TIFFErrorExtR(tif, module, "JxlEncoderFrameSettingsCreate() failed");
     801           0 :         JxlEncoderDestroy(enc);
     802           0 :         return 0;
     803             :     }
     804             : 
     805         202 :     JxlPixelFormat format = {0};
     806         202 :     format.data_type = GetJXLDataType(tif);
     807         202 :     format.endianness = JXL_NATIVE_ENDIAN;
     808         202 :     format.align = 0;
     809             : 
     810             : #ifdef HAVE_JxlEncoderSetCodestreamLevel
     811         202 :     if (td->td_bitspersample > 12)
     812             :     {
     813          90 :         JxlEncoderSetCodestreamLevel(enc, 10);
     814             :     }
     815             : #endif
     816         202 :     JxlBasicInfo basic_info = {0};
     817         202 :     JxlEncoderInitBasicInfo(&basic_info);
     818         202 :     basic_info.xsize = sp->segment_width;
     819         202 :     basic_info.ysize = sp->segment_height;
     820         202 :     basic_info.bits_per_sample = td->td_bitspersample;
     821         202 :     basic_info.orientation = JXL_ORIENT_IDENTITY;
     822         202 :     if (td->td_sampleformat == SAMPLEFORMAT_IEEEFP)
     823             :     {
     824          45 :         if (td->td_bitspersample == 32)
     825          44 :             basic_info.exponent_bits_per_sample = 8;
     826             :         else
     827           1 :             basic_info.exponent_bits_per_sample = 5;
     828             :     }
     829             :     else
     830             :     {
     831         157 :         basic_info.exponent_bits_per_sample = 0;
     832             :     }
     833             : 
     834         202 :     int bAlphaEmbedded = 0;
     835         202 :     const int bAlphaDistanceSameAsMainChannel =
     836         217 :         (sp->alpha_distance < 0.0f) ||
     837          15 :         ((sp->lossless && sp->alpha_distance == 0.0f) ||
     838          15 :          (!sp->lossless && sp->alpha_distance == sp->distance));
     839             : #ifndef HAVE_JxlEncoderSetExtraChannelDistance
     840             :     if (!bAlphaDistanceSameAsMainChannel)
     841             :     {
     842             :         TIFFWarningExtR(tif, module,
     843             :                         "AlphaDistance ignored due to "
     844             :                         "JxlEncoderSetExtraChannelDistance() not being "
     845             :                         "available. Please upgrade libjxl to > 0.8.1");
     846             :     }
     847             : #endif
     848             : 
     849         202 :     if (td->td_planarconfig == PLANARCONFIG_SEPARATE)
     850             :     {
     851         102 :         format.num_channels = 1;
     852         102 :         basic_info.num_color_channels = 1;
     853         102 :         basic_info.num_extra_channels = 0;
     854         102 :         basic_info.alpha_bits = 0;
     855         102 :         basic_info.alpha_exponent_bits = 0;
     856             :     }
     857             :     else
     858             :     {
     859         100 :         if (td->td_photometric == PHOTOMETRIC_MINISBLACK &&
     860          28 :             td->td_extrasamples > 0 &&
     861          14 :             td->td_extrasamples == td->td_samplesperpixel - 1 &&
     862          14 :             td->td_sampleinfo[0] == EXTRASAMPLE_UNASSALPHA &&
     863             :             bAlphaDistanceSameAsMainChannel)
     864             :         {  // gray with alpha
     865           6 :             format.num_channels = 2;
     866           6 :             basic_info.num_color_channels = 1;
     867           6 :             basic_info.num_extra_channels = td->td_extrasamples;
     868           6 :             basic_info.alpha_bits = td->td_bitspersample;
     869           6 :             basic_info.alpha_exponent_bits =
     870           6 :                 basic_info.exponent_bits_per_sample;
     871           6 :             bAlphaEmbedded = 1;
     872             :         }
     873          94 :         else if (td->td_photometric == PHOTOMETRIC_RGB &&
     874          72 :                  td->td_extrasamples > 0 &&
     875          30 :                  td->td_extrasamples == td->td_samplesperpixel - 3 &&
     876          30 :                  td->td_sampleinfo[0] == EXTRASAMPLE_UNASSALPHA &&
     877             :                  bAlphaDistanceSameAsMainChannel)
     878             :         {  // rgb with alpha, and same distance for alpha vs non-alpha channels
     879           9 :             format.num_channels = 4;
     880           9 :             basic_info.num_color_channels = 3;
     881           9 :             basic_info.num_extra_channels = td->td_samplesperpixel - 3;
     882           9 :             basic_info.alpha_bits = td->td_bitspersample;
     883           9 :             basic_info.alpha_exponent_bits =
     884           9 :                 basic_info.exponent_bits_per_sample;
     885           9 :             bAlphaEmbedded = 1;
     886             :         }
     887          85 :         else if (td->td_photometric == PHOTOMETRIC_RGB &&
     888          63 :                  ((td->td_extrasamples == 0) ||
     889          21 :                   (td->td_extrasamples > 0 &&
     890          21 :                    td->td_extrasamples == td->td_samplesperpixel - 3 &&
     891          21 :                    (td->td_sampleinfo[0] != EXTRASAMPLE_UNASSALPHA ||
     892             :                     !bAlphaDistanceSameAsMainChannel))))
     893             :         {  // rgb without alpha, or differente distance for alpha vs non-alpha
     894             :             // channels
     895          63 :             format.num_channels = 3;
     896          63 :             basic_info.num_color_channels = 3;
     897          63 :             basic_info.num_extra_channels = td->td_samplesperpixel - 3;
     898          63 :             basic_info.alpha_bits = 0;
     899          63 :             basic_info.alpha_exponent_bits = 0;
     900             :         }
     901             :         else
     902             :         {  // fallback to gray without alpha and with eventual extra channels
     903          22 :             format.num_channels = 1;
     904          22 :             basic_info.num_color_channels = 1;
     905          22 :             basic_info.num_extra_channels = td->td_samplesperpixel - 1;
     906          22 :             basic_info.alpha_bits = 0;
     907          22 :             basic_info.alpha_exponent_bits = 0;
     908             :         }
     909             : #ifndef HAVE_JxlExtraChannels
     910             :         if (basic_info.num_extra_channels > 1 ||
     911             :             (basic_info.num_extra_channels == 1 && !bAlphaEmbedded))
     912             :         {
     913             :             TIFFErrorExtR(
     914             :                 tif, module,
     915             :                 "JXL: INTERLEAVE=PIXEL does not support this combination of "
     916             :                 "bands. Please upgrade libjxl to 0.8+");
     917             :             return 0;
     918             :         }
     919             : #endif
     920             :     }
     921             : 
     922         202 :     if (sp->lossless)
     923             :     {
     924             : #ifdef HAVE_JxlEncoderSetFrameLossless
     925         166 :         JxlEncoderSetFrameLossless(opts, TRUE);
     926             : #else
     927             :         JxlEncoderOptionsSetLossless(opts, TRUE);
     928             : #endif
     929             : #ifdef HAVE_JxlEncoderSetFrameDistance
     930         166 :         JxlEncoderSetFrameDistance(opts, 0);
     931             : #else
     932             :         JxlEncoderOptionsSetDistance(opts, 0);
     933             : #endif
     934         166 :         basic_info.uses_original_profile = JXL_TRUE;
     935             :     }
     936             :     else
     937             :     {
     938             : #ifdef HAVE_JxlEncoderSetFrameDistance
     939          36 :         if (JxlEncoderSetFrameDistance(opts, sp->distance) != JXL_ENC_SUCCESS)
     940             : #else
     941             :         if (JxlEncoderOptionsSetDistance(opts, sp->distance) != JXL_ENC_SUCCESS)
     942             : #endif
     943             :         {
     944           0 :             TIFFErrorExtR(tif, module, "JxlEncoderSetFrameDistance() failed");
     945           0 :             JxlEncoderDestroy(enc);
     946           0 :             return 0;
     947             :         }
     948             :     }
     949             : #ifdef HAVE_JxlEncoderFrameSettingsSetOption
     950         202 :     if (JxlEncoderFrameSettingsSetOption(opts, JXL_ENC_FRAME_SETTING_EFFORT,
     951         202 :                                          sp->effort) != JXL_ENC_SUCCESS)
     952             : #else
     953             :     if (JxlEncoderOptionsSetEffort(opts, sp->effort) != JXL_ENC_SUCCESS)
     954             : #endif
     955             :     {
     956           0 :         TIFFErrorExtR(tif, module, "JxlEncoderFrameSettingsSetOption() failed");
     957           0 :         JxlEncoderDestroy(enc);
     958           0 :         return 0;
     959             :     }
     960             : 
     961         202 :     if (JXL_ENC_SUCCESS != JxlEncoderSetBasicInfo(enc, &basic_info))
     962             :     {
     963           0 :         TIFFErrorExtR(tif, module, "JxlEncoderSetBasicInfo() failed");
     964           0 :         JxlEncoderDestroy(enc);
     965           0 :         return 0;
     966             :     }
     967             : 
     968         202 :     JxlColorEncoding color_encoding = {0};
     969         202 :     JxlColorEncodingSetToSRGB(&color_encoding, /*is_gray*/
     970         202 :                               (td->td_planarconfig == PLANARCONFIG_SEPARATE ||
     971         100 :                                basic_info.num_color_channels == 1));
     972         202 :     if (JXL_ENC_SUCCESS != JxlEncoderSetColorEncoding(enc, &color_encoding))
     973             :     {
     974           0 :         TIFFErrorExtR(tif, module, "JxlEncoderSetColorEncoding() failed");
     975           0 :         JxlEncoderDestroy(enc);
     976           0 :         return 0;
     977             :     }
     978             : 
     979         202 :     uint8_t *main_buffer = sp->uncompressed_buffer;
     980         202 :     unsigned int main_size = sp->uncompressed_size;
     981             : 
     982             : #ifdef HAVE_JxlExtraChannels
     983         202 :     int nBytesPerSample = GetJXLDataTypeSize(format.data_type);
     984         202 :     if (td->td_planarconfig == PLANARCONFIG_CONTIG &&
     985         100 :         (basic_info.num_extra_channels > 1 ||
     986          86 :          (basic_info.num_extra_channels == 1 && !bAlphaEmbedded)))
     987             :     {
     988          32 :         main_size = (sp->uncompressed_size / td->td_samplesperpixel);
     989          32 :         int nMainChannels = basic_info.num_color_channels;
     990          32 :         if (bAlphaEmbedded)
     991           3 :             nMainChannels++;
     992          32 :         main_size *= nMainChannels;
     993          32 :         main_buffer = _TIFFmallocExt(tif, main_size);
     994          32 :         if (main_buffer == NULL)
     995           0 :             return 0;
     996          32 :         int outChunkSize = nBytesPerSample * nMainChannels;
     997          32 :         int inStep = nBytesPerSample * td->td_samplesperpixel;
     998          32 :         uint8_t *cur_outbuffer = main_buffer;
     999          32 :         uint8_t *cur_inbuffer = sp->uncompressed_buffer;
    1000     1286700 :         for (; cur_outbuffer - main_buffer < main_size;
    1001     1286670 :              cur_outbuffer += outChunkSize, cur_inbuffer += inStep)
    1002             :         {
    1003     1286670 :             memcpy(cur_outbuffer, cur_inbuffer, outChunkSize);
    1004             :         }
    1005          85 :         for (int iChannel = nMainChannels; iChannel < td->td_samplesperpixel;
    1006          53 :              iChannel++)
    1007             :         {
    1008             :             JxlExtraChannelInfo extra_channel_info;
    1009          53 :             int channelType = JXL_CHANNEL_OPTIONAL;
    1010          53 :             const int iExtraChannel = iChannel - nMainChannels + bAlphaEmbedded;
    1011          53 :             if (iExtraChannel < td->td_extrasamples &&
    1012          53 :                 (td->td_sampleinfo[iExtraChannel] == EXTRASAMPLE_UNASSALPHA ||
    1013          32 :                  td->td_sampleinfo[iExtraChannel] == EXTRASAMPLE_ASSOCALPHA))
    1014             :             {
    1015          21 :                 channelType = JXL_CHANNEL_ALPHA;
    1016             :             }
    1017          53 :             JxlEncoderInitExtraChannelInfo(channelType, &extra_channel_info);
    1018          53 :             extra_channel_info.bits_per_sample = basic_info.bits_per_sample;
    1019          53 :             extra_channel_info.exponent_bits_per_sample =
    1020          53 :                 basic_info.exponent_bits_per_sample;
    1021          53 :             if (iExtraChannel < td->td_extrasamples &&
    1022          53 :                 td->td_sampleinfo[iExtraChannel] == EXTRASAMPLE_ASSOCALPHA)
    1023             :             {
    1024           0 :                 extra_channel_info.alpha_premultiplied = JXL_TRUE;
    1025             :             }
    1026             : 
    1027          53 :             if (JXL_ENC_SUCCESS != JxlEncoderSetExtraChannelInfo(
    1028             :                                        enc, iExtraChannel, &extra_channel_info))
    1029             :             {
    1030           0 :                 TIFFErrorExtR(tif, module,
    1031             :                               "JxlEncoderSetExtraChannelInfo(%d) failed",
    1032             :                               iChannel);
    1033           0 :                 JxlEncoderDestroy(enc);
    1034           0 :                 _TIFFfreeExt(tif, main_buffer);
    1035           0 :                 return 0;
    1036             :             }
    1037             : #if HAVE_JxlEncoderSetExtraChannelDistance
    1038          53 :             if (channelType == JXL_CHANNEL_ALPHA && sp->alpha_distance >= 0.0f)
    1039             :             {
    1040          15 :                 if (JXL_ENC_SUCCESS !=
    1041          15 :                     JxlEncoderSetExtraChannelDistance(opts, iExtraChannel,
    1042             :                                                       sp->alpha_distance))
    1043             :                 {
    1044           0 :                     TIFFErrorExtR(
    1045             :                         tif, module,
    1046             :                         "JxlEncoderSetExtraChannelDistance(%d) failed",
    1047             :                         iChannel);
    1048           0 :                     JxlEncoderDestroy(enc);
    1049           0 :                     _TIFFfreeExt(tif, main_buffer);
    1050           0 :                     return 0;
    1051             :                 }
    1052             :             }
    1053          38 :             else if (!(sp->lossless))
    1054             :             {
    1055             :                 // By default libjxl applies lossless encoding for extra channels
    1056           4 :                 if (JXL_ENC_SUCCESS != JxlEncoderSetExtraChannelDistance(
    1057             :                                            opts, iExtraChannel, sp->distance))
    1058             :                 {
    1059           0 :                     TIFFErrorExtR(
    1060             :                         tif, module,
    1061             :                         "JxlEncoderSetExtraChannelDistance(%d) failed",
    1062             :                         iChannel);
    1063           0 :                     JxlEncoderDestroy(enc);
    1064           0 :                     _TIFFfreeExt(tif, main_buffer);
    1065           0 :                     return 0;
    1066             :                 }
    1067             :             }
    1068             : #endif
    1069             :         }
    1070             :     }
    1071             : #endif
    1072             : 
    1073         202 :     int retCode =
    1074         202 :         JxlEncoderAddImageFrame(opts, &format, main_buffer, main_size);
    1075             :     // cleanup now
    1076         202 :     if (main_buffer != sp->uncompressed_buffer)
    1077             :     {
    1078          32 :         _TIFFfreeExt(tif, main_buffer);
    1079             :     }
    1080         202 :     if (retCode != JXL_ENC_SUCCESS)
    1081             :     {
    1082           0 :         TIFFErrorExtR(tif, module, "JxlEncoderAddImageFrame() failed");
    1083           0 :         JxlEncoderDestroy(enc);
    1084           0 :         return 0;
    1085             :     }
    1086             : 
    1087             : #ifdef HAVE_JxlExtraChannels
    1088         202 :     if (td->td_planarconfig == PLANARCONFIG_CONTIG &&
    1089         100 :         (basic_info.num_extra_channels > 1 ||
    1090          86 :          (basic_info.num_extra_channels == 1 && !bAlphaEmbedded)))
    1091             :     {
    1092          32 :         int nMainChannels = basic_info.num_color_channels;
    1093          32 :         if (bAlphaEmbedded)
    1094           3 :             nMainChannels++;
    1095          32 :         int extra_channel_size =
    1096          32 :             (sp->uncompressed_size / td->td_samplesperpixel);
    1097          32 :         uint8_t *extra_channel_buffer = _TIFFmallocExt(tif, extra_channel_size);
    1098          32 :         if (extra_channel_buffer == NULL)
    1099           0 :             return 0;
    1100          32 :         int inStep = nBytesPerSample * td->td_samplesperpixel;
    1101          32 :         int outStep = nBytesPerSample;
    1102          85 :         for (int iChannel = nMainChannels; iChannel < td->td_samplesperpixel;
    1103          53 :              iChannel++)
    1104             :         {
    1105          53 :             uint8_t *cur_outbuffer = extra_channel_buffer;
    1106          53 :             uint8_t *cur_inbuffer =
    1107          53 :                 sp->uncompressed_buffer + iChannel * outStep;
    1108     2272160 :             for (; cur_outbuffer - extra_channel_buffer < extra_channel_size;
    1109     2272110 :                  cur_outbuffer += outStep, cur_inbuffer += inStep)
    1110             :             {
    1111     2272110 :                 memcpy(cur_outbuffer, cur_inbuffer, outStep);
    1112             :             }
    1113         106 :             if (JxlEncoderSetExtraChannelBuffer(
    1114             :                     opts, &format, extra_channel_buffer, extra_channel_size,
    1115             :                     (bAlphaEmbedded)
    1116           3 :                         ? iChannel - nMainChannels + 1
    1117          50 :                         : iChannel - nMainChannels) != JXL_ENC_SUCCESS)
    1118             :             {
    1119           0 :                 TIFFErrorExtR(tif, module,
    1120             :                               "JxlEncoderSetExtraChannelBuffer() failed");
    1121           0 :                 _TIFFfreeExt(tif, extra_channel_buffer);
    1122           0 :                 JxlEncoderDestroy(enc);
    1123           0 :                 return 0;
    1124             :             }
    1125             :         }
    1126          32 :         _TIFFfreeExt(tif, extra_channel_buffer);
    1127             :     }
    1128             : #endif
    1129             : 
    1130         202 :     JxlEncoderCloseInput(enc);
    1131             : 
    1132             :     while (TRUE)
    1133           0 :     {
    1134         202 :         size_t len = (size_t)tif->tif_rawdatasize;
    1135         202 :         uint8_t *buf = (uint8_t *)tif->tif_rawdata;
    1136             :         JxlEncoderStatus process_result =
    1137         202 :             JxlEncoderProcessOutput(enc, &buf, &len);
    1138         202 :         if (process_result == JXL_ENC_ERROR)
    1139             :         {
    1140           0 :             TIFFErrorExtR(tif, module, "JxlEncoderProcessOutput() failed");
    1141           0 :             JxlEncoderDestroy(enc);
    1142           0 :             return 0;
    1143             :         }
    1144         202 :         tif->tif_rawcc = tif->tif_rawdatasize - len;
    1145         202 :         if (!TIFFFlushData1(tif))
    1146             :         {
    1147           0 :             JxlEncoderDestroy(enc);
    1148           0 :             return 0;
    1149             :         }
    1150         202 :         if (process_result != JXL_ENC_NEED_MORE_OUTPUT)
    1151         202 :             break;
    1152             :     }
    1153             : 
    1154         202 :     JxlEncoderDestroy(enc);
    1155         202 :     return 1;
    1156             : }
    1157             : 
    1158         541 : static void JXLCleanup(TIFF *tif)
    1159             : {
    1160         541 :     JXLState *sp = LState(tif);
    1161             : 
    1162         541 :     assert(sp != 0);
    1163             : 
    1164         541 :     tif->tif_tagmethods.vgetfield = sp->vgetparent;
    1165         541 :     tif->tif_tagmethods.vsetfield = sp->vsetparent;
    1166             : 
    1167         541 :     _TIFFfreeExt(tif, sp->uncompressed_buffer);
    1168             : 
    1169         541 :     if (sp->decoder)
    1170         117 :         JxlDecoderDestroy(sp->decoder);
    1171             : 
    1172         541 :     _TIFFfreeExt(tif, sp);
    1173         541 :     tif->tif_data = NULL;
    1174             : 
    1175         541 :     _TIFFSetDefaultCompressionState(tif);
    1176         541 : }
    1177             : 
    1178             : static const TIFFField JXLFields[] = {
    1179             :     {TIFFTAG_JXL_LOSSYNESS, 0, 0, TIFF_ANY, 0, TIFF_SETGET_UINT32, FIELD_PSEUDO,
    1180             :      FALSE, FALSE, "Lossyness", NULL},
    1181             :     {TIFFTAG_JXL_EFFORT, 0, 0, TIFF_ANY, 0, TIFF_SETGET_UINT32, FIELD_PSEUDO,
    1182             :      FALSE, FALSE, "Effort", NULL},
    1183             :     {TIFFTAG_JXL_DISTANCE, 0, 0, TIFF_ANY, 0, TIFF_SETGET_FLOAT, FIELD_PSEUDO,
    1184             :      FALSE, FALSE, "Distance", NULL},
    1185             :     {TIFFTAG_JXL_ALPHA_DISTANCE, 0, 0, TIFF_ANY, 0, TIFF_SETGET_FLOAT,
    1186             :      FIELD_PSEUDO, FALSE, FALSE, "AlphaDistance", NULL},
    1187             : };
    1188             : 
    1189        5747 : static int JXLVSetField(TIFF *tif, uint32_t tag, va_list ap)
    1190             : {
    1191             :     static const char module[] = "JXLVSetField";
    1192        5747 :     JXLState *sp = LState(tif);
    1193             : 
    1194        5747 :     switch (tag)
    1195             :     {
    1196         270 :         case TIFFTAG_JXL_LOSSYNESS:
    1197             :         {
    1198         270 :             uint32_t lossyness = va_arg(ap, uint32_t);
    1199         270 :             if (lossyness == JXL_LOSSLESS)
    1200         209 :                 sp->lossless = TRUE;
    1201          61 :             else if (lossyness == JXL_LOSSY)
    1202          61 :                 sp->lossless = FALSE;
    1203             :             else
    1204             :             {
    1205           0 :                 TIFFErrorExtR(tif, module, "Invalid value for Lossyness: %u",
    1206             :                               lossyness);
    1207           0 :                 return 0;
    1208             :             }
    1209         270 :             return 1;
    1210             :         }
    1211             : 
    1212         265 :         case TIFFTAG_JXL_EFFORT:
    1213             :         {
    1214         265 :             uint32_t effort = va_arg(ap, uint32_t);
    1215         265 :             if (effort < 1 || effort > 9)
    1216             :             {
    1217           0 :                 TIFFErrorExtR(tif, module, "Invalid value for Effort: %u",
    1218             :                               effort);
    1219           0 :                 return 0;
    1220             :             }
    1221         265 :             sp->effort = effort;
    1222         265 :             return 1;
    1223             :         }
    1224             : 
    1225         266 :         case TIFFTAG_JXL_DISTANCE:
    1226             :         {
    1227         266 :             float distance = (float)va_arg(ap, double);
    1228         266 :             if (distance < 0 || distance > 15)
    1229             :             {
    1230           0 :                 TIFFErrorExtR(tif, module, "Invalid value for Distance: %f",
    1231             :                               distance);
    1232           0 :                 return 0;
    1233             :             }
    1234         266 :             sp->distance = distance;
    1235         266 :             return 1;
    1236             :         }
    1237             : 
    1238         266 :         case TIFFTAG_JXL_ALPHA_DISTANCE:
    1239             :         {
    1240         266 :             float alpha_distance = (float)va_arg(ap, double);
    1241         266 :             if (alpha_distance != -1 &&
    1242           6 :                 (alpha_distance < 0 || alpha_distance > 15))
    1243             :             {
    1244           0 :                 TIFFErrorExtR(tif, module,
    1245             :                               "Invalid value for AlphaDistance: %f",
    1246             :                               alpha_distance);
    1247           0 :                 return 0;
    1248             :             }
    1249         266 :             sp->alpha_distance = alpha_distance;
    1250         266 :             return 1;
    1251             :         }
    1252             : 
    1253        4680 :         default:
    1254             :         {
    1255        4680 :             return (*sp->vsetparent)(tif, tag, ap);
    1256             :         }
    1257             :     }
    1258             :     /*NOTREACHED*/
    1259             : }
    1260             : 
    1261        8233 : static int JXLVGetField(TIFF *tif, uint32_t tag, va_list ap)
    1262             : {
    1263        8233 :     JXLState *sp = LState(tif);
    1264             : 
    1265        8233 :     switch (tag)
    1266             :     {
    1267           0 :         case TIFFTAG_JXL_LOSSYNESS:
    1268           0 :             *va_arg(ap, uint32_t *) = sp->lossless ? JXL_LOSSLESS : JXL_LOSSY;
    1269           0 :             break;
    1270           0 :         case TIFFTAG_JXL_EFFORT:
    1271           0 :             *va_arg(ap, uint32_t *) = sp->effort;
    1272           0 :             break;
    1273           0 :         case TIFFTAG_JXL_DISTANCE:
    1274           0 :             *va_arg(ap, float *) = sp->distance;
    1275           0 :             break;
    1276           0 :         case TIFFTAG_JXL_ALPHA_DISTANCE:
    1277           0 :             *va_arg(ap, float *) = sp->alpha_distance;
    1278           0 :             break;
    1279        8233 :         default:
    1280        8233 :             return (*sp->vgetparent)(tif, tag, ap);
    1281             :     }
    1282           0 :     return 1;
    1283             : }
    1284             : 
    1285         541 : int TIFFInitJXL(TIFF *tif, int scheme)
    1286             : {
    1287             :     static const char module[] = "TIFFInitJXL";
    1288             :     JXLState *sp;
    1289             : 
    1290             :     (void)scheme;
    1291         541 :     assert(scheme == COMPRESSION_JXL || scheme == COMPRESSION_JXL_DNG_1_7);
    1292             : 
    1293             :     /*
    1294             :      * Merge codec-specific tag information.
    1295             :      */
    1296         541 :     if (!_TIFFMergeFields(tif, JXLFields, TIFFArrayCount(JXLFields)))
    1297             :     {
    1298           0 :         TIFFErrorExtR(tif, module, "Merging JXL codec-specific tags failed");
    1299           0 :         return 0;
    1300             :     }
    1301             : 
    1302             :     /*
    1303             :      * Allocate state block so tag methods have storage to record values.
    1304             :      */
    1305         541 :     tif->tif_data = (uint8_t *)_TIFFcallocExt(tif, 1, sizeof(JXLState));
    1306         541 :     if (tif->tif_data == NULL)
    1307           0 :         goto bad;
    1308         541 :     sp = LState(tif);
    1309             : 
    1310             :     /*
    1311             :      * Override parent get/set field methods.
    1312             :      */
    1313         541 :     sp->vgetparent = tif->tif_tagmethods.vgetfield;
    1314         541 :     tif->tif_tagmethods.vgetfield = JXLVGetField; /* hook for codec tags */
    1315         541 :     sp->vsetparent = tif->tif_tagmethods.vsetfield;
    1316         541 :     tif->tif_tagmethods.vsetfield = JXLVSetField; /* hook for codec tags */
    1317             : 
    1318             :     /*
    1319             :      * Install codec methods.
    1320             :      */
    1321         541 :     tif->tif_fixuptags = JXLFixupTags;
    1322         541 :     tif->tif_setupdecode = JXLSetupDecode;
    1323         541 :     tif->tif_predecode = JXLPreDecode;
    1324         541 :     tif->tif_decoderow = JXLDecode;
    1325         541 :     tif->tif_decodestrip = JXLDecode;
    1326         541 :     tif->tif_decodetile = JXLDecode;
    1327         541 :     tif->tif_setupencode = JXLSetupEncode;
    1328         541 :     tif->tif_preencode = JXLPreEncode;
    1329         541 :     tif->tif_postencode = JXLPostEncode;
    1330         541 :     tif->tif_encoderow = JXLEncode;
    1331         541 :     tif->tif_encodestrip = JXLEncode;
    1332         541 :     tif->tif_encodetile = JXLEncode;
    1333         541 :     tif->tif_cleanup = JXLCleanup;
    1334             : 
    1335             :     /* Default values for codec-specific fields */
    1336         541 :     sp->decoder = NULL;
    1337             : 
    1338         541 :     sp->state = 0;
    1339         541 :     sp->lossless = TRUE;
    1340         541 :     sp->effort = 5;
    1341         541 :     sp->distance = 1.0;
    1342         541 :     sp->alpha_distance = -1.0;
    1343             : 
    1344         541 :     return 1;
    1345           0 : bad:
    1346           0 :     TIFFErrorExtR(tif, module, "No space for JXL state block");
    1347           0 :     return 0;
    1348             : }

Generated by: LCOV version 1.14