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
2.31k
  : fXResolution ()
68
2.31k
  , fYResolution ()
69
  
70
2.31k
  , fResolutionUnit (0)
71
  
72
2.31k
  {
73
  
74
2.31k
  }
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
2.31k
  {
83
  
84
2.31k
  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
2.31k
  if (preview)
144
0
    {
145
    
146
0
    preview->SpoolAdobeThumbnail (stream);
147
        
148
0
    }
149
    
150
2.31k
  if (metadata && metadata->IPTCLength ())
151
0
    {
152
    
153
0
    dng_fingerprint iptcDigest = metadata->IPTCDigest ();
154
155
0
    if (iptcDigest.IsValid ())
156
0
      {
157
      
158
0
      stream.Put_uint32 (DNG_CHAR4 ('8','B','I','M'));
159
0
      stream.Put_uint16 (1061);
160
0
      stream.Put_uint16 (0);
161
      
162
0
      stream.Put_uint32 (16);
163
      
164
0
      stream.Put (iptcDigest.data, 16);
165
      
166
0
      }
167
      
168
0
    }
169
    
170
2.31k
  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
2.31k
  }
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
2.31k
  {
191
  
192
2.31k
  dng_memory_stream stream (host.Allocator ());
193
  
194
2.31k
  SpoolAdobeData (stream,
195
2.31k
          metadata,
196
2.31k
          preview,
197
2.31k
          imageResources);
198
          
199
2.31k
  return stream.AsMemoryBlock (host.Allocator ());
200
  
201
2.31k
  }
202
203
/*****************************************************************************/
204
205
tag_string::tag_string (uint16 code,
206
              const dng_string &s,
207
              bool forceASCII)
208
              
209
97.9k
  : tiff_tag (code, ttAscii, 0)
210
  
211
97.9k
  , fString (s)
212
  
213
97.9k
  {
214
      
215
97.9k
  if (forceASCII)
216
93.5k
    {
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
93.5k
    }
227
    
228
4.46k
  else if (!fString.IsASCII ())
229
0
    {
230
    
231
0
    fType = ttByte;
232
    
233
0
    }
234
    
235
97.9k
  fCount = fString.Length () + 1;
236
  
237
97.9k
  }
238
239
/*****************************************************************************/
240
241
void tag_string::Put (dng_stream &stream) const
242
4.73k
  {
243
  
244
4.73k
  stream.Put (fString.Get (), Size ());
245
  
246
4.73k
  }
247
248
/*****************************************************************************/
249
250
tag_encoded_text::tag_encoded_text (uint16 code,
251
                  const dng_string &text)
252
  
253
9.17k
  : tiff_tag (code, ttUndefined, 0)
254
  
255
9.17k
  , fText (text)
256
  
257
9.17k
  , fUTF16 ()
258
  
259
9.17k
  {
260
      
261
9.17k
  if (fText.IsASCII ())
262
9.16k
    {
263
  
264
9.16k
    fCount = 8 + fText.Length ();
265
    
266
9.16k
    }
267
    
268
15
  else
269
15
    {
270
    
271
15
    fCount = 8 + fText.Get_UTF16 (fUTF16) * 2;
272
    
273
15
    }
274
  
275
9.17k
  }
276
277
/*****************************************************************************/
278
279
void tag_encoded_text::Put (dng_stream &stream) const
280
26
  {
281
  
282
26
  if (fUTF16.Buffer ())
283
15
    {
284
    
285
15
    stream.Put ("UNICODE\000", 8);
286
    
287
15
    uint32 chars = (fCount - 8) >> 1;
288
    
289
15
    const uint16 *buf = fUTF16.Buffer_uint16 ();
290
    
291
138
    for (uint32 j = 0; j < chars; j++)
292
123
      {
293
      
294
123
      stream.Put_uint16 (buf [j]);
295
      
296
123
      }
297
    
298
15
    }
299
    
300
11
  else
301
11
    {
302
    
303
11
    stream.Put ("ASCII\000\000\000", 8);
304
    
305
11
    stream.Put (fText.Get (), fCount - 8);
306
    
307
11
    }
308
    
309
26
  }
310
  
311
/*****************************************************************************/
312
313
void tag_data_ptr::Put (dng_stream &stream) const
314
63.5k
  {
315
  
316
  // If we are swapping bytes, we need to swap with the right size
317
  // entries.
318
  
319
63.5k
  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
63.5k
  stream.Put (fData, Size ());
410
        
411
63.5k
  }
412
413
/******************************************************************************/
414
415
tag_matrix::tag_matrix (uint16 code,
416
              const dng_matrix &m)
417
             
418
40
  : tag_srational_ptr (code, fEntry, m.Rows () * m.Cols ())
419
  
420
40
  {
421
  
422
40
  uint32 index = 0;
423
  
424
60
  for (uint32 r = 0; r < m.Rows (); r++)
425
80
    for (uint32 c = 0; c < m.Cols (); c++)
426
60
      {
427
      
428
60
      fEntry [index].Set_real64 (m [r] [c], 10000);
429
      
430
60
      index++;
431
      
432
60
      }
433
  
434
40
  }
435
436
/******************************************************************************/
437
438
tag_icc_profile::tag_icc_profile (const void *profileData,
439
                  uint32 profileSize)
440
      
441
2.31k
  : tag_data_ptr (tcICCProfile, 
442
2.31k
            ttUndefined,
443
2.31k
            0,
444
2.31k
            NULL)
445
  
446
2.31k
  {
447
  
448
2.31k
  if (profileData && profileSize)
449
710
    {
450
    
451
710
    SetCount (profileSize);
452
710
    SetData  (profileData);
453
    
454
710
    }
455
  
456
2.31k
  }
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
9.17k
  : tag_data_ptr (code, ttAscii, 20, fData)
482
  
483
9.17k
  {
484
  
485
9.17k
  if (dt.IsValid ())
486
0
    {
487
  
488
0
    sprintf (fData,
489
0
         "%04d:%02d:%02d %02d:%02d:%02d",
490
0
         (int) dt.fYear,
491
0
         (int) dt.fMonth,
492
0
         (int) dt.fDay,
493
0
         (int) dt.fHour,
494
0
         (int) dt.fMinute,
495
0
         (int) dt.fSecond);
496
         
497
0
    }
498
  
499
9.17k
  }
500
501
/******************************************************************************/
502
503
tag_iptc::tag_iptc (const void *data,
504
            uint32 length)
505
  
506
2.31k
  : tiff_tag (tcIPTC_NAA, ttLong, (length + 3) >> 2)
507
  
508
2.31k
  , fData   (data  )
509
2.31k
  , fLength (length)
510
  
511
2.31k
  {
512
  
513
2.31k
  }
514
515
/******************************************************************************/
516
517
void tag_iptc::Put (dng_stream &stream) const
518
0
  {
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
0
  stream.Put (fData, fLength);
526
  
527
  // Pad with zeros to get to long word boundary.
528
  
529
0
  uint32 extra = fCount * 4 - fLength;
530
  
531
0
  while (extra--)
532
0
    {
533
0
    stream.Put_uint8 (0);
534
0
    }
535
536
0
  }
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
68.3k
  {
574
  
575
68.3k
  if (fEntries >= kMaxEntries)
576
0
    {
577
0
    ThrowProgramError ();
578
0
    }
579
  
580
  // Tags must be sorted in increasing order of tag code.
581
  
582
68.3k
  uint32 index = fEntries;
583
  
584
567k
  for (uint32 j = 0; j < fEntries; j++)
585
528k
    {
586
    
587
528k
    if (tag->Code () < fTag [j]->Code ())
588
29.9k
      {
589
29.9k
      index = j;
590
29.9k
      break;
591
29.9k
      }
592
    
593
528k
    }
594
    
595
191k
  for (uint32 k = fEntries; k > index; k--)
596
122k
    {
597
    
598
122k
    fTag [k] = fTag [k - 1];
599
    
600
122k
    }
601
  
602
68.3k
  fTag [index] = tag;
603
  
604
68.3k
  fEntries++;
605
  
606
68.3k
  }
607
    
608
/******************************************************************************/
609
610
uint32 dng_tiff_directory::Size () const
611
19.3k
  {
612
  
613
19.3k
  if (!fEntries) return 0;
614
  
615
4.79k
  uint32 size = fEntries * 12 + 6;
616
  
617
74.2k
  for (uint32 index = 0; index < fEntries; index++)
618
69.4k
    {
619
    
620
69.4k
    uint32 tagSize = fTag [index]->Size ();
621
    
622
69.4k
    if (tagSize > 4)
623
17.0k
      {
624
      
625
17.0k
      size += (tagSize + 1) & ~1;
626
      
627
17.0k
      }
628
    
629
69.4k
    }
630
    
631
4.79k
  return size;
632
  
633
19.3k
  }
634
    
635
/******************************************************************************/
636
637
void dng_tiff_directory::Put (dng_stream &stream,
638
                  OffsetsBase offsetsBase,
639
                  uint32 explicitBase) const
640
10.1k
  {
641
  
642
10.1k
  if (!fEntries) return;
643
  
644
4.29k
  uint32 index;
645
  
646
4.29k
  uint32 bigData = fEntries * 12 + 6;
647
  
648
4.29k
  if (offsetsBase == offsetsRelativeToStream)
649
4.29k
    bigData += (uint32) stream.Position ();
650
651
0
  else if (offsetsBase == offsetsRelativeToExplicitBase)
652
0
    bigData += explicitBase;
653
654
4.29k
  stream.Put_uint16 ((uint16) fEntries);
655
  
656
72.6k
  for (index = 0; index < fEntries; index++)
657
68.3k
    {
658
    
659
68.3k
    const tiff_tag &tag = *fTag [index];
660
    
661
68.3k
    stream.Put_uint16 (tag.Code  ());
662
68.3k
    stream.Put_uint16 (tag.Type  ());
663
68.3k
    stream.Put_uint32 (tag.Count ());
664
    
665
68.3k
    uint32 size = tag.Size ();
666
    
667
68.3k
    if (size <= 4)
668
51.9k
      {
669
      
670
51.9k
      tag.Put (stream);
671
      
672
96.5k
      while (size < 4)
673
44.5k
        {
674
44.5k
        stream.Put_uint8 (0);
675
44.5k
        size++;
676
44.5k
        }
677
        
678
51.9k
      }
679
      
680
16.3k
    else
681
16.3k
      {
682
      
683
16.3k
      stream.Put_uint32 (bigData);
684
      
685
16.3k
      bigData += (size + 1) & ~1;
686
            
687
16.3k
      }
688
    
689
68.3k
    }
690
    
691
4.29k
  stream.Put_uint32 (fChained);   // Next IFD offset
692
  
693
72.6k
  for (index = 0; index < fEntries; index++)
694
68.3k
    {
695
    
696
68.3k
    const tiff_tag &tag = *fTag [index];
697
    
698
68.3k
    uint32 size = tag.Size ();
699
    
700
68.3k
    if (size > 4)
701
16.3k
      {
702
      
703
16.3k
      tag.Put (stream);
704
      
705
16.3k
      if (size & 1)
706
2.73k
        stream.Put_uint8 (0);
707
      
708
16.3k
      }
709
    
710
68.3k
    }
711
  
712
4.29k
  }
713
714
/******************************************************************************/
715
716
dng_basic_tag_set::dng_basic_tag_set (dng_tiff_directory &directory,
717
                    const dng_ifd &info)
718
              
719
4.04k
  : fNewSubFileType (tcNewSubFileType, info.fNewSubFileType)
720
  
721
4.04k
  , fImageWidth  (tcImageWidth , info.fImageWidth )
722
4.04k
  , fImageLength (tcImageLength, info.fImageLength)
723
  
724
4.04k
  , fPhotoInterpretation (tcPhotometricInterpretation,
725
4.04k
                (uint16) info.fPhotometricInterpretation)
726
  
727
4.04k
  , fFillOrder (tcFillOrder, 1)
728
  
729
4.04k
  , fSamplesPerPixel (tcSamplesPerPixel, (uint16) info.fSamplesPerPixel)
730
  
731
4.04k
  , fBitsPerSample (tcBitsPerSample,
732
4.04k
            fBitsPerSampleData,
733
4.04k
            info.fSamplesPerPixel)
734
            
735
4.04k
  , fStrips (info.fUsesStrips)
736
            
737
4.04k
  , fTileWidth (tcTileWidth, info.fTileWidth)
738
  
739
4.04k
  , fTileLength (fStrips ? tcRowsPerStrip : tcTileLength, 
740
4.04k
           info.fTileLength)
741
  
742
4.04k
  , fTileInfoBuffer (info.TilesPerImage (), 8)
743
  
744
4.04k
  , fTileOffsetData (fTileInfoBuffer.Buffer_uint32 ())
745
  
746
4.04k
  , fTileOffsets (fStrips ? tcStripOffsets : tcTileOffsets,
747
4.04k
            fTileOffsetData,
748
4.04k
            info.TilesPerImage ())
749
             
750
4.04k
  , fTileByteCountData (fTileOffsetData + info.TilesPerImage ())
751
  
752
4.04k
  , fTileByteCounts (fStrips ? tcStripByteCounts : tcTileByteCounts,
753
4.04k
             fTileByteCountData,
754
4.04k
             info.TilesPerImage ())
755
              
756
4.04k
  , fPlanarConfiguration (tcPlanarConfiguration, pcInterleaved)
757
  
758
4.04k
  , fCompression (tcCompression, (uint16) info.fCompression)
759
4.04k
  , fPredictor   (tcPredictor  , (uint16) info.fPredictor  )
760
  
761
4.04k
  , fExtraSamples (tcExtraSamples,
762
4.04k
             fExtraSamplesData,
763
4.04k
             info.fExtraSamplesCount)
764
             
765
4.04k
  , fSampleFormat (tcSampleFormat,
766
4.04k
             fSampleFormatData,
767
4.04k
             info.fSamplesPerPixel)
768
             
769
4.04k
  , fRowInterleaveFactor (tcRowInterleaveFactor,
770
4.04k
                (uint16) info.fRowInterleaveFactor)
771
                
772
4.04k
  , fSubTileBlockSize (tcSubTileBlockSize,
773
4.04k
               fSubTileBlockSizeData,
774
4.04k
               2)
775
                 
