Coverage Report

Created: 2025-01-23 06:29

/src/dng_sdk/source/dng_image_writer.cpp
Line
Count
Source (jump to first uncovered line)
1
/*****************************************************************************/
2
// Copyright 2006-2012 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_image_writer.cpp#4 $ */ 
10
/* $DateTime: 2012/06/14 20:24:41 $ */
11
/* $Change: 835078 $ */
12
/* $Author: tknoll $ */
13
14
/*****************************************************************************/
15
16
#include "dng_image_writer.h"
17
18
#include "dng_abort_sniffer.h"
19
#include "dng_area_task.h"
20
#include "dng_bottlenecks.h"
21
#include "dng_camera_profile.h"
22
#include "dng_color_space.h"
23
#include "dng_exif.h"
24
#include "dng_flags.h"
25
#include "dng_exceptions.h"
26
#include "dng_host.h"
27
#include "dng_ifd.h"
28
#include "dng_image.h"
29
#include "dng_jpeg_image.h"
30
#include "dng_lossless_jpeg.h"
31
#include "dng_memory.h"
32
#include "dng_memory_stream.h"
33
#include "dng_negative.h"
34
#include "dng_pixel_buffer.h"
35
#include "dng_preview.h"
36
#include "dng_read_image.h"
37
#include "dng_safe_arithmetic.h"
38
#include "dng_stream.h"
39
#include "dng_string_list.h"
40
#include "dng_tag_codes.h"
41
#include "dng_tag_values.h"
42
#include "dng_utils.h"
43
44
#if qDNGUseXMP
45
#include "dng_xmp.h"
46
#endif
47
48
#include "zlib.h"
49
50
#if qDNGUseLibJPEG
51
#include "dng_jpeglib.h"
52
#endif
53
54
/*****************************************************************************/
55
56
// Defines for testing DNG 1.2 features.
57
58
//#define qTestRowInterleave 2
59
60
//#define qTestSubTileBlockRows 2
61
//#define qTestSubTileBlockCols 2
62
63
/*****************************************************************************/
64
65
dng_resolution::dng_resolution ()
66
67
8.75k
  : fXResolution ()
68
8.75k
  , fYResolution ()
69
  
70
8.75k
  , fResolutionUnit (0)
71
  
72
8.75k
  {
73
  
74
8.75k
  }
75
76
/******************************************************************************/
77
78
static void SpoolAdobeData (dng_stream &stream,
79
              const dng_metadata *metadata,
80
              const dng_jpeg_preview *preview,
81
              const dng_memory_block *imageResources)
82
8.75k
  {
83
  
84
8.75k
  TempBigEndian tempEndian (stream);
85
  
86
  #if qDNGUseXMP
87
88
  if (metadata && metadata->GetXMP ())
89
    {
90
    
91
    bool marked = false;
92
    
93
    if (metadata->GetXMP ()->GetBoolean (XMP_NS_XAP_RIGHTS,
94
                       "Marked",
95
                       marked))
96
      {
97
      
98
      stream.Put_uint32 (DNG_CHAR4 ('8','B','I','M'));
99
      stream.Put_uint16 (1034);
100
      stream.Put_uint16 (0);
101
      
102
      stream.Put_uint32 (1);
103
      
104
      stream.Put_uint8 (marked ? 1 : 0);
105
      
106
      stream.Put_uint8 (0);
107
      
108
      }
109
      
110
    dng_string webStatement;
111
    
112
    if (metadata->GetXMP ()->GetString (XMP_NS_XAP_RIGHTS,
113
                      "WebStatement",
114
                      webStatement))
115
      {
116
      
117
      dng_memory_data buffer;
118
      
119
      uint32 size = webStatement.Get_SystemEncoding (buffer);
120
      
121
      if (size > 0)
122
        {
123
        
124
        stream.Put_uint32 (DNG_CHAR4 ('8','B','I','M'));
125
        stream.Put_uint16 (1035);
126
        stream.Put_uint16 (0);
127
        
128
        stream.Put_uint32 (size);
129
        
130
        stream.Put (buffer.Buffer (), size);
131
        
132
        if (size & 1)
133
          stream.Put_uint8 (0);
134
          
135
        }
136
      
137
      }
138
    
139
    }
140
  
141
  #endif
142
  
143
8.75k
  if (preview)
144
0
    {
145
    
146
0
    preview->SpoolAdobeThumbnail (stream);
147
        
148
0
    }
149
    
150
8.75k
  if (metadata && metadata->IPTCLength ())
151
41
    {
152
    
153
41
    dng_fingerprint iptcDigest = metadata->IPTCDigest ();
154
155
41
    if (iptcDigest.IsValid ())
156
41
      {
157
      
158
41
      stream.Put_uint32 (DNG_CHAR4 ('8','B','I','M'));
159
41
      stream.Put_uint16 (1061);
160
41
      stream.Put_uint16 (0);
161
      
162
41
      stream.Put_uint32 (16);
163
      
164
41
      stream.Put (iptcDigest.data, 16);
165
      
166
41
      }
167
      
168
41
    }
169
    
170
8.75k
  if (imageResources)
171
0
    {
172
    
173
0
    uint32 size = imageResources->LogicalSize ();
174
    
175
0
    stream.Put (imageResources->Buffer (), size);
176
    
177
0
    if (size & 1)
178
0
      stream.Put_uint8 (0);
179
    
180
0
    }
181
    
182
8.75k
  }
183
184
/******************************************************************************/
185
186
static dng_memory_block * BuildAdobeData (dng_host &host,
187
                      const dng_metadata *metadata,
188
                      const dng_jpeg_preview *preview,
189
                      const dng_memory_block *imageResources)
190
8.75k
  {
191
  
192
8.75k
  dng_memory_stream stream (host.Allocator ());
193
  
194
8.75k
  SpoolAdobeData (stream,
195
8.75k
          metadata,
196
8.75k
          preview,
197
8.75k
          imageResources);
198
          
199
8.75k
  return stream.AsMemoryBlock (host.Allocator ());
200
  
201
8.75k
  }
202
203
/*****************************************************************************/
204
205
tag_string::tag_string (uint16 code,
206
              const dng_string &s,
207
              bool forceASCII)
208
              
209
344k
  : tiff_tag (code, ttAscii, 0)
210
  
211
344k
  , fString (s)
212
  
213
344k
  {
214
      
215
344k
  if (forceASCII)
216
330k
    {
217
    
218
    // Metadata working group recommendation - go ahead
219
    // write UTF-8 into ASCII tag strings, rather than
220
    // actually force the strings to ASCII.  There is a matching
221
    // change on the reading side to assume UTF-8 if the string
222
    // contains a valid UTF-8 string.
223
    //
224
    // fString.ForceASCII ();
225
    
226
330k
    }
227
    
228
13.8k
  else if (!fString.IsASCII ())
229
2
    {
230
    
231
2
    fType = ttByte;
232
    
233
2
    }
234
    
235
344k
  fCount = fString.Length () + 1;
236
  
237
344k
  }
238
239
/*****************************************************************************/
240
241
void tag_string::Put (dng_stream &stream) const
242
13.5k
  {
243
  
244
13.5k
  stream.Put (fString.Get (), Size ());
245
  
246
13.5k
  }
247
248
/*****************************************************************************/
249
250
tag_encoded_text::tag_encoded_text (uint16 code,
251
                  const dng_string &text)
252
  
253
32.4k
  : tiff_tag (code, ttUndefined, 0)
254
  
255
32.4k
  , fText (text)
256
  
257
32.4k
  , fUTF16 ()
258
  
259
32.4k
  {
260
      
261
32.4k
  if (fText.IsASCII ())
262
32.4k
    {
263
  
264
32.4k
    fCount = 8 + fText.Length ();
265
    
266
32.4k
    }
267
    
268
53
  else
269
53
    {
270
    
271
53
    fCount = 8 + fText.Get_UTF16 (fUTF16) * 2;
272
    
273
53
    }
274
  
275
32.4k
  }
276
277
/*****************************************************************************/
278
279
void tag_encoded_text::Put (dng_stream &stream) const
280
302
  {
281
  
282
302
  if (fUTF16.Buffer ())
283
53
    {
284
    
285
53
    stream.Put ("UNICODE\000", 8);
286
    
287
53
    uint32 chars = (fCount - 8) >> 1;
288
    
289
53
    const uint16 *buf = fUTF16.Buffer_uint16 ();
290
    
291
1.43k
    for (uint32 j = 0; j < chars; j++)
292
1.38k
      {
293
      
294
1.38k
      stream.Put_uint16 (buf [j]);
295
      
296
1.38k
      }
297
    
298
53
    }
299
    
300
249
  else
301
249
    {
302
    
303
249
    stream.Put ("ASCII\000\000\000", 8);
304
    
305
249
    stream.Put (fText.Get (), fCount - 8);
306
    
307
249
    }
308
    
309
302
  }
310
  
311
/*****************************************************************************/
312
313
void tag_data_ptr::Put (dng_stream &stream) const
314
215k
  {
315
  
316
  // If we are swapping bytes, we need to swap with the right size
317
  // entries.
318
  
319
215k
  if (stream.SwapBytes ())
320
0
    {
321
  
322
0
    switch (Type ())
323
0
      {
324
      
325
      // Two byte entries.
326
      
327
0
      case ttShort:
328
0
      case ttSShort:
329
0
      case ttUnicode:
330
0
        {
331
        
332
0
        const uint16 *p = (const uint16 *) fData;
333
        
334
0
        uint32 entries = (Size () >> 1);
335
        
336
0
        for (uint32 j = 0; j < entries; j++)
337
0
          {
338
          
339
0
          stream.Put_uint16 (p [j]);
340
          
341
0
          }
342
        
343
0
        return;
344
        
345
0
        }
346
      
347
      // Four byte entries.
348
      
349
0
      case ttLong:
350
0
      case ttSLong:
351
0
      case ttRational:
352
0
      case ttSRational:
353
0
      case ttIFD:
354
0
      case ttFloat:
355
0
      case ttComplex:
356
0
        {
357
        
358
0
        const uint32 *p = (const uint32 *) fData;
359
        
360
0
        uint32 entries = (Size () >> 2);
361
        
362
0
        for (uint32 j = 0; j < entries; j++)
363
0
          {
364
          
365
0
          stream.Put_uint32 (p [j]);
366
          
367
0
          }
368
        
369
0
        return;
370
        
371
0
        }
372
        
373
      // Eight byte entries.
374
      
375
0
      case ttDouble:
376
0
        {
377
        
378
0
        const real64 *p = (const real64 *) fData;
379
        
380
0
        uint32 entries = (Size () >> 3);
381
        
382
0
        for (uint32 j = 0; j < entries; j++)
383
0
          {
384
          
385
0
          stream.Put_real64 (p [j]);
386
          
387
0
          }
388
        
389
0
        return;
390
        
391
0
        }
392
      
393
      // Entries don't need to be byte swapped.  Fall through
394
      // to non-byte swapped case.
395
        
396
0
      default:
397
0
        {
398
        
399
0
        break;
400
        
401
0
        }
402
403
0
      }
404
      
405
0
    }
406
    
407
  // Non-byte swapped case.
408
    
409
215k
  stream.Put (fData, Size ());
410
        
411
215k
  }
412
413
/******************************************************************************/
414
415
tag_matrix::tag_matrix (uint16 code,
416
              const dng_matrix &m)
417
             
418
6.60k
  : tag_srational_ptr (code, fEntry, m.Rows () * m.Cols ())
419
  
420
6.60k
  {
421
  
422
6.60k
  uint32 index = 0;
423
  
424
9.26k
  for (uint32 r = 0; r < m.Rows (); r++)
425
10.6k
    for (uint32 c = 0; c < m.Cols (); c++)
426
7.98k
      {
427
      
428
7.98k
      fEntry [index].Set_real64 (m [r] [c], 10000);
429
      
430
7.98k
      index++;
431
      
432
7.98k
      }
433
  
434
6.60k
  }
435
436
/******************************************************************************/
437
438
tag_icc_profile::tag_icc_profile (const void *profileData,
439
                  uint32 profileSize)
440
      
441
8.75k
  : tag_data_ptr (tcICCProfile, 
442
8.75k
            ttUndefined,
443
8.75k
            0,
444
8.75k
            NULL)
445
  
446
8.75k
  {
447
  
448
8.75k
  if (profileData && profileSize)
449
1.94k
    {
450
    
451
1.94k
    SetCount (profileSize);
452
1.94k
    SetData  (profileData);
453
    
454
1.94k
    }
455
  
456
8.75k
  }
457
      
458
/******************************************************************************/
459
460
void tag_cfa_pattern::Put (dng_stream &stream) const
461
0
  {
462
  
463
0
  stream.Put_uint16 ((uint16) fCols);
464
0
  stream.Put_uint16 ((uint16) fRows);
465
  
466
0
  for (uint32 col = 0; col < fCols; col++)
467
0
    for (uint32 row = 0; row < fRows; row++)
468
0
      {
469
      
470
0
      stream.Put_uint8 (fPattern [row * kMaxCFAPattern + col]);
471
      
472
0
      }
473
474
0
  }
475
476
/******************************************************************************/
477
478
tag_exif_date_time::tag_exif_date_time (uint16 code,
479
                          const dng_date_time &dt)
480
             
481
32.4k
  : tag_data_ptr (code, ttAscii, 20, fData)
482
  
483
32.4k
  {
484
  
485
32.4k
  if (dt.IsValid ())
486
811
    {
487
  
488
811
    sprintf (fData,
489
811
         "%04d:%02d:%02d %02d:%02d:%02d",
490
811
         (int) dt.fYear,
491
811
         (int) dt.fMonth,
492
811
         (int) dt.fDay,
493
811
         (int) dt.fHour,
494
811
         (int) dt.fMinute,
495
811
         (int) dt.fSecond);
496
         
497
811
    }
498
  
499
32.4k
  }
500
501
/******************************************************************************/
502
503
tag_iptc::tag_iptc (const void *data,
504
            uint32 length)
505
  
506
8.75k
  : tiff_tag (tcIPTC_NAA, ttLong, (length + 3) >> 2)
507
  
508
8.75k
  , fData   (data  )
509
8.75k
  , fLength (length)
510
  
511
8.75k
  {
512
  
513
8.75k
  }
514
515
/******************************************************************************/
516
517
void tag_iptc::Put (dng_stream &stream) const
518
41
  {
519
  
520
  // Note: For historical compatiblity reasons, the standard TIFF data 
521
  // type for IPTC data is ttLong, but without byte swapping.  This really
522
  // should be ttUndefined, but doing the right thing would break some
523
  // existing readers.
524
  
525
41
  stream.Put (fData, fLength);
526
  
527
  // Pad with zeros to get to long word boundary.
528
  
529
41
  uint32 extra = fCount * 4 - fLength;
530
  
531
94
  while (extra--)
532
53
    {
533
53
    stream.Put_uint8 (0);
534
53
    }
535
536
41
  }
537
538
/******************************************************************************/
539
540
tag_xmp::tag_xmp (const dng_xmp *xmp)
541
  
542
0
  : tag_uint8_ptr (tcXMP, NULL, 0)
543
  
544
0
  , fBuffer ()
545
  
546
0
  {
547
  
548
  #if qDNGUseXMP
549
  
550
  if (xmp)
551
    {
552
    
553
    fBuffer.Reset (xmp->Serialize (true));
554
    
555
    if (fBuffer.Get ())
556
      {
557
      
558
      SetData (fBuffer->Buffer_uint8 ());
559
      
560
      SetCount (fBuffer->LogicalSize ());
561
      
562
      }
563
    
564
    }
565
  
566
  #endif
567
  
568
0
  }
569
570
/******************************************************************************/
571
572
void dng_tiff_directory::Add (const tiff_tag *tag)
573
229k
  {
574
  
575
229k
  if (fEntries >= kMaxEntries)
576
0
    {
577
0
    ThrowProgramError ();
578
0
    }
579
  
580
  // Tags must be sorted in increasing order of tag code.
581
  
582
229k
  uint32 index = fEntries;
583
  
584
1.97M
  for (uint32 j = 0; j < fEntries; j++)
585
1.84M
    {
586
    
587
1.84M
    if (tag->Code () < fTag [j]->Code ())
588
104k
      {
589
104k
      index = j;
590
104k
      break;
591
104k
      }
592
    
593
1.84M
    }
594
    
595
733k
  for (uint32 k = fEntries; k > index; k--)
596
503k
    {
597
    
598
503k
    fTag [k] = fTag [k - 1];
599
    
600
503k
    }
601
  
602
229k
  fTag [index] = tag;
603
  
604
229k
  fEntries++;
605
  
606
229k
  }
607
    
608
/******************************************************************************/
609
610
uint32 dng_tiff_directory::Size () const
611
67.3k
  {
612
  
613
67.3k
  if (!fEntries) return 0;
614
  
615
18.4k
  uint32 size = fEntries * 12 + 6;
616
  
617
262k
  for (uint32 index = 0; index < fEntries; index++)
618
243k
    {
619
    
620
243k
    uint32 tagSize = fTag [index]->Size ();
621
    
622
243k
    if (tagSize > 4)
623
66.9k
      {
624
      
625
66.9k
      size += (tagSize + 1) & ~1;
626
      
627
66.9k
      }
628
    
629
243k
    }
630
    
631
18.4k
  return size;
632
  
633
67.3k
  }
634
    
635
/******************************************************************************/
636
637
void dng_tiff_directory::Put (dng_stream &stream,
638
                  OffsetsBase offsetsBase,
639
                  uint32 explicitBase) const
640
34.8k
  {
641
  
642
34.8k
  if (!fEntries) return;
643
  
644
14.9k
  uint32 index;
645
  
646
14.9k
  uint32 bigData = fEntries * 12 + 6;
647
  
648
14.9k
  if (offsetsBase == offsetsRelativeToStream)
649
14.9k
    bigData += (uint32) stream.Position ();
650
651
0
  else if (offsetsBase == offsetsRelativeToExplicitBase)
652
0
    bigData += explicitBase;
653
654
14.9k
  stream.Put_uint16 ((uint16) fEntries);
655
  
656
244k
  for (index = 0; index < fEntries; index++)
657
229k
    {
658
    
659
229k
    const tiff_tag &tag = *fTag [index];
660
    
661
229k
    stream.Put_uint16 (tag.Code  ());
662
229k
    stream.Put_uint16 (tag.Type  ());
663
229k
    stream.Put_uint32 (tag.Count ());
664
    
665
229k
    uint32 size = tag.Size ();
666
    
667
229k
    if (size <= 4)
668
168k
      {
669
      
670
168k
      tag.Put (stream);
671
      
672
312k
      while (size < 4)
673
143k
        {
674
143k
        stream.Put_uint8 (0);
675
143k
        size++;
676
143k
        }
677
        
678
168k
      }
679
      
680
60.4k
    else
681
60.4k
      {
682
      
683
60.4k
      stream.Put_uint32 (bigData);
684
      
685
60.4k
      bigData += (size + 1) & ~1;
686
            
687
60.4k
      }
688
    
689
229k
    }
690
    
691
14.9k
  stream.Put_uint32 (fChained);   // Next IFD offset
692
  
693
244k
  for (index = 0; index < fEntries; index++)
694
229k
    {
695
    
696
229k
    const tiff_tag &tag = *fTag [index];
697
    
698
229k
    uint32 size = tag.Size ();
699
    
700
229k
    if (size > 4)
701
60.4k
      {
702
      
703
60.4k
      tag.Put (stream);
704
      
705
60.4k
      if (size & 1)
706
7.93k
        stream.Put_uint8 (0);
707
      
708
60.4k
      }
709
    
710
229k
    }
711
  
712
14.9k
  }
713
714
/******************************************************************************/
715
716
dng_basic_tag_set::dng_basic_tag_set (dng_tiff_directory &directory,
717
                    const dng_ifd &info)
718
              
719
13.2k
  : fNewSubFileType (tcNewSubFileType, info.fNewSubFileType)
720
  
721
13.2k
  , fImageWidth  (tcImageWidth , info.fImageWidth )
722
13.2k
  , fImageLength (tcImageLength, info.fImageLength)
723
  
724
13.2k
  , fPhotoInterpretation (tcPhotometricInterpretation,
725
13.2k
                (uint16) info.fPhotometricInterpretation)
726
  
727
13.2k
  , fFillOrder (tcFillOrder, 1)
728
  
729
13.2k
  , fSamplesPerPixel (tcSamplesPerPixel, (uint16) info.fSamplesPerPixel)
730
  
731
13.2k
  , fBitsPerSample (tcBitsPerSample,
732
13.2k
            fBitsPerSampleData,
733
13.2k
            info.fSamplesPerPixel)
734
            
735
13.2k
  , fStrips (info.fUsesStrips)
736
            
737
13.2k
  , fTileWidth (tcTileWidth, info.fTileWidth)
738
  
739
13.2k
  , fTileLength (fStrips ? tcRowsPerStrip : tcTileLength, 
740
13.2k
           info.fTileLength)
741
  
742
13.2k
  , fTileInfoBuffer (info.TilesPerImage (), 8)
743
  
744
13.2k
  , fTileOffsetData (fTileInfoBuffer.Buffer_uint32 ())
745
  
746
13.2k
  , fTileOffsets (fStrips ? tcStripOffsets : tcTileOffsets,
747
13.2k
            fTileOffsetData,
748
13.2k
            info.TilesPerImage ())
749
             
750
13.2k
  , fTileByteCountData (fTileOffsetData + info.TilesPerImage ())
751
  
752
13.2k
  , fTileByteCounts (fStrips ? tcStripByteCounts : tcTileByteCounts,
753
13.2k
             fTileByteCountData,
754
13.2k
             info.TilesPerImage ())
755
              
756
13.2k
  , fPlanarConfiguration (tcPlanarConfiguration, pcInterleaved)
757
  
758
13.2k
  , fCompression (tcCompression, (uint16) info.fCompression)
759
13.2k
  , fPredictor   (tcPredictor  , (uint16) info.fPredictor  )
760
  
761
13.2k
  , fExtraSamples (tcExtraSamples,
762
13.2k
             fExtraSamplesData,
763
13.2k
             info.fExtraSamplesCount)
764
             
765
13.2k
  , fSampleFormat (tcSampleFormat,
766
13.2k
             fSampleFormatData,
767
13.2k
             info.fSamplesPerPixel)
768
             
769
13.2k
  , fRowInterleaveFactor (tcRowInterleaveFactor,
770
13.2k
                (uint16) info.fRowInterleaveFactor)
771
                
772
13.2k
  , fSubTileBlockSize (tcSubTileBlockSize,
773
13.2k
               fSubTileBlockSizeData,
774
13.2k
               2)
775
                 
776
13.2k
  {
777
  
778
13.2k
  uint32 j;
779
  
780
39.1k
  for (j = 0; j < info.fSamplesPerPixel; j++)
781
25.9k
    {
782
  
783
25.9k
    fBitsPerSampleData [j] = (uint16) info.fBitsPerSample [0];
784
    
785
25.9k
    }
786
  
787
13.2k
  directory.Add (&fNewSubFileType);
788
  
789
13.2k
  directory.Add (&fImageWidth);
790
13.2k
  directory.Add (&fImageLength);
791
  
792
13.2k
  directory.Add (&fPhotoInterpretation);
793
  
794
13.2k
  directory.Add (&fSamplesPerPixel);
795
  
796
13.2k
  directory.Add (&fBitsPerSample);
797
  
798
13.2k
  if (info.fBitsPerSample [0] !=  8 &&
799
13.2k
      info.fBitsPerSample [0] != 16 &&
800
13.2k
      info.fBitsPerSample [0] != 32)
801
0
    {
802
  
803
0
    directory.Add (&fFillOrder);
804
    
805
0
    }
806
    
807
13.2k
  if (!fStrips)
808
2.07k
    {
809
    
810
2.07k
    directory.Add (&fTileWidth);
811
  
812
2.07k
    }
813
814
13.2k
  directory.Add (&fTileLength);
815
  
816
13.2k
  directory.Add (&fTileOffsets);
817
13.2k
  directory.Add (&fTileByteCounts);
818
  
819
13.2k
  directory.Add (&fPlanarConfiguration);
820
  
821
13.2k
  directory.Add (&fCompression);
822
  
823
13.2k
  if (info.fPredictor != cpNullPredictor)
824
495
    {
825
    
826
495
    directory.Add (&fPredictor);
827
    
828
495
    }
829
    
830
13.2k
  if (info.fExtraSamplesCount != 0)
831
0
    {
832
    
833
0
    for (j = 0; j < info.fExtraSamplesCount; j++)
834
0
      {
835
0
      fExtraSamplesData [j] = (uint16) info.fExtraSamples [j];
836
0
      }
837
      
838
0
    directory.Add (&fExtraSamples);
839
    
840
0
    }
841
    
842
13.2k
  if (info.fSampleFormat [0] != sfUnsignedInteger)
843
1.97k
    {
844
    
845
4.57k
    for (j = 0; j < info.fSamplesPerPixel; j++)
846
2.59k
      {
847
2.59k
      fSampleFormatData [j] = (uint16) info.fSampleFormat [j];
848
2.59k
      }
849
      
850
1.97k
    directory.Add (&fSampleFormat);
851
    
852
1.97k
    }
853
    
854
13.2k
  if (info.fRowInterleaveFactor != 1)
855
0
    {
856
    
857
0
    directory.Add (&fRowInterleaveFactor);
858
    
859
0
    }
860
    
861
13.2k
  if (info.fSubTileBlockRows != 1 ||
862
13.2k
    info.fSubTileBlockCols != 1)
863
0
    {
864
    
865
0
    fSubTileBlockSizeData [0] = (uint16) info.fSubTileBlockRows;
866
0
    fSubTileBlockSizeData [1] = (uint16) info.fSubTileBlockCols;
867
    
868
0
    directory.Add (&fSubTileBlockSize);
869
    
870
0
    }
871
  
872
13.2k
  }
873
874
/******************************************************************************/
875
876
exif_tag_set::exif_tag_set (dng_tiff_directory &directory,
877
                const dng_exif &exif,
878
              bool makerNoteSafe,
879
              const void *makerNoteData,
880
              uint32 makerNoteLength,
881
                bool insideDNG)
882
883
10.8k
  : fExifIFD ()
884
10.8k
  , fGPSIFD  ()
885
  
886
10.8k
  , fExifLink (tcExifIFD, 0)
887
10.8k
  , fGPSLink  (tcGPSInfo, 0)
888
  
889
10.8k
  , fAddedExifLink (false)
890
10.8k
  , fAddedGPSLink  (false)
891
  
892
10.8k
  , fExifVersion (tcExifVersion, ttUndefined, 4, fExifVersionData)
893
  
894
10.8k
  , fExposureTime      (tcExposureTime     , exif.fExposureTime     )
895
10.8k
  , fShutterSpeedValue (tcShutterSpeedValue, exif.fShutterSpeedValue)
896
  
897
10.8k
  , fFNumber     (tcFNumber      , exif.fFNumber      )
898
10.8k
  , fApertureValue (tcApertureValue, exif.fApertureValue)
899
  
900
10.8k
  , fBrightnessValue (tcBrightnessValue, exif.fBrightnessValue)
901
  
902
10.8k
  , fExposureBiasValue (tcExposureBiasValue, exif.fExposureBiasValue)
903
  
904
10.8k
  , fMaxApertureValue (tcMaxApertureValue , exif.fMaxApertureValue)
905
  
906
10.8k
  , fSubjectDistance (tcSubjectDistance, exif.fSubjectDistance)
907
  
908
10.8k
  , fFocalLength (tcFocalLength, exif.fFocalLength)
909
  
910
  // Special case: the EXIF 2.2 standard represents ISO speed ratings with 2 bytes,
911
  // which cannot hold ISO speed ratings above 65535 (e.g., 102400). In these
912
  // cases, we write the maximum representable ISO speed rating value in the EXIF
913
  // tag, i.e., 65535.
914
915
10.8k
  , fISOSpeedRatings (tcISOSpeedRatings, 
916
10.8k
              (uint16) Min_uint32 (65535, 
917
10.8k
                         exif.fISOSpeedRatings [0]))
918
919
10.8k
  , fSensitivityType (tcSensitivityType, (uint16) exif.fSensitivityType)
920
921
10.8k
  , fStandardOutputSensitivity (tcStandardOutputSensitivity, exif.fStandardOutputSensitivity)
922
  
923
10.8k
  , fRecommendedExposureIndex (tcRecommendedExposureIndex, exif.fRecommendedExposureIndex)
924
925
10.8k
  , fISOSpeed (tcISOSpeed, exif.fISOSpeed)
926
927
10.8k
  , fISOSpeedLatitudeyyy (tcISOSpeedLatitudeyyy, exif.fISOSpeedLatitudeyyy)
928
929
10.8k
  , fISOSpeedLatitudezzz (tcISOSpeedLatitudezzz, exif.fISOSpeedLatitudezzz)
930
931
10.8k
  , fFlash (tcFlash, (uint16) exif.fFlash)
932
  
933
10.8k
  , fExposureProgram (tcExposureProgram, (uint16) exif.fExposureProgram)
934
  
935
10.8k
  , fMeteringMode (tcMeteringMode, (uint16) exif.fMeteringMode)
936
  
937
10.8k
  , fLightSource (tcLightSource, (uint16) exif.fLightSource)
938
  
939
10.8k
  , fSensingMethod (tcSensingMethodExif, (uint16) exif.fSensingMethod)
940
  
941
10.8k
  , fFocalLength35mm (tcFocalLengthIn35mmFilm, (uint16) exif.fFocalLengthIn35mmFilm)
942
  
943
10.8k
  , fFileSourceData ((uint8) exif.fFileSource)
944
10.8k
  , fFileSource     (tcFileSource, ttUndefined, 1, &fFileSourceData)
945
946
10.8k
  , fSceneTypeData ((uint8) exif.fSceneType)
947
10.8k
  , fSceneType     (tcSceneType, ttUndefined, 1, &fSceneTypeData)
948
  
949
10.8k
  , fCFAPattern (tcCFAPatternExif,
950
10.8k
           exif.fCFARepeatPatternRows,
951
10.8k
           exif.fCFARepeatPatternCols,
952
10.8k
           &exif.fCFAPattern [0] [0])
953
  
954
10.8k
  , fCustomRendered     (tcCustomRendered    , (uint16) exif.fCustomRendered    )
955
10.8k
  , fExposureMode       (tcExposureMode    , (uint16) exif.fExposureMode      )
956
10.8k
  , fWhiteBalance       (tcWhiteBalance    , (uint16) exif.fWhiteBalance      )
957
10.8k
  , fSceneCaptureType     (tcSceneCaptureType  , (uint16) exif.fSceneCaptureType    )
958
10.8k
  , fGainControl      (tcGainControl     , (uint16) exif.fGainControl     )
959
10.8k
  , fContrast         (tcContrast      , (uint16) exif.fContrast        )
960
10.8k
  , fSaturation       (tcSaturation      , (uint16) exif.fSaturation      )
961
10.8k
  , fSharpness        (tcSharpness       , (uint16) exif.fSharpness       )
