Coverage Report

Created: 2026-02-26 06:56

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.25M
  : fBigEndian (false)
34
  
35
6.25M
  , fColorPlanes (0)
36
  
37
6.25M
  , fCalibrationIlluminant1 (lsUnknown)
38
6.25M
  , fCalibrationIlluminant2 (lsUnknown)
39
40
6.25M
  , fColorMatrix1 ()
41
6.25M
  , fColorMatrix2 ()
42
    
43
6.25M
  , fForwardMatrix1 ()
44
6.25M
  , fForwardMatrix2 ()
45
    
46
6.25M
  , fReductionMatrix1 ()
47
6.25M
  , fReductionMatrix2 ()
48
    
49
6.25M
  , fProfileCalibrationSignature ()
50
  
51
6.25M
  , fProfileName ()
52
  
53
6.25M
  , fProfileCopyright ()
54
55
6.25M
  , fEmbedPolicy (pepAllowCopying)
56
  
57
6.25M
  , fProfileHues (0)
58
6.25M
  , fProfileSats (0)
59
6.25M
  , fProfileVals (0)
60
61
6.25M
  , fHueSatDeltas1Offset (0)
62
6.25M
  , fHueSatDeltas1Count  (0)
63
64
6.25M
  , fHueSatDeltas2Offset (0)
65
6.25M
  , fHueSatDeltas2Count  (0)
66
67
6.25M
  , fHueSatMapEncoding (encoding_Linear)
68
  
69
6.25M
  , fLookTableHues (0)
70
6.25M
  , fLookTableSats (0)
71
6.25M
  , fLookTableVals (0)
72
    
73
6.25M
  , fLookTableOffset (0)
74
6.25M
  , fLookTableCount  (0)
75
76
6.25M
  , fLookTableEncoding (encoding_Linear)
77
78
6.25M
  , fBaselineExposureOffset (0, 100)
79
  
80
6.25M
  , fDefaultBlackRender (defaultBlackRender_Auto)
81
82
6.25M
  , fToneCurveOffset     (0)
83
6.25M
  , fToneCurveCount      (0)
84
  
85
6.25M
  , fUniqueCameraModel ()
86
87
6.25M
  {
88
  
89
6.25M
  }
90
                   
