Coverage Report

Created: 2026-02-26 06:56

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