962
10.8k
  , fSubjectDistanceRange (tcSubjectDistanceRange, (uint16) exif.fSubjectDistanceRange)
963
    
964
10.8k
  , fDigitalZoomRatio (tcDigitalZoomRatio, exif.fDigitalZoomRatio)
965
  
966
10.8k
  , fExposureIndex (tcExposureIndexExif, exif.fExposureIndex)
967
  
968
10.8k
  , fImageNumber (tcImageNumber, exif.fImageNumber)
969
  
970
10.8k
  , fSelfTimerMode (tcSelfTimerMode, (uint16) exif.fSelfTimerMode)
971
  
972
10.8k
  , fBatteryLevelA (tcBatteryLevel, exif.fBatteryLevelA)
973
10.8k
  , fBatteryLevelR (tcBatteryLevel, exif.fBatteryLevelR)
974
  
975
10.8k
  , fFocalPlaneXResolution (tcFocalPlaneXResolutionExif, exif.fFocalPlaneXResolution)
976
10.8k
  , fFocalPlaneYResolution (tcFocalPlaneYResolutionExif, exif.fFocalPlaneYResolution)
977
  
978
10.8k
  , fFocalPlaneResolutionUnit (tcFocalPlaneResolutionUnitExif, (uint16) exif.fFocalPlaneResolutionUnit)
979
  
980
10.8k
  , fSubjectArea (tcSubjectArea, fSubjectAreaData, exif.fSubjectAreaCount)
981
982
10.8k
  , fLensInfo (tcLensInfo, fLensInfoData, 4)
983
  
984
10.8k
  , fDateTime      (tcDateTime       , exif.fDateTime         .DateTime ())
985
10.8k
  , fDateTimeOriginal  (tcDateTimeOriginal , exif.fDateTimeOriginal .DateTime ())
986
10.8k
  , fDateTimeDigitized (tcDateTimeDigitized, exif.fDateTimeDigitized.DateTime ())
987
  
988
10.8k
  , fSubsecTime      (tcSubsecTime,      exif.fDateTime         .Subseconds ())
989
10.8k
  , fSubsecTimeOriginal  (tcSubsecTimeOriginal,  exif.fDateTimeOriginal .Subseconds ())
990
10.8k
  , fSubsecTimeDigitized (tcSubsecTimeDigitized, exif.fDateTimeDigitized.Subseconds ())
991
  
992
10.8k
  , fMake (tcMake, exif.fMake)
993
994
10.8k
  , fModel (tcModel, exif.fModel)
995
  
996
10.8k
  , fArtist (tcArtist, exif.fArtist)
997
  
998
10.8k
  , fSoftware (tcSoftware, exif.fSoftware)
999
  
1000
10.8k
  , fCopyright (tcCopyright, exif.fCopyright)
1001
  
1002
10.8k
  , fMakerNoteSafety (tcMakerNoteSafety, makerNoteSafe ? 1 : 0)
1003
  
1004
10.8k
  , fMakerNote (tcMakerNote, ttUndefined, makerNoteLength, makerNoteData)
1005
          
1006
10.8k
  , fImageDescription (tcImageDescription, exif.fImageDescription)
1007
  
1008
10.8k
  , fSerialNumber (tcCameraSerialNumber, exif.fCameraSerialNumber)
1009
  
1010
10.8k
  , fUserComment (tcUserComment, exif.fUserComment)
1011
  
1012
10.8k
  , fImageUniqueID (tcImageUniqueID, ttAscii, 33, fImageUniqueIDData)
1013
1014
  // EXIF 2.3 tags.
1015
1016
10.8k
  , fCameraOwnerName   (tcCameraOwnerNameExif,    exif.fOwnerName     )
1017
10.8k
  , fBodySerialNumber  (tcCameraSerialNumberExif, exif.fCameraSerialNumber)
1018
10.8k
  , fLensSpecification (tcLensSpecificationExif,  fLensInfoData, 4      )
1019
10.8k
  , fLensMake      (tcLensMakeExif,       exif.fLensMake      )
1020
10.8k
  , fLensModel       (tcLensModelExif,      exif.fLensName      )
1021
10.8k
  , fLensSerialNumber  (tcLensSerialNumberExif,   exif.fLensSerialNumber  )
1022
1023
10.8k
  , fGPSVersionID (tcGPSVersionID, fGPSVersionData, 4)
1024
  
1025
10.8k
  , fGPSLatitudeRef (tcGPSLatitudeRef, exif.fGPSLatitudeRef)
1026
10.8k
  , fGPSLatitude    (tcGPSLatitude,    exif.fGPSLatitude, 3)
1027
  
1028
10.8k
  , fGPSLongitudeRef (tcGPSLongitudeRef, exif.fGPSLongitudeRef)
1029
10.8k
  , fGPSLongitude    (tcGPSLongitude,    exif.fGPSLongitude, 3)
1030
  
1031
10.8k
  , fGPSAltitudeRef (tcGPSAltitudeRef, (uint8) exif.fGPSAltitudeRef)
1032
10.8k
  , fGPSAltitude    (tcGPSAltitude,            exif.fGPSAltitude   )
1033
  
1034
10.8k
  , fGPSTimeStamp (tcGPSTimeStamp, exif.fGPSTimeStamp, 3)
1035
    
1036
10.8k
  , fGPSSatellites  (tcGPSSatellites , exif.fGPSSatellites )
1037
10.8k
  , fGPSStatus      (tcGPSStatus     , exif.fGPSStatus     )
1038
10.8k
  , fGPSMeasureMode (tcGPSMeasureMode, exif.fGPSMeasureMode)
1039
  
1040
10.8k
  , fGPSDOP (tcGPSDOP, exif.fGPSDOP)
1041
    
1042
10.8k
  , fGPSSpeedRef (tcGPSSpeedRef, exif.fGPSSpeedRef)
1043
10.8k
  , fGPSSpeed    (tcGPSSpeed   , exif.fGPSSpeed   )
1044
    
1045
10.8k
  , fGPSTrackRef (tcGPSTrackRef, exif.fGPSTrackRef)
1046
10.8k
  , fGPSTrack    (tcGPSTrack   , exif.fGPSTrack   )
1047
    
1048
10.8k
  , fGPSImgDirectionRef (tcGPSImgDirectionRef, exif.fGPSImgDirectionRef)
1049
10.8k
  , fGPSImgDirection    (tcGPSImgDirection   , exif.fGPSImgDirection   )
1050
  
1051
10.8k
  , fGPSMapDatum (tcGPSMapDatum, exif.fGPSMapDatum)
1052
    
1053
10.8k
  , fGPSDestLatitudeRef (tcGPSDestLatitudeRef, exif.fGPSDestLatitudeRef)
1054
10.8k
  , fGPSDestLatitude    (tcGPSDestLatitude,    exif.fGPSDestLatitude, 3)
1055
  
1056
10.8k
  , fGPSDestLongitudeRef (tcGPSDestLongitudeRef, exif.fGPSDestLongitudeRef)
1057
10.8k
  , fGPSDestLongitude    (tcGPSDestLongitude,    exif.fGPSDestLongitude, 3)
1058
  
1059
10.8k
  , fGPSDestBearingRef (tcGPSDestBearingRef, exif.fGPSDestBearingRef)
1060
10.8k
  , fGPSDestBearing    (tcGPSDestBearing   , exif.fGPSDestBearing   )
1061
    
1062
10.8k
  , fGPSDestDistanceRef (tcGPSDestDistanceRef, exif.fGPSDestDistanceRef)
1063
10.8k
  , fGPSDestDistance    (tcGPSDestDistance   , exif.fGPSDestDistance   )
1064
    
1065
10.8k
  , fGPSProcessingMethod (tcGPSProcessingMethod, exif.fGPSProcessingMethod)
1066
10.8k
  , fGPSAreaInformation  (tcGPSAreaInformation , exif.fGPSAreaInformation )
1067
  
1068
10.8k
  , fGPSDateStamp (tcGPSDateStamp, exif.fGPSDateStamp)
1069
  
1070
10.8k
  , fGPSDifferential (tcGPSDifferential, (uint16) exif.fGPSDifferential)
1071
    
1072
10.8k
  , fGPSHPositioningError (tcGPSHPositioningError, exif.fGPSHPositioningError)
1073
    
1074
10.8k
  {
1075
  
1076
10.8k
  if (exif.fExifVersion)
1077
48
    {
1078
    
1079
48
    fExifVersionData [0] = (uint8) (exif.fExifVersion >> 24);
1080
48
    fExifVersionData [1] = (uint8) (exif.fExifVersion >> 16);
1081
48
    fExifVersionData [2] = (uint8) (exif.fExifVersion >>  8);
1082
48
    fExifVersionData [3] = (uint8) (exif.fExifVersion      );
1083
    
1084
48
    fExifIFD.Add (&fExifVersion);
1085
1086
48
    }
1087
  
1088
10.8k
  if (exif.fExposureTime.IsValid ())
1089
42
    {
1090
42
    fExifIFD.Add (&fExposureTime);
1091
42
    }
1092
    
1093
10.8k
  if (exif.fShutterSpeedValue.IsValid ())
1094
42
    {
1095
42
    fExifIFD.Add (&fShutterSpeedValue);
1096
42
    }
1097
    
1098
10.8k
  if (exif.fFNumber.IsValid ())
1099
50
    {
1100
50
    fExifIFD.Add (&fFNumber);
1101
50
    }
1102
    
1103
10.8k
  if (exif.fApertureValue.IsValid ())
1104
39
    {
1105
39
    fExifIFD.Add (&fApertureValue);
1106
39
    }
1107
    
1108
10.8k
  if (exif.fBrightnessValue.IsValid ())
1109
372
    {
1110
372
    fExifIFD.Add (&fBrightnessValue);
1111
372
    }
1112
    
1113
10.8k
  if (exif.fExposureBiasValue.IsValid ())
1114
185
    {
1115
185
    fExifIFD.Add (&fExposureBiasValue);
1116
185
    }
1117
    
1118
10.8k
  if (exif.fMaxApertureValue.IsValid ())
1119
429
    {
1120
429
    fExifIFD.Add (&fMaxApertureValue);
1121
429
    }
1122
    
1123
10.8k
  if (exif.fSubjectDistance.IsValid ())
1124
30
    {
1125
30
    fExifIFD.Add (&fSubjectDistance);
1126
30
    }
1127
    
1128
10.8k
  if (exif.fFocalLength.IsValid ())
1129
625
    {
1130
625
    fExifIFD.Add (&fFocalLength);
1131
625
    }
1132
  
1133
10.8k
  if (exif.fISOSpeedRatings [0] != 0)
1134
461
    {
1135
461
    fExifIFD.Add (&fISOSpeedRatings);
1136
461
    }
1137
    
1138
10.8k
  if (exif.fFlash <= 0x0FFFF)
1139
439
    {
1140
439
    fExifIFD.Add (&fFlash);
1141
439
    }
1142
    
1143
10.8k
  if (exif.fExposureProgram <= 0x0FFFF)
1144
30
    {
1145
30
    fExifIFD.Add (&fExposureProgram);
1146
30
    }
1147
    
1148
10.8k
  if (exif.fMeteringMode <= 0x0FFFF)
1149
491
    {
1150
491
    fExifIFD.Add (&fMeteringMode);
1151
491
    }
1152
    
1153
10.8k
  if (exif.fLightSource <= 0x0FFFF)
1154
641
    {
1155
641
    fExifIFD.Add (&fLightSource);
1156
641
    }
1157
    
1158
10.8k
  if (exif.fSensingMethod <= 0x0FFFF)
1159
238
    {
1160
238
    fExifIFD.Add (&fSensingMethod);
1161
238
    }
1162
    
1163
10.8k
  if (exif.fFocalLengthIn35mmFilm != 0)
1164
34
    {
1165
34
    fExifIFD.Add (&fFocalLength35mm);
1166
34
    }
1167
    
1168
10.8k
  if (exif.fFileSource <= 0x0FF)
1169
74
    {
1170
74
    fExifIFD.Add (&fFileSource);
1171
74
    }
1172
    
1173
10.8k
  if (exif.fSceneType <= 0x0FF)
1174
290
    {
1175
290
    fExifIFD.Add (&fSceneType);
1176
290
    }
1177
    
1178
10.8k
  if (exif.fCFARepeatPatternRows &&
1179
10.8k
      exif.fCFARepeatPatternCols)
1180
0
    {
1181
0
    fExifIFD.Add (&fCFAPattern);
1182
0
    }
1183
    
1184
10.8k
  if (exif.fCustomRendered <= 0x0FFFF)
1185
79
    {
1186
79
    fExifIFD.Add (&fCustomRendered);
1187
79
    }
1188
    
1189
10.8k
  if (exif.fExposureMode <= 0x0FFFF)
1190
622
    {
1191
622
    fExifIFD.Add (&fExposureMode);
1192
622
    }
1193
    
1194
10.8k
  if (exif.fWhiteBalance <= 0x0FFFF)
1195
300
    {
1196
300
    fExifIFD.Add (&fWhiteBalance);
1197
300
    }
1198
    
1199
10.8k
  if (exif.fSceneCaptureType <= 0x0FFFF)
1200
100
    {
1201
100
    fExifIFD.Add (&fSceneCaptureType);
1202
100
    }
1203
    
1204
10.8k
  if (exif.fGainControl <= 0x0FFFF)
1205
66
    {
1206
66
    fExifIFD.Add (&fGainControl);
1207
66
    }
1208
    
1209
10.8k
  if (exif.fContrast <= 0x0FFFF)
1210
33
    {
1211
33
    fExifIFD.Add (&fContrast);
1212
33
    }
1213
    
1214
10.8k
  if (exif.fSaturation <= 0x0FFFF)
1215
45
    {
1216
45
    fExifIFD.Add (&fSaturation);
1217
45
    }
1218
    
1219
10.8k
  if (exif.fSharpness <= 0x0FFFF)
1220
39
    {
1221
39
    fExifIFD.Add (&fSharpness);
1222
39
    }
1223
    
1224
10.8k
  if (exif.fSubjectDistanceRange <= 0x0FFFF)
1225
2
    {
1226
2
    fExifIFD.Add (&fSubjectDistanceRange);
1227
2
    }
1228
    
1229
10.8k
  if (exif.fDigitalZoomRatio.IsValid ())
1230
126
    {
1231
126
    fExifIFD.Add (&fDigitalZoomRatio);
1232
126
    }
1233
    
1234
10.8k
  if (exif.fExposureIndex.IsValid ())
1235
590
    {
1236
590
    fExifIFD.Add (&fExposureIndex);
1237
590
    }
1238
    
1239
10.8k
  if (insideDNG)  // TIFF-EP only tags
1240
2.07k
    {
1241
    
1242
2.07k
    if (exif.fImageNumber != 0xFFFFFFFF)
1243
19
      {
1244
19
      directory.Add (&fImageNumber);
1245
19
      }
1246
      
1247
2.07k
    if (exif.fSelfTimerMode <= 0x0FFFF)
1248
0
      {
1249
0
      directory.Add (&fSelfTimerMode);
1250
0
      }
1251
      
1252
2.07k
    if (exif.fBatteryLevelA.NotEmpty ())
1253
6
      {
1254
6
      directory.Add (&fBatteryLevelA);
1255
6
      }
1256
      
1257
2.06k
    else if (exif.fBatteryLevelR.IsValid ())
1258
64
      {
1259
64
      directory.Add (&fBatteryLevelR);
1260
64
      }
1261
    
1262
2.07k
    }
1263
    
1264
10.8k
  if (exif.fFocalPlaneXResolution.IsValid ())
1265
173
    {
1266
173
    fExifIFD.Add (&fFocalPlaneXResolution);
1267
173
    }
1268
  
1269
10.8k
  if (exif.fFocalPlaneYResolution.IsValid ())
1270
251
    {
1271
251
    fExifIFD.Add (&fFocalPlaneYResolution);
1272
251
    }
1273
  
1274
10.8k
  if (exif.fFocalPlaneResolutionUnit <= 0x0FFFF)
1275
20
    {
1276
20
    fExifIFD.Add (&fFocalPlaneResolutionUnit);
1277
20
    }
1278
    
1279
10.8k
  if (exif.fSubjectAreaCount)
1280
13
    {
1281
    
1282
13
    fSubjectAreaData [0] = (uint16) exif.fSubjectArea [0];
1283
13
    fSubjectAreaData [1] = (uint16) exif.fSubjectArea [1];
1284
13
    fSubjectAreaData [2] = (uint16) exif.fSubjectArea [2];
1285
13
    fSubjectAreaData [3] = (uint16) exif.fSubjectArea [3];
1286
    
1287
13
    fExifIFD.Add (&fSubjectArea);
1288
    
1289
13
    }
1290
  
1291
10.8k
  if (exif.fLensInfo [0].IsValid () &&
1292
10.8k
    exif.fLensInfo [1].IsValid ())
1293
62
    {
1294
    
1295
62
    fLensInfoData [0] = exif.fLensInfo [0];
1296
62
    fLensInfoData [1] = exif.fLensInfo [1];
1297
62
    fLensInfoData [2] = exif.fLensInfo [2];
1298
62
    fLensInfoData [3] = exif.fLensInfo [3];
1299
1300
62
    if (insideDNG)
1301
31
      {
1302
31
      directory.Add (&fLensInfo);
1303
31
      }
1304
    
1305
62
    }
1306
    
1307
10.8k
  if (exif.fDateTime.IsValid ())
1308
811
    {
1309
    
1310
811
    directory.Add (&fDateTime);
1311
    
1312
811
    if (exif.fDateTime.Subseconds ().NotEmpty ())
1313
11
      {
1314
11
      fExifIFD.Add (&fSubsecTime);
1315
11
      }
1316
    
1317
811
    }
1318
    
1319
10.8k
  if (exif.fDateTimeOriginal.IsValid ())
1320
0
    {
1321
    
1322
0
    fExifIFD.Add (&fDateTimeOriginal);
1323
    
1324
0
    if (exif.fDateTimeOriginal.Subseconds ().NotEmpty ())
1325
0
      {
1326
0
      fExifIFD.Add (&fSubsecTimeOriginal);
1327
0
      }
1328
    
1329
0
    }
1330
    
1331
10.8k
  if (exif.fDateTimeDigitized.IsValid ())
1332
0
    {
1333
    
1334
0
    fExifIFD.Add (&fDateTimeDigitized);
1335
    
1336
0
    if (exif.fDateTimeDigitized.Subseconds ().NotEmpty ())
1337
0
      {
1338
0
      fExifIFD.Add (&fSubsecTimeDigitized);
1339
0
      }
1340
    
1341
0
    }
1342
    
1343
10.8k
  if (exif.fMake.NotEmpty ())
1344
13
    {
1345
13
    directory.Add (&fMake);
1346
13
    }
1347
    
1348
10.8k
  if (exif.fModel.NotEmpty ())
1349
27
    {
1350
27
    directory.Add (&fModel);
1351
27
    }
1352
    
1353
10.8k
  if (exif.fArtist.NotEmpty ())
1354
30
    {
1355
30
    directory.Add (&fArtist);
1356
30
    }
1357
  
1358
10.8k
  if (exif.fSoftware.NotEmpty ())
1359
528
    {
1360
528
    directory.Add (&fSoftware);
1361
528
    }
1362
  
1363
10.8k
  if (exif.fCopyright.NotEmpty ())
1364
6
    {
1365
6
    directory.Add (&fCopyright);
1366
6
    }
1367
  
1368
10.8k
  if (exif.fImageDescription.NotEmpty ())
1369
20
    {
1370
20
    directory.Add (&fImageDescription);
1371
20
    }
1372
  
1373
10.8k
  if (exif.fCameraSerialNumber.NotEmpty () && insideDNG)
1374
260
    {
1375
260
    directory.Add (&fSerialNumber);
1376
260
    }
1377
    
1378
10.8k
  if (makerNoteSafe && makerNoteData)
1379
0
    {
1380
    
1381
0
    directory.Add (&fMakerNoteSafety);
1382
    
1383
0
    fExifIFD.Add (&fMakerNote);
1384
    
1385
0
    }
1386
    
1387
10.8k
  if (exif.fUserComment.NotEmpty ())
1388
302
    {
1389
302
    fExifIFD.Add (&fUserComment);
1390
302
    }
1391
    
1392
10.8k
  if (exif.fImageUniqueID.IsValid ())
1393
0
    {
1394
    
1395
0
    for (uint32 j = 0; j < 16; j++)
1396
0
      {
1397
      
1398
0
      sprintf (fImageUniqueIDData + j * 2,
1399
0
           "%02X",
1400
0
           (unsigned) exif.fImageUniqueID.data [j]);
1401
           
1402
0
      }
1403
    
1404
0
    fExifIFD.Add (&fImageUniqueID);
1405
    
1406
0
    }
1407
1408
10.8k
  if (exif.AtLeastVersion0230 ())
1409
46
    {
1410
1411
46
    if (exif.fSensitivityType != 0)
1412
0
      {
1413
      
1414
0
      fExifIFD.Add (&fSensitivityType);
1415
      
1416
0
      }
1417
1418
    // Sensitivity tags. Do not write these extra tags unless the SensitivityType
1419
    // and PhotographicSensitivity (i.e., ISOSpeedRatings) values are valid.
1420
1421
46
    if (exif.fSensitivityType   != 0 &&
1422
46
      exif.fISOSpeedRatings [0] != 0)
1423
0
      {
1424
1425
      // Standard Output Sensitivity (SOS).
1426
1427
0
      if (exif.fStandardOutputSensitivity != 0)
1428
0
        {
1429
0
        fExifIFD.Add (&fStandardOutputSensitivity); 
1430
0
        }
1431
1432
      // Recommended Exposure Index (REI).
1433
1434
0
      if (exif.fRecommendedExposureIndex != 0)
1435
0
        {
1436
0
        fExifIFD.Add (&fRecommendedExposureIndex);
1437
0
        }
1438
1439
      // ISO Speed.
1440
1441
0
      if (exif.fISOSpeed != 0)
1442
0
        {
1443
1444
0
        fExifIFD.Add (&fISOSpeed);
1445
1446
0
        if (exif.fISOSpeedLatitudeyyy != 0 &&
1447
0
          exif.fISOSpeedLatitudezzz != 0)
1448
0
          {
1449
            
1450
0
          fExifIFD.Add (&fISOSpeedLatitudeyyy);
1451
0
          fExifIFD.Add (&fISOSpeedLatitudezzz);
1452
            
1453
0
          }
1454
1455
0
        }
1456
1457
0
      }
1458
    
1459
46
    if (exif.fOwnerName.NotEmpty ())
1460
0
      {
1461
0
      fExifIFD.Add (&fCameraOwnerName);
1462
0
      }
1463
    
1464
46
    if (exif.fCameraSerialNumber.NotEmpty ())
1465
2
      {
1466
2
      fExifIFD.Add (&fBodySerialNumber);
1467
2
      }
1468
1469
46
    if (exif.fLensInfo [0].IsValid () &&
1470
46
      exif.fLensInfo [1].IsValid ())
1471
0
      {
1472
0
      fExifIFD.Add (&fLensSpecification);
1473
0
      }
1474
    
1475
46
    if (exif.fLensMake.NotEmpty ())
1476
0
      {
1477
0
      fExifIFD.Add (&fLensMake);
1478
0
      }
1479
    
1480
46
    if (exif.fLensName.NotEmpty ())
1481
2
      {
1482
2
      fExifIFD.Add (&fLensModel);
1483
2
      }
1484
    
1485
46
    if (exif.fLensSerialNumber.NotEmpty ())
1486
0
      {
1487
0
      fExifIFD.Add (&fLensSerialNumber);
1488
0
      }
1489
    
1490
46
    }
1491
    
1492
10.8k
  if (exif.fGPSVersionID)
1493
0
    {
1494
    
1495
0
    fGPSVersionData [0] = (uint8) (exif.fGPSVersionID >> 24);
1496
0
    fGPSVersionData [1] = (uint8) (exif.fGPSVersionID >> 16);
1497
0
    fGPSVersionData [2] = (uint8) (exif.fGPSVersionID >>  8);
1498
0
    fGPSVersionData [3] = (uint8) (exif.fGPSVersionID      );
1499
    
1500
0
    fGPSIFD.Add (&fGPSVersionID);
1501
    
1502
0
    }
1503
    
1504
10.8k
  if (exif.fGPSLatitudeRef.NotEmpty () &&
1505
10.8k
    exif.fGPSLatitude [0].IsValid ())
1506
0
    {
1507
0
    fGPSIFD.Add (&fGPSLatitudeRef);
1508
0
    fGPSIFD.Add (&fGPSLatitude   );
1509
0
    }
1510
    
1511
10.8k
  if (exif.fGPSLongitudeRef.NotEmpty () &&
1512
10.8k
    exif.fGPSLongitude [0].IsValid ())
1513
0
    {
1514
0
    fGPSIFD.Add (&fGPSLongitudeRef);
1515
0
    fGPSIFD.Add (&fGPSLongitude   );
1516
0
    }
1517
    
1518
10.8k
  if (exif.fGPSAltitudeRef <= 0x0FF)
1519
0
    {
1520
0
    fGPSIFD.Add (&fGPSAltitudeRef);
1521
0
    }
1522
    
1523
10.8k
  if (exif.fGPSAltitude.IsValid ())
1524
0
    {
1525
0
    fGPSIFD.Add (&fGPSAltitude);
1526
0
    }
1527
    
1528
10.8k
  if (exif.fGPSTimeStamp [0].IsValid ())
1529
0
    {
1530
0
    fGPSIFD.Add (&fGPSTimeStamp);
1531
0
    }
1532
    
1533
10.8k
  if (exif.fGPSSatellites.NotEmpty ())
1534
0
    {
1535
0
    fGPSIFD.Add (&fGPSSatellites);
1536
0
    }
1537
    
1538
10.8k
  if (exif.fGPSStatus.NotEmpty ())
1539
0
    {
1540
0
    fGPSIFD.Add (&fGPSStatus);
1541
0
    }
1542
    
1543
10.8k
  if (exif.fGPSMeasureMode.NotEmpty ())
1544
0
    {
1545
0
    fGPSIFD.Add (&fGPSMeasureMode);
1546
0
    }
1547
    
1548
10.8k
  if (exif.fGPSDOP.IsValid ())
1549
0
    {
1550
0
    fGPSIFD.Add (&fGPSDOP);
1551
0
    }
1552
    
1553
10.8k
  if (exif.fGPSSpeedRef.NotEmpty ())
1554
0
    {
1555
0
    fGPSIFD.Add (&fGPSSpeedRef);
1556
0
    }
1557
    
1558
10.8k
  if (exif.fGPSSpeed.IsValid ())
1559
0
    {
1560
0
    fGPSIFD.Add (&fGPSSpeed);
1561
0
    }
1562
    
1563
10.8k
  if (exif.fGPSTrackRef.NotEmpty ())
1564
0
    {
1565
0
    fGPSIFD.Add (&fGPSTrackRef);
1566
0
    }
1567
    
1568
10.8k
  if (exif.fGPSTrack.IsValid ())
1569
0
    {
1570
0
    fGPSIFD.Add (&fGPSTrack);
1571
0
    }
1572
    
1573
10.8k
  if (exif.fGPSImgDirectionRef.NotEmpty ())
1574
0
    {
1575
0
    fGPSIFD.Add (&fGPSImgDirectionRef);
1576
0
    }
1577
    
1578
10.8k
  if (exif.fGPSImgDirection.IsValid ())
1579
0
    {
1580
0
    fGPSIFD.Add (&fGPSImgDirection);
1581
0
    }
1582
1583
10.8k
  if (exif.fGPSMapDatum.NotEmpty ())
1584
0
    {
1585
0
    fGPSIFD.Add (&fGPSMapDatum);
1586
0
    }
1587
    
1588
10.8k
  if (exif.fGPSDestLatitudeRef.NotEmpty () &&
1589
10.8k
    exif.fGPSDestLatitude [0].IsValid ())
1590
0
    {
1591
0
    fGPSIFD.Add (&fGPSDestLatitudeRef);
1592
0
    fGPSIFD.Add (&fGPSDestLatitude   );
1593
0
    }
1594
    
1595
10.8k
  if (exif.fGPSDestLongitudeRef.NotEmpty () &&
1596
10.8k
    exif.fGPSDestLongitude [0].IsValid ())
1597
0
    {
1598
0
    fGPSIFD.Add (&fGPSDestLongitudeRef);
1599
0
    fGPSIFD.Add (&fGPSDestLongitude   );
1600
0
    }
1601
    
1602
10.8k
  if (exif.fGPSDestBearingRef.NotEmpty ())
1603
0
    {
1604
0
    fGPSIFD.Add (&fGPSDestBearingRef);
1605
0
    }
1606
    
1607
10.8k
  if (exif.fGPSDestBearing.IsValid ())
1608
0
    {
1609
0
    fGPSIFD.Add (&fGPSDestBearing);
1610
0
    }
1611
1612
10.8k
  if (exif.fGPSDestDistanceRef.NotEmpty ())
1613
0
    {
1614
0
    fGPSIFD.Add (&fGPSDestDistanceRef);
1615
0
    }
1616
    
1617
10.8k
  if (exif.fGPSDestDistance.IsValid ())
1618
0
    {
1619
0
    fGPSIFD.Add (&fGPSDestDistance);
1620
0
    }
1621
    
1622
10.8k
  if (exif.fGPSProcessingMethod.NotEmpty ())
1623
0
    {
1624
0
    fGPSIFD.Add (&fGPSProcessingMethod);
1625
0
    }
1626
1627
10.8k
  if (exif.fGPSAreaInformation.NotEmpty ())
1628
0
    {
1629
0
    fGPSIFD.Add (&fGPSAreaInformation);
1630
0
    }
1631
  
1632
10.8k
  if (exif.fGPSDateStamp.NotEmpty ())
1633
0
    {
1634
0
    fGPSIFD.Add (&fGPSDateStamp);
1635
0
    }
1636
  
1637
10.8k
  if (exif.fGPSDifferential <= 0x0FFFF)
1638
0
    {
1639
0
    fGPSIFD.Add (&fGPSDifferential);
1640
0
    }
1641
1642
10.8k
  if (exif.AtLeastVersion0230 ())
1643
46
    {
1644
    
1645
46
    if (exif.fGPSHPositioningError.IsValid ())
1646
0
      {
1647
0
      fGPSIFD.Add (&fGPSHPositioningError);
1648
0
      }
1649
1650
46
    }
1651
    
1652
10.8k
  AddLinks (directory);
1653
  
1654
10.8k
  }
1655
1656
/******************************************************************************/
1657
1658
void exif_tag_set::AddLinks (dng_tiff_directory &directory)
1659
10.8k
  {
1660
  
1661
10.8k
  if (fExifIFD.Size () != 0 && !fAddedExifLink)
1662
1.73k
    {
1663
    
1664
1.73k
    directory.Add (&fExifLink);
1665
    
1666
1.73k
    fAddedExifLink = true;
1667
    
1668
1.73k
    }
1669
  
1670
10.8k
  if (fGPSIFD.Size () != 0 && !fAddedGPSLink)
1671
0
    {
1672
    
1673
0
    directory.Add (&fGPSLink);
1674
    
1675
0
    fAddedGPSLink = true;
1676
    
1677
0
    }
1678
  
1679
10.8k
  }
