Coverage Report

Created: 2026-01-17 06:44

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