Coverage Report

Created: 2026-02-14 06:39

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/dng_sdk/source/dng_shared.cpp
Line
Count
Source
1
/*****************************************************************************/
2
// Copyright 2006-2008 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_shared.cpp#2 $ */ 
10
/* $DateTime: 2012/06/14 20:24:41 $ */
11
/* $Change: 835078 $ */
12
/* $Author: tknoll $ */
13
14
/*****************************************************************************/
15
16
#include "dng_shared.h"
17
18
#include "dng_camera_profile.h"
19
#include "dng_exceptions.h"
20
#include "dng_globals.h"
21
#include "dng_memory.h"
22
#include "dng_parse_utils.h"
23
#include "dng_safe_arithmetic.h"
24
#include "dng_tag_codes.h"
25
#include "dng_tag_types.h"
26
#include "dng_tag_values.h"
27
#include "dng_utils.h"
28
                   
29
/*****************************************************************************/
30
31
dng_camera_profile_info::dng_camera_profile_info ()
32
33
6.92M
  : fBigEndian (false)
34
  
35
6.92M
  , fColorPlanes (0)
36
  
37
6.92M
  , fCalibrationIlluminant1 (lsUnknown)
38
6.92M
  , fCalibrationIlluminant2 (lsUnknown)
39
40
6.92M
  , fColorMatrix1 ()
41
6.92M
  , fColorMatrix2 ()
42
    
43
6.92M
  , fForwardMatrix1 ()
44
6.92M
  , fForwardMatrix2 ()
45
    
46
6.92M
  , fReductionMatrix1 ()
47
6.92M
  , fReductionMatrix2 ()
48
    
49
6.92M
  , fProfileCalibrationSignature ()
50
  
51
6.92M
  , fProfileName ()
52
  
53
6.92M
  , fProfileCopyright ()
54
55
6.92M
  , fEmbedPolicy (pepAllowCopying)
56
  
57
6.92M
  , fProfileHues (0)
58
6.92M
  , fProfileSats (0)
59
6.92M
  , fProfileVals (0)
60
61
6.92M
  , fHueSatDeltas1Offset (0)
62
6.92M
  , fHueSatDeltas1Count  (0)
63
64
6.92M
  , fHueSatDeltas2Offset (0)
65
6.92M
  , fHueSatDeltas2Count  (0)
66
67
6.92M
  , fHueSatMapEncoding (encoding_Linear)
68
  
69
6.92M
  , fLookTableHues (0)
70
6.92M
  , fLookTableSats (0)
71
6.92M
  , fLookTableVals (0)
72
    
73
6.92M
  , fLookTableOffset (0)
74
6.92M
  , fLookTableCount  (0)
75
76
6.92M
  , fLookTableEncoding (encoding_Linear)
77
78
6.92M
  , fBaselineExposureOffset (0, 100)
79
  
80
6.92M
  , fDefaultBlackRender (defaultBlackRender_Auto)
81
82
6.92M
  , fToneCurveOffset     (0)
83
6.92M
  , fToneCurveCount      (0)
84
  
85
6.92M
  , fUniqueCameraModel ()
86
87
6.92M
  {
88
  
89
6.92M
  }
90
                   
91
/*****************************************************************************/
92
93
dng_camera_profile_info::~dng_camera_profile_info ()
94
7.37M
  {
95
  
96
7.37M
  }
97
                   
98
/*****************************************************************************/
99
100
bool dng_camera_profile_info::ParseTag (dng_stream &stream,
101
                    uint32 parentCode,
102
                    uint32 tagCode,
103
                    uint32 tagType,
104
                    uint32 tagCount,
105
                    uint64 tagOffset)