1680
  
1681
/******************************************************************************/
1682
1683
class range_tag_set
1684
  {
1685
  
1686
  private:
1687
  
1688
    uint32 fActiveAreaData [4];
1689
  
1690
    tag_uint32_ptr fActiveArea;
1691
  
1692
    uint32 fMaskedAreaData [kMaxMaskedAreas * 4];
1693
  
1694
    tag_uint32_ptr fMaskedAreas;
1695
                
1696
    tag_uint16_ptr fLinearizationTable;
1697
  
1698
    uint16 fBlackLevelRepeatDimData [2];
1699
    
1700
    tag_uint16_ptr fBlackLevelRepeatDim;
1701
    
1702
    dng_urational fBlackLevelData [kMaxBlackPattern *
1703
                       kMaxBlackPattern *
1704
                       kMaxSamplesPerPixel];
1705
    
1706
    tag_urational_ptr fBlackLevel;
1707
    
1708
    dng_memory_data fBlackLevelDeltaHData;
1709
    dng_memory_data fBlackLevelDeltaVData;
1710
    
1711
    tag_srational_ptr fBlackLevelDeltaH;
1712
    tag_srational_ptr fBlackLevelDeltaV;
1713
    
1714
    uint16 fWhiteLevelData16 [kMaxSamplesPerPixel];
1715
    uint32 fWhiteLevelData32 [kMaxSamplesPerPixel];
1716
    
1717
    tag_uint16_ptr fWhiteLevel16;
1718
    tag_uint32_ptr fWhiteLevel32;
1719
    
1720
  public:
1721
  
1722
    range_tag_set (dng_tiff_directory &directory,
1723
               const dng_negative &negative);
1724
    
1725
  };
1726
  
1727
/******************************************************************************/
1728
1729
range_tag_set::range_tag_set (dng_tiff_directory &directory,
1730
                    const dng_negative &negative)
1731
                
1732
2.07k
  : fActiveArea (tcActiveArea,
1733
2.07k
           fActiveAreaData,
1734
2.07k
           4)
1735
  
1736
2.07k
  , fMaskedAreas (tcMaskedAreas,
1737
2.07k
            fMaskedAreaData,
1738
2.07k
            0)
1739
  
1740
2.07k
  , fLinearizationTable (tcLinearizationTable,
1741
2.07k
               NULL,
1742
2.07k
               0)
1743
                
1744
2.07k
  , fBlackLevelRepeatDim (tcBlackLevelRepeatDim,
1745
2.07k
                fBlackLevelRepeatDimData,
1746
2.07k
                2)
1747
               
1748
2.07k
  , fBlackLevel (tcBlackLevel,
1749
2.07k
           fBlackLevelData)
1750
           
1751
2.07k
  , fBlackLevelDeltaHData ()
1752
2.07k
  , fBlackLevelDeltaVData ()
1753
           
1754
2.07k
  , fBlackLevelDeltaH (tcBlackLevelDeltaH)
1755
2.07k
  , fBlackLevelDeltaV (tcBlackLevelDeltaV)
1756
 
1757
2.07k
  , fWhiteLevel16 (tcWhiteLevel,
1758
2.07k
             fWhiteLevelData16)
1759
           
1760
2.07k
  , fWhiteLevel32 (tcWhiteLevel,
1761
2.07k
             fWhiteLevelData32)
1762
           
1763
2.07k
  {
1764
  
1765
2.07k
  const dng_image &rawImage (negative.RawImage ());
1766
  
1767
2.07k
  const dng_linearization_info *rangeInfo = negative.GetLinearizationInfo ();
1768
  
1769
2.07k
  if (rangeInfo)
1770
2.07k
    {
1771
    
1772
    // ActiveArea:
1773
    
1774
2.07k
      {
1775
    
1776
2.07k
      const dng_rect &r = rangeInfo->fActiveArea;
1777
      
1778
2.07k
      if (r.NotEmpty ())
1779
16
        {
1780
      
1781
16
        fActiveAreaData [0] = r.t;
1782
16
        fActiveAreaData [1] = r.l;
1783
16
        fActiveAreaData [2] = r.b;
1784
16
        fActiveAreaData [3] = r.r;
1785
        
1786
16
        directory.Add (&fActiveArea);
1787
        
1788
16
        }
1789
        
1790
2.07k
      }
1791
      
1792
    // MaskedAreas:
1793
      
1794
2.07k
    if (rangeInfo->fMaskedAreaCount)
1795
0
      {
1796
      
1797
0
      fMaskedAreas.SetCount (rangeInfo->fMaskedAreaCount * 4);
1798
      
1799
0
      for (uint32 index = 0; index < rangeInfo->fMaskedAreaCount; index++)
1800
0
        {
1801
        
1802
0
        const dng_rect &r = rangeInfo->fMaskedArea [index];
1803
        
1804
0
        fMaskedAreaData [index * 4 + 0] = r.t;
1805
0
        fMaskedAreaData [index * 4 + 1] = r.l;
1806
0
        fMaskedAreaData [index * 4 + 2] = r.b;
1807
0
        fMaskedAreaData [index * 4 + 3] = r.r;
1808
        
1809
0
        }
1810
        
1811
0
      directory.Add (&fMaskedAreas);
1812
      
1813
0
      }
1814
      
1815
    // LinearizationTable:
1816
1817
2.07k
    if (rangeInfo->fLinearizationTable.Get ())
1818
0
      {
1819
      
1820
0
      fLinearizationTable.SetData  (rangeInfo->fLinearizationTable->Buffer_uint16 ()     );
1821
0
      fLinearizationTable.SetCount (rangeInfo->fLinearizationTable->LogicalSize   () >> 1);
1822
      
1823
0
      directory.Add (&fLinearizationTable);
1824
      
1825
0
      }
1826
      
1827
    // BlackLevelRepeatDim:
1828
    
1829
2.07k
      {
1830
    
1831
2.07k
      fBlackLevelRepeatDimData [0] = (uint16) rangeInfo->fBlackLevelRepeatRows;
1832
2.07k
      fBlackLevelRepeatDimData [1] = (uint16) rangeInfo->fBlackLevelRepeatCols;
1833
      
1834
2.07k
      directory.Add (&fBlackLevelRepeatDim);
1835
      
1836
2.07k
      }
1837
    
1838
    // BlackLevel:
1839
    
1840
2.07k
      {
1841
    
1842
2.07k
      uint32 index = 0;
1843
      
1844
4.14k
      for (uint16 v = 0; v < rangeInfo->fBlackLevelRepeatRows; v++)
1845
2.07k
        {
1846
        
1847
4.14k
        for (uint32 h = 0; h < rangeInfo->fBlackLevelRepeatCols; h++)
1848
2.07k
          {
1849
          
1850
5.79k
          for (uint32 c = 0; c < rawImage.Planes (); c++)
1851
3.72k
            {
1852
          
1853
3.72k
            fBlackLevelData [index++] = rangeInfo->BlackLevel (v, h, c);
1854
          
1855
3.72k
            }
1856
            
1857
2.07k
          }
1858
          
1859
2.07k
        }
1860
        
1861
2.07k
      fBlackLevel.SetCount (rangeInfo->fBlackLevelRepeatRows *
1862
2.07k
                  rangeInfo->fBlackLevelRepeatCols * rawImage.Planes ());
1863
      
1864
2.07k
      directory.Add (&fBlackLevel);
1865
      
1866
2.07k
      }
1867
    
1868
    // BlackLevelDeltaH:
1869
        
1870
2.07k
    if (rangeInfo->ColumnBlackCount ())
1871
0
      {
1872
      
1873
0
      uint32 count = rangeInfo->ColumnBlackCount ();
1874
    
1875
0
      fBlackLevelDeltaHData.Allocate (count, sizeof (dng_srational));
1876
                         
1877
0
      dng_srational *blacks = (dng_srational *) fBlackLevelDeltaHData.Buffer ();
1878
      
1879
0
      for (uint32 col = 0; col < count; col++)
1880
0
        {
1881
        
1882
0
        blacks [col] = rangeInfo->ColumnBlack (col);
1883
        
1884
0
        }
1885
                         
1886
0
      fBlackLevelDeltaH.SetData  (blacks);
1887
0
      fBlackLevelDeltaH.SetCount (count );
1888
      
1889
0
      directory.Add (&fBlackLevelDeltaH);
1890
      
1891
0
      }
1892
    
1893
    // BlackLevelDeltaV:
1894
        
1895
2.07k
    if (rangeInfo->RowBlackCount ())
1896
0
      {
1897
      
1898
0
      uint32 count = rangeInfo->RowBlackCount ();
1899
    
1900
0
      fBlackLevelDeltaVData.Allocate (count, sizeof (dng_srational));
1901
                         
1902
0
      dng_srational *blacks = (dng_srational *) fBlackLevelDeltaVData.Buffer ();
1903
      
1904
0
      for (uint32 row = 0; row < count; row++)
1905
0
        {
1906
        
1907
0
        blacks [row] = rangeInfo->RowBlack (row);
1908
        
1909
0
        }
1910
                         
1911
0
      fBlackLevelDeltaV.SetData  (blacks);
1912
0
      fBlackLevelDeltaV.SetCount (count );
1913
      
1914
0
      directory.Add (&fBlackLevelDeltaV);
1915
      
1916
0
      }
1917
      
1918
2.07k
    }
1919
    
1920
  // WhiteLevel:
1921
  
1922
  // Only use the 32-bit data type if we must use it since there
1923
  // are some lazy (non-Adobe) DNG readers out there.
1924
  
1925
2.07k
  bool needs32 = false;
1926
    
1927
2.07k
  fWhiteLevel16.SetCount (rawImage.Planes ());
1928
2.07k
  fWhiteLevel32.SetCount (rawImage.Planes ());
1929
  
1930
5.79k
  for (uint32 c = 0; c < fWhiteLevel16.Count (); c++)
1931
3.72k
    {
1932
    
1933
3.72k
    fWhiteLevelData32 [c] = negative.WhiteLevel (c);
1934
    
1935
3.72k
    if (fWhiteLevelData32 [c] > 0x0FFFF)
1936
0
      {
1937
0
      needs32 = true;
1938
0
      }
1939
      
1940
3.72k
    fWhiteLevelData16 [c] = (uint16) fWhiteLevelData32 [c];
1941
    
1942
3.72k
    }
1943
    
1944
2.07k
  if (needs32)
1945
0
    {
1946
0
    directory.Add (&fWhiteLevel32);
1947
0
    }
1948
    
1949
2.07k
  else
1950
2.07k
    {
1951
2.07k
    directory.Add (&fWhiteLevel16);
1952
2.07k
    }
1953
  
1954
2.07k
  }
1955
1956
/******************************************************************************/
1957
1958
class mosaic_tag_set
1959
  {
1960
  
1961
  private:
1962
  
1963
    uint16 fCFARepeatPatternDimData [2];
1964
    
1965
    tag_uint16_ptr fCFARepeatPatternDim;
1966
                       
1967
    uint8 fCFAPatternData [kMaxCFAPattern *
1968
                 kMaxCFAPattern];
1969
    
1970
    tag_uint8_ptr fCFAPattern;
1971
                 
1972
    uint8 fCFAPlaneColorData [kMaxColorPlanes];
1973
    
1974
    tag_uint8_ptr fCFAPlaneColor;
1975
                  
1976
    tag_uint16 fCFALayout;
1977
    
1978
    tag_uint32 fGreenSplit;
1979
    
1980
  public:
1981
  
1982
    mosaic_tag_set (dng_tiff_directory &directory,
1983
                const dng_mosaic_info &info);
1984
    
1985
  };
1986
  
1987
/******************************************************************************/
1988
1989
mosaic_tag_set::mosaic_tag_set (dng_tiff_directory &directory,
1990
                      const dng_mosaic_info &info)
1991
1992
2.07k
  : fCFARepeatPatternDim (tcCFARepeatPatternDim,
1993
2.07k
                  fCFARepeatPatternDimData,
1994
2.07k
                  2)
1995
                  
1996
2.07k
  , fCFAPattern (tcCFAPattern,
1997
2.07k
           fCFAPatternData)
1998
          
1999
2.07k
  , fCFAPlaneColor (tcCFAPlaneColor,
2000
2.07k
            fCFAPlaneColorData)
2001
            
2002
2.07k
  , fCFALayout (tcCFALayout,
2003
2.07k
          (uint16) info.fCFALayout)
2004
  
2005
2.07k
  , fGreenSplit (tcBayerGreenSplit,
2006
2.07k
           info.fBayerGreenSplit)
2007
  
2008
2.07k
  {
2009
  
2010
2.07k
  if (info.IsColorFilterArray ())
2011
0
    {
2012
  
2013
    // CFARepeatPatternDim:
2014
    
2015
0
    fCFARepeatPatternDimData [0] = (uint16) info.fCFAPatternSize.v;
2016
0
    fCFARepeatPatternDimData [1] = (uint16) info.fCFAPatternSize.h;
2017
        
2018
0
    directory.Add (&fCFARepeatPatternDim);
2019
    
2020
    // CFAPattern:
2021
    
2022
0
    fCFAPattern.SetCount (info.fCFAPatternSize.v *
2023
0
                info.fCFAPatternSize.h);
2024
                
2025
0
    for (int32 r = 0; r < info.fCFAPatternSize.v; r++)
2026
0
      {
2027
      
2028
0
      for (int32 c = 0; c < info.fCFAPatternSize.h; c++)
2029
0
        {
2030
        
2031
0
        fCFAPatternData [r * info.fCFAPatternSize.h + c] = info.fCFAPattern [r] [c];
2032
        
2033
0
        }
2034
        
2035
0
      }
2036
        
2037
0
    directory.Add (&fCFAPattern);
2038
    
2039
    // CFAPlaneColor:
2040
    
2041
0
    fCFAPlaneColor.SetCount (info.fColorPlanes);
2042
    
2043
0
    for (uint32 j = 0; j < info.fColorPlanes; j++)
2044
0
      {
2045
    
2046
0
      fCFAPlaneColorData [j] = info.fCFAPlaneColor [j];
2047
      
2048
0
      }
2049
    
2050
0
    directory.Add (&fCFAPlaneColor);
2051
    
2052
    // CFALayout:
2053
    
2054
0
    fCFALayout.Set ((uint16) info.fCFALayout);
2055
    
2056
0
    directory.Add (&fCFALayout);
2057
    
2058
    // BayerGreenSplit:  (only include if the pattern is a Bayer pattern)
2059
      
2060
0
    if (info.fCFAPatternSize == dng_point (2, 2) &&
2061
0
      info.fColorPlanes    == 3)
2062
0
      {
2063
      
2064
0
      directory.Add (&fGreenSplit);
2065
      
2066
0
      }
2067
      
2068
0
    }
2069
2070
2.07k
  }
2071
  
2072
/******************************************************************************/
2073
2074
class color_tag_set
2075
  {
2076
  
2077
  private:
2078
  
2079
    uint32 fColorChannels;
2080
    
2081
    tag_matrix fCameraCalibration1;
2082
    tag_matrix fCameraCalibration2;
2083
                     
2084
    tag_string fCameraCalibrationSignature;
2085
    
2086
    tag_string fAsShotProfileName;
2087
2088
    dng_urational fAnalogBalanceData [4];
2089
  
2090
    tag_urational_ptr fAnalogBalance;
2091
                    
2092
    dng_urational fAsShotNeutralData [4];
2093
    
2094
    tag_urational_ptr fAsShotNeutral;
2095
                      
2096
    dng_urational fAsShotWhiteXYData [2];
2097
    
2098
    tag_urational_ptr fAsShotWhiteXY;
2099
                      
2100
    tag_urational fLinearResponseLimit;
2101
                      
2102
  public:
2103
  
2104
    color_tag_set (dng_tiff_directory &directory,
2105
               const dng_negative &negative);
2106
    
2107
  };
2108
  
2109
/******************************************************************************/
2110
2111
color_tag_set::color_tag_set (dng_tiff_directory &directory,
2112
                    const dng_negative &negative)
2113
                
2114
826
  : fColorChannels (negative.ColorChannels ())
2115
  
2116
826
  , fCameraCalibration1 (tcCameraCalibration1,
2117
826
                 negative.CameraCalibration1 ())
2118
            
2119
826
  , fCameraCalibration2 (tcCameraCalibration2,
2120
826
                 negative.CameraCalibration2 ())
2121
               
2122
826
  , fCameraCalibrationSignature (tcCameraCalibrationSignature,
2123
826
                   negative.CameraCalibrationSignature ())
2124
                   
2125
826
  , fAsShotProfileName (tcAsShotProfileName,
2126
826
              negative.AsShotProfileName ())
2127
2128
826
  , fAnalogBalance (tcAnalogBalance,
2129
826
            fAnalogBalanceData,
2130
826
            fColorChannels)
2131
            
2132
826
  , fAsShotNeutral (tcAsShotNeutral,
2133
826
            fAsShotNeutralData,
2134
826
            fColorChannels)
2135
            
2136
826
  , fAsShotWhiteXY (tcAsShotWhiteXY,
2137
826
            fAsShotWhiteXYData,
2138
826
            2)
2139
                    
2140
826
  , fLinearResponseLimit (tcLinearResponseLimit,
2141
826
                  negative.LinearResponseLimitR ())
2142
                  
2143
826
  {
2144
  
2145
826
  if (fColorChannels > 1)
2146
826
    {
2147
    
2148
826
    uint32 channels2 = fColorChannels * fColorChannels;
2149
    
2150
826
    if (fCameraCalibration1.Count () == channels2)
2151
56
      {
2152
      
2153
56
      directory.Add (&fCameraCalibration1);
2154
    
2155
56
      }
2156
      
2157
826
    if (fCameraCalibration2.Count () == channels2)
2158
0
      {
2159
      
2160
0
      directory.Add (&fCameraCalibration2);
2161
    
2162
0
      }
2163
      
2164
826
    if (fCameraCalibration1.Count () == channels2 ||
2165
826
      fCameraCalibration2.Count () == channels2)
2166
56
      {
2167
      
2168
56
      if (negative.CameraCalibrationSignature ().NotEmpty ())
2169
17
        {
2170
        
2171
17
        directory.Add (&fCameraCalibrationSignature);
2172
        
2173
17
        }
2174
      
2175
56
      }
2176
      
2177
826
    if (negative.AsShotProfileName ().NotEmpty ())
2178
56
      {
2179
      
2180
56
      directory.Add (&fAsShotProfileName);
2181
        
2182
56
      }
2183
      
2184
3.30k
    for (uint32 j = 0; j < fColorChannels; j++)
2185
2.47k
      {
2186
      
2187
2.47k
      fAnalogBalanceData [j] = negative.AnalogBalanceR (j);
2188
      
2189
2.47k
      }
2190
      
2191
826
    directory.Add (&fAnalogBalance);
2192
    
2193
826
    if (negative.HasCameraNeutral ())
2194
46
      {
2195
      
2196
184
      for (uint32 k = 0; k < fColorChannels; k++)
2197
138
        {
2198
        
2199
138
        fAsShotNeutralData [k] = negative.CameraNeutralR (k);
2200
        
2201
138
        }
2202
      
2203
46
      directory.Add (&fAsShotNeutral);
2204
      
2205
46
      }
2206
      
2207
780
    else if (negative.HasCameraWhiteXY ())
2208
568
      {
2209
      
2210
568
      negative.GetCameraWhiteXY (fAsShotWhiteXYData [0],
2211
568
                     fAsShotWhiteXYData [1]);
2212
                     
2213
568
      directory.Add (&fAsShotWhiteXY);
2214
      
2215
568
      }
2216
    
2217
826
    directory.Add (&fLinearResponseLimit);
2218
    
2219
826
    }
2220
  
2221
826
  }
2222
2223
/******************************************************************************/
2224
2225
class profile_tag_set
2226
  {
2227
  
2228
  private:
2229
  
2230
    tag_uint16 fCalibrationIlluminant1;
2231
    tag_uint16 fCalibrationIlluminant2;
2232
    
2233
    tag_matrix fColorMatrix1;
2234
    tag_matrix fColorMatrix2;
2235
    
2236
    tag_matrix fForwardMatrix1;
2237
    tag_matrix fForwardMatrix2;
2238
2239
    tag_matrix fReductionMatrix1;
2240
    tag_matrix fReductionMatrix2;
2241
    
2242
    tag_string fProfileName;
2243
    
2244
    tag_string fProfileCalibrationSignature;
2245
    
2246
    tag_uint32 fEmbedPolicyTag;
2247
    
2248
    tag_string fCopyrightTag;
2249
    
2250
    uint32 fHueSatMapDimData [3];
2251
    
2252
    tag_uint32_ptr fHueSatMapDims;
2253
2254
    tag_data_ptr fHueSatData1;
2255
    tag_data_ptr fHueSatData2;
2256
    
2257
    tag_uint32 fHueSatMapEncodingTag;
2258
    
2259
    uint32 fLookTableDimData [3];
2260
    
2261
    tag_uint32_ptr fLookTableDims;
2262
2263
    tag_data_ptr fLookTableData;
2264
    
2265
    tag_uint32 fLookTableEncodingTag;
2266
2267
    tag_srational fBaselineExposureOffsetTag;
2268
    
2269
    tag_uint32 fDefaultBlackRenderTag;
2270
2271
    dng_memory_data fToneCurveBuffer;
2272
    
2273
    tag_data_ptr fToneCurveTag;
2274
2275
  public:
2276
  
2277
    profile_tag_set (dng_tiff_directory &directory,
2278
             const dng_camera_profile &profile);
2279
    
2280
  };
2281
  
2282
/******************************************************************************/
2283
2284
profile_tag_set::profile_tag_set (dng_tiff_directory &directory,
2285
                        const dng_camera_profile &profile)
2286
                
2287
826
  : fCalibrationIlluminant1 (tcCalibrationIlluminant1,
2288
826
                 (uint16) profile.CalibrationIlluminant1 ())
2289
                 
2290
826
  , fCalibrationIlluminant2 (tcCalibrationIlluminant2,
2291
826
                 (uint16) profile.CalibrationIlluminant2 ())
2292
  
2293
826
  , fColorMatrix1 (tcColorMatrix1,
2294
826
             profile.ColorMatrix1 ())
2295
            
2296
826
  , fColorMatrix2 (tcColorMatrix2,
2297
826
             profile.ColorMatrix2 ())
2298
2299
826
  , fForwardMatrix1 (tcForwardMatrix1,
2300
826
             profile.ForwardMatrix1 ())
2301
            
2302
826
  , fForwardMatrix2 (tcForwardMatrix2,
2303
826
             profile.ForwardMatrix2 ())
2304
            
2305
826
  , fReductionMatrix1 (tcReductionMatrix1,
2306
826
               profile.ReductionMatrix1 ())
2307
            
2308
826
  , fReductionMatrix2 (tcReductionMatrix2,
2309
826
               profile.ReductionMatrix2 ())
2310
               
2311
826
  , fProfileName (tcProfileName,
2312
826
            profile.Name (),
2313
826
            false)
2314
               
2315
826
  , fProfileCalibrationSignature (tcProfileCalibrationSignature,
2316
826
                    profile.ProfileCalibrationSignature (),
2317
826
                    false)
2318
            
2319
826
  , fEmbedPolicyTag (tcProfileEmbedPolicy,
2320
826
             profile.EmbedPolicy ())
2321
             
2322
826
  , fCopyrightTag (tcProfileCopyright,
2323
826
             profile.Copyright (),
2324
826
             false)
2325
             
2326
826
  , fHueSatMapDims (tcProfileHueSatMapDims, 
2327
826
            fHueSatMapDimData,
2328
826
            3)
2329
    
2330
826
  , fHueSatData1 (tcProfileHueSatMapData1,
2331
826
            ttFloat,
2332
826
            profile.HueSatDeltas1 ().DeltasCount () * 3,
2333
826
            profile.HueSatDeltas1 ().GetConstDeltas ())
2334
            
2335
826
  , fHueSatData2 (tcProfileHueSatMapData2,
2336
826
            ttFloat,
2337
826
            profile.HueSatDeltas2 ().DeltasCount () * 3,
2338
826
            profile.HueSatDeltas2 ().GetConstDeltas ())
2339
            
2340
826
  , fHueSatMapEncodingTag (tcProfileHueSatMapEncoding,
2341
826
                 profile.HueSatMapEncoding ())
2342
             
2343
826
  , fLookTableDims (tcProfileLookTableDims,
2344
826
            fLookTableDimData,
2345
826
            3)
2346
            
2347
826
  , fLookTableData (tcProfileLookTableData,
2348
826
            ttFloat,
2349
826
            profile.LookTable ().DeltasCount () * 3,
2350
826
              profile.LookTable ().GetConstDeltas ())
2351
            
2352
826
  , fLookTableEncodingTag (tcProfileLookTableEncoding,
2353
826
                 profile.LookTableEncoding ())
2354
             
2355
826
  , fBaselineExposureOffsetTag (tcBaselineExposureOffset,
2356
826
                  profile.BaselineExposureOffset ())
2357
             
2358
826
  , fDefaultBlackRenderTag (tcDefaultBlackRender,
2359
826
                profile.DefaultBlackRender ())
2360
             
2361
826
  , fToneCurveBuffer ()
2362
            
2363
826
  , fToneCurveTag (tcProfileToneCurve,
2364
826
             ttFloat,
2365
826
             0,
2366
826
             NULL)
2367
2368
826
  {
2369
  
2370
826
  if (profile.HasColorMatrix1 ())
2371
826
    {
2372
  
2373
826
    uint32 colorChannels = profile.ColorMatrix1 ().Rows ();
2374
    
2375
826
    directory.Add (&fCalibrationIlluminant1);
2376
    
2377
826
    directory.Add (&fColorMatrix1);
2378
    
2379
826
    if (fForwardMatrix1.Count () == colorChannels * 3)
2380
0
      {
2381
      
2382
0
      directory.Add (&fForwardMatrix1);
2383
2384
0
      }
2385
    
2386
826
    if (colorChannels > 3 && fReductionMatrix1.Count () == colorChannels * 3)
2387
0
      {
2388
      
2389
0
      directory.Add (&fReductionMatrix1);
2390
      
2391
0
      }
2392
      
2393
826
    if (profile.HasColorMatrix2 ())
2394
1
      {
2395
    
2396
1
      directory.Add (&fCalibrationIlluminant2);
2397
    
2398
1
      directory.Add (&fColorMatrix2);
2399
        
2400
1
      if (fForwardMatrix2.Count () == colorChannels * 3)
2401
0
        {
2402
        
2403
0
        directory.Add (&fForwardMatrix2);
2404
2405
0
        }
2406
    
2407
1
      if (colorChannels > 3 && fReductionMatrix2.Count () == colorChannels * 3)
2408
0
        {
2409
        
2410
0
        directory.Add (&fReductionMatrix2);
2411
        
2412
0
        }
2413
  
2414
1
      }
2415
      
2416
826
    if (profile.Name ().NotEmpty ())
2417
826
      {
2418
      
2419
826
      directory.Add (&fProfileName);
2420
2421
826
      }
2422
      
2423
826
    if (profile.ProfileCalibrationSignature ().NotEmpty ())
2424
1
      {
2425
      
2426
1
      directory.Add (&fProfileCalibrationSignature);
2427
      
2428
1
      }
2429
      
2430
826
    directory.Add (&fEmbedPolicyTag);
2431
    
2432
826
    if (profile.Copyright ().NotEmpty ())
2433
1
      {
2434
      
2435
1
      directory.Add (&fCopyrightTag);
2436
      
2437
1
      }
2438
    
2439
826
    bool haveHueSat1 = profile.HueSatDeltas1 ().IsValid ();
2440
    
2441
826
    bool haveHueSat2 = profile.HueSatDeltas2 ().IsValid () &&
2442
826
               profile.HasColorMatrix2 ();
2443
2444
826
    if (haveHueSat1 || haveHueSat2)
2445
0
      {
2446
      
2447
0
      uint32 hueDivs = 0;
2448
0
      uint32 satDivs = 0;
2449
0
      uint32 valDivs = 0;
2450
2451
0
      if (haveHueSat1)
2452
0
        {
2453
2454
0
        profile.HueSatDeltas1 ().GetDivisions (hueDivs,
2455
0
                             satDivs,
2456
0
                             valDivs);
2457
2458
0
        }
2459
2460
0
      else
2461
0
        {
2462
2463
0
        profile.HueSatDeltas2 ().GetDivisions (hueDivs,
2464
0
                             satDivs,
2465
0
                             valDivs);
2466
2467
0
        }
2468
        
2469
0
      fHueSatMapDimData [0] = hueDivs;
2470
0
      fHueSatMapDimData [1] = satDivs;
2471
0
      fHueSatMapDimData [2] = valDivs;
2472
      
2473
0
      directory.Add (&fHueSatMapDims);
2474
2475
      // Don't bother including the ProfileHueSatMapEncoding tag unless it's
2476
      // non-linear.
2477
2478
0
      if (profile.HueSatMapEncoding () != encoding_Linear)
2479
0
        {
2480
2481
0
        directory.Add (&fHueSatMapEncodingTag);
2482
2483
0
        }
2484
    
2485
0
      }
2486
      
2487
826
    if (haveHueSat1)
2488
0
      {
2489
      
2490
0
      directory.Add (&fHueSatData1);
2491
      
2492
0
      }
2493
      
2494
826
    if (haveHueSat2)
2495
0
      {
2496
      
2497
0
      directory.Add (&fHueSatData2);
2498
      
2499
0
      }
2500
      
2501
826
    if (profile.HasLookTable ())
2502
0
      {
2503
      
2504
0
      uint32 hueDivs = 0;
2505
0
      uint32 satDivs = 0;
2506
0
      uint32 valDivs = 0;
2507
2508
0
      profile.LookTable ().GetDivisions (hueDivs,
2509
0
                         satDivs,
2510
0
                         valDivs);
2511
2512
0
      fLookTableDimData [0] = hueDivs;
2513
0
      fLookTableDimData [1] = satDivs;
2514
0
      fLookTableDimData [2] = valDivs;
2515
      
2516
0
      directory.Add (&fLookTableDims);
2517
      
2518
0
      directory.Add (&fLookTableData);
2519
      
2520
      // Don't bother including the ProfileLookTableEncoding tag unless it's
2521
      // non-linear.
2522
2523
0
      if (profile.LookTableEncoding () != encoding_Linear)
2524
0
        {
2525
2526
0
        directory.Add (&fLookTableEncodingTag);
2527
2528
0
        }
2529
    
2530
0
      }
2531
2532
    // Don't bother including the BaselineExposureOffset tag unless it's both
2533
    // valid and non-zero.
2534
2535
826
    if (profile.BaselineExposureOffset ().IsValid ())
2536
826
      {
2537
2538
826
      if (profile.BaselineExposureOffset ().As_real64 () != 0.0)
2539
0
        {
2540
      
2541
0
        directory.Add (&fBaselineExposureOffsetTag);
2542
2543
0
        }
2544
        
2545
826
      }
2546
      
2547
826
    if (profile.DefaultBlackRender () != defaultBlackRender_Auto)
2548
9
      {
2549
2550
9
      directory.Add (&fDefaultBlackRenderTag);
2551
2552
9
      }
2553
    
2554
826
    if (profile.ToneCurve ().IsValid ())
2555
0
      {
2556
      
2557
      // Tone curve stored as pairs of 32-bit coordinates.  Probably could do with
2558
      // 16-bits here, but should be small number of points so...
2559
      
2560
0
      uint32 toneCurvePoints = (uint32) (profile.ToneCurve ().fCoord.size ());
2561
2562
0
      fToneCurveBuffer.Allocate (SafeUint32Mult(toneCurvePoints, 2),
2563
0
                     sizeof (real32));
2564
2565
0
      real32 *points = fToneCurveBuffer.Buffer_real32 ();
2566
      
2567
0
      fToneCurveTag.SetCount (toneCurvePoints * 2);
2568
0
      fToneCurveTag.SetData  (points);
2569
      
2570
0
      for (uint32 i = 0; i < toneCurvePoints; i++)
2571
0
        {
2572
2573
        // Transpose coordinates so they are in a more expected
2574
        // order (domain -> range).
2575
2576
0
        points [i * 2    ] = (real32) profile.ToneCurve ().fCoord [i].h;
2577
0
        points [i * 2 + 1] = (real32) profile.ToneCurve ().fCoord [i].v;
2578
2579
0
        }
2580
2581
0
      directory.Add (&fToneCurveTag);
2582
2583
0
      }
2584
2585
826
    }
2586
  
2587
826
  }