776
4.04k
  {
777
  
778
4.04k
  uint32 j;
779
  
780
9.37k
  for (j = 0; j < info.fSamplesPerPixel; j++)
781
5.33k
    {
782
  
783
5.33k
    fBitsPerSampleData [j] = (uint16) info.fBitsPerSample [0];
784
    
785
5.33k
    }
786
  
787
4.04k
  directory.Add (&fNewSubFileType);
788
  
789
4.04k
  directory.Add (&fImageWidth);
790
4.04k
  directory.Add (&fImageLength);
791
  
792
4.04k
  directory.Add (&fPhotoInterpretation);
793
  
794
4.04k
  directory.Add (&fSamplesPerPixel);
795
  
796
4.04k
  directory.Add (&fBitsPerSample);
797
  
798
4.04k
  if (info.fBitsPerSample [0] !=  8 &&
799
4.04k
      info.fBitsPerSample [0] != 16 &&
800
4.04k
      info.fBitsPerSample [0] != 32)
801
7
    {
802
  
803
7
    directory.Add (&fFillOrder);
804
    
805
7
    }
806
    
807
4.04k
  if (!fStrips)
808
741
    {
809
    
810
741
    directory.Add (&fTileWidth);
811
  
812
741
    }
813
814
4.04k
  directory.Add (&fTileLength);
815
  
816
4.04k
  directory.Add (&fTileOffsets);
817
4.04k
  directory.Add (&fTileByteCounts);
818
  
819
4.04k
  directory.Add (&fPlanarConfiguration);
820
  
821
4.04k
  directory.Add (&fCompression);
822
  
823
4.04k
  if (info.fPredictor != cpNullPredictor)
824
284
    {
825
    
826
284
    directory.Add (&fPredictor);
827
    
828
284
    }
829
    
830
4.04k
  if (info.fExtraSamplesCount != 0)
831
15
    {
832
    
833
30
    for (j = 0; j < info.fExtraSamplesCount; j++)
834
15
      {
835
15
      fExtraSamplesData [j] = (uint16) info.fExtraSamples [j];
836
15
      }
837
      
838
15
    directory.Add (&fExtraSamples);
839
    
840
15
    }
841
    
842
4.04k
  if (info.fSampleFormat [0] != sfUnsignedInteger)
843
630
    {
844
    
845
1.26k
    for (j = 0; j < info.fSamplesPerPixel; j++)
846
630
      {
847
630
      fSampleFormatData [j] = (uint16) info.fSampleFormat [j];
848
630
      }
849
      
850
630
    directory.Add (&fSampleFormat);
851
    
852
630
    }
853
    
854
4.04k
  if (info.fRowInterleaveFactor != 1)
855
0
    {
856
    
857
0
    directory.Add (&fRowInterleaveFactor);
858
    
859
0
    }
860
    
861
4.04k
  if (info.fSubTileBlockRows != 1 ||
862
4.04k
    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
4.04k
  }
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
3.05k
  : fExifIFD ()
884
3.05k
  , fGPSIFD  ()
885
  
886
3.05k
  , fExifLink (tcExifIFD, 0)
887
3.05k
  , fGPSLink  (tcGPSInfo, 0)
888
  
889
3.05k
  , fAddedExifLink (false)
890
3.05k
  , fAddedGPSLink  (false)
891
  
892
3.05k
  , fExifVersion (tcExifVersion, ttUndefined, 4, fExifVersionData)
893
  
894
3.05k
  , fExposureTime      (tcExposureTime     , exif.fExposureTime     )
895
3.05k
  , fShutterSpeedValue (tcShutterSpeedValue, exif.fShutterSpeedValue)
896
  
897
3.05k
  , fFNumber     (tcFNumber      , exif.fFNumber      )
898
3.05k
  , fApertureValue (tcApertureValue, exif.fApertureValue)
899
  
900
3.05k
  , fBrightnessValue (tcBrightnessValue, exif.fBrightnessValue)
901
  
902
3.05k
  , fExposureBiasValue (tcExposureBiasValue, exif.fExposureBiasValue)
903
  
904
3.05k
  , fMaxApertureValue (tcMaxApertureValue , exif.fMaxApertureValue)
905
  
906
3.05k
  , fSubjectDistance (tcSubjectDistance, exif.fSubjectDistance)
907
  
908
3.05k
  , 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
3.05k
  , fISOSpeedRatings (tcISOSpeedRatings, 
916
3.05k
              (uint16) Min_uint32 (65535, 
917
3.05k
                         exif.fISOSpeedRatings [0]))
918
919
3.05k
  , fSensitivityType (tcSensitivityType, (uint16) exif.fSensitivityType)
920
921
3.05k
  , fStandardOutputSensitivity (tcStandardOutputSensitivity, exif.fStandardOutputSensitivity)
922
  
923
3.05k
  , fRecommendedExposureIndex (tcRecommendedExposureIndex, exif.fRecommendedExposureIndex)
924
925
3.05k
  , fISOSpeed (tcISOSpeed, exif.fISOSpeed)
926
927
3.05k
  , fISOSpeedLatitudeyyy (tcISOSpeedLatitudeyyy, exif.fISOSpeedLatitudeyyy)
928
929
3.05k
  , fISOSpeedLatitudezzz (tcISOSpeedLatitudezzz, exif.fISOSpeedLatitudezzz)
930
931
3.05k
  , fFlash (tcFlash, (uint16) exif.fFlash)
932
  
933
3.05k
  , fExposureProgram (tcExposureProgram, (uint16) exif.fExposureProgram)
934
  
935
3.05k
  , fMeteringMode (tcMeteringMode, (uint16) exif.fMeteringMode)
936
  
937
3.05k
  , fLightSource (tcLightSource, (uint16) exif.fLightSource)
938
  
939
3.05k
  , fSensingMethod (tcSensingMethodExif, (uint16) exif.fSensingMethod)
940
  
941
3.05k
  , fFocalLength35mm (tcFocalLengthIn35mmFilm, (uint16) exif.fFocalLengthIn35mmFilm)
942
  
943
3.05k
  , fFileSourceData ((uint8) exif.fFileSource)
944
3.05k
  , fFileSource     (tcFileSource, ttUndefined, 1, &fFileSourceData)
945
946
3.05k
  , fSceneTypeData ((uint8) exif.fSceneType)
947
3.05k
  , fSceneType     (tcSceneType, ttUndefined, 1, &fSceneTypeData)
948
  
949
3.05k
  , fCFAPattern (tcCFAPatternExif,
950
3.05k
           exif.fCFARepeatPatternRows,
951
3.05k
           exif.fCFARepeatPatternCols,
952
3.05k
           &exif.fCFAPattern [0] [0])
953
  
954
3.05k
  , fCustomRendered     (tcCustomRendered    , (uint16) exif.fCustomRendered    )
955
3.05k
  , fExposureMode       (tcExposureMode    , (uint16) exif.fExposureMode      )
956
3.05k
  , fWhiteBalance       (tcWhiteBalance    , (uint16) exif.fWhiteBalance      )
957
3.05k
  , fSceneCaptureType     (tcSceneCaptureType  , (uint16) exif.fSceneCaptureType    )
958
3.05k
  , fGainControl      (tcGainControl     , (uint16) exif.fGainControl     )
959
3.05k
  , fContrast         (tcContrast      , (uint16) exif.fContrast        )
960
3.05k
  , fSaturation       (tcSaturation      , (uint16) exif.fSaturation      )
961
3.05k
  , fSharpness        (tcSharpness       , (uint16) exif.fSharpness       )
962
3.05k
  , fSubjectDistanceRange (tcSubjectDistanceRange, (uint16) exif.fSubjectDistanceRange)
963
    
964
3.05k
  , fDigitalZoomRatio (tcDigitalZoomRatio, exif.fDigitalZoomRatio)
965
  
966
3.05k
  , fExposureIndex (tcExposureIndexExif, exif.fExposureIndex)
967
  
968
3.05k
  , fImageNumber (tcImageNumber, exif.fImageNumber)
969
  
970
3.05k
  , fSelfTimerMode (tcSelfTimerMode, (uint16) exif.fSelfTimerMode)
971
  
972
3.05k
  , fBatteryLevelA (tcBatteryLevel, exif.fBatteryLevelA)
973
3.05k
  , fBatteryLevelR (tcBatteryLevel, exif.fBatteryLevelR)
974
  
975
3.05k
  , fFocalPlaneXResolution (tcFocalPlaneXResolutionExif, exif.fFocalPlaneXResolution)
976
3.05k
  , fFocalPlaneYResolution (tcFocalPlaneYResolutionExif, exif.fFocalPlaneYResolution)
977
  
978
3.05k
  , fFocalPlaneResolutionUnit (tcFocalPlaneResolutionUnitExif, (uint16) exif.fFocalPlaneResolutionUnit)
979
  
980
3.05k
  , fSubjectArea (tcSubjectArea, fSubjectAreaData, exif.fSubjectAreaCount)
981
982
3.05k
  , fLensInfo (tcLensInfo, fLensInfoData, 4)
983
  
984
3.05k
  , fDateTime      (tcDateTime       , exif.fDateTime         .DateTime ())
985
3.05k
  , fDateTimeOriginal  (tcDateTimeOriginal , exif.fDateTimeOriginal .DateTime ())
986
3.05k
  , fDateTimeDigitized (tcDateTimeDigitized, exif.fDateTimeDigitized.DateTime ())
987
  
988
3.05k
  , fSubsecTime      (tcSubsecTime,      exif.fDateTime         .Subseconds ())
989
3.05k
  , fSubsecTimeOriginal  (tcSubsecTimeOriginal,  exif.fDateTimeOriginal .Subseconds ())
990
3.05k
  , fSubsecTimeDigitized (tcSubsecTimeDigitized, exif.fDateTimeDigitized.Subseconds ())
991
  
992
3.05k
  , fMake (tcMake, exif.fMake)
993
994
3.05k
  , fModel (tcModel, exif.fModel)
995
  
996
3.05k
  , fArtist (tcArtist, exif.fArtist)
997
  
998
3.05k
  , fSoftware (tcSoftware, exif.fSoftware)
999
  
1000
3.05k
  , fCopyright (tcCopyright, exif.fCopyright)
1001
  
1002
3.05k
  , fMakerNoteSafety (tcMakerNoteSafety, makerNoteSafe ? 1 : 0)
1003
  
1004
3.05k
  , fMakerNote (tcMakerNote, ttUndefined, makerNoteLength, makerNoteData)
1005
          
1006
3.05k
  , fImageDescription (tcImageDescription, exif.fImageDescription)
1007
  
1008
3.05k
  , fSerialNumber (tcCameraSerialNumber, exif.fCameraSerialNumber)
1009
  
1010
3.05k
  , fUserComment (tcUserComment, exif.fUserComment)
1011
  
1012
3.05k
  , fImageUniqueID (tcImageUniqueID, ttAscii, 33, fImageUniqueIDData)
1013
1014
  // EXIF 2.3 tags.
1015
1016
3.05k
  , fCameraOwnerName   (tcCameraOwnerNameExif,    exif.fOwnerName     )
1017
3.05k
  , fBodySerialNumber  (tcCameraSerialNumberExif, exif.fCameraSerialNumber)
1018
3.05k
  , fLensSpecification (tcLensSpecificationExif,  fLensInfoData, 4      )
1019
3.05k
  , fLensMake      (tcLensMakeExif,       exif.fLensMake      )
1020
3.05k
  , fLensModel       (tcLensModelExif,      exif.fLensName      )
1021
3.05k
  , fLensSerialNumber  (tcLensSerialNumberExif,   exif.fLensSerialNumber  )
1022
1023
3.05k
  , fGPSVersionID (tcGPSVersionID, fGPSVersionData, 4)
1024
  
1025
3.05k
  , fGPSLatitudeRef (tcGPSLatitudeRef, exif.fGPSLatitudeRef)
1026
3.05k
  , fGPSLatitude    (tcGPSLatitude,    exif.fGPSLatitude, 3)
1027
  
1028
3.05k
  , fGPSLongitudeRef (tcGPSLongitudeRef, exif.fGPSLongitudeRef)
1029
3.05k
  , fGPSLongitude    (tcGPSLongitude,    exif.fGPSLongitude, 3)
1030
  
1031
3.05k
  , fGPSAltitudeRef (tcGPSAltitudeRef, (uint8) exif.fGPSAltitudeRef)
1032
3.05k
  , fGPSAltitude    (tcGPSAltitude,            exif.fGPSAltitude   )
1033
  
1034
3.05k
  , fGPSTimeStamp (tcGPSTimeStamp, exif.fGPSTimeStamp, 3)
1035
    
1036
3.05k
  , fGPSSatellites  (tcGPSSatellites , exif.fGPSSatellites )
1037
3.05k
  , fGPSStatus      (tcGPSStatus     , exif.fGPSStatus     )
1038
3.05k
  , fGPSMeasureMode (tcGPSMeasureMode, exif.fGPSMeasureMode)
1039
  
1040
3.05k
  , fGPSDOP (tcGPSDOP, exif.fGPSDOP)
1041
    
1042
3.05k
  , fGPSSpeedRef (tcGPSSpeedRef, exif.fGPSSpeedRef)
1043
3.05k
  , fGPSSpeed    (tcGPSSpeed   , exif.fGPSSpeed   )
1044
    
1045
3.05k
  , fGPSTrackRef (tcGPSTrackRef, exif.fGPSTrackRef)
1046
3.05k
  , fGPSTrack    (tcGPSTrack   , exif.fGPSTrack   )
1047
    
1048
3.05k
  , fGPSImgDirectionRef (tcGPSImgDirectionRef, exif.fGPSImgDirectionRef)
1049
3.05k
  , fGPSImgDirection    (tcGPSImgDirection   , exif.fGPSImgDirection   )
1050
  
1051
3.05k
  , fGPSMapDatum (tcGPSMapDatum, exif.fGPSMapDatum)
1052
    
1053
3.05k
  , fGPSDestLatitudeRef (tcGPSDestLatitudeRef, exif.fGPSDestLatitudeRef)
1054
3.05k
  , fGPSDestLatitude    (tcGPSDestLatitude,    exif.fGPSDestLatitude, 3)
1055
  
1056
3.05k
  , fGPSDestLongitudeRef (tcGPSDestLongitudeRef, exif.fGPSDestLongitudeRef)
1057
3.05k
  , fGPSDestLongitude    (tcGPSDestLongitude,    exif.fGPSDestLongitude, 3)
1058
  
1059
3.05k
  , fGPSDestBearingRef (tcGPSDestBearingRef, exif.fGPSDestBearingRef)
1060
3.05k
  , fGPSDestBearing    (tcGPSDestBearing   , exif.fGPSDestBearing   )
1061
    
1062
3.05k
  , fGPSDestDistanceRef (tcGPSDestDistanceRef, exif.fGPSDestDistanceRef)
1063
3.05k
  , fGPSDestDistance    (tcGPSDestDistance   , exif.fGPSDestDistance   )
1064
    
1065
3.05k
  , fGPSProcessingMethod (tcGPSProcessingMethod, exif.fGPSProcessingMethod)
1066
3.05k
  , fGPSAreaInformation  (tcGPSAreaInformation , exif.fGPSAreaInformation )
1067
  
1068
3.05k
  , fGPSDateStamp (tcGPSDateStamp, exif.fGPSDateStamp)
1069
  
1070
3.05k
  , fGPSDifferential (tcGPSDifferential, (uint16) exif.fGPSDifferential)
1071
    
1072
3.05k
  , fGPSHPositioningError (tcGPSHPositioningError, exif.fGPSHPositioningError)
1073
    
1074
3.05k
  {
1075
  
1076
3.05k
  if (exif.fExifVersion)
1077
5
    {
1078
    
1079
5
    fExifVersionData [0] = (uint8) (exif.fExifVersion >> 24);
1080
5
    fExifVersionData [1] = (uint8) (exif.fExifVersion >> 16);
1081
5
    fExifVersionData [2] = (uint8) (exif.fExifVersion >>  8);
1082
5
    fExifVersionData [3] = (uint8) (exif.fExifVersion      );
1083
    
1084
5
    fExifIFD.Add (&fExifVersion);
1085
1086
5
    }
1087
  
1088
3.05k
  if (exif.fExposureTime.IsValid ())
1089
41
    {
1090
41
    fExifIFD.Add (&fExposureTime);
1091
41
    }
1092
    
1093
3.05k
  if (exif.fShutterSpeedValue.IsValid ())
1094
41
    {
1095
41
    fExifIFD.Add (&fShutterSpeedValue);
1096
41
    }
1097
    
1098
3.05k
  if (exif.fFNumber.IsValid ())
1099
33
    {
1100
33
    fExifIFD.Add (&fFNumber);
1101
33
    }
1102
    
1103
3.05k
  if (exif.fApertureValue.IsValid ())
1104
28
    {
1105
28
    fExifIFD.Add (&fApertureValue);
1106
28
    }
1107
    
1108
3.05k
  if (exif.fBrightnessValue.IsValid ())
1109
13
    {
1110
13
    fExifIFD.Add (&fBrightnessValue);
1111
13
    }
1112
    
1113
3.05k
  if (exif.fExposureBiasValue.IsValid ())
1114
47
    {
1115
47
    fExifIFD.Add (&fExposureBiasValue);
1116
47
    }
1117
    
1118
3.05k
  if (exif.fMaxApertureValue.IsValid ())
1119
2
    {
1120
2
    fExifIFD.Add (&fMaxApertureValue);
1121
2
    }
1122
    
1123
3.05k
  if (exif.fSubjectDistance.IsValid ())
1124
11
    {
1125
11
    fExifIFD.Add (&fSubjectDistance);
1126
11
    }
1127
    
1128
3.05k
  if (exif.fFocalLength.IsValid ())
1129
11
    {
1130
11
    fExifIFD.Add (&fFocalLength);
1131
11
    }
1132
  
1133
3.05k
  if (exif.fISOSpeedRatings [0] != 0)
1134
44
    {
1135
44
    fExifIFD.Add (&fISOSpeedRatings);
1136
44
    }
1137
    
1138
3.05k
  if (exif.fFlash <= 0x0FFFF)
1139
11
    {
1140
11
    fExifIFD.Add (&fFlash);
1141
11
    }
1142
    
1143
3.05k
  if (exif.fExposureProgram <= 0x0FFFF)
1144
7
    {
1145
7
    fExifIFD.Add (&fExposureProgram);
1146
7
    }
1147
    
1148
3.05k
  if (exif.fMeteringMode <= 0x0FFFF)
1149
4
    {
1150
4
    fExifIFD.Add (&fMeteringMode);
1151
4
    }
1152
    
1153
3.05k
  if (exif.fLightSource <= 0x0FFFF)
1154
9
    {
1155
9
    fExifIFD.Add (&fLightSource);
1156
9
    }
1157
    
1158
3.05k
  if (exif.fSensingMethod <= 0x0FFFF)
1159
35
    {
1160
35
    fExifIFD.Add (&fSensingMethod);
1161
35
    }
1162
    
1163
3.05k
  if (exif.fFocalLengthIn35mmFilm != 0)
1164
4
    {
1165
4
    fExifIFD.Add (&fFocalLength35mm);
1166
4
    }
1167
    
1168
3.05k
  if (exif.fFileSource <= 0x0FF)
1169
23
    {
1170
23
    fExifIFD.Add (&fFileSource);
1171
23
    }
1172
    
1173
3.05k
  if (exif.fSceneType <= 0x0FF)
1174
19
    {
1175
19
    fExifIFD.Add (&fSceneType);
1176
19
    }
1177
    
1178
3.05k
  if (exif.fCFARepeatPatternRows &&
1179
3.05k
      exif.fCFARepeatPatternCols)
1180
0
    {
1181
0
    fExifIFD.Add (&fCFAPattern);
1182
0
    }
1183
    
1184
3.05k
  if (exif.fCustomRendered <= 0x0FFFF)
1185
7
    {
1186
7
    fExifIFD.Add (&fCustomRendered);
1187
7
    }
1188
    
1189
3.05k
  if (exif.fExposureMode <= 0x0FFFF)
1190
4
    {
1191
4
    fExifIFD.Add (&fExposureMode);
1192
4
    }
1193
    
1194
3.05k
  if (exif.fWhiteBalance <= 0x0FFFF)
1195
7
    {
1196
7
    fExifIFD.Add (&fWhiteBalance);
1197
7
    }
1198
    
1199
3.05k
  if (exif.fSceneCaptureType <= 0x0FFFF)
1200
0
    {
1201
0
    fExifIFD.Add (&fSceneCaptureType);
1202
0
    }
1203
    
1204
3.05k
  if (exif.fGainControl <= 0x0FFFF)
1205
21
    {
1206
21
    fExifIFD.Add (&fGainControl);
1207
21
    }
1208
    
1209
3.05k
  if (exif.fContrast <= 0x0FFFF)
1210
10
    {
1211
10
    fExifIFD.Add (&fContrast);
1212
10
    }
1213
    
1214
3.05k
  if (exif.fSaturation <= 0x0FFFF)
1215
8
    {
1216
8
    fExifIFD.Add (&fSaturation);
1217
8
    }
1218
    
1219
3.05k
  if (exif.fSharpness <= 0x0FFFF)
1220
1
    {
1221
1
    fExifIFD.Add (&fSharpness);
1222
1
    }
1223
    
1224
3.05k
  if (exif.fSubjectDistanceRange <= 0x0FFFF)
1225
5
    {
1226
5
    fExifIFD.Add (&fSubjectDistanceRange);
1227
5
    }
1228
    
1229
3.05k
  if (exif.fDigitalZoomRatio.IsValid ())
1230
15
    {
1231
15
    fExifIFD.Add (&fDigitalZoomRatio);
1232
15
    }
1233
    
1234
3.05k
  if (exif.fExposureIndex.IsValid ())
1235
23
    {
1236
23
    fExifIFD.Add (&fExposureIndex);
1237
23
    }
1238
    
1239
3.05k
  if (insideDNG)  // TIFF-EP only tags
1240
741
    {
1241
    
1242
741
    if (exif.fImageNumber != 0xFFFFFFFF)
1243
0
      {
1244
0
      directory.Add (&fImageNumber);
1245
0
      }
1246
      
1247
741
    if (exif.fSelfTimerMode <= 0x0FFFF)
1248
3
      {
1249
3
      directory.Add (&fSelfTimerMode);
1250
3
      }
1251
      
1252
741
    if (exif.fBatteryLevelA.NotEmpty ())
1253
0
      {
1254
0
      directory.Add (&fBatteryLevelA);
1255
0
      }
1256
      
1257
741
    else if (exif.fBatteryLevelR.IsValid ())
1258
7
      {
1259
7
      directory.Add (&fBatteryLevelR);
1260
7
      }
1261
    
1262
741
    }
1263
    
1264
3.05k
  if (exif.fFocalPlaneXResolution.IsValid ())
1265
14
    {
1266
14
    fExifIFD.Add (&fFocalPlaneXResolution);
1267
14
    }
1268
  
1269
3.05k
  if (exif.fFocalPlaneYResolution.IsValid ())
1270
4
    {
1271
4
    fExifIFD.Add (&fFocalPlaneYResolution);
1272
4
    }
1273
  
1274
3.05k
  if (exif.fFocalPlaneResolutionUnit <= 0x0FFFF)
1275
14
    {
1276
14
    fExifIFD.Add (&fFocalPlaneResolutionUnit);
1277
14
    }
1278
    
1279
3.05k
  if (exif.fSubjectAreaCount)
1280
14
    {
1281
    
1282
14
    fSubjectAreaData [0] = (uint16) exif.fSubjectArea [0];
1283
14
    fSubjectAreaData [1] = (uint16) exif.fSubjectArea [1];
1284
14
    fSubjectAreaData [2] = (uint16) exif.fSubjectArea [2];
1285
14
    fSubjectAreaData [3] = (uint16) exif.fSubjectArea [3];
1286
    
1287
14
    fExifIFD.Add (&fSubjectArea);
1288
    
1289
14
    }
1290
  
1291
3.05k
  if (exif.fLensInfo [0].IsValid () &&
1292
3.05k
    exif.fLensInfo [1].IsValid ())
1293
1
    {
1294
    
1295
1
    fLensInfoData [0] = exif.fLensInfo [0];
1296
1
    fLensInfoData [1] = exif.fLensInfo [1];
1297
1
    fLensInfoData [2] = exif.fLensInfo [2];
1298
1
    fLensInfoData [3] = exif.fLensInfo [3];
1299
1300
1
    if (insideDNG)
1301
0
      {
1302
0
      directory.Add (&fLensInfo);
1303
0
      }
1304
    
1305
1
    }
1306
    
1307
3.05k
  if (exif.fDateTime.IsValid ())
1308
0
    {
1309
    
1310
0
    directory.Add (&fDateTime);
1311
    
1312
0
    if (exif.fDateTime.Subseconds ().NotEmpty ())
1313
0
      {
1314
0
      fExifIFD.Add (&fSubsecTime);
1315
0
      }
1316
    
1317
0
    }
1318
    
1319
3.05k
  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
3.05k
  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
3.05k
  if (exif.fMake.NotEmpty ())
1344
6
    {
1345
6
    directory.Add (&fMake);
1346
6
    }
1347
    
1348
3.05k
  if (exif.fModel.NotEmpty ())
1349
16
    {
1350
16
    directory.Add (&fModel);
1351
16
    }
1352
    
1353
3.05k
  if (exif.fArtist.NotEmpty ())
1354
8
    {
1355
8
    directory.Add (&fArtist);
1356
8
    }
1357
  
1358
3.05k
  if (exif.fSoftware.NotEmpty ())
1359
7
    {
1360
7
    directory.Add (&fSoftware);
1361
7
    }
1362
  
1363
3.05k
  if (exif.fCopyright.NotEmpty ())
1364
0
    {
1365
0
    directory.Add (&fCopyright);
1366
0
    }
1367
  
1368
3.05k
  if (exif.fImageDescription.NotEmpty ())
1369
1
    {
1370
1
    directory.Add (&fImageDescription);
1371
1
    }
1372
  
1373
3.05k
  if (exif.fCameraSerialNumber.NotEmpty () && insideDNG)
1374
0
    {
1375
0
    directory.Add (&fSerialNumber);
1376
0
    }
1377
    
1378
3.05k
  if (makerNoteSafe && makerNoteData)
1379
0
    {
1380
    
1381
0
    directory.Add (&fMakerNoteSafety);
1382
    
1383
0
    fExifIFD.Add (&fMakerNote);
1384
    
1385
0
    }
1386
    
1387
3.05k
  if (exif.fUserComment.NotEmpty ())
1388
26
    {
1389
26
    fExifIFD.Add (&fUserComment);
1390
26
    }
1391
    
1392
3.05k
  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
3.05k
  if (exif.AtLeastVersion0230 ())
1409
4
    {
1410
1411
4
    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
4
    if (exif.fSensitivityType   != 0 &&
1422
4
      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
4
    if (exif.fOwnerName.NotEmpty ())
1460
0
      {
1461
0
      fExifIFD.Add (&fCameraOwnerName);
1462
0
      }
1463
    
1464
4
    if (exif.fCameraSerialNumber.NotEmpty ())
1465
0
      {
1466
0
      fExifIFD.Add (&fBodySerialNumber);
1467
0
      }
1468
1469
4
    if (exif.fLensInfo [0].IsValid () &&
1470
4
      exif.fLensInfo [1].IsValid ())
1471
0
      {
1472
0
      fExifIFD.Add (&fLensSpecification);
1473
0
      }
1474
    
1475
4
    if (exif.fLensMake.NotEmpty ())
1476
0
      {
1477
0
      fExifIFD.Add (&fLensMake);
1478
0
      }
1479
    
1480
4
    if (exif.fLensName.NotEmpty ())
1481
0
      {
1482
0
      fExifIFD.Add (&fLensModel);
1483
0
      }
1484
    
1485
4
    if (exif.fLensSerialNumber.NotEmpty ())
1486
0
      {
1487
0
      fExifIFD.Add (&fLensSerialNumber);
1488
0
      }
1489
    
1490
4
    }
1491
    
1492
3.05k
  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
3.05k
  if (exif.fGPSLatitudeRef.NotEmpty () &&
1505
3.05k
    exif.fGPSLatitude [0].IsValid ())
1506
0
    {
1507
0
    fGPSIFD.Add (&fGPSLatitudeRef);
1508
0
    fGPSIFD.Add (&fGPSLatitude   );
1509
0
    }
1510
    
1511
3.05k
  if (exif.fGPSLongitudeRef.NotEmpty () &&
1512
3.05k
    exif.fGPSLongitude [0].IsValid ())
1513
0
    {
1514
0
    fGPSIFD.Add (&fGPSLongitudeRef);
1515
0
    fGPSIFD.Add (&fGPSLongitude   );
1516
0
    }
1517
    
1518
3.05k
  if (exif.fGPSAltitudeRef <= 0x0FF)
1519
0
    {
1520
0
    fGPSIFD.Add (&fGPSAltitudeRef);
1521
0
    }
1522
    
1523
3.05k
  if (exif.fGPSAltitude.IsValid ())
1524
4
    {
1525
4
    fGPSIFD.Add (&fGPSAltitude);
1526
4
    }
1527
    
1528
3.05k
  if (exif.fGPSTimeStamp [0].IsValid ())
1529
0
    {
1530
0
    fGPSIFD.Add (&fGPSTimeStamp);
1531
0
    }
1532
    
1533
3.05k
  if (exif.fGPSSatellites.NotEmpty ())
1534
0
    {
1535
0
    fGPSIFD.Add (&fGPSSatellites);
1536
0
    }
1537
    
1538
3.05k
  if (exif.fGPSStatus.NotEmpty ())
1539
0
    {
1540
0
    fGPSIFD.Add (&fGPSStatus);
1541
0
    }
1542
    
1543
3.05k
  if (exif.fGPSMeasureMode.NotEmpty ())
1544
0
    {
1545
0
    fGPSIFD.Add (&fGPSMeasureMode);
1546
0
    }
1547
    
1548
3.05k
  if (exif.fGPSDOP.IsValid ())
1549
0
    {
1550
0
    fGPSIFD.Add (&fGPSDOP);
1551
0
    }
1552
    
1553
3.05k
  if (exif.fGPSSpeedRef.NotEmpty ())
1554
0
    {
1555
0
    fGPSIFD.Add (&fGPSSpeedRef);
1556
0
    }
1557
    
1558
3.05k
  if (exif.fGPSSpeed.IsValid ())
1559
0
    {
1560
0
    fGPSIFD.Add (&fGPSSpeed);
1561
0
    }
1562
    
1563
3.05k
  if (exif.fGPSTrackRef.NotEmpty ())
1564
0
    {
1565
0
    fGPSIFD.Add (&fGPSTrackRef);
1566
0
    }
1567
    
1568
3.05k
  if (exif.fGPSTrack.IsValid ())
1569
0
    {
1570
0
    fGPSIFD.Add (&fGPSTrack);
1571
0
    }
1572
    
1573
3.05k
  if (exif.fGPSImgDirectionRef.NotEmpty ())
1574
0
    {
1575
0
    fGPSIFD.Add (&fGPSImgDirectionRef);
1576
0
    }
1577
    
1578
3.05k
  if (exif.fGPSImgDirection.IsValid ())
1579
0
    {
1580
0
    fGPSIFD.Add (&fGPSImgDirection);
1581
0
    }
1582
1583
3.05k
  if (exif.fGPSMapDatum.NotEmpty ())
1584
0
    {
1585
0
    fGPSIFD.Add (&fGPSMapDatum);
1586
0
    }
1587
    
1588
3.05k
  if (exif.fGPSDestLatitudeRef.NotEmpty () &&
1589
3.05k
    exif.fGPSDestLatitude [0].IsValid ())
1590
0
    {
1591
0
    fGPSIFD.Add (&fGPSDestLatitudeRef);
1592
0
    fGPSIFD.Add (&fGPSDestLatitude   );
1593
0
    }
1594
    
1595
3.05k
  if (exif.fGPSDestLongitudeRef.NotEmpty () &&
1596
3.05k
    exif.fGPSDestLongitude [0].IsValid ())
1597
0
    {
1598
0
    fGPSIFD.Add (&fGPSDestLongitudeRef);
1599
0
    fGPSIFD.Add (&fGPSDestLongitude   );
1600
0
    }
1601
    
1602
3.05k
  if (exif.fGPSDestBearingRef.NotEmpty ())
1603
0
    {
1604
0
    fGPSIFD.Add (&fGPSDestBearingRef);
1605
0
    }
1606
    
1607
3.05k
  if (exif.fGPSDestBearing.IsValid ())
1608
0
    {
1609
0
    fGPSIFD.Add (&fGPSDestBearing);
1610
0
    }
1611
1612
3.05k
  if (exif.fGPSDestDistanceRef.NotEmpty ())
1613
0
    {
1614
0
    fGPSIFD.Add (&fGPSDestDistanceRef);
1615
0
    }
1616
    
1617
3.05k
  if (exif.fGPSDestDistance.IsValid ())
1618
0
    {
1619
0
    fGPSIFD.Add (&fGPSDestDistance);
1620
0
    }
1621
    
1622
3.05k
  if (exif.fGPSProcessingMethod.NotEmpty ())
1623
0
    {
1624
0
    fGPSIFD.Add (&fGPSProcessingMethod);
1625
0
    }
1626
1627
3.05k
  if (exif.fGPSAreaInformation.NotEmpty ())
1628
0
    {
1629
0
    fGPSIFD.Add (&fGPSAreaInformation);
1630
0
    }
1631
  
1632
3.05k
  if (exif.fGPSDateStamp.NotEmpty ())
1633
0
    {
1634
0
    fGPSIFD.Add (&fGPSDateStamp);
1635
0
    }
1636
  
1637
3.05k
  if (exif.fGPSDifferential <= 0x0FFFF)
1638
0
    {
1639
0
    fGPSIFD.Add (&fGPSDifferential);
1640
0
    }
1641
1642
3.05k
  if (exif.AtLeastVersion0230 ())
1643
4
    {
1644
    
1645
4
    if (exif.fGPSHPositioningError.IsValid ())
1646
0
      {
1647
0
      fGPSIFD.Add (&fGPSHPositioningError);
1648
0
      }
1649
1650
4
    }
1651
    
1652
3.05k
  AddLinks (directory);
1653
  
1654
3.05k
  }
1655
1656
/******************************************************************************/
1657
1658
void exif_tag_set::AddLinks (dng_tiff_directory &directory)
1659
3.05k
  {
1660
  
1661
3.05k
  if (fExifIFD.Size () != 0 && !fAddedExifLink)
1662
245
    {
1663
    
1664
245
    directory.Add (&fExifLink);
1665
    
1666
245
    fAddedExifLink = true;
1667
    
1668
245
    }
1669
  
1670
3.05k
  if (fGPSIFD.Size () != 0 && !fAddedGPSLink)
1671
4
    {
1672
    
1673
4
    directory.Add (&fGPSLink);
1674
    
1675
4
    fAddedGPSLink = true;
1676
    
1677
4
    }
1678
  
1679
3.05k
  }
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
741
  : fActiveArea (tcActiveArea,
1733
741
           fActiveAreaData,
1734
741
           4)
1735
  
1736
741
  , fMaskedAreas (tcMaskedAreas,
1737
741
            fMaskedAreaData,
1738
741
            0)
1739
  
1740
741
  , fLinearizationTable (tcLinearizationTable,
1741
741
               NULL,
1742
741
               0)
1743
                
1744
741
  , fBlackLevelRepeatDim (tcBlackLevelRepeatDim,
1745
741
                fBlackLevelRepeatDimData,
1746
741
                2)
1747
               
1748
741
  , fBlackLevel (tcBlackLevel,
1749
741
           fBlackLevelData)
1750
           
1751
741
  , fBlackLevelDeltaHData ()
1752
741
  , fBlackLevelDeltaVData ()
1753
           
1754
741
  , fBlackLevelDeltaH (tcBlackLevelDeltaH)
1755
741
  , fBlackLevelDeltaV (tcBlackLevelDeltaV)
1756
 
1757
741
  , fWhiteLevel16 (tcWhiteLevel,
1758
741
             fWhiteLevelData16)
1759
           
1760
741
  , fWhiteLevel32 (tcWhiteLevel,
1761
741
             fWhiteLevelData32)
1762
           
1763
741
  {
1764
  
1765
741
  const dng_image &rawImage (negative.RawImage ());
1766
  
1767
741
  const dng_linearization_info *rangeInfo = negative.GetLinearizationInfo ();
1768
  
1769
741
  if (rangeInfo)
1770
741
    {
1771
    
1772
    // ActiveArea:
1773
    
1774
741
      {
1775
    
1776
741
      const dng_rect &r = rangeInfo->fActiveArea;
1777
      
1778
741
      if (r.NotEmpty ())
1779
274
        {
1780
      
1781
274
        fActiveAreaData [0] = r.t;
1782
274
        fActiveAreaData [1] = r.l;
1783
274
        fActiveAreaData [2] = r.b;
1784
274
        fActiveAreaData [3] = r.r;
1785
        
1786
274
        directory.Add (&fActiveArea);
1787
        
1788
274
        }
1789
        
1790
741
      }
1791
      
1792
    // MaskedAreas:
1793
      
1794
741
    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
741
    if (rangeInfo->fLinearizationTable.Get ())
1818
1
      {
1819
      
1820
1
      fLinearizationTable.SetData  (rangeInfo->fLinearizationTable->Buffer_uint16 ()     );
1821
1
      fLinearizationTable.SetCount (rangeInfo->fLinearizationTable->LogicalSize   () >> 1);
1822
      
1823
1
      directory.Add (&fLinearizationTable);
1824
      
1825
1
      }
1826
      
1827
    // BlackLevelRepeatDim:
1828
    
1829
741
      {
1830
    
1831
741
      fBlackLevelRepeatDimData [0] = (uint16) rangeInfo->fBlackLevelRepeatRows;
1832
741
      fBlackLevelRepeatDimData [1] = (uint16) rangeInfo->fBlackLevelRepeatCols;
1833
      
1834
741
      directory.Add (&fBlackLevelRepeatDim);
1835
      
1836
741
      }
1837
    
1838
    // BlackLevel:
1839
    
1840
741
      {
1841
    
1842
741
      uint32 index = 0;
1843
      
1844
1.49k
      for (uint16 v = 0; v < rangeInfo->fBlackLevelRepeatRows; v++)
1845
756
        {
1846
        
1847
1.59k
        for (uint32 h = 0; h < rangeInfo->fBlackLevelRepeatCols; h++)
1848
837
          {
1849
          
1850
1.68k
          for (uint32 c = 0; c < rawImage.Planes (); c++)
1851
843
            {
1852
          
1853
843
            fBlackLevelData [index++] = rangeInfo->BlackLevel (v, h, c);
1854
          
1855
843
            }
1856
            
1857
837
          }
1858
          
1859
756
        }
1860
        
1861
741
      fBlackLevel.SetCount (rangeInfo->fBlackLevelRepeatRows *
1862
741
                  rangeInfo->fBlackLevelRepeatCols * rawImage.Planes ());
1863
      
1864
741
      directory.Add (&fBlackLevel);
1865
      
1866
741
      }
1867
    
1868
    // BlackLevelDeltaH:
1869
        
1870
741
    if (rangeInfo->ColumnBlackCount ())
1871
1
      {
1872
      
1873
1
      uint32 count = rangeInfo->ColumnBlackCount ();
1874
    
1875
1
      fBlackLevelDeltaHData.Allocate (count, sizeof (dng_srational));
1876
                         
1877
1
      dng_srational *blacks = (dng_srational *) fBlackLevelDeltaHData.Buffer ();
1878
      
1879
3
      for (uint32 col = 0; col < count; col++)
1880
2
        {
1881
        
1882
2
        blacks [col] = rangeInfo->ColumnBlack (col);
1883
        
1884
2
        }
1885
                         
1886
1
      fBlackLevelDeltaH.SetData  (blacks);
1887
1
      fBlackLevelDeltaH.SetCount (count );
1888
      
1889
1
      directory.Add (&fBlackLevelDeltaH);
1890
      
1891
1
      }
1892
    
1893
    // BlackLevelDeltaV:
1894
        
1895
741
    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
741
    }
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
741
  bool needs32 = false;
1926
    
1927
741
  fWhiteLevel16.SetCount (rawImage.Planes ());
1928
741
  fWhiteLevel32.SetCount (rawImage.Planes ());
1929
  
1930
1.48k
  for (uint32 c = 0; c < fWhiteLevel16.Count (); c++)
1931
747
    {
1932
    
1933
747
    fWhiteLevelData32 [c] = negative.WhiteLevel (c);
1934
    
1935
747
    if (fWhiteLevelData32 [c] > 0x0FFFF)
1936
40
      {
1937
40
      needs32 = true;
1938
40
      }
1939
      
1940
747
    fWhiteLevelData16 [c] = (uint16) fWhiteLevelData32 [c];
1941
    
1942
747
    }
1943
    
1944
741
  if (needs32)
1945
40
    {
1946
40
    directory.Add (&fWhiteLevel32);
1947
40
    }
1948
    
1949
701
  else
1950
701
    {
1951
701
    directory.Add (&fWhiteLevel16);
1952
701
    }
1953
  
1954
741
  }
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
741
  : fCFARepeatPatternDim (tcCFARepeatPatternDim,
1993
741
                  fCFARepeatPatternDimData,
1994
741
                  2)
1995
                  
1996
741
  , fCFAPattern (tcCFAPattern,
1997
741
           fCFAPatternData)
1998
          
1999
741
  , fCFAPlaneColor (tcCFAPlaneColor,
2000
741
            fCFAPlaneColorData)
2001
            
2002
741
  , fCFALayout (tcCFALayout,
2003
741
          (uint16) info.fCFALayout)
2004
  
2005
741
  , fGreenSplit (tcBayerGreenSplit,
2006
741
           info.fBayerGreenSplit)
2007
  
2008
741
  {
2009
  
2010
741
  if (info.IsColorFilterArray ())
2011
3
    {
2012
  
2013
    // CFARepeatPatternDim:
2014
    
2015
3
    fCFARepeatPatternDimData [0] = (uint16) info.fCFAPatternSize.v;
2016
3
    fCFARepeatPatternDimData [1] = (uint16) info.fCFAPatternSize.h;
2017
        
2018
3
    directory.Add (&fCFARepeatPatternDim);
2019
    
2020
    // CFAPattern:
2021
    
2022
3
    fCFAPattern.SetCount (info.fCFAPatternSize.v *
2023
3
                info.fCFAPatternSize.h);
2024
                
2025
9
    for (int32 r = 0; r < info.fCFAPatternSize.v; r++)
2026
6
      {
2027
      
2028
18
      for (int32 c = 0; c < info.fCFAPatternSize.h; c++)
2029
12
        {
2030
        
2031
12
        fCFAPatternData [r * info.fCFAPatternSize.h + c] = info.fCFAPattern [r] [c];
2032
        
2033
12
        }
2034
        
2035
6
      }
2036
        
2037
3
    directory.Add (&fCFAPattern);
2038
    
2039
    // CFAPlaneColor:
2040
    
2041
3
    fCFAPlaneColor.SetCount (info.fColorPlanes);
2042
    
2043
15
    for (uint32 j = 0; j < info.fColorPlanes; j++)
2044
12
      {
2045
    
2046
12
      fCFAPlaneColorData [j] = info.fCFAPlaneColor [j];
2047
      
2048
12
      }
2049
    
2050
3
    directory.Add (&fCFAPlaneColor);
2051
    
2052
    // CFALayout:
2053
    
2054
3
    fCFALayout.Set ((uint16) info.fCFALayout);
2055
    
2056
3
    directory.Add (&fCFALayout);
2057
    
2058
    // BayerGreenSplit:  (only include if the pattern is a Bayer pattern)
2059
      
2060
3
    if (info.fCFAPatternSize == dng_point (2, 2) &&
2061
3
      info.fColorPlanes    == 3)
2062
0
      {
2063
      
2064
0
      directory.Add (&fGreenSplit);
2065
      
2066
0
      }
2067
      
2068
3
    }
2069
2070
741
  }
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
5
  : fColorChannels (negative.ColorChannels ())
2115
  
2116
5
  , fCameraCalibration1 (tcCameraCalibration1,
2117
5
                 negative.CameraCalibration1 ())
2118
            
2119
5
  , fCameraCalibration2 (tcCameraCalibration2,
2120
5
                 negative.CameraCalibration2 ())
2121
               
2122
5
  , fCameraCalibrationSignature (tcCameraCalibrationSignature,
2123
5
                   negative.CameraCalibrationSignature ())
2124
                   
2125
5
  , fAsShotProfileName (tcAsShotProfileName,
2126
5
              negative.AsShotProfileName ())
2127
2128
5
  , fAnalogBalance (tcAnalogBalance,
2129
5
            fAnalogBalanceData,
2130
5
            fColorChannels)
2131
            
2132
5
  , fAsShotNeutral (tcAsShotNeutral,
2133
5
            fAsShotNeutralData,
2134
5
            fColorChannels)
2135
            
2136
5
  , fAsShotWhiteXY (tcAsShotWhiteXY,
2137
5
            fAsShotWhiteXYData,
2138
5
            2)
2139
                    
2140
5
  , fLinearResponseLimit (tcLinearResponseLimit,
2141
5
                  negative.LinearResponseLimitR ())
2142
                  
2143
5
  {
2144
  
2145
5
  if (fColorChannels > 1)
2146
5
    {
2147
    
2148
5
    uint32 channels2 = fColorChannels * fColorChannels;
2149
    
2150
5
    if (fCameraCalibration1.Count () == channels2)
2151
0
      {
2152
      
2153
0
      directory.Add (&fCameraCalibration1);
2154
    
2155
0
      }
2156
      
2157
5
    if (fCameraCalibration2.Count () == channels2)
2158
0
      {
2159
      
2160
0
      directory.Add (&fCameraCalibration2);
2161
    
2162
0
      }
2163
      
2164
5
    if (fCameraCalibration1.Count () == channels2 ||
2165
5
      fCameraCalibration2.Count () == channels2)
2166
0
      {
2167
      
2168
0
      if (negative.CameraCalibrationSignature ().NotEmpty ())
2169
0
        {
2170
        
2171
0
        directory.Add (&fCameraCalibrationSignature);
2172
        
2173
0
        }
2174
      
2175
0
      }
2176
      
2177
5
    if (negative.AsShotProfileName ().NotEmpty ())
2178
0
      {
2179
      
2180
0
      directory.Add (&fAsShotProfileName);
2181
        
2182
0
      }
2183
      
2184
25
    for (uint32 j = 0; j < fColorChannels; j++)
2185
20
      {
2186
      
2187
20
      fAnalogBalanceData [j] = negative.AnalogBalanceR (j);
2188
      
2189
20
      }
2190
      
2191
5
    directory.Add (&fAnalogBalance);
2192
    
2193
5
    if (negative.HasCameraNeutral ())
2194
0
      {
2195
      
2196
0
      for (uint32 k = 0; k < fColorChannels; k++)
2197
0
        {
2198
        
2199
0
        fAsShotNeutralData [k] = negative.CameraNeutralR (k);
2200
        
2201
0
        }
2202
      
2203
0
      directory.Add (&fAsShotNeutral);
2204
      
2205
0
      }
2206
      
2207
5
    else if (negative.HasCameraWhiteXY ())
2208
0
      {
2209
      
2210
0
      negative.GetCameraWhiteXY (fAsShotWhiteXYData [0],
2211
0
                     fAsShotWhiteXYData [1]);
2212
                     
2213
0
      directory.Add (&fAsShotWhiteXY);
2214
      
2215
0
      }
2216
    
2217
5
    directory.Add (&fLinearResponseLimit);
2218
    
2219
5
    }
2220
  
2221
5
  }
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
5
  : fCalibrationIlluminant1 (tcCalibrationIlluminant1,
2288
5
                 (uint16) profile.CalibrationIlluminant1 ())
2289
                 
2290
5
  , fCalibrationIlluminant2 (tcCalibrationIlluminant2,
2291
5
                 (uint16) profile.CalibrationIlluminant2 ())
2292
  
2293
5
  , fColorMatrix1 (tcColorMatrix1,
2294
5
             profile.ColorMatrix1 ())
2295
            
2296
5
  , fColorMatrix2 (tcColorMatrix2,
2297
5
             profile.ColorMatrix2 ())
2298
2299
5
  , fForwardMatrix1 (tcForwardMatrix1,
2300
5
             profile.ForwardMatrix1 ())
2301
            
2302
5
  , fForwardMatrix2 (tcForwardMatrix2,
2303
5
             profile.ForwardMatrix2 ())
2304
            
2305
5
  , fReductionMatrix1 (tcReductionMatrix1,
2306
5
               profile.ReductionMatrix1 ())
2307
            
2308
5
  , fReductionMatrix2 (tcReductionMatrix2,
2309
5
               profile.ReductionMatrix2 ())
2310
               
2311
5
  , fProfileName (tcProfileName,
2312
5
            profile.Name (),
2313
5
            false)
2314
               
2315
5
  , fProfileCalibrationSignature (tcProfileCalibrationSignature,
2316
5
                    profile.ProfileCalibrationSignature (),
2317
5
                    false)
2318
            
2319
5
  , fEmbedPolicyTag (tcProfileEmbedPolicy,
2320
5
             profile.EmbedPolicy ())
2321
             
2322
5
  , fCopyrightTag (tcProfileCopyright,
2323
5
             profile.Copyright (),
2324
5
             false)
2325
             
2326
5
  , fHueSatMapDims (tcProfileHueSatMapDims, 
2327
5
            fHueSatMapDimData,
2328
5
            3)
2329
    
2330
5
  , fHueSatData1 (tcProfileHueSatMapData1,
2331
5
            ttFloat,
2332
5
            profile.HueSatDeltas1 ().DeltasCount () * 3,
2333
5
            profile.HueSatDeltas1 ().GetConstDeltas ())
2334
            
2335
5
  , fHueSatData2 (tcProfileHueSatMapData2,
2336
5
            ttFloat,
2337
5
            profile.HueSatDeltas2 ().DeltasCount () * 3,
2338
5
            profile.HueSatDeltas2 ().GetConstDeltas ())
2339
            
2340
5
  , fHueSatMapEncodingTag (tcProfileHueSatMapEncoding,
2341
5
                 profile.HueSatMapEncoding ())
2342
             
2343
5
  , fLookTableDims (tcProfileLookTableDims,
2344
5
            fLookTableDimData,
2345
5
            3)
2346
            
2347
5
  , fLookTableData (tcProfileLookTableData,
2348
5
            ttFloat,
2349
5
            profile.LookTable ().DeltasCount () * 3,
2350
5
              profile.LookTable ().GetConstDeltas ())
2351
            
2352
5
  , fLookTableEncodingTag (tcProfileLookTableEncoding,
2353
5
                 profile.LookTableEncoding ())
2354
             
2355
5
  , fBaselineExposureOffsetTag (tcBaselineExposureOffset,
2356
5
                  profile.BaselineExposureOffset ())
2357
             
2358
5
  , fDefaultBlackRenderTag (tcDefaultBlackRender,
2359
5
                profile.DefaultBlackRender ())
2360
             
2361
5
  , fToneCurveBuffer ()
2362
            
2363
5
  , fToneCurveTag (tcProfileToneCurve,
2364
5
             ttFloat,
2365
5
             0,
2366
5
             NULL)
2367
2368
5
  {
2369
  
2370
5
  if (profile.HasColorMatrix1 ())
2371
5
    {
2372
  
2373
5
    uint32 colorChannels = profile.ColorMatrix1 ().Rows ();
2374
    
2375
5
    directory.Add (&fCalibrationIlluminant1);
2376
    
2377
5
    directory.Add (&fColorMatrix1);
2378
    
2379
5
    if (fForwardMatrix1.Count () == colorChannels * 3)
2380
0
      {
2381
      
2382
0
      directory.Add (&fForwardMatrix1);
2383
2384
0
      }
2385
    
2386
5
    if (colorChannels > 3 && fReductionMatrix1.Count () == colorChannels * 3)
2387
0
      {
2388
      
2389
0
      directory.Add (&fReductionMatrix1);
2390
      
2391
0
      }
2392
      
2393
5
    if (profile.HasColorMatrix2 ())
2394
0
      {
2395
    
2396
0
      directory.Add (&fCalibrationIlluminant2);
2397
    
2398
0
      directory.Add (&fColorMatrix2);
2399
        
2400
0
      if (fForwardMatrix2.Count () == colorChannels * 3)
2401
0
        {
2402
        
2403
0
        directory.Add (&fForwardMatrix2);
2404
2405
0
        }
2406
    
2407
0
      if (colorChannels > 3 && fReductionMatrix2.Count () == colorChannels * 3)
2408
0
        {
2409
        
2410
0
        directory.Add (&fReductionMatrix2);
2411
        
2412
0
        }
2413
  
2414
0
      }
2415
      
2416
5
    if (profile.Name ().NotEmpty ())
2417
5
      {
2418
      
2419
5
      directory.Add (&fProfileName);
2420
2421
5
      }
2422
      
2423
5
    if (profile.ProfileCalibrationSignature ().NotEmpty ())
2424
0
      {
2425
      
2426
0
      directory.Add (&fProfileCalibrationSignature);
2427
      
2428
0
      }
2429
      
2430
5
    directory.Add (&fEmbedPolicyTag);
2431
    
2432
5
    if (profile.Copyright ().NotEmpty ())
2433
0
      {
2434
      
2435
0
      directory.Add (&fCopyrightTag);
2436
      
2437
0
      }
2438
    
2439
5
    bool haveHueSat1 = profile.HueSatDeltas1 ().IsValid ();
2440
    
2441
5
    bool haveHueSat2 = profile.HueSatDeltas2 ().IsValid () &&
2442
5
               profile.HasColorMatrix2 ();
2443
2444
5
    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
5
    if (haveHueSat1)
2488
0
      {
2489
      
2490
0
      directory.Add (&fHueSatData1);
2491
      
2492
0
      }
2493
      
2494
5
    if (haveHueSat2)
2495
0
      {
2496
      
2497
0
      directory.Add (&fHueSatData2);
2498
      
2499
0
      }
2500
      
2501
5
    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
5
    if (profile.BaselineExposureOffset ().IsValid ())
2536
5
      {
2537
2538
5
      if (profile.BaselineExposureOffset ().As_real64 () != 0.0)
2539
0
        {
2540
      
2541
0
        directory.Add (&fBaselineExposureOffsetTag);
2542
2543
0
        }
2544
        
2545
5
      }
2546
      
2547
5
    if (profile.DefaultBlackRender () != defaultBlackRender_Auto)
2548
0
      {
2549
2550
0
      directory.Add (&fDefaultBlackRenderTag);
2551
2552
0
      }
2553
    
2554
5
    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
5
    }
2586
  
2587
5
  }
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
741
  : tag_data_ptr (tcNoiseProfile,
2646
741
            ttDouble,
2647
741
            2 * profile.NumFunctions (),
2648
741
            fValues)
2649
2650
741
  {
2651
2652
741
  DNG_REQUIRE (profile.NumFunctions () <= kMaxColorPlanes,
2653
741
         "Too many noise functions in tag_dng_noise_profile.");
2654
2655
741
  for (uint32 i = 0; i < profile.NumFunctions (); i++)
2656
0
    {
2657
2658
0
    fValues [(2 * i)  ] = profile.NoiseFunction (i).Scale  ();
2659
0
    fValues [(2 * i) + 1] = profile.NoiseFunction (i).Offset ();
2660
2661
0
    }
2662
  
2663
741
  }
2664
    
2665
/*****************************************************************************/
2666
2667
dng_image_writer::dng_image_writer ()
2668
4.79k
  {
2669
  
2670
4.79k
  }
2671
2672
/*****************************************************************************/
2673
2674
dng_image_writer::~dng_image_writer ()
2675
4.79k
  {
2676
  
2677
4.79k
  }
2678
                
2679
/*****************************************************************************/
2680
2681
uint32 dng_image_writer::CompressedBufferSize (const dng_ifd &ifd,
2682
                         uint32 uncompressedSize)
2683
3.32k
  {
2684
  
2685
3.32k
  switch (ifd.fCompression)
2686
3.32k
    {
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
284
    case ccDeflate:
2698
284
      {
2699
    
2700
      // ZLib says maximum is source size + 0.1% + 12 bytes.
2701
      
2702
284
      return SafeUint32Add (SafeUint32Add (uncompressedSize,
2703
284
                         uncompressedSize >> 8), 64);
2704
      
2705
0
      }
2706
      
2707
217
    case ccJPEG:
2708
217
      {
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
217
      if (ifd.fBitsPerSample [0] <= 8)
2714
41
        {
2715
        
2716
41
        return SafeUint32Mult (uncompressedSize, 2);
2717
        
2718
41
        }
2719
        
2720
176
      break;
2721
  
2722
217
      }
2723
      
2724
2.81k
    default:
2725
2.81k
      break;
2726
    
2727
3.32k
    }
2728
  
2729
2.99k
  return 0;
2730
  
2731
3.32k
  }
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
3.90k
  {
2802
  
2803
3.90k
  const uint32 dRowStep = cols * channels;
2804
  
2805
1.39M
  for (uint32 row = 0; row < rows; row++)
2806
1.39M
    {
2807
    
2808
480M
    for (uint32 col = cols - 1; col > 0; col--)
2809
479M
      {
2810
      
2811
958M
      for (uint32 channel = 0; channel < channels; channel++)
2812
479M
        {
2813
        
2814
479M
        dPtr [col * channels + channel] -= dPtr [(col - 1) * channels + channel];
2815
        
2816
479M
        }
2817
      
2818
479M
      }
2819
    
2820
1.39M
    dPtr += dRowStep;
2821
    
2822
1.39M
    }
2823
2824
3.90k
  }
2825
  
2826
/*****************************************************************************/
2827
2828
inline void EncodeDeltaBytes (uint8 *bytePtr, int32 cols, int32 channels)
2829
602k
  {
2830
  
2831
602k
  if (channels == 1)
2832
602k
    {
2833
    
2834
602k
    bytePtr += (cols - 1);
2835
    
2836
602k
    uint8 this0 = bytePtr [0];
2837
    
2838
284M
    for (int32 col = 1; col < cols; col++)
2839
284M
      {
2840
      
2841
284M
      uint8 prev0 = bytePtr [-1];
2842
      
2843
284M
      this0 -= prev0;
2844
      
2845
284M
      bytePtr [0] = this0;
2846
      
2847
284M
      this0 = prev0;
2848
      
2849
284M
      bytePtr -= 1;
2850
2851
284M
      }
2852
  
2853
602k
    }
2854
    
2855
0
  else if (channels == 3)
2856
0
    {
2857
    
2858
0
    bytePtr += (cols - 1) * 3;
2859
    
2860
0
    uint8 this0 = bytePtr [0];
2861
0
    uint8 this1 = bytePtr [1];
2862
0
    uint8 this2 = bytePtr [2];
2863
    
2864
0
    for (int32 col = 1; col < cols; col++)
2865
0
      {
2866
      
2867
0
      uint8 prev0 = bytePtr [-3];
2868
0
      uint8 prev1 = bytePtr [-2];
2869
0
      uint8 prev2 = bytePtr [-1];
2870
      
2871
0
      this0 -= prev0;
2872
0
      this1 -= prev1;
2873
0
      this2 -= prev2;
2874
      
2875
0
      bytePtr [0] = this0;
2876
0
      bytePtr [1] = this1;
2877
0
      bytePtr [2] = this2;
2878
      
2879
0
      this0 = prev0;
2880
0
      this1 = prev1;
2881
0
      this2 = prev2;
2882
      
2883
0
      bytePtr -= 3;
2884
2885
0
      }
2886
  
2887
0
    }
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
602k
  }
2908
2909
/*****************************************************************************/
2910
2911
static void EncodeFPDelta (uint8 *buffer,
2912
               uint8 *temp,
2913
               int32 cols,
2914
               int32 channels,
2915
               int32 bytesPerSample)
2916
602k
  {
2917
  
2918
602k
  int32 rowIncrement = cols * channels;
2919
  
2920
602k
  if (bytesPerSample == 2)
2921
602k
    {
2922
    
2923
602k
    const uint8 *src = buffer;
2924
    
2925
    #if qDNGBigEndian
2926
    uint8 *dst0 = temp;
2927
    uint8 *dst1 = temp + rowIncrement;
2928
    #else
2929
602k
    uint8 *dst1 = temp;
2930
602k
    uint8 *dst0 = temp + rowIncrement;
2931
602k
    #endif
2932
        
2933
143M
    for (int32 col = 0; col < rowIncrement; ++col)
2934
142M
      {
2935
      
2936
142M
      dst0 [col] = src [0];
2937
142M
      dst1 [col] = src [1];
2938
      
2939
142M
      src += 2;
2940
      
2941
142M
      }
2942
      
2943
602k
    }
2944
    
2945
112
  else if (bytesPerSample == 3)
2946
112
    {
2947
    
2948
112
    const uint8 *src = buffer;
2949
    
2950
112
    uint8 *dst0 = temp;
2951
112
    uint8 *dst1 = temp + rowIncrement;
2952
112
    uint8 *dst2 = temp + rowIncrement * 2;
2953
        
2954
2.16k
    for (int32 col = 0; col < rowIncrement; ++col)
2955
2.04k
      {
2956
      
2957
2.04k
      dst0 [col] = src [0];
2958
2.04k
      dst1 [col] = src [1];
2959
2.04k
      dst2 [col] = src [2];
2960
      
2961
2.04k
      src += 3;
2962
      
2963
2.04k
      }
2964
      
2965
112
    }
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
602k
  EncodeDeltaBytes (temp, cols*bytesPerSample, channels);
2999
  
3000
602k
  memcpy (buffer, temp, cols*bytesPerSample*channels);
3001
  
3002
602k
  }
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
112k
  {
3011
  
3012
112k
  switch (ifd.fPredictor)
3013
112k
    {
3014
    
3015
3.90k
    case cpHorizontalDifference:
3016
3.90k
    case cpHorizontalDifferenceX2:
3017
3.90k
    case cpHorizontalDifferenceX4:
3018
3.90k
      {
3019
      
3020
3.90k
      int32 xFactor = 1;
3021
      
3022
3.90k
      if (ifd.fPredictor == cpHorizontalDifferenceX2)
3023
0
        {
3024
0
        xFactor = 2;
3025
0
        }
3026
        
3027
3.90k
      else if (ifd.fPredictor == cpHorizontalDifferenceX4)
3028
0
        {
3029
0
        xFactor = 4;
3030
0
        }
3031
      
3032
3.90k
      switch (buffer.fPixelType)
3033
3.90k
        {
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
3.90k
        case ttLong:
3060
3.90k
          {
3061
          
3062
3.90k
          EncodeDelta32 ((uint32 *) buffer.fData,
3063
3.90k
                   buffer.fArea.H (),
3064
3.90k
                   buffer.fArea.W () / xFactor,
3065
3.90k
                   buffer.fPlanes    * xFactor);
3066
          
3067
3.90k
          return;
3068
          
3069
0
          }
3070
          
3071
0
        default:
3072
0
          break;
3073
          
3074
3.90k
        }
3075
      
3076
0
      break;
3077
      
3078
3.90k
      }
3079
      
3080
810
    case cpFloatingPoint:
3081
810
    case cpFloatingPointX2:
3082
810
    case cpFloatingPointX4:
3083
810
      {
3084
      
3085
810
      int32 xFactor = 1;
3086
      
3087
810
      if (ifd.fPredictor == cpFloatingPointX2)
3088
0
        {
3089
0
        xFactor = 2;
3090
0
        }
3091
        
3092
810
      else if (ifd.fPredictor == cpFloatingPointX4)
3093
0
        {
3094
0
        xFactor = 4;
3095
0
        }
3096
      
3097
810
      if (buffer.fRowStep < 0)
3098
0
        {
3099
0
        ThrowProgramError ("Row step may not be negative");
3100
0
        }
3101
810
      uint32 tempBufferSize = SafeUint32Mult (
3102
810
        static_cast<uint32>(buffer.fRowStep),
3103
810
        buffer.fPixelSize);
3104
      
3105
810
      if (!tempBuffer.Get () || tempBuffer->LogicalSize () < tempBufferSize)
3106
244
        {
3107
        
3108
244
        tempBuffer.Reset (host.Allocate (tempBufferSize));
3109
        
3110
244
        }
3111
        
3112
603k
      for (int32 row = buffer.fArea.t; row < buffer.fArea.b; row++)
3113
602k
        {
3114
        
3115
602k
        EncodeFPDelta ((uint8 *) buffer.DirtyPixel (row, buffer.fArea.l, buffer.fPlane),
3116
602k
                 tempBuffer->Buffer_uint8 (),
3117
602k
                 buffer.fArea.W () / xFactor,
3118
602k
                 buffer.fPlanes    * xFactor,
3119
602k
                 buffer.fPixelSize);
3120
        
3121
602k
        }
3122
      
3123
810
      return;
3124
      
3125
810
      }
3126
      
3127
107k
    default:
3128
107k
      break;
3129
    
3130
112k
    }
3131
  
3132
107k
  if (ifd.fPredictor != cpNullPredictor)
3133
0
    {
3134
    
3135
0
    ThrowProgramError ();
3136
    
3137
0
    }
3138
  
3139
107k
  }
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
5.38k
  {
3623
  
3624
5.38k
  dng_jpeg_stream_dest *dest = (dng_jpeg_stream_dest *) cinfo->dest;
3625
3626
5.38k
  dest->pub.next_output_byte = dest->fBuffer;
3627
5.38k
  dest->pub.free_in_buffer   = sizeof (dest->fBuffer);
3628
  
3629
5.38k
  }
3630
3631
/*****************************************************************************/
3632
3633
static boolean dng_empty_output_buffer (j_compress_ptr cinfo)
3634
33.2k
  {
3635
  
3636
33.2k
  dng_jpeg_stream_dest *dest = (dng_jpeg_stream_dest *) cinfo->dest;
3637
  
3638
33.2k
  dest->fStream->Put (dest->fBuffer, sizeof (dest->fBuffer));
3639
3640
33.2k
  dest->pub.next_output_byte = dest->fBuffer;
3641
33.2k
  dest->pub.free_in_buffer   = sizeof (dest->fBuffer);
3642
3643
33.2k
  return TRUE;
3644
  
3645
33.2k
  }
3646
3647
/*****************************************************************************/
3648
3649
static void dng_term_destination (j_compress_ptr cinfo)
3650
5.38k
  {
3651
  
3652
5.38k
  dng_jpeg_stream_dest *dest = (dng_jpeg_stream_dest *) cinfo->dest;
3653
  
3654
5.38k
  uint32 datacount = sizeof (dest->fBuffer) -
3655
5.38k
             (uint32) dest->pub.free_in_buffer;
3656
  
3657
5.38k
  if (datacount)
3658
5.38k
    {
3659
5.38k
    dest->fStream->Put (dest->fBuffer, datacount);
3660
5.38k
    }
3661
3662
5.38k
  }
3663
3664
/*****************************************************************************/
3665
3666
static void jpeg_set_adobe_quality (struct jpeg_compress_struct *cinfo,
3667
                  int32 quality)
3668
5.38k
  {
3669
  
3670
  // If out of range, map to default.
3671
    
3672
5.38k
  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
5.38k
  bool useChromaDownsampling = (quality <= 6);
3680
    
3681
  // Approximate mapping from Adobe quality levels to LibJPEG levels.
3682
  
3683
5.38k
  const int kLibJPEGQuality [13] =
3684
5.38k
    {
3685
5.38k
    5, 11, 23, 34, 46, 63, 76, 77, 86, 90, 94, 97, 99
3686
5.38k
    };
3687
    
3688
5.38k
  quality = kLibJPEGQuality [quality];
3689
  
3690
5.38k
  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
5.38k
  if (!useChromaDownsampling)
3696
5.13k
    {
3697
    
3698
5.13k
    cinfo->comp_info [0].h_samp_factor = 1;
3699
5.13k
    cinfo->comp_info [0].h_samp_factor = 1;
3700
    
3701
5.13k
    }
3702
        
3703
5.38k
  }
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
112k
  {
3717
  
3718
112k
  switch (ifd.fCompression)
3719
112k
    {
3720
    
3721
92.4k
    case ccUncompressed:
3722
92.4k
      {
3723
      
3724
      // Special case support for when we save to 8-bits from
3725
      // 16-bit data.
3726
      
3727
92.4k
      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
92.4k
      else
3745
92.4k
        {
3746
  
3747
        // Swap bytes if required.
3748
        
3749
92.4k
        if (stream.SwapBytes ())
3750
0
          {
3751
          
3752
0
          ByteSwapBuffer (host, buffer);
3753
                    
3754
0
          }
3755
      
3756
        // Write the bytes.
3757
        
3758
92.4k
        stream.Put (buffer.fData, buffer.fRowStep *
3759
92.4k
                      buffer.fArea.H () *
3760
92.4k
                      buffer.fPixelSize);
3761
                      
3762
92.4k
        }
3763
      
3764
92.4k
      break;
3765
      
3766
0
      }
3767
      
3768
0
    case ccLZW:
3769
4.71k
    case ccDeflate:
3770
4.71k
      {
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
4.71k
      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
4.71k
      uint32 sBytes = buffer.fRowStep *
3787
4.71k
              buffer.fArea.H () *
3788
4.71k
              buffer.fPixelSize;
3789
        
3790
4.71k
      uint8 *sBuffer = (uint8 *) buffer.fData;
3791
        
3792
4.71k
      uint32 dBytes = 0;
3793
        
3794
4.71k
      uint8 *dBuffer = compressedBuffer->Buffer_uint8 ();
3795
      
3796
4.71k
      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
4.71k
      else
3809
4.71k
        {
3810
        
3811
4.71k
        uLongf dCount = compressedBuffer->LogicalSize ();
3812
        
3813
4.71k
        int32 level = Z_DEFAULT_COMPRESSION;
3814
        
3815
4.71k
        if (ifd.fCompressionQuality >= Z_BEST_SPEED &&
3816
4.71k
          ifd.fCompressionQuality <= Z_BEST_COMPRESSION)
3817
0
          {
3818
          
3819
0
          level = ifd.fCompressionQuality;
3820
          
3821
0
          }
3822
        
3823
4.71k
        int zResult = ::compress2 (dBuffer,
3824
4.71k
                       &dCount,
3825
4.71k
                       sBuffer,
3826
4.71k
                       sBytes,
3827
4.71k
                       level);
3828
                      
3829
4.71k
        if (zResult != Z_OK)
3830
0
          {
3831
          
3832
0
          ThrowMemoryFull ();
3833
          
3834
0
          }
3835
3836
4.71k
        dBytes = (uint32) dCount;
3837
        
3838
4.71k
        }
3839
                    
3840
4.71k
      if (dBytes > compressedBuffer->LogicalSize ())
3841
0
        {
3842
        
3843
0
        DNG_REPORT ("Compression output buffer overflow");
3844
        
3845
0
        ThrowProgramError ();
3846
        
3847
0
        }
3848
                    
3849
4.71k
      stream.Put (dBuffer, dBytes);
3850
        
3851
4.71k
      return;
3852
3853
0
      }
3854
      
3855
10.5k
    case ccJPEG:
3856
10.5k
      {
3857
      
3858
10.5k
      dng_pixel_buffer temp (buffer);
3859
        
3860
10.5k
      if (buffer.fPixelType == ttByte)
3861
354
        {
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
354
        temp.fData = compressedBuffer->Buffer ();
3867
        
3868
354
        temp.fPixelType = ttShort;
3869
354
        temp.fPixelSize = 2;
3870
        
3871
354
        temp.CopyArea (buffer,
3872
354
                 buffer.fArea,
3873
354
                 buffer.fPlane,
3874
354
                 buffer.fPlanes);
3875
        
3876
354
        }
3877
        
3878
10.5k
      EncodeLosslessJPEG ((const uint16 *) temp.fData,
3879
10.5k
                temp.fArea.H (),
3880
10.5k
                temp.fArea.W (),
3881
10.5k
                temp.fPlanes,
3882
10.5k
                ifd.fBitsPerSample [0],
3883
10.5k
                temp.fRowStep,
3884
10.5k
                temp.fColStep,
3885
10.5k
                stream);
3886
                    
3887
10.5k
      break;
3888
      
3889
0
      }
3890
      
3891
0
    #if qDNGUseLibJPEG
3892
    
3893
4.89k
    case ccLossyJPEG:
3894
4.89k
      {
3895
      
3896
4.89k
      struct jpeg_compress_struct cinfo;
3897
  
3898
      // Setup the error manager.
3899
      
3900
4.89k
      struct jpeg_error_mgr jerr;
3901
3902
4.89k
      cinfo.err = jpeg_std_error (&jerr);
3903
      
3904
4.89k
      jerr.error_exit     = dng_error_exit;
3905
4.89k
      jerr.output_message = dng_output_message;
3906
  
3907
4.89k
      try
3908
4.89k
        {
3909
        
3910
        // Create the compression context.
3911
3912
4.89k
        jpeg_create_compress (&cinfo);
3913
        
3914
        // Setup the destination manager to write to stream.
3915
        
3916
4.89k
        dng_jpeg_stream_dest dest;
3917
        
3918
4.89k
        dest.fStream = &stream;
3919
        
3920
4.89k
        dest.pub.init_destination    = dng_init_destination;
3921
4.89k
        dest.pub.empty_output_buffer = dng_empty_output_buffer;
3922
4.89k
        dest.pub.term_destination    = dng_term_destination;
3923
        
3924
4.89k
        cinfo.dest = &dest.pub;
3925
        
3926
        // Setup basic image info.
3927
        
3928
4.89k
        cinfo.image_width      = buffer.fArea.W ();
3929
4.89k
        cinfo.image_height     = buffer.fArea.H ();
3930
4.89k
        cinfo.input_components = buffer.fPlanes;
3931
        
3932
4.89k
        switch (buffer.fPlanes)
3933
4.89k
          {
3934
          
3935
4.89k
          case 1:
3936
4.89k
            cinfo.in_color_space = JCS_GRAYSCALE;
3937
4.89k
            break;
3938
            
3939
0
          case 3:
3940
0
            cinfo.in_color_space = JCS_RGB;
3941
0
            break;
3942
            
3943
2
          case 4:
3944
2
            cinfo.in_color_space = JCS_CMYK;
3945
2
            break;
3946
            
3947
0
          default:
3948
0
            ThrowProgramError ();
3949
            
3950
4.89k
          }
3951
          
3952
        // Setup the compression parameters.
3953
3954
4.89k
        jpeg_set_defaults (&cinfo);
3955
        
3956
4.89k
        jpeg_set_adobe_quality (&cinfo, ifd.fCompressionQuality);
3957
        
3958
        // Write the JPEG header.
3959
        
3960
4.89k
        jpeg_start_compress (&cinfo, TRUE);
3961
        
3962
        // Write the scanlines.
3963
        
3964
2.90M
        for (int32 row = buffer.fArea.t; row < buffer.fArea.b; row++)
3965
2.90M
          {
3966
          
3967
2.90M
          uint8 *sampArray [1];
3968
    
3969
2.90M
          sampArray [0] = buffer.DirtyPixel_uint8 (row,
3970
2.90M
                               buffer.fArea.l,
3971
2.90M
                               0);
3972
3973
2.90M
          jpeg_write_scanlines (&cinfo, sampArray, 1);
3974
          
3975
2.90M
          }
3976
3977
        // Cleanup.
3978
          
3979
4.89k
        jpeg_finish_compress (&cinfo);
3980
3981
4.89k
        jpeg_destroy_compress (&cinfo);
3982
          
3983
4.89k
        }
3984
        
3985
4.89k
      catch (...)
3986
4.89k
        {
3987
        
3988
0
        jpeg_destroy_compress (&cinfo);
3989
        
3990
0
        throw;
3991
        
3992
0
        }
3993
        
3994
4.89k
      return;
3995
      
3996
4.89k
      }
3997
      
3998
0
    #endif
3999
      
4000
0
    default:
4001
0
      {
4002
      
4003
0
      ThrowProgramError ();
4004
      
4005
0
      }
4006
      
4007
112k
    }
4008
  
4009
112k
  }
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
487
  {
4018
  
4019
487
  #if qDNGUseLibJPEG
4020
    
4021
487
  dng_memory_stream stream (host.Allocator ());
4022
  
4023
487
  struct jpeg_compress_struct cinfo;
4024
4025
  // Setup the error manager.
4026
  
4027
487
  struct jpeg_error_mgr jerr;
4028
4029
487
  cinfo.err = jpeg_std_error (&jerr);
4030
  
4031
487
  jerr.error_exit     = dng_error_exit;
4032
487
  jerr.output_message = dng_output_message;
4033
4034
487
  try
4035
487
    {
4036
    
4037
    // Create the compression context.
4038
4039
487
    jpeg_create_compress (&cinfo);
4040
    
4041
    // Setup the destination manager to write to stream.
4042
    
4043
487
    dng_jpeg_stream_dest dest;
4044
    
4045
487
    dest.fStream = &stream;
4046
    
4047
487
    dest.pub.init_destination    = dng_init_destination;
4048
487
    dest.pub.empty_output_buffer = dng_empty_output_buffer;
4049
487
    dest.pub.term_destination    = dng_term_destination;
4050
    
4051
487
    cinfo.dest = &dest.pub;
4052
    
4053
    // Setup basic image info.
4054
    
4055
487
    cinfo.image_width      = image.Bounds ().W ();
4056
487
    cinfo.image_height     = image.Bounds ().H ();
4057
487
    cinfo.input_components = image.Planes ();
4058
    
4059
487
    switch (image.Planes ())
4060
487
      {
4061
      
4062
485
      case 1:
4063
485
        cinfo.in_color_space = JCS_GRAYSCALE;
4064
485
        break;
4065
        
4066
2
      case 3:
4067
2
        cinfo.in_color_space = JCS_RGB;
4068
2
        break;
4069
        
4070
0
      default:
4071
0
        ThrowProgramError ();
4072
        
4073
487
      }
4074
      
4075
    // Setup the compression parameters.
4076
4077
487
    jpeg_set_defaults (&cinfo);
4078
    
4079
487
    jpeg_set_adobe_quality (&cinfo, quality);
4080
    
4081
    // Find some preview information based on the compression settings.
4082
    
4083
487
    preview.fPreviewSize = image.Size ();
4084
  
4085
487
    if (image.Planes () == 1)
4086
485
      {
4087
      
4088
485
      preview.fPhotometricInterpretation = piBlackIsZero;
4089
      
4090
485
      }
4091
      
4092
2
    else
4093
2
      {
4094
      
4095
2
      preview.fPhotometricInterpretation = piYCbCr;
4096
      
4097
2
      preview.fYCbCrSubSampling.h  = cinfo.comp_info [0].h_samp_factor;
4098
2
      preview.fYCbCrSubSampling.v  = cinfo.comp_info [0].v_samp_factor;
4099
      
4100
2
      }
4101
    
4102
    // Write the JPEG header.
4103
    
4104
487
    jpeg_start_compress (&cinfo, TRUE);
4105
    
4106
    // Write the scanlines.
4107
    
4108
487
    dng_pixel_buffer buffer (image.Bounds (), 0, image.Planes (), ttByte,
4109
487
       pcInterleaved, NULL);
4110
    
4111
487
    AutoPtr<dng_memory_block> bufferData (host.Allocate (buffer.fRowStep));
4112
    
4113
487
    buffer.fData = bufferData->Buffer ();
4114
    
4115
209k
    for (uint32 row = 0; row < cinfo.image_height; row++)
4116
208k
      {
4117
      
4118
208k
      buffer.fArea.t = row;
4119
208k
      buffer.fArea.b = row + 1;
4120
      
4121
208k
      image.Get (buffer);
4122
      
4123
208k
      uint8 *sampArray [1];
4124
4125
208k
      sampArray [0] = buffer.DirtyPixel_uint8 (row,
4126
208k
                           buffer.fArea.l,
4127
208k
                           0);
4128
4129
208k
      jpeg_write_scanlines (&cinfo, sampArray, 1);
4130
      
4131
208k
      }
4132
4133
    // Cleanup.
4134
      
4135
487
    jpeg_finish_compress (&cinfo);
4136
4137
487
    jpeg_destroy_compress (&cinfo);
4138
      
4139
487
    }
4140
    
4141
487
  catch (...)
4142
487
    {
4143
    
4144
0
    jpeg_destroy_compress (&cinfo);
4145
    
4146
0
    throw;
4147
    
4148
0
    }
4149
           
4150
487
  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
487
  }
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
112k
  {
4178
  
4179
  // Create pixel buffer to hold uncompressed tile.
4180
  
4181
112k
  dng_pixel_buffer buffer (tileArea, 0, ifd.fSamplesPerPixel,
4182
112k
     image.PixelType(), pcInterleaved, uncompressedBuffer->Buffer());
4183
  
4184
  // Get the uncompressed data.
4185
  
4186
112k
  image.Get (buffer, dng_image::edge_zero);
4187
  
4188
  // Deal with sub-tile blocks.
4189
  
4190
112k
  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
112k
  if (ifd.fSampleFormat [0] == sfFloatingPoint)
4203
8.51k
    {
4204
    
4205
8.51k
    if (ifd.fBitsPerSample [0] == 16)
4206
803
      {
4207
      
4208
803
      uint32 *srcPtr = (uint32 *) buffer.fData;
4209
803
      uint16 *dstPtr = (uint16 *) buffer.fData;
4210
      
4211
803
      uint32 pixels = tileArea.W () * tileArea.H () * buffer.fPlanes;
4212
      
4213
142M
      for (uint32 j = 0; j < pixels; j++)
4214
142M
        {
4215
        
4216
142M
        dstPtr [j] = DNG_FloatToHalf (srcPtr [j]);
4217
        
4218
142M
        }
4219
        
4220
803
      buffer.fPixelSize = 2;
4221
      
4222
803
      }
4223
      
4224
8.51k
    if (ifd.fBitsPerSample [0] == 24)
4225
7
      {
4226
      
4227
7
      uint32 *srcPtr = (uint32 *) buffer.fData;
4228
7
      uint8  *dstPtr = (uint8  *) buffer.fData;
4229
      
4230
7
      uint32 pixels = tileArea.W () * tileArea.H () * buffer.fPlanes;
4231
      
4232
7
      if (stream.BigEndian () || ifd.fPredictor == cpFloatingPoint   ||
4233
7
                     ifd.fPredictor == cpFloatingPointX2 ||
4234
7
                     ifd.fPredictor == cpFloatingPointX4)
4235
7
        {
4236
      
4237
2.05k
        for (uint32 j = 0; j < pixels; j++)
4238
2.04k
          {
4239
          
4240
2.04k
          DNG_FloatToFP24 (srcPtr [j], dstPtr);
4241
          
4242
2.04k
          dstPtr += 3;
4243
          
4244
2.04k
          }
4245
          
4246
7
        }
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
7
      buffer.fPixelSize = 3;
4269
      
4270
7
      }
4271
    
4272
8.51k
    }
4273
  
4274
  // Run predictor.
4275
  
4276
112k
  EncodePredictor (host,
4277
112k
           ifd,
4278
112k
           buffer,
4279
112k
           tempBuffer);
4280
    
4281
  // Adjust pixel buffer for fake channels.
4282
  
4283
112k
  if (fakeChannels > 1)
4284
3
    {
4285
    
4286
3
    buffer.fPlanes  *= fakeChannels;
4287
3
    buffer.fColStep *= fakeChannels;
4288
    
4289
3
    buffer.fArea.r = buffer.fArea.l + (buffer.fArea.W () / fakeChannels);
4290
    
4291
3
    }
4292
    
4293
  // Compress (if required) and write out the data.
4294
  
4295
112k
  WriteData (host,
4296
112k
         ifd,
4297
112k
         stream,
4298
112k
         buffer,
4299
112k
         compressedBuffer);
4300
         
4301
112k
  }
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
3.32k
  {
4575
  
4576
  // Deal with row interleaved images.
4577
  
4578
3.32k
  if (ifd.fRowInterleaveFactor > 1 &&
4579
3.32k
    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
3.32k
  uint32 bytesPerSample = TagTypeSize (image.PixelType ());
4603
  
4604
3.32k
  uint32 bytesPerPixel = SafeUint32Mult (ifd.fSamplesPerPixel,
4605
3.32k
                       bytesPerSample);
4606
  
4607
3.32k
  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
3.32k
  uint32 subTileLength = ifd.fTileLength;
4613
  
4614
3.32k
  if (ifd.TileByteCount (ifd.TileArea (0, 0)) != 0)
4615
2.81k
    {
4616
    
4617
2.81k
    subTileLength = Pin_uint32 (ifd.fSubTileBlockRows,
4618
2.81k
                  kImageBufferSize / tileRowBytes, 
4619
2.81k
                  ifd.fTileLength);
4620
          
4621
    // Don't split sub-tiles across subTileBlocks.
4622
    
4623
2.81k
    subTileLength = subTileLength / ifd.fSubTileBlockRows
4624
2.81k
                    * ifd.fSubTileBlockRows;
4625
                  
4626
2.81k
    }
4627
    
4628
  // Find size of uncompressed buffer.
4629
  
4630
3.32k
  uint32 uncompressedSize = SafeUint32Mult(subTileLength, tileRowBytes);
4631
  
4632
  // Find size of compressed buffer, if required.
4633
  
4634
3.32k
  uint32 compressedSize = CompressedBufferSize (ifd, uncompressedSize);
4635
      
4636
  // See if we can do this write using multiple threads.
4637
  
4638
3.32k
  uint32 tilesAcross = ifd.TilesAcross ();
4639
3.32k
  uint32 tilesDown   = ifd.TilesDown   ();
4640
                 
4641
3.32k
  bool useMultipleThreads = (tilesDown * tilesAcross >= 2) &&
4642
3.32k
                (host.PerformAreaTaskThreads () > 1) &&
4643
3.32k
                (subTileLength == ifd.fTileLength) &&
4644
3.32k
                (ifd.fCompression != ccUncompressed);
4645
  
4646
    
4647
#if qImagecore
4648
  useMultipleThreads = false; 
4649
#endif
4650
    
4651
3.32k
  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
3.32k
  else
4675
3.32k
    {
4676
                  
4677
3.32k
    AutoPtr<dng_memory_block> compressedBuffer;
4678
3.32k
    AutoPtr<dng_memory_block> uncompressedBuffer;
4679
3.32k
    AutoPtr<dng_memory_block> subTileBlockBuffer;
4680
3.32k
    AutoPtr<dng_memory_block> tempBuffer;
4681
    
4682
3.32k
    if (compressedSize)
4683
325
      {
4684
325
      compressedBuffer.Reset (host.Allocate (compressedSize));
4685
325
      }
4686
    
4687
3.32k
    if (uncompressedSize)
4688
3.32k
      {
4689
3.32k
      uncompressedBuffer.Reset (host.Allocate (uncompressedSize));
4690
3.32k
      }
4691
    
4692
3.32k
    if (ifd.fSubTileBlockRows > 1 && uncompressedSize)
4693
0
      {
4694
0
      subTileBlockBuffer.Reset (host.Allocate (uncompressedSize));
4695
0
      }
4696
        
4697
    // Write out each tile.
4698
    
4699
3.32k
    uint32 tileIndex = 0;
4700
    
4701
10.4k
    for (uint32 rowIndex = 0; rowIndex < tilesDown; rowIndex++)
4702
7.13k
      {
4703
      
4704
25.2k
      for (uint32 colIndex = 0; colIndex < tilesAcross; colIndex++)
4705
18.0k
        {
4706
        
4707
        // Remember this offset.
4708
        
4709
18.0k
        uint32 tileOffset = (uint32) stream.Position ();
4710
      
4711
18.0k
        basic.SetTileOffset (tileIndex, tileOffset);
4712
        
4713
        // Split tile into sub-tiles if possible.
4714
        
4715
18.0k
        dng_rect tileArea = ifd.TileArea (rowIndex, colIndex);
4716
        
4717
18.0k
        uint32 subTileCount = (tileArea.H () + subTileLength - 1) /
4718
18.0k
                    subTileLength;
4719
                    
4720
125k
        for (uint32 subIndex = 0; subIndex < subTileCount; subIndex++)
4721
107k
          {
4722
          
4723
107k
          host.SniffForAbort ();
4724
        
4725
107k
          dng_rect subArea (tileArea);
4726
          
4727
107k
          subArea.t = tileArea.t + subIndex * subTileLength;
4728
          
4729
107k
          subArea.b = Min_int32 (subArea.t + subTileLength,
4730
107k
                       tileArea.b);
4731
                       
4732
          // Write the sub-tile.
4733
          
4734
107k
          WriteTile (host,
4735
107k
                 ifd,
4736
107k
                 stream,
4737
107k
                 image,
4738
107k
                 subArea,
4739
107k
                 fakeChannels,
4740
107k
                 compressedBuffer,
4741
107k
                 uncompressedBuffer,
4742
107k
                 subTileBlockBuffer,
4743
107k
                 tempBuffer);
4744
                 
4745
107k
          }
4746
          
4747
        // Update tile count.
4748
          
4749
18.0k
        uint32 tileByteCount = (uint32) stream.Position () - tileOffset;
4750
          
4751
18.0k
        basic.SetTileByteCount (tileIndex, tileByteCount);
4752
        
4753
18.0k
        tileIndex++;
4754
        
4755
        // Keep the tiles on even byte offsets.
4756
                           
4757
18.0k
        if (tileByteCount & 1)
4758
12.3k
          {
4759
12.3k
          stream.Put_uint8 (0);
4760
12.3k
          }
4761
          
4762
18.0k
        }
4763
4764
7.13k
      }
4765
      
4766
3.32k
    }
4767
    
4768
3.32k
  }
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
1.45k
  {
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
1.45k
  }
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
710
  {
5268
  
5269
710
  WriteTIFF (host,
5270
710
         stream,
5271
710
         image,
5272
710
         photometricInterpretation,
5273
710
         compression,
5274
710
         negative ? &(negative->Metadata ()) : NULL,
5275
710
         space,
5276
710
         resolution,
5277
710
         thumbnail,
5278
710
         imageResources,
5279
710
         metadataSubset);
5280
  
5281
710
  }
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
2.31k
  {
5297
  
5298
2.31k
  const void *profileData = NULL;
5299
2.31k
  uint32 profileSize = 0;
5300
  
5301
2.31k
  const uint8 *data = NULL;
5302
2.31k
  uint32 size = 0;
5303
  
5304
2.31k
  if (space && space->ICCProfile (size, data))
5305
710
    {
5306
    
5307
710
    profileData = data;
5308
710
    profileSize = size;
5309
    
5310
710
    }
5311
    
5312
2.31k
  WriteTIFFWithProfile (host,
5313
2.31k
              stream,
5314
2.31k
              image,
5315
2.31k
              photometricInterpretation,
5316
2.31k
              compression,
5317
2.31k
              metadata,
5318
2.31k
              profileData,
5319
2.31k
              profileSize,
5320
2.31k
              resolution,
5321
2.31k
              thumbnail,
5322
2.31k
              imageResources,
5323
2.31k
              metadataSubset);
5324
  
5325
2.31k
  }
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
2.31k
  {
5373
  
5374
2.31k
  uint32 j;
5375
  
5376
2.31k
  AutoPtr<dng_metadata> metadata;
5377
  
5378
2.31k
  if (constMetadata)
5379
710
    {
5380
    
5381
710
    metadata.Reset (constMetadata->Clone (host.Allocator ()));
5382
    
5383
710
    CleanUpMetadata (host, 
5384
710
             *metadata,
5385
710
             metadataSubset,
5386
710
             "image/tiff");
5387
    
5388
710
    }
5389
  
5390
2.31k
  dng_ifd ifd;
5391
  
5392
2.31k
  ifd.fNewSubFileType = sfMainImage;
5393
  
5394
2.31k
  ifd.fImageWidth  = image.Bounds ().W ();
5395
2.31k
  ifd.fImageLength = image.Bounds ().H ();
5396
  
5397
2.31k
  ifd.fSamplesPerPixel = image.Planes ();
5398
  
5399
2.31k
  ifd.fBitsPerSample [0] = TagTypeSize (image.PixelType ()) * 8;
5400
  
5401
3.58k
  for (j = 1; j < ifd.fSamplesPerPixel; j++)
5402
1.26k
    {
5403
1.26k
    ifd.fBitsPerSample [j] = ifd.fBitsPerSample [0];
5404
1.26k
    }
5405
    
5406
2.31k
  ifd.fPhotometricInterpretation = photometricInterpretation;
5407
  
5408
2.31k
  ifd.fCompression = compression;
5409
  
5410
2.31k
  if (ifd.fCompression == ccUncompressed)
5411
2.31k
    {
5412
    
5413
2.31k
    ifd.SetSingleStrip ();
5414
    
5415
2.31k
    }
5416
    
5417
0
  else
5418
0
    {
5419
    
5420
0
    ifd.FindStripSize (128 * 1024);
5421
    
5422
0
    ifd.fPredictor = cpHorizontalDifference;
5423
    
5424
0
    }
5425
5426
2.31k
  uint32 extraSamples = 0;
5427
  
5428
2.31k
  switch (photometricInterpretation)
5429
2.31k
    {
5430
    
5431
1.69k
    case piBlackIsZero:
5432
1.69k
      {
5433
1.69k
      extraSamples = image.Planes () - 1;
5434
1.69k
      break;
5435
0
      }
5436
      
5437
626
    case piRGB:
5438
626
      {
5439
626
      extraSamples = image.Planes () - 3;
5440
626
      break;
5441
0
      }
5442
      
5443
0
    default:
5444
0
      break;
5445
      
5446
2.31k
    }
5447
    
5448
2.31k
  ifd.fExtraSamplesCount = extraSamples;
5449
  
5450
2.31k
  if (image.PixelType () == ttFloat)
5451
386
    {
5452
    
5453
772
    for (j = 0; j < ifd.fSamplesPerPixel; j++)
5454
386
      {
5455
386
      ifd.fSampleFormat [j] = sfFloatingPoint;
5456
386
      }
5457
      
5458
386
    }
5459
  
5460
2.31k
  dng_tiff_directory mainIFD;
5461
  
5462
2.31k
  dng_basic_tag_set basic (mainIFD, ifd);
5463
  
5464
  // Resolution.
5465
  
5466
2.31k
  dng_resolution res;
5467
  
5468
2.31k
  if (resolution)
5469
0
    {
5470
0
    res = *resolution;
5471
0
    }
5472
  
5473
2.31k
  tag_urational tagXResolution (tcXResolution, res.fXResolution);
5474
2.31k
  tag_urational tagYResolution (tcYResolution, res.fYResolution);
5475
  
5476
2.31k
  tag_uint16 tagResolutionUnit (tcResolutionUnit, res.fResolutionUnit);
5477
  
5478
2.31k
  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
2.31k
  tag_icc_profile iccProfileTag (profileData, profileSize);
5488
  
5489
2.31k
  if (iccProfileTag.Count ())
5490
710
    {
5491
710
    mainIFD.Add (&iccProfileTag);
5492
710
    }
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
2.31k
  tag_iptc tagIPTC (metadata.Get () ? metadata->IPTCData   () : NULL,
5510
2.31k
            metadata.Get () ? metadata->IPTCLength () : 0);
5511
    
5512
2.31k
  if (tagIPTC.Count ())
5513
0
    {
5514
0
    mainIFD.Add (&tagIPTC);
5515
0
    }
5516
    
5517
  // Adobe data (thumbnail and IPTC digest)
5518
  
5519
2.31k
  AutoPtr<dng_memory_block> adobeData (BuildAdobeData (host,
5520
2.31k
                             metadata.Get (),
5521
2.31k
                             thumbnail,
5522
2.31k
                             imageResources));
5523
                             
5524
2.31k
  tag_uint8_ptr tagAdobe (tcAdobeData,
5525
2.31k
              adobeData->Buffer_uint8 (),
5526
2.31k
              adobeData->LogicalSize ());
5527
                   
5528
2.31k
  if (tagAdobe.Count ())
5529
0
    {
5530
0
    mainIFD.Add (&tagAdobe);
5531
0
    }
5532
    
5533
  // Exif metadata.
5534
  
5535
2.31k
  exif_tag_set exifSet (mainIFD,
5536
2.31k
              metadata.Get () && metadata->GetExif () ? *metadata->GetExif () 
5537
2.31k
                                  : dng_exif (),
5538
2.31k
              metadata.Get () ? metadata->IsMakerNoteSafe () : false,
5539
2.31k
              metadata.Get () ? metadata->MakerNoteData   () : NULL,
5540
2.31k
              metadata.Get () ? metadata->MakerNoteLength () : 0,
5541
2.31k
              false);
5542
5543
  // Find offset to main image data.
5544
  
5545
2.31k
  uint32 offsetMainIFD = 8;
5546
  
5547
2.31k
  uint32 offsetExifData = offsetMainIFD + mainIFD.Size ();
5548
  
5549
2.31k
  exifSet.Locate (offsetExifData);
5550
  
5551
2.31k
  uint32 offsetMainData = offsetExifData + exifSet.Size ();
5552
    
5553
2.31k
  stream.SetWritePosition (offsetMainData);
5554
  
5555
  // Write the main image data.
5556
  
5557
2.31k
  WriteImage (host,
5558
2.31k
        ifd,
5559
2.31k
        basic,
5560
2.31k
        stream,
5561
2.31k
        image);
5562
        
5563
  // Trim the file to this length.
5564
  
5565
2.31k
  stream.SetLength (stream.Position ());
5566
  
5567
  // TIFF has a 4G size limit.
5568
  
5569
2.31k
  if (stream.Length () > 0x0FFFFFFFFL)
5570
0
    {
5571
0
    ThrowImageTooBigTIFF ();
5572
0
    }
5573
  
5574
  // Write TIFF Header.
5575
  
5576
2.31k
  stream.SetWritePosition (0);
5577
  
5578
2.31k
  stream.Put_uint16 (stream.BigEndian () ? byteOrderMM : byteOrderII);
5579
  
5580
2.31k
  stream.Put_uint16 (42);
5581
  
5582
2.31k
  stream.Put_uint32 (offsetMainIFD);
5583
  
5584
  // Write the IFDs.
5585
  
5586
2.31k
  mainIFD.Put (stream);
5587
  
5588
2.31k
  exifSet.Put (stream);
5589
  
5590
2.31k
  stream.Flush ();
5591
  
5592
2.31k
  }
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
741
  {
5603
  
5604
741
  WriteDNG (host,
5605
741
        stream,
5606
741
        negative,
5607
741
        negative.Metadata (),
5608
741
        previewList,
5609
741
        maxBackwardVersion,
5610
741
        uncompressed);
5611
  
5612
741
  }
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
741
  {
5624
5625
741
  uint32 j;
5626
  
5627
  // Clean up metadata per MWG recommendations.
5628
  
5629
741
  AutoPtr<dng_metadata> metadata (constMetadata.Clone (host.Allocator ()));
5630
5631
741
  CleanUpMetadata (host, 
5632
741
           *metadata,
5633
741
           kMetadataSubset_All,
5634
741
           "image/dng");
5635
           
5636
  // Figure out the compression to use.  Most of the time this is lossless
5637
  // JPEG.
5638
  
5639
741
  uint32 compression = uncompressed ? ccUncompressed : ccJPEG;
5640
    
5641
  // Was the the original file lossy JPEG compressed?
5642
  
5643
741
  const dng_jpeg_image *rawJPEGImage = negative.RawJPEGImage ();
5644
  
5645
  // If so, can we save it using the requested compression and DNG version?
5646
  
5647
741
  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
741
  else if (rawJPEGImage)
5664
240
    {
5665
    
5666
240
    compression = ccLossyJPEG;
5667
    
5668
240
    }
5669
    
5670
  // Are we saving the original size tags?
5671
  
5672
741
  bool saveOriginalDefaultFinalSize     = false;
5673
741
  bool saveOriginalBestQualityFinalSize = false;
5674
741
  bool saveOriginalDefaultCropSize      = false;
5675
  
5676
741
    {
5677
    
5678
    // See if we are saving a proxy image.
5679
    
5680
741
    dng_point defaultFinalSize (negative.DefaultFinalHeight (),
5681
741
                  negative.DefaultFinalWidth  ());
5682
                  
5683
741
    saveOriginalDefaultFinalSize = (negative.OriginalDefaultFinalSize () !=
5684
741
                    defaultFinalSize);
5685
    
5686
741
    if (saveOriginalDefaultFinalSize)
5687
76
      {
5688
      
5689
      // If the save OriginalDefaultFinalSize tag, this changes the defaults
5690
      // for the OriginalBestQualityFinalSize and OriginalDefaultCropSize tags.
5691
      
5692
76
      saveOriginalBestQualityFinalSize = (negative.OriginalBestQualityFinalSize () != 
5693
76
                        defaultFinalSize);
5694
                        
5695
76
      saveOriginalDefaultCropSize = (negative.OriginalDefaultCropSizeV () !=
5696
76
                       dng_urational (defaultFinalSize.v, 1)) ||
5697
76
                      (negative.OriginalDefaultCropSizeH () !=
5698
12
                       dng_urational (defaultFinalSize.h, 1));
5699
5700
76
      }
5701
      
5702
665
    else
5703
665
      {
5704
      
5705
      // Else these two tags default to the normal non-proxy size image values.
5706
      
5707
665
      dng_point bestQualityFinalSize (negative.BestQualityFinalHeight (),
5708
665
                      negative.BestQualityFinalWidth  ());
5709
                      
5710
665
      saveOriginalBestQualityFinalSize = (negative.OriginalBestQualityFinalSize () != 
5711
665
                        bestQualityFinalSize);
5712
                        
5713
665
      saveOriginalDefaultCropSize = (negative.OriginalDefaultCropSizeV () !=
5714
665
                       negative.DefaultCropSizeV ()) ||
5715
665
                      (negative.OriginalDefaultCropSizeH () !=
5716
665
                       negative.DefaultCropSizeH ());
5717
      
5718
665
      }
5719
    
5720
741
    }
5721
    
5722
  // Is this a floating point image that we are saving?
5723
  
5724
741
  bool isFloatingPoint = (negative.RawImage ().PixelType () == ttFloat);
5725
  
5726
  // Does this image have a transparency mask?
5727
  
5728
741
  bool hasTransparencyMask = (negative.RawTransparencyMask () != NULL);
5729
  
5730
  // Should we save a compressed 32-bit integer file?
5731
  
5732
741
  bool isCompressed32BitInteger = (negative.RawImage ().PixelType () == ttLong) &&
5733
741
                    (maxBackwardVersion >= dngVersion_1_4_0_0) &&
5734
741
                  (!uncompressed);
5735
  
5736
  // Figure out what main version to use.
5737
  
5738
741
  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
741
  if (dngVersion == dngVersion_1_4_0_0)
5743
741
    {
5744
    
5745
741
    if (!rawJPEGImage                     &&
5746
741
      !isFloatingPoint          &&
5747
741
      !hasTransparencyMask        &&
5748
741
      !isCompressed32BitInteger      &&
5749
741
      !saveOriginalDefaultFinalSize     &&
5750
741
      !saveOriginalBestQualityFinalSize &&
5751
741
      !saveOriginalDefaultCropSize      )
5752
217
      {
5753
        
5754
217
      dngVersion = dngVersion_1_3_0_0;
5755
        
5756
217
      }
5757
      
5758
741
    }
5759
  
5760
  // Figure out what backward version to use.
5761
  
5762
741
  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
741
  dngBackwardVersion = Max_uint32 (dngBackwardVersion,
5769
741
                   negative.OpcodeList1 ().MinVersion (false));
5770
5771
741
  dngBackwardVersion = Max_uint32 (dngBackwardVersion,
5772
741
                   negative.OpcodeList2 ().MinVersion (false));
5773
5774
741
  dngBackwardVersion = Max_uint32 (dngBackwardVersion,
5775
741
                   negative.OpcodeList3 ().MinVersion (false));
5776
                   
5777
741
  if (negative.GetMosaicInfo () &&
5778
741
    negative.GetMosaicInfo ()->fCFALayout >= 6)
5779
0
    {
5780
0
    dngBackwardVersion = Max_uint32 (dngBackwardVersion, dngVersion_1_3_0_0);
5781
0
    }
5782
    
5783
741
  if (rawJPEGImage || isFloatingPoint || hasTransparencyMask || isCompressed32BitInteger)
5784
524
    {
5785
524
    dngBackwardVersion = Max_uint32 (dngBackwardVersion, dngVersion_1_4_0_0);
5786
524
    }
5787
                   
5788
741
  if (dngBackwardVersion > dngVersion)
5789
0
    {
5790
0
    ThrowProgramError ();
5791
0
    }
5792
    
5793
  // Find best thumbnail from preview list, if any.
5794
5795
741
  const dng_preview *thumbnail = NULL;
5796
  
5797
741
  if (previewList)
5798
741
    {
5799
    
5800
741
    uint32 thumbArea = 0;
5801
    
5802
1.72k
    for (j = 0; j < previewList->Count (); j++)
5803
988
      {
5804
      
5805
988
      const dng_image_preview *imagePreview = dynamic_cast<const dng_image_preview *>(&previewList->Preview (j));
5806
      
5807
988
      if (imagePreview)
5808
501
        {
5809
        
5810
501
        uint32 thisArea = imagePreview->fImage->Bounds ().W () *
5811
501
                  imagePreview->fImage->Bounds ().H ();
5812
                  
5813
501
        if (!thumbnail || thisArea < thumbArea)
5814
501
          {
5815
          
5816
501
          thumbnail = &previewList->Preview (j);
5817
          
5818
501
          thumbArea = thisArea;
5819
          
5820
501
          }
5821
      
5822
501
        }
5823
        
5824
988
      const dng_jpeg_preview *jpegPreview = dynamic_cast<const dng_jpeg_preview *>(&previewList->Preview (j));
5825
      
5826
988
      if (jpegPreview)
5827
487
        {
5828
        
5829
487
        uint32 thisArea = jpegPreview->fPreviewSize.h *
5830
487
                  jpegPreview->fPreviewSize.v;
5831
                  
5832
487
        if (!thumbnail || thisArea < thumbArea)
5833
240
          {
5834
          
5835
240
          thumbnail = &previewList->Preview (j);
5836
          
5837
240
          thumbArea = thisArea;
5838
          
5839
240
          }
5840
        
5841
487
        }
5842
        
5843
988
      }
5844
    
5845
741
    }
5846
    
5847
  // Create the main IFD
5848
                     
5849
741
  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
741
  AutoPtr<dng_tiff_directory> rawIFD_IfNotMain;
5855
  
5856
741
  if (thumbnail)
5857
741
    {
5858
741
    rawIFD_IfNotMain.Reset (new dng_tiff_directory);
5859
741
    }
5860
5861
741
  dng_tiff_directory &rawIFD (thumbnail ? *rawIFD_IfNotMain : mainIFD);
5862
  
5863
  // Include DNG version tags.
5864
  
5865
741
  uint8 dngVersionData [4];
5866
  
5867
741
  dngVersionData [0] = (uint8) (dngVersion >> 24);
5868
741
  dngVersionData [1] = (uint8) (dngVersion >> 16);
5869
741
  dngVersionData [2] = (uint8) (dngVersion >>  8);
5870
741
  dngVersionData [3] = (uint8) (dngVersion      );
5871
  
5872
741
  tag_uint8_ptr tagDNGVersion (tcDNGVersion, dngVersionData, 4);
5873
  
5874
741
  mainIFD.Add (&tagDNGVersion);
5875
  
5876
741
  uint8 dngBackwardVersionData [4];
5877
5878
741
  dngBackwardVersionData [0] = (uint8) (dngBackwardVersion >> 24);
5879
741
  dngBackwardVersionData [1] = (uint8) (dngBackwardVersion >> 16);
5880
741
  dngBackwardVersionData [2] = (uint8) (dngBackwardVersion >>  8);
5881
741
  dngBackwardVersionData [3] = (uint8) (dngBackwardVersion      );
5882
  
5883
741
  tag_uint8_ptr tagDNGBackwardVersion (tcDNGBackwardVersion, dngBackwardVersionData, 4);
5884
  
5885
741
  mainIFD.Add (&tagDNGBackwardVersion);
5886
  
5887
  // The main IFD contains the thumbnail, if there is a thumbnail.
5888
                
5889
741
  AutoPtr<dng_basic_tag_set> thmBasic;
5890
  
5891
741
  if (thumbnail)
5892
741
    {
5893
741
    thmBasic.Reset (thumbnail->AddTagSet (mainIFD));
5894
741
    }
5895
              
5896
  // Get the raw image we are writing.
5897
  
5898
741
  const dng_image &rawImage (negative.RawImage ());
5899
  
5900
  // For floating point, we only support ZIP compression.
5901
  
5902
741
  if (isFloatingPoint && !uncompressed)
5903
244
    {
5904
    
5905
244
    compression = ccDeflate;
5906
    
5907
244
    }
5908
  
5909
  // For 32-bit integer images, we only support ZIP and uncompressed.
5910
  
5911
741
  if (rawImage.PixelType () == ttLong)
5912
40
    {
5913
    
5914
40
    if (isCompressed32BitInteger)
5915
40
      {
5916
40
      compression = ccDeflate;
5917
40
      }
5918
      
5919
0
    else
5920
0
      {
5921
0
      compression = ccUncompressed;
5922
0
      }
5923
  
5924
40
    }
5925
  
5926
  // Get a copy of the mosaic info.
5927
  
5928
741
  dng_mosaic_info mosaicInfo;
5929
  
5930
741
  if (negative.GetMosaicInfo ())
5931
3
    {
5932
3
    mosaicInfo = *(negative.GetMosaicInfo ());
5933
3
    }
5934
    
5935
  // Create a dng_ifd record for the raw image.
5936
  
5937
741
  dng_ifd info;
5938
  
5939
741
  info.fImageWidth  = rawImage.Width  ();
5940
741
  info.fImageLength = rawImage.Height ();
5941
  
5942
741
  info.fSamplesPerPixel = rawImage.Planes ();
5943
  
5944
741
  info.fPhotometricInterpretation = mosaicInfo.IsColorFilterArray () ? piCFA
5945
741
                                     : piLinearRaw;
5946
      
5947
741
  info.fCompression = compression;
5948
  
5949
741
  if (isFloatingPoint && compression == ccDeflate)
5950
244
    {
5951
    
5952
244
    info.fPredictor = cpFloatingPoint;
5953
    
5954
244
    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
244
    } 
5970
    
5971
741
  if (isCompressed32BitInteger)
5972
40
    {
5973
    
5974
40
    info.fPredictor = cpHorizontalDifference;
5975
    
5976
40
    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
40
    }
5992
  
5993
741
  uint32 rawPixelType = rawImage.PixelType ();
5994
  
5995
741
  if (rawPixelType == ttShort)
5996
177
    {
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
177
    const dng_linearization_info *rangeInfo = negative.GetLinearizationInfo ();
6002
  
6003
177
    if (rangeInfo)
6004
177
      {
6005
6006
177
      if (rangeInfo->fLinearizationTable.Get ())
6007
1
        {
6008
        
6009
1
        uint32 entries = rangeInfo->fLinearizationTable->LogicalSize () >> 1;
6010
        
6011
1
        if (entries <= 256)
6012
1
          {
6013
          
6014
1
          rawPixelType = ttByte;
6015
          
6016
1
          }
6017
                        
6018
1
        }
6019
        
6020
177
      }
6021
6022
177
    }
6023
  
6024
741
  switch (rawPixelType)
6025
741
    {
6026
    
6027
281
    case ttByte:
6028
281
      {
6029
281
      info.fBitsPerSample [0] = 8;
6030
281
      break;
6031
0
      }
6032
    
6033
176
    case ttShort:
6034
176
      {
6035
176
      info.fBitsPerSample [0] = 16;
6036
176
      break;
6037
0
      }
6038
      
6039
40
    case ttLong:
6040
40
      {
6041
40
      info.fBitsPerSample [0] = 32;
6042
40
      break;
6043
0
      }
6044
      
6045
244
    case ttFloat:
6046
244
      {
6047
      
6048
244
      if (negative.RawFloatBitDepth () == 16)
6049
237
        {
6050
237
        info.fBitsPerSample [0] = 16;
6051
237
        }
6052
        
6053
7
      else if (negative.RawFloatBitDepth () == 24)
6054
7
        {
6055
7
        info.fBitsPerSample [0] = 24;
6056
7
        }
6057
        
6058
0
      else
6059
0
        {
6060
0
        info.fBitsPerSample [0] = 32;
6061
0
        }
6062
6063
488
      for (j = 0; j < info.fSamplesPerPixel; j++)
6064
244
        {
6065
244
        info.fSampleFormat [j] = sfFloatingPoint;
6066
244
        }
6067
6068
244
      break;
6069
      
6070
0
      }
6071
      
6072
0
    default:
6073
0
      {
6074
0
      ThrowProgramError ();
6075
0
      }
6076
      
6077
741
    }
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
741
  uint32 fakeChannels = 1;
6084
  
6085
741
  if (info.fCompression == ccJPEG)
6086
217
    {
6087
    
6088
217
    if (mosaicInfo.IsColorFilterArray ())
6089
3
      {
6090
      
6091
3
      if (mosaicInfo.fCFAPatternSize.h == 4)
6092
0
        {
6093
0
        fakeChannels = 4;
6094
0
        }
6095
        
6096
3
      else if (mosaicInfo.fCFAPatternSize.h == 2)
6097
3
        {
6098
3
        fakeChannels = 2;
6099
3
        }
6100
      
6101
      // However, lossless JEPG is limited to four channels,
6102
      // so compromise might be required.
6103
      
6104
3
      while (fakeChannels * info.fSamplesPerPixel > 4 &&
6105
3
           fakeChannels > 1)
6106
0
        {
6107
        
6108
0
        fakeChannels >>= 1;
6109
        
6110
0
        }
6111
      
6112
3
      }
6113
  
6114
217
    }
6115
    
6116
  // Figure out tile sizes.
6117
  
6118
741
  if (rawJPEGImage)
6119
240
    {
6120
    
6121
240
    DNG_ASSERT (rawPixelType == ttByte,
6122
240
          "Unexpected jpeg pixel type");
6123
    
6124
240
    DNG_ASSERT (info.fImageWidth  == (uint32) rawJPEGImage->fImageSize.h &&
6125
240
          info.fImageLength == (uint32) rawJPEGImage->fImageSize.v,
6126
240
          "Unexpected jpeg image size");
6127
          
6128
240
    info.fTileWidth  = rawJPEGImage->fTileSize.h;
6129
240
    info.fTileLength = rawJPEGImage->fTileSize.v;
6130
6131
240
    info.fUsesStrips = rawJPEGImage->fUsesStrips;
6132
    
6133
240
    info.fUsesTiles = !info.fUsesStrips;
6134
    
6135
240
    }
6136
  
6137
501
  else if (info.fCompression == ccJPEG)
6138
217
    {
6139
    
6140
217
    info.FindTileSize (128 * 1024);
6141
    
6142
217
    }
6143
    
6144
284
  else if (info.fCompression == ccDeflate)
6145
284
    {
6146
    
6147
284
    info.FindTileSize (512 * 1024);
6148
    
6149
284
    }
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
741
  dng_basic_tag_set rawBasic (rawIFD, info);
6186
  
6187
  // JPEG tables, if any.
6188
  
6189
741
  tag_data_ptr tagJPEGTables (tcJPEGTables,
6190
741
                ttUndefined,
6191
741
                0,
6192
741
                NULL);
6193
                
6194
741
  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
741
  dng_urational defaultScaleData [2];
6208
  
6209
741
  defaultScaleData [0] = negative.DefaultScaleH ();
6210
741
  defaultScaleData [1] = negative.DefaultScaleV ();
6211
                        
6212
741
  tag_urational_ptr tagDefaultScale (tcDefaultScale,
6213
741
                       defaultScaleData,
6214
741
                       2);
6215
6216
741
  rawIFD.Add (&tagDefaultScale);
6217
  
6218
  // Best quality scale tag.
6219
  
6220
741
  tag_urational tagBestQualityScale (tcBestQualityScale,
6221
741
                     negative.BestQualityScale ());
6222
                    
6223
741
  rawIFD.Add (&tagBestQualityScale);
6224
  
6225
  // DefaultCropOrigin tag.
6226
6227
741
  dng_urational defaultCropOriginData [2];
6228
  
6229
741
  defaultCropOriginData [0] = negative.DefaultCropOriginH ();
6230
741
  defaultCropOriginData [1] = negative.DefaultCropOriginV ();
6231
                        
6232
741
  tag_urational_ptr tagDefaultCropOrigin (tcDefaultCropOrigin,
6233
741
                            defaultCropOriginData,
6234
741
                            2);
6235
6236
741
  rawIFD.Add (&tagDefaultCropOrigin);
6237
  
6238
  // DefaultCropSize tag.
6239
6240
741
  dng_urational defaultCropSizeData [2];
6241
  
6242
741
  defaultCropSizeData [0] = negative.DefaultCropSizeH ();
6243
741
  defaultCropSizeData [1] = negative.DefaultCropSizeV ();
6244
                        
6245
741
  tag_urational_ptr tagDefaultCropSize (tcDefaultCropSize,
6246
741
                          defaultCropSizeData,
6247
741
                          2);
6248
6249
741
  rawIFD.Add (&tagDefaultCropSize);
6250
  
6251
  // DefaultUserCrop tag.
6252
6253
741
  dng_urational defaultUserCropData [4];
6254
  
6255
741
  defaultUserCropData [0] = negative.DefaultUserCropT ();
6256
741
  defaultUserCropData [1] = negative.DefaultUserCropL ();
6257
741
  defaultUserCropData [2] = negative.DefaultUserCropB ();
6258
741
  defaultUserCropData [3] = negative.DefaultUserCropR ();
6259
                        
6260
741
  tag_urational_ptr tagDefaultUserCrop (tcDefaultUserCrop,
6261
741
                      defaultUserCropData,
6262
741
                      4);
6263
6264
741
  rawIFD.Add (&tagDefaultUserCrop);
6265
  
6266
  // Range mapping tag set.
6267
  
6268
741
  range_tag_set rangeSet (rawIFD, negative);
6269
              
6270
  // Mosaic pattern information.
6271
  
6272
741
  mosaic_tag_set mosaicSet (rawIFD, mosaicInfo);
6273
      
6274
  // Chroma blur radius.
6275
  
6276
741
  tag_urational tagChromaBlurRadius (tcChromaBlurRadius,
6277
741
                     negative.ChromaBlurRadius ());
6278
                    
6279
741
  if (negative.ChromaBlurRadius ().IsValid ())
6280
7
    {
6281
    
6282
7
    rawIFD.Add (&tagChromaBlurRadius);
6283
    
6284
7
    }
6285
  
6286
  // Anti-alias filter strength.
6287
  
6288
741
  tag_urational tagAntiAliasStrength (tcAntiAliasStrength,
6289
741
                      negative.AntiAliasStrength ());
6290
                    
6291
741
  if (negative.AntiAliasStrength ().IsValid ())
6292
741
    {
6293
    
6294
741
    rawIFD.Add (&tagAntiAliasStrength);
6295
    
6296
741
    }
6297
    
6298
  // Profile and other color related tags.
6299
  
6300
741
  AutoPtr<profile_tag_set> profileSet;
6301
  
6302
741
  AutoPtr<color_tag_set> colorSet;
6303
  
6304
741
  dng_std_vector<uint32> extraProfileIndex;
6305
  
6306
741
  if (!negative.IsMonochrome ())
6307
5
    {
6308
    
6309
5
    const dng_camera_profile &mainProfile (*negative.ComputeCameraProfileToEmbed (constMetadata));
6310
    
6311
5
    profileSet.Reset (new profile_tag_set (mainIFD,
6312
5
                         mainProfile));
6313
    
6314
5
    colorSet.Reset (new color_tag_set (mainIFD,
6315
5
                       negative));
6316
                       
6317
    // Build list of profile indices to include in extra profiles tag.
6318
                       
6319
5
    uint32 profileCount = negative.ProfileCount ();
6320
    
6321
10
    for (uint32 index = 0; index < profileCount; index++)
6322
5
      {
6323
      
6324
5
      const dng_camera_profile &profile (negative.ProfileByIndex (index));
6325
      
6326
5
      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
5
      }
6339
                       
6340
5
    }
6341
    
6342
  // Extra camera profiles tag.
6343
  
6344
741
  uint32 extraProfileCount = (uint32) extraProfileIndex.size ();
6345
  
6346
741
  dng_memory_data extraProfileOffsets (extraProfileCount, sizeof (uint32));
6347
  
6348
741
  tag_uint32_ptr extraProfileTag (tcExtraCameraProfiles,
6349
741
                  extraProfileOffsets.Buffer_uint32 (),
6350
741
                  extraProfileCount);
6351
                  
6352
741
  if (extraProfileCount)
6353
0
    {
6354
    
6355
0
    mainIFD.Add (&extraProfileTag);
6356
6357
0
    }
6358
      
6359
  // Other tags.
6360
  
6361
741
  tag_uint16 tagOrientation (tcOrientation,
6362
741
                   (uint16) negative.ComputeOrientation (constMetadata).GetTIFF ());
6363
                 
6364
741
  mainIFD.Add (&tagOrientation);
6365
6366
741
  tag_srational tagBaselineExposure (tcBaselineExposure,
6367
741
                       negative.BaselineExposureR ());
6368
                      
6369
741
  mainIFD.Add (&tagBaselineExposure);
6370
6371
741
  tag_urational tagBaselineNoise (tcBaselineNoise,
6372
741
                      negative.BaselineNoiseR ());
6373
                      
6374
741
  mainIFD.Add (&tagBaselineNoise);
6375
  
6376
741
  tag_urational tagNoiseReductionApplied (tcNoiseReductionApplied,
6377
741
                      negative.NoiseReductionApplied ());
6378
                      
6379
741
  if (negative.NoiseReductionApplied ().IsValid ())
6380
0
    {
6381
    
6382
0
    mainIFD.Add (&tagNoiseReductionApplied);
6383
  
6384
0
    }
6385
6386
741
  tag_dng_noise_profile tagNoiseProfile (negative.NoiseProfile ());
6387
    
6388
741
  if (negative.NoiseProfile ().IsValidForNegative (negative))
6389
0
    {
6390
6391
0
    mainIFD.Add (&tagNoiseProfile);
6392
    
6393
0
    }
6394
6395
741
  tag_urational tagBaselineSharpness (tcBaselineSharpness,
6396
741
                        negative.BaselineSharpnessR ());
6397
                      
6398
741
  mainIFD.Add (&tagBaselineSharpness);
6399
6400
741
  tag_string tagUniqueName (tcUniqueCameraModel,
6401
741
                  negative.ModelName (),
6402
741
                  true);
6403
                
6404
741
  mainIFD.Add (&tagUniqueName);
6405
  
6406
741
  tag_string tagLocalName (tcLocalizedCameraModel,
6407
741
                 negative.LocalName (),
6408
741
                 false);
6409
               
6410
741
  if (negative.LocalName ().NotEmpty ())
6411
1
    {
6412
    
6413
1
    mainIFD.Add (&tagLocalName);
6414
  
6415
1
    }
6416
  
6417
741
  tag_urational tagShadowScale (tcShadowScale,
6418
741
                    negative.ShadowScaleR ());
6419
                      
6420
741
  mainIFD.Add (&tagShadowScale);
6421
  
6422
741
  tag_uint16 tagColorimetricReference (tcColorimetricReference,
6423
741
                     (uint16) negative.ColorimetricReference ());
6424
                     
6425
741
  if (negative.ColorimetricReference () != crSceneReferred)
6426
0
    {
6427
    
6428
0
    mainIFD.Add (&tagColorimetricReference);
6429
    
6430
0
    }
6431
    
6432
741
  bool useNewDigest = (maxBackwardVersion >= dngVersion_1_4_0_0);
6433
    
6434
741
  if (compression == ccLossyJPEG)
6435
240
    {
6436
    
6437
240
    negative.FindRawJPEGImageDigest (host);
6438
    
6439
240
    }
6440
    
6441
501
  else
6442
501
    {
6443
    
6444
501
    if (useNewDigest)
6445
501
      {
6446
501
      negative.FindNewRawImageDigest (host);
6447
501
      }
6448
0
    else
6449
0
      {
6450
0
      negative.FindRawImageDigest (host);
6451
0
      }
6452
    
6453
501
    }
6454
  
6455
741
  tag_uint8_ptr tagRawImageDigest (useNewDigest ? tcNewRawImageDigest : tcRawImageDigest,
6456
741
                   compression == ccLossyJPEG ?
6457
240
                   negative.RawJPEGImageDigest ().data :
6458
741
                   (useNewDigest ? negative.NewRawImageDigest ().data
6459
501
                           : negative.RawImageDigest    ().data),
6460
741
                     16);
6461
                      
6462
741
  mainIFD.Add (&tagRawImageDigest);
6463
  
6464
741
  negative.FindRawDataUniqueID (host);
6465
  
6466
741
  tag_uint8_ptr tagRawDataUniqueID (tcRawDataUniqueID,
6467
741
                      negative.RawDataUniqueID ().data,
6468
741
                      16);
6469
                      
6470
741
  if (negative.RawDataUniqueID ().IsValid ())
6471
741
    {
6472
                 
6473
741
    mainIFD.Add (&tagRawDataUniqueID);
6474
    
6475
741
    }
6476
  
6477
741
  tag_string tagOriginalRawFileName (tcOriginalRawFileName,
6478
741
                       negative.OriginalRawFileName (),
6479
741
                       false);
6480
               
6481
741
  if (negative.HasOriginalRawFileName ())
6482
0
    {
6483
    
6484
0
    mainIFD.Add (&tagOriginalRawFileName);
6485
  
6486
0
    }
6487
    
6488
741
  negative.FindOriginalRawFileDigest ();
6489
    
6490
741
  tag_data_ptr tagOriginalRawFileData (tcOriginalRawFileData,
6491
741
                     ttUndefined,
6492
741
                     negative.OriginalRawFileDataLength (),
6493
741
                     negative.OriginalRawFileData       ());
6494
                     
6495
741
  tag_uint8_ptr tagOriginalRawFileDigest (tcOriginalRawFileDigest,
6496
741
                      negative.OriginalRawFileDigest ().data,
6497
741
                      16);
6498
                     
6499
741
  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
741
  exif_tag_set exifSet (mainIFD,
6526
741
              *metadata->GetExif (),
6527
741
              metadata->IsMakerNoteSafe (),
6528
741
              metadata->MakerNoteData   (),
6529
741
              metadata->MakerNoteLength (),
6530
741
              true);
6531
            
6532
  // Private data.
6533
  
6534
741
  tag_uint8_ptr tagPrivateData (tcDNGPrivateData,
6535
741
                    negative.PrivateData (),
6536
741
                    negative.PrivateLength ());
6537
               
6538
741
  if (negative.PrivateLength ())
6539
1
    {
6540
    
6541
1
    mainIFD.Add (&tagPrivateData);
6542
    
6543
1
    }
6544
    
6545
  // Proxy size tags.
6546
  
6547
741
  uint32 originalDefaultFinalSizeData [2];
6548
  
6549
741
  originalDefaultFinalSizeData [0] = negative.OriginalDefaultFinalSize ().h;
6550
741
  originalDefaultFinalSizeData [1] = negative.OriginalDefaultFinalSize ().v;
6551
  
6552
741
  tag_uint32_ptr tagOriginalDefaultFinalSize (tcOriginalDefaultFinalSize,
6553
741
                        originalDefaultFinalSizeData,
6554
741
                        2);
6555
  
6556
741
  if (saveOriginalDefaultFinalSize)
6557
76
    {
6558
    
6559
76
    mainIFD.Add (&tagOriginalDefaultFinalSize);
6560
    
6561
76
    }
6562
    
6563
741
  uint32 originalBestQualityFinalSizeData [2];
6564
  
6565
741
  originalBestQualityFinalSizeData [0] = negative.OriginalBestQualityFinalSize ().h;
6566
741
  originalBestQualityFinalSizeData [1] = negative.OriginalBestQualityFinalSize ().v;
6567
  
6568
741
  tag_uint32_ptr tagOriginalBestQualityFinalSize (tcOriginalBestQualityFinalSize,
6569
741
                          originalBestQualityFinalSizeData,
6570
741
                          2);
6571
  
6572
741
  if (saveOriginalBestQualityFinalSize)
6573
76
    {
6574
    
6575
76
    mainIFD.Add (&tagOriginalBestQualityFinalSize);
6576
    
6577
76
    }
6578
    
6579
741
  dng_urational originalDefaultCropSizeData [2];
6580
  
6581
741
  originalDefaultCropSizeData [0] = negative.OriginalDefaultCropSizeH ();
6582
741
  originalDefaultCropSizeData [1] = negative.OriginalDefaultCropSizeV ();
6583
  
6584
741
  tag_urational_ptr tagOriginalDefaultCropSize (tcOriginalDefaultCropSize,
6585
741
                          originalDefaultCropSizeData,
6586
741
                          2);
6587
  
6588
741
  if (saveOriginalDefaultCropSize)
6589
77
    {
6590
    
6591
77
    mainIFD.Add (&tagOriginalDefaultCropSize);
6592
    
6593
77
    }
6594
    
6595
  // Opcode list 1.
6596
  
6597
741
  AutoPtr<dng_memory_block> opcodeList1Data (negative.OpcodeList1 ().Spool (host));
6598
  
6599
741
  tag_data_ptr tagOpcodeList1 (tcOpcodeList1,
6600
741
                 ttUndefined,
6601
741
                 opcodeList1Data.Get () ? opcodeList1Data->LogicalSize () : 0,
6602
741
                 opcodeList1Data.Get () ? opcodeList1Data->Buffer      () : NULL);
6603
                 
6604
741
  if (opcodeList1Data.Get ())
6605
0
    {
6606
    
6607
0
    rawIFD.Add (&tagOpcodeList1);
6608
    
6609
0
    }
6610
    
6611
  // Opcode list 2.
6612
  
6613
741
  AutoPtr<dng_memory_block> opcodeList2Data (negative.OpcodeList2 ().Spool (host));
6614
  
6615
741
  tag_data_ptr tagOpcodeList2 (tcOpcodeList2,
6616
741
                 ttUndefined,
6617
741
                 opcodeList2Data.Get () ? opcodeList2Data->LogicalSize () : 0,
6618
741
                 opcodeList2Data.Get () ? opcodeList2Data->Buffer      () : NULL);
6619
                 
6620
741
  if (opcodeList2Data.Get ())
6621
240
    {
6622
    
6623
240
    rawIFD.Add (&tagOpcodeList2);
6624
    
6625
240
    }
6626
    
6627
  // Opcode list 3.
6628
  
6629
741
  AutoPtr<dng_memory_block> opcodeList3Data (negative.OpcodeList3 ().Spool (host));
6630
  
6631
741
  tag_data_ptr tagOpcodeList3 (tcOpcodeList3,
6632
741
                 ttUndefined,
6633
741
                 opcodeList3Data.Get () ? opcodeList3Data->LogicalSize () : 0,
6634
741
                 opcodeList3Data.Get () ? opcodeList3Data->Buffer      () : NULL);
6635
                 
6636
741
  if (opcodeList3Data.Get ())
6637
0
    {
6638
    
6639
0
    rawIFD.Add (&tagOpcodeList3);
6640
    
6641
0
    }
6642
    
6643
  // Transparency mask, if any.
6644
  
6645
741
  AutoPtr<dng_ifd> maskInfo;
6646
  
6647
741
  AutoPtr<dng_tiff_directory> maskIFD;
6648
  
6649
741
  AutoPtr<dng_basic_tag_set> maskBasic;
6650
  
6651
741
  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
741
  uint32 subFileCount = thumbnail ? 1 : 0;
6706
  
6707
741
  if (hasTransparencyMask)
6708
0
    {
6709
0
    subFileCount++;
6710
0
    }
6711
  
6712
  // Add previews.
6713
  
6714
741
  uint32 previewCount = previewList ? previewList->Count () : 0;
6715
  
6716
741
  AutoPtr<dng_tiff_directory> previewIFD [kMaxDNGPreviews];
6717
  
6718
741
  AutoPtr<dng_basic_tag_set> previewBasic [kMaxDNGPreviews];
6719
  
6720
1.72k
  for (j = 0; j < previewCount; j++)
6721
988
    {
6722
    
6723
988
    if (thumbnail != &previewList->Preview (j))
6724
247
      {
6725
    
6726
247
      previewIFD [j] . Reset (new dng_tiff_directory);
6727
      
6728
247
      previewBasic [j] . Reset (previewList->Preview (j).AddTagSet (*previewIFD [j]));
6729
        
6730
247
      subFileCount++;
6731
      
6732
247
      }
6733
    
6734
988
    }
6735
    
6736
  // And a link to the raw and JPEG image IFDs.
6737
  
6738
741
  uint32 subFileData [kMaxDNGPreviews + 2];
6739
  
6740
741
  tag_uint32_ptr tagSubFile (tcSubIFDs,
6741
741
                 subFileData,
6742
741
                 subFileCount);
6743
                 
6744
741
  if (subFileCount)
6745
741
    {
6746
  
6747
741
    mainIFD.Add (&tagSubFile);
6748
    
6749
741
    }
6750
  
6751
  // Skip past the header and IFDs for now.
6752
  
6753
741
  uint32 currentOffset = 8;
6754
  
6755
741
  currentOffset += mainIFD.Size ();
6756
  
6757
741
  uint32 subFileIndex = 0;
6758
  
6759
741
  if (thumbnail)
6760
741
    {
6761
  
6762
741
    subFileData [subFileIndex++] = currentOffset;
6763
  
6764
741
    currentOffset += rawIFD.Size ();
6765
    
6766
741
    }
6767
    
6768
741
  if (hasTransparencyMask)
6769
0
    {
6770
    
6771
0
    subFileData [subFileIndex++] = currentOffset;
6772
    
6773
0
    currentOffset += maskIFD->Size ();
6774
    
6775
0
    }
6776
  
6777
1.72k
  for (j = 0; j < previewCount; j++)
6778
988
    {
6779
    
6780
988
    if (thumbnail != &previewList->Preview (j))
6781
247
      {
6782
  
6783
247
      subFileData [subFileIndex++] = currentOffset;
6784
6785
247
      currentOffset += previewIFD [j]->Size ();
6786
      
6787
247
      }
6788
    
6789
988
    }
6790
    
6791
741
  exifSet.Locate (currentOffset);
6792
  
6793
741
  currentOffset += exifSet.Size ();
6794
  
6795
741
  stream.SetWritePosition (currentOffset);
6796
  
6797
  // Write the extra profiles.
6798
  
6799
741
  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
741
  if (thumbnail)
6822
741
    {
6823
  
6824
741
    thumbnail->WriteData (host,
6825
741
                *this,
6826
741
                *thmBasic,
6827
741
                stream);
6828
               
6829
741
    }
6830
  
6831
  // Write the preview data.
6832
  
6833
1.72k
  for (j = 0; j < previewCount; j++)
6834
988
    {
6835
    
6836
988
    if (thumbnail != &previewList->Preview (j))
6837
247
      {
6838
  
6839
247
      previewList->Preview (j).WriteData (host,
6840
247
                        *this,
6841
247
                        *previewBasic [j],
6842
247
                        stream);
6843
                        
6844
247
      }
6845
      
6846
988
    }
6847
    
6848
  // Write the raw data.
6849
  
6850
741
  if (rawJPEGImage)
6851
240
    {
6852
    
6853
240
    uint32 tileCount = info.TilesAcross () *
6854
240
               info.TilesDown   ();
6855
              
6856
5.13k
    for (uint32 tileIndex = 0; tileIndex < tileCount; tileIndex++)
6857
4.89k
      {
6858
      
6859
      // Remember this offset.
6860
      
6861
4.89k
      uint32 tileOffset = (uint32) stream.Position ();
6862
    
6863
4.89k
      rawBasic.SetTileOffset (tileIndex, tileOffset);
6864
      
6865
      // Write JPEG data.
6866
      
6867
4.89k
      stream.Put (rawJPEGImage->fJPEGData [tileIndex]->Buffer      (),
6868
4.89k
            rawJPEGImage->fJPEGData [tileIndex]->LogicalSize ());
6869
              
6870
      // Update tile count.
6871
        
6872
4.89k
      uint32 tileByteCount = (uint32) stream.Position () - tileOffset;
6873
        
6874
4.89k
      rawBasic.SetTileByteCount (tileIndex, tileByteCount);
6875
      
6876
      // Keep the tiles on even byte offsets.
6877
                         
6878
4.89k
      if (tileByteCount & 1)
6879
1.31k
        {
6880
1.31k
        stream.Put_uint8 (0);
6881
1.31k
        }
6882
6883
4.89k
      }
6884
    
6885
240
    }
6886
    
6887
501
  else
6888
501
    {
6889
    
6890
    #if qDNGValidate
6891
    dng_timer timer ("Write raw image time");
6892
    #endif
6893
  
6894
501
    WriteImage (host,
6895
501
          info,
6896
501
          rawBasic,
6897
501
          stream,
6898
501
          rawImage,
6899
501
          fakeChannels);
6900
          
6901
501
    }
6902
    
6903
  // Write transparency mask image.
6904
  
6905
741
  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
741
  stream.SetLength (stream.Position ());
6923
  
6924
  // DNG has a 4G size limit.
6925
  
6926
741
  if (stream.Length () > 0x0FFFFFFFFL)
6927
0
    {
6928
0
    ThrowImageTooBigDNG ();
6929
0
    }
6930
  
6931
  // Write TIFF Header.
6932
  
6933
741
  stream.SetWritePosition (0);
6934
  
6935
741
  stream.Put_uint16 (stream.BigEndian () ? byteOrderMM : byteOrderII);
6936
  
6937
741
  stream.Put_uint16 (42);
6938
  
6939
741
  stream.Put_uint32 (8);
6940
  
6941
  // Write the IFDs.
6942
  
6943
741
  mainIFD.Put (stream);
6944
  
6945
741
  if (thumbnail)
6946
741
    {
6947
  
6948
741
    rawIFD.Put (stream);
6949
    
6950
741
    }
6951
  
6952
741
  if (hasTransparencyMask)
6953
0
    {
6954
    
6955
0
    maskIFD->Put (stream);
6956
    
6957
0
    }
6958
    
6959
1.72k
  for (j = 0; j < previewCount; j++)
6960
988
    {
6961
    
6962
988
    if (thumbnail != &previewList->Preview (j))
6963
247
      {
6964
  
6965
247
      previewIFD [j]->Put (stream);
6966
      
6967
247
      }
6968
    
6969
988
    }
6970
    
6971
741
  exifSet.Put (stream);
6972
  
6973
741
  stream.Flush ();
6974
  
6975
741
  }
6976
                 
6977
/*****************************************************************************/