106
44.8M
  {
107
108
44.8M
  switch (tagCode)
109
44.8M
    {
110
      
111
98.2k
    case tcCalibrationIlluminant1:
112
98.2k
      {
113
      
114
98.2k
      CheckTagType (parentCode, tagCode, tagType, ttShort);
115
      
116
98.2k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
117
      
118
98.2k
      fCalibrationIlluminant1 = stream.TagValue_uint32 (tagType);
119
      
120
      #if qDNGValidate
121
122
      if (gVerbose)
123
        {
124
        
125
        printf ("CalibrationIlluminant1: %s\n",
126
            LookupLightSource (fCalibrationIlluminant1));
127
        
128
        }
129
        
130
      #endif
131
        
132
98.2k
      break;
133
      
134
0
      }
135
      
136
5.04k
    case tcCalibrationIlluminant2:
137
5.04k
      {
138
      
139
5.04k
      CheckTagType (parentCode, tagCode, tagType, ttShort);
140
      
141
5.04k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
142
      
143
5.04k
      fCalibrationIlluminant2 = stream.TagValue_uint32 (tagType);
144
      
145
      #if qDNGValidate
146
147
      if (gVerbose)
148
        {
149
        
150
        printf ("CalibrationIlluminant2: %s\n",
151
            LookupLightSource (fCalibrationIlluminant2));
152
        
153
        }
154
        
155
      #endif
156
        
157
5.04k
      break;
158
      
159
0
      }
160
161
186k
    case tcColorMatrix1:
162
186k
      {
163
      
164
186k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
165
      
166
186k
      if (fColorPlanes == 0)
167
141k
        {
168
      
169
141k
        fColorPlanes = Pin_uint32 (0, tagCount / 3, kMaxColorPlanes);
170
                  
171
141k
        }
172
        
173
186k
      if (!CheckColorImage (parentCode, tagCode, fColorPlanes))
174
2.33k
        return false;
175
        
176
184k
      if (!ParseMatrixTag (stream,
177
184k
                 parentCode,
178
184k
                 tagCode,
179
184k
                 tagType,
180
184k
                 tagCount,
181
184k
                 fColorPlanes,
182
184k
                 3,
183
184k
                 fColorMatrix1))
184
41.2k
        return false;
185
        
186
      #if qDNGValidate
187
      
188
      if (gVerbose)
189
        {
190
        
191
        printf ("ColorMatrix1:\n");
192
        
193
        DumpMatrix (fColorMatrix1);
194
                
195
        }
196
        
197
      #endif
198
        
199
143k
      break;
200
      
201
184k
      }
202
203
222k
    case tcColorMatrix2:
204
222k
      {
205
      
206
222k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
207
      
208
      // Kludge - Hasselblad FFF files are very DNG-like, but sometimes
209
      // only have a ColorMatrix2 tag and no ColorMatrix1 tag.
210
      
211
222k
      bool hasselbladHack = (fColorPlanes == 0);
212
      
213
222k
      if (hasselbladHack)
214
61.9k
        {
215
        
216
61.9k
        fColorPlanes = Pin_uint32 (0, tagCount / 3, kMaxColorPlanes);
217
        
218
        #if qDNGValidate
219
        
220
        ReportWarning ("ColorMatrix2 without ColorMatrix1");
221
        
222
        #endif
223
      
224
61.9k
        }
225
      
226
222k
      if (!CheckColorImage (parentCode, tagCode, fColorPlanes))
227
4.44k
        return false;
228
        
229
218k
      if (!ParseMatrixTag (stream,
230
218k
                 parentCode,
231
218k
                 tagCode,
232
218k
                 tagType,
233
218k
                 tagCount,
234
218k
                 fColorPlanes,
235
218k
                 3,
236
218k
                 fColorMatrix2))
237
185k
        return false;
238
        
239
      #if qDNGValidate
240
      
241
      if (gVerbose)
242
        {
243
        
244
        printf ("ColorMatrix2:\n");
245
        
246
        DumpMatrix (fColorMatrix2);
247
                
248
        }
249
        
250
      #endif
251
        
252
33.2k
      if (hasselbladHack)
253
3.97k
        {
254
        
255
3.97k
        fColorMatrix1 = fColorMatrix2;
256
        
257
3.97k
        fColorMatrix2 = dng_matrix ();
258
        
259
3.97k
        }
260
        
261
33.2k
      break;
262
      
263
218k
      }
264
265
72.8k
    case tcForwardMatrix1:
266
72.8k
      {
267
      
268
72.8k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
269
      
270
72.8k
      if (!CheckColorImage (parentCode, tagCode, fColorPlanes))
271
60.5k
        return false;
272
        
273
12.3k
      if (!ParseMatrixTag (stream,
274
12.3k
                 parentCode,
275
12.3k
                 tagCode,
276
12.3k
                 tagType,
277
12.3k
                 tagCount,
278
12.3k
                 3,
279
12.3k
                 fColorPlanes,
280
12.3k
                 fForwardMatrix1))
281
4.98k
        return false;
282
        
283
      #if qDNGValidate
284
      
285
      if (gVerbose)
286
        {
287
        
288
        printf ("ForwardMatrix1:\n");
289
        
290
        DumpMatrix (fForwardMatrix1);
291
                
292
        }
293
        
294
      #endif
295
        
296
7.33k
      break;
297
      
298
12.3k
      }
299
300
22.4k
    case tcForwardMatrix2:
301
22.4k
      {
302
      
303
22.4k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
304
      
305
22.4k
      if (!CheckColorImage (parentCode, tagCode, fColorPlanes))
306
19.7k
        return false;
307
        
308
2.71k
      if (!ParseMatrixTag (stream,
309
2.71k
                 parentCode,
310
2.71k
                 tagCode,
311
2.71k
                 tagType,
312
2.71k
                 tagCount,
313
2.71k
                 3,
314
2.71k
                 fColorPlanes,
315
2.71k
                 fForwardMatrix2))
316
399
        return false;
317
        
318
      #if qDNGValidate
319
      
320
      if (gVerbose)
321
        {
322
        
323
        printf ("ForwardMatrix2:\n");
324
        
325
        DumpMatrix (fForwardMatrix2);
326
                
327
        }
328
        
329
      #endif
330
        
331
2.31k
      break;
332
      
333
2.71k
      }
334
335
65.6k
    case tcReductionMatrix1:
336
65.6k
      {
337
      
338
65.6k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
339
      
340
65.6k
      if (!CheckColorImage (parentCode, tagCode, fColorPlanes))
341
56.2k
        return false;
342
        
343
9.42k
      if (!ParseMatrixTag (stream,
344
9.42k
                 parentCode,
345
9.42k
                 tagCode,
346
9.42k
                 tagType,
347
9.42k
                 tagCount,
348
9.42k
                 3,
349
9.42k
                 fColorPlanes,
350
9.42k
                 fReductionMatrix1))
351
5.82k
        return false;
352
        
353
      #if qDNGValidate
354
      
355
      if (gVerbose)
356
        {
357
        
358
        printf ("ReductionMatrix1:\n");
359
        
360
        DumpMatrix (fReductionMatrix1);
361
                
362
        }
363
        
364
      #endif
365
        
366
3.59k
      break;
367
      
368
9.42k
      }
369
370
315k
    case tcReductionMatrix2:
371
315k
      {
372
      
373
315k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
374
      
375
315k
      if (!CheckColorImage (parentCode, tagCode, fColorPlanes))
376
287k
        return false;
377
        
378
27.5k
      if (!ParseMatrixTag (stream,
379
27.5k
                 parentCode,
380
27.5k
                 tagCode,
381
27.5k
                 tagType,
382
27.5k
                 tagCount,
383
27.5k
                 3,
384
27.5k
                 fColorPlanes,
385
27.5k
                 fReductionMatrix2))
386
4.59k
        return false;
387
        
388
      #if qDNGValidate
389
      
390
      if (gVerbose)
391
        {
392
        
393
        printf ("ReductionMatrix2:\n");
394
        
395
        DumpMatrix (fReductionMatrix2);
396
                
397
        }
398
        
399
      #endif
400
        
401
22.9k
      break;
402
      
403
27.5k
      }
404
405
73.3k
    case tcProfileCalibrationSignature:
406
73.3k
      {
407
408
73.3k
      CheckTagType (parentCode, tagCode, tagType, ttAscii, ttByte);
409
      
410
73.3k
      ParseStringTag (stream,
411
73.3k
              parentCode,
412
73.3k
              tagCode,
413
73.3k
              tagCount,
414
73.3k
              fProfileCalibrationSignature,
415
73.3k
              false);
416
      
417
      #if qDNGValidate
418
            
419
      if (gVerbose)
420
        {
421
        
422
        printf ("ProfileCalibrationSignature: ");
423
        
424
        DumpString (fProfileCalibrationSignature);
425
        
426
        printf ("\n");
427
        
428
        }
429
430
      #endif
431
      
432
73.3k
      break;
433
      
434
27.5k
      }
435
      
436
83.0k
    case tcProfileName:
437
83.0k
      {
438
      
439
83.0k
      CheckTagType (parentCode, tagCode, tagType, ttAscii, ttByte);
440
      
441
83.0k
      ParseStringTag (stream,
442
83.0k
              parentCode,
443
83.0k
              tagCode,
444
83.0k
              tagCount,
445
83.0k
              fProfileName,
446
83.0k
              false);
447
      
448
      #if qDNGValidate
449
      
450
      if (gVerbose)
451
        {
452
        
453
        printf ("ProfileName: ");
454
        
455
        DumpString (fProfileName);
456
        
457
        printf ("\n");
458
        
459
        }
460
        
461
      #endif
462
      
463
83.0k
      break;
464
      
465
27.5k
      }
466
      
467
8.98k
    case tcProfileCopyright:
468
8.98k
      {
469
470
8.98k
      CheckTagType (parentCode, tagCode, tagType, ttAscii, ttByte);
471
      
472
8.98k
      ParseStringTag (stream,
473
8.98k
              parentCode,
474
8.98k
              tagCode,
475
8.98k
              tagCount,
476
8.98k
              fProfileCopyright,
477
8.98k
              false);
478
      
479
      #if qDNGValidate
480
            
481
      if (gVerbose)
482
        {
483
        
484
        printf ("ProfileCopyright: ");
485
        
486
        DumpString (fProfileCopyright);
487
        
488
        printf ("\n");
489
        
490
        }
491
492
      #endif
493
      
494
8.98k
      break;
495
      
496
27.5k
      }
497
498
49.7k
    case tcProfileEmbedPolicy:
499
49.7k
      {
500
        
501
49.7k
      CheckTagType (parentCode, tagCode, tagType, ttLong);
502
      
503
49.7k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
504
      
505
49.7k
      fEmbedPolicy = stream.TagValue_uint32 (tagType);
506
507
      #if qDNGValidate
508
509
      if (gVerbose)
510
        {
511
        
512
        const char *policy;
513
514
        switch (fEmbedPolicy)
515
          {
516
517
          case pepAllowCopying:
518
            policy = "Allow copying";
519
            break;
520
521
          case pepEmbedIfUsed:
522
            policy = "Embed if used";
523
            break;
524
525
          case pepEmbedNever:
526
            policy = "Embed never";
527
            break;
528
529
          case pepNoRestrictions:
530
            policy = "No restrictions";
531
            break;
532
533
          default:
534
            policy = "INVALID VALUE";
535
  
536
          }
537
538
        printf ("ProfileEmbedPolicy: %s\n", policy);
539
                
540
        }
541
542
      #endif
543
544
49.7k
      break;
545
      
546
27.5k
      }
547
548
351k
    case tcProfileHueSatMapDims:
549
351k
      {
550
        
551
351k
      CheckTagType (parentCode, tagCode, tagType, ttLong);
552
      
553
351k
      CheckTagCount (parentCode, tagCode, tagCount, 2, 3);
554
      
555
351k
      fProfileHues = stream.TagValue_uint32 (tagType);
556
351k
      fProfileSats = stream.TagValue_uint32 (tagType);
557
558
351k
      if (tagCount > 2)
559
94.8k
        fProfileVals = stream.TagValue_uint32 (tagType);
560
256k
      else
561
256k
        fProfileVals = 1;
562
      
563
      #if qDNGValidate
564
565
      if (gVerbose)
566
        {
567
        
568
        printf ("ProfileHueSatMapDims: Hues = %u, Sats = %u, Vals = %u\n",
569
            (unsigned) fProfileHues,
570
            (unsigned) fProfileSats,
571
            (unsigned) fProfileVals);
572
                
573
        }
574
        
575
      #endif
576
577
351k
      break;
578
      
579
27.5k
      }
580
581
57.8k
    case tcProfileHueSatMapData1:
582
57.8k
      {
583
584
57.8k
      if (!CheckTagType (parentCode, tagCode, tagType, ttFloat))
585
8.48k
        return false;
586
        
587
49.4k
      bool skipSat0 = (tagCount ==
588
49.4k
               SafeUint32Mult(fProfileHues,
589
49.4k
                      SafeUint32Sub(fProfileSats, 1u),
590
49.4k
                      fProfileVals,
591
49.4k
                      3u));
592
      
593
49.4k
      if (!skipSat0)
594
38.2k
        {
595
      
596
38.2k
        if (!CheckTagCount (parentCode, tagCode, tagCount,
597
38.2k
                  SafeUint32Mult(fProfileHues, fProfileSats, fProfileVals, 3)))
598
35.4k
          return false;
599
          
600
38.2k
        }
601
        
602
13.9k
      fBigEndian = stream.BigEndian ();
603
        
604
13.9k
      fHueSatDeltas1Offset = tagOffset;
605
13.9k
      fHueSatDeltas1Count  = tagCount;
606
      
607
      #if qDNGValidate
608
609
      if (gVerbose)
610
        {
611
        
612
        printf ("ProfileHueSatMapData1:\n");
613
        
614
        DumpHueSatMap (stream,
615
                 fProfileHues,
616
                 fProfileSats,
617
                 fProfileVals,
618
                 skipSat0);
619
          
620
        }
621
        
622
      #endif
623
624
13.9k
      break;
625
      
626
49.4k
      }
627
628
237k
    case tcProfileHueSatMapData2:
629
237k
      {
630
631
237k
      if (!CheckTagType (parentCode, tagCode, tagType, ttFloat))
632
7.00k
        return false;
633
      
634
230k
      bool skipSat0 = (tagCount ==
635
230k
               SafeUint32Mult(fProfileHues,
636
230k
                      SafeUint32Sub(fProfileSats, 1u),
637
230k
                      fProfileVals,
638
230k
                      3u));
639
      
640
230k
      if (!skipSat0)
641
183k
        {
642
      
643
183k
        if (!CheckTagCount (parentCode, tagCode, tagCount,
644
183k
                  SafeUint32Mult(fProfileHues, fProfileSats, fProfileVals, 3)))
645
179k
          return false;
646
          
647
183k
        }
648
      
649
50.6k
      fBigEndian = stream.BigEndian ();
650
        
651
50.6k
      fHueSatDeltas2Offset = tagOffset;
652
50.6k
      fHueSatDeltas2Count  = tagCount;
653
      
654
      #if qDNGValidate
655
656
      if (gVerbose)
657
        {
658
        
659
        printf ("ProfileHueSatMapData2:\n");
660
        
661
        DumpHueSatMap (stream,
662
                 fProfileHues,
663
                 fProfileSats,
664
                 fProfileVals,
665
                 skipSat0);
666
          
667
        }
668
        
669
      #endif
670
671
50.6k
      break;
672
      
673
230k
      }
674
675
1.23k
    case tcProfileHueSatMapEncoding:
676
1.23k
      {
677
678
1.23k
      CheckTagType (parentCode, tagCode, tagType, ttLong);
679
      
680
1.23k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
681
      
682
1.23k
      fHueSatMapEncoding = stream.TagValue_uint32 (tagType);
683
684
      #if qDNGValidate
685
686
      if (gVerbose)
687
        {
688
        
689
        const char *encoding = NULL;
690
691
        switch (fHueSatMapEncoding)
692
          {
693
694
          case encoding_Linear:
695
            encoding = "Linear";
696
            break;
697
698
          case encoding_sRGB:
699
            encoding = "sRGB";
700
            break;
701
702
          default:
703
            encoding = "INVALID VALUE";
704
  
705
          }
706
707
        printf ("ProfileHueSatMapEncoding: %s\n", encoding);
708
                
709
        }
710
711
      #endif
712
713
1.23k
      break;
714
      
715
230k
      }
716
717
199k
    case tcProfileLookTableDims:
718
199k
      {
719
        
720
199k
      CheckTagType (parentCode, tagCode, tagType, ttLong);
721
      
722
199k
      CheckTagCount (parentCode, tagCode, tagCount, 2, 3);
723
      
724
199k
      fLookTableHues = stream.TagValue_uint32 (tagType);
725
199k
      fLookTableSats = stream.TagValue_uint32 (tagType);
726
727
199k
      if (tagCount > 2)
728
123k
        fLookTableVals = stream.TagValue_uint32 (tagType);
729
76.3k
      else
730
76.3k
        fLookTableVals = 1;
731
      
732
      #if qDNGValidate
733
734
      if (gVerbose)
735
        {
736
        
737
        printf ("ProfileLookTableDims: Hues = %u, Sats = %u, Vals = %u\n",
738
            (unsigned) fLookTableHues,
739
            (unsigned) fLookTableSats,
740
            (unsigned) fLookTableVals);
741
                
742
        }
743
        
744
      #endif
745
746
199k
      break;
747
      
748
230k
      }
749
750
81.3k
    case tcProfileLookTableData:
751
81.3k
      {
752
753
81.3k
      if (!CheckTagType (parentCode, tagCode, tagType, ttFloat))
754
17.6k
        return false;
755
      
756
63.7k
      bool skipSat0 = (tagCount ==
757
63.7k
               SafeUint32Mult(fLookTableHues,
758
63.7k
                      SafeUint32Sub(fLookTableSats, 1u),
759
63.7k
                      fLookTableVals,
760
63.7k
                      3u));
761
      
762
63.7k
      if (!skipSat0)
763
47.5k
        {
764
      
765
47.5k
        if (!CheckTagCount (parentCode, tagCode, tagCount,
766
47.5k
                  SafeUint32Mult(fLookTableHues,
767
47.5k
                          fLookTableSats,
768
47.5k
                          fLookTableVals, 3)))
769
18.8k
          return false;
770
          
771
47.5k
        }
772
      
773
44.8k
      fBigEndian = stream.BigEndian ();
774
        
775
44.8k
      fLookTableOffset = tagOffset;
776
44.8k
      fLookTableCount  = tagCount;
777
      
778
      #if qDNGValidate
779
780
      if (gVerbose)
781
        {
782
        
783
        printf ("ProfileLookTableData:\n");
784
        
785
        DumpHueSatMap (stream,
786
                 fLookTableHues,
787
                 fLookTableSats,
788
                 fLookTableVals,
789
                 skipSat0);
790
          
791
        }
792
        
793
      #endif
794
795
44.8k
      break;
796
      
797
63.7k
      }
798
799
5.81k
    case tcProfileLookTableEncoding:
800
5.81k
      {
801
802
5.81k
      CheckTagType (parentCode, tagCode, tagType, ttLong);
803
      
804
5.81k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
805
      
806
5.81k
      fLookTableEncoding = stream.TagValue_uint32 (tagType);
807
808
      #if qDNGValidate
809
810
      if (gVerbose)
811
        {
812
        
813
        const char *encoding = NULL;
814
815
        switch (fLookTableEncoding)
816
          {
817
818
          case encoding_Linear:
819
            encoding = "Linear";
820
            break;
821
822
          case encoding_sRGB:
823
            encoding = "sRGB";
824
            break;
825
826
          default:
827
            encoding = "INVALID VALUE";
828
  
829
          }
830
831
        printf ("ProfileLookTableEncoding: %s\n", encoding);
832
                
833
        }
834
835
      #endif
836
837
5.81k
      break;
838
      
839
63.7k
      }
840
841
28.2k
    case tcBaselineExposureOffset:
842
28.2k
      {
843
      
844
28.2k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
845
      
846
28.2k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
847
      
848
28.2k
      fBaselineExposureOffset = stream.TagValue_srational (tagType);
849
      
850
      #if qDNGValidate
851
      
852
      if (gVerbose)
853
        {
854
        
855
        printf ("BaselineExposureOffset: %+0.2f\n",
856
              fBaselineExposureOffset.As_real64 ());
857
              
858
        }
859
        
860
      #endif
861
        
862
28.2k
      break;
863
      
864
63.7k
      }
865
      
866
18.7k
    case tcDefaultBlackRender:
867
18.7k
      {
868
869
18.7k
      CheckTagType (parentCode, tagCode, tagType, ttLong);
870
      
871
18.7k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
872
      
873
18.7k
      fDefaultBlackRender = stream.TagValue_uint32 (tagType);
874
875
      #if qDNGValidate
876
877
      if (gVerbose)
878
        {
879
        
880
        const char *setting = NULL;
881
882
        switch (fDefaultBlackRender)
883
          {
884
885
          case defaultBlackRender_Auto:
886
            setting = "Auto";
887
            break;
888
889
          case defaultBlackRender_None:
890
            setting = "None";
891
            break;
892
893
          default:
894
            setting = "INVALID VALUE";
895
  
896
          }
897
898
        printf ("DefaultBlackRender: %s\n", 
899
            setting);
900
                
901
        }
902
903
      #endif
904
905
18.7k
      break;
906
      
907
63.7k
      }
908
909
86.2k
    case tcProfileToneCurve:
910
86.2k
      {
911
        
912
86.2k
      if (!CheckTagType (parentCode, tagCode, tagType, ttFloat))
913
5.83k
        return false;
914
      
915
80.4k
      if (!CheckTagCount (parentCode, tagCode, tagCount, 4, tagCount))
916
26.1k
        return false;
917
    
918
54.3k
      if ((tagCount & 1) != 0)
919
8.11k
        {
920
        
921
        #if qDNGValidate
922
        
923
          {
924
            
925
          char message [256];
926
          
927
          sprintf (message,
928
               "%s %s has odd count (%u)",
929
               LookupParentCode (parentCode),
930
               LookupTagCode (parentCode, tagCode),
931
               (unsigned) tagCount);
932
               
933
          ReportWarning (message);
934
                 
935
          }
936
          
937
        #endif
938
          
939
8.11k
        return false;
940
        
941
8.11k
        }
942
      
943
46.1k
      fBigEndian = stream.BigEndian ();
944
        
945
46.1k
      fToneCurveOffset = tagOffset;
946
46.1k
      fToneCurveCount  = tagCount;
947
      
948
      #if qDNGValidate
949
950
      if (gVerbose)
951
        {
952
        
953
        DumpTagValues (stream,
954
                 "Coord",
955
                 parentCode,
956
                 tagCode,
957
                 tagType,
958
                 tagCount);
959
          
960
                
961
        }
962
        
963
      #endif
964
965
46.1k
      break;
966
      
967
54.3k
      }
968
969
306k
    case tcUniqueCameraModel:
970
306k
      {
971
      
972
      // Note: This code is only used when parsing stand-alone
973
      // profiles.  The embedded profiles are assumed to be restricted
974
      // to the model they are embedded in.
975
      
976
306k
      CheckTagType (parentCode, tagCode, tagType, ttAscii);
977
      
978
306k
      ParseStringTag (stream,
979
306k
              parentCode,
980
306k
              tagCode,
981
306k
              tagCount,
982
306k
              fUniqueCameraModel,
983
306k
              false);
984
              
985
306k
      bool didTrim = fUniqueCameraModel.TrimTrailingBlanks ();
986
      
987
      #if qDNGValidate
988
      
989
      if (didTrim)
990
        {
991
        
992
        ReportWarning ("UniqueCameraModel string has trailing blanks");
993
        
994
        }
995
      
996
      if (gVerbose)
997
        {
998
        
999
        printf ("UniqueCameraModel: ");
1000
        
1001
        DumpString (fUniqueCameraModel);
1002
        
1003
        printf ("\n");
1004
        
1005
        }
1006
        
1007
      #else
1008
      
1009
306k
      (void) didTrim;   // Unused
1010
        
1011
306k
      #endif
1012
      
1013
306k
      break;
1014
      
1015
54.3k
      }
1016
1017
42.2M
    default:
1018
42.2M
      {
1019
      
1020
42.2M
      return false;
1021
      
1022
54.3k
      }
1023
      
1024
44.8M
    }
1025
1026
1.56M
  return true;
1027
  
1028
44.8M
  }
