Coverage Report

Created: 2025-11-09 06:37

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/dng_sdk/source/dng_jpeg_image.cpp
Line
Count
Source
1
/*****************************************************************************/
2
// Copyright 2011 Adobe Systems Incorporated
3
// All Rights Reserved.
4
//
5
// NOTICE:  Adobe permits you to use, modify, and distribute this file in
6
// accordance with the terms of the Adobe license agreement accompanying it.
7
/*****************************************************************************/
8
9
/* $Id: //mondo/dng_sdk_1_4/dng_sdk/source/dng_jpeg_image.cpp#1 $ */ 
10
/* $DateTime: 2012/05/30 13:28:51 $ */
11
/* $Change: 832332 $ */
12
/* $Author: tknoll $ */
13
14
/*****************************************************************************/
15
16
#include "dng_jpeg_image.h"
17
18
#include "dng_abort_sniffer.h"
19
#include "dng_area_task.h"
20
#include "dng_assertions.h"
21
#include "dng_host.h"
22
#include "dng_ifd.h"
23
#include "dng_image.h"
24
#include "dng_image_writer.h"
25
#include "dng_memory_stream.h"
26
#include "dng_mutex.h"
27
#include "dng_safe_arithmetic.h"
28
29
/*****************************************************************************/
30
31
dng_jpeg_image::dng_jpeg_image ()
32
33
6.07k
  : fImageSize  ()
34
6.07k
  , fTileSize   ()
35
6.07k
  , fUsesStrips (false)
36
6.07k
  , fJPEGTables ()
37
6.07k
  , fJPEGData   ()
38
  
39
6.07k
  {
40
  
41
6.07k
  }
42
43
/*****************************************************************************/
44
45
class dng_jpeg_image_encode_task : public dng_area_task
46
  {
47
  
48
  private:
49
  
50
    dng_host &fHost;
51
    
52
    dng_image_writer &fWriter;
53
    
54
    const dng_image &fImage;
55
  
56
    dng_jpeg_image &fJPEGImage;
57
    
58
    uint32 fTileCount;
59
    
60
    const dng_ifd &fIFD;
61
        
62
    dng_mutex fMutex;
63
    
64
    uint32 fNextTileIndex;
65
    
66
  public:
67
  
68
    dng_jpeg_image_encode_task (dng_host &host,
69
                  dng_image_writer &writer,
70
                  const dng_image &image,
71
                  dng_jpeg_image &jpegImage,
72
                  uint32 tileCount,
73
                  const dng_ifd &ifd)
74
    
75
5.65k
      : fHost       (host)
76
5.65k
      , fWriter       (writer)
77
5.65k
      , fImage        (image)
78
5.65k
      , fJPEGImage        (jpegImage)
79
5.65k
      , fTileCount      (tileCount)
80
5.65k
      , fIFD          (ifd)
81
5.65k
      , fMutex        ("dng_jpeg_image_encode_task")
82
5.65k
      , fNextTileIndex    (0)
83
      
84
5.65k
      {
85
      
86
5.65k
      fMinTaskArea = 16 * 16;
87
5.65k
      fUnitCell    = dng_point (16, 16);
88
5.65k
      fMaxTileSize = dng_point (16, 16);
89
      
90
5.65k
      }
91
  
92
    void Process (uint32 /* threadIndex */,
93
            const dng_rect & /* tile */,
94
            dng_abort_sniffer *sniffer)
95
5.65k
      {
96
      
97
5.65k
      AutoPtr<dng_memory_block> compressedBuffer;
98
5.65k
      AutoPtr<dng_memory_block> uncompressedBuffer;
99
5.65k
      AutoPtr<dng_memory_block> subTileBlockBuffer;
100
5.65k
      AutoPtr<dng_memory_block> tempBuffer;
101
      
102
5.65k
      uint32 uncompressedSize = SafeUint32Mult (
103
5.65k
        fIFD.fTileLength, fIFD.fTileWidth, fIFD.fSamplesPerPixel);
104
      
105
5.65k
      uncompressedBuffer.Reset (fHost.Allocate (uncompressedSize));
106
      
107
5.65k
      uint32 tilesAcross = fIFD.TilesAcross ();
108
  
109
22.5k
      while (true)
110
22.5k
        {
111
        
112
22.5k
        uint32 tileIndex;
113
        
114
22.5k
          {
115
          
116
22.5k
          dng_lock_mutex lock (&fMutex);
117
          
118
22.5k
          if (fNextTileIndex == fTileCount)
119
5.64k
            {
120
5.64k
            return;
121
5.64k
            }
122
            
123
16.9k
          tileIndex = fNextTileIndex++;
124
                    
125
16.9k
          }
126
          
127
0
        dng_abort_sniffer::SniffForAbort (sniffer);
128
        
129
16.9k
        uint32 rowIndex = tileIndex / tilesAcross;
130
16.9k
        uint32 colIndex = tileIndex % tilesAcross;
131
        
132
16.9k
        dng_rect tileArea = fIFD.TileArea (rowIndex, colIndex);
133
        
134
16.9k
        dng_memory_stream stream (fHost.Allocator ());
135
        
136
16.9k
        fWriter.WriteTile (fHost,
137
16.9k
                   fIFD,
138
16.9k
                   stream,
139
16.9k
                   fImage,
140
16.9k
                   tileArea,
141
16.9k
                   1,
142
16.9k
                   compressedBuffer,
143
16.9k
                   uncompressedBuffer,
144
16.9k
                   subTileBlockBuffer,
145
16.9k
                   tempBuffer);
146
                  
147
16.9k
        fJPEGImage.fJPEGData [tileIndex].Reset (stream.AsMemoryBlock (fHost.Allocator ()));
148
          
149
16.9k
        }
150
      
151
5.65k
      }
152
    
153
  private:
154
155
    // Hidden copy constructor and assignment operator.
156
157
    dng_jpeg_image_encode_task (const dng_jpeg_image_encode_task &);
158
159
    dng_jpeg_image_encode_task & operator= (const dng_jpeg_image_encode_task &);
160
    
161
  };