2588
2589
/******************************************************************************/
2590
2591
tiff_dng_extended_color_profile::tiff_dng_extended_color_profile 
2592
                 (const dng_camera_profile &profile)
2593
2594
0
  : fProfile (profile)
2595
2596
0
  {
2597
  
2598
0
  }
2599
2600
/******************************************************************************/
2601
2602
void tiff_dng_extended_color_profile::Put (dng_stream &stream,
2603
                       bool includeModelRestriction)
2604
0
  {
2605
  
2606
  // Profile header.
2607
2608
0
  stream.Put_uint16 (stream.BigEndian () ? byteOrderMM : byteOrderII);
2609
2610
0
  stream.Put_uint16 (magicExtendedProfile);
2611
2612
0
  stream.Put_uint32 (8);
2613
  
2614
  // Profile tags.
2615
  
2616
0
  profile_tag_set tagSet (*this, fProfile);
2617
2618
  // Camera this profile is for.
2619
2620
0
  tag_string cameraModelTag (tcUniqueCameraModel, 
2621
0
                 fProfile.UniqueCameraModelRestriction ());
2622
                 
2623
0
  if (includeModelRestriction)
2624
0
    {
2625
    
2626
0
    if (fProfile.UniqueCameraModelRestriction ().NotEmpty ())
2627
0
      {
2628
      
2629
0
      Add (&cameraModelTag);
2630
      
2631
0
      }
2632
      
2633
0
    }
2634
2635
  // Write it all out.
2636
2637
0
  dng_tiff_directory::Put (stream, offsetsRelativeToExplicitBase, 8);
2638
2639
0
  }
2640
2641
/*****************************************************************************/
2642
2643
tag_dng_noise_profile::tag_dng_noise_profile (const dng_noise_profile &profile)
2644
2645
2.07k
  : tag_data_ptr (tcNoiseProfile,
2646
2.07k
            ttDouble,
2647
2.07k
            2 * profile.NumFunctions (),
2648
2.07k
            fValues)
2649
2650
2.07k
  {
2651
2652
2.07k
  DNG_REQUIRE (profile.NumFunctions () <= kMaxColorPlanes,
2653
2.07k
         "Too many noise functions in tag_dng_noise_profile.");
2654
2655
2.08k
  for (uint32 i = 0; i < profile.NumFunctions (); i++)
2656
10
    {
2657
2658
10
    fValues [(2 * i)  ] = profile.NoiseFunction (i).Scale  ();
2659
10
    fValues [(2 * i) + 1] = profile.NoiseFunction (i).Offset ();
2660
2661
10
    }
2662
  
2663
2.07k
  }
2664
    
2665
/*****************************************************************************/
2666
2667
dng_image_writer::dng_image_writer ()
2668
14.9k
  {
2669
  
2670
14.9k
  }
2671
2672
/*****************************************************************************/
2673
2674
dng_image_writer::~dng_image_writer ()
2675
14.9k
  {
2676
  
2677
14.9k
  }
2678
                
2679
/*****************************************************************************/
2680
2681
uint32 dng_image_writer::CompressedBufferSize (const dng_ifd &ifd,
2682
                         uint32 uncompressedSize)
2683
9.74k
  {
2684
  
2685
9.74k
  switch (ifd.fCompression)
2686
9.74k
    {
2687
    
2688
0
    case ccLZW:
2689
0
      {
2690
      
2691
      // Add lots of slop for LZW to expand data.
2692
        
2693
0
      return SafeUint32Add (SafeUint32Mult (uncompressedSize, 2), 1024);
2694
      
2695
0
      }
2696
      
2697
495
    case ccDeflate:
2698
495
      {
2699
    
2700
      // ZLib says maximum is source size + 0.1% + 12 bytes.
2701
      
2702
495
      return SafeUint32Add (SafeUint32Add (uncompressedSize,
2703
495
                         uncompressedSize >> 8), 64);
2704
      
2705
0
      }
2706
      
2707
0
    case ccJPEG:
2708
0
      {
2709
      
2710
      // If we are saving lossless JPEG from an 8-bit image, reserve
2711
      // space to pad the data out to 16-bits.
2712
      
2713
0
      if (ifd.fBitsPerSample [0] <= 8)
2714
0
        {
2715
        
2716
0
        return SafeUint32Mult (uncompressedSize, 2);
2717
        
2718
0
        }
2719
        
2720
0
      break;
2721
  
2722
0
      }
2723
      
2724
9.25k
    default:
2725
9.25k
      break;
2726
    
2727
9.74k
    }
2728
  
2729
9.25k
  return 0;
2730
  
2731
9.74k
  }
2732
                
2733
/******************************************************************************/
2734
2735
static void EncodeDelta8 (uint8 *dPtr,
2736
              uint32 rows,
2737
              uint32 cols,
2738
              uint32 channels)
2739
0
  {
2740
  
2741
0
  const uint32 dRowStep = cols * channels;
2742
  
2743
0
  for (uint32 row = 0; row < rows; row++)
2744
0
    {
2745
    
2746
0
    for (uint32 col = cols - 1; col > 0; col--)
2747
0
      {
2748
      
2749
0
      for (uint32 channel = 0; channel < channels; channel++)
2750
0
        {
2751
        
2752
0
        dPtr [col * channels + channel] -= dPtr [(col - 1) * channels + channel];
2753
        
2754
0
        }
2755
      
2756
0
      }
2757
    
2758
0
    dPtr += dRowStep;
2759
    
2760
0
    }
2761
2762
0
  }
2763
2764
/******************************************************************************/
2765
2766
static void EncodeDelta16 (uint16 *dPtr,
2767
               uint32 rows,
2768
               uint32 cols,
2769
               uint32 channels)
2770
0
  {
2771
  
2772
0
  const uint32 dRowStep = cols * channels;
2773
  
2774
0
  for (uint32 row = 0; row < rows; row++)
2775
0
    {
2776
    
2777
0
    for (uint32 col = cols - 1; col > 0; col--)
2778
0
      {
2779
      
2780
0
      for (uint32 channel = 0; channel < channels; channel++)
2781
0
        {
2782
        
2783
0
        dPtr [col * channels + channel] -= dPtr [(col - 1) * channels + channel];
2784
        
2785
0
        }
2786
      
2787
0
      }
2788
    
2789
0
    dPtr += dRowStep;
2790
    
2791
0
    }
2792
2793
0
  }
2794
  
2795
/******************************************************************************/
2796
2797
static void EncodeDelta32 (uint32 *dPtr,
2798
               uint32 rows,
2799
               uint32 cols,
2800
               uint32 channels)
2801
0
  {
2802
  
2803
0
  const uint32 dRowStep = cols * channels;
2804
  
2805
0
  for (uint32 row = 0; row < rows; row++)
2806
0
    {
2807
    
2808
0
    for (uint32 col = cols - 1; col > 0; col--)
2809
0
      {
2810
      
2811
0
      for (uint32 channel = 0; channel < channels; channel++)
2812
0
        {
2813
        
2814
0
        dPtr [col * channels + channel] -= dPtr [(col - 1) * channels + channel];
2815
        
2816
0
        }
2817
      
2818
0
      }
2819
    
2820
0
    dPtr += dRowStep;
2821
    
2822
0
    }
2823
2824
0
  }
2825
  
2826
/*****************************************************************************/
2827
2828
inline void EncodeDeltaBytes (uint8 *bytePtr, int32 cols, int32 channels)
2829
141k
  {
2830
  
2831
141k
  if (channels == 1)
2832
123k
    {
2833
    
2834
123k
    bytePtr += (cols - 1);
2835
    
2836
123k
    uint8 this0 = bytePtr [0];
2837
    
2838
5.73M
    for (int32 col = 1; col < cols; col++)
2839
5.60M
      {
2840
      
2841
5.60M
      uint8 prev0 = bytePtr [-1];
2842
      
2843
5.60M
      this0 -= prev0;
2844
      
2845
5.60M
      bytePtr [0] = this0;
2846
      
2847
5.60M
      this0 = prev0;
2848
      
2849
5.60M
      bytePtr -= 1;
2850
2851
5.60M
      }
2852
  
2853
123k
    }
2854
    
2855
17.8k
  else if (channels == 3)
2856
17.8k
    {
2857
    
2858
17.8k
    bytePtr += (cols - 1) * 3;
2859
    
2860
17.8k
    uint8 this0 = bytePtr [0];
2861
17.8k
    uint8 this1 = bytePtr [1];
2862
17.8k
    uint8 this2 = bytePtr [2];
2863
    
2864
8.18M
    for (int32 col = 1; col < cols; col++)
2865
8.17M
      {
2866
      
2867
8.17M
      uint8 prev0 = bytePtr [-3];
2868
8.17M
      uint8 prev1 = bytePtr [-2];
2869
8.17M
      uint8 prev2 = bytePtr [-1];
2870
      
2871
8.17M
      this0 -= prev0;
2872
8.17M
      this1 -= prev1;
2873
8.17M
      this2 -= prev2;
2874
      
2875
8.17M
      bytePtr [0] = this0;
2876
8.17M
      bytePtr [1] = this1;
2877
8.17M
      bytePtr [2] = this2;
2878
      
2879
8.17M
      this0 = prev0;
2880
8.17M
      this1 = prev1;
2881
8.17M
      this2 = prev2;
2882
      
2883
8.17M
      bytePtr -= 3;
2884
2885
8.17M
      }
2886
  
2887
17.8k
    }
2888
    
2889
0
  else
2890
0
    {
2891
  
2892
0
    uint32 rowBytes = cols * channels;
2893
    
2894
0
    bytePtr += rowBytes - 1;
2895
    
2896
0
    for (uint32 col = channels; col < rowBytes; col++)
2897
0
      {
2898
      
2899
0
      bytePtr [0] -= bytePtr [-channels];
2900
        
2901
0
      bytePtr--;
2902
2903
0
      }
2904
      
2905
0
    }
2906
2907
141k
  }
2908
2909
/*****************************************************************************/
2910
2911
static void EncodeFPDelta (uint8 *buffer,
2912
               uint8 *temp,
2913
               int32 cols,
2914
               int32 channels,
2915
               int32 bytesPerSample)
2916
141k
  {
2917
  
2918
141k
  int32 rowIncrement = cols * channels;
2919
  
2920
141k
  if (bytesPerSample == 2)
2921
141k
    {
2922
    
2923
141k
    const uint8 *src = buffer;
2924
    
2925
    #if qDNGBigEndian
2926
    uint8 *dst0 = temp;
2927
    uint8 *dst1 = temp + rowIncrement;
2928
    #else
2929
141k
    uint8 *dst1 = temp;
2930
141k
    uint8 *dst0 = temp + rowIncrement;
2931
141k
    #endif
2932
        
2933
15.2M
    for (int32 col = 0; col < rowIncrement; ++col)
2934
15.1M
      {
2935
      
2936
15.1M
      dst0 [col] = src [0];
2937
15.1M
      dst1 [col] = src [1];
2938
      
2939
15.1M
      src += 2;
2940
      
2941
15.1M
      }
2942
      
2943
141k
    }
2944
    
2945
0
  else if (bytesPerSample == 3)
2946
0
    {
2947
    
2948
0
    const uint8 *src = buffer;
2949
    
2950
0
    uint8 *dst0 = temp;
2951
0
    uint8 *dst1 = temp + rowIncrement;
2952
0
    uint8 *dst2 = temp + rowIncrement * 2;
2953
        
2954
0
    for (int32 col = 0; col < rowIncrement; ++col)
2955
0
      {
2956
      
2957
0
      dst0 [col] = src [0];
2958
0
      dst1 [col] = src [1];
2959
0
      dst2 [col] = src [2];
2960
      
2961
0
      src += 3;
2962
      
2963
0
      }
2964
      
2965
0
    }
2966
    
2967
0
  else
2968
0
    {
2969
    
2970
0
    const uint8 *src = buffer;
2971
    
2972
    #if qDNGBigEndian
2973
    uint8 *dst0 = temp;
2974
    uint8 *dst1 = temp + rowIncrement;
2975
    uint8 *dst2 = temp + rowIncrement * 2;
2976
    uint8 *dst3 = temp + rowIncrement * 3;
2977
    #else
2978
0
    uint8 *dst3 = temp;
2979
0
    uint8 *dst2 = temp + rowIncrement;
2980
0
    uint8 *dst1 = temp + rowIncrement * 2;
2981
0
    uint8 *dst0 = temp + rowIncrement * 3;
2982
0
    #endif
2983
        
2984
0
    for (int32 col = 0; col < rowIncrement; ++col)
2985
0
      {
2986
      
2987
0
      dst0 [col] = src [0];
2988
0
      dst1 [col] = src [1];
2989
0
      dst2 [col] = src [2];
2990
0
      dst3 [col] = src [3];
2991
      
2992
0
      src += 4;
2993
      
2994
0
      }
2995
      
2996
0
    }
2997
    
2998
141k
  EncodeDeltaBytes (temp, cols*bytesPerSample, channels);
2999
  
3000
141k
  memcpy (buffer, temp, cols*bytesPerSample*channels);
3001
  
3002
141k
  }
3003
3004
/*****************************************************************************/
3005
3006
void dng_image_writer::EncodePredictor (dng_host &host,
3007
                      const dng_ifd &ifd,
3008
                          dng_pixel_buffer &buffer,
3009
                    AutoPtr<dng_memory_block> &tempBuffer)
3010
68.4k
  {
3011
  
3012
68.4k
  switch (ifd.fPredictor)
3013
68.4k
    {
3014
    
3015
0
    case cpHorizontalDifference:
3016
0
    case cpHorizontalDifferenceX2:
3017
0
    case cpHorizontalDifferenceX4:
3018
0
      {
3019
      
3020
0
      int32 xFactor = 1;
3021
      
3022
0
      if (ifd.fPredictor == cpHorizontalDifferenceX2)
3023
0
        {
3024
0
        xFactor = 2;
3025
0
        }
3026
        
3027
0
      else if (ifd.fPredictor == cpHorizontalDifferenceX4)
3028
0
        {
3029
0
        xFactor = 4;
3030
0
        }
3031
      
3032
0
      switch (buffer.fPixelType)
3033
0
        {
3034
        
3035
0
        case ttByte:
3036
0
          {
3037
          
3038
0
          EncodeDelta8 ((uint8 *) buffer.fData,
3039
0
                  buffer.fArea.H (),
3040
0
                  buffer.fArea.W () / xFactor,
3041
0
                  buffer.fPlanes    * xFactor);
3042
          
3043
0
          return;
3044
          
3045
0
          }
3046
          
3047
0
        case ttShort:
3048
0
          {
3049
          
3050
0
          EncodeDelta16 ((uint16 *) buffer.fData,
3051
0
                   buffer.fArea.H (),
3052
0
                   buffer.fArea.W () / xFactor,
3053
0
                   buffer.fPlanes    * xFactor);
3054
          
3055
0
          return;
3056
          
3057
0
          }
3058
          
3059
0
        case ttLong:
3060
0
          {
3061
          
3062
0
          EncodeDelta32 ((uint32 *) buffer.fData,
3063
0
                   buffer.fArea.H (),
3064
0
                   buffer.fArea.W () / xFactor,
3065
0
                   buffer.fPlanes    * xFactor);
3066
          
3067
0
          return;
3068
          
3069
0
          }
3070
          
3071
0
        default:
3072
0
          break;
3073
          
3074
0
        }
3075
      
3076
0
      break;
3077
      
3078
0
      }
3079
      
3080
721
    case cpFloatingPoint:
3081
721
    case cpFloatingPointX2:
3082
721
    case cpFloatingPointX4:
3083
721
      {
3084
      
3085
721
      int32 xFactor = 1;
3086
      
3087
721
      if (ifd.fPredictor == cpFloatingPointX2)
3088
0
        {
3089
0
        xFactor = 2;
3090
0
        }
3091
        
3092
721
      else if (ifd.fPredictor == cpFloatingPointX4)
3093
0
        {
3094
0
        xFactor = 4;
3095
0
        }
3096
      
3097
721
      if (buffer.fRowStep < 0)
3098
0
        {
3099
0
        ThrowProgramError ("Row step may not be negative");
3100
0
        }
3101
721
      uint32 tempBufferSize = SafeUint32Mult (
3102
721
        static_cast<uint32>(buffer.fRowStep),
3103
721
        buffer.fPixelSize);
3104
      
3105
721
      if (!tempBuffer.Get () || tempBuffer->LogicalSize () < tempBufferSize)
3106
495
        {
3107
        
3108
495
        tempBuffer.Reset (host.Allocate (tempBufferSize));
3109
        
3110
495
        }
3111
        
3112
142k
      for (int32 row = buffer.fArea.t; row < buffer.fArea.b; row++)
3113
141k
        {
3114
        
3115
141k
        EncodeFPDelta ((uint8 *) buffer.DirtyPixel (row, buffer.fArea.l, buffer.fPlane),
3116
141k
                 tempBuffer->Buffer_uint8 (),
3117
141k
                 buffer.fArea.W () / xFactor,
3118
141k
                 buffer.fPlanes    * xFactor,
3119
141k
                 buffer.fPixelSize);
3120
        
3121
141k
        }
3122
      
3123
721
      return;
3124
      
3125
721
      }
3126
      
3127
67.7k
    default:
3128
67.7k
      break;
3129
    
3130
68.4k
    }
3131
  
3132
67.7k
  if (ifd.fPredictor != cpNullPredictor)
3133
0
    {
3134
    
3135
0
    ThrowProgramError ();
3136
    
3137
0
    }
3138
  
3139
67.7k
  }
3140
                
3141
/*****************************************************************************/
3142
3143
void dng_image_writer::ByteSwapBuffer (dng_host & /* host */,
3144
                     dng_pixel_buffer &buffer)
3145
0
  {
3146
  
3147
0
  uint32 pixels = buffer.fRowStep * buffer.fArea.H ();
3148
  
3149
0
  switch (buffer.fPixelSize)
3150
0
    {
3151
    
3152
0
    case 2:
3153
0
      {
3154
      
3155
0
      DoSwapBytes16 ((uint16 *) buffer.fData,
3156
0
               pixels);
3157
               
3158
0
      break;
3159
      
3160
0
      }
3161
      
3162
0
    case 4:
3163
0
      {
3164
      
3165
0
      DoSwapBytes32 ((uint32 *) buffer.fData,
3166
0
               pixels);
3167
               
3168
0
      break;
3169
      
3170
0
      }
3171
      
3172
0
    default:
3173
0
      break;
3174
      
3175
0
    }
3176
3177
0
  }
3178
                
3179
/*****************************************************************************/
3180
3181
void dng_image_writer::ReorderSubTileBlocks (const dng_ifd &ifd,
3182
                       dng_pixel_buffer &buffer,
3183
                       AutoPtr<dng_memory_block> &uncompressedBuffer,
3184
                       AutoPtr<dng_memory_block> &subTileBlockBuffer)
3185
0
  {
3186
  
3187
0
  uint32 blockRows = ifd.fSubTileBlockRows;
3188
0
  uint32 blockCols = ifd.fSubTileBlockCols;
3189
  
3190
0
  uint32 rowBlocks = buffer.fArea.H () / blockRows;
3191
0
  uint32 colBlocks = buffer.fArea.W () / blockCols;
3192
  
3193
0
  int32 rowStep = buffer.fRowStep * buffer.fPixelSize;
3194
0
  int32 colStep = buffer.fColStep * buffer.fPixelSize;
3195
  
3196
0
  int32 rowBlockStep = rowStep * blockRows;
3197
0
  int32 colBlockStep = colStep * blockCols;
3198
  
3199
0
  uint32 blockColBytes = blockCols * buffer.fPlanes * buffer.fPixelSize;
3200
  
3201
0
  const uint8 *s0 = uncompressedBuffer->Buffer_uint8 ();
3202
0
        uint8 *d0 = subTileBlockBuffer->Buffer_uint8 ();
3203
  
3204
0
  for (uint32 rowBlock = 0; rowBlock < rowBlocks; rowBlock++)
3205
0
    {
3206
    
3207
0
    const uint8 *s1 = s0;
3208
    
3209
0
    for (uint32 colBlock = 0; colBlock < colBlocks; colBlock++)
3210
0
      {
3211
      
3212
0
      const uint8 *s2 = s1;
3213
      
3214
0
      for (uint32 blockRow = 0; blockRow < blockRows; blockRow++)
3215
0
        {
3216
        
3217
0
        for (uint32 j = 0; j < blockColBytes; j++)
3218
0
          {
3219
          
3220
0
          d0 [j] = s2 [j];
3221
          
3222
0
          }
3223
          
3224
0
        d0 += blockColBytes;
3225
        
3226
0
        s2 += rowStep;
3227
        
3228
0
        }
3229
      
3230
0
      s1 += colBlockStep;
3231
      
3232
0
      }
3233
      
3234
0
    s0 += rowBlockStep;
3235
    
3236
0
    }
3237
    
3238
  // Copy back reordered pixels.
3239
    
3240
0
  DoCopyBytes (subTileBlockBuffer->Buffer      (),
3241
0
         uncompressedBuffer->Buffer      (),
3242
0
         uncompressedBuffer->LogicalSize ());
3243
  
3244
0
  }
3245
                
3246
/******************************************************************************/
3247
3248
class dng_lzw_compressor
3249
  {
3250
  
3251
  private:
3252
  
3253
    enum
3254
      {
3255
      kResetCode = 256,
3256
      kEndCode   = 257,
3257
      kTableSize = 4096
3258
      };
3259
3260
    // Compressor nodes have two son pointers.  The low order bit of
3261
    // the next code determines which pointer is used.  This cuts the
3262
    // number of nodes searched for the next code by two on average.
3263
3264
    struct LZWCompressorNode
3265
      {
3266
      int16 final;
3267
      int16 son0;
3268
      int16 son1;
3269
      int16 brother;
3270
      };
3271
      
3272
    dng_memory_data fBuffer;
3273
3274
    LZWCompressorNode *fTable;
3275
    
3276
    uint8 *fDstPtr;
3277
    
3278
    int32 fDstCount;
3279
    
3280
    int32 fBitOffset;
3281
3282
    int32 fNextCode;
3283
    
3284
    int32 fCodeSize;
3285
    
3286
  public:
3287
  
3288
    dng_lzw_compressor ();
3289
    
3290
    void Compress (const uint8 *sPtr,
3291
             uint8 *dPtr,
3292
             uint32 sCount,
3293
             uint32 &dCount);
3294
 
3295
  private:
3296
    
3297
    void InitTable ();
3298
  
3299
    int32 SearchTable (int32 w, int32 k) const
3300
0
      {
3301
      
3302
0
      DNG_ASSERT ((w >= 0) && (w <= kTableSize),
3303
0
            "Bad w value in dng_lzw_compressor::SearchTable");
3304
      
3305
0
      int32 son0 = fTable [w] . son0;
3306
0
      int32 son1 = fTable [w] . son1;
3307
      
3308
      // Branchless version of:
3309
      // int32 code = (k & 1) ? son1 : son0;
3310
      
3311
0
      int32 code = son0 + ((-((int32) (k & 1))) & (son1 - son0));
3312
3313
0
      while (code > 0 && fTable [code].final != k)
3314
0
        {
3315
0
        code = fTable [code].brother;
3316
0
        }
3317
3318
0
      return code;
3319
3320
0
      }
3321
3322
    void AddTable (int32 w, int32 k);
3323
    
3324
    void PutCodeWord (int32 code);
3325
3326
    // Hidden copy constructor and assignment operator.
3327
  
3328
    dng_lzw_compressor (const dng_lzw_compressor &compressor);
3329
    
3330
    dng_lzw_compressor & operator= (const dng_lzw_compressor &compressor);
3331
3332
  };
3333
3334
/******************************************************************************/
3335
3336
dng_lzw_compressor::dng_lzw_compressor ()
3337
3338
0
  : fBuffer    ()
3339
  , fTable     (NULL)
3340
  , fDstPtr    (NULL)
3341
0
  , fDstCount  (0)
3342
0
  , fBitOffset (0)
3343
0
  , fNextCode  (0)
3344
0
  , fCodeSize  (0)
3345
  
3346
0
  {
3347
  
3348
0
  fBuffer.Allocate (kTableSize, sizeof (LZWCompressorNode));
3349
  
3350
0
  fTable = (LZWCompressorNode *) fBuffer.Buffer ();
3351
  
3352
0
  }
3353
3354
/******************************************************************************/
3355
3356
void dng_lzw_compressor::InitTable ()
3357
0
  {
3358
3359
0
  fCodeSize = 9;
3360
3361
0
  fNextCode = 258;
3362
    
3363
0
  LZWCompressorNode *node = &fTable [0];
3364
  
3365
0
  for (int32 code = 0; code < 256; ++code)
3366
0
    {
3367
    
3368
0
    node->final   = (int16) code;
3369
0
    node->son0    = -1;
3370
0
    node->son1    = -1;
3371
0
    node->brother = -1;
3372
    
3373
0
    node++;
3374
    
3375
0
    }
3376
    
3377
0
  }
3378
3379
/******************************************************************************/
3380
3381
void dng_lzw_compressor::AddTable (int32 w, int32 k)
3382
0
  {
3383
  
3384
0
  DNG_ASSERT ((w >= 0) && (w <= kTableSize),
3385
0
        "Bad w value in dng_lzw_compressor::AddTable");
3386
3387
0
  LZWCompressorNode *node = &fTable [w];
3388
3389
0
  int32 nextCode = fNextCode;
3390
3391
0
  DNG_ASSERT ((nextCode >= 0) && (nextCode <= kTableSize),
3392
0
        "Bad fNextCode value in dng_lzw_compressor::AddTable");
3393
  
3394
0
  LZWCompressorNode *node2 = &fTable [nextCode];
3395
  
3396
0
  fNextCode++;
3397
  
3398
0
  int32 oldSon;
3399
  
3400
0
  if( k&1 )
3401
0
    {
3402
0
    oldSon = node->son1;
3403
0
    node->son1 = (int16) nextCode;
3404
0
    }
3405
0
  else
3406
0
    {
3407
0
    oldSon = node->son0;
3408
0
    node->son0 = (int16) nextCode;
3409
0
    }
3410
  
3411
0
  node2->final   = (int16) k;
3412
0
  node2->son0    = -1;
3413
0
  node2->son1    = -1;
3414
0
  node2->brother = (int16) oldSon;
3415
  
3416
0
  if (nextCode == (1 << fCodeSize) - 1)
3417
0
    {
3418
0
    if (fCodeSize != 12)
3419
0
      fCodeSize++;
3420
0
    }
3421
    
3422
0
  }
3423
3424
/******************************************************************************/
3425
3426
void dng_lzw_compressor::PutCodeWord (int32 code)
3427
0
  {
3428
  
3429
0
  int32 bit = (int32) (fBitOffset & 7);
3430
  
3431
0
  int32 offset1 = fBitOffset >> 3;
3432
0
  int32 offset2 = (fBitOffset + fCodeSize - 1) >> 3;
3433
    
3434
0
  int32 shift1 = (fCodeSize + bit) -  8;
3435
0
  int32 shift2 = (fCodeSize + bit) - 16;
3436
  
3437
0
  uint8 byte1 = (uint8) (code >> shift1);
3438
  
3439
0
  uint8 *dstPtr1 = fDstPtr + offset1;
3440
0
  uint8 *dstPtr3 = fDstPtr + offset2;
3441
  
3442
0
  if (offset1 + 1 == offset2)
3443
0
    {
3444
    
3445
0
    uint8 byte2 = (uint8) (code << (-shift2));
3446
    
3447
0
    if (bit)
3448
0
      *dstPtr1 |= byte1;
3449
0
    else
3450
0
      *dstPtr1 = byte1;
3451
    
3452
0
    *dstPtr3 = byte2;
3453
    
3454
0
    }
3455
3456
0
  else
3457
0
    {
3458
    
3459
0
    int32 shift3 = (fCodeSize + bit) - 24;
3460
    
3461
0
    uint8 byte2 = (uint8) (code >> shift2);
3462
0
    uint8 byte3 = (uint8) (code << (-shift3));
3463
    
3464
0
    uint8 *dstPtr2 = fDstPtr + (offset1 + 1);
3465
    
3466
0
    if (bit)
3467
0
      *dstPtr1 |= byte1;
3468
0
    else
3469
0
      *dstPtr1 = byte1;
3470
    
3471
0
    *dstPtr2 = byte2;
3472
    
3473
0
    *dstPtr3 = byte3;
3474
    
3475
0
    }
3476
    
3477
0
  fBitOffset += fCodeSize;
3478
  
3479
0
  }
3480
3481
/******************************************************************************/
3482
3483
void dng_lzw_compressor::Compress (const uint8 *sPtr,
3484
                       uint8 *dPtr,
3485
                       uint32 sCount,
3486
                       uint32 &dCount)