1029
1030
/*****************************************************************************/
1031
1032
bool dng_camera_profile_info::ParseExtended (dng_stream &stream)
1033
6.82M
  {
1034
1035
6.82M
  try
1036
6.82M
    {
1037
    
1038
    // Offsets are relative to the start of this structure, not the entire file.
1039
1040
6.82M
    uint64 startPosition = stream.Position ();
1041
1042
    // Read header. Like a TIFF header, but with different magic number
1043
    // Plus all offsets are relative to the start of the IFD, not to the
1044
    // stream or file.
1045
1046
6.82M
    uint16 byteOrder = stream.Get_uint16 ();
1047
1048
6.82M
    if (byteOrder == byteOrderMM)
1049
98.6k
      fBigEndian = true;
1050
      
1051
6.72M
    else if (byteOrder == byteOrderII)
1052
2.44M
      fBigEndian = false;
1053
      
1054
4.28M
    else
1055
4.28M
      return false;
1056
1057
2.54M
    TempBigEndian setEndianness (stream, fBigEndian);
1058
1059
2.54M
    uint16 magicNumber = stream.Get_uint16 ();
1060
1061
2.54M
    if (magicNumber != magicExtendedProfile)
1062
1.91M
      {
1063
1.91M
      return false;
1064
1.91M
      }
1065
1066
631k
    uint32 offset = stream.Get_uint32 ();
1067
1068
631k
    stream.Skip (SafeUint32Sub(offset, 8u));
1069
1070
    // Start on IFD entries.
1071
1072
631k
    uint32 ifdEntries = stream.Get_uint16 ();
1073
  
1074
631k
    if (ifdEntries < 1)
1075
675
      {
1076
675
      return false;
1077
675
      }
1078
    
1079
45.4M
    for (uint32 tag_index = 0; tag_index < ifdEntries; tag_index++)
1080
44.8M
      {
1081
      
1082
44.8M
      stream.SetReadPosition (startPosition + 8 + 2 + tag_index * 12);
1083
      
1084
44.8M
      uint16 tagCode  = stream.Get_uint16 ();
1085
44.8M
      uint32 tagType  = stream.Get_uint16 ();
1086
44.8M
      uint32 tagCount = stream.Get_uint32 ();
1087
      
1088
44.8M
      uint64 tagOffset = stream.Position ();
1089
1090
44.8M
      if (SafeUint32Mult(TagTypeSize (tagType), tagCount) > 4)
1091
5.17M
        {
1092
1093
5.17M
        tagOffset = startPosition + stream.Get_uint32 ();
1094
1095
5.17M
        stream.SetReadPosition (tagOffset);
1096
1097
5.17M
        }
1098
        
1099
44.8M
      if (!ParseTag (stream,
1100
44.8M
               0,
1101
44.8M
               tagCode,
1102
44.8M
               tagType,
1103
44.8M
               tagCount,
1104
44.8M
               tagOffset))
1105
42.9M
        {
1106
        
1107
        #if qDNGValidate
1108
    
1109
        if (gVerbose)
1110
          {
1111
              
1112
          stream.SetReadPosition (tagOffset);
1113
        
1114
          printf ("*");
1115
            
1116
          DumpTagValues (stream,
1117
                   LookupTagType (tagType),
1118
                   0,
1119
                   tagCode,
1120
                   tagType,
1121
                   tagCount);
1122
          
1123
          }
1124
          
1125
        #endif
1126
1127
42.9M
        }
1128
1129
44.8M
      }
1130
      
1131
630k
    return true;
1132
1133
631k
    }
1134
    
1135
6.82M
  catch (...)
1136
6.82M
    {
1137
    
1138
    // Eat parsing errors.
1139
    
1140
354k
    }
1141
1142
354k
  return false;
1143
  
1144
6.82M
  }
1145
1146
/*****************************************************************************/
1147
1148
dng_shared::dng_shared ()
1149
  
1150
91.6k
  : fExifIFD       (0)
1151
91.6k
  , fGPSInfo       (0)
1152
91.6k
  , fInteroperabilityIFD (0)
1153
91.6k
  , fKodakDCRPrivateIFD  (0)
1154
91.6k
  , fKodakKDCPrivateIFD  (0)
1155
    
1156
91.6k
  , fXMPCount  (0)
1157
91.6k
  , fXMPOffset (0)
1158
    
1159
91.6k
  , fIPTC_NAA_Count  (0)
1160
91.6k
  , fIPTC_NAA_Offset (0)
1161
  
1162
91.6k
  , fMakerNoteCount  (0)
1163
91.6k
  , fMakerNoteOffset (0)
1164
91.6k
  , fMakerNoteSafety (0)
1165
  
1166
91.6k
  , fDNGVersion         (0)
1167
91.6k
  , fDNGBackwardVersion (0)
1168
  
1169
91.6k
  , fUniqueCameraModel    ()
1170
91.6k
  , fLocalizedCameraModel ()
1171
  
1172
91.6k
  , fCameraProfile ()
1173
  
1174
91.6k
  , fExtraCameraProfiles ()
1175
  
1176
91.6k
  , fCameraCalibration1 ()
1177
91.6k
  , fCameraCalibration2 ()
1178
    
1179
91.6k
  , fCameraCalibrationSignature  ()
1180
1181
91.6k
  , fAnalogBalance ()
1182
    
1183
91.6k
  , fAsShotNeutral ()
1184
    
1185
91.6k
  , fAsShotWhiteXY ()
1186
  
1187
91.6k
  , fBaselineExposure      (0, 1)
1188
91.6k
  , fBaselineNoise         (1, 1)
1189
91.6k
  , fNoiseReductionApplied (0, 0)
1190
91.6k
  , fBaselineSharpness     (1, 1)
1191
91.6k
  , fLinearResponseLimit   (1, 1)
1192
91.6k
  , fShadowScale           (1, 1)
1193
  
1194
91.6k
  , fHasBaselineExposure (false)
1195
91.6k
  , fHasShadowScale      (false)
1196
  
1197
91.6k
  , fDNGPrivateDataCount  (0)
1198
91.6k
  , fDNGPrivateDataOffset (0)
1199
  
1200
91.6k
  , fRawImageDigest    ()
1201
91.6k
  , fNewRawImageDigest ()
1202
  
1203
91.6k
  , fRawDataUniqueID ()
1204
  
1205
91.6k
  , fOriginalRawFileName ()
1206
    
1207
91.6k
  , fOriginalRawFileDataCount  (0)
1208
91.6k
  , fOriginalRawFileDataOffset (0)
1209
  
1210
91.6k
  , fOriginalRawFileDigest ()
1211
    
1212
91.6k
  , fAsShotICCProfileCount  (0)
1213
91.6k
  , fAsShotICCProfileOffset (0)
1214
  
1215
91.6k
  , fAsShotPreProfileMatrix ()
1216
    
1217
91.6k
  , fCurrentICCProfileCount  (0)
1218
91.6k
  , fCurrentICCProfileOffset (0)
1219
  
1220
91.6k
  , fCurrentPreProfileMatrix ()
1221
  
1222
91.6k
  , fColorimetricReference (crSceneReferred)
1223
1224
91.6k
  , fAsShotProfileName ()
1225
1226
91.6k
  , fNoiseProfile ()
1227
  
1228
91.6k
  , fOriginalDefaultFinalSize     ()
1229
91.6k
  , fOriginalBestQualityFinalSize ()
1230
    
1231
91.6k
  , fOriginalDefaultCropSizeH ()
1232
91.6k
  , fOriginalDefaultCropSizeV ()
1233
    
1234
91.6k
  {
1235
  
1236
91.6k
  }
1237
  
1238
/*****************************************************************************/
1239
1240
dng_shared::~dng_shared ()
1241
91.6k
  {
1242
  
1243
91.6k
  }
1244
    
1245
/*****************************************************************************/
1246
1247
bool dng_shared::ParseTag (dng_stream &stream,
1248
               dng_exif &exif,
1249
               uint32 parentCode,
1250
               bool /* isMainIFD */,
1251
               uint32 tagCode,
1252
               uint32 tagType,
1253
               uint32 tagCount,
1254
               uint64 tagOffset,
1255
               int64 /* offsetDelta */)
1256
1.74M
  {
1257
  
1258
1.74M
  if (parentCode == 0)
1259
767k
    {
1260
    
1261
767k
    if (Parse_ifd0 (stream,
1262
767k
            exif,
1263
767k
            parentCode,
1264
767k
            tagCode,
1265
767k
            tagType,
1266
767k
            tagCount,
1267
767k
            tagOffset))
1268
418k
      {
1269
      
1270
418k
      return true;
1271
      
1272
418k
      }
1273
1274
767k
    }
1275
    
1276
1.32M
  if (parentCode == 0 ||
1277
973k
    parentCode == tcExifIFD)
1278
415k
    {
1279
    
1280
415k
    if (Parse_ifd0_exif (stream,
1281
415k
               exif,
1282
415k
               parentCode,
1283
415k
               tagCode,
1284
415k
               tagType,
1285
415k
               tagCount,
1286
415k
               tagOffset))
1287
14.2k
      {
1288
      
1289
14.2k
      return true;
1290
      
1291
14.2k
      }
1292
1293
415k
    }
1294
    
1295
1.30M
  return false;
1296
    
1297
1.32M
  }