162
163
/*****************************************************************************/
164
165
void dng_jpeg_image::Encode (dng_host &host,
166
               const dng_negative &negative,
167
               dng_image_writer &writer,
168
               const dng_image &image)
169
5.65k
  {
170
  
171
  #if qDNGValidate
172
  dng_timer timer ("Encode JPEG Proxy time");
173
  #endif
174
  
175
5.65k
  DNG_ASSERT (image.PixelType () == ttByte, "Cannot JPEG encode non-byte image");
176
  
177
5.65k
  fImageSize = image.Bounds ().Size ();
178
  
179
5.65k
  dng_ifd ifd;
180
  
181
5.65k
  ifd.fImageWidth  = fImageSize.h;
182
5.65k
  ifd.fImageLength = fImageSize.v;
183
  
184
5.65k
  ifd.fSamplesPerPixel = image.Planes ();
185
  
186
5.65k
  ifd.fBitsPerSample [0] = 8;
187
5.65k
  ifd.fBitsPerSample [1] = 8;
188
5.65k
  ifd.fBitsPerSample [2] = 8;
189
5.65k
  ifd.fBitsPerSample [3] = 8;
190
  
191
5.65k
  ifd.fPhotometricInterpretation = piLinearRaw;
192
  
193
5.65k
  ifd.fCompression = ccLossyJPEG;
194
  
195
5.65k
  ifd.FindTileSize (512 * 512 * ifd.fSamplesPerPixel);
196
  
197
5.65k
  fTileSize.h = ifd.fTileWidth;
198
5.65k
  fTileSize.v = ifd.fTileLength;
199
  
200
  // Need a higher quality for raw proxies than non-raw proxies,
201
  // since users often perform much greater color changes.  Also, use
202
  // we are targeting a "large" size proxy (larger than 5MP pixels), or this
203
  // is a full size proxy, then use a higher quality.
204
  
205
5.65k
  bool useHigherQuality = (uint64) ifd.fImageWidth *
206
5.65k
              (uint64) ifd.fImageLength > 5000000 ||
207
5.62k
              image.Bounds ().Size () == negative.OriginalDefaultFinalSize ();
208
  
209
5.65k
  if (negative.ColorimetricReference () == crSceneReferred)
210
5.04k
    {
211
5.04k
    ifd.fCompressionQuality = useHigherQuality ? 11 : 10;
212
5.04k
    }
213
609
  else
214
609
    {
215
609
    ifd.fCompressionQuality = useHigherQuality ? 10 : 8;
216
609
    }
217
  
218
5.65k
  uint32 tilesAcross = ifd.TilesAcross ();
219
5.65k
  uint32 tilesDown   = ifd.TilesDown   ();
220
  
221
5.65k
  uint32 tileCount = tilesAcross * tilesDown;
222
  
223
5.65k
  fJPEGData.Reset (tileCount);
224
  
225
5.65k
  uint32 threadCount = Min_uint32 (tileCount,
226
5.65k
                   host.PerformAreaTaskThreads ());
227
                     
228
5.65k
  dng_jpeg_image_encode_task task (host,
229
5.65k
                   writer,
230
5.65k
                   image,
231
5.65k
                   *this,
232
5.65k
                   tileCount,
233
5.65k
                   ifd);
234
                    
235
5.65k
  host.PerformAreaTask (task,
236
5.65k
              dng_rect (0, 0, 16, 16 * threadCount));
237
    
238
5.65k
  }
239
      
