Coverage Report

Created: 2025-10-10 07:04

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