1298
1299
/*****************************************************************************/
1300
1301
// Parses tags that should only appear in IFD 0.
1302
1303
bool dng_shared::Parse_ifd0 (dng_stream &stream,
1304
               dng_exif & /* exif */,
1305
               uint32 parentCode,
1306
               uint32 tagCode,
1307
               uint32 tagType,
1308
               uint32 tagCount,
1309
               uint64 tagOffset)
1310
767k
  {
1311
  
1312
767k
  switch (tagCode)
1313
767k
    {
1314
      
1315
15.0k
    case tcXMP:
1316
15.0k
      {
1317
      
1318
15.0k
      CheckTagType (parentCode, tagCode, tagType, ttByte, ttUndefined);
1319
      
1320
15.0k
      fXMPCount  = tagCount;
1321
15.0k
      fXMPOffset = fXMPCount ? tagOffset : 0;
1322
      
1323
      #if qDNGValidate
1324
      
1325
      if (gVerbose)
1326
        {
1327
        
1328
        printf ("XMP: Count = %u, Offset = %u\n",
1329
            (unsigned) fXMPCount,
1330
            (unsigned) fXMPOffset);
1331
            
1332
        if (fXMPCount)
1333
          {
1334
            
1335
          DumpXMP (stream, fXMPCount);
1336
          
1337
          }
1338
            
1339
        }
1340
        
1341
      #endif
1342
        
1343
15.0k
      break;
1344
      
1345
0
      }
1346
1347
318
    case tcIPTC_NAA:
1348
318
      {
1349
      
1350
318
      CheckTagType (parentCode, tagCode, tagType, ttLong, ttAscii, ttUndefined);
1351
      
1352
318
      fIPTC_NAA_Count = SafeUint32Mult(tagCount,
1353
318
                              TagTypeSize(tagType));
1354
318
      fIPTC_NAA_Offset = fIPTC_NAA_Count ? tagOffset : 0;
1355
      
1356
      #if qDNGValidate
1357
      
1358
      if (gVerbose)
1359
        {
1360
        
1361
        printf ("IPTC/NAA: Count = %u, Offset = %u\n",
1362
            (unsigned) fIPTC_NAA_Count,
1363
            (unsigned) fIPTC_NAA_Offset);
1364
            
1365
        if (fIPTC_NAA_Count)
1366
          {
1367
            
1368
          DumpHexAscii (stream, fIPTC_NAA_Count);
1369
          
1370
          }
1371
          
1372
        // Compute and output the digest.
1373
        
1374
        dng_memory_data buffer (fIPTC_NAA_Count);
1375
        
1376
        stream.SetReadPosition (fIPTC_NAA_Offset);
1377
        
1378
        stream.Get (buffer.Buffer (), fIPTC_NAA_Count);
1379
        
1380
        const uint8 *data = buffer.Buffer_uint8 ();
1381
        
1382
        uint32 count = fIPTC_NAA_Count;
1383
        
1384
        // Method 1: Counting all bytes (this is correct).
1385
        
1386
          {
1387
          
1388
          dng_md5_printer printer;
1389
    
1390
          printer.Process (data, count);
1391
          
1392
          printf ("IPTCDigest: ");
1393
          
1394
          DumpFingerprint (printer.Result ());
1395
          
1396
          printf ("\n");
1397
          
1398
          }
1399
          
1400
        // Method 2: Ignoring zero padding.
1401
          
1402
          {
1403
          
1404
          uint32 removed = 0;
1405
      
1406
          while ((removed < 3) && (count > 0) && (data [count - 1] == 0))
1407
            {
1408
            removed++;
1409
            count--;
1410
            }
1411
            
1412
          if (removed != 0)
1413
            {
1414
          
1415
            dng_md5_printer printer;
1416
      
1417
            printer.Process (data, count);
1418
            
1419
            printf ("IPTCDigest (ignoring zero padding): ");
1420
            
1421
            DumpFingerprint (printer.Result ());
1422
            
1423
            printf ("\n");
1424
            
1425
            }
1426
          
1427
          }
1428
          
1429
        }
1430
        
1431
      #endif
1432
        
1433
318
      break;
1434
      
1435
0
      }
1436
1437
15.4k
    case tcExifIFD:
1438
15.4k
      {
1439
      
1440
15.4k
      CheckTagType (parentCode, tagCode, tagType, ttLong, ttIFD);
1441
      
1442
15.4k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
1443
      
1444
15.4k
      fExifIFD = stream.TagValue_uint32 (tagType);
1445
      
1446
      #if qDNGValidate
1447
      
1448
      if (gVerbose)
1449
        {
1450
        printf ("ExifIFD: %u\n", (unsigned) fExifIFD);
1451
        }
1452
        
1453
      #endif
1454
        
1455
15.4k
      break;
1456
      
1457
0
      }
1458
    
1459
7.63k
    case tcGPSInfo:
1460
7.63k
      {
1461
      
1462
7.63k
      CheckTagType (parentCode, tagCode, tagType, ttLong, ttIFD);
1463
      
1464
7.63k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
1465
      
1466
7.63k
      fGPSInfo = stream.TagValue_uint32 (tagType);
1467
      
1468
      #if qDNGValidate
1469
      
1470
      if (gVerbose)
1471
        {
1472
        printf ("GPSInfo: %u\n", (unsigned) fGPSInfo);
1473
        }
1474
        
1475
      #endif
1476
        
1477
7.63k
      break;
1478
      
1479
0
      }
1480
        
1481
699
    case tcKodakDCRPrivateIFD:
1482
699
      {
1483
      
1484
699
      CheckTagType (parentCode, tagCode, tagType, ttLong, ttIFD);
1485
      
1486
699
      CheckTagCount (parentCode, tagCode, tagCount, 1);
1487
      
1488
699
      fKodakDCRPrivateIFD = stream.TagValue_uint32 (tagType);
1489
      
1490
      #if qDNGValidate
1491
      
1492
      if (gVerbose)
1493
        {
1494
        printf ("KodakDCRPrivateIFD: %u\n", (unsigned) fKodakDCRPrivateIFD);
1495
        }
1496
        
1497
      #endif
1498
        
1499
699
      break;
1500
      
1501
0
      }
1502
    
1503
1.67k
    case tcKodakKDCPrivateIFD:
1504
1.67k
      {
1505
      
1506
1.67k
      CheckTagType (parentCode, tagCode, tagType, ttLong, ttIFD);
1507
      
1508
1.67k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
1509
      
1510
1.67k
      fKodakKDCPrivateIFD = stream.TagValue_uint32 (tagType);
1511
      
1512
      #if qDNGValidate
1513
      
1514
      if (gVerbose)
1515
        {
1516
        printf ("KodakKDCPrivateIFD: %u\n", (unsigned) fKodakKDCPrivateIFD);
1517
        }
1518
        
1519
      #endif
1520
        
1521
1.67k
      break;
1522
      
1523
0
      }
1524
    
1525
80.6k
    case tcDNGVersion:
1526
80.6k
      {
1527
      
1528
80.6k
      CheckTagType (parentCode, tagCode, tagType, ttByte);
1529
      
1530
80.6k
      CheckTagCount (parentCode, tagCode, tagCount, 4);
1531
      
1532
80.6k
      uint32 b0 = stream.Get_uint8 ();
1533
80.6k
      uint32 b1 = stream.Get_uint8 ();
1534
80.6k
      uint32 b2 = stream.Get_uint8 ();
1535
80.6k
      uint32 b3 = stream.Get_uint8 ();
1536
      
1537
80.6k
      fDNGVersion = (b0 << 24) | (b1 << 16) | (b2 << 8) | b3;
1538
      
1539
      #if qDNGValidate
1540
      
1541
      if (gVerbose)
1542
        {
1543
        printf ("DNGVersion: %u.%u.%u.%u\n",
1544
            (unsigned) b0,
1545
            (unsigned) b1,
1546
            (unsigned) b2,
1547
            (unsigned) b3);
1548
        }
1549
        
1550
      #endif
1551
      
1552
80.6k
      break;
1553
      
1554
0
      }
1555
        
1556
26.3k
    case tcDNGBackwardVersion:
1557
26.3k
      {
1558
      
1559
26.3k
      CheckTagType (parentCode, tagCode, tagType, ttByte);
1560
      
1561
26.3k
      CheckTagCount (parentCode, tagCode, tagCount, 4);
1562
      
1563
26.3k
      uint32 b0 = stream.Get_uint8 ();
1564
26.3k
      uint32 b1 = stream.Get_uint8 ();
1565
26.3k
      uint32 b2 = stream.Get_uint8 ();
1566
26.3k
      uint32 b3 = stream.Get_uint8 ();
1567
      
1568
26.3k
      fDNGBackwardVersion = (b0 << 24) | (b1 << 16) | (b2 << 8) | b3;
1569
      
1570
      #if qDNGValidate
1571
      
1572
      if (gVerbose)
1573
        {
1574
        printf ("DNGBackwardVersion: %u.%u.%u.%u\n", 
1575
            (unsigned) b0, 
1576
            (unsigned) b1, 
1577
            (unsigned) b2, 
1578
            (unsigned) b3);
1579
        }
1580
        
1581
      #endif
1582
      
1583
26.3k
      break;
1584
      
1585
0
      }
1586
        
1587
15.3k
    case tcUniqueCameraModel:
1588
15.3k
      {
1589
      
1590
15.3k
      CheckTagType (parentCode, tagCode, tagType, ttAscii);
1591
      
1592
15.3k
      ParseStringTag (stream,
1593
15.3k
              parentCode,
1594
15.3k
              tagCode,
1595
15.3k
              tagCount,
1596
15.3k
              fUniqueCameraModel,
1597
15.3k
              false);
1598
              
1599
15.3k
      bool didTrim = fUniqueCameraModel.TrimTrailingBlanks ();
1600
      
1601
      #if qDNGValidate
1602
      
1603
      if (didTrim)
1604
        {
1605
        
1606
        ReportWarning ("UniqueCameraModel string has trailing blanks");
1607
        
1608
        }
1609
      
1610
      if (gVerbose)
1611
        {
1612
        
1613
        printf ("UniqueCameraModel: ");
1614
        
1615
        DumpString (fUniqueCameraModel);
1616
        
1617
        printf ("\n");
1618
        
1619
        }
1620
        
1621
      #else
1622
      
1623
15.3k
      (void) didTrim;   // Unused
1624
        
1625
15.3k
      #endif
1626
      
1627
15.3k
      break;
1628
      
1629
0
      }
1630
1631
1.25k
    case tcLocalizedCameraModel:
1632
1.25k
      {
1633
      
1634
1.25k
      CheckTagType (parentCode, tagCode, tagType, ttAscii, ttByte);
1635
      
1636
1.25k
      ParseStringTag (stream,
1637
1.25k
              parentCode,
1638
1.25k
              tagCode,
1639
1.25k
              tagCount,
1640
1.25k
              fLocalizedCameraModel,
1641
1.25k
              false);
1642
      
1643
1.25k
      bool didTrim = fLocalizedCameraModel.TrimTrailingBlanks ();
1644
      
1645
      #if qDNGValidate
1646
      
1647
      if (didTrim)
1648
        {
1649
        
1650
        ReportWarning ("LocalizedCameraModel string has trailing blanks");
1651
        
1652
        }
1653
      
1654
      if (gVerbose)
1655
        {
1656
        
1657
        printf ("LocalizedCameraModel: ");
1658
        
1659
        DumpString (fLocalizedCameraModel);
1660
        
1661
        printf ("\n");
1662
        
1663
        }
1664
        
1665
      #else
1666
      
1667
1.25k
      (void) didTrim;   // Unused
1668
        
1669
1.25k
      #endif
1670
      
1671
1.25k
      break;
1672
      
1673
0
      }
1674
      
1675
4.50k
    case tcCameraCalibration1:
1676
4.50k
      {
1677
      
1678
4.50k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
1679
      
1680
4.50k
      if (!CheckColorImage (parentCode, tagCode, fCameraProfile.fColorPlanes))
1681
2.12k
        return false;
1682
        
1683
2.38k
      if (!ParseMatrixTag (stream,
1684
2.38k
                 parentCode,
1685
2.38k
                 tagCode,
1686
2.38k
                 tagType,
1687
2.38k
                 tagCount,
1688
2.38k
                 fCameraProfile.fColorPlanes,
1689
2.38k
                 fCameraProfile.fColorPlanes,
1690
2.38k
                 fCameraCalibration1))
1691
915
        return false;
1692
        
1693
      #if qDNGValidate
1694
      
1695
      if (gVerbose)
1696
        {
1697
        
1698
        printf ("CameraCalibration1:\n");
1699
        
1700
        DumpMatrix (fCameraCalibration1);
1701
                
1702
        }
1703
        
1704
      #endif
1705
        
1706
1.46k
      break;
1707
      
1708
2.38k
      }
1709
1710
4.21k
    case tcCameraCalibration2:
1711
4.21k
      {
1712
      
1713
4.21k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
1714
      
1715
4.21k
      if (!CheckColorImage (parentCode, tagCode, fCameraProfile.fColorPlanes))
1716
419
        return false;
1717
        
1718
3.79k
      if (!ParseMatrixTag (stream,
1719
3.79k
                 parentCode,
1720
3.79k
                 tagCode,
1721
3.79k
                 tagType,
1722
3.79k
                 tagCount,
1723
3.79k
                 fCameraProfile.fColorPlanes,
1724
3.79k
                 fCameraProfile.fColorPlanes,
1725
3.79k
                 fCameraCalibration2))
1726
3.09k
        return false;
1727
        
1728
      #if qDNGValidate
1729
      
1730
      if (gVerbose)
1731
        {
1732
        
1733
        printf ("CameraCalibration2:\n");
1734
        
1735
        DumpMatrix (fCameraCalibration2);
1736
                
1737
        }
1738
        
1739
      #endif
1740
        
1741
699
      break;
1742
      
1743
3.79k
      }
1744
1745
3.07k
    case tcCameraCalibrationSignature:
1746
3.07k
      {
1747
1748
3.07k
      CheckTagType (parentCode, tagCode, tagType, ttAscii, ttByte);
1749
      
1750
3.07k
      ParseStringTag (stream,
1751
3.07k
              parentCode,
1752
3.07k
              tagCode,
1753
3.07k
              tagCount,
1754
3.07k
              fCameraCalibrationSignature,
1755
3.07k
              false);
1756
      
1757
      #if qDNGValidate
1758
            
1759
      if (gVerbose)
1760
        {
1761
        
1762
        printf ("CameraCalibrationSignature: ");
1763
        
1764
        DumpString (fCameraCalibrationSignature);
1765
        
1766
        printf ("\n");
1767
        
1768
        }
1769
1770
      #endif
1771
      
1772
3.07k
      break;
1773
      
1774
3.79k
      }
1775
      
1776
28.7k
    case tcAnalogBalance:
1777
28.7k
      {
1778
      
1779
28.7k
      CheckTagType (parentCode, tagCode, tagType, ttRational);
1780
      
1781
      // Kludge - Hasselblad FFF files are very DNG-like, but sometimes
1782
      // they don't have any ColorMatrix tags.
1783
      
1784
28.7k
      bool hasselbladHack = (fDNGVersion == 0 &&
1785
1.73k
                   fCameraProfile.fColorPlanes == 0);
1786
      
1787
28.7k
      if (hasselbladHack)
1788
590
        {
1789
        
1790
590
        fCameraProfile.fColorPlanes = Pin_uint32 (0, tagCount, kMaxColorPlanes);
1791
        
1792
        #if qDNGValidate
1793
        
1794
        ReportWarning ("AnalogBalance without ColorMatrix1");
1795
        
1796
        #endif
1797
      
1798
590
        }
1799
      
1800
28.7k
      if (!CheckColorImage (parentCode, tagCode, fCameraProfile.fColorPlanes))
1801
4.02k
        return false;
1802
        
1803
24.7k
      if (!ParseVectorTag (stream,
1804
24.7k
                 parentCode,
1805
24.7k
                 tagCode,
1806
24.7k
                 tagType,
1807
24.7k
                 tagCount,
1808
24.7k
                 fCameraProfile.fColorPlanes,
1809
24.7k
                 fAnalogBalance))
1810
9.59k
        return false;
1811
  
1812
      #if qDNGValidate
1813
      
1814
      if (gVerbose)
1815
        {
1816
        
1817
        printf ("AnalogBalance:");
1818
        
1819
        DumpVector (fAnalogBalance);
1820
                
1821
        }
1822
        
1823
      #endif
1824
        
1825
15.1k
      break;
1826
      
1827
24.7k
      }
1828
      
1829
15.1k
    case tcAsShotNeutral:
1830
12.6k
      {
1831
      
1832
12.6k
      CheckTagType (parentCode, tagCode, tagType, ttRational);
1833
      
1834
      // Kludge - Hasselblad FFF files are very DNG-like, but sometimes
1835
      // they don't have any ColorMatrix tags.
1836
      
1837
12.6k
      bool hasselbladHack = (fDNGVersion == 0 &&
1838
2.19k
                   fCameraProfile.fColorPlanes == 0);
1839
      
1840
12.6k
      if (hasselbladHack)
1841
901
        {
1842
        
1843
901
        fCameraProfile.fColorPlanes = Pin_uint32 (0, tagCount, kMaxColorPlanes);
1844
        
1845
        #if qDNGValidate
1846
        
1847
        ReportWarning ("AsShotNeutral without ColorMatrix1");
1848
        
1849
        #endif
1850
      
1851
901
        }
1852
      
1853
12.6k
      if (!CheckColorImage (parentCode, tagCode, fCameraProfile.fColorPlanes))
1854
3.63k
        return false;
1855
        
1856
9.05k
      if (!ParseVectorTag (stream,
1857
9.05k
                 parentCode,
1858
9.05k
                 tagCode,
1859
9.05k
                 tagType,
1860
9.05k
                 tagCount,
1861
9.05k
                 fCameraProfile.fColorPlanes,
1862
9.05k
                 fAsShotNeutral))
1863
1.63k
        return false;
1864
  
1865
      #if qDNGValidate
1866
      
1867
      if (gVerbose)
1868
        {
1869
        
1870
        printf ("AsShotNeutral:");
1871
        
1872
        DumpVector (fAsShotNeutral);
1873
                
1874
        }
1875
        
1876
      #endif
1877
        
1878
7.42k
      break;
1879
      
1880
9.05k
      }
1881
      
1882
13.7k
    case tcAsShotWhiteXY:
1883
13.7k
      {
1884
      
1885
13.7k
      CheckTagType (parentCode, tagCode, tagType, ttRational);
1886
      
1887
13.7k
      if (!CheckColorImage (parentCode, tagCode, fCameraProfile.fColorPlanes))
1888
1.22k
        return false;
1889
        
1890
12.5k
      if (!CheckTagCount (parentCode, tagCode, tagCount, 2))
1891
3.62k
        return false;
1892
        
1893
8.89k
      fAsShotWhiteXY.x = stream.TagValue_real64 (tagType);
1894
8.89k
      fAsShotWhiteXY.y = stream.TagValue_real64 (tagType);
1895
      
1896
      #if qDNGValidate
1897
      
1898
      if (gVerbose)
1899
        {
1900
        
1901
        printf ("AsShotWhiteXY: %0.4f %0.4f\n",
1902
            fAsShotWhiteXY.x,
1903
            fAsShotWhiteXY.y);
1904
            
1905
        }
1906
        
1907
      #endif
1908
        
1909
8.89k
      break;
1910
      
1911
12.5k
      }
1912
      
1913
19.7k
    case tcBaselineExposure:
1914
19.7k
      {
1915
      
1916
19.7k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
1917
      
1918
19.7k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
1919
      
1920
19.7k
      fBaselineExposure = stream.TagValue_srational (tagType);
1921
      
1922
19.7k
      fHasBaselineExposure = true;
1923
      
1924
      #if qDNGValidate
1925
      
1926
      if (gVerbose)
1927
        {
1928
        
1929
        printf ("BaselineExposure: %+0.2f\n",
1930
              fBaselineExposure.As_real64 ());
1931
              
1932
        }
1933
        
1934
      #endif
1935
        
1936
19.7k
      break;
1937
      
1938
12.5k
      }
1939
      
1940
14.4k
    case tcBaselineNoise:
1941
14.4k
      {
1942
      
1943
14.4k
      CheckTagType (parentCode, tagCode, tagType, ttRational);
1944
      
1945
14.4k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
1946
      
1947
14.4k
      fBaselineNoise = stream.TagValue_urational (tagType);
1948
      
1949
      #if qDNGValidate
1950
1951
      if (gVerbose)
1952
        {
1953
        
1954
        printf ("BaselineNoise: %0.2f\n",
1955
            fBaselineNoise.As_real64 ());
1956
            
1957
        }
1958
        
1959
      #endif
1960
        
1961
14.4k
      break;
1962
      
1963
12.5k
      }
1964
      
1965
3.23k
    case tcNoiseReductionApplied:
1966
3.23k
      {
1967
      
1968
3.23k
      if (!CheckTagType (parentCode, tagCode, tagType, ttRational))
1969
230
        return false;
1970
      
1971
3.00k
      if (!CheckTagCount (parentCode, tagCode, tagCount, 1))
1972
48
        return false;
1973
      
1974
2.95k
      fNoiseReductionApplied = stream.TagValue_urational (tagType);
1975
      
1976
      #if qDNGValidate
1977
1978
      if (gVerbose)
1979
        {
1980
        
1981
        printf ("NoiseReductionApplied: %u/%u\n",
1982
            (unsigned) fNoiseReductionApplied.n,
1983
            (unsigned) fNoiseReductionApplied.d);
1984
            
1985
        }
1986
        
1987
      #endif
1988
        
1989
2.95k
      break;
1990
      
1991
3.00k
      }
1992
1993
3.36k
    case tcNoiseProfile:
1994
3.36k
      {
1995
1996
3.36k
      if (!CheckTagType (parentCode, tagCode, tagType, ttDouble))
1997
300
        return false;
1998
1999
      // Must be an even, positive number of doubles in a noise profile.
2000
      
2001
3.06k
      if (!tagCount || (tagCount & 1))
2002
492
        return false;
2003
2004
      // Determine number of planes (i.e., half the number of doubles).
2005
2006
2.57k
      const uint32 numPlanes = Pin_uint32 (0, 
2007
2.57k
                         tagCount >> 1, 
2008
2.57k
                         kMaxColorPlanes);
2009
2010
      // Parse the noise function parameters.
2011
2012
2.57k
      dng_std_vector<dng_noise_function> noiseFunctions;
2013
2014
12.2k
      for (uint32 i = 0; i < numPlanes; i++)
2015
9.72k
        {
2016
2017
9.72k
        const real64 scale  = stream.TagValue_real64 (tagType);
2018
9.72k
        const real64 offset = stream.TagValue_real64 (tagType);
2019
2020
9.72k
        noiseFunctions.push_back (dng_noise_function (scale, offset));
2021
2022
9.72k
        }
2023
2024
      // Store the noise profile.
2025
2026
2.57k
      fNoiseProfile = dng_noise_profile (noiseFunctions);
2027
2028
      // Debug.
2029
2030
      #if qDNGValidate
2031
2032
      if (gVerbose)
2033
        {
2034
        
2035
        printf ("NoiseProfile:\n");
2036
        
2037
        printf ("  Planes: %u\n", (unsigned) numPlanes);
2038
          
2039
        for (uint32 plane = 0; plane < numPlanes; plane++)
2040
          {
2041
2042
          printf ("  Noise function for plane %u: scale = %.8lf, offset = %.8lf\n",
2043
              (unsigned) plane,
2044
              noiseFunctions [plane].Scale  (),
2045
              noiseFunctions [plane].Offset ());
2046
2047
          }
2048
        
2049
        }
2050
2051
      #endif
2052
      
2053
2.57k
      break;
2054
      
2055
3.06k
      }
2056
      
2057
16.1k
    case tcBaselineSharpness:
2058
16.1k
      {
2059
      
2060
16.1k
      CheckTagType (parentCode, tagCode, tagType, ttRational);
2061
      
2062
16.1k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
2063
      
2064
16.1k
      fBaselineSharpness = stream.TagValue_urational (tagType);
2065
      
2066
      #if qDNGValidate
2067
2068
      if (gVerbose)
2069
        {
2070
        
2071
        printf ("BaselineSharpness: %0.2f\n",
2072
              fBaselineSharpness.As_real64 ());
2073
        
2074
        }
2075
        
2076
      #endif
2077
        
2078
16.1k
      break;
2079
      
2080
3.06k
      }
2081
      
2082
12.6k
    case tcLinearResponseLimit:
2083
12.6k
      {
2084
      
2085
12.6k
      CheckTagType (parentCode, tagCode, tagType, ttRational);
2086
      
2087
12.6k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
2088
      
2089
12.6k
      fLinearResponseLimit = stream.TagValue_urational (tagType);
2090
      
2091
      #if qDNGValidate
2092
2093
      if (gVerbose)
2094
        {
2095
        
2096
        printf ("LinearResponseLimit: %0.2f\n",
2097
            fLinearResponseLimit.As_real64 ());
2098
        
2099
        }
2100
        
2101
      #endif
2102
        
2103
12.6k
      break;
2104
      
2105
3.06k
      }
2106
      
2107
4.13k
    case tcShadowScale:
2108
4.13k
      {
2109
      
2110
4.13k
      CheckTagType (parentCode, tagCode, tagType, ttRational);
2111
      
2112
4.13k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
2113
      
2114
4.13k
      fShadowScale = stream.TagValue_urational (tagType);
2115
      
2116
4.13k
      fHasShadowScale = true;
2117
      
2118
      #if qDNGValidate
2119
2120
      if (gVerbose)
2121
        {
2122
        
2123
        printf ("ShadowScale: %0.4f\n",
2124
            fShadowScale.As_real64 ());
2125
        
2126
        }
2127
        
2128
      #endif
2129
        
2130
4.13k
      break;
2131
      
2132
3.06k
      }
2133
      
2134
20.7k
    case tcDNGPrivateData:
2135
20.7k
      {
2136
      
2137
20.7k
      CheckTagType (parentCode, tagCode, tagType, ttByte);
2138
      
2139
20.7k
      fDNGPrivateDataCount  = tagCount;
2140
20.7k
      fDNGPrivateDataOffset = tagOffset;
2141
      
2142
      #if qDNGValidate
2143
2144
      if (gVerbose)
2145
        {
2146
        
2147
        printf ("DNGPrivateData: Count = %u, Offset = %u\n",
2148
            (unsigned) fDNGPrivateDataCount,
2149
            (unsigned) fDNGPrivateDataOffset);
2150
            
2151
        DumpHexAscii (stream, tagCount);
2152
        
2153
        }
2154
        
2155
      #endif
2156
      
2157
20.7k
      break;
2158
      
2159
3.06k
      }
2160
      
2161
2.30k
    case tcMakerNoteSafety:
2162
2.30k
      {
2163
      
2164
2.30k
      CheckTagType (parentCode, tagCode, tagType, ttShort);
2165
      
2166
2.30k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
2167
      
2168
2.30k
      fMakerNoteSafety = stream.TagValue_uint32 (tagType);
2169
      
2170
      #if qDNGValidate
2171
2172
      if (gVerbose)
2173
        {
2174
        
2175
        printf ("MakerNoteSafety: %s\n",
2176
            LookupMakerNoteSafety (fMakerNoteSafety));
2177
        
2178
        }
2179
        
2180
      #endif
2181
      
2182
2.30k
      break;
2183
      
2184
3.06k
      }
2185
      
2186
6.46k
    case tcRawImageDigest:
2187
6.46k
      {
2188
      
2189
6.46k
      if (!CheckTagType (parentCode, tagCode, tagType, ttByte))
2190
341
        return false;
2191
        
2192
6.12k
      if (!CheckTagCount (parentCode, tagCode, tagCount, 16))
2193
513
        return false;
2194
        
2195
5.61k
      stream.Get (fRawImageDigest.data, 16);
2196
        
2197
      #if qDNGValidate
2198
2199
      if (gVerbose)
2200
        {
2201
        
2202
        printf ("RawImageDigest: ");
2203
        
2204
        DumpFingerprint (fRawImageDigest);
2205
                  
2206
        printf ("\n");
2207
        
2208
        }
2209
        
2210
      #endif
2211
        
2212
5.61k
      break;
2213
      
2214
6.12k
      }
2215
      
2216
470
    case tcNewRawImageDigest:
2217
470
      {
2218
      
2219
470
      if (!CheckTagType (parentCode, tagCode, tagType, ttByte))
2220
37
        return false;
2221
        
2222
433
      if (!CheckTagCount (parentCode, tagCode, tagCount, 16))
2223
188
        return false;
2224
        
2225
245
      stream.Get (fNewRawImageDigest.data, 16);
2226
      
2227
      #if qDNGValidate
2228
2229
      if (gVerbose)
2230
        {
2231
        
2232
        printf ("NewRawImageDigest: ");
2233
        
2234
        DumpFingerprint (fNewRawImageDigest);
2235
                  
2236
        printf ("\n");
2237
        
2238
        }
2239
        
2240
      #endif
2241
        
2242
245
      break;
2243
      
2244
433
      }
2245
      
2246
15.1k
    case tcRawDataUniqueID:
2247
15.1k
      {
2248
      
2249
15.1k
      if (!CheckTagType (parentCode, tagCode, tagType, ttByte))
2250
1.89k
        return false;
2251
        
2252
13.2k
      if (!CheckTagCount (parentCode, tagCode, tagCount, 16))
2253
4.57k
        return false;
2254
        
2255
8.64k
      stream.Get (fRawDataUniqueID.data, 16);
2256
        
2257
      #if qDNGValidate
2258
2259
      if (gVerbose)
2260
        {
2261
        
2262
        printf ("RawDataUniqueID: ");
2263
        
2264
        DumpFingerprint (fRawDataUniqueID);
2265
                  
2266
        printf ("\n");
2267
        
2268
        }
2269
        
2270
      #endif
2271
        
2272
8.64k
      break;
2273
      
2274
13.2k
      }
2275
      
2276
283
    case tcOriginalRawFileName:
2277
283
      {
2278
      
2279
283
      CheckTagType (parentCode, tagCode, tagType, ttAscii, ttByte);
2280
      
2281
283
      ParseStringTag (stream,
2282
283
              parentCode,
2283
283
              tagCode,
2284
283
              tagCount,
2285
283
              fOriginalRawFileName,
2286
283
              false);
2287
      
2288
      #if qDNGValidate
2289
2290
      if (gVerbose)
2291
        {
2292
        
2293
        printf ("OriginalRawFileName: ");
2294
        
2295
        DumpString (fOriginalRawFileName);
2296
        
2297
        printf ("\n");
2298
        
2299
        }
2300
        
2301
      #endif
2302
      
2303
283
      break;
2304
      
2305
13.2k
      }
2306
      
2307
350
    case tcOriginalRawFileData:
2308
350
      {
2309
      
2310
350
      CheckTagType (parentCode, tagCode, tagType, ttUndefined);
2311
      
2312
350
      fOriginalRawFileDataCount  = tagCount;
2313
350
      fOriginalRawFileDataOffset = tagOffset;
2314
      
2315
      #if qDNGValidate
2316
2317
      if (gVerbose)
2318
        {
2319
        
2320
        printf ("OriginalRawFileData: Count = %u, Offset = %u\n",
2321
            (unsigned) fOriginalRawFileDataCount,
2322
            (unsigned) fOriginalRawFileDataOffset);
2323
            
2324
        DumpHexAscii (stream, tagCount);
2325
        
2326
        }
2327
        
2328
      #endif
2329
        
2330
350
      break;
2331
      
2332
13.2k
      }
2333
      
2334
1.42k
    case tcOriginalRawFileDigest:
2335
1.42k
      {
2336
      
2337
1.42k
      if (!CheckTagType (parentCode, tagCode, tagType, ttByte))
2338
760
        return false;
2339
        
2340
669
      if (!CheckTagCount (parentCode, tagCode, tagCount, 16))
2341
489
        return false;
2342
        
2343
180
      stream.Get (fOriginalRawFileDigest.data, 16);
2344
        
2345
      #if qDNGValidate
2346
2347
      if (gVerbose)
2348
        {
2349
        
2350
        printf ("OriginalRawFileDigest: ");
2351
        
2352
        DumpFingerprint (fOriginalRawFileDigest);
2353
                  
2354
        printf ("\n");
2355
        
2356
        }
2357
        
2358
      #endif
2359
        
2360
180
      break;
2361
      
2362
669
      }
2363
      
2364
64
    case tcAsShotICCProfile:
2365
64
      {
2366
      
2367
64
      CheckTagType (parentCode, tagCode, tagType, ttUndefined);
2368
      
2369
64
      fAsShotICCProfileCount  = tagCount;
2370
64
      fAsShotICCProfileOffset = tagOffset;
2371
      
2372
      #if qDNGValidate
2373
2374
      if (gVerbose)
2375
        {
2376
        
2377
        printf ("AsShotICCProfile: Count = %u, Offset = %u\n",
2378
            (unsigned) fAsShotICCProfileCount,
2379
            (unsigned) fAsShotICCProfileOffset);
2380
            
2381
        DumpHexAscii (stream, tagCount);
2382
        
2383
        }
2384
        
2385
      #endif
2386
        
2387
64
      break;
2388
      
2389
669
      }
2390
      
2391
6.62k
    case tcAsShotPreProfileMatrix:
2392
6.62k
      {
2393
      
2394
6.62k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
2395
      
2396
6.62k
      if (!CheckColorImage (parentCode, tagCode, fCameraProfile.fColorPlanes))
2397
1.50k
        return false;
2398
        
2399
5.12k
      uint32 rows = fCameraProfile.fColorPlanes;
2400
      
2401
5.12k
      if (tagCount == fCameraProfile.fColorPlanes * 3)
2402
425
        {
2403
425
        rows = 3;
2404
425
        }
2405
      
2406
5.12k
      if (!ParseMatrixTag (stream,
2407
5.12k
                 parentCode,
2408
5.12k
                 tagCode,
2409
5.12k
                 tagType,
2410
5.12k
                 tagCount,
2411
5.12k
                 rows,
2412
5.12k
                 fCameraProfile.fColorPlanes,
2413
5.12k
                 fAsShotPreProfileMatrix))
2414
4.31k
        return false;
2415
        
2416
      #if qDNGValidate
2417
2418
      if (gVerbose)
2419
        {
2420
        
2421
        printf ("AsShotPreProfileMatrix:\n");
2422
        
2423
        DumpMatrix (fAsShotPreProfileMatrix);
2424
                
2425
        }
2426
        
2427
      #endif
2428
        
2429
811
      break;
2430
      
2431
5.12k
      }
2432
      
2433
811
    case tcCurrentICCProfile:
2434
619
      {
2435
      
2436
619
      CheckTagType (parentCode, tagCode, tagType, ttUndefined);
2437
      
2438
619
      fCurrentICCProfileCount  = tagCount;
2439
619
      fCurrentICCProfileOffset = tagOffset;
2440
      
2441
      #if qDNGValidate
2442
2443
      if (gVerbose)
2444
        {
2445
        
2446
        printf ("CurrentICCProfile: Count = %u, Offset = %u\n",
2447
            (unsigned) fCurrentICCProfileCount,
2448
            (unsigned) fCurrentICCProfileOffset);
2449
            
2450
        DumpHexAscii (stream, tagCount);
2451
        
2452
        }
2453
        
2454
      #endif
2455
        
2456
619
      break;
2457
      
2458
5.12k
      }
2459
      
2460
30.8k
    case tcCurrentPreProfileMatrix:
2461
30.8k
      {
2462
      
2463
30.8k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
2464
      
2465
30.8k
      if (!CheckColorImage (parentCode, tagCode, fCameraProfile.fColorPlanes))
2466
6.74k
        return false;
2467
        
2468
24.1k
      uint32 rows = fCameraProfile.fColorPlanes;
2469
      
2470
24.1k
      if (tagCount == fCameraProfile.fColorPlanes * 3)
2471
11.0k
        {
2472
11.0k
        rows = 3;
2473
11.0k
        }
2474
      
2475
24.1k
      if (!ParseMatrixTag (stream,
2476
24.1k
                 parentCode,
2477
24.1k
                 tagCode,
2478
24.1k
                 tagType,
2479
24.1k
                 tagCount,
2480
24.1k
                 rows,
2481
24.1k
                 fCameraProfile.fColorPlanes,
2482
24.1k
                 fCurrentPreProfileMatrix))
2483
13.0k
        return false;
2484
        
2485
      #if qDNGValidate
2486
2487
      if (gVerbose)
2488
        {
2489
        
2490
        printf ("CurrentPreProfileMatrix:\n");
2491
        
2492
        DumpMatrix (fCurrentPreProfileMatrix);
2493
                
2494
        }
2495
        
2496
      #endif
2497
        
2498
11.1k
      break;
2499
      
2500
24.1k
      }
2501
      
2502
19.1k
    case tcColorimetricReference:
2503
19.1k
      {
2504
      
2505
19.1k
      CheckTagType (parentCode, tagCode, tagType, ttShort);
2506
      
2507
19.1k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
2508
      
2509
19.1k
      fColorimetricReference = stream.TagValue_uint32 (tagType);
2510
      
2511
      #if qDNGValidate
2512
2513
      if (gVerbose)
2514
        {
2515
        
2516
        printf ("ColorimetricReference: %s\n",
2517
            LookupColorimetricReference (fColorimetricReference));
2518
        
2519
        }
2520
        
2521
      #endif
2522
        
2523
19.1k
      break;
2524
      
2525
24.1k
      }
2526
2527
8.74k
    case tcExtraCameraProfiles:
2528
8.74k
      {
2529
        
2530
8.74k
      CheckTagType (parentCode, tagCode, tagType, ttLong);
2531
      
2532
8.74k
      CheckTagCount (parentCode, tagCode, tagCount, 1, tagCount);
2533
      
2534
      #if qDNGValidate
2535
2536
      if (gVerbose)
2537
        {
2538
        
2539
        printf ("ExtraCameraProfiles: %u\n", (unsigned) tagCount);
2540
        
2541
        }
2542
        
2543
      #endif
2544
      
2545
8.74k
      fExtraCameraProfiles.reserve (tagCount);
2546
      
2547
6.83M
      for (uint32 index = 0; index < tagCount; index++)
2548
6.83M
        {
2549
        
2550
        #if qDNGValidate
2551
2552
        if (gVerbose)
2553
          {
2554
          
2555
          printf ("\nExtraCameraProfile [%u]:\n\n", (unsigned) index);
2556
          
2557
          }
2558
          
2559
        #endif
2560
      
2561
6.83M
        stream.SetReadPosition (tagOffset + index * 4);
2562
        
2563
6.83M
        uint32 profileOffset = stream.TagValue_uint32 (tagType);
2564
        
2565
6.83M
        dng_camera_profile_info profileInfo;
2566
        
2567
6.83M
        stream.SetReadPosition (profileOffset);
2568
        
2569
6.83M
        if (profileInfo.ParseExtended (stream))
2570
277k
          {
2571
          
2572
277k
          fExtraCameraProfiles.push_back (profileInfo);
2573
          
2574
277k
          }
2575
          
2576
6.55M
        else
2577
6.55M
          {
2578
          
2579
          #if qDNGValidate
2580
2581
          ReportWarning ("Unable to parse extra camera profile");
2582
          
2583
          #endif
2584
      
2585
6.55M
          }
2586
        
2587
6.83M
        }
2588
2589
      #if qDNGValidate
2590
2591
      if (gVerbose)
2592
        {
2593
        
2594
        printf ("\nDone with ExtraCameraProfiles\n\n");
2595
        
2596
        }
2597
        
2598
      #endif
2599
      
2600
8.74k
      break;
2601
      
2602
24.1k
      }
2603
      
2604
1.38k
    case tcAsShotProfileName:
2605
1.38k
      {
2606
            
2607
1.38k
      CheckTagType (parentCode, tagCode, tagType, ttAscii, ttByte);
2608
      
2609
1.38k
      ParseStringTag (stream,
2610
1.38k
              parentCode,
2611
1.38k
              tagCode,
2612
1.38k
              tagCount,
2613
1.38k
              fAsShotProfileName,
2614
1.38k
              false);
2615
      
2616
      #if qDNGValidate
2617
      
2618
      if (gVerbose)
2619
        {
2620
        
2621
        printf ("AsShotProfileName: ");
2622
        
2623
        DumpString (fAsShotProfileName);
2624
        
2625
        printf ("\n");
2626
        
2627
        }
2628
        
2629
      #endif
2630
      
2631
1.38k
      break;
2632
      
2633
24.1k
      }
2634
2635
925
    case tcOriginalDefaultFinalSize:
2636
925
      {
2637
      
2638
925
      CheckTagType (parentCode, tagCode, tagType, ttShort, ttLong);
2639
      
2640
925
      if (!CheckTagCount (parentCode, tagCode, tagCount, 2))
2641
107
        return false;
2642
        
2643
818
      fOriginalDefaultFinalSize.h = stream.TagValue_int32 (tagType);
2644
818
      fOriginalDefaultFinalSize.v = stream.TagValue_int32 (tagType);
2645
      
2646
      #if qDNGValidate
2647
        
2648
      if (gVerbose)
2649
        {
2650
        
2651
        printf ("OriginalDefaultFinalSize: H = %d V = %d\n",
2652
            (int) fOriginalDefaultFinalSize.h,
2653
            (int) fOriginalDefaultFinalSize.v);
2654
            
2655
        }
2656
        
2657
      #endif
2658
      
2659
818
      break;
2660
      
2661
925
      }
2662
        
2663
622
    case tcOriginalBestQualityFinalSize:
2664
622
      {
2665
      
2666
622
      CheckTagType (parentCode, tagCode, tagType, ttShort, ttLong);
2667
      
2668
622
      if (!CheckTagCount (parentCode, tagCode, tagCount, 2))
2669
170
        return false;
2670
        
2671
452
      fOriginalBestQualityFinalSize.h = stream.TagValue_int32 (tagType);
2672
452
      fOriginalBestQualityFinalSize.v = stream.TagValue_int32 (tagType);
2673
      
2674
      #if qDNGValidate
2675
        
2676
      if (gVerbose)
2677
        {
2678
        
2679
        printf ("OriginalBestQualityFinalSize: H = %d V = %d\n",
2680
            (int) fOriginalBestQualityFinalSize.h,
2681
            (int) fOriginalBestQualityFinalSize.v);
2682
            
2683
        }
2684
        
2685
      #endif
2686
      
2687
452
      break;
2688
      
2689
622
      }
2690
        
2691
774
    case tcOriginalDefaultCropSize:
2692
774
      {
2693
      
2694
774
      CheckTagType (parentCode, tagCode, tagType, ttShort, ttLong, ttRational);
2695
      
2696
774
      if (!CheckTagCount (parentCode, tagCode, tagCount, 2))
2697
116
        return false;
2698
        
2699
658
      fOriginalDefaultCropSizeH = stream.TagValue_urational (tagType);
2700
658
      fOriginalDefaultCropSizeV = stream.TagValue_urational (tagType);
2701
      
2702
      #if qDNGValidate
2703
        
2704
      if (gVerbose)
2705
        {
2706
        
2707
        printf ("OriginalDefaultCropSize: H = %0.2f V = %0.2f\n",
2708
            fOriginalDefaultCropSizeH.As_real64 (),
2709
            fOriginalDefaultCropSizeV.As_real64 ());
2710
            
2711
        }
2712
        
2713
      #endif
2714
      
2715
658
      break;
2716
      
2717
774
      }
2718
      
2719
345k
    default:
2720
345k
      {
2721
      
2722
      // The main camera profile tags also appear in IFD 0
2723
      
2724
345k
      return fCameraProfile.ParseTag (stream,
2725
345k
                      parentCode,
2726
345k
                      tagCode,
2727
345k
                      tagType,
2728
345k
                      tagCount,
2729
345k
                      tagOffset);
2730
      
2731
774
      }
2732
      
2733
767k
    }
2734
2735
352k
  return true;
2736
  
2737
767k
  }