3487
0
  {
3488
  
3489
0
  fDstPtr = dPtr;
3490
  
3491
0
  fBitOffset = 0;
3492
  
3493
0
  InitTable ();
3494
  
3495
0
  PutCodeWord (kResetCode);
3496
  
3497
0
  int32 code = -1;
3498
  
3499
0
  int32 pixel;
3500
  
3501
0
  if (sCount > 0)
3502
0
    {
3503
    
3504
0
    pixel = *sPtr;
3505
0
    sPtr = sPtr + 1;
3506
0
    code = pixel;
3507
3508
0
    sCount--;
3509
3510
0
    while (sCount--)
3511
0
      {
3512
3513
0
      pixel = *sPtr;
3514
0
      sPtr = sPtr + 1;
3515
      
3516
0
      int32 newCode = SearchTable (code, pixel);
3517
      
3518
0
      if (newCode == -1)
3519
0
        {
3520
        
3521
0
        PutCodeWord (code);
3522
        
3523
0
        if (fNextCode < 4093)
3524
0
          {
3525
0
          AddTable (code, pixel);
3526
0
          }
3527
0
        else
3528
0
          {
3529
0
          PutCodeWord (kResetCode);
3530
0
          InitTable ();
3531
0
          }
3532
          
3533
0
        code = pixel;
3534
        
3535
0
        }
3536
        
3537
0
      else
3538
0
        code = newCode;
3539
        
3540
0
      }
3541
    
3542
0
    }
3543
    
3544
0
  if (code != -1)
3545
0
    {
3546
0
    PutCodeWord (code);
3547
0
    AddTable (code, 0);
3548
0
    }
3549
    
3550
0
  PutCodeWord (kEndCode);
3551
3552
0
  dCount = (fBitOffset + 7) >> 3;
3553
3554
0
  }
3555
3556
/*****************************************************************************/
3557
3558
#if qDNGUseLibJPEG
3559
3560
/*****************************************************************************/
3561
3562
static void dng_error_exit (j_common_ptr cinfo)
3563
0
  {
3564
  
3565
  // Output message.
3566
  
3567
0
  (*cinfo->err->output_message) (cinfo);
3568
  
3569
  // Convert to a dng_exception.
3570
3571
0
  switch (cinfo->err->msg_code)
3572
0
    {
3573
    
3574
0
    case JERR_OUT_OF_MEMORY:
3575
0
      {
3576
0
      ThrowMemoryFull ();
3577
0
      break;
3578
0
      }
3579
      
3580
0
    default:
3581
0
      {
3582
0
      ThrowBadFormat ();
3583
0
      }
3584
      
3585
0
    }
3586
      
3587
0
  }
3588
3589
/*****************************************************************************/
3590
3591
static void dng_output_message (j_common_ptr cinfo)
3592
0
  {
3593
  
3594
  // Format message to string.
3595
  
3596
0
  char buffer [JMSG_LENGTH_MAX];
3597
3598
0
  (*cinfo->err->format_message) (cinfo, buffer);
3599
  
3600
  // Report the libjpeg message as a warning.
3601
  
3602
0
  ReportWarning ("libjpeg", buffer);
3603
3604
0
  }
3605
3606
/*****************************************************************************/
3607
3608
struct dng_jpeg_stream_dest
3609
  {
3610
  
3611
  struct jpeg_destination_mgr pub;
3612
  
3613
  dng_stream *fStream;
3614
  
3615
  uint8 fBuffer [4096];
3616
  
3617
  };
3618
3619
/*****************************************************************************/
3620
3621
static void dng_init_destination (j_compress_ptr cinfo)
3622
3.80k
  {
3623
  
3624
3.80k
  dng_jpeg_stream_dest *dest = (dng_jpeg_stream_dest *) cinfo->dest;
3625
3626
3.80k
  dest->pub.next_output_byte = dest->fBuffer;
3627
3.80k
  dest->pub.free_in_buffer   = sizeof (dest->fBuffer);
3628
  
3629
3.80k
  }
3630
3631
/*****************************************************************************/
3632
3633
static boolean dng_empty_output_buffer (j_compress_ptr cinfo)
3634
16.0k
  {
3635
  
3636
16.0k
  dng_jpeg_stream_dest *dest = (dng_jpeg_stream_dest *) cinfo->dest;
3637
  
3638
16.0k
  dest->fStream->Put (dest->fBuffer, sizeof (dest->fBuffer));
3639
3640
16.0k
  dest->pub.next_output_byte = dest->fBuffer;
3641
16.0k
  dest->pub.free_in_buffer   = sizeof (dest->fBuffer);
3642
3643
16.0k
  return TRUE;
3644
  
3645
16.0k
  }
3646
3647
/*****************************************************************************/
3648
3649
static void dng_term_destination (j_compress_ptr cinfo)
3650
3.80k
  {
3651
  
3652
3.80k
  dng_jpeg_stream_dest *dest = (dng_jpeg_stream_dest *) cinfo->dest;
3653
  
3654
3.80k
  uint32 datacount = sizeof (dest->fBuffer) -
3655
3.80k
             (uint32) dest->pub.free_in_buffer;
3656
  
3657
3.80k
  if (datacount)
3658
3.79k
    {
3659
3.79k
    dest->fStream->Put (dest->fBuffer, datacount);
3660
3.79k
    }
3661
3662
3.80k
  }
3663
3664
/*****************************************************************************/
3665
3666
static void jpeg_set_adobe_quality (struct jpeg_compress_struct *cinfo,
3667
                  int32 quality)
3668
3.80k
  {
3669
  
3670
  // If out of range, map to default.
3671
    
3672
3.80k
  if (quality < 0 || quality > 12)
3673
0
    {
3674
0
    quality = 10;
3675
0
    }
3676
    
3677
  // Adobe turns off chroma downsampling at high quality levels.
3678
  
3679
3.80k
  bool useChromaDownsampling = (quality <= 6);
3680
    
3681
  // Approximate mapping from Adobe quality levels to LibJPEG levels.
3682
  
3683
3.80k
  const int kLibJPEGQuality [13] =
3684
3.80k
    {
3685
3.80k
    5, 11, 23, 34, 46, 63, 76, 77, 86, 90, 94, 97, 99
3686
3.80k
    };
3687
    
3688
3.80k
  quality = kLibJPEGQuality [quality];
3689
  
3690
3.80k
  jpeg_set_quality (cinfo, quality, TRUE);
3691
  
3692
  // LibJPEG defaults to always using chroma downsampling.  Turn if off
3693
  // if we need it off to match Adobe.
3694
  
3695
3.80k
  if (!useChromaDownsampling)
3696
3.46k
    {
3697
    
3698
3.46k
    cinfo->comp_info [0].h_samp_factor = 1;
3699
3.46k
    cinfo->comp_info [0].h_samp_factor = 1;
3700
    
3701
3.46k
    }
3702
        
3703
3.80k
  }
3704
3705
/*****************************************************************************/
3706
3707
#endif
3708
3709
/*****************************************************************************/
3710
3711
void dng_image_writer::WriteData (dng_host &host,
3712
                  const dng_ifd &ifd,
3713
                      dng_stream &stream,
3714
                      dng_pixel_buffer &buffer,
3715
                  AutoPtr<dng_memory_block> &compressedBuffer)
3716
68.4k
  {
3717
  
3718
68.4k
  switch (ifd.fCompression)
3719
68.4k
    {
3720
    
3721
65.8k
    case ccUncompressed:
3722
65.8k
      {
3723
      
3724
      // Special case support for when we save to 8-bits from
3725
      // 16-bit data.
3726
      
3727
65.8k
      if (ifd.fBitsPerSample [0] == 8 && buffer.fPixelType == ttShort)
3728
0
        {
3729
        
3730
0
        uint32 count = buffer.fRowStep *
3731
0
                 buffer.fArea.H ();
3732
                 
3733
0
        const uint16 *sPtr = (const uint16 *) buffer.fData;
3734
        
3735
0
        for (uint32 j = 0; j < count; j++)
3736
0
          {
3737
          
3738
0
          stream.Put_uint8 ((uint8) sPtr [j]);
3739
          
3740
0
          }
3741
        
3742
0
        }
3743
        
3744
65.8k
      else
3745
65.8k
        {
3746
  
3747
        // Swap bytes if required.
3748
        
3749
65.8k
        if (stream.SwapBytes ())
3750
0
          {
3751
          
3752
0
          ByteSwapBuffer (host, buffer);
3753
                    
3754
0
          }
3755
      
3756
        // Write the bytes.
3757
        
3758
65.8k
        stream.Put (buffer.fData, buffer.fRowStep *
3759
65.8k
                      buffer.fArea.H () *
3760
65.8k
                      buffer.fPixelSize);
3761
                      
3762
65.8k
        }
3763
      
3764
65.8k
      break;
3765
      
3766
0
      }
3767
      
3768
0
    case ccLZW:
3769
721
    case ccDeflate:
3770
721
      {
3771
      
3772
      // Both these compression algorithms are byte based.  The floating
3773
      // point predictor already does byte ordering, so don't ever swap
3774
      // when using it.
3775
      
3776
721
      if (stream.SwapBytes () && ifd.fPredictor != cpFloatingPoint)
3777
0
        {
3778
        
3779
0
        ByteSwapBuffer (host,
3780
0
                buffer);
3781
                
3782
0
        }
3783
      
3784
      // Run the compression algorithm.
3785
        
3786
721
      uint32 sBytes = buffer.fRowStep *
3787
721
              buffer.fArea.H () *
3788
721
              buffer.fPixelSize;
3789
        
3790
721
      uint8 *sBuffer = (uint8 *) buffer.fData;
3791
        
3792
721
      uint32 dBytes = 0;
3793
        
3794
721
      uint8 *dBuffer = compressedBuffer->Buffer_uint8 ();
3795
      
3796
721
      if (ifd.fCompression == ccLZW)
3797
0
        {
3798
        
3799
0
        dng_lzw_compressor lzwCompressor;
3800
        
3801
0
        lzwCompressor.Compress (sBuffer,
3802
0
                    dBuffer,
3803
0
                    sBytes,
3804
0
                    dBytes);
3805
                    
3806
0
        }
3807
        
3808
721
      else
3809
721
        {
3810
        
3811
721
        uLongf dCount = compressedBuffer->LogicalSize ();
3812
        
3813
721
        int32 level = Z_DEFAULT_COMPRESSION;
3814
        
3815
721
        if (ifd.fCompressionQuality >= Z_BEST_SPEED &&
3816
721
          ifd.fCompressionQuality <= Z_BEST_COMPRESSION)
3817
0
          {
3818
          
3819
0
          level = ifd.fCompressionQuality;
3820
          
3821
0
          }
3822
        
3823
721
        int zResult = ::compress2 (dBuffer,
3824
721
                       &dCount,
3825
721
                       sBuffer,
3826
721
                       sBytes,
3827
721
                       level);
3828
                      
3829
721
        if (zResult != Z_OK)
3830
0
          {
3831
          
3832
0
          ThrowMemoryFull ();
3833
          
3834
0
          }
3835
3836
721
        dBytes = (uint32) dCount;
3837
        
3838
721
        }
3839
                    
3840
721
      if (dBytes > compressedBuffer->LogicalSize ())
3841
0
        {
3842
        
3843
0
        DNG_REPORT ("Compression output buffer overflow");
3844
        
3845
0
        ThrowProgramError ();
3846
        
3847
0
        }
3848
                    
3849
721
      stream.Put (dBuffer, dBytes);
3850
        
3851
721
      return;
3852
3853
0
      }
3854
      
3855
0
    case ccJPEG:
3856
0
      {
3857
      
3858
0
      dng_pixel_buffer temp (buffer);
3859
        
3860
0
      if (buffer.fPixelType == ttByte)
3861
0
        {
3862
        
3863
        // The lossless JPEG encoder needs 16-bit data, so if we are
3864
        // are saving 8 bit data, we need to pad it out to 16-bits.
3865
        
3866
0
        temp.fData = compressedBuffer->Buffer ();
3867
        
3868
0
        temp.fPixelType = ttShort;
3869
0
        temp.fPixelSize = 2;
3870
        
3871
0
        temp.CopyArea (buffer,
3872
0
                 buffer.fArea,
3873
0
                 buffer.fPlane,
3874
0
                 buffer.fPlanes);
3875
        
3876
0
        }
3877
        
3878
0
      EncodeLosslessJPEG ((const uint16 *) temp.fData,
3879
0
                temp.fArea.H (),
3880
0
                temp.fArea.W (),
3881
0
                temp.fPlanes,
3882
0
                ifd.fBitsPerSample [0],
3883
0
                temp.fRowStep,
3884
0
                temp.fColStep,
3885
0
                stream);
3886
                    
3887
0
      break;
3888
      
3889
0
      }
3890
      
3891
0
    #if qDNGUseLibJPEG
3892
    
3893
1.88k
    case ccLossyJPEG:
3894
1.88k
      {
3895
      
3896
1.88k
      struct jpeg_compress_struct cinfo;
3897
  
3898
      // Setup the error manager.
3899
      
3900
1.88k
      struct jpeg_error_mgr jerr;
3901
3902
1.88k
      cinfo.err = jpeg_std_error (&jerr);
3903
      
3904
1.88k
      jerr.error_exit     = dng_error_exit;
3905
1.88k
      jerr.output_message = dng_output_message;
3906
  
3907
1.88k
      try
3908
1.88k
        {
3909
        
3910
        // Create the compression context.
3911
3912
1.88k
        jpeg_create_compress (&cinfo);
3913
        
3914
        // Setup the destination manager to write to stream.
3915
        
3916
1.88k
        dng_jpeg_stream_dest dest;
3917
        
3918
1.88k
        dest.fStream = &stream;
3919
        
3920
1.88k
        dest.pub.init_destination    = dng_init_destination;
3921
1.88k
        dest.pub.empty_output_buffer = dng_empty_output_buffer;
3922
1.88k
        dest.pub.term_destination    = dng_term_destination;
3923
        
3924
1.88k
        cinfo.dest = &dest.pub;
3925
        
3926
        // Setup basic image info.
3927
        
3928
1.88k
        cinfo.image_width      = buffer.fArea.W ();
3929
1.88k
        cinfo.image_height     = buffer.fArea.H ();
3930
1.88k
        cinfo.input_components = buffer.fPlanes;
3931
        
3932
1.88k
        switch (buffer.fPlanes)
3933
1.88k
          {
3934
          
3935
1.00k
          case 1:
3936
1.00k
            cinfo.in_color_space = JCS_GRAYSCALE;
3937
1.00k
            break;
3938
            
3939
882
          case 3:
3940
882
            cinfo.in_color_space = JCS_RGB;
3941
882
            break;
3942
            
3943
0
          case 4:
3944
0
            cinfo.in_color_space = JCS_CMYK;
3945
0
            break;
3946
            
3947
0
          default:
3948
0
            ThrowProgramError ();
3949
            
3950
1.88k
          }
3951
          
3952
        // Setup the compression parameters.
3953
3954
1.88k
        jpeg_set_defaults (&cinfo);
3955
        
3956
1.88k
        jpeg_set_adobe_quality (&cinfo, ifd.fCompressionQuality);
3957
        
3958
        // Write the JPEG header.
3959
        
3960
1.88k
        jpeg_start_compress (&cinfo, TRUE);
3961
        
3962
        // Write the scanlines.
3963
        
3964
398k
        for (int32 row = buffer.fArea.t; row < buffer.fArea.b; row++)
3965
396k
          {
3966
          
3967
396k
          uint8 *sampArray [1];
3968
    
3969
396k
          sampArray [0] = buffer.DirtyPixel_uint8 (row,
3970
396k
                               buffer.fArea.l,
3971
396k
                               0);
3972
3973
396k
          jpeg_write_scanlines (&cinfo, sampArray, 1);
3974
          
3975
396k
          }
3976
3977
        // Cleanup.
3978
          
3979
1.88k
        jpeg_finish_compress (&cinfo);
3980
3981
1.88k
        jpeg_destroy_compress (&cinfo);
3982
          
3983
1.88k
        }
3984
        
3985
1.88k
      catch (...)
3986
1.88k
        {
3987
        
3988
0
        jpeg_destroy_compress (&cinfo);
3989
        
3990
0
        throw;
3991
        
3992
0
        }
3993
        
3994
1.88k
      return;
3995
      
3996
1.88k
      }
3997
      
3998
0
    #endif
3999
      
4000
0
    default:
4001
0
      {
4002
      
4003
0
      ThrowProgramError ();
4004
      
4005
0
      }
4006
      
4007
68.4k
    }
4008
  
4009
68.4k
  }
4010
                
4011
/******************************************************************************/
4012
4013
void dng_image_writer::EncodeJPEGPreview (dng_host &host,
4014
                      const dng_image &image,
4015
                      dng_jpeg_preview &preview,
4016
                      int32 quality)
4017
1.91k
  {
4018
  
4019
1.91k
  #if qDNGUseLibJPEG
4020
    
4021
1.91k
  dng_memory_stream stream (host.Allocator ());
4022
  
4023
1.91k
  struct jpeg_compress_struct cinfo;
4024
4025
  // Setup the error manager.
4026
  
4027
1.91k
  struct jpeg_error_mgr jerr;
4028
4029
1.91k
  cinfo.err = jpeg_std_error (&jerr);
4030
  
4031
1.91k
  jerr.error_exit     = dng_error_exit;
4032
1.91k
  jerr.output_message = dng_output_message;
4033
4034
1.91k
  try
4035
1.91k
    {
4036
    
4037
    // Create the compression context.
4038
4039
1.91k
    jpeg_create_compress (&cinfo);
4040
    
4041
    // Setup the destination manager to write to stream.
4042
    
4043
1.91k
    dng_jpeg_stream_dest dest;
4044
    
4045
1.91k
    dest.fStream = &stream;
4046
    
4047
1.91k
    dest.pub.init_destination    = dng_init_destination;
4048
1.91k
    dest.pub.empty_output_buffer = dng_empty_output_buffer;
4049
1.91k
    dest.pub.term_destination    = dng_term_destination;
4050
    
4051
1.91k
    cinfo.dest = &dest.pub;
4052
    
4053
    // Setup basic image info.
4054
    
4055
1.91k
    cinfo.image_width      = image.Bounds ().W ();
4056
1.91k
    cinfo.image_height     = image.Bounds ().H ();
4057
1.91k
    cinfo.input_components = image.Planes ();
4058
    
4059
1.91k
    switch (image.Planes ())
4060
1.91k
      {
4061
      
4062
1.10k
      case 1:
4063
1.10k
        cinfo.in_color_space = JCS_GRAYSCALE;
4064
1.10k
        break;
4065
        
4066
814
      case 3:
4067
814
        cinfo.in_color_space = JCS_RGB;
4068
814
        break;
4069
        
4070
0
      default:
4071
0
        ThrowProgramError ();
4072
        
4073
1.91k
      }
4074
      
4075
    // Setup the compression parameters.
4076
4077
1.91k
    jpeg_set_defaults (&cinfo);
4078
    
4079
1.91k
    jpeg_set_adobe_quality (&cinfo, quality);
4080
    
4081
    // Find some preview information based on the compression settings.
4082
    
4083
1.91k
    preview.fPreviewSize = image.Size ();
4084
  
4085
1.91k
    if (image.Planes () == 1)
4086
1.10k
      {
4087
      
4088
1.10k
      preview.fPhotometricInterpretation = piBlackIsZero;
4089
      
4090
1.10k
      }
4091
      
4092
814
    else
4093
814
      {
4094
      
4095
814
      preview.fPhotometricInterpretation = piYCbCr;
4096
      
4097
814
      preview.fYCbCrSubSampling.h  = cinfo.comp_info [0].h_samp_factor;
4098
814
      preview.fYCbCrSubSampling.v  = cinfo.comp_info [0].v_samp_factor;
4099
      
4100
814
      }
4101
    
4102
    // Write the JPEG header.
4103
    
4104
1.91k
    jpeg_start_compress (&cinfo, TRUE);
4105
    
4106
    // Write the scanlines.
4107
    
4108
1.91k
    dng_pixel_buffer buffer (image.Bounds (), 0, image.Planes (), ttByte,
4109
1.91k
       pcInterleaved, NULL);
4110
    
4111
1.91k
    AutoPtr<dng_memory_block> bufferData (host.Allocate (buffer.fRowStep));
4112
    
4113
1.91k
    buffer.fData = bufferData->Buffer ();
4114
    
4115
302k
    for (uint32 row = 0; row < cinfo.image_height; row++)
4116
300k
      {
4117
      
4118
300k
      buffer.fArea.t = row;
4119
300k
      buffer.fArea.b = row + 1;
4120
      
4121
300k
      image.Get (buffer);
4122
      
4123
300k
      uint8 *sampArray [1];
4124
4125
300k
      sampArray [0] = buffer.DirtyPixel_uint8 (row,
4126
300k
                           buffer.fArea.l,
4127
300k
                           0);
4128
4129
300k
      jpeg_write_scanlines (&cinfo, sampArray, 1);
4130
      
4131
300k
      }
4132
4133
    // Cleanup.
4134
      
4135
1.91k
    jpeg_finish_compress (&cinfo);
4136
4137
1.91k
    jpeg_destroy_compress (&cinfo);
4138
      
4139
1.91k
    }
4140
    
4141
1.91k
  catch (...)
4142
1.91k
    {
4143
    
4144
0
    jpeg_destroy_compress (&cinfo);
4145
    
4146
0
    throw;
4147
    
4148
0
    }
4149
           
4150
1.91k
  preview.fCompressedData.Reset (stream.AsMemoryBlock (host.Allocator ()));
4151
4152
  #else
4153
  
4154
  (void) host;
4155
  (void) image;
4156
  (void) preview;
4157
  (void) quality;
4158
  
4159
  ThrowProgramError ("No JPEG encoder");
4160
  
4161
  #endif
4162
    
4163
1.91k
  }
4164
                
4165
/*****************************************************************************/
4166
4167
void dng_image_writer::WriteTile (dng_host &host,
4168
                      const dng_ifd &ifd,
4169
                      dng_stream &stream,
4170
                      const dng_image &image,
4171
                      const dng_rect &tileArea,
4172
                      uint32 fakeChannels,
4173
                  AutoPtr<dng_memory_block> &compressedBuffer,
4174
                  AutoPtr<dng_memory_block> &uncompressedBuffer,
4175
                  AutoPtr<dng_memory_block> &subTileBlockBuffer,
4176
                  AutoPtr<dng_memory_block> &tempBuffer)
4177
68.4k
  {
4178
  
4179
  // Create pixel buffer to hold uncompressed tile.
4180
  
4181
68.4k
  dng_pixel_buffer buffer (tileArea, 0, ifd.fSamplesPerPixel,
4182
68.4k
     image.PixelType(), pcInterleaved, uncompressedBuffer->Buffer());
4183
  
4184
  // Get the uncompressed data.
4185
  
4186
68.4k
  image.Get (buffer, dng_image::edge_zero);
4187
  
4188
  // Deal with sub-tile blocks.
4189
  
4190
68.4k
  if (ifd.fSubTileBlockRows > 1)
4191
0
    {
4192
    
4193
0
    ReorderSubTileBlocks (ifd,
4194
0
                buffer,
4195
0
                uncompressedBuffer,
4196
0
                subTileBlockBuffer);
4197
    
4198
0
    }
4199
    
4200
  // Floating point depth conversion.
4201
  
4202
68.4k
  if (ifd.fSampleFormat [0] == sfFloatingPoint)
4203
7.93k
    {
4204
    
4205
7.93k
    if (ifd.fBitsPerSample [0] == 16)
4206
721
      {
4207
      
4208
721
      uint32 *srcPtr = (uint32 *) buffer.fData;
4209
721
      uint16 *dstPtr = (uint16 *) buffer.fData;
4210
      
4211
721
      uint32 pixels = tileArea.W () * tileArea.H () * buffer.fPlanes;
4212
      
4213
15.1M
      for (uint32 j = 0; j < pixels; j++)
4214
15.1M
        {
4215
        
4216
15.1M
        dstPtr [j] = DNG_FloatToHalf (srcPtr [j]);
4217
        
4218
15.1M
        }
4219
        
4220
721
      buffer.fPixelSize = 2;
4221
      
4222
721
      }
4223
      
4224
7.93k
    if (ifd.fBitsPerSample [0] == 24)
4225
0
      {
4226
      
4227
0
      uint32 *srcPtr = (uint32 *) buffer.fData;
4228
0
      uint8  *dstPtr = (uint8  *) buffer.fData;
4229
      
4230
0
      uint32 pixels = tileArea.W () * tileArea.H () * buffer.fPlanes;
4231
      
4232
0
      if (stream.BigEndian () || ifd.fPredictor == cpFloatingPoint   ||
4233
0
                     ifd.fPredictor == cpFloatingPointX2 ||
4234
0
                     ifd.fPredictor == cpFloatingPointX4)
4235
0
        {
4236
      
4237
0
        for (uint32 j = 0; j < pixels; j++)
4238
0
          {
4239
          
4240
0
          DNG_FloatToFP24 (srcPtr [j], dstPtr);
4241
          
4242
0
          dstPtr += 3;
4243
          
4244
0
          }
4245
          
4246
0
        }
4247
        
4248
0
      else
4249
0
        {
4250
      
4251
0
        for (uint32 j = 0; j < pixels; j++)
4252
0
          {
4253
          
4254
0
          uint8 output [3];
4255
          
4256
0
          DNG_FloatToFP24 (srcPtr [j], output);
4257
          
4258
0
          dstPtr [0] = output [2];
4259
0
          dstPtr [1] = output [1];
4260
0
          dstPtr [2] = output [0];
4261
          
4262
0
          dstPtr += 3;
4263
          
4264
0
          }
4265
          
4266
0
        }
4267
        
4268
0
      buffer.fPixelSize = 3;
4269
      
4270
0
      }
4271
    
4272
7.93k
    }
4273
  
4274
  // Run predictor.
4275
  
4276
68.4k
  EncodePredictor (host,
4277
68.4k
           ifd,
4278
68.4k
           buffer,
4279
68.4k
           tempBuffer);
4280
    
4281
  // Adjust pixel buffer for fake channels.
4282
  
4283
68.4k
  if (fakeChannels > 1)
4284
0
    {
4285
    
4286
0
    buffer.fPlanes  *= fakeChannels;
4287
0
    buffer.fColStep *= fakeChannels;
4288
    
4289
0
    buffer.fArea.r = buffer.fArea.l + (buffer.fArea.W () / fakeChannels);
4290
    
4291
0
    }
4292
    
4293
  // Compress (if required) and write out the data.
4294
  
4295
68.4k
  WriteData (host,
4296
68.4k
         ifd,
4297
68.4k
         stream,
4298
68.4k
         buffer,
4299
68.4k
         compressedBuffer);
4300
         
4301
68.4k
  }
4302
4303
/*****************************************************************************/
4304
4305
class dng_write_tiles_task : public dng_area_task
4306
  {
4307
  
4308
  private:
4309
  
4310
    dng_image_writer &fImageWriter;
4311
    
4312
    dng_host &fHost;
4313
    
4314
    const dng_ifd &fIFD;
4315
    
4316
    dng_basic_tag_set &fBasic;
4317
    
4318
    dng_stream &fStream;
4319
    
4320
    const dng_image &fImage;
4321
    
4322
    uint32 fFakeChannels;
4323
    
4324
    uint32 fTilesDown;
4325
    
4326
    uint32 fTilesAcross;
4327
    
4328
    uint32 fCompressedSize;
4329
    
4330
    uint32 fUncompressedSize;
4331
    
4332
    dng_mutex fMutex1;
4333
    
4334
    uint32 fNextTileIndex;
4335
    
4336
    dng_mutex fMutex2;
4337
    
4338
    dng_condition fCondition;
4339
    
4340
    bool fTaskFailed;
4341
4342
    uint32 fWriteTileIndex;
4343
    
4344
  public:
4345
  
4346
    dng_write_tiles_task (dng_image_writer &imageWriter,
4347
                dng_host &host,
4348
                const dng_ifd &ifd,
4349
                dng_basic_tag_set &basic,
4350
                dng_stream &stream,
4351
                const dng_image &image,
4352
                uint32 fakeChannels,
4353
                uint32 tilesDown,
4354
                uint32 tilesAcross,
4355
                uint32 compressedSize,
4356
                uint32 uncompressedSize)
4357
    
4358
0
      : fImageWriter      (imageWriter)
4359
0
      , fHost         (host)
4360
0
      , fIFD          (ifd)
4361
0
      , fBasic        (basic)
4362
0
      , fStream         (stream)
4363
0
      , fImage          (image)
4364
0
      , fFakeChannels   (fakeChannels)
4365
0
      , fTilesDown        (tilesDown)
4366
0
      , fTilesAcross    (tilesAcross)
4367
0
      , fCompressedSize   (compressedSize)
4368
0
      , fUncompressedSize (uncompressedSize)
4369
0
      , fMutex1       ("dng_write_tiles_task_1")
4370
0
      , fNextTileIndex    (0)
4371
0
      , fMutex2       ("dng_write_tiles_task_2")
4372
0
      , fCondition      ()
4373
0
      , fTaskFailed     (false)
4374
0
      , fWriteTileIndex   (0)
4375
      
4376
0
      {
4377
      
4378
0
      fMinTaskArea = 16 * 16;
4379
0
      fUnitCell    = dng_point (16, 16);
4380
0
      fMaxTileSize = dng_point (16, 16);
4381
      
4382
0
      }
4383
  
4384
    void Process (uint32 /* threadIndex */,
4385
            const dng_rect & /* tile */,
4386
            dng_abort_sniffer *sniffer)
4387
0
      {
4388
      
4389
0
      try
4390
0
        {
4391
      
4392
0
        AutoPtr<dng_memory_block> compressedBuffer;
4393
0
        AutoPtr<dng_memory_block> uncompressedBuffer;
4394
0
        AutoPtr<dng_memory_block> subTileBlockBuffer;
4395
0
        AutoPtr<dng_memory_block> tempBuffer;
4396
        
4397
0
        if (fCompressedSize)
4398
0
          {
4399
0
          compressedBuffer.Reset (fHost.Allocate (fCompressedSize));
4400
0
          }
4401
        
4402
0
        if (fUncompressedSize)
4403
0
          {
4404
0
          uncompressedBuffer.Reset (fHost.Allocate (fUncompressedSize));
4405
0
          }
4406
        
4407
0
        if (fIFD.fSubTileBlockRows > 1 && fUncompressedSize)
4408
0
          {
4409
0
          subTileBlockBuffer.Reset (fHost.Allocate (fUncompressedSize));
4410
0
          }
4411
        
4412
0
        while (true)
4413
0
          {
4414
          
4415
          // Find tile index to compress.
4416
          
4417
0
          uint32 tileIndex;
4418
          
4419
0
            {
4420
            
4421
0
            dng_lock_mutex lock (&fMutex1);
4422
            
4423
0
            if (fNextTileIndex == fTilesDown * fTilesAcross)
4424
0
              {
4425
0
              return;
4426
0
              }
4427
              
4428
0
            tileIndex = fNextTileIndex++;
4429
            
4430
0
            }
4431
            
4432
0
          dng_abort_sniffer::SniffForAbort (sniffer);
4433
          
4434
          // Compress tile.
4435
          
4436
0
          uint32 rowIndex = tileIndex / fTilesAcross;
4437
          
4438
0
          uint32 colIndex = tileIndex - rowIndex * fTilesAcross;
4439
          
4440
0
          dng_rect tileArea = fIFD.TileArea (rowIndex, colIndex);
4441
          
4442
0
          dng_memory_stream tileStream (fHost.Allocator ());
4443
                       
4444
0
          tileStream.SetLittleEndian (fStream.LittleEndian ());
4445
          
4446
0
          dng_host host (&fHost.Allocator (),
4447
0
                   sniffer);
4448
                
4449
0
          fImageWriter.WriteTile (host,
4450
0
                      fIFD,
4451
0
                      tileStream,
4452
0
                      fImage,
4453
0
                      tileArea,
4454
0
                      fFakeChannels,
4455
0
                      compressedBuffer,
4456
0
                      uncompressedBuffer,
4457
0
                      subTileBlockBuffer,
4458
0
                      tempBuffer);
4459
                      
4460
0
          tileStream.Flush ();
4461
                      
4462
0
          uint32 tileByteCount = (uint32) tileStream.Length ();
4463
          
4464
0
          tileStream.SetReadPosition (0);
4465
          
4466
          // Wait until it is our turn to write tile.
4467
4468
0
            {
4469
          
4470
0
            dng_lock_mutex lock (&fMutex2);
4471
          
4472
0
            while (!fTaskFailed &&
4473
0
                 fWriteTileIndex != tileIndex)
4474
0
              {
4475
4476
0
              fCondition.Wait (fMutex2);
4477
              
4478
0
              }
4479
4480
            // If the task failed in another thread, that thread already threw an exception.
4481
4482
0
            if (fTaskFailed)
4483
0
              return;
4484
4485
0
            }           
4486
          
4487
0
          dng_abort_sniffer::SniffForAbort (sniffer);
4488
          
4489
          // Remember this offset.
4490
        
4491
0
          uint32 tileOffset = (uint32) fStream.Position ();
4492
        
4493
0
          fBasic.SetTileOffset (tileIndex, tileOffset);
4494
            
4495
          // Copy tile stream for tile into main stream.
4496
              
4497
0
          tileStream.CopyToStream (fStream, tileByteCount);
4498
              
4499
          // Update tile count.
4500
            
4501
0
          fBasic.SetTileByteCount (tileIndex, tileByteCount);
4502
          
4503
          // Keep the tiles on even byte offsets.
4504
                             
4505
0
          if (tileByteCount & 1)
4506
0
            {
4507
0
            fStream.Put_uint8 (0);
4508
0
            }
4509
              
4510
          // Let other threads know it is safe to write to stream.
4511
          
4512
0
            {
4513
            
4514
0
            dng_lock_mutex lock (&fMutex2);
4515
            
4516
            // If the task failed in another thread, that thread already threw an exception.
4517
4518
0
            if (fTaskFailed)
4519
0
              return;
4520
4521
0
            fWriteTileIndex++;
4522
            
4523
0
            fCondition.Broadcast ();
4524
            
4525
0
            }
4526
            
4527
0
          }
4528
          
4529
0
        }
4530
        
4531
0
      catch (...)
4532
0
        {
4533
        
4534
        // If first to fail, wake up any threads waiting on condition.
4535
        
4536
0
        bool needBroadcast = false;
4537
4538
0
          {
4539
          
4540
0
          dng_lock_mutex lock (&fMutex2);
4541
4542
0
          needBroadcast = !fTaskFailed;
4543
0
          fTaskFailed = true;
4544
          
4545
0
          }
4546
4547
0
        if (needBroadcast)
4548
0
          fCondition.Broadcast ();
4549
        
4550
0
        throw;
4551
        
4552
0
        }
4553
      
4554
0
      }
4555
    
4556
  private:
4557
4558
    // Hidden copy constructor and assignment operator.
4559
4560
    dng_write_tiles_task (const dng_write_tiles_task &);
4561
4562
    dng_write_tiles_task & operator= (const dng_write_tiles_task &);
4563
    
4564
  };
