Coverage Report

Created: 2025-11-11 06:48

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
4.00k
  : fImageSize  ()
34
4.00k
  , fTileSize   ()
35
4.00k
  , fUsesStrips (false)
36
4.00k
  , fJPEGTables ()
37
4.00k
  , fJPEGData   ()
38
  
39
4.00k
  {
40
  
41
4.00k
  }
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
3.78k
      : fHost       (host)
76
3.78k
      , fWriter       (writer)
77
3.78k
      , fImage        (image)
78
3.78k
      , fJPEGImage        (jpegImage)
79
3.78k
      , fTileCount      (tileCount)
80
3.78k
      , fIFD          (ifd)
81
3.78k
      , fMutex        ("dng_jpeg_image_encode_task")
82
3.78k
      , fNextTileIndex    (0)
83
      
84
3.78k
      {
85
      
86
3.78k
      fMinTaskArea = 16 * 16;
87
3.78k
      fUnitCell    = dng_point (16, 16);
88
3.78k
      fMaxTileSize = dng_point (16, 16);
89
      
90
3.78k
      }
91
  
92
    void Process (uint32 /* threadIndex */,
93
            const dng_rect & /* tile */,
94
            dng_abort_sniffer *sniffer)
95
3.78k
      {
96
      
97
3.78k
      AutoPtr<dng_memory_block> compressedBuffer;
98
3.78k
      AutoPtr<dng_memory_block> uncompressedBuffer;
99
3.78k
      AutoPtr<dng_memory_block> subTileBlockBuffer;
100
3.78k
      AutoPtr<dng_memory_block> tempBuffer;
101
      
102
3.78k
      uint32 uncompressedSize = SafeUint32Mult (
103
3.78k
        fIFD.fTileLength, fIFD.fTileWidth, fIFD.fSamplesPerPixel);
104
      
105
3.78k
      uncompressedBuffer.Reset (fHost.Allocate (uncompressedSize));
106
      
107
3.78k
      uint32 tilesAcross = fIFD.TilesAcross ();
108
  
109
18.3k
      while (true)
110
18.3k
        {
111
        
112
18.3k
        uint32 tileIndex;
113
        
114
18.3k
          {
115
          
116
18.3k
          dng_lock_mutex lock (&fMutex);
117
          
118
18.3k
          if (fNextTileIndex == fTileCount)
119
3.77k
            {
120
3.77k
            return;
121
3.77k
            }
122
            
123
14.5k
          tileIndex = fNextTileIndex++;
124
                    
125
14.5k
          }
126
          
127
0
        dng_abort_sniffer::SniffForAbort (sniffer);
128
        
129
14.5k
        uint32 rowIndex = tileIndex / tilesAcross;
130
14.5k
        uint32 colIndex = tileIndex % tilesAcross;
131
        
132
14.5k
        dng_rect tileArea = fIFD.TileArea (rowIndex, colIndex);
133
        
134
14.5k
        dng_memory_stream stream (fHost.Allocator ());
135
        
136
14.5k
        fWriter.WriteTile (fHost,
137
14.5k
                   fIFD,
138
14.5k
                   stream,
139
14.5k
                   fImage,
140
14.5k
                   tileArea,
141
14.5k
                   1,
142
14.5k
                   compressedBuffer,
143
14.5k
                   uncompressedBuffer,
144
14.5k
                   subTileBlockBuffer,
145
14.5k
                   tempBuffer);
146
                  
147
14.5k
        fJPEGImage.fJPEGData [tileIndex].Reset (stream.AsMemoryBlock (fHost.Allocator ()));
148
          
149
14.5k
        }
150
      
151
3.78k
      }
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
3.78k
  {
170
  
171
  #if qDNGValidate
172
  dng_timer timer ("Encode JPEG Proxy time");
173
  #endif
174
  
175
3.78k
  DNG_ASSERT (image.PixelType () == ttByte, "Cannot JPEG encode non-byte image");
176
  
177
3.78k
  fImageSize = image.Bounds ().Size ();
178
  
179
3.78k
  dng_ifd ifd;
180
  
181
3.78k
  ifd.fImageWidth  = fImageSize.h;
182
3.78k
  ifd.fImageLength = fImageSize.v;
183
  
184
3.78k
  ifd.fSamplesPerPixel = image.Planes ();
185
  
186
3.78k
  ifd.fBitsPerSample [0] = 8;
187
3.78k
  ifd.fBitsPerSample [1] = 8;
188
3.78k
  ifd.fBitsPerSample [2] = 8;
189
3.78k
  ifd.fBitsPerSample [3] = 8;
190
  
191
3.78k
  ifd.fPhotometricInterpretation = piLinearRaw;
192
  
193
3.78k
  ifd.fCompression = ccLossyJPEG;
194
  
195
3.78k
  ifd.FindTileSize (512 * 512 * ifd.fSamplesPerPixel);
196
  
197
3.78k
  fTileSize.h = ifd.fTileWidth;
198
3.78k
  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
3.78k
  bool useHigherQuality = (uint64) ifd.fImageWidth *
206
3.78k
              (uint64) ifd.fImageLength > 5000000 ||
207
3.75k
              image.Bounds ().Size () == negative.OriginalDefaultFinalSize ();
208
  
209
3.78k
  if (negative.ColorimetricReference () == crSceneReferred)
210
3.50k
    {
211
3.50k
    ifd.fCompressionQuality = useHigherQuality ? 11 : 10;
212
3.50k
    }
213
276
  else
214
276
    {
215
276
    ifd.fCompressionQuality = useHigherQuality ? 10 : 8;
216
276
    }
217
  
218
3.78k
  uint32 tilesAcross = ifd.TilesAcross ();
219
3.78k
  uint32 tilesDown   = ifd.TilesDown   ();
220
  
221
3.78k
  uint32 tileCount = tilesAcross * tilesDown;
222
  
223
3.78k
  fJPEGData.Reset (tileCount);
224
  
225
3.78k
  uint32 threadCount = Min_uint32 (tileCount,
226
3.78k
                   host.PerformAreaTaskThreads ());
227
                     
228
3.78k
  dng_jpeg_image_encode_task task (host,
229
3.78k
                   writer,
230
3.78k
                   image,
231
3.78k
                   *this,
232
3.78k
                   tileCount,
233
3.78k
                   ifd);
234
                    
235
3.78k
  host.PerformAreaTask (task,
236
3.78k
              dng_rect (0, 0, 16, 16 * threadCount));
237
    
238
3.78k
  }
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
3.77k
      : fJPEGImage        (jpegImage)