2738
  
2739
/*****************************************************************************/
2740
2741
// Parses tags that should only appear in IFD 0 or EXIF IFD.
2742
2743
bool dng_shared::Parse_ifd0_exif (dng_stream &stream,
2744
                  dng_exif & /* exif */,
2745
                      uint32 parentCode,
2746
                      uint32 tagCode,
2747
                      uint32 tagType,
2748
                      uint32 tagCount,
2749
                      uint64 tagOffset)
2750
415k
  {
2751
  
2752
415k
  switch (tagCode)
2753
415k
    {
2754
    
2755
11.3k
    case tcMakerNote:
2756
11.3k
      {
2757
      
2758
11.3k
      CheckTagType (parentCode, tagCode, tagType, ttUndefined);
2759
      
2760
11.3k
      fMakerNoteCount  = tagCount;
2761
11.3k
      fMakerNoteOffset = tagOffset;
2762
      
2763
      #if qDNGValidate
2764
2765
      if (gVerbose)
2766
        {
2767
        
2768
        printf ("MakerNote: Count = %u, Offset = %u\n",
2769
            (unsigned) fMakerNoteCount,
2770
            (unsigned) fMakerNoteOffset);
2771
            
2772
        DumpHexAscii (stream, tagCount);
2773
        
2774
        }
2775
        
2776
      #endif
2777
        
2778
11.3k
      break;
2779
      
2780
0
      }
2781
      
2782
2.89k
    case tcInteroperabilityIFD:
2783
2.89k
      {
2784
      
2785
2.89k
      CheckTagType (parentCode, tagCode, tagType, ttLong, ttIFD);
2786
      
2787
2.89k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
2788
      
2789
2.89k
      fInteroperabilityIFD = stream.TagValue_uint32 (tagType);
2790
      
2791
      #if qDNGValidate
2792
2793
      if (gVerbose)
2794
        {
2795
        printf ("InteroperabilityIFD: %u\n", (unsigned) fInteroperabilityIFD);
2796
        }
2797
        
2798
      #endif
2799
      
2800
2.89k
      break;
2801
      
2802
0
      }
2803
      
2804
401k
    default:
2805
401k
      {
2806
      
2807
401k
      return false;
2808
      
2809
0
      }
2810
      
2811
415k
    }
2812
    
2813
14.2k
  return true;
2814
  
2815
415k
  }