4565
4566
/*****************************************************************************/
4567
4568
void dng_image_writer::WriteImage (dng_host &host,
4569
                       const dng_ifd &ifd,
4570
                       dng_basic_tag_set &basic,
4571
                       dng_stream &stream,
4572
                       const dng_image &image,
4573
                       uint32 fakeChannels)
4574
9.74k
  {
4575
  
4576
  // Deal with row interleaved images.
4577
  
4578
9.74k
  if (ifd.fRowInterleaveFactor > 1 &&
4579
9.74k
    ifd.fRowInterleaveFactor < ifd.fImageLength)
4580
0
    {
4581
    
4582
0
    dng_ifd tempIFD (ifd);
4583
    
4584
0
    tempIFD.fRowInterleaveFactor = 1;
4585
    
4586
0
    dng_row_interleaved_image tempImage (*((dng_image *) &image),
4587
0
                       ifd.fRowInterleaveFactor);
4588
    
4589
0
    WriteImage (host,
4590
0
          tempIFD,
4591
0
          basic,
4592
0
          stream,
4593
0
          tempImage,
4594
0
          fakeChannels);
4595
        
4596
0
    return;
4597
    
4598
0
    }
4599
  
4600
  // Compute basic information.
4601
  
4602
9.74k
  uint32 bytesPerSample = TagTypeSize (image.PixelType ());
4603
  
4604
9.74k
  uint32 bytesPerPixel = SafeUint32Mult (ifd.fSamplesPerPixel,
4605
9.74k
                       bytesPerSample);
4606
  
4607
9.74k
  uint32 tileRowBytes = SafeUint32Mult (ifd.fTileWidth, bytesPerPixel);
4608
  
4609
  // If we can compute the number of bytes needed to store the
4610
  // data, we can split the write for each tile into sub-tiles.
4611
  
4612
9.74k
  uint32 subTileLength = ifd.fTileLength;
4613
  
4614
9.74k
  if (ifd.TileByteCount (ifd.TileArea (0, 0)) != 0)
4615
9.25k
    {
4616
    
4617
9.25k
    subTileLength = Pin_uint32 (ifd.fSubTileBlockRows,
4618
9.25k
                  kImageBufferSize / tileRowBytes, 
4619
9.25k
                  ifd.fTileLength);
4620
          
4621
    // Don't split sub-tiles across subTileBlocks.
4622
    
4623
9.25k
    subTileLength = subTileLength / ifd.fSubTileBlockRows
4624
9.25k
                    * ifd.fSubTileBlockRows;
4625
                  
4626
9.25k
    }
4627
    
4628
  // Find size of uncompressed buffer.
4629
  
4630
9.74k
  uint32 uncompressedSize = SafeUint32Mult(subTileLength, tileRowBytes);
4631
  
4632
  // Find size of compressed buffer, if required.
4633
  
4634
9.74k
  uint32 compressedSize = CompressedBufferSize (ifd, uncompressedSize);
4635
      
4636
  // See if we can do this write using multiple threads.
4637
  
4638
9.74k
  uint32 tilesAcross = ifd.TilesAcross ();
4639
9.74k
  uint32 tilesDown   = ifd.TilesDown   ();
4640
                 
4641
9.74k
  bool useMultipleThreads = (tilesDown * tilesAcross >= 2) &&
4642
9.74k
                (host.PerformAreaTaskThreads () > 1) &&
4643
9.74k
                (subTileLength == ifd.fTileLength) &&
4644
9.74k
                (ifd.fCompression != ccUncompressed);
4645
  
4646
    
4647
#if qImagecore
4648
  useMultipleThreads = false; 
4649
#endif
4650
    
4651
9.74k
  if (useMultipleThreads)
4652
0
    {
4653
    
4654
0
    uint32 threadCount = Min_uint32 (tilesDown * tilesAcross,
4655
0
                     host.PerformAreaTaskThreads ());
4656
                     
4657
0
    dng_write_tiles_task task (*this,
4658
0
                   host,
4659
0
                   ifd,
4660
0
                   basic,
4661
0
                   stream,
4662
0
                   image,
4663
0
                   fakeChannels,
4664
0
                   tilesDown,
4665
0
                   tilesAcross,
4666
0
                   compressedSize,
4667
0
                   uncompressedSize);
4668
                  
4669
0
    host.PerformAreaTask (task,
4670
0
                dng_rect (0, 0, 16, 16 * threadCount));
4671
    
4672
0
    }
4673
    
4674
9.74k
  else
4675
9.74k
    {
4676
                  
4677
9.74k
    AutoPtr<dng_memory_block> compressedBuffer;
4678
9.74k
    AutoPtr<dng_memory_block> uncompressedBuffer;
4679
9.74k
    AutoPtr<dng_memory_block> subTileBlockBuffer;
4680
9.74k
    AutoPtr<dng_memory_block> tempBuffer;
4681
    
4682
9.74k
    if (compressedSize)
4683
495
      {
4684
495
      compressedBuffer.Reset (host.Allocate (compressedSize));
4685
495
      }
4686
    
4687
9.74k
    if (uncompressedSize)
4688
9.74k
      {
4689
9.74k
      uncompressedBuffer.Reset (host.Allocate (uncompressedSize));
4690
9.74k
      }
4691
    
4692
9.74k
    if (ifd.fSubTileBlockRows > 1 && uncompressedSize)
4693
0
      {
4694
0
      subTileBlockBuffer.Reset (host.Allocate (uncompressedSize));
4695
0
      }
4696
        
4697
    // Write out each tile.
4698
    
4699
9.74k
    uint32 tileIndex = 0;
4700
    
4701
19.4k
    for (uint32 rowIndex = 0; rowIndex < tilesDown; rowIndex++)
4702
9.74k
      {
4703
      
4704
19.7k
      for (uint32 colIndex = 0; colIndex < tilesAcross; colIndex++)
4705
9.97k
        {
4706
        
4707
        // Remember this offset.
4708
        
4709
9.97k
        uint32 tileOffset = (uint32) stream.Position ();
4710
      
4711
9.97k
        basic.SetTileOffset (tileIndex, tileOffset);
4712
        
4713
        // Split tile into sub-tiles if possible.
4714
        
4715
9.97k
        dng_rect tileArea = ifd.TileArea (rowIndex, colIndex);
4716
        
4717
9.97k
        uint32 subTileCount = (tileArea.H () + subTileLength - 1) /
4718
9.97k
                    subTileLength;
4719
                    
4720
76.5k
        for (uint32 subIndex = 0; subIndex < subTileCount; subIndex++)
4721
66.5k
          {
4722
          
4723
66.5k
          host.SniffForAbort ();
4724
        
4725
66.5k
          dng_rect subArea (tileArea);
4726
          
4727
66.5k
          subArea.t = tileArea.t + subIndex * subTileLength;
4728
          
4729
66.5k
          subArea.b = Min_int32 (subArea.t + subTileLength,
4730
66.5k
                       tileArea.b);
4731
                       
4732
          // Write the sub-tile.
4733
          
4734
66.5k
          WriteTile (host,
4735
66.5k
                 ifd,
4736
66.5k
                 stream,
4737
66.5k
                 image,
4738
66.5k
                 subArea,
4739
66.5k
                 fakeChannels,
4740
66.5k
                 compressedBuffer,
4741
66.5k
                 uncompressedBuffer,
4742
66.5k
                 subTileBlockBuffer,
4743
66.5k
                 tempBuffer);
4744
                 
4745
66.5k
          }
4746
          
4747
        // Update tile count.
4748
          
4749
9.97k
        uint32 tileByteCount = (uint32) stream.Position () - tileOffset;
4750
          
4751
9.97k
        basic.SetTileByteCount (tileIndex, tileByteCount);
4752
        
4753
9.97k
        tileIndex++;
4754
        
4755
        // Keep the tiles on even byte offsets.
4756
                           
4757
9.97k
        if (tileByteCount & 1)
4758
1.02k
          {
4759
1.02k
          stream.Put_uint8 (0);
4760
1.02k
          }
4761
          
4762
9.97k
        }
4763
4764
9.74k
      }
4765
      
4766
9.74k
    }
4767
    
4768
9.74k
  }
4769
4770
/*****************************************************************************/
4771
4772
#if qDNGUseXMP
4773
4774
static void CopyString (const dng_xmp &oldXMP,
4775
            dng_xmp &newXMP,
4776
            const char *ns,
4777
            const char *path,
4778
            dng_string *exif = NULL)
4779
  {
4780
  
4781
  dng_string s;
4782
  
4783
  if (oldXMP.GetString (ns, path, s))
4784
    {
4785
    
4786
    if (s.NotEmpty ())
4787
      {
4788
      
4789
      newXMP.SetString (ns, path, s);
4790
      
4791
      if (exif)
4792
        {
4793
        
4794
        *exif = s;
4795
        
4796
        }
4797
      
4798
      }
4799
      
4800
    }
4801
  
4802
  }
4803
                 
4804
/*****************************************************************************/
4805
4806
static void CopyStringList (const dng_xmp &oldXMP,
4807
              dng_xmp &newXMP,
4808
              const char *ns,
4809
              const char *path,
4810
              bool isBag)
4811
  {
4812
  
4813
  dng_string_list list;
4814
  
4815
  if (oldXMP.GetStringList (ns, path, list))
4816
    {
4817
    
4818
    if (list.Count ())
4819
      {
4820
      
4821
      newXMP.SetStringList (ns, path, list, isBag);
4822
            
4823
      }
4824
      
4825
    }
4826
  
4827
  }
4828
                 
4829
/*****************************************************************************/
4830
4831
static void CopyAltLangDefault (const dng_xmp &oldXMP,
4832
                dng_xmp &newXMP,
4833
                const char *ns,
4834
                const char *path,
4835
                dng_string *exif = NULL)
4836
  {
4837
  
4838
  dng_string s;
4839
  
4840
  if (oldXMP.GetAltLangDefault (ns, path, s))
4841
    {
4842
    
4843
    if (s.NotEmpty ())
4844
      {
4845
      
4846
      newXMP.SetAltLangDefault (ns, path, s);
4847
      
4848
      if (exif)
4849
        {
4850
        
4851
        *exif = s;
4852
        
4853
        }
4854
      
4855
      }
4856
      
4857
    }
4858
  
4859
  }
4860
                 
4861
/*****************************************************************************/
4862
4863
static void CopyStructField (const dng_xmp &oldXMP,
4864
               dng_xmp &newXMP,
4865
               const char *ns,
4866
               const char *path,
4867
               const char *field)
4868
  {
4869
  
4870
  dng_string s;
4871
  
4872
  if (oldXMP.GetStructField (ns, path, ns, field, s))
4873
    {
4874
    
4875
    if (s.NotEmpty ())
4876
      {
4877
      
4878
      newXMP.SetStructField (ns, path, ns, field, s);
4879
      
4880
      }
4881
      
4882
    }
4883
  
4884
  }
4885
4886
/*****************************************************************************/
4887
4888
static void CopyBoolean (const dng_xmp &oldXMP,
4889
             dng_xmp &newXMP,
4890
             const char *ns,
4891
             const char *path)
4892
  {
4893
  
4894
  bool b;
4895
  
4896
  if (oldXMP.GetBoolean (ns, path, b))
4897
    {
4898
    
4899
    newXMP.SetBoolean (ns, path, b);
4900
            
4901
    }
4902
  
4903
  }
4904
  
4905
#endif
4906
               
4907
/*****************************************************************************/
4908
4909
void dng_image_writer::CleanUpMetadata (dng_host &host,
4910
                    dng_metadata &metadata,
4911
                    dng_metadata_subset metadataSubset,
4912
                    const char *dstMIMI,
4913
                    const char *software)
4914
4.01k
  {
4915
  
4916
  #if qDNGUseXMP
4917
  
4918
  if (metadata.GetXMP () && metadata.GetExif ())
4919
    {
4920
    
4921
    dng_xmp  &newXMP  (*metadata.GetXMP  ());
4922
    dng_exif &newEXIF (*metadata.GetExif ());
4923
    
4924
    // Update software tag.
4925
    
4926
    if (software)
4927
      {
4928
  
4929
      newEXIF.fSoftware.Set (software);
4930
      
4931
      newXMP.Set (XMP_NS_XAP,
4932
            "CreatorTool",
4933
            software);
4934
      
4935
      }
4936
    
4937
    #if qDNGXMPDocOps
4938
    
4939
    newXMP.DocOpsPrepareForSave (metadata.SourceMIMI ().Get (),
4940
                   dstMIMI);
4941
                          
4942
    #else
4943
    
4944
    metadata.UpdateDateTimeToNow ();
4945
    
4946
    #endif
4947
    
4948
    // Update EXIF version to at least 2.3 so all the exif tags
4949
    // can be written.
4950
    
4951
    if (newEXIF.fExifVersion < DNG_CHAR4 ('0','2','3','0'))
4952
      {
4953
    
4954
      newEXIF.fExifVersion = DNG_CHAR4 ('0','2','3','0');
4955
      
4956
      newXMP.Set (XMP_NS_EXIF, "ExifVersion", "0230");
4957
      
4958
      }
4959
      
4960
    // Resync EXIF, remove EXIF tags from XMP.
4961
  
4962
    newXMP.SyncExif (newEXIF,
4963
             metadata.GetOriginalExif (),
4964
             false,
4965
             true);
4966
                    
4967
    // Deal with ImageIngesterPro bug.  This program is adding lots of
4968
    // empty metadata strings into the XMP, which is screwing up Adobe CS4.
4969
    // We are saving a new file, so this is a chance to clean up this mess.
4970
    
4971
    newXMP.RemoveEmptyStringsAndArrays (XMP_NS_DC);
4972
    newXMP.RemoveEmptyStringsAndArrays (XMP_NS_XAP);
4973
    newXMP.RemoveEmptyStringsAndArrays (XMP_NS_PHOTOSHOP);
4974
    newXMP.RemoveEmptyStringsAndArrays (XMP_NS_IPTC);
4975
    newXMP.RemoveEmptyStringsAndArrays (XMP_NS_XAP_RIGHTS);
4976
    newXMP.RemoveEmptyStringsAndArrays ("http://ns.iview-multimedia.com/mediapro/1.0/");
4977
4978
    // Process metadata subset.
4979
    
4980
    if (metadataSubset == kMetadataSubset_CopyrightOnly ||
4981
      metadataSubset == kMetadataSubset_CopyrightAndContact)
4982
      {
4983
      
4984
      dng_xmp  oldXMP  (newXMP );
4985
      dng_exif oldEXIF (newEXIF);
4986
      
4987
      // For these options, we start from nothing, and only fill in the
4988
      // fields that we absolutely need.
4989
      
4990
      newXMP.RemoveProperties (NULL);
4991
      
4992
      newEXIF.SetEmpty ();
4993
      
4994
      metadata.ClearMakerNote ();
4995
      
4996
      // Move copyright related fields over.
4997
      
4998
      CopyAltLangDefault (oldXMP,
4999
                newXMP,
5000
                XMP_NS_DC,
5001
                "rights",
5002
                &newEXIF.fCopyright);
5003
                        
5004
      CopyAltLangDefault (oldXMP,
5005
                newXMP,
5006
                XMP_NS_XAP_RIGHTS,
5007
                "UsageTerms");
5008
                
5009
      CopyString (oldXMP,
5010
            newXMP,
5011
            XMP_NS_XAP_RIGHTS,
5012
            "WebStatement");
5013
            
5014
      CopyBoolean (oldXMP,
5015
             newXMP,
5016
             XMP_NS_XAP_RIGHTS,
5017
             "Marked");
5018
             
5019
      #if qDNGXMPDocOps
5020
      
5021
      // Include basic DocOps fields, but not the full history.
5022
      
5023
      CopyString (oldXMP,
5024
            newXMP,
5025
            XMP_NS_MM,
5026
            "OriginalDocumentID");
5027
            
5028
      CopyString (oldXMP,
5029
            newXMP,
5030
            XMP_NS_MM,
5031
            "DocumentID");
5032
      
5033
      CopyString (oldXMP,
5034
            newXMP,
5035
            XMP_NS_MM,
5036
            "InstanceID");
5037
      
5038
      CopyString (oldXMP,
5039
            newXMP,
5040
            XMP_NS_XAP,
5041
            "MetadataDate");
5042
      
5043
      #endif
5044
      
5045
      // Copyright and Contact adds the contact info fields.
5046
      
5047
      if (metadataSubset == kMetadataSubset_CopyrightAndContact)
5048
        {
5049
        
5050
        // Note: Save for Web is not including the dc:creator list, but it
5051
        // is part of the IPTC contract info metadata panel, so I 
5052
        // think it should be copied as part of the contact info.
5053
        
5054
        CopyStringList (oldXMP,
5055
                newXMP,
5056
                XMP_NS_DC,
5057
                "creator",
5058
                false);
5059
                
5060
        // The first string dc:creator list is mirrored to the
5061
        // the exif artist tag, so copy that also.
5062
                
5063
        newEXIF.fArtist = oldEXIF.fArtist;
5064
        
5065
        // Copy other contact fields.
5066
        
5067
        CopyString (oldXMP,
5068
              newXMP,
5069
              XMP_NS_PHOTOSHOP,
5070
              "AuthorsPosition");
5071
              
5072
        CopyStructField (oldXMP,
5073
                 newXMP,
5074
                 XMP_NS_IPTC,
5075
                 "CreatorContactInfo",
5076
                 "CiEmailWork");
5077
              
5078
        CopyStructField (oldXMP,
5079
                 newXMP,
5080
                 XMP_NS_IPTC,
5081
                 "CreatorContactInfo",
5082
                 "CiAdrExtadr");
5083
              
5084
        CopyStructField (oldXMP,
5085
                 newXMP,
5086
                 XMP_NS_IPTC,
5087
                 "CreatorContactInfo",
5088
                 "CiAdrCity");
5089
              
5090
        CopyStructField (oldXMP,
5091
                 newXMP,
5092
                 XMP_NS_IPTC,
5093
                 "CreatorContactInfo",
5094
                 "CiAdrRegion");
5095
              
5096
        CopyStructField (oldXMP,
5097
                 newXMP,
5098
                 XMP_NS_IPTC,
5099
                 "CreatorContactInfo",
5100
                 "CiAdrPcode");
5101
              
5102
        CopyStructField (oldXMP,
5103
                 newXMP,
5104
                 XMP_NS_IPTC,
5105
                 "CreatorContactInfo",
5106
                 "CiAdrCtry");
5107
              
5108
        CopyStructField (oldXMP,
5109
                 newXMP,
5110
                 XMP_NS_IPTC,
5111
                 "CreatorContactInfo",
5112
                 "CiTelWork");
5113
              
5114
        CopyStructField (oldXMP,
5115
                 newXMP,
5116
                 XMP_NS_IPTC,
5117
                 "CreatorContactInfo",
5118
                 "CiUrlWork");
5119
                 
5120
        CopyAltLangDefault (oldXMP,
5121
                  newXMP,
5122
                  XMP_NS_DC,
5123
                  "title");
5124
                        
5125
        }
5126
5127
      }
5128
      
5129
    else if (metadataSubset == kMetadataSubset_AllExceptCameraInfo        ||
5130
         metadataSubset == kMetadataSubset_AllExceptCameraAndLocation ||
5131
         metadataSubset == kMetadataSubset_AllExceptLocationInfo)
5132
      {
5133
      
5134
      dng_xmp  oldXMP  (newXMP );
5135
      dng_exif oldEXIF (newEXIF);
5136
      
5137
      if (metadataSubset == kMetadataSubset_AllExceptCameraInfo ||
5138
        metadataSubset == kMetadataSubset_AllExceptCameraAndLocation)
5139
        {
5140
      
5141
        // This removes most of the EXIF info, so just copy the fields
5142
        // we are not deleting.
5143
        
5144
        newEXIF.SetEmpty ();
5145
        
5146
        newEXIF.fImageDescription  = oldEXIF.fImageDescription;   // Note: Differs from SFW
5147
        newEXIF.fSoftware          = oldEXIF.fSoftware;
5148
        newEXIF.fArtist            = oldEXIF.fArtist;
5149
        newEXIF.fCopyright         = oldEXIF.fCopyright;
5150
        newEXIF.fCopyright2        = oldEXIF.fCopyright2;
5151
        newEXIF.fDateTime          = oldEXIF.fDateTime;
5152
        newEXIF.fDateTimeOriginal  = oldEXIF.fDateTimeOriginal;
5153
        newEXIF.fDateTimeDigitized = oldEXIF.fDateTimeDigitized;
5154
        newEXIF.fExifVersion       = oldEXIF.fExifVersion;
5155
        newEXIF.fImageUniqueID     = oldEXIF.fImageUniqueID;
5156
        
5157
        newEXIF.CopyGPSFrom (oldEXIF);
5158
        
5159
        // Remove exif info from XMP.
5160
        
5161
        newXMP.RemoveProperties (XMP_NS_EXIF);
5162
        newXMP.RemoveProperties (XMP_NS_AUX);
5163
        
5164
        // Remove Camera Raw info
5165
        
5166
        newXMP.RemoveProperties (XMP_NS_CRS);
5167
        newXMP.RemoveProperties (XMP_NS_CRSS);
5168
        newXMP.RemoveProperties (XMP_NS_CRX);
5169
        
5170
        // Remove DocOps history, since it contains the original
5171
        // camera format.
5172
        
5173
        newXMP.Remove (XMP_NS_MM, "History");
5174
        
5175
        // MakerNote contains camera info.
5176
        
5177
        metadata.ClearMakerNote ();
5178
      
5179
        }
5180
        
5181
      if (metadataSubset == kMetadataSubset_AllExceptLocationInfo ||
5182
        metadataSubset == kMetadataSubset_AllExceptCameraAndLocation)
5183
        {
5184
        
5185
        // Remove GPS fields.
5186
        
5187
        dng_exif blankExif;
5188
        
5189
        newEXIF.CopyGPSFrom (blankExif);
5190
        
5191
        // Remove MakerNote just in case, because we don't know
5192
        // all of what is in it.
5193
        
5194
        metadata.ClearMakerNote ();
5195
        
5196
        // Remove XMP & IPTC location fields.
5197
        
5198
        newXMP.Remove (XMP_NS_PHOTOSHOP, "City");
5199
        newXMP.Remove (XMP_NS_PHOTOSHOP, "State");
5200
        newXMP.Remove (XMP_NS_PHOTOSHOP, "Country");
5201
        newXMP.Remove (XMP_NS_IPTC, "Location");
5202
        newXMP.Remove (XMP_NS_IPTC, "CountryCode");
5203
        newXMP.Remove (XMP_NS_IPTC_EXT, "LocationCreated");
5204
        newXMP.Remove (XMP_NS_IPTC_EXT, "LocationShown");
5205
        
5206
        }
5207
      
5208
      }
5209
                    
5210
    // Rebuild the legacy IPTC block, if needed.
5211
    
5212
    bool isTIFF = (strcmp (dstMIMI, "image/tiff") == 0);
5213
    bool isDNG  = (strcmp (dstMIMI, "image/dng" ) == 0);
5214
5215
    if (!isDNG)
5216
      {
5217
    
5218
      metadata.RebuildIPTC (host.Allocator (),
5219
                  isTIFF);
5220
                  
5221
      }
5222
      
5223
    else
5224
      {
5225
      
5226
      metadata.ClearIPTC ();
5227
      
5228
      }
5229
  
5230
    // Clear format related XMP.
5231
                    
5232
    newXMP.ClearOrientation ();
5233
    
5234
    newXMP.ClearImageInfo ();
5235
    
5236
    newXMP.RemoveProperties (XMP_NS_DNG);
5237
    
5238
    // All the formats we care about already keep the IPTC digest
5239
    // elsewhere, do we don't need to write it to the XMP.
5240
    
5241
    newXMP.ClearIPTCDigest ();
5242
    
5243
    // Make sure that sidecar specific tags never get written to files.
5244
    
5245
    newXMP.Remove (XMP_NS_PHOTOSHOP, "SidecarForExtension");
5246
    newXMP.Remove (XMP_NS_PHOTOSHOP, "EmbeddedXMPDigest");
5247
    
5248
    }
5249
  
5250
  #endif
5251
  
5252
4.01k
  }
5253
5254
/*****************************************************************************/
5255
5256
void dng_image_writer::WriteTIFF (dng_host &host,
5257
                  dng_stream &stream,
5258
                  const dng_image &image,
5259
                  uint32 photometricInterpretation,
5260
                  uint32 compression,
5261
                  dng_negative *negative,
5262
                  const dng_color_space *space,
5263
                  const dng_resolution *resolution,
5264
                  const dng_jpeg_preview *thumbnail,
5265
                  const dng_memory_block *imageResources,
5266
                  dng_metadata_subset metadataSubset)
5267
1.94k
  {
5268
  
5269
1.94k
  WriteTIFF (host,
5270
1.94k
         stream,
5271
1.94k
         image,
5272
1.94k
         photometricInterpretation,
5273
1.94k
         compression,
5274
1.94k
         negative ? &(negative->Metadata ()) : NULL,
5275
1.94k
         space,
5276
1.94k
         resolution,
5277
1.94k
         thumbnail,
5278
1.94k
         imageResources,
5279
1.94k
         metadataSubset);
5280
  
5281
1.94k
  }
5282
  
5283
/*****************************************************************************/
5284
5285
void dng_image_writer::WriteTIFF (dng_host &host,
5286
                  dng_stream &stream,
5287
                  const dng_image &image,
5288
                  uint32 photometricInterpretation,
5289
                  uint32 compression,
5290
                  const dng_metadata *metadata,
5291
                  const dng_color_space *space,
5292
                  const dng_resolution *resolution,
5293
                  const dng_jpeg_preview *thumbnail,
5294
                  const dng_memory_block *imageResources,
5295
                  dng_metadata_subset metadataSubset)
5296
8.75k
  {
5297
  
5298
8.75k
  const void *profileData = NULL;
5299
8.75k
  uint32 profileSize = 0;
5300
  
5301
8.75k
  const uint8 *data = NULL;
5302
8.75k
  uint32 size = 0;
5303
  
5304
8.75k
  if (space && space->ICCProfile (size, data))
5305
1.94k
    {
5306
    
5307
1.94k
    profileData = data;
5308
1.94k
    profileSize = size;
5309
    
5310
1.94k
    }
5311
    
5312
8.75k
  WriteTIFFWithProfile (host,
5313
8.75k
              stream,
5314
8.75k
              image,
5315
8.75k
              photometricInterpretation,
5316
8.75k
              compression,
5317
8.75k
              metadata,
5318
8.75k
              profileData,
5319
8.75k
              profileSize,
5320
8.75k
              resolution,
5321
8.75k
              thumbnail,
5322
8.75k
              imageResources,
5323
8.75k
              metadataSubset);
5324
  
5325
8.75k
  }
5326
5327
/*****************************************************************************/
5328
5329
void dng_image_writer::WriteTIFFWithProfile (dng_host &host,
5330
                       dng_stream &stream,
5331
                       const dng_image &image,
5332
                       uint32 photometricInterpretation,
5333
                       uint32 compression,
5334
                       dng_negative *negative,
5335
                       const void *profileData,
5336
                       uint32 profileSize,
5337
                       const dng_resolution *resolution,
5338
                       const dng_jpeg_preview *thumbnail,
5339
                       const dng_memory_block *imageResources,
5340
                       dng_metadata_subset metadataSubset)