91
/*****************************************************************************/
92
93
dng_camera_profile_info::~dng_camera_profile_info ()
94
6.84M
  {
95
  
96
6.84M
  }
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.9M
  {
107
108
44.9M
  switch (tagCode)
109
44.9M
    {
110
      
111
108k
    case tcCalibrationIlluminant1:
112
108k
      {
113
      
114
108k
      CheckTagType (parentCode, tagCode, tagType, ttShort);
115
      
116
108k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
117
      
118
108k
      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
108k
      break;
133
      
134
0
      }
135
      
136
6.74k
    case tcCalibrationIlluminant2:
137
6.74k
      {
138
      
139
6.74k
      CheckTagType (parentCode, tagCode, tagType, ttShort);
140
      
141
6.74k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
142
      
143
6.74k
      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
6.74k
      break;
158
      
159
0
      }
160
161
122k
    case tcColorMatrix1:
162
122k
      {
163
      
164
122k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
165
      
166
122k
      if (fColorPlanes == 0)
167
93.2k
        {
168
      
169
93.2k
        fColorPlanes = Pin_uint32 (0, tagCount / 3, kMaxColorPlanes);
170
                  
171
93.2k
        }
172
        
173
122k
      if (!CheckColorImage (parentCode, tagCode, fColorPlanes))
174
3.05k
        return false;
175
        
176
119k
      if (!ParseMatrixTag (stream,
177
119k
                 parentCode,
178
119k
                 tagCode,
179
119k
                 tagType,
180
119k
                 tagCount,
181
119k
                 fColorPlanes,
182
119k
                 3,
183
119k
                 fColorMatrix1))
184
15.3k
        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
104k
      break;
200
      
201
119k
      }
202
203
299k
    case tcColorMatrix2:
204
299k
      {
205
      
206
299k
      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
299k
      bool hasselbladHack = (fColorPlanes == 0);
212
      
213
299k
      if (hasselbladHack)
214
99.3k
        {
215
        
216
99.3k
        fColorPlanes = Pin_uint32 (0, tagCount / 3, kMaxColorPlanes);
217
        
218
        #if qDNGValidate
219
        
220
        ReportWarning ("ColorMatrix2 without ColorMatrix1");
221
        
222
        #endif
223
      
224
99.3k
        }
225
      
226
299k
      if (!CheckColorImage (parentCode, tagCode, fColorPlanes))
227
2.71k
        return false;
228
        
229
296k
      if (!ParseMatrixTag (stream,
230
296k
                 parentCode,
231
296k
                 tagCode,
232
296k
                 tagType,
233
296k
                 tagCount,
234
296k
                 fColorPlanes,
235
296k
                 3,
236
296k
                 fColorMatrix2))
237
275k
        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
20.7k
      if (hasselbladHack)
253
2.98k
        {
254
        
255
2.98k
        fColorMatrix1 = fColorMatrix2;
256
        
257
2.98k
        fColorMatrix2 = dng_matrix ();
258
        
259
2.98k
        }
260
        
261
20.7k
      break;
262
      
263
296k
      }
264
265
57.4k
    case tcForwardMatrix1:
266
57.4k
      {
267
      
268
57.4k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
269
      
270
57.4k
      if (!CheckColorImage (parentCode, tagCode, fColorPlanes))
271
45.6k
        return false;
272
        
273
11.8k
      if (!ParseMatrixTag (stream,
274
11.8k
                 parentCode,
275
11.8k
                 tagCode,
276
11.8k
                 tagType,
277
11.8k
                 tagCount,
278
11.8k
                 3,
279
11.8k
                 fColorPlanes,
280
11.8k
                 fForwardMatrix1))
281
3.86k
        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
8.02k
      break;
297
      
298
11.8k
      }
299
300
26.4k
    case tcForwardMatrix2:
301
26.4k
      {
302
      
303
26.4k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
304
      
305
26.4k
      if (!CheckColorImage (parentCode, tagCode, fColorPlanes))
306
23.9k
        return false;
307
        
308
2.50k
      if (!ParseMatrixTag (stream,
309
2.50k
                 parentCode,
310
2.50k
                 tagCode,
311
2.50k
                 tagType,
312
2.50k
                 tagCount,
313
2.50k
                 3,
314
2.50k
                 fColorPlanes,
315
2.50k
                 fForwardMatrix2))
316
652
        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
1.84k
      break;
332
      
333
2.50k
      }
334
335
183k
    case tcReductionMatrix1:
336
183k
      {
337
      
338
183k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
339
      
340
183k
      if (!CheckColorImage (parentCode, tagCode, fColorPlanes))
341
47.9k
        return false;
342
        
343
135k
      if (!ParseMatrixTag (stream,
344
135k
                 parentCode,
345
135k
                 tagCode,
346
135k
                 tagType,
347
135k
                 tagCount,
348
135k
                 3,
349
135k
                 fColorPlanes,
350
135k
                 fReductionMatrix1))
351
133k
        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
1.58k
      break;
367
      
368
135k
      }
369
370
301k
    case tcReductionMatrix2:
371
301k
      {
372
      
373
301k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
374
      
375
301k
      if (!CheckColorImage (parentCode, tagCode, fColorPlanes))
376
288k
        return false;
377
        
378
13.7k
      if (!ParseMatrixTag (stream,
379
13.7k
                 parentCode,
380
13.7k
                 tagCode,
381
13.7k
                 tagType,
382
13.7k
                 tagCount,
383
13.7k
                 3,
384
13.7k
                 fColorPlanes,
385
13.7k
                 fReductionMatrix2))
386
1.62k
        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
12.1k
      break;
402
      
403
13.7k
      }
404
405
86.5k
    case tcProfileCalibrationSignature:
406
86.5k
      {
407
408
86.5k
      CheckTagType (parentCode, tagCode, tagType, ttAscii, ttByte);
409
      
410
86.5k
      ParseStringTag (stream,
411
86.5k
              parentCode,
412
86.5k
              tagCode,
413
86.5k
              tagCount,
414
86.5k
              fProfileCalibrationSignature,
415
86.5k
              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
86.5k
      break;
433
      
434
13.7k
      }
435
      
436
87.1k
    case tcProfileName:
437
87.1k
      {
438
      
439
87.1k
      CheckTagType (parentCode, tagCode, tagType, ttAscii, ttByte);
440
      
441
87.1k
      ParseStringTag (stream,
442
87.1k
              parentCode,
443
87.1k
              tagCode,
444
87.1k
              tagCount,
445
87.1k
              fProfileName,
446
87.1k
              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
87.1k
      break;
464
      
465
13.7k
      }
466
      
467
9.45k
    case tcProfileCopyright:
468
9.45k
      {
469
470
9.45k
      CheckTagType (parentCode, tagCode, tagType, ttAscii, ttByte);
471
      
472
9.45k
      ParseStringTag (stream,
473
9.45k
              parentCode,
474
9.45k
              tagCode,
475
9.45k
              tagCount,
476
9.45k
              fProfileCopyright,
477
9.45k
              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
9.45k
      break;
495
      
496
13.7k
      }
497
498
42.9k
    case tcProfileEmbedPolicy:
499
42.9k
      {
500
        
501
42.9k
      CheckTagType (parentCode, tagCode, tagType, ttLong);
502
      
503
42.9k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
504
      
505
42.9k
      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
42.9k
      break;
545
      
546
13.7k
      }
547
548
333k
    case tcProfileHueSatMapDims:
549
333k
      {
550
        
551
333k
      CheckTagType (parentCode, tagCode, tagType, ttLong);
552
      
553
333k
      CheckTagCount (parentCode, tagCode, tagCount, 2, 3);
554
      
555
333k
      fProfileHues = stream.TagValue_uint32 (tagType);
556
333k
      fProfileSats = stream.TagValue_uint32 (tagType);
557
558
333k
      if (tagCount > 2)
559
80.5k
        fProfileVals = stream.TagValue_uint32 (tagType);
560
253k
      else
561
253k
        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
333k
      break;
578
      
579
13.7k
      }
580
581
154k
    case tcProfileHueSatMapData1:
582
154k
      {
583
584
154k
      if (!CheckTagType (parentCode, tagCode, tagType, ttFloat))
585
132k
        return false;
586
        
587
21.5k
      bool skipSat0 = (tagCount ==
588
21.5k
               SafeUint32Mult(fProfileHues,
589
21.5k
                      SafeUint32Sub(fProfileSats, 1u),
590
21.5k
                      fProfileVals,
591
21.5k
                      3u));
592
      
593
21.5k
      if (!skipSat0)
594
10.9k
        {
595
      
596
10.9k
        if (!CheckTagCount (parentCode, tagCode, tagCount,
597
10.9k
                  SafeUint32Mult(fProfileHues, fProfileSats, fProfileVals, 3)))
598
9.05k
          return false;
599
          
600
10.9k
        }
601
        
602
12.4k
      fBigEndian = stream.BigEndian ();
603
        
604
12.4k
      fHueSatDeltas1Offset = tagOffset;
605
12.4k
      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
12.4k
      break;
625
      
626
21.5k
      }
627
628
208k
    case tcProfileHueSatMapData2:
629
208k
      {
630
631
208k
      if (!CheckTagType (parentCode, tagCode, tagType, ttFloat))
632
7.32k
        return false;
633
      
634
200k
      bool skipSat0 = (tagCount ==
635
200k
               SafeUint32Mult(fProfileHues,
636
200k
                      SafeUint32Sub(fProfileSats, 1u),
637
200k
                      fProfileVals,
638
200k
                      3u));
639
      
640
200k
      if (!skipSat0)
641
157k
        {
642
      
643
157k
        if (!CheckTagCount (parentCode, tagCode, tagCount,
644
157k
                  SafeUint32Mult(fProfileHues, fProfileSats, fProfileVals, 3)))
645
155k
          return false;
646
          
647
157k
        }
648
      
649
45.3k
      fBigEndian = stream.BigEndian ();
650
        
651
45.3k
      fHueSatDeltas2Offset = tagOffset;
652
45.3k
      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
45.3k
      break;
672
      
673
200k
      }
674
675
3.49k
    case tcProfileHueSatMapEncoding:
676
3.49k
      {
677
678
3.49k
      CheckTagType (parentCode, tagCode, tagType, ttLong);
679
      
680
3.49k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
681
      
682
3.49k
      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
3.49k
      break;
714
      
715
200k
      }
716
717
150k
    case tcProfileLookTableDims:
718
150k
      {
719
        
720
150k
      CheckTagType (parentCode, tagCode, tagType, ttLong);
721
      
722
150k
      CheckTagCount (parentCode, tagCode, tagCount, 2, 3);
723
      
724
150k
      fLookTableHues = stream.TagValue_uint32 (tagType);
725
150k
      fLookTableSats = stream.TagValue_uint32 (tagType);
726
727
150k
      if (tagCount > 2)
728
100k
        fLookTableVals = stream.TagValue_uint32 (tagType);
729
49.6k
      else
730
49.6k
        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
150k
      break;
747
      
748
200k
      }
749
750
68.5k
    case tcProfileLookTableData:
751
68.5k
      {
752
753
68.5k
      if (!CheckTagType (parentCode, tagCode, tagType, ttFloat))
754
7.74k
        return false;
755
      
756
60.8k
      bool skipSat0 = (tagCount ==
757
60.8k
               SafeUint32Mult(fLookTableHues,
758
60.8k
                      SafeUint32Sub(fLookTableSats, 1u),
759
60.8k
                      fLookTableVals,
760
60.8k
                      3u));
761
      
762
60.8k
      if (!skipSat0)
763
58.2k
        {
764
      
765
58.2k
        if (!CheckTagCount (parentCode, tagCode, tagCount,
766
58.2k
                  SafeUint32Mult(fLookTableHues,
767
58.2k
                          fLookTableSats,
768
58.2k
                          fLookTableVals, 3)))
769
23.8k
          return false;
770
          
771
58.2k
        }
772
      
773
36.9k
      fBigEndian = stream.BigEndian ();
774
        
775
36.9k
      fLookTableOffset = tagOffset;
776
36.9k
      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
36.9k
      break;
796
      
797
60.8k
      }
798
799
2.77k
    case tcProfileLookTableEncoding:
800
2.77k
      {
801
802
2.77k
      CheckTagType (parentCode, tagCode, tagType, ttLong);
803
      
804
2.77k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
805
      
806
2.77k
      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
2.77k
      break;
838
      
839
60.8k
      }
840
841
32.5k
    case tcBaselineExposureOffset:
842
32.5k
      {
843
      
844
32.5k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
845
      
846
32.5k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
847
      
848
32.5k
      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
32.5k
      break;
863
      
864
60.8k
      }
865
      
866
24.4k
    case tcDefaultBlackRender:
867
24.4k
      {
868
869
24.4k
      CheckTagType (parentCode, tagCode, tagType, ttLong);
870
      
871
24.4k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
872
      
873
24.4k
      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
24.4k
      break;
906
      
907
60.8k
      }
908
909
73.8k
    case tcProfileToneCurve:
910
73.8k
      {
911
        
912
73.8k
      if (!CheckTagType (parentCode, tagCode, tagType, ttFloat))
913
5.35k
        return false;
914
      
915
68.4k
      if (!CheckTagCount (parentCode, tagCode, tagCount, 4, tagCount))
916
24.9k
        return false;
917
    
918
43.5k
      if ((tagCount & 1) != 0)
919
1.01k
        {
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
1.01k
        return false;
940
        
941
1.01k
        }
942
      
943
42.5k
      fBigEndian = stream.BigEndian ();
944
        
945
42.5k
      fToneCurveOffset = tagOffset;
946
42.5k
      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
42.5k
      break;
966
      
967
43.5k
      }
968
969
249k
    case tcUniqueCameraModel:
970
249k
      {
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
249k
      CheckTagType (parentCode, tagCode, tagType, ttAscii);
977
      
978
249k
      ParseStringTag (stream,
979
249k
              parentCode,
980
249k
              tagCode,
981
249k
              tagCount,
982
249k
              fUniqueCameraModel,
983
249k
              false);
984
              
985
249k
      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
249k
      (void) didTrim;   // Unused
1010
        
1011
249k
      #endif
1012
      
1013
249k
      break;
1014
      
1015
43.5k
      }
1016
1017
42.3M
    default:
1018
42.3M
      {
1019
      
1020
42.3M
      return false;
1021
      
1022
43.5k
      }
1023
      
1024
44.9M
    }
1025
1026
1.41M
  return true;
1027
  
1028
44.9M
  }
1029
1030
/*****************************************************************************/
1031
1032
bool dng_camera_profile_info::ParseExtended (dng_stream &stream)
1033
6.23M
  {
1034
1035
6.23M
  try
1036
6.23M
    {
1037
    
1038
    // Offsets are relative to the start of this structure, not the entire file.
1039
1040
6.23M
    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.23M
    uint16 byteOrder = stream.Get_uint16 ();
1047
1048
6.23M
    if (byteOrder == byteOrderMM)
1049
162k
      fBigEndian = true;
1050
      
1051
6.07M
    else if (byteOrder == byteOrderII)
1052
2.32M
      fBigEndian = false;
1053
      
1054
3.74M
    else
1055
3.74M
      return false;
1056
1057
2.48M
    TempBigEndian setEndianness (stream, fBigEndian);
1058
1059
2.48M
    uint16 magicNumber = stream.Get_uint16 ();
1060
1061
2.48M
    if (magicNumber != magicExtendedProfile)
1062
1.92M
      {
1063
1.92M
      return false;
1064
1.92M
      }
1065
1066
561k
    uint32 offset = stream.Get_uint32 ();
1067
1068
561k
    stream.Skip (SafeUint32Sub(offset, 8u));
1069
1070
    // Start on IFD entries.
1071
1072
561k
    uint32 ifdEntries = stream.Get_uint16 ();
1073
  
1074
561k
    if (ifdEntries < 1)
1075
529
      {
1076
529
      return false;
1077
529
      }
1078
    
1079
45.5M
    for (uint32 tag_index = 0; tag_index < ifdEntries; tag_index++)
1080
44.9M
      {
1081
      
1082
44.9M
      stream.SetReadPosition (startPosition + 8 + 2 + tag_index * 12);
1083
      
1084
44.9M
      uint16 tagCode  = stream.Get_uint16 ();
1085
44.9M
      uint32 tagType  = stream.Get_uint16 ();
1086
44.9M
      uint32 tagCount = stream.Get_uint32 ();
1087
      
1088
44.9M
      uint64 tagOffset = stream.Position ();
1089
1090
44.9M
      if (SafeUint32Mult(TagTypeSize (tagType), tagCount) > 4)
1091
4.60M
        {
1092
1093
4.60M
        tagOffset = startPosition + stream.Get_uint32 ();
1094
1095
4.60M
        stream.SetReadPosition (tagOffset);
1096
1097
4.60M
        }
1098
        
1099
44.9M
      if (!ParseTag (stream,
1100
44.9M
               0,
1101
44.9M
               tagCode,
1102
44.9M
               tagType,
1103
44.9M
               tagCount,
1104
44.9M
               tagOffset))
1105
43.4M
        {
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
43.4M
        }
1128
1129
44.9M
      }
1130
      
1131
560k
    return true;
1132
1133
561k
    }
1134
    
1135
6.23M
  catch (...)
1136
6.23M
    {
1137
    
1138
    // Eat parsing errors.
1139
    
1140
216k
    }
1141
1142
216k
  return false;
1143
  
1144
6.23M
  }
1145
1146
/*****************************************************************************/
1147
1148
dng_shared::dng_shared ()
1149
  
1150
23.8k
  : fExifIFD       (0)
1151
23.8k
  , fGPSInfo       (0)
1152
23.8k
  , fInteroperabilityIFD (0)
1153
23.8k
  , fKodakDCRPrivateIFD  (0)
1154
23.8k
  , fKodakKDCPrivateIFD  (0)
1155
    
1156
23.8k
  , fXMPCount  (0)
1157
23.8k
  , fXMPOffset (0)
1158
    
1159
23.8k
  , fIPTC_NAA_Count  (0)
1160
23.8k
  , fIPTC_NAA_Offset (0)
1161
  
1162
23.8k
  , fMakerNoteCount  (0)
1163
23.8k
  , fMakerNoteOffset (0)
1164
23.8k
  , fMakerNoteSafety (0)
1165
  
1166
23.8k
  , fDNGVersion         (0)
1167
23.8k
  , fDNGBackwardVersion (0)
1168
  
1169
23.8k
  , fUniqueCameraModel    ()
1170
23.8k
  , fLocalizedCameraModel ()
1171
  
1172
23.8k
  , fCameraProfile ()
1173
  
1174
23.8k
  , fExtraCameraProfiles ()
1175
  
1176
23.8k
  , fCameraCalibration1 ()
1177
23.8k
  , fCameraCalibration2 ()
1178
    
1179
23.8k
  , fCameraCalibrationSignature  ()
1180
1181
23.8k
  , fAnalogBalance ()
1182
    
1183
23.8k
  , fAsShotNeutral ()
1184
    
1185
23.8k
  , fAsShotWhiteXY ()
1186
  
1187
23.8k
  , fBaselineExposure      (0, 1)
1188
23.8k
  , fBaselineNoise         (1, 1)
1189
23.8k
  , fNoiseReductionApplied (0, 0)
1190
23.8k
  , fBaselineSharpness     (1, 1)
1191
23.8k
  , fLinearResponseLimit   (1, 1)
1192
23.8k
  , fShadowScale           (1, 1)
1193
  
1194
23.8k
  , fHasBaselineExposure (false)
1195
23.8k
  , fHasShadowScale      (false)
1196
  
1197
23.8k
  , fDNGPrivateDataCount  (0)
1198
23.8k
  , fDNGPrivateDataOffset (0)
1199
  
1200
23.8k
  , fRawImageDigest    ()
1201
23.8k
  , fNewRawImageDigest ()
1202
  
1203
23.8k
  , fRawDataUniqueID ()
1204
  
1205
23.8k
  , fOriginalRawFileName ()
1206
    
1207
23.8k
  , fOriginalRawFileDataCount  (0)
1208
23.8k
  , fOriginalRawFileDataOffset (0)
1209
  
1210
23.8k
  , fOriginalRawFileDigest ()
1211
    
1212
23.8k
  , fAsShotICCProfileCount  (0)
1213
23.8k
  , fAsShotICCProfileOffset (0)
1214
  
1215
23.8k
  , fAsShotPreProfileMatrix ()
1216
    
1217
23.8k
  , fCurrentICCProfileCount  (0)
1218
23.8k
  , fCurrentICCProfileOffset (0)
1219
  
1220
23.8k
  , fCurrentPreProfileMatrix ()
1221
  
1222
23.8k
  , fColorimetricReference (crSceneReferred)
1223
1224
23.8k
  , fAsShotProfileName ()
1225
1226
23.8k
  , fNoiseProfile ()
1227
  
1228
23.8k
  , fOriginalDefaultFinalSize     ()
1229
23.8k
  , fOriginalBestQualityFinalSize ()
1230
    
1231
23.8k
  , fOriginalDefaultCropSizeH ()
1232
23.8k
  , fOriginalDefaultCropSizeV ()
1233
    
1234
23.8k
  {
1235
  
1236
23.8k
  }
1237
  
1238
/*****************************************************************************/
1239
1240
dng_shared::~dng_shared ()
1241
23.8k
  {
1242
  
1243
23.8k
  }
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.84M
  {
1257
  
1258
1.84M
  if (parentCode == 0)
1259
541k
    {
1260
    
1261
541k
    if (Parse_ifd0 (stream,
1262
541k
            exif,
1263
541k
            parentCode,
1264
541k
            tagCode,
1265
541k
            tagType,
1266
541k
            tagCount,
1267
541k
            tagOffset))
1268
315k
      {
1269
      
1270
315k
      return true;
1271
      
1272
315k
      }
1273
1274
541k
    }
1275
    
1276
1.52M
  if (parentCode == 0 ||
1277
1.30M
    parentCode == tcExifIFD)
1278
302k
    {
1279
    
1280
302k
    if (Parse_ifd0_exif (stream,
1281
302k
               exif,
1282
302k
               parentCode,
1283
302k
               tagCode,
1284
302k
               tagType,
1285
302k
               tagCount,
1286
302k
               tagOffset))
1287
7.33k
      {
1288
      
1289
7.33k
      return true;
1290
      
1291
7.33k
      }
1292
1293
302k
    }
1294
    
1295
1.52M
  return false;
1296
    
1297
1.52M
  }
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
541k
  {
1311
  
1312
541k
  switch (tagCode)
1313
541k
    {
1314
      
1315
16.2k
    case tcXMP:
1316
16.2k
      {
1317
      
1318
16.2k
      CheckTagType (parentCode, tagCode, tagType, ttByte, ttUndefined);
1319
      
1320
16.2k
      fXMPCount  = tagCount;
1321
16.2k
      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
16.2k
      break;
1344
      
1345
0
      }
1346
1347
326
    case tcIPTC_NAA:
1348
326
      {
1349
      
1350
326
      CheckTagType (parentCode, tagCode, tagType, ttLong, ttAscii, ttUndefined);
1351
      
1352
326
      fIPTC_NAA_Count = SafeUint32Mult(tagCount,
1353
326
                              TagTypeSize(tagType));
1354
326
      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
326
      break;
1434
      
1435
0
      }
1436
1437
14.4k
    case tcExifIFD:
1438
14.4k
      {
1439
      
1440
14.4k
      CheckTagType (parentCode, tagCode, tagType, ttLong, ttIFD);
1441
      
1442
14.4k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
1443
      
1444
14.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
14.4k
      break;
1456
      
1457
0
      }
1458
    
1459
3.38k
    case tcGPSInfo:
1460
3.38k
      {
1461
      
1462
3.38k
      CheckTagType (parentCode, tagCode, tagType, ttLong, ttIFD);
1463
      
1464
3.38k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
1465
      
1466
3.38k
      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
3.38k
      break;
1478
      
1479
0
      }
1480
        
1481
1.15k
    case tcKodakDCRPrivateIFD:
1482
1.15k
      {
1483
      
1484
1.15k
      CheckTagType (parentCode, tagCode, tagType, ttLong, ttIFD);
1485
      
1486
1.15k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
1487
      
1488
1.15k
      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
1.15k
      break;
1500
      
1501
0
      }
1502
    
1503
1.69k
    case tcKodakKDCPrivateIFD:
1504
1.69k
      {
1505
      
1506
1.69k
      CheckTagType (parentCode, tagCode, tagType, ttLong, ttIFD);
1507
      
1508
1.69k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
1509
      
1510
1.69k
      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.69k
      break;
1522
      
1523
0
      }
1524
    
1525
38.6k
    case tcDNGVersion:
1526
38.6k
      {
1527
      
1528
38.6k
      CheckTagType (parentCode, tagCode, tagType, ttByte);
1529
      
1530
38.6k
      CheckTagCount (parentCode, tagCode, tagCount, 4);
1531
      
1532
38.6k
      uint32 b0 = stream.Get_uint8 ();
1533
38.6k
      uint32 b1 = stream.Get_uint8 ();
1534
38.6k
      uint32 b2 = stream.Get_uint8 ();
1535
38.6k
      uint32 b3 = stream.Get_uint8 ();
1536
      
1537
38.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
38.6k
      break;
1553
      
1554
0
      }
1555
        
1556
21.4k
    case tcDNGBackwardVersion:
1557
21.4k
      {
1558
      
1559
21.4k
      CheckTagType (parentCode, tagCode, tagType, ttByte);
1560
      
1561
21.4k
      CheckTagCount (parentCode, tagCode, tagCount, 4);
1562
      
1563
21.4k
      uint32 b0 = stream.Get_uint8 ();
1564
21.4k
      uint32 b1 = stream.Get_uint8 ();
1565
21.4k
      uint32 b2 = stream.Get_uint8 ();
1566
21.4k
      uint32 b3 = stream.Get_uint8 ();
1567
      
1568
21.4k
      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
21.4k
      break;
1584
      
1585
0
      }
1586
        
1587
16.5k
    case tcUniqueCameraModel:
1588
16.5k
      {
1589
      
1590
16.5k
      CheckTagType (parentCode, tagCode, tagType, ttAscii);
1591
      
1592
16.5k
      ParseStringTag (stream,
1593
16.5k
              parentCode,
1594
16.5k
              tagCode,
1595
16.5k
              tagCount,
1596
16.5k
              fUniqueCameraModel,
1597
16.5k
              false);
1598
              
1599
16.5k
      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
16.5k
      (void) didTrim;   // Unused
1624
        
1625
16.5k
      #endif
1626
      
1627
16.5k
      break;
1628
      
1629
0
      }
1630
1631
598
    case tcLocalizedCameraModel:
1632
598
      {
1633
      
1634
598
      CheckTagType (parentCode, tagCode, tagType, ttAscii, ttByte);
1635
      
1636
598
      ParseStringTag (stream,
1637
598
              parentCode,
1638
598
              tagCode,
1639
598
              tagCount,
1640
598
              fLocalizedCameraModel,
1641
598
              false);
1642
      
1643
598
      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
598
      (void) didTrim;   // Unused
1668
        
1669
598
      #endif
1670
      
1671
598
      break;
1672
      
1673
0
      }
1674
      
1675
5.61k
    case tcCameraCalibration1:
1676
5.61k
      {
1677
      
1678
5.61k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
1679
      
1680
5.61k
      if (!CheckColorImage (parentCode, tagCode, fCameraProfile.fColorPlanes))
1681
2.78k
        return false;
1682
        
1683
2.82k
      if (!ParseMatrixTag (stream,
1684
2.82k
                 parentCode,
1685
2.82k
                 tagCode,
1686
2.82k
                 tagType,
1687
2.82k
                 tagCount,
1688
2.82k
                 fCameraProfile.fColorPlanes,
1689
2.82k
                 fCameraProfile.fColorPlanes,
1690
2.82k
                 fCameraCalibration1))
1691
1.02k
        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.79k
      break;
1707
      
1708
2.82k
      }
1709
1710
4.41k
    case tcCameraCalibration2:
1711
4.41k
      {
1712
      
1713
4.41k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
1714
      
1715
4.41k
      if (!CheckColorImage (parentCode, tagCode, fCameraProfile.fColorPlanes))
1716
413
        return false;
1717
        
1718
4.00k
      if (!ParseMatrixTag (stream,
1719
4.00k
                 parentCode,
1720
4.00k
                 tagCode,
1721
4.00k
                 tagType,
1722
4.00k
                 tagCount,
1723
4.00k
                 fCameraProfile.fColorPlanes,
1724
4.00k
                 fCameraProfile.fColorPlanes,
1725
4.00k
                 fCameraCalibration2))
1726
3.39k
        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
604
      break;
1742
      
1743
4.00k
      }
1744
1745
604
    case tcCameraCalibrationSignature:
1746
215
      {
1747
1748
215
      CheckTagType (parentCode, tagCode, tagType, ttAscii, ttByte);
1749
      
1750
215
      ParseStringTag (stream,
1751
215
              parentCode,
1752
215
              tagCode,
1753
215
              tagCount,
1754
215
              fCameraCalibrationSignature,
1755
215
              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
215
      break;
1773
      
1774
4.00k
      }
1775
      
1776
29.8k
    case tcAnalogBalance:
1777
29.8k
      {
1778
      
1779
29.8k
      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
29.8k
      bool hasselbladHack = (fDNGVersion == 0 &&
1785
1.92k
                   fCameraProfile.fColorPlanes == 0);
1786
      
1787
29.8k
      if (hasselbladHack)
1788
376
        {
1789
        
1790
376
        fCameraProfile.fColorPlanes = Pin_uint32 (0, tagCount, kMaxColorPlanes);
1791
        
1792
        #if qDNGValidate
1793
        
1794
        ReportWarning ("AnalogBalance without ColorMatrix1");
1795
        
1796
        #endif
1797
      
1798
376
        }
1799
      
1800
29.8k
      if (!CheckColorImage (parentCode, tagCode, fCameraProfile.fColorPlanes))
1801
5.41k
        return false;
1802
        
1803
24.3k
      if (!ParseVectorTag (stream,
1804
24.3k
                 parentCode,
1805
24.3k
                 tagCode,
1806
24.3k
                 tagType,
1807
24.3k
                 tagCount,
1808
24.3k
                 fCameraProfile.fColorPlanes,
1809
24.3k
                 fAnalogBalance))
1810
8.69k
        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.6k
      break;
1826
      
1827
24.3k
      }
1828
      
1829
15.6k
    case tcAsShotNeutral:
1830
13.2k
      {
1831
      
1832
13.2k
      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
13.2k
      bool hasselbladHack = (fDNGVersion == 0 &&
1838
3.48k
                   fCameraProfile.fColorPlanes == 0);
1839
      
1840
13.2k
      if (hasselbladHack)
1841
288
        {
1842
        
1843
288
        fCameraProfile.fColorPlanes = Pin_uint32 (0, tagCount, kMaxColorPlanes);
1844
        
1845
        #if qDNGValidate
1846
        
1847
        ReportWarning ("AsShotNeutral without ColorMatrix1");
1848
        
1849
        #endif
1850
      
1851
288
        }
1852
      
1853
13.2k
      if (!CheckColorImage (parentCode, tagCode, fCameraProfile.fColorPlanes))
1854
3.58k
        return false;
1855
        
1856
9.71k
      if (!ParseVectorTag (stream,
1857
9.71k
                 parentCode,
1858
9.71k
                 tagCode,
1859
9.71k
                 tagType,
1860
9.71k
                 tagCount,
1861
9.71k
                 fCameraProfile.fColorPlanes,
1862
9.71k
                 fAsShotNeutral))
1863
1.15k
        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
8.55k
      break;
1879
      
1880
9.71k
      }
1881
      
1882
12.4k
    case tcAsShotWhiteXY:
1883
12.4k
      {
1884
      
1885
12.4k
      CheckTagType (parentCode, tagCode, tagType, ttRational);
1886
      
1887
12.4k
      if (!CheckColorImage (parentCode, tagCode, fCameraProfile.fColorPlanes))
1888
1.89k
        return false;
1889
        
1890
10.5k
      if (!CheckTagCount (parentCode, tagCode, tagCount, 2))
1891
2.47k
        return false;
1892
        
1893
8.10k
      fAsShotWhiteXY.x = stream.TagValue_real64 (tagType);
1894
8.10k
      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.10k
      break;
1910
      
1911
10.5k
      }
1912
      
1913
19.6k
    case tcBaselineExposure:
1914
19.6k
      {
1915
      
1916
19.6k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
1917
      
1918
19.6k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
1919
      
1920
19.6k
      fBaselineExposure = stream.TagValue_srational (tagType);
1921
      
1922
19.6k
      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.6k
      break;
1937
      
1938
10.5k
      }
1939
      
1940
14.1k
    case tcBaselineNoise:
1941
14.1k
      {
1942
      
1943
14.1k
      CheckTagType (parentCode, tagCode, tagType, ttRational);
1944
      
1945
14.1k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
1946
      
1947
14.1k
      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.1k
      break;
1962
      
1963
10.5k
      }
1964
      
1965
2.86k
    case tcNoiseReductionApplied:
1966
2.86k
      {
1967
      
1968
2.86k
      if (!CheckTagType (parentCode, tagCode, tagType, ttRational))
1969
212
        return false;
1970
      
1971
2.65k
      if (!CheckTagCount (parentCode, tagCode, tagCount, 1))
1972
119
        return false;
1973
      
1974
2.53k
      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.53k
      break;
1990
      
1991
2.65k
      }
1992
1993
321
    case tcNoiseProfile:
1994
321
      {
1995
1996
321
      if (!CheckTagType (parentCode, tagCode, tagType, ttDouble))
1997
6
        return false;
1998
1999
      // Must be an even, positive number of doubles in a noise profile.
2000
      
2001
315
      if (!tagCount || (tagCount & 1))
2002
55
        return false;
2003
2004
      // Determine number of planes (i.e., half the number of doubles).
2005
2006
260
      const uint32 numPlanes = Pin_uint32 (0, 
2007
260
                         tagCount >> 1, 
2008
260
                         kMaxColorPlanes);
2009
2010
      // Parse the noise function parameters.
2011
2012
260
      dng_std_vector<dng_noise_function> noiseFunctions;
2013
2014
1.20k
      for (uint32 i = 0; i < numPlanes; i++)
2015
940
        {
2016
2017
940
        const real64 scale  = stream.TagValue_real64 (tagType);
2018
940
        const real64 offset = stream.TagValue_real64 (tagType);
2019
2020
940
        noiseFunctions.push_back (dng_noise_function (scale, offset));
2021
2022
940
        }
2023
2024
      // Store the noise profile.
2025
2026
260
      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
260
      break;
2054
      
2055
315
      }
2056
      
2057
13.0k
    case tcBaselineSharpness:
2058
13.0k
      {
2059
      
2060
13.0k
      CheckTagType (parentCode, tagCode, tagType, ttRational);
2061
      
2062
13.0k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
2063
      
2064
13.0k
      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
13.0k
      break;
2079
      
2080
315
      }
2081
      
2082
12.5k
    case tcLinearResponseLimit:
2083
12.5k
      {
2084
      
2085
12.5k
      CheckTagType (parentCode, tagCode, tagType, ttRational);
2086
      
2087
12.5k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
2088
      
2089
12.5k
      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.5k
      break;
2104
      
2105
315
      }
2106
      
2107
2.11k
    case tcShadowScale:
2108
2.11k
      {
2109
      
2110
2.11k
      CheckTagType (parentCode, tagCode, tagType, ttRational);
2111
      
2112
2.11k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
2113
      
2114
2.11k
      fShadowScale = stream.TagValue_urational (tagType);
2115
      
2116
2.11k
      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
2.11k
      break;
2131
      
2132
315
      }
2133
      
2134
5.99k
    case tcDNGPrivateData:
2135
5.99k
      {
2136
      
2137
5.99k
      CheckTagType (parentCode, tagCode, tagType, ttByte);
2138
      
2139
5.99k
      fDNGPrivateDataCount  = tagCount;
2140
5.99k
      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
5.99k
      break;
2158
      
2159
315
      }
2160
      
2161
1.97k
    case tcMakerNoteSafety:
2162
1.97k
      {
2163
      
2164
1.97k
      CheckTagType (parentCode, tagCode, tagType, ttShort);
2165
      
2166
1.97k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
2167
      
2168
1.97k
      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
1.97k
      break;
2183
      
2184
315
      }
2185
      
2186
5.52k
    case tcRawImageDigest:
2187
5.52k
      {
2188
      
2189
5.52k
      if (!CheckTagType (parentCode, tagCode, tagType, ttByte))
2190
748
        return false;
2191
        
2192
4.77k
      if (!CheckTagCount (parentCode, tagCode, tagCount, 16))
2193
384
        return false;
2194
        
2195
4.38k
      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
4.38k
      break;
2213
      
2214
4.77k
      }
2215
      
2216
355
    case tcNewRawImageDigest:
2217
355
      {
2218
      
2219
355
      if (!CheckTagType (parentCode, tagCode, tagType, ttByte))
2220
28
        return false;
2221
        
2222
327
      if (!CheckTagCount (parentCode, tagCode, tagCount, 16))
2223
121
        return false;
2224
        
2225
206
      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
206
      break;
2243
      
2244
327
      }
2245
      
2246
12.2k
    case tcRawDataUniqueID:
2247
12.2k
      {
2248
      
2249
12.2k
      if (!CheckTagType (parentCode, tagCode, tagType, ttByte))
2250
2.22k
        return false;
2251
        
2252
10.0k
      if (!CheckTagCount (parentCode, tagCode, tagCount, 16))
2253
5.12k
        return false;
2254
        
2255
4.92k
      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
4.92k
      break;
2273
      
2274
10.0k
      }
2275
      
2276
193
    case tcOriginalRawFileName:
2277
193
      {
2278
      
2279
193
      CheckTagType (parentCode, tagCode, tagType, ttAscii, ttByte);
2280
      
2281
193
      ParseStringTag (stream,
2282
193
              parentCode,
2283
193
              tagCode,
2284
193
              tagCount,
2285
193
              fOriginalRawFileName,
2286
193
              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
193
      break;
2304
      
2305
10.0k
      }
2306
      
2307
45
    case tcOriginalRawFileData:
2308
45
      {
2309
      
2310
45
      CheckTagType (parentCode, tagCode, tagType, ttUndefined);
2311
      
2312
45
      fOriginalRawFileDataCount  = tagCount;
2313
45
      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
45
      break;
2331
      
2332
10.0k
      }
2333
      
2334
1.90k
    case tcOriginalRawFileDigest:
2335
1.90k
      {
2336
      
2337
1.90k
      if (!CheckTagType (parentCode, tagCode, tagType, ttByte))
2338
1.14k
        return false;
2339
        
2340
763
      if (!CheckTagCount (parentCode, tagCode, tagCount, 16))
2341
529
        return false;
2342
        
2343
234
      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
234
      break;
2361
      
2362
763
      }
2363
      
2364
16
    case tcAsShotICCProfile:
2365
16
      {
2366
      
2367
16
      CheckTagType (parentCode, tagCode, tagType, ttUndefined);
2368
      
2369
16
      fAsShotICCProfileCount  = tagCount;
2370
16
      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
16
      break;
2388
      
2389
763
      }
2390
      
2391
5.75k
    case tcAsShotPreProfileMatrix:
2392
5.75k
      {
2393
      
2394
5.75k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
2395
      
2396
5.75k
      if (!CheckColorImage (parentCode, tagCode, fCameraProfile.fColorPlanes))
2397
693
        return false;
2398
        
2399
5.05k
      uint32 rows = fCameraProfile.fColorPlanes;
2400
      
2401
5.05k
      if (tagCount == fCameraProfile.fColorPlanes * 3)
2402
262
        {
2403
262
        rows = 3;
2404
262
        }
2405
      
2406
5.05k
      if (!ParseMatrixTag (stream,
2407
5.05k
                 parentCode,
2408
5.05k
                 tagCode,
2409
5.05k
                 tagType,
2410
5.05k
                 tagCount,
2411
5.05k
                 rows,
2412
5.05k
                 fCameraProfile.fColorPlanes,
2413
5.05k
                 fAsShotPreProfileMatrix))
2414
4.51k
        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
538
      break;
2430
      
2431
5.05k
      }
2432
      
2433
559
    case tcCurrentICCProfile:
2434
559
      {
2435
      
2436
559
      CheckTagType (parentCode, tagCode, tagType, ttUndefined);
2437
      
2438
559
      fCurrentICCProfileCount  = tagCount;
2439
559
      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
559
      break;
2457
      
2458
5.05k
      }
2459
      
2460
33.5k
    case tcCurrentPreProfileMatrix:
2461
33.5k
      {
2462
      
2463
33.5k
      CheckTagType (parentCode, tagCode, tagType, ttSRational);
2464
      
2465
33.5k
      if (!CheckColorImage (parentCode, tagCode, fCameraProfile.fColorPlanes))
2466
4.13k
        return false;
2467
        
2468
29.4k
      uint32 rows = fCameraProfile.fColorPlanes;
2469
      
2470
29.4k
      if (tagCount == fCameraProfile.fColorPlanes * 3)
2471
13.2k
        {
2472
13.2k
        rows = 3;
2473
13.2k
        }
2474
      
2475
29.4k
      if (!ParseMatrixTag (stream,
2476
29.4k
                 parentCode,
2477
29.4k
                 tagCode,
2478
29.4k
                 tagType,
2479
29.4k
                 tagCount,
2480
29.4k
                 rows,
2481
29.4k
                 fCameraProfile.fColorPlanes,
2482
29.4k
                 fCurrentPreProfileMatrix))
2483
16.1k
        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
13.2k
      break;
2499
      
2500
29.4k
      }
2501
      
2502
14.2k
    case tcColorimetricReference:
2503
14.2k
      {
2504
      
2505
14.2k
      CheckTagType (parentCode, tagCode, tagType, ttShort);
2506
      
2507
14.2k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
2508
      
2509
14.2k
      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
14.2k
      break;
2524
      
2525
29.4k
      }
2526
2527
4.35k
    case tcExtraCameraProfiles:
2528
4.35k
      {
2529
        
2530
4.35k
      CheckTagType (parentCode, tagCode, tagType, ttLong);
2531
      
2532
4.35k
      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
4.35k
      fExtraCameraProfiles.reserve (tagCount);
2546
      
2547
6.23M
      for (uint32 index = 0; index < tagCount; index++)
2548
6.23M
        {
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.23M
        stream.SetReadPosition (tagOffset + index * 4);
2562
        
2563
6.23M
        uint32 profileOffset = stream.TagValue_uint32 (tagType);
2564
        
2565
6.23M
        dng_camera_profile_info profileInfo;
2566
        
2567
6.23M
        stream.SetReadPosition (profileOffset);
2568
        
2569
6.23M
        if (profileInfo.ParseExtended (stream))
2570
344k
          {
2571
          
2572
344k
          fExtraCameraProfiles.push_back (profileInfo);
2573
          
2574
344k
          }
2575
          
2576
5.88M
        else
2577
5.88M
          {
2578
          
2579
          #if qDNGValidate
2580
2581
          ReportWarning ("Unable to parse extra camera profile");
2582
          
2583
          #endif
2584
      
2585
5.88M
          }
2586
        
2587
6.23M
        }
2588
2589
      #if qDNGValidate
2590
2591
      if (gVerbose)
2592
        {
2593
        
2594
        printf ("\nDone with ExtraCameraProfiles\n\n");
2595
        
2596
        }
2597
        
2598
      #endif
2599
      
2600
4.35k
      break;
2601
      
2602
29.4k
      }
2603
      
2604
1.27k
    case tcAsShotProfileName:
2605
1.27k
      {
2606
            
2607
1.27k
      CheckTagType (parentCode, tagCode, tagType, ttAscii, ttByte);
2608
      
2609
1.27k
      ParseStringTag (stream,
2610
1.27k
              parentCode,
2611
1.27k
              tagCode,
2612
1.27k
              tagCount,
2613
1.27k
              fAsShotProfileName,
2614
1.27k
              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.27k
      break;
2632
      
2633
29.4k
      }
2634
2635
1.22k
    case tcOriginalDefaultFinalSize:
2636
1.22k
      {
2637
      
2638
1.22k
      CheckTagType (parentCode, tagCode, tagType, ttShort, ttLong);
2639
      
2640
1.22k
      if (!CheckTagCount (parentCode, tagCode, tagCount, 2))
2641
289
        return false;
2642
        
2643
933
      fOriginalDefaultFinalSize.h = stream.TagValue_int32 (tagType);
2644
933
      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
933
      break;
2660
      
2661
1.22k
      }
2662
        
2663
395
    case tcOriginalBestQualityFinalSize:
2664
395
      {
2665
      
2666
395
      CheckTagType (parentCode, tagCode, tagType, ttShort, ttLong);
2667
      
2668
395
      if (!CheckTagCount (parentCode, tagCode, tagCount, 2))
2669
196
        return false;
2670
        
2671
199
      fOriginalBestQualityFinalSize.h = stream.TagValue_int32 (tagType);
2672
199
      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
199
      break;
2688
      
2689
395
      }
2690
        
2691
1.00k
    case tcOriginalDefaultCropSize:
2692
1.00k
      {
2693
      
2694
1.00k
      CheckTagType (parentCode, tagCode, tagType, ttShort, ttLong, ttRational);
2695
      
2696
1.00k
      if (!CheckTagCount (parentCode, tagCode, tagCount, 2))
2697
29
        return false;
2698
        
2699
974
      fOriginalDefaultCropSizeH = stream.TagValue_urational (tagType);
2700
974
      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
974
      break;
2716
      
2717
1.00k
      }
2718
      
2719
205k
    default:
2720
205k
      {
2721
      
2722
      // The main camera profile tags also appear in IFD 0
2723
      
2724
205k
      return fCameraProfile.ParseTag (stream,
2725
205k
                      parentCode,
2726
205k
                      tagCode,
2727
205k
                      tagType,
2728
205k
                      tagCount,
2729
205k
                      tagOffset);
2730
      
2731
1.00k
      }
2732
      
2733
541k
    }
2734
2735
266k
  return true;
2736
  
2737
541k
  }
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
302k
  {
2751
  
2752
302k
  switch (tagCode)
2753
302k
    {
2754
    
2755
5.07k
    case tcMakerNote:
2756
5.07k
      {
2757
      
2758
5.07k
      CheckTagType (parentCode, tagCode, tagType, ttUndefined);
2759
      
2760
5.07k
      fMakerNoteCount  = tagCount;
2761
5.07k
      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
5.07k
      break;
2779
      
2780
0
      }
2781
      
2782
2.26k
    case tcInteroperabilityIFD:
2783
2.26k
      {
2784
      
2785
2.26k
      CheckTagType (parentCode, tagCode, tagType, ttLong, ttIFD);
2786
      
2787
2.26k
      CheckTagCount (parentCode, tagCode, tagCount, 1);
2788
      
2789
2.26k
      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.26k
      break;
2801
      
2802
0
      }
2803
      
2804
295k
    default:
2805
295k
      {
2806
      
2807
295k
      return false;
2808
      
2809
0
      }
2810
      
2811
302k
    }
2812
    
2813
7.33k
  return true;
2814
  
2815
302k
  }
2816
                 
2817
/*****************************************************************************/
2818
2819
void dng_shared::PostParse (dng_host & /* host */,
2820
              dng_exif & /* exif */)
2821
17.9k
  {
2822
  
2823
  // Fill in default values for DNG images.
2824
  
2825
17.9k
  if (fDNGVersion != 0)
2826
15.5k
    {
2827
    
2828
    // Support for DNG versions before 1.0.0.0.
2829
    
2830
15.5k
    if (fDNGVersion < dngVersion_1_0_0_0)
2831
2.63k
      {
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
2.63k
      fCameraProfile.fCalibrationIlluminant1 = lsStandardLightA;
2843
2.63k
      fCameraProfile.fCalibrationIlluminant2 = lsD65;
2844
      
2845
2.63k
      fDNGVersion = dngVersion_1_0_0_0;
2846
2847
2.63k
      }
2848
    
2849
    // Default value for DNGBackwardVersion tag.
2850
    
2851
15.5k
    if (fDNGBackwardVersion == 0)
2852
8.39k
      {
2853
      
2854
8.39k
      fDNGBackwardVersion = fDNGVersion & 0xFFFF0000;
2855
      
2856
8.39k
      }
2857
  
2858
    // Check DNGBackwardVersion value.
2859
    
2860
15.5k
    if (fDNGBackwardVersion < dngVersion_1_0_0_0)
2861
795
      {
2862
      
2863
      #if qDNGValidate
2864
      
2865
      ReportWarning ("DNGBackwardVersion less than 1.0.0.0");
2866
             
2867
      #endif
2868
      
2869
795
      fDNGBackwardVersion = dngVersion_1_0_0_0;
2870
        
2871
795
      }
2872
2873
15.5k
    if (fDNGBackwardVersion > fDNGVersion)
2874
1.78k
      {
2875
      
2876
      #if qDNGValidate
2877
      
2878
      ReportWarning ("DNGBackwardVersion > DNGVersion");
2879
             
2880
      #endif
2881
      
2882
1.78k
      fDNGBackwardVersion = fDNGVersion;
2883
          
2884
1.78k
      }
2885
2886
    // Check UniqueCameraModel.
2887
    
2888
15.5k
    if (fUniqueCameraModel.IsEmpty ())
2889
14.3k
      {
2890
      
2891
      #if qDNGValidate
2892
      
2893
      ReportWarning ("Missing or invalid UniqueCameraModel");
2894
             
2895
      #endif
2896
      
2897
14.3k
      fUniqueCameraModel.Set ("Digital Negative");
2898
      
2899
14.3k
      }
2900
    
2901
    // If we don't know the color depth yet, it must be a monochrome DNG.
2902
    
2903
15.5k
    if (fCameraProfile.fColorPlanes == 0)
2904
7.81k
      {
2905
      
2906
7.81k
      fCameraProfile.fColorPlanes = 1;
2907
      
2908
7.81k
      }
2909
      
2910
    // Check color info.
2911
    
2912
15.5k
    if (fCameraProfile.fColorPlanes > 1)
2913
7.71k
      {
2914
      
2915
      // Check illuminant pair.
2916
      
2917
7.71k
      if (fCameraProfile.fColorMatrix2.NotEmpty ())
2918
92
        {
2919
        
2920
92
        if (fCameraProfile.fCalibrationIlluminant1 == lsUnknown ||
2921
88
          (fCameraProfile.fCalibrationIlluminant2 == lsUnknown ||
2922
83
          (fCameraProfile.fCalibrationIlluminant1 == fCameraProfile.fCalibrationIlluminant2)))
2923
11
          {
2924
            
2925
          #if qDNGValidate
2926
      
2927
          ReportWarning ("Invalid CalibrationIlluminant pair");
2928
                 
2929
          #endif
2930
                 
2931
11
          fCameraProfile.fColorMatrix2 = dng_matrix ();
2932
        
2933
11
          }
2934
2935
92
        }
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
7.71k
      if (fColorimetricReference == crICCProfilePCS)
2942
2.56k
        {
2943
        
2944
2.56k
        if (fAsShotNeutral.NotEmpty ())
2945
2
          {
2946
          
2947
          #if qDNGValidate
2948
        
2949
          ReportWarning ("AsShotNeutral not allowed for this "
2950
                   "ColorimetricReference value");
2951
                 
2952
          #endif
2953
          
2954
2
          fAsShotNeutral.Clear ();
2955
          
2956
2
          }
2957
          
2958
2.56k
        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
2.56k
        fAsShotWhiteXY = pcs;
2978
2979
2.56k
        }
2980
        
2981
5.14k
      else
2982
5.14k
        {
2983
        
2984
        // Warn if both AsShotNeutral and AsShotWhiteXY are specified.
2985
          
2986
5.14k
        if (fAsShotNeutral.NotEmpty () && fAsShotWhiteXY.IsValid ())
2987
29
          {
2988
            
2989
          #if qDNGValidate
2990
        
2991
          ReportWarning ("Both AsShotNeutral and AsShotWhiteXY included");
2992
                 
2993
          #endif
2994
          
2995
29
          fAsShotWhiteXY = dng_xy_coord ();
2996
                   
2997
29
          }
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
5.14k
        }
3014
        
3015
      // Default values of calibration signatures are required for legacy
3016
      // compatiblity.
3017
3018
7.71k
      if (fCameraProfile.fCalibrationIlluminant1 == lsStandardLightA &&
3019
395
        fCameraProfile.fCalibrationIlluminant2 == lsD65            &&
3020
367
        fCameraCalibration1.Rows () == fCameraProfile.fColorPlanes &&
3021
8
        fCameraCalibration1.Cols () == fCameraProfile.fColorPlanes &&
3022
8
        fCameraCalibration2.Rows () == fCameraProfile.fColorPlanes &&
3023
5
        fCameraCalibration2.Cols () == fCameraProfile.fColorPlanes &&
3024
5
        fCameraCalibrationSignature.IsEmpty ()                     &&
3025
4
        fCameraProfile.fProfileCalibrationSignature.IsEmpty ()     )
3026
2
        {
3027
3028
2
        fCameraCalibrationSignature.Set (kAdobeCalibrationSignature);
3029
          
3030
2
        fCameraProfile.fProfileCalibrationSignature.Set (kAdobeCalibrationSignature);
3031
3032
2
        }
3033
3034
7.71k
      }
3035
      
3036
    // Check BaselineNoise.
3037
3038
15.5k
    if (fBaselineNoise.As_real64 () <= 0.0)
3039
164
      {
3040
      
3041
      #if qDNGValidate
3042
      
3043
      ReportWarning ("Invalid BaselineNoise");
3044
             
3045
      #endif
3046
      
3047
164
      fBaselineNoise = dng_urational (1, 1);
3048
               
3049
164
      }
3050
      
3051
    // Check BaselineSharpness.
3052
    
3053
15.5k
    if (fBaselineSharpness.As_real64 () <= 0.0)
3054
466
      {
3055
      
3056
      #if qDNGValidate
3057
      
3058
      ReportWarning ("Invalid BaselineSharpness");
3059
             
3060
      #endif
3061
      
3062
466
      fBaselineSharpness = dng_urational (1, 1);
3063
               
3064
466
      }
3065
3066
    // Check NoiseProfile.
3067
3068
15.5k
    if (!fNoiseProfile.IsValid () && fNoiseProfile.NumFunctions () != 0)
3069
22
      {
3070
      
3071
      #if qDNGValidate
3072
      
3073
      ReportWarning ("Invalid NoiseProfile");
3074
             
3075
      #endif
3076
      
3077
22
      fNoiseProfile = dng_noise_profile ();
3078
               
3079
22
      }
3080
      
3081
    // Check LinearResponseLimit.
3082
    
3083
15.5k
    if (fLinearResponseLimit.As_real64 () < 0.5 ||
3084
13.9k
      fLinearResponseLimit.As_real64 () > 1.0)
3085
2.48k
      {
3086
      
3087
      #if qDNGValidate
3088
      
3089
      ReportWarning ("Invalid LinearResponseLimit");
3090
             
3091
      #endif
3092
      
3093
2.48k
      fLinearResponseLimit = dng_urational (1, 1);
3094
               
3095
2.48k
      }
3096
      
3097
    // Check ShadowScale.
3098
    
3099
15.5k
    if (fShadowScale.As_real64 () <= 0.0)
3100
108
      {
3101
      
3102
      #if qDNGValidate
3103
      
3104
      ReportWarning ("Invalid ShadowScale");
3105
             
3106
      #endif
3107
               
3108
108
      fShadowScale = dng_urational (1, 1);
3109
               
3110
108
      }
3111
    
3112
15.5k
    }
3113
  
3114
17.9k
  }
3115
                 
3116
/*****************************************************************************/
3117
3118
bool dng_shared::IsValidDNG ()
3119
17.4k
  {
3120
  
3121
  // Check DNGVersion value.
3122
  
3123
17.4k
  if (fDNGVersion < dngVersion_1_0_0_0)
3124
2.38k
    {
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
2.38k
    return false;
3149
      
3150
2.38k
    }
3151
    
3152
  // Check DNGBackwardVersion value.
3153
  
3154
15.0k
  if (fDNGBackwardVersion > dngVersion_Current)
3155
433
    {
3156
    
3157
    #if qDNGValidate
3158
    
3159
    ReportError ("DNGBackwardVersion (or DNGVersion) is too high");
3160
    
3161
    #endif
3162
    
3163
433
    ThrowUnsupportedDNG ();
3164
      
3165
433
    }
3166
    
3167
  // Check color transform info.
3168
  
3169
15.0k
  if (fCameraProfile.fColorPlanes > 1)
3170
7.19k
    {
3171
    
3172
    // CameraCalibration1 is optional, but it must be valid if present.
3173
    
3174
7.19k
    if (fCameraCalibration1.Cols () != 0 ||
3175
7.08k
      fCameraCalibration1.Rows () != 0)
3176
105
      {
3177
      
3178
105
      if (fCameraCalibration1.Cols () != fCameraProfile.fColorPlanes ||
3179
105
        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
105
      try
3195
105
        {
3196
        
3197
105
        (void) Invert (fCameraCalibration1);
3198
        
3199
105
        }
3200
        
3201
105
      catch (...)
3202
105
        {
3203
        
3204
        #if qDNGValidate
3205
    
3206
        ReportError ("CameraCalibration1 is not invertable");
3207
               
3208
        #endif
3209
               
3210
2
        return false;
3211
      
3212
2
        }
3213
        
3214
105
      }
3215
    
3216
    // CameraCalibration2 is optional, but it must be valid if present.
3217
    
3218
7.19k
    if (fCameraCalibration2.Cols () != 0 ||
3219
7.17k
      fCameraCalibration2.Rows () != 0)
3220
20
      {
3221
      
3222
20
      if (fCameraCalibration2.Cols () != fCameraProfile.fColorPlanes ||
3223
20
        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
20
      try
3239
20
        {
3240
        
3241
20
        (void) Invert (fCameraCalibration2);
3242
        
3243
20
        }
3244
        
3245
20
      catch (...)
3246
20
        {
3247
        
3248
        #if qDNGValidate
3249
  
3250
        ReportError ("CameraCalibration2 is not invertable");
3251
               
3252
        #endif
3253
               
3254
2
        return false;
3255
      
3256
2
        }
3257
          
3258
20
      }
3259
      
3260
    // Check analog balance
3261
    
3262
7.19k
    dng_matrix analogBalance;
3263
    
3264
7.19k
    if (fAnalogBalance.NotEmpty ())
3265
3.95k
      {
3266
      
3267
3.95k
      analogBalance = fAnalogBalance.AsDiagonal ();
3268
      
3269
3.95k
      try
3270
3.95k
        {
3271
        
3272
3.95k
        (void) Invert (analogBalance);
3273
        
3274
3.95k
        }
3275
        
3276
3.95k
      catch (...)
3277
3.95k
        {
3278
        
3279
        #if qDNGValidate
3280
    
3281
        ReportError ("AnalogBalance is not invertable");
3282
               
3283
        #endif
3284
               
3285
21
        return false;
3286
      
3287
21
        }
3288
        
3289
3.95k
      }
3290
        
3291
7.19k
    }
3292
      
3293
15.0k
  return true;
3294
  
3295
15.0k
  }
3296
  
3297
/*****************************************************************************/