2816
                 
2817
/*****************************************************************************/
2818
2819
void dng_shared::PostParse (dng_host & /* host */,
2820
              dng_exif & /* exif */)
2821
68.7k
  {
2822
  
2823
  // Fill in default values for DNG images.
2824
  
2825
68.7k
  if (fDNGVersion != 0)
2826
51.7k
    {
2827
    
2828
    // Support for DNG versions before 1.0.0.0.
2829
    
2830
51.7k
    if (fDNGVersion < dngVersion_1_0_0_0)
2831
21.3k
      {
2832
      
2833
      #if qDNGValidate
2834
      
2835
      ReportWarning ("DNGVersion less than 1.0.0.0");
2836
             
2837
      #endif
2838
      
2839
      // The CalibrationIlluminant tags were added just before
2840
      // DNG version 1.0.0.0, and were hardcoded before that.
2841
      
2842
21.3k
      fCameraProfile.fCalibrationIlluminant1 = lsStandardLightA;
2843
21.3k
      fCameraProfile.fCalibrationIlluminant2 = lsD65;
2844
      
2845
21.3k
      fDNGVersion = dngVersion_1_0_0_0;
2846
2847
21.3k
      }
2848
    
2849
    // Default value for DNGBackwardVersion tag.
2850
    
2851
51.7k
    if (fDNGBackwardVersion == 0)
2852
38.3k
      {
2853
      
2854
38.3k
      fDNGBackwardVersion = fDNGVersion & 0xFFFF0000;
2855
      
2856
38.3k
      }
2857
  
2858
    // Check DNGBackwardVersion value.
2859
    
2860
51.7k
    if (fDNGBackwardVersion < dngVersion_1_0_0_0)
2861
1.12k
      {
2862
      
2863
      #if qDNGValidate
2864
      
2865
      ReportWarning ("DNGBackwardVersion less than 1.0.0.0");
2866
             
2867
      #endif
2868
      
2869
1.12k
      fDNGBackwardVersion = dngVersion_1_0_0_0;
2870
        
2871
1.12k
      }
2872
2873
51.7k
    if (fDNGBackwardVersion > fDNGVersion)
2874
5.49k
      {
2875
      
2876
      #if qDNGValidate
2877
      
2878
      ReportWarning ("DNGBackwardVersion > DNGVersion");
2879
             
2880
      #endif
2881
      
2882
5.49k
      fDNGBackwardVersion = fDNGVersion;
2883
          
2884
5.49k
      }
2885
2886
    // Check UniqueCameraModel.
2887
    
2888
51.7k
    if (fUniqueCameraModel.IsEmpty ())
2889
48.8k
      {
2890
      
2891
      #if qDNGValidate
2892
      
2893
      ReportWarning ("Missing or invalid UniqueCameraModel");
2894
             
2895
      #endif
2896
      
2897
48.8k
      fUniqueCameraModel.Set ("Digital Negative");
2898
      
2899
48.8k
      }
2900
    
2901
    // If we don't know the color depth yet, it must be a monochrome DNG.
2902
    
2903
51.7k
    if (fCameraProfile.fColorPlanes == 0)
2904
35.2k
      {
2905
      
2906
35.2k
      fCameraProfile.fColorPlanes = 1;
2907
      
2908
35.2k
      }
2909
      
2910
    // Check color info.
2911
    
2912
51.7k
    if (fCameraProfile.fColorPlanes > 1)
2913
16.4k
      {
2914
      
2915
      // Check illuminant pair.
2916
      
2917
16.4k
      if (fCameraProfile.fColorMatrix2.NotEmpty ())
2918
786
        {
2919
        
2920
786
        if (fCameraProfile.fCalibrationIlluminant1 == lsUnknown ||
2921
688
          (fCameraProfile.fCalibrationIlluminant2 == lsUnknown ||
2922
616
          (fCameraProfile.fCalibrationIlluminant1 == fCameraProfile.fCalibrationIlluminant2)))
2923
178
          {
2924
            
2925
          #if qDNGValidate
2926
      
2927
          ReportWarning ("Invalid CalibrationIlluminant pair");
2928
                 
2929
          #endif
2930
                 
2931
178
          fCameraProfile.fColorMatrix2 = dng_matrix ();
2932
        
2933
178
          }
2934
2935
786
        }
2936
        
2937
      // If the colorimetric reference is the ICC profile PCS, then the
2938
      // data must already be white balanced.  The "AsShotWhiteXY" is required
2939
      // to be the ICC Profile PCS white point.
2940
      
2941
16.4k
      if (fColorimetricReference == crICCProfilePCS)
2942
3.57k
        {
2943
        
2944
3.57k
        if (fAsShotNeutral.NotEmpty ())
2945
145
          {
2946
          
2947
          #if qDNGValidate
2948
        
2949
          ReportWarning ("AsShotNeutral not allowed for this "
2950
                   "ColorimetricReference value");
2951
                 
2952
          #endif
2953
          
2954
145
          fAsShotNeutral.Clear ();
2955
          
2956
145
          }
2957
          
2958
3.57k
        dng_xy_coord pcs = PCStoXY ();
2959
        
2960
        #if qDNGValidate
2961
          
2962
        if (fAsShotWhiteXY.IsValid ())
2963
          {
2964
          
2965
          if (Abs_real64 (fAsShotWhiteXY.x - pcs.x) > 0.01 ||
2966
            Abs_real64 (fAsShotWhiteXY.y - pcs.y) > 0.01)
2967
            {
2968
            
2969
            ReportWarning ("AsShotWhiteXY does not match the ICC Profile PCS");
2970
2971
            }
2972
            
2973
          }
2974
                   
2975
        #endif
2976
        
2977
3.57k
        fAsShotWhiteXY = pcs;
2978
2979
3.57k
        }
2980
        
2981
12.9k
      else
2982
12.9k
        {
2983
        
2984
        // Warn if both AsShotNeutral and AsShotWhiteXY are specified.
2985
          
2986
12.9k
        if (fAsShotNeutral.NotEmpty () && fAsShotWhiteXY.IsValid ())
2987
89
          {
2988
            
2989
          #if qDNGValidate
2990
        
2991
          ReportWarning ("Both AsShotNeutral and AsShotWhiteXY included");
2992
                 
2993
          #endif
2994
          
2995
89
          fAsShotWhiteXY = dng_xy_coord ();
2996
                   
2997
89
          }
2998
          
2999
        // Warn if neither AsShotNeutral nor AsShotWhiteXY are specified.
3000
          
3001
        #if qDNGValidate
3002
        
3003
        if (fAsShotNeutral.IsEmpty () && !fAsShotWhiteXY.IsValid ())
3004
          {
3005
          
3006
          ReportWarning ("Neither AsShotNeutral nor AsShotWhiteXY included",
3007
                   "legal but not recommended");
3008
                 
3009
          }
3010
              
3011
        #endif
3012
          
3013
12.9k
        }
3014
        
3015
      // Default values of calibration signatures are required for legacy
3016
      // compatiblity.
3017
3018
16.4k
      if (fCameraProfile.fCalibrationIlluminant1 == lsStandardLightA &&
3019
1.04k
        fCameraProfile.fCalibrationIlluminant2 == lsD65            &&
3020
878
        fCameraCalibration1.Rows () == fCameraProfile.fColorPlanes &&
3021
187
        fCameraCalibration1.Cols () == fCameraProfile.fColorPlanes &&
3022
187
        fCameraCalibration2.Rows () == fCameraProfile.fColorPlanes &&
3023
81
        fCameraCalibration2.Cols () == fCameraProfile.fColorPlanes &&
3024
81
        fCameraCalibrationSignature.IsEmpty ()                     &&
3025
73
        fCameraProfile.fProfileCalibrationSignature.IsEmpty ()     )
3026
64
        {
3027
3028
64
        fCameraCalibrationSignature.Set (kAdobeCalibrationSignature);
3029
          
3030
64
        fCameraProfile.fProfileCalibrationSignature.Set (kAdobeCalibrationSignature);
3031
3032
64
        }
3033
3034
16.4k
      }
3035
      
3036
    // Check BaselineNoise.
3037
3038
51.7k
    if (fBaselineNoise.As_real64 () <= 0.0)
3039
343
      {
3040
      
3041
      #if qDNGValidate
3042
      
3043
      ReportWarning ("Invalid BaselineNoise");
3044
             
3045
      #endif
3046
      
3047
343
      fBaselineNoise = dng_urational (1, 1);
3048
               
3049
343
      }
3050
      
3051
    // Check BaselineSharpness.
3052
    
3053
51.7k
    if (fBaselineSharpness.As_real64 () <= 0.0)
3054
836
      {
3055
      
3056
      #if qDNGValidate
3057
      
3058
      ReportWarning ("Invalid BaselineSharpness");
3059
             
3060
      #endif
3061
      
3062
836
      fBaselineSharpness = dng_urational (1, 1);
3063
               
3064
836
      }
3065
3066
    // Check NoiseProfile.
3067
3068
51.7k
    if (!fNoiseProfile.IsValid () && fNoiseProfile.NumFunctions () != 0)
3069
162
      {
3070
      
3071
      #if qDNGValidate
3072
      
3073
      ReportWarning ("Invalid NoiseProfile");
3074
             
3075
      #endif
3076
      
3077
162
      fNoiseProfile = dng_noise_profile ();
3078
               
3079
162
      }
3080
      
3081
    // Check LinearResponseLimit.
3082
    
3083
51.7k
    if (fLinearResponseLimit.As_real64 () < 0.5 ||
3084
49.2k
      fLinearResponseLimit.As_real64 () > 1.0)
3085
3.43k
      {
3086
      
3087
      #if qDNGValidate
3088
      
3089
      ReportWarning ("Invalid LinearResponseLimit");
3090
             
3091
      #endif
3092
      
3093
3.43k
      fLinearResponseLimit = dng_urational (1, 1);
3094
               
3095
3.43k
      }
3096
      
3097
    // Check ShadowScale.
3098
    
3099
51.7k
    if (fShadowScale.As_real64 () <= 0.0)
3100
764
      {
3101
      
3102
      #if qDNGValidate
3103
      
3104
      ReportWarning ("Invalid ShadowScale");
3105
             
3106
      #endif
3107
               
3108
764
      fShadowScale = dng_urational (1, 1);
3109
               
3110
764
      }
3111
    
3112
51.7k
    }
3113
  
3114
68.7k
  }