5341
0
  {
5342
  
5343
0
  WriteTIFFWithProfile (host,
5344
0
              stream,
5345
0
              image,
5346
0
              photometricInterpretation,
5347
0
              compression,
5348
0
              negative ? &(negative->Metadata ()) : NULL,
5349
0
              profileData,
5350
0
              profileSize,
5351
0
              resolution,
5352
0
              thumbnail,
5353
0
              imageResources,
5354
0
              metadataSubset);
5355
  
5356
0
  }
5357
  
5358
/*****************************************************************************/
5359
5360
void dng_image_writer::WriteTIFFWithProfile (dng_host &host,
5361
                       dng_stream &stream,
5362
                       const dng_image &image,
5363
                       uint32 photometricInterpretation,
5364
                       uint32 compression,
5365
                       const dng_metadata *constMetadata,
5366
                       const void *profileData,
5367
                       uint32 profileSize,
5368
                       const dng_resolution *resolution,
5369
                       const dng_jpeg_preview *thumbnail,
5370
                       const dng_memory_block *imageResources,
5371
                       dng_metadata_subset metadataSubset)
5372
8.75k
  {
5373
  
5374
8.75k
  uint32 j;
5375
  
5376
8.75k
  AutoPtr<dng_metadata> metadata;
5377
  
5378
8.75k
  if (constMetadata)
5379
1.94k
    {
5380
    
5381
1.94k
    metadata.Reset (constMetadata->Clone (host.Allocator ()));
5382
    
5383
1.94k
    CleanUpMetadata (host, 
5384
1.94k
             *metadata,
5385
1.94k
             metadataSubset,
5386
1.94k
             "image/tiff");
5387
    
5388
1.94k
    }
5389
  
5390
8.75k
  dng_ifd ifd;
5391
  
5392
8.75k
  ifd.fNewSubFileType = sfMainImage;
5393
  
5394
8.75k
  ifd.fImageWidth  = image.Bounds ().W ();
5395
8.75k
  ifd.fImageLength = image.Bounds ().H ();
5396
  
5397
8.75k
  ifd.fSamplesPerPixel = image.Planes ();
5398
  
5399
8.75k
  ifd.fBitsPerSample [0] = TagTypeSize (image.PixelType ()) * 8;
5400
  
5401
17.9k
  for (j = 1; j < ifd.fSamplesPerPixel; j++)
5402
9.19k
    {
5403
9.19k
    ifd.fBitsPerSample [j] = ifd.fBitsPerSample [0];
5404
9.19k
    }
5405
    
5406
8.75k
  ifd.fPhotometricInterpretation = photometricInterpretation;
5407
  
5408
8.75k
  ifd.fCompression = compression;
5409
  
5410
8.75k
  if (ifd.fCompression == ccUncompressed)
5411
8.75k
    {
5412
    
5413
8.75k
    ifd.SetSingleStrip ();
5414
    
5415
8.75k
    }
5416
    
5417
0
  else
5418
0
    {
5419
    
5420
0
    ifd.FindStripSize (128 * 1024);
5421
    
5422
0
    ifd.fPredictor = cpHorizontalDifference;
5423
    
5424
0
    }
5425
5426
8.75k
  uint32 extraSamples = 0;
5427
  
5428
8.75k
  switch (photometricInterpretation)
5429
8.75k
    {
5430
    
5431
4.15k
    case piBlackIsZero:
5432
4.15k
      {
5433
4.15k
      extraSamples = image.Planes () - 1;
5434
4.15k
      break;
5435
0
      }
5436
      
5437
4.59k
    case piRGB:
5438
4.59k
      {
5439
4.59k
      extraSamples = image.Planes () - 3;
5440
4.59k
      break;
5441
0
      }
5442
      
5443
0
    default:
5444
0
      break;
5445
      
5446
8.75k
    }
5447
    
5448
8.75k
  ifd.fExtraSamplesCount = extraSamples;
5449
  
5450
8.75k
  if (image.PixelType () == ttFloat)
5451
1.48k
    {
5452
    
5453
3.38k
    for (j = 0; j < ifd.fSamplesPerPixel; j++)
5454
1.90k
      {
5455
1.90k
      ifd.fSampleFormat [j] = sfFloatingPoint;
5456
1.90k
      }
5457
      
5458
1.48k
    }
5459
  
5460
8.75k
  dng_tiff_directory mainIFD;
5461
  
5462
8.75k
  dng_basic_tag_set basic (mainIFD, ifd);
5463
  
5464
  // Resolution.
5465
  
5466
8.75k
  dng_resolution res;
5467
  
5468
8.75k
  if (resolution)
5469
0
    {
5470
0
    res = *resolution;
5471
0
    }
5472
  
5473
8.75k
  tag_urational tagXResolution (tcXResolution, res.fXResolution);
5474
8.75k
  tag_urational tagYResolution (tcYResolution, res.fYResolution);
5475
  
5476
8.75k
  tag_uint16 tagResolutionUnit (tcResolutionUnit, res.fResolutionUnit);
5477
  
5478
8.75k
  if (resolution)
5479
0
    {
5480
0
    mainIFD.Add (&tagXResolution   );
5481
0
    mainIFD.Add (&tagYResolution   );
5482
0
    mainIFD.Add (&tagResolutionUnit);
5483
0
    }
5484
5485
  // ICC Profile.
5486
  
5487
8.75k
  tag_icc_profile iccProfileTag (profileData, profileSize);
5488
  
5489
8.75k
  if (iccProfileTag.Count ())
5490
1.94k
    {
5491
1.94k
    mainIFD.Add (&iccProfileTag);
5492
1.94k
    }
5493
    
5494
  // XMP metadata.
5495
  
5496
  #if qDNGUseXMP
5497
  
5498
  tag_xmp tagXMP (metadata.Get () ? metadata->GetXMP () : NULL);
5499
  
5500
  if (tagXMP.Count ())
5501
    {
5502
    mainIFD.Add (&tagXMP);
5503
    }
5504
    
5505
  #endif
5506
  
5507
  // IPTC metadata.
5508
  
5509
8.75k
  tag_iptc tagIPTC (metadata.Get () ? metadata->IPTCData   () : NULL,
5510
8.75k
            metadata.Get () ? metadata->IPTCLength () : 0);
5511
    
5512
8.75k
  if (tagIPTC.Count ())
5513
41
    {
5514
41
    mainIFD.Add (&tagIPTC);
5515
41
    }
5516
    
5517
  // Adobe data (thumbnail and IPTC digest)
5518
  
5519
8.75k
  AutoPtr<dng_memory_block> adobeData (BuildAdobeData (host,
5520
8.75k
                             metadata.Get (),
5521
8.75k
                             thumbnail,
5522
8.75k
                             imageResources));
5523
                             
5524
8.75k
  tag_uint8_ptr tagAdobe (tcAdobeData,
5525
8.75k
              adobeData->Buffer_uint8 (),
5526
8.75k
              adobeData->LogicalSize ());
5527
                   
5528
8.75k
  if (tagAdobe.Count ())
5529
41
    {
5530
41
    mainIFD.Add (&tagAdobe);
5531
41
    }
5532
    
5533
  // Exif metadata.
5534
  
5535
8.75k
  exif_tag_set exifSet (mainIFD,
5536
8.75k
              metadata.Get () && metadata->GetExif () ? *metadata->GetExif () 
5537
8.75k
                                  : dng_exif (),
5538
8.75k
              metadata.Get () ? metadata->IsMakerNoteSafe () : false,
5539
8.75k
              metadata.Get () ? metadata->MakerNoteData   () : NULL,
5540
8.75k
              metadata.Get () ? metadata->MakerNoteLength () : 0,
5541
8.75k
              false);
5542
5543
  // Find offset to main image data.
5544
  
5545
8.75k
  uint32 offsetMainIFD = 8;
5546
  
5547
8.75k
  uint32 offsetExifData = offsetMainIFD + mainIFD.Size ();
5548
  
5549
8.75k
  exifSet.Locate (offsetExifData);
5550
  
5551
8.75k
  uint32 offsetMainData = offsetExifData + exifSet.Size ();
5552
    
5553
8.75k
  stream.SetWritePosition (offsetMainData);
5554
  
5555
  // Write the main image data.
5556
  
5557
8.75k
  WriteImage (host,
5558
8.75k
        ifd,
5559
8.75k
        basic,
5560
8.75k
        stream,
5561
8.75k
        image);
5562
        
5563
  // Trim the file to this length.
5564
  
5565
8.75k
  stream.SetLength (stream.Position ());
5566
  
5567
  // TIFF has a 4G size limit.
5568
  
5569
8.75k
  if (stream.Length () > 0x0FFFFFFFFL)
5570
0
    {
5571
0
    ThrowImageTooBigTIFF ();
5572
0
    }
5573
  
5574
  // Write TIFF Header.
5575
  
5576
8.75k
  stream.SetWritePosition (0);
5577
  
5578
8.75k
  stream.Put_uint16 (stream.BigEndian () ? byteOrderMM : byteOrderII);
5579
  
5580
8.75k
  stream.Put_uint16 (42);
5581
  
5582
8.75k
  stream.Put_uint32 (offsetMainIFD);
5583
  
5584
  // Write the IFDs.
5585
  
5586
8.75k
  mainIFD.Put (stream);
5587
  
5588
8.75k
  exifSet.Put (stream);
5589
  
5590
8.75k
  stream.Flush ();
5591
  
5592
8.75k
  }
5593
                 
5594
/*****************************************************************************/
5595
5596
void dng_image_writer::WriteDNG (dng_host &host,
5597
                   dng_stream &stream,
5598
                   dng_negative &negative,
5599
                   const dng_preview_list *previewList,
5600
                 uint32 maxBackwardVersion,
5601
                   bool uncompressed)
5602
2.07k
  {
5603
  
5604
2.07k
  WriteDNG (host,
5605
2.07k
        stream,
5606
2.07k
        negative,
5607
2.07k
        negative.Metadata (),
5608
2.07k
        previewList,
5609
2.07k
        maxBackwardVersion,
5610
2.07k
        uncompressed);
5611
  
5612
2.07k
  }
5613
  
5614
/*****************************************************************************/
5615
5616
void dng_image_writer::WriteDNG (dng_host &host,
5617
                   dng_stream &stream,
5618
                   const dng_negative &negative,
5619
                 const dng_metadata &constMetadata,
5620
                 const dng_preview_list *previewList,
5621
                 uint32 maxBackwardVersion,
5622
                   bool uncompressed)