240
/*****************************************************************************/
241
242
class dng_jpeg_image_find_digest_task : public dng_area_task
243
  {
244
  
245
  private:
246
  
247
    const dng_jpeg_image &fJPEGImage;
248
    
249
    uint32 fTileCount;
250
    
251
    dng_fingerprint *fDigests;
252
        
253
    dng_mutex fMutex;
254
    
255
    uint32 fNextTileIndex;
256
    
257
  public:
258
  
259
    dng_jpeg_image_find_digest_task (const dng_jpeg_image &jpegImage,
260
                     uint32 tileCount,
261
                     dng_fingerprint *digests)
262
    
263
5.64k
      : fJPEGImage        (jpegImage)
264
5.64k
      , fTileCount      (tileCount)
265
5.64k
      , fDigests      (digests)
266
5.64k
      , fMutex        ("dng_jpeg_image_find_digest_task")
267
5.64k
      , fNextTileIndex    (0)
268
      
269
5.64k
      {
270
      
271
5.64k
      fMinTaskArea = 16 * 16;
272
5.64k
      fUnitCell    = dng_point (16, 16);
273
5.64k
      fMaxTileSize = dng_point (16, 16);
274
      
275
5.64k
      }
276
  
277
    void Process (uint32 /* threadIndex */,
278
            const dng_rect & /* tile */,
279
            dng_abort_sniffer *sniffer)
280
5.64k
      {
281
      
282
22.5k
      while (true)
283
22.5k
        {
284
        
285
22.5k
        uint32 tileIndex;
286
        
287
22.5k
          {
288
          
289
22.5k
          dng_lock_mutex lock (&fMutex);
290
          
291
22.5k
          if (fNextTileIndex == fTileCount)
292
5.64k
            {
293
5.64k
            return;
294
5.64k
            }
295
            
296
16.9k
          tileIndex = fNextTileIndex++;
297
                    
298
16.9k
          }
299
          
300
0
        dng_abort_sniffer::SniffForAbort (sniffer);
301
        
302
16.9k
        dng_md5_printer printer;
303
        
304
16.9k
        printer.Process (fJPEGImage.fJPEGData [tileIndex]->Buffer      (),
305
16.9k
                 fJPEGImage.fJPEGData [tileIndex]->LogicalSize ());
306
                 
307
16.9k
        fDigests [tileIndex] = printer.Result ();
308
          
309
16.9k
        }
310
      
311
5.64k
      }
312
    
313
  private:
314
315
    // Hidden copy constructor and assignment operator.
316
317
    dng_jpeg_image_find_digest_task (const dng_jpeg_image_find_digest_task &);
318
319
    dng_jpeg_image_find_digest_task & operator= (const dng_jpeg_image_find_digest_task &);
320
    
321
  };
322
323
/*****************************************************************************/
324
325
dng_fingerprint dng_jpeg_image::FindDigest (dng_host &host) const
326
5.64k
  {
327
  
328
5.64k
  uint32 tileCount = TileCount ();
329
  
330
5.64k
  uint32 arrayCount = tileCount + (fJPEGTables.Get () ? 1 : 0);
331
  
332
5.64k
  AutoArray<dng_fingerprint> digests (arrayCount);
333
  
334
  // Compute digest of each compressed tile.
335
336
5.64k
    {
337
    
338
5.64k
    uint32 threadCount = Min_uint32 (tileCount,
339
5.64k
                     host.PerformAreaTaskThreads ());
340
                     
341
5.64k
    dng_jpeg_image_find_digest_task task (*this,
342
5.64k
                        tileCount,
343
5.64k
                        digests.Get ());
344
                      
345
5.64k
    host.PerformAreaTask (task,
346
5.64k
                dng_rect (0, 0, 16, 16 * threadCount));
347
    
348
5.64k
    }
349
  
350
  // Compute digest of JPEG tables, if any.
351
    
352
5.64k
  if (fJPEGTables.Get ())
353
0
    {
354
    
355
0
    dng_md5_printer printer;
356
    
357
0
    printer.Process (fJPEGTables->Buffer      (),
358
0
             fJPEGTables->LogicalSize ());
359
             
360
0
    digests [tileCount] = printer.Result ();
361
    
362
0
    }
363
    
364
  // Combine digests into a single digest.
365
  
366
5.64k
    {
367
    
368
5.64k
    dng_md5_printer printer;
369
    
370
22.5k
    for (uint32 k = 0; k < arrayCount; k++)
371
16.9k
      {
372
    
373
16.9k
      printer.Process (digests [k].data,
374
16.9k
               dng_fingerprint::kDNGFingerprintSize);
375
               
376
16.9k
      }
377
      
378
5.64k
    return printer.Result ();
379
    
380
5.64k
    }
381
  
382
5.64k
  }
383
      
384
/*****************************************************************************/
385