3115
                 
3116
/*****************************************************************************/
3117
3118
bool dng_shared::IsValidDNG ()
3119
66.2k
  {
3120
  
3121
  // Check DNGVersion value.
3122
  
3123
66.2k
  if (fDNGVersion < dngVersion_1_0_0_0)
3124
16.7k
    {
3125
    
3126
    #if qDNGValidate
3127
    
3128
    if (fDNGVersion != dngVersion_None)
3129
      {
3130
3131
      ReportError ("Invalid DNGVersion");
3132
3133
      }
3134
           
3135
    #if qDNGValidateTarget
3136
    
3137
    else
3138
      {
3139
3140
      ReportError ("Missing DNGVersion");
3141
3142
      }
3143
      
3144
    #endif
3145
    
3146
    #endif
3147
           
3148
16.7k
    return false;
3149
      
3150
16.7k
    }
3151
    
3152
  // Check DNGBackwardVersion value.
3153
  
3154
49.4k
  if (fDNGBackwardVersion > dngVersion_Current)
3155
4.05k
    {
3156
    
3157
    #if qDNGValidate
3158
    
3159
    ReportError ("DNGBackwardVersion (or DNGVersion) is too high");
3160
    
3161
    #endif
3162
    
3163
4.05k
    ThrowUnsupportedDNG ();
3164
      
3165
4.05k
    }
3166
    
3167
  // Check color transform info.
3168
  
3169
49.4k
  if (fCameraProfile.fColorPlanes > 1)
3170
14.8k
    {
3171
    
3172
    // CameraCalibration1 is optional, but it must be valid if present.
3173
    
3174
14.8k
    if (fCameraCalibration1.Cols () != 0 ||
3175
14.6k
      fCameraCalibration1.Rows () != 0)
3176
249
      {
3177
      
3178
249
      if (fCameraCalibration1.Cols () != fCameraProfile.fColorPlanes ||
3179
249
        fCameraCalibration1.Rows () != fCameraProfile.fColorPlanes)
3180
0
        {
3181
        
3182
        #if qDNGValidate
3183
    
3184
        ReportError ("CameraCalibration1 is wrong size");
3185
               
3186
        #endif
3187
3188
0
        return false;
3189
        
3190
0
        }
3191
        
3192
      // Make sure it is invertable.
3193
      
3194
249
      try
3195
249
        {
3196
        
3197
249
        (void) Invert (fCameraCalibration1);
3198
        
3199
249
        }
3200
        
3201
249
      catch (...)
3202
249
        {
3203
        
3204
        #if qDNGValidate
3205
    
3206
        ReportError ("CameraCalibration1 is not invertable");
3207
               
3208
        #endif
3209
               
3210
14
        return false;
3211
      
3212
14
        }
3213
        
3214
249
      }
3215
    
3216
    // CameraCalibration2 is optional, but it must be valid if present.
3217
    
3218
14.8k
    if (fCameraCalibration2.Cols () != 0 ||
3219
14.6k
      fCameraCalibration2.Rows () != 0)
3220
192
      {
3221
      
3222
192
      if (fCameraCalibration2.Cols () != fCameraProfile.fColorPlanes ||
3223
192
        fCameraCalibration2.Rows () != fCameraProfile.fColorPlanes)
3224
0
        {
3225
        
3226
        #if qDNGValidate
3227
    
3228
        ReportError ("CameraCalibration2 is wrong size");
3229
               
3230
        #endif
3231
3232
0
        return false;
3233
        
3234
0
        }
3235
      
3236
      // Make sure it is invertable.
3237
      
3238
192
      try
3239
192
        {
3240
        
3241
192
        (void) Invert (fCameraCalibration2);
3242
        
3243
192
        }
3244
        
3245
192
      catch (...)
3246
192
        {
3247
        
3248
        #if qDNGValidate
3249
  
3250
        ReportError ("CameraCalibration2 is not invertable");
3251
               
3252
        #endif
3253
               
3254
8
        return false;
3255
      
3256
8
        }
3257
          
3258
192
      }
3259
      
3260
    // Check analog balance
3261
    
3262
14.8k
    dng_matrix analogBalance;
3263
    
3264
14.8k
    if (fAnalogBalance.NotEmpty ())
3265
4.89k
      {
3266
      
3267
4.89k
      analogBalance = fAnalogBalance.AsDiagonal ();
3268
      
3269
4.89k
      try
3270
4.89k
        {
3271
        
3272
4.89k
        (void) Invert (analogBalance);
3273
        
3274
4.89k
        }
3275
        
3276
4.89k
      catch (...)
3277
4.89k
        {
3278
        
3279
        #if qDNGValidate
3280
    
3281
        ReportError ("AnalogBalance is not invertable");
3282
               
3283
        #endif
3284
               
3285
53
        return false;
3286
      
3287
53
        }
3288
        
3289
4.89k
      }
3290
        
3291
14.8k
    }
3292
      
3293
49.4k
  return true;
3294
  
3295
49.4k
  }
3296
  
3297
/*****************************************************************************/