5623
2.07k
  {
5624
5625
2.07k
  uint32 j;
5626
  
5627
  // Clean up metadata per MWG recommendations.
5628
  
5629
2.07k
  AutoPtr<dng_metadata> metadata (constMetadata.Clone (host.Allocator ()));
5630
5631
2.07k
  CleanUpMetadata (host, 
5632
2.07k
           *metadata,
5633
2.07k
           kMetadataSubset_All,
5634
2.07k
           "image/dng");
5635
           
5636
  // Figure out the compression to use.  Most of the time this is lossless
5637
  // JPEG.
5638
  
5639
2.07k
  uint32 compression = uncompressed ? ccUncompressed : ccJPEG;
5640
    
5641
  // Was the the original file lossy JPEG compressed?
5642
  
5643
2.07k
  const dng_jpeg_image *rawJPEGImage = negative.RawJPEGImage ();
5644
  
5645
  // If so, can we save it using the requested compression and DNG version?
5646
  
5647
2.07k
  if (uncompressed || maxBackwardVersion < dngVersion_1_4_0_0)
5648
0
    {
5649
    
5650
0
    if (rawJPEGImage || negative.RawJPEGImageDigest ().IsValid ())
5651
0
      {
5652
      
5653
0
      rawJPEGImage = NULL;
5654
      
5655
0
      negative.ClearRawJPEGImageDigest ();
5656
      
5657
0
      negative.ClearRawImageDigest ();
5658
      
5659
0
      }
5660
    
5661
0
    }
5662
    
5663
2.07k
  else if (rawJPEGImage)
5664
1.57k
    {
5665
    
5666
1.57k
    compression = ccLossyJPEG;
5667
    
5668
1.57k
    }
5669
    
5670
  // Are we saving the original size tags?
5671
  
5672
2.07k
  bool saveOriginalDefaultFinalSize     = false;
5673
2.07k
  bool saveOriginalBestQualityFinalSize = false;
5674
2.07k
  bool saveOriginalDefaultCropSize      = false;
5675
  
5676
2.07k
    {
5677
    
5678
    // See if we are saving a proxy image.
5679
    
5680
2.07k
    dng_point defaultFinalSize (negative.DefaultFinalHeight (),
5681
2.07k
                  negative.DefaultFinalWidth  ());
5682
                  
5683
2.07k
    saveOriginalDefaultFinalSize = (negative.OriginalDefaultFinalSize () !=
5684
2.07k
                    defaultFinalSize);
5685
    
5686
2.07k
    if (saveOriginalDefaultFinalSize)
5687
459
      {
5688
      
5689
      // If the save OriginalDefaultFinalSize tag, this changes the defaults
5690
      // for the OriginalBestQualityFinalSize and OriginalDefaultCropSize tags.
5691
      
5692
459
      saveOriginalBestQualityFinalSize = (negative.OriginalBestQualityFinalSize () != 
5693
459
                        defaultFinalSize);
5694
                        
5695
459
      saveOriginalDefaultCropSize = (negative.OriginalDefaultCropSizeV () !=
5696
459
                       dng_urational (defaultFinalSize.v, 1)) ||
5697
459
                      (negative.OriginalDefaultCropSizeH () !=
5698
124
                       dng_urational (defaultFinalSize.h, 1));
5699
5700
459
      }
5701
      
5702
1.61k
    else
5703
1.61k
      {
5704
      
5705
      // Else these two tags default to the normal non-proxy size image values.
5706
      
5707
1.61k
      dng_point bestQualityFinalSize (negative.BestQualityFinalHeight (),
5708
1.61k
                      negative.BestQualityFinalWidth  ());
5709
                      
5710
1.61k
      saveOriginalBestQualityFinalSize = (negative.OriginalBestQualityFinalSize () != 
5711
1.61k
                        bestQualityFinalSize);
5712
                        
5713
1.61k
      saveOriginalDefaultCropSize = (negative.OriginalDefaultCropSizeV () !=
5714
1.61k
                       negative.DefaultCropSizeV ()) ||
5715
1.61k
                      (negative.OriginalDefaultCropSizeH () !=
5716
1.50k
                       negative.DefaultCropSizeH ());
5717
      
5718
1.61k
      }
5719
    
5720
2.07k
    }
5721
    
5722
  // Is this a floating point image that we are saving?
5723
  
5724
2.07k
  bool isFloatingPoint = (negative.RawImage ().PixelType () == ttFloat);
5725
  
5726
  // Does this image have a transparency mask?
5727
  
5728
2.07k
  bool hasTransparencyMask = (negative.RawTransparencyMask () != NULL);
5729
  
5730
  // Should we save a compressed 32-bit integer file?
5731
  
5732
2.07k
  bool isCompressed32BitInteger = (negative.RawImage ().PixelType () == ttLong) &&
5733
2.07k
                    (maxBackwardVersion >= dngVersion_1_4_0_0) &&
5734
2.07k
                  (!uncompressed);
5735
  
5736
  // Figure out what main version to use.
5737
  
5738
2.07k
  uint32 dngVersion = dngVersion_Current;
5739
  
5740
  // Don't write version 1.4 files unless we actually use some feature of the 1.4 spec.
5741
  
5742
2.07k
  if (dngVersion == dngVersion_1_4_0_0)
5743
2.07k
    {
5744
    
5745
2.07k
    if (!rawJPEGImage                     &&
5746
2.07k
      !isFloatingPoint          &&
5747
2.07k
      !hasTransparencyMask        &&
5748
2.07k
      !isCompressed32BitInteger      &&
5749
2.07k
      !saveOriginalDefaultFinalSize     &&
5750
2.07k
      !saveOriginalBestQualityFinalSize &&
5751
2.07k
      !saveOriginalDefaultCropSize      )
5752
0
      {
5753
        
5754
0
      dngVersion = dngVersion_1_3_0_0;
5755
        
5756
0
      }
5757
      
5758
2.07k
    }
5759
  
5760
  // Figure out what backward version to use.
5761
  
5762
2.07k
  uint32 dngBackwardVersion = dngVersion_1_1_0_0;
5763
  
5764
  #if defined(qTestRowInterleave) || defined(qTestSubTileBlockRows) || defined(qTestSubTileBlockCols)
5765
  dngBackwardVersion = Max_uint32 (dngBackwardVersion, dngVersion_1_2_0_0);
5766
  #endif
5767
  
5768
2.07k
  dngBackwardVersion = Max_uint32 (dngBackwardVersion,
5769
2.07k
                   negative.OpcodeList1 ().MinVersion (false));
5770
5771
2.07k
  dngBackwardVersion = Max_uint32 (dngBackwardVersion,
5772
2.07k
                   negative.OpcodeList2 ().MinVersion (false));
5773
5774
2.07k
  dngBackwardVersion = Max_uint32 (dngBackwardVersion,
5775
2.07k
                   negative.OpcodeList3 ().MinVersion (false));
5776
                   
5777
2.07k
  if (negative.GetMosaicInfo () &&
5778
2.07k
    negative.GetMosaicInfo ()->fCFALayout >= 6)
5779
0
    {
5780
0
    dngBackwardVersion = Max_uint32 (dngBackwardVersion, dngVersion_1_3_0_0);
5781
0
    }
5782
    
5783
2.07k
  if (rawJPEGImage || isFloatingPoint || hasTransparencyMask || isCompressed32BitInteger)
5784
2.07k
    {
5785
2.07k
    dngBackwardVersion = Max_uint32 (dngBackwardVersion, dngVersion_1_4_0_0);
5786
2.07k
    }
5787
                   
5788
2.07k
  if (dngBackwardVersion > dngVersion)
5789
0
    {
5790
0
    ThrowProgramError ();
5791
0
    }
5792
    
5793
  // Find best thumbnail from preview list, if any.
5794
5795
2.07k
  const dng_preview *thumbnail = NULL;
5796
  
5797
2.07k
  if (previewList)
5798
2.07k
    {
5799
    
5800
2.07k
    uint32 thumbArea = 0;
5801
    
5802
4.48k
    for (j = 0; j < previewList->Count (); j++)
5803
2.41k
      {
5804
      
5805
2.41k
      const dng_image_preview *imagePreview = dynamic_cast<const dng_image_preview *>(&previewList->Preview (j));
5806
      
5807
2.41k
      if (imagePreview)
5808
495
        {
5809
        
5810
495
        uint32 thisArea = imagePreview->fImage->Bounds ().W () *
5811
495
                  imagePreview->fImage->Bounds ().H ();
5812
                  
5813
495
        if (!thumbnail || thisArea < thumbArea)
5814
495
          {
5815
          
5816
495
          thumbnail = &previewList->Preview (j);
5817
          
5818
495
          thumbArea = thisArea;
5819
          
5820
495
          }
5821
      
5822
495
        }
5823
        
5824
2.41k
      const dng_jpeg_preview *jpegPreview = dynamic_cast<const dng_jpeg_preview *>(&previewList->Preview (j));
5825
      
5826
2.41k
      if (jpegPreview)
5827
1.91k
        {
5828
        
5829
1.91k
        uint32 thisArea = jpegPreview->fPreviewSize.h *
5830
1.91k
                  jpegPreview->fPreviewSize.v;
5831
                  
5832
1.91k
        if (!thumbnail || thisArea < thumbArea)
5833
1.57k
          {
5834
          
5835
1.57k
          thumbnail = &previewList->Preview (j);
5836
          
5837
1.57k
          thumbArea = thisArea;
5838
          
5839
1.57k
          }
5840
        
5841
1.91k
        }
5842
        
5843
2.41k
      }
5844
    
5845
2.07k
    }
5846
    
5847
  // Create the main IFD
5848
                     
5849
2.07k
  dng_tiff_directory mainIFD;
5850
  
5851
  // Create the IFD for the raw data. If there is no thumnail, this is
5852
  // just a reference the main IFD.  Otherwise allocate a new one.
5853
  
5854
2.07k
  AutoPtr<dng_tiff_directory> rawIFD_IfNotMain;
5855
  
5856
2.07k
  if (thumbnail)
5857
2.07k
    {
5858
2.07k
    rawIFD_IfNotMain.Reset (new dng_tiff_directory);
5859
2.07k
    }
5860
5861
2.07k
  dng_tiff_directory &rawIFD (thumbnail ? *rawIFD_IfNotMain : mainIFD);
5862
  
5863
  // Include DNG version tags.
5864
  
5865
2.07k
  uint8 dngVersionData [4];
5866
  
5867
2.07k
  dngVersionData [0] = (uint8) (dngVersion >> 24);
5868
2.07k
  dngVersionData [1] = (uint8) (dngVersion >> 16);
5869
2.07k
  dngVersionData [2] = (uint8) (dngVersion >>  8);
5870
2.07k
  dngVersionData [3] = (uint8) (dngVersion      );
5871
  
5872
2.07k
  tag_uint8_ptr tagDNGVersion (tcDNGVersion, dngVersionData, 4);
5873
  
5874
2.07k
  mainIFD.Add (&tagDNGVersion);
5875
  
5876
2.07k
  uint8 dngBackwardVersionData [4];
5877
5878
2.07k
  dngBackwardVersionData [0] = (uint8) (dngBackwardVersion >> 24);
5879
2.07k
  dngBackwardVersionData [1] = (uint8) (dngBackwardVersion >> 16);
5880
2.07k
  dngBackwardVersionData [2] = (uint8) (dngBackwardVersion >>  8);
5881
2.07k
  dngBackwardVersionData [3] = (uint8) (dngBackwardVersion      );
5882
  
5883
2.07k
  tag_uint8_ptr tagDNGBackwardVersion (tcDNGBackwardVersion, dngBackwardVersionData, 4);
5884
  
5885
2.07k
  mainIFD.Add (&tagDNGBackwardVersion);
5886
  
5887
  // The main IFD contains the thumbnail, if there is a thumbnail.
5888
                
5889
2.07k
  AutoPtr<dng_basic_tag_set> thmBasic;
5890
  
5891
2.07k
  if (thumbnail)
5892
2.07k
    {
5893
2.07k
    thmBasic.Reset (thumbnail->AddTagSet (mainIFD));
5894
2.07k
    }
5895
              
5896
  // Get the raw image we are writing.
5897
  
5898
2.07k
  const dng_image &rawImage (negative.RawImage ());
5899
  
5900
  // For floating point, we only support ZIP compression.
5901
  
5902
2.07k
  if (isFloatingPoint && !uncompressed)
5903
495
    {
5904
    
5905
495
    compression = ccDeflate;
5906
    
5907
495
    }
5908
  
5909
  // For 32-bit integer images, we only support ZIP and uncompressed.
5910
  
5911
2.07k
  if (rawImage.PixelType () == ttLong)
5912
0
    {
5913
    
5914
0
    if (isCompressed32BitInteger)
5915
0
      {
5916
0
      compression = ccDeflate;
5917
0
      }
5918
      
5919
0
    else
5920
0
      {
5921
0
      compression = ccUncompressed;
5922
0
      }
5923
  
5924
0
    }
5925
  
5926
  // Get a copy of the mosaic info.
5927
  
5928
2.07k
  dng_mosaic_info mosaicInfo;
5929
  
5930
2.07k
  if (negative.GetMosaicInfo ())
5931
0
    {
5932
0
    mosaicInfo = *(negative.GetMosaicInfo ());
5933
0
    }
5934
    
5935
  // Create a dng_ifd record for the raw image.
5936
  
5937
2.07k
  dng_ifd info;
5938
  
5939
2.07k
  info.fImageWidth  = rawImage.Width  ();
5940
2.07k
  info.fImageLength = rawImage.Height ();
5941
  
5942
2.07k
  info.fSamplesPerPixel = rawImage.Planes ();
5943
  
5944
2.07k
  info.fPhotometricInterpretation = mosaicInfo.IsColorFilterArray () ? piCFA
5945
2.07k
                                     : piLinearRaw;
5946
      
5947
2.07k
  info.fCompression = compression;
5948
  
5949
2.07k
  if (isFloatingPoint && compression == ccDeflate)
5950
495
    {
5951
    
5952
495
    info.fPredictor = cpFloatingPoint;
5953
    
5954
495
    if (mosaicInfo.IsColorFilterArray ())
5955
0
      {
5956
      
5957
0
      if (mosaicInfo.fCFAPatternSize.h == 2)
5958
0
        {
5959
0
        info.fPredictor = cpFloatingPointX2;
5960
0
        }
5961
        
5962
0
      else if (mosaicInfo.fCFAPatternSize.h == 4)
5963
0
        {
5964
0
        info.fPredictor = cpFloatingPointX4;
5965
0
        }
5966
        
5967
0
      }
5968
      
5969
495
    } 
5970
    
5971
2.07k
  if (isCompressed32BitInteger)
5972
0
    {
5973
    
5974
0
    info.fPredictor = cpHorizontalDifference;
5975
    
5976
0
    if (mosaicInfo.IsColorFilterArray ())
5977
0
      {
5978
      
5979
0
      if (mosaicInfo.fCFAPatternSize.h == 2)
5980
0
        {
5981
0
        info.fPredictor = cpHorizontalDifferenceX2;
5982
0
        }
5983
        
5984
0
      else if (mosaicInfo.fCFAPatternSize.h == 4)
5985
0
        {
5986
0
        info.fPredictor = cpHorizontalDifferenceX4;
5987
0
        }
5988
        
5989
0
      }
5990
      
5991
0
    }
5992
  
5993
2.07k
  uint32 rawPixelType = rawImage.PixelType ();
5994
  
5995
2.07k
  if (rawPixelType == ttShort)
5996
0
    {
5997
    
5998
    // See if we are using a linearization table with <= 256 entries, in which
5999
    // case the useful data will all fit within 8-bits.
6000
    
6001
0
    const dng_linearization_info *rangeInfo = negative.GetLinearizationInfo ();
6002
  
6003
0
    if (rangeInfo)
6004
0
      {
6005
6006
0
      if (rangeInfo->fLinearizationTable.Get ())
6007
0
        {
6008
        
6009
0
        uint32 entries = rangeInfo->fLinearizationTable->LogicalSize () >> 1;
6010
        
6011
0
        if (entries <= 256)
6012
0
          {
6013
          
6014
0
          rawPixelType = ttByte;
6015
          
6016
0
          }
6017
                        
6018
0
        }
6019
        
6020
0
      }
6021
6022
0
    }
6023
  
6024
2.07k
  switch (rawPixelType)
6025
2.07k
    {
6026
    
6027
1.57k
    case ttByte:
6028
1.57k
      {
6029
1.57k
      info.fBitsPerSample [0] = 8;
6030
1.57k
      break;
6031
0
      }
6032
    
6033
0
    case ttShort:
6034
0
      {
6035
0
      info.fBitsPerSample [0] = 16;
6036
0
      break;
6037
0
      }
6038
      
6039
0
    case ttLong:
6040
0
      {
6041
0
      info.fBitsPerSample [0] = 32;
6042
0
      break;
6043
0
      }
6044
      
6045
495
    case ttFloat:
6046
495
      {
6047
      
6048
495
      if (negative.RawFloatBitDepth () == 16)
6049
495
        {
6050
495
        info.fBitsPerSample [0] = 16;
6051
495
        }
6052
        
6053
0
      else if (negative.RawFloatBitDepth () == 24)
6054
0
        {
6055
0
        info.fBitsPerSample [0] = 24;
6056
0
        }
6057
        
6058
0
      else
6059
0
        {
6060
0
        info.fBitsPerSample [0] = 32;
6061
0
        }
6062
6063
1.18k
      for (j = 0; j < info.fSamplesPerPixel; j++)
6064
687
        {
6065
687
        info.fSampleFormat [j] = sfFloatingPoint;
6066
687
        }
6067
6068
495
      break;
6069
      
6070
0
      }
6071
      
6072
0
    default:
6073
0
      {
6074
0
      ThrowProgramError ();
6075
0
      }
6076
      
6077
2.07k
    }
6078
  
6079
  // For lossless JPEG compression, we often lie about the
6080
  // actual channel count to get the predictors to work across
6081
  // same color mosaic pixels.
6082
  
6083
2.07k
  uint32 fakeChannels = 1;
6084
  
6085
2.07k
  if (info.fCompression == ccJPEG)
6086
0
    {
6087
    
6088
0
    if (mosaicInfo.IsColorFilterArray ())
6089
0
      {
6090
      
6091
0
      if (mosaicInfo.fCFAPatternSize.h == 4)
6092
0
        {
6093
0
        fakeChannels = 4;
6094
0
        }
6095
        
6096
0
      else if (mosaicInfo.fCFAPatternSize.h == 2)
6097
0
        {
6098
0
        fakeChannels = 2;
6099
0
        }
6100
      
6101
      // However, lossless JEPG is limited to four channels,
6102
      // so compromise might be required.
6103
      
6104
0
      while (fakeChannels * info.fSamplesPerPixel > 4 &&
6105
0
           fakeChannels > 1)
6106
0
        {
6107
        
6108
0
        fakeChannels >>= 1;
6109
        
6110
0
        }
6111
      
6112
0
      }
6113
  
6114
0
    }
6115
    
6116
  // Figure out tile sizes.
6117
  
6118
2.07k
  if (rawJPEGImage)
6119
1.57k
    {
6120
    
6121
1.57k
    DNG_ASSERT (rawPixelType == ttByte,
6122
1.57k
          "Unexpected jpeg pixel type");
6123
    
6124
1.57k
    DNG_ASSERT (info.fImageWidth  == (uint32) rawJPEGImage->fImageSize.h &&
6125
1.57k
          info.fImageLength == (uint32) rawJPEGImage->fImageSize.v,
6126
1.57k
          "Unexpected jpeg image size");
6127
          
6128
1.57k
    info.fTileWidth  = rawJPEGImage->fTileSize.h;
6129
1.57k
    info.fTileLength = rawJPEGImage->fTileSize.v;
6130
6131
1.57k
    info.fUsesStrips = rawJPEGImage->fUsesStrips;
6132
    
6133
1.57k
    info.fUsesTiles = !info.fUsesStrips;
6134
    
6135
1.57k
    }
6136
  
6137
495
  else if (info.fCompression == ccJPEG)
6138
0
    {
6139
    
6140
0
    info.FindTileSize (128 * 1024);
6141
    
6142
0
    }
6143
    
6144
495
  else if (info.fCompression == ccDeflate)
6145
495
    {
6146
    
6147
495
    info.FindTileSize (512 * 1024);
6148
    
6149
495
    }
6150
    
6151
0
  else if (info.fCompression == ccLossyJPEG)
6152
0
    {
6153
    
6154
0
    ThrowProgramError ("No JPEG compressed image");
6155
        
6156
0
    }
6157
    
6158
  // Don't use tiles for uncompressed images.
6159
    
6160
0
  else
6161
0
    {
6162
    
6163
0
    info.SetSingleStrip ();
6164
    
6165
0
    }
6166
    
6167
  #ifdef qTestRowInterleave
6168
  
6169
  info.fRowInterleaveFactor = qTestRowInterleave;
6170
  
6171
  #endif
6172
      
6173
  #if defined(qTestSubTileBlockRows) && defined(qTestSubTileBlockCols)
6174
  
6175
  info.fSubTileBlockRows = qTestSubTileBlockRows;
6176
  info.fSubTileBlockCols = qTestSubTileBlockCols;
6177
  
6178
  if (fakeChannels == 2)
6179
    fakeChannels = 4;
6180
  
6181
  #endif
6182
  
6183
  // Basic information.
6184
  
6185
2.07k
  dng_basic_tag_set rawBasic (rawIFD, info);
6186
  
6187
  // JPEG tables, if any.
6188
  
6189
2.07k
  tag_data_ptr tagJPEGTables (tcJPEGTables,
6190
2.07k
                ttUndefined,
6191
2.07k
                0,
6192
2.07k
                NULL);
6193
                
6194
2.07k
  if (rawJPEGImage && rawJPEGImage->fJPEGTables.Get ())
6195
0
    {
6196
    
6197
0
    tagJPEGTables.SetData (rawJPEGImage->fJPEGTables->Buffer ());
6198
    
6199
0
    tagJPEGTables.SetCount (rawJPEGImage->fJPEGTables->LogicalSize ());
6200
    
6201
0
    rawIFD.Add (&tagJPEGTables);
6202
    
6203
0
    }
6204
              
6205
  // DefaultScale tag.
6206
6207
2.07k
  dng_urational defaultScaleData [2];
6208
  
6209
2.07k
  defaultScaleData [0] = negative.DefaultScaleH ();
6210
2.07k
  defaultScaleData [1] = negative.DefaultScaleV ();
6211
                        
6212
2.07k
  tag_urational_ptr tagDefaultScale (tcDefaultScale,
6213
2.07k
                       defaultScaleData,
6214
2.07k
                       2);
6215
6216
2.07k
  rawIFD.Add (&tagDefaultScale);
6217
  
6218
  // Best quality scale tag.
6219
  
6220
2.07k
  tag_urational tagBestQualityScale (tcBestQualityScale,
6221
2.07k
                     negative.BestQualityScale ());
6222
                    
6223
2.07k
  rawIFD.Add (&tagBestQualityScale);
6224
  
6225
  // DefaultCropOrigin tag.
6226
6227
2.07k
  dng_urational defaultCropOriginData [2];
6228
  
6229
2.07k
  defaultCropOriginData [0] = negative.DefaultCropOriginH ();
6230
2.07k
  defaultCropOriginData [1] = negative.DefaultCropOriginV ();
6231
                        
6232
2.07k
  tag_urational_ptr tagDefaultCropOrigin (tcDefaultCropOrigin,
6233
2.07k
                            defaultCropOriginData,
6234
2.07k
                            2);
6235
6236
2.07k
  rawIFD.Add (&tagDefaultCropOrigin);
6237
  
6238
  // DefaultCropSize tag.
6239
6240
2.07k
  dng_urational defaultCropSizeData [2];
6241
  
6242
2.07k
  defaultCropSizeData [0] = negative.DefaultCropSizeH ();
6243
2.07k
  defaultCropSizeData [1] = negative.DefaultCropSizeV ();
6244
                        
6245
2.07k
  tag_urational_ptr tagDefaultCropSize (tcDefaultCropSize,
6246
2.07k
                          defaultCropSizeData,
6247
2.07k
                          2);
6248
6249
2.07k
  rawIFD.Add (&tagDefaultCropSize);
6250
  
6251
  // DefaultUserCrop tag.
6252
6253
2.07k
  dng_urational defaultUserCropData [4];
6254
  
6255
2.07k
  defaultUserCropData [0] = negative.DefaultUserCropT ();
6256
2.07k
  defaultUserCropData [1] = negative.DefaultUserCropL ();
6257
2.07k
  defaultUserCropData [2] = negative.DefaultUserCropB ();
6258
2.07k
  defaultUserCropData [3] = negative.DefaultUserCropR ();
6259
                        
6260
2.07k
  tag_urational_ptr tagDefaultUserCrop (tcDefaultUserCrop,
6261
2.07k
                      defaultUserCropData,
6262
2.07k
                      4);
6263
6264
2.07k
  rawIFD.Add (&tagDefaultUserCrop);
6265
  
6266
  // Range mapping tag set.
6267
  
6268
2.07k
  range_tag_set rangeSet (rawIFD, negative);
6269
              
6270
  // Mosaic pattern information.
6271
  
6272
2.07k
  mosaic_tag_set mosaicSet (rawIFD, mosaicInfo);
6273
      
6274
  // Chroma blur radius.
6275
  
6276
2.07k
  tag_urational tagChromaBlurRadius (tcChromaBlurRadius,
6277
2.07k
                     negative.ChromaBlurRadius ());
6278
                    
6279
2.07k
  if (negative.ChromaBlurRadius ().IsValid ())
6280
107
    {
6281
    
6282
107
    rawIFD.Add (&tagChromaBlurRadius);
6283
    
6284
107
    }
6285
  
6286
  // Anti-alias filter strength.
6287
  
6288
2.07k
  tag_urational tagAntiAliasStrength (tcAntiAliasStrength,
6289
2.07k
                      negative.AntiAliasStrength ());
6290
                    
6291
2.07k
  if (negative.AntiAliasStrength ().IsValid ())
6292
2.01k
    {
6293
    
6294
2.01k
    rawIFD.Add (&tagAntiAliasStrength);
6295
    
6296
2.01k
    }
6297
    
6298
  // Profile and other color related tags.
6299
  
6300
2.07k
  AutoPtr<profile_tag_set> profileSet;
6301
  
6302
2.07k
  AutoPtr<color_tag_set> colorSet;
6303
  
6304
2.07k
  dng_std_vector<uint32> extraProfileIndex;
6305
  
6306
2.07k
  if (!negative.IsMonochrome ())
6307
826
    {
6308
    
6309
826
    const dng_camera_profile &mainProfile (*negative.ComputeCameraProfileToEmbed (constMetadata));
6310
    
6311
826
    profileSet.Reset (new profile_tag_set (mainIFD,
6312
826
                         mainProfile));
6313
    
6314
826
    colorSet.Reset (new color_tag_set (mainIFD,
6315
826
                       negative));
6316
                       
6317
    // Build list of profile indices to include in extra profiles tag.
6318
                       
6319
826
    uint32 profileCount = negative.ProfileCount ();
6320
    
6321
1.65k
    for (uint32 index = 0; index < profileCount; index++)
6322
826
      {
6323
      
6324
826
      const dng_camera_profile &profile (negative.ProfileByIndex (index));
6325
      
6326
826
      if (&profile != &mainProfile)
6327
0
        {
6328
        
6329
0
        if (profile.WasReadFromDNG ())
6330
0
          {
6331
          
6332
0
          extraProfileIndex.push_back (index);
6333
          
6334
0
          }
6335
        
6336
0
        }
6337
        
6338
826
      }
6339
                       
6340
826
    }
6341
    
6342
  // Extra camera profiles tag.
6343
  
6344
2.07k
  uint32 extraProfileCount = (uint32) extraProfileIndex.size ();
6345
  
6346
2.07k
  dng_memory_data extraProfileOffsets (extraProfileCount, sizeof (uint32));
6347
  
6348
2.07k
  tag_uint32_ptr extraProfileTag (tcExtraCameraProfiles,
6349
2.07k
                  extraProfileOffsets.Buffer_uint32 (),
6350
2.07k
                  extraProfileCount);
6351
                  
6352
2.07k
  if (extraProfileCount)
6353
0
    {
6354
    
6355
0
    mainIFD.Add (&extraProfileTag);
6356
6357
0
    }
6358
      
6359
  // Other tags.
6360
  
6361
2.07k
  tag_uint16 tagOrientation (tcOrientation,
6362
2.07k
                   (uint16) negative.ComputeOrientation (constMetadata).GetTIFF ());
6363
                 
6364
2.07k
  mainIFD.Add (&tagOrientation);
6365
6366
2.07k
  tag_srational tagBaselineExposure (tcBaselineExposure,
6367
2.07k
                       negative.BaselineExposureR ());
6368
                      
6369
2.07k
  mainIFD.Add (&tagBaselineExposure);
6370
6371
2.07k
  tag_urational tagBaselineNoise (tcBaselineNoise,
6372
2.07k
                      negative.BaselineNoiseR ());
6373
                      
6374
2.07k
  mainIFD.Add (&tagBaselineNoise);
6375
  
6376
2.07k
  tag_urational tagNoiseReductionApplied (tcNoiseReductionApplied,
6377
2.07k
                      negative.NoiseReductionApplied ());
6378
                      
6379
2.07k
  if (negative.NoiseReductionApplied ().IsValid ())
6380
35
    {
6381
    
6382
35
    mainIFD.Add (&tagNoiseReductionApplied);
6383
  
6384
35
    }
6385
6386
2.07k
  tag_dng_noise_profile tagNoiseProfile (negative.NoiseProfile ());
6387
    
6388
2.07k
  if (negative.NoiseProfile ().IsValidForNegative (negative))
6389
1
    {
6390
6391
1
    mainIFD.Add (&tagNoiseProfile);
6392
    
6393
1
    }
6394
6395
2.07k
  tag_urational tagBaselineSharpness (tcBaselineSharpness,
6396
2.07k
                        negative.BaselineSharpnessR ());
6397
                      
6398
2.07k
  mainIFD.Add (&tagBaselineSharpness);
6399
6400
2.07k
  tag_string tagUniqueName (tcUniqueCameraModel,
6401
2.07k
                  negative.ModelName (),
6402
2.07k
                  true);
6403
                
6404
2.07k
  mainIFD.Add (&tagUniqueName);
6405
  
6406
2.07k
  tag_string tagLocalName (tcLocalizedCameraModel,
6407
2.07k
                 negative.LocalName (),
6408
2.07k
                 false);
6409
               
6410
2.07k
  if (negative.LocalName ().NotEmpty ())
6411
3
    {
6412
    
6413
3
    mainIFD.Add (&tagLocalName);
6414
  
6415
3
    }
6416
  
6417
2.07k
  tag_urational tagShadowScale (tcShadowScale,
6418
2.07k
                    negative.ShadowScaleR ());
6419
                      
6420
2.07k
  mainIFD.Add (&tagShadowScale);
6421
  
6422
2.07k
  tag_uint16 tagColorimetricReference (tcColorimetricReference,
6423
2.07k
                     (uint16) negative.ColorimetricReference ());
6424
                     
6425
2.07k
  if (negative.ColorimetricReference () != crSceneReferred)
6426
572
    {
6427
    
6428
572
    mainIFD.Add (&tagColorimetricReference);
6429
    
6430
572
    }
6431
    
6432
2.07k
  bool useNewDigest = (maxBackwardVersion >= dngVersion_1_4_0_0);
6433
    
6434
2.07k
  if (compression == ccLossyJPEG)
6435
1.57k
    {
6436
    
6437
1.57k
    negative.FindRawJPEGImageDigest (host);
6438
    
6439
1.57k
    }
6440
    
6441
495
  else
6442
495
    {
6443
    
6444
495
    if (useNewDigest)
6445
495
      {
6446
495
      negative.FindNewRawImageDigest (host);
6447
495
      }
6448
0
    else
6449
0
      {
6450
0
      negative.FindRawImageDigest (host);
6451
0
      }
6452
    
6453
495
    }
6454
  
6455
2.07k
  tag_uint8_ptr tagRawImageDigest (useNewDigest ? tcNewRawImageDigest : tcRawImageDigest,
6456
2.07k
                   compression == ccLossyJPEG ?
6457
1.57k
                   negative.RawJPEGImageDigest ().data :
6458
2.07k
                   (useNewDigest ? negative.NewRawImageDigest ().data
6459
495
                           : negative.RawImageDigest    ().data),
6460
2.07k
                     16);
6461
                      
6462
2.07k
  mainIFD.Add (&tagRawImageDigest);
6463
  
6464
2.07k
  negative.FindRawDataUniqueID (host);
6465
  
6466
2.07k
  tag_uint8_ptr tagRawDataUniqueID (tcRawDataUniqueID,
6467
2.07k
                      negative.RawDataUniqueID ().data,
6468
2.07k
                      16);
6469
                      
6470
2.07k
  if (negative.RawDataUniqueID ().IsValid ())
6471
2.07k
    {
6472
                 
6473
2.07k
    mainIFD.Add (&tagRawDataUniqueID);
6474
    
6475
2.07k
    }
6476
  
6477
2.07k
  tag_string tagOriginalRawFileName (tcOriginalRawFileName,
6478
2.07k
                       negative.OriginalRawFileName (),
6479
2.07k
                       false);
6480
               
6481
2.07k
  if (negative.HasOriginalRawFileName ())
6482
33
    {
6483
    
6484
33
    mainIFD.Add (&tagOriginalRawFileName);
6485
  
6486
33
    }
6487
    
6488
2.07k
  negative.FindOriginalRawFileDigest ();
6489
    
6490
2.07k
  tag_data_ptr tagOriginalRawFileData (tcOriginalRawFileData,
6491
2.07k
                     ttUndefined,
6492
2.07k
                     negative.OriginalRawFileDataLength (),
6493
2.07k
                     negative.OriginalRawFileData       ());
6494
                     
6495
2.07k
  tag_uint8_ptr tagOriginalRawFileDigest (tcOriginalRawFileDigest,
6496
2.07k
                      negative.OriginalRawFileDigest ().data,
6497
2.07k
                      16);
6498
                     
6499
2.07k
  if (negative.OriginalRawFileData ())
6500
0
    {
6501
    
6502
0
    mainIFD.Add (&tagOriginalRawFileData);
6503
    
6504
0
    mainIFD.Add (&tagOriginalRawFileDigest);
6505
  
6506
0
    }
6507
6508
  // XMP metadata.
6509
6510
  #if qDNGUseXMP
6511
    
6512
  tag_xmp tagXMP (metadata->GetXMP ());
6513
  
6514
  if (tagXMP.Count ())
6515
    {
6516
    
6517
    mainIFD.Add (&tagXMP);
6518
    
6519
    }
6520
    
6521
  #endif
6522
  
6523
  // Exif tags.
6524
  
6525
2.07k
  exif_tag_set exifSet (mainIFD,
6526
2.07k
              *metadata->GetExif (),
6527
2.07k
              metadata->IsMakerNoteSafe (),
6528
2.07k
              metadata->MakerNoteData   (),
6529
2.07k
              metadata->MakerNoteLength (),
6530
2.07k
              true);
6531
            
6532
  // Private data.
6533
  
6534
2.07k
  tag_uint8_ptr tagPrivateData (tcDNGPrivateData,
6535
2.07k
                    negative.PrivateData (),
6536
2.07k
                    negative.PrivateLength ());
6537
               
6538
2.07k
  if (negative.PrivateLength ())
6539
0
    {
6540
    
6541
0
    mainIFD.Add (&tagPrivateData);
6542
    
6543
0
    }
6544
    
6545
  // Proxy size tags.
6546
  
6547
2.07k
  uint32 originalDefaultFinalSizeData [2];
6548
  
6549
2.07k
  originalDefaultFinalSizeData [0] = negative.OriginalDefaultFinalSize ().h;
6550
2.07k
  originalDefaultFinalSizeData [1] = negative.OriginalDefaultFinalSize ().v;
6551
  
6552
2.07k
  tag_uint32_ptr tagOriginalDefaultFinalSize (tcOriginalDefaultFinalSize,
6553
2.07k
                        originalDefaultFinalSizeData,
6554
2.07k
                        2);
6555
  
6556
2.07k
  if (saveOriginalDefaultFinalSize)
6557
459
    {
6558
    
6559
459
    mainIFD.Add (&tagOriginalDefaultFinalSize);
6560
    
6561
459
    }
6562
    
6563
2.07k
  uint32 originalBestQualityFinalSizeData [2];
6564
  
6565
2.07k
  originalBestQualityFinalSizeData [0] = negative.OriginalBestQualityFinalSize ().h;
6566
2.07k
  originalBestQualityFinalSizeData [1] = negative.OriginalBestQualityFinalSize ().v;
6567
  
6568
2.07k
  tag_uint32_ptr tagOriginalBestQualityFinalSize (tcOriginalBestQualityFinalSize,
6569
2.07k
                          originalBestQualityFinalSizeData,
6570
2.07k
                          2);
6571
  
6572
2.07k
  if (saveOriginalBestQualityFinalSize)
6573
460
    {
6574
    
6575
460
    mainIFD.Add (&tagOriginalBestQualityFinalSize);
6576
    
6577
460
    }
6578
    
6579
2.07k
  dng_urational originalDefaultCropSizeData [2];
6580
  
6581
2.07k
  originalDefaultCropSizeData [0] = negative.OriginalDefaultCropSizeH ();
6582
2.07k
  originalDefaultCropSizeData [1] = negative.OriginalDefaultCropSizeV ();
6583
  
6584
2.07k
  tag_urational_ptr tagOriginalDefaultCropSize (tcOriginalDefaultCropSize,
6585
2.07k
                          originalDefaultCropSizeData,
6586
2.07k
                          2);
6587
  
6588
2.07k
  if (saveOriginalDefaultCropSize)
6589
565
    {
6590
    
6591
565
    mainIFD.Add (&tagOriginalDefaultCropSize);
6592
    
6593
565
    }
6594
    
6595
  // Opcode list 1.
6596
  
6597
2.07k
  AutoPtr<dng_memory_block> opcodeList1Data (negative.OpcodeList1 ().Spool (host));
6598
  
6599
2.07k
  tag_data_ptr tagOpcodeList1 (tcOpcodeList1,
6600
2.07k
                 ttUndefined,
6601
2.07k
                 opcodeList1Data.Get () ? opcodeList1Data->LogicalSize () : 0,
6602
2.07k
                 opcodeList1Data.Get () ? opcodeList1Data->Buffer      () : NULL);
6603
                 
6604
2.07k
  if (opcodeList1Data.Get ())
6605
0
    {
6606
    
6607
0
    rawIFD.Add (&tagOpcodeList1);
6608
    
6609
0
    }
6610
    
6611
  // Opcode list 2.
6612
  
6613
2.07k
  AutoPtr<dng_memory_block> opcodeList2Data (negative.OpcodeList2 ().Spool (host));
6614
  
6615
2.07k
  tag_data_ptr tagOpcodeList2 (tcOpcodeList2,
6616
2.07k
                 ttUndefined,
6617
2.07k
                 opcodeList2Data.Get () ? opcodeList2Data->LogicalSize () : 0,
6618
2.07k
                 opcodeList2Data.Get () ? opcodeList2Data->Buffer      () : NULL);
6619
                 
6620
2.07k
  if (opcodeList2Data.Get ())
6621
1.57k
    {
6622
    
6623
1.57k
    rawIFD.Add (&tagOpcodeList2);
6624
    
6625
1.57k
    }
6626
    
6627
  // Opcode list 3.
6628
  
6629
2.07k
  AutoPtr<dng_memory_block> opcodeList3Data (negative.OpcodeList3 ().Spool (host));
6630
  
6631
2.07k
  tag_data_ptr tagOpcodeList3 (tcOpcodeList3,
6632
2.07k
                 ttUndefined,
6633
2.07k
                 opcodeList3Data.Get () ? opcodeList3Data->LogicalSize () : 0,
6634
2.07k
                 opcodeList3Data.Get () ? opcodeList3Data->Buffer      () : NULL);
6635
                 
6636
2.07k
  if (opcodeList3Data.Get ())
6637
0
    {
6638
    
6639
0
    rawIFD.Add (&tagOpcodeList3);
6640
    
6641
0
    }
6642
    
6643
  // Transparency mask, if any.
6644
  
6645
2.07k
  AutoPtr<dng_ifd> maskInfo;
6646
  
6647
2.07k
  AutoPtr<dng_tiff_directory> maskIFD;
6648
  
6649
2.07k
  AutoPtr<dng_basic_tag_set> maskBasic;
6650
  
6651
2.07k
  if (hasTransparencyMask)
6652
0
    {
6653
    
6654
    // Create mask IFD.
6655
    
6656
0
    maskInfo.Reset (new dng_ifd);
6657
    
6658
0
    maskInfo->fNewSubFileType = sfTransparencyMask;
6659
    
6660
0
    maskInfo->fImageWidth  = negative.RawTransparencyMask ()->Bounds ().W ();
6661
0
    maskInfo->fImageLength = negative.RawTransparencyMask ()->Bounds ().H ();
6662
    
6663
0
    maskInfo->fSamplesPerPixel = 1;
6664
    
6665
0
    maskInfo->fBitsPerSample [0] = negative.RawTransparencyMaskBitDepth ();
6666
    
6667
0
    maskInfo->fPhotometricInterpretation = piTransparencyMask;
6668
    
6669
0
    maskInfo->fCompression = uncompressed ? ccUncompressed  : ccDeflate;
6670
0
    maskInfo->fPredictor   = uncompressed ? cpNullPredictor : cpHorizontalDifference;
6671
    
6672
0
    if (negative.RawTransparencyMask ()->PixelType () == ttFloat)
6673
0
      {
6674
      
6675
0
      maskInfo->fSampleFormat [0] = sfFloatingPoint;
6676
      
6677
0
      if (maskInfo->fCompression == ccDeflate)
6678
0
        {
6679
0
        maskInfo->fPredictor = cpFloatingPoint;
6680
0
        }
6681
      
6682
0
      }
6683
        
6684
0
    if (maskInfo->fCompression == ccDeflate)
6685
0
      {
6686
0
      maskInfo->FindTileSize (512 * 1024);
6687
0
      }
6688
0
    else
6689
0
      {
6690
0
      maskInfo->SetSingleStrip ();
6691
0
      }
6692
      
6693
    // Create mask tiff directory.
6694
      
6695
0
    maskIFD.Reset (new dng_tiff_directory);
6696
    
6697
    // Add mask basic tag set.
6698
    
6699
0
    maskBasic.Reset (new dng_basic_tag_set (*maskIFD, *maskInfo));
6700
        
6701
0
    }
6702
    
6703
  // Add other subfiles.
6704
    
6705
2.07k
  uint32 subFileCount = thumbnail ? 1 : 0;
6706
  
6707
2.07k
  if (hasTransparencyMask)
6708
0
    {
6709
0
    subFileCount++;
6710
0
    }
6711
  
6712
  // Add previews.
6713
  
6714
2.07k
  uint32 previewCount = previewList ? previewList->Count () : 0;
6715
  
6716
2.07k
  AutoPtr<dng_tiff_directory> previewIFD [kMaxDNGPreviews];
6717
  
6718
2.07k
  AutoPtr<dng_basic_tag_set> previewBasic [kMaxDNGPreviews];
6719
  
6720
4.48k
  for (j = 0; j < previewCount; j++)
6721
2.41k
    {
6722
    
6723
2.41k
    if (thumbnail != &previewList->Preview (j))
6724
340
      {
6725
    
6726
340
      previewIFD [j] . Reset (new dng_tiff_directory);
6727
      
6728
340
      previewBasic [j] . Reset (previewList->Preview (j).AddTagSet (*previewIFD [j]));
6729
        
6730
340
      subFileCount++;
6731
      
6732
340
      }
6733
    
6734
2.41k
    }
6735
    
6736
  // And a link to the raw and JPEG image IFDs.
6737
  
6738
2.07k
  uint32 subFileData [kMaxDNGPreviews + 2];
6739
  
6740
2.07k
  tag_uint32_ptr tagSubFile (tcSubIFDs,
6741
2.07k
                 subFileData,
6742
2.07k
                 subFileCount);
6743
                 
6744
2.07k
  if (subFileCount)
6745
2.07k
    {
6746
  
6747
2.07k
    mainIFD.Add (&tagSubFile);
6748
    
6749
2.07k
    }
6750
  
6751
  // Skip past the header and IFDs for now.
6752
  
6753
2.07k
  uint32 currentOffset = 8;
6754
  
6755
2.07k
  currentOffset += mainIFD.Size ();
6756
  
6757
2.07k
  uint32 subFileIndex = 0;
6758
  
6759
2.07k
  if (thumbnail)
6760
2.07k
    {
6761
  
6762
2.07k
    subFileData [subFileIndex++] = currentOffset;
6763
  
6764
2.07k
    currentOffset += rawIFD.Size ();
6765
    
6766
2.07k
    }
6767
    
6768
2.07k
  if (hasTransparencyMask)
6769
0
    {
6770
    
6771
0
    subFileData [subFileIndex++] = currentOffset;
6772
    
6773
0
    currentOffset += maskIFD->Size ();
6774
    
6775
0
    }
6776
  
6777
4.48k
  for (j = 0; j < previewCount; j++)
6778
2.41k
    {
6779
    
6780
2.41k
    if (thumbnail != &previewList->Preview (j))
6781
340
      {
6782
  
6783
340
      subFileData [subFileIndex++] = currentOffset;
6784
6785
340
      currentOffset += previewIFD [j]->Size ();
6786
      
6787
340
      }
6788
    
6789
2.41k
    }
6790
    
6791
2.07k
  exifSet.Locate (currentOffset);
6792
  
6793
2.07k
  currentOffset += exifSet.Size ();
6794
  
6795
2.07k
  stream.SetWritePosition (currentOffset);
6796
  
6797
  // Write the extra profiles.
6798
  
6799
2.07k
  if (extraProfileCount)
6800
0
    {
6801
    
6802
0
    for (j = 0; j < extraProfileCount; j++)
6803
0
      {
6804
      
6805
0
      extraProfileOffsets.Buffer_uint32 () [j] = (uint32) stream.Position ();
6806
      
6807
0
      uint32 index = extraProfileIndex [j];
6808
      
6809
0
      const dng_camera_profile &profile (negative.ProfileByIndex (index));
6810
      
6811
0
      tiff_dng_extended_color_profile extraWriter (profile);
6812
      
6813
0
      extraWriter.Put (stream, false);
6814
      
6815
0
      }
6816
    
6817
0
    }
6818
  
6819
  // Write the thumbnail data.
6820
  
6821
2.07k
  if (thumbnail)
6822
2.07k
    {
6823
  
6824
2.07k
    thumbnail->WriteData (host,
6825
2.07k
                *this,
6826
2.07k
                *thmBasic,
6827
2.07k
                stream);
6828
               
6829
2.07k
    }
6830
  
6831
  // Write the preview data.
6832
  
6833
4.48k
  for (j = 0; j < previewCount; j++)
6834
2.41k
    {
6835
    
6836
2.41k
    if (thumbnail != &previewList->Preview (j))
6837
340
      {
6838
  
6839
340
      previewList->Preview (j).WriteData (host,
6840
340
                        *this,
6841
340
                        *previewBasic [j],
6842
340
                        stream);
6843
                        
6844
340
      }
6845
      
6846
2.41k
    }
6847
    
6848
  // Write the raw data.
6849
  
6850
2.07k
  if (rawJPEGImage)
6851
1.57k
    {
6852
    
6853
1.57k
    uint32 tileCount = info.TilesAcross () *
6854
1.57k
               info.TilesDown   ();
6855
              
6856
3.43k
    for (uint32 tileIndex = 0; tileIndex < tileCount; tileIndex++)
6857
1.85k
      {
6858
      
6859
      // Remember this offset.
6860
      
6861
1.85k
      uint32 tileOffset = (uint32) stream.Position ();
6862
    
6863
1.85k
      rawBasic.SetTileOffset (tileIndex, tileOffset);
6864
      
6865
      // Write JPEG data.
6866
      
6867
1.85k
      stream.Put (rawJPEGImage->fJPEGData [tileIndex]->Buffer      (),
6868
1.85k
            rawJPEGImage->fJPEGData [tileIndex]->LogicalSize ());
6869
              
6870
      // Update tile count.
6871
        
6872
1.85k
      uint32 tileByteCount = (uint32) stream.Position () - tileOffset;
6873
        
6874
1.85k
      rawBasic.SetTileByteCount (tileIndex, tileByteCount);
6875
      
6876
      // Keep the tiles on even byte offsets.
6877
                         
6878
1.85k
      if (tileByteCount & 1)
6879
965
        {
6880
965
        stream.Put_uint8 (0);
6881
965
        }
6882
6883
1.85k
      }
6884
    
6885
1.57k
    }
6886
    
6887
495
  else
6888
495
    {
6889
    
6890
    #if qDNGValidate
6891
    dng_timer timer ("Write raw image time");
6892
    #endif
6893
  
6894
495
    WriteImage (host,
6895
495
          info,
6896
495
          rawBasic,
6897
495
          stream,
6898
495
          rawImage,
6899
495
          fakeChannels);
6900
          
6901
495
    }
6902
    
6903
  // Write transparency mask image.
6904
  
6905
2.07k
  if (hasTransparencyMask)
6906
0
    {
6907
    
6908
    #if qDNGValidate
6909
    dng_timer timer ("Write transparency mask time");
6910
    #endif
6911
  
6912
0
    WriteImage (host,
6913
0
          *maskInfo,
6914
0
          *maskBasic,
6915
0
          stream,
6916
0
          *negative.RawTransparencyMask ());
6917
          
6918
0
    }
6919
          
6920
  // Trim the file to this length.
6921
  
6922
2.07k
  stream.SetLength (stream.Position ());
6923
  
6924
  // DNG has a 4G size limit.
6925
  
6926
2.07k
  if (stream.Length () > 0x0FFFFFFFFL)
6927
0
    {
6928
0
    ThrowImageTooBigDNG ();
6929
0
    }
6930
  
6931
  // Write TIFF Header.
6932
  
6933
2.07k
  stream.SetWritePosition (0);
6934
  
6935
2.07k
  stream.Put_uint16 (stream.BigEndian () ? byteOrderMM : byteOrderII);
6936
  
6937
2.07k
  stream.Put_uint16 (42);
6938
  
6939
2.07k
  stream.Put_uint32 (8);
6940
  
6941
  // Write the IFDs.
6942
  
6943
2.07k
  mainIFD.Put (stream);
6944
  
6945
2.07k
  if (thumbnail)
6946
2.07k
    {
6947
  
6948
2.07k
    rawIFD.Put (stream);
6949
    
6950
2.07k
    }
6951
  
6952
2.07k
  if (hasTransparencyMask)
6953
0
    {
6954
    
6955
0
    maskIFD->Put (stream);
6956
    
6957
0
    }
6958
    
6959
4.48k
  for (j = 0; j < previewCount; j++)
6960
2.41k
    {
6961
    
6962
2.41k
    if (thumbnail != &previewList->Preview (j))
6963
340
      {
6964
  
6965
340
      previewIFD [j]->Put (stream);
6966
      
6967
340
      }
6968
    
6969
2.41k
    }
6970
    
6971
2.07k
  exifSet.Put (stream);
6972
  
6973
2.07k
  stream.Flush ();
6974
  
6975
2.07k
  }
6976
                 
6977
/*****************************************************************************/