264
3.77k
      , fTileCount      (tileCount)
265
3.77k
      , fDigests      (digests)
266
3.77k
      , fMutex        ("dng_jpeg_image_find_digest_task")
267
3.77k
      , fNextTileIndex    (0)
268
      
269
3.77k
      {
270
      
271
3.77k
      fMinTaskArea = 16 * 16;
272
3.77k
      fUnitCell    = dng_point (16, 16);
273
3.77k
      fMaxTileSize = dng_point (16, 16);
274
      
275
3.77k
      }
276
  
277
    void Process (uint32 /* threadIndex */,
278
            const dng_rect & /* tile */,
279
            dng_abort_sniffer *sniffer)
280
3.77k
      {
281
      
282
18.3k
      while (true)
283
18.3k
        {
284
        
285
18.3k
        uint32 tileIndex;
286
        
287
18.3k
          {
288
          
289
18.3k
          dng_lock_mutex lock (&fMutex);
290
          
291
18.3k
          if (fNextTileIndex == fTileCount)
292
3.77k
            {
293
3.77k
            return;
294
3.77k
            }
295
            
296
14.5k
          tileIndex = fNextTileIndex++;
297
                    
298
14.5k
          }
299
          
300
0
        dng_abort_sniffer::SniffForAbort (sniffer);
301
        
302
14.5k
        dng_md5_printer printer;
303
        
304
14.5k
        printer.Process (fJPEGImage.fJPEGData [tileIndex]->Buffer      (),
305
14.5k
                 fJPEGImage.fJPEGData [tileIndex]->LogicalSize ());
306
                 
307
14.5k
        fDigests [tileIndex] = printer.Result ();
308
          
309
14.5k
        }
310
      
311
3.77k
      }
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
3.77k
  {
327
  
328
3.77k
  uint32 tileCount = TileCount ();
329
  
330
3.77k
  uint32 arrayCount = tileCount + (fJPEGTables.Get () ? 1 : 0);
331
  
332
3.77k
  AutoArray<dng_fingerprint> digests (arrayCount);
333
  
334
  // Compute digest of each compressed tile.
335
336
3.77k
    {
337
    
338
3.77k
    uint32 threadCount = Min_uint32 (tileCount,
339
3.77k
                     host.PerformAreaTaskThreads ());
340
                     
341
3.77k
    dng_jpeg_image_find_digest_task task (*this,
342
3.77k
                        tileCount,
343
3.77k
                        digests.Get ());
344
                      
345
3.77k
    host.PerformAreaTask (task,
346
3.77k
                dng_rect (0, 0, 16, 16 * threadCount));
347
    
348
3.77k
    }
349
  
350
  // Compute digest of JPEG tables, if any.
351
    
352
3.77k
  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
3.77k
    {
367
    
368
3.77k
    dng_md5_printer printer;
369
    
370
18.3k
    for (uint32 k = 0; k < arrayCount; k++)
371
14.5k
      {
372
    
373
14.5k
      printer.Process (digests [k].data,
374
14.5k
               dng_fingerprint::kDNGFingerprintSize);
375
               
376
14.5k
      }
377
      
378
3.77k
    return printer.Result ();
379
    
380
3.77k
    }
381
  
382
3.77k
  }
383
      
384
/*****************************************************************************/
385