1 /*****************************************************************************/
2 // Copyright 2006-2007 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_info.cpp#1 $ */
10 /* $DateTime: 2012/05/30 13:28:51 $ */
11 /* $Change: 832332 $ */
12 /* $Author: tknoll $ */
13 
14 /*****************************************************************************/
15 
16 #include "dng_info.h"
17 
18 #include "dng_camera_profile.h"
19 #include "dng_exceptions.h"
20 #include "dng_globals.h"
21 #include "dng_host.h"
22 #include "dng_tag_codes.h"
23 #include "dng_parse_utils.h"
24 #include "dng_safe_arithmetic.h"
25 #include "dng_tag_types.h"
26 #include "dng_tag_values.h"
27 #include "dng_utils.h"
28 
29 /*****************************************************************************/
30 
31 dng_info::dng_info ()
32 
33 	:	fTIFFBlockOffset         (0)
34 	,	fTIFFBlockOriginalOffset (kDNGStreamInvalidOffset)
35 	,	fBigEndian				 (false)
36 	,	fMagic					 (0)
37 	,	fExif					 ()
38 	,	fShared					 ()
39 	,	fMainIndex				 (-1)
40 	,	fMaskIndex				 (-1)
41 	,	fIFDCount				 (0)
42 	,	fChainedIFDCount		 (0)
43 	,	fMakerNoteNextIFD		 (0)
44 
45 	{
46 
47 	}
48 
49 /*****************************************************************************/
50 
51 dng_info::~dng_info ()
52 	{
53 
54 	}
55 
56 /*****************************************************************************/
57 
58 void dng_info::ValidateMagic ()
59 	{
60 
61 	switch (fMagic)
62 		{
63 
64 		case magicTIFF:
65 		case magicExtendedProfile:
66 		case magicRawCache:
67 		case magicPanasonic:
68 		case magicOlympusA:
69 		case magicOlympusB:
70 			{
71 
72 			return;
73 
74 			}
75 
76 		default:
77 			{
78 
79 			#if qDNGValidate
80 
81 			ReportError ("Invalid TIFF magic number");
82 
83 			#endif
84 
85 			ThrowBadFormat ();
86 
87 			}
88 
89 		}
90 
91 	}
92 
93 /*****************************************************************************/
94 
95 void dng_info::ParseTag (dng_host &host,
96 						 dng_stream &stream,
97 						 dng_exif *exif,
98 						 dng_shared *shared,
99 						 dng_ifd *ifd,
100 						 uint32 parentCode,
101 						 uint32 tagCode,
102 						 uint32 tagType,
103 						 uint32 tagCount,
104 						 uint64 tagOffset,
105 						 int64 offsetDelta)
106 	{
107 
108 	bool isSubIFD = parentCode >= tcFirstSubIFD &&
109 					parentCode <= tcLastSubIFD;
110 
111 	bool isMainIFD = (parentCode == 0 || isSubIFD) &&
112 					 ifd &&
113 					 ifd->fUsesNewSubFileType &&
114 			 		 ifd->fNewSubFileType == sfMainImage;
115 
116 	// Panasonic RAW format stores private tags using tag codes < 254 in
117 	// IFD 0.  Redirect the parsing of these tags into a logical
118 	// "PanasonicRAW" IFD.
119 
120 	// Panasonic is starting to use some higher numbers also (280..283).
121 
122 	if (fMagic == 85 && parentCode == 0 && (tagCode < tcNewSubFileType ||
123 											(tagCode >= 280 && tagCode <= 283)))
124 		{
125 
126 		parentCode = tcPanasonicRAW;
127 
128 		ifd = NULL;
129 
130 		}
131 
132 	stream.SetReadPosition (tagOffset);
133 
134 	if (ifd && ifd->ParseTag (stream,
135 						 	  parentCode,
136 						 	  tagCode,
137 						 	  tagType,
138 						 	  tagCount,
139 						 	  tagOffset))
140 		{
141 
142 		return;
143 
144 		}
145 
146 	stream.SetReadPosition (tagOffset);
147 
148 	if (exif && shared && exif->ParseTag (stream,
149 										  *shared,
150 										  parentCode,
151 										  isMainIFD,
152 										  tagCode,
153 										  tagType,
154 										  tagCount,
155 										  tagOffset))
156 		{
157 
158 		return;
159 
160 		}
161 
162 	stream.SetReadPosition (tagOffset);
163 
164 	if (shared && exif && shared->ParseTag (stream,
165 											*exif,
166 								    		parentCode,
167 								    		isMainIFD,
168 								    		tagCode,
169 								    		tagType,
170 								    		tagCount,
171 								    		tagOffset,
172 								    		offsetDelta))
173 		{
174 
175 		return;
176 
177 		}
178 
179 	if (parentCode == tcLeicaMakerNote &&
180 		tagType == ttUndefined &&
181 		tagCount >= 14)
182 		{
183 
184 		if (ParseMakerNoteIFD (host,
185 							   stream,
186 							   tagCount,
187 							   tagOffset,
188 							   offsetDelta,
189 							   tagOffset,
190 							   stream.Length (),
191 							   tcLeicaMakerNote))
192 			{
193 
194 			return;
195 
196 			}
197 
198 		}
199 
200 	if (parentCode == tcOlympusMakerNote &&
201 		tagType == ttUndefined &&
202 		tagCount >= 14)
203 		{
204 
205 		uint32 olympusMakerParent = 0;
206 
207 		switch (tagCode)
208 			{
209 
210 			case 8208:
211 				olympusMakerParent = tcOlympusMakerNote8208;
212 				break;
213 
214 			case 8224:
215 				olympusMakerParent = tcOlympusMakerNote8224;
216 				break;
217 
218 			case 8240:
219 				olympusMakerParent = tcOlympusMakerNote8240;
220 				break;
221 
222 			case 8256:
223 				olympusMakerParent = tcOlympusMakerNote8256;
224 				break;
225 
226 			case 8272:
227 				olympusMakerParent = tcOlympusMakerNote8272;
228 				break;
229 
230 			case 12288:
231 				olympusMakerParent = tcOlympusMakerNote12288;
232 				break;
233 
234 			default:
235 				break;
236 
237 			}
238 
239 		if (olympusMakerParent)
240 			{
241 
242 			// Olympus made a mistake in some camera models in computing
243 			// the size of these sub-tags, so we fudge the count.
244 
245 			if (ParseMakerNoteIFD (host,
246 								   stream,
247 							       stream.Length () - tagOffset,
248 				   	  		       tagOffset,
249 				   	  		       offsetDelta,
250 				   	  		       tagOffset,
251 				   	  		       stream.Length (),
252 				   	  		       olympusMakerParent))
253 				{
254 
255 				return;
256 
257 				}
258 
259 			}
260 
261 		}
262 
263 	if (parentCode == tcRicohMakerNote &&
264 		tagCode == 0x2001 &&
265 		tagType == ttUndefined &&
266 		tagCount > 22)
267 		{
268 
269 		char header [20];
270 
271 		stream.SetReadPosition (tagOffset);
272 
273 		stream.Get (header, sizeof (header));
274 
275 		if (memcmp (header, "[Ricoh Camera Info]", 19) == 0)
276 			{
277 
278 			ParseMakerNoteIFD (host,
279 							   stream,
280 							   tagCount - 20,
281 				   	  		   tagOffset + 20,
282 				   	  		   offsetDelta,
283 				   	  		   tagOffset + 20,
284 				   	  		   tagOffset + tagCount,
285 				   	  		   tcRicohMakerNoteCameraInfo);
286 
287 			return;
288 
289 			}
290 
291 		}
292 
293 	#if qDNGValidate
294 
295 		{
296 
297 		stream.SetReadPosition (tagOffset);
298 
299 		if (gVerbose)
300 			{
301 
302 			printf ("*");
303 
304 			DumpTagValues (stream,
305 						   LookupTagType (tagType),
306 						   parentCode,
307 						   tagCode,
308 						   tagType,
309 						   tagCount);
310 
311 			}
312 
313 		// If type is ASCII, then parse anyway so we report any ASCII
314 		// NULL termination or character set errors.
315 
316 		else if (tagType == ttAscii)
317 			{
318 
319 			dng_string s;
320 
321 			ParseStringTag (stream,
322 							parentCode,
323 							tagCode,
324 							tagCount,
325 							s,
326 							false);
327 
328 			}
329 
330 		}
331 
332 	#endif
333 
334 	}
335 
336 /*****************************************************************************/
337 
338 bool dng_info::ValidateIFD (dng_stream &stream,
339 						    uint64 ifdOffset,
340 						    int64 offsetDelta)
341 	{
342 
343 	// Make sure we have a count.
344 
345 	if (ifdOffset + 2 > stream.Length ())
346 		{
347 		return false;
348 		}
349 
350 	// Get entry count.
351 
352 	stream.SetReadPosition (ifdOffset);
353 
354 	uint32 ifdEntries = stream.Get_uint16 ();
355 
356 	if (ifdEntries < 1)
357 		{
358 		return false;
359 		}
360 
361 	// Make sure we have room for all entries and next IFD link.
362 
363 	if (ifdOffset + 2 + ifdEntries * 12 + 4 > stream.Length ())
364 		{
365 		return false;
366 		}
367 
368 	// Check each entry.
369 
370 	for (uint32 tag_index = 0; tag_index < ifdEntries; tag_index++)
371 		{
372 
373 		stream.SetReadPosition (ifdOffset + 2 + tag_index * 12);
374 
375 		stream.Skip (2);		// Ignore tag code.
376 
377 		uint32 tagType  = stream.Get_uint16 ();
378 		uint32 tagCount = stream.Get_uint32 ();
379 
380 		uint32 tag_type_size = TagTypeSize (tagType);
381 
382 		if (tag_type_size == 0)
383 			{
384 			return false;
385 			}
386 
387 		uint32 tag_data_size = SafeUint32Mult(tagCount, tag_type_size);
388 
389 		if (tag_data_size > 4)
390 			{
391 
392 			uint64 tagOffset = stream.Get_uint32 ();
393 
394 			tagOffset += offsetDelta;
395 
396 			if (SafeUint64Add(tagOffset, tag_data_size) > stream.Length())
397 				{
398 				return false;
399 				}
400 
401 			}
402 
403 		}
404 
405 	return true;
406 
407 	}
408 
409 /*****************************************************************************/
410 
411 void dng_info::ParseIFD (dng_host &host,
412 						 dng_stream &stream,
413 						 dng_exif *exif,
414 						 dng_shared *shared,
415 						 dng_ifd *ifd,
416 						 uint64 ifdOffset,
417 						 int64 offsetDelta,
418 						 uint32 parentCode)
419 	{
420 
421 	#if qDNGValidate
422 
423 	bool isMakerNote = (parentCode >= tcFirstMakerNoteIFD &&
424 						parentCode <= tcLastMakerNoteIFD);
425 
426 	#endif
427 
428 	stream.SetReadPosition (ifdOffset);
429 
430 	if (ifd)
431 		{
432 		ifd->fThisIFD = ifdOffset;
433 		}
434 
435 	uint32 ifdEntries = stream.Get_uint16 ();
436 
437 	#if qDNGValidate
438 
439 	if (gVerbose)
440 		{
441 
442 		printf ("%s: Offset = %u, Entries = %u\n\n",
443 				LookupParentCode (parentCode),
444 			    (unsigned) ifdOffset,
445 			    (unsigned) ifdEntries);
446 
447 		}
448 
449 	if ((ifdOffset & 1) && !isMakerNote)
450 		{
451 
452 		char message [256];
453 
454 		sprintf (message,
455 				 "%s has odd offset (%u)",
456 				 LookupParentCode (parentCode),
457 				 (unsigned) ifdOffset);
458 
459 		ReportWarning (message);
460 
461 		}
462 
463 	#endif
464 
465 	uint32 prev_tag_code = 0;
466 
467 	for (uint32 tag_index = 0; tag_index < ifdEntries; tag_index++)
468 		{
469 
470 		stream.SetReadPosition (ifdOffset + 2 + tag_index * 12);
471 
472 		uint32 tagCode  = stream.Get_uint16 ();
473 		uint32 tagType  = stream.Get_uint16 ();
474 
475 		// Minolta 7D files have a bug in the EXIF block where the count
476 		// is wrong, and we run off into next IFD link.  So if abort parsing
477 		// if we get a zero code/type combinations.
478 
479 		if (tagCode == 0 && tagType == 0)
480 			{
481 
482 			#if qDNGValidate
483 
484 			char message [256];
485 
486 			sprintf (message,
487 					 "%s had zero/zero tag code/type entry",
488 					 LookupParentCode (parentCode));
489 
490 			ReportWarning (message);
491 
492 			#endif
493 
494 			return;
495 
496 			}
497 
498 		uint32 tagCount = stream.Get_uint32 ();
499 
500 		#if qDNGValidate
501 
502 			{
503 
504 			if (tag_index > 0 && tagCode <= prev_tag_code && !isMakerNote)
505 				{
506 
507 				char message [256];
508 
509 				sprintf (message,
510 						 "%s tags are not sorted in ascending numerical order",
511 						 LookupParentCode (parentCode));
512 
513 				ReportWarning (message);
514 
515 				}
516 
517 			}
518 
519 		#endif
520 
521 		prev_tag_code = tagCode;
522 
523 		uint32 tag_type_size = TagTypeSize (tagType);
524 
525 		if (tag_type_size == 0)
526 			{
527 
528 			#if qDNGValidate
529 
530 				{
531 
532 				char message [256];
533 
534 				sprintf (message,
535 						 "%s %s has unknown type (%u)",
536 						 LookupParentCode (parentCode),
537 						 LookupTagCode (parentCode, tagCode),
538 						 (unsigned) tagType);
539 
540 				ReportWarning (message);
541 
542 				}
543 
544 			#endif
545 
546 			continue;
547 
548 			}
549 
550 		uint64 tagOffset = ifdOffset + 2 + tag_index * 12 + 8;
551 
552 		if (SafeUint32Mult(tagCount, tag_type_size) > 4)
553 			{
554 
555 			tagOffset = stream.Get_uint32 ();
556 
557 			#if qDNGValidate
558 
559 				{
560 
561 				if (!(ifdOffset & 1) &&
562 				     (tagOffset & 1) &&
563 				    !isMakerNote     &&
564 				    parentCode != tcKodakDCRPrivateIFD &&
565 					parentCode != tcKodakKDCPrivateIFD)
566 					{
567 
568 					char message [256];
569 
570 					sprintf (message,
571 							 "%s %s has odd data offset (%u)",
572 						 	 LookupParentCode (parentCode),
573 						 	 LookupTagCode (parentCode, tagCode),
574 							 (unsigned) tagOffset);
575 
576 					ReportWarning (message);
577 
578 					}
579 
580 				}
581 
582 			#endif
583 
584 			tagOffset += offsetDelta;
585 
586 			stream.SetReadPosition (tagOffset);
587 
588 			}
589 
590 		ParseTag (host,
591 				  stream,
592 			      exif,
593 				  shared,
594 				  ifd,
595 				  parentCode,
596 				  tagCode,
597 				  tagType,
598 				  tagCount,
599 				  tagOffset,
600 				  offsetDelta);
601 
602 		}
603 
604 	stream.SetReadPosition (ifdOffset + 2 + ifdEntries * 12);
605 
606 	uint32 nextIFD = stream.Get_uint32 ();
607 
608 	#if qDNGValidate
609 
610 	if (gVerbose)
611 		{
612 		printf ("NextIFD = %u\n", (unsigned) nextIFD);
613 		}
614 
615 	#endif
616 
617 	if (ifd)
618 		{
619 		ifd->fNextIFD = nextIFD;
620 		}
621 
622 	#if qDNGValidate
623 
624 	if (nextIFD)
625 		{
626 
627 		if (parentCode != 0 &&
628 				(parentCode < tcFirstChainedIFD ||
629 				 parentCode > tcLastChainedIFD  ))
630 			{
631 
632 			char message [256];
633 
634 			sprintf (message,
635 					 "%s has an unexpected non-zero NextIFD (%u)",
636 				 	 LookupParentCode (parentCode),
637 				 	 (unsigned) nextIFD);
638 
639 			ReportWarning (message);
640 
641 			}
642 
643 		}
644 
645 	if (gVerbose)
646 		{
647 		printf ("\n");
648 		}
649 
650 	#endif
651 
652 	}
653 
654 /*****************************************************************************/
655 
656 bool dng_info::ParseMakerNoteIFD (dng_host &host,
657 								  dng_stream &stream,
658 								  uint64 ifdSize,
659 						 		  uint64 ifdOffset,
660 								  int64 offsetDelta,
661 								  uint64 minOffset,
662 								  uint64 maxOffset,
663 						 		  uint32 parentCode)
664 	{
665 
666 	uint32 tagIndex;
667 	uint32 tagCode;
668 	uint32 tagType;
669 	uint32 tagCount;
670 
671 	// Assume there is no next IFD pointer.
672 
673 	fMakerNoteNextIFD = 0;
674 
675 	// If size is too small to hold a single entry IFD, abort.
676 
677 	if (ifdSize < 14)
678 		{
679 		return false;
680 		}
681 
682 	// Get entry count.
683 
684 	stream.SetReadPosition (ifdOffset);
685 
686 	uint32 ifdEntries = stream.Get_uint16 ();
687 
688 	// Make the entry count if reasonable for the MakerNote size.
689 
690 	if (ifdEntries < 1 || 2 + ifdEntries * 12 > ifdSize)
691 		{
692 		return false;
693 		}
694 
695 	// Scan IFD to verify all the tag types are all valid.
696 
697 	for (tagIndex = 0; tagIndex < ifdEntries; tagIndex++)
698 		{
699 
700 		stream.SetReadPosition (ifdOffset + 2 + tagIndex * 12 + 2);
701 
702 		tagType = stream.Get_uint16 ();
703 
704 		// Kludge: Some Canon MakerNotes contain tagType = 0 tags, so we
705 		// need to ignore them.  This was a "firmware 1.0.4" Canon 40D raw file.
706 
707 		if (parentCode == tcCanonMakerNote && tagType == 0)
708 			{
709 			continue;
710 			}
711 
712 		if (TagTypeSize (tagType) == 0)
713 			{
714 			return false;
715 			}
716 
717 		}
718 
719 	// OK, the IFD looks reasonable enough to parse.
720 
721 	#if qDNGValidate
722 
723 	if (gVerbose)
724 		{
725 
726 		printf ("%s: Offset = %u, Entries = %u\n\n",
727 				LookupParentCode (parentCode),
728 			    (unsigned) ifdOffset,
729 			    (unsigned) ifdEntries);
730 
731 		}
732 
733 	#endif
734 
735 	for (tagIndex = 0; tagIndex < ifdEntries; tagIndex++)
736 		{
737 
738 		stream.SetReadPosition (ifdOffset + 2 + tagIndex * 12);
739 
740 		tagCode  = stream.Get_uint16 ();
741 		tagType  = stream.Get_uint16 ();
742 		tagCount = stream.Get_uint32 ();
743 
744 		if (tagType == 0)
745 			{
746 			continue;
747 			}
748 
749 		uint32 tagSize = SafeUint32Mult(tagCount, TagTypeSize (tagType));
750 
751 		uint64 tagOffset = ifdOffset + 2 + tagIndex * 12 + 8;
752 
753 		if (tagSize > 4)
754 			{
755 
756 			tagOffset = stream.Get_uint32 () + offsetDelta;
757 
758 			if (tagOffset           < minOffset ||
759 				SafeUint64Add(tagOffset, tagSize) > maxOffset)
760 				{
761 
762 				// Tag data is outside the valid offset range,
763 				// so ignore this tag.
764 
765 				continue;
766 
767 				}
768 
769 			stream.SetReadPosition (tagOffset);
770 
771 			}
772 
773 		// Olympus switched to using IFDs in version 3 makernotes.
774 
775 		if (parentCode == tcOlympusMakerNote &&
776 			tagType == ttIFD &&
777 			tagCount == 1)
778 			{
779 
780 			uint32 olympusMakerParent = 0;
781 
782 			switch (tagCode)
783 				{
784 
785 				case 8208:
786 					olympusMakerParent = tcOlympusMakerNote8208;
787 					break;
788 
789 				case 8224:
790 					olympusMakerParent = tcOlympusMakerNote8224;
791 					break;
792 
793 				case 8240:
794 					olympusMakerParent = tcOlympusMakerNote8240;
795 					break;
796 
797 				case 8256:
798 					olympusMakerParent = tcOlympusMakerNote8256;
799 					break;
800 
801 				case 8272:
802 					olympusMakerParent = tcOlympusMakerNote8272;
803 					break;
804 
805 				case 12288:
806 					olympusMakerParent = tcOlympusMakerNote12288;
807 					break;
808 
809 				default:
810 					break;
811 
812 				}
813 
814 			if (olympusMakerParent)
815 				{
816 
817 				stream.SetReadPosition (tagOffset);
818 
819 				uint64 subMakerNoteOffset = stream.Get_uint32 () + offsetDelta;
820 
821 				if (subMakerNoteOffset >= minOffset &&
822 					subMakerNoteOffset <  maxOffset)
823 					{
824 
825 					if (ParseMakerNoteIFD (host,
826 										   stream,
827 										   maxOffset - subMakerNoteOffset,
828 										   subMakerNoteOffset,
829 										   offsetDelta,
830 										   minOffset,
831 										   maxOffset,
832 										   olympusMakerParent))
833 						{
834 
835 						continue;
836 
837 						}
838 
839 					}
840 
841 				}
842 
843 			stream.SetReadPosition (tagOffset);
844 
845 			}
846 
847 		ParseTag (host,
848 				  stream,
849 				  fExif.Get (),
850 				  fShared.Get (),
851 				  NULL,
852 				  parentCode,
853 				  tagCode,
854 				  tagType,
855 				  tagCount,
856 				  tagOffset,
857 				  offsetDelta);
858 
859 		}
860 
861 	// Grab next IFD pointer, for possible use.
862 
863 	if (ifdSize >= 2 + ifdEntries * 12 + 4)
864 		{
865 
866 		stream.SetReadPosition (ifdOffset + 2 + ifdEntries * 12);
867 
868 		fMakerNoteNextIFD = stream.Get_uint32 ();
869 
870 		}
871 
872 	#if qDNGValidate
873 
874 	if (gVerbose)
875 		{
876 		printf ("\n");
877 		}
878 
879 	#endif
880 
881 	return true;
882 
883 	}
884 
885 /*****************************************************************************/
886 
887 void dng_info::ParseMakerNote (dng_host &host,
888 							   dng_stream &stream,
889 							   uint32 makerNoteCount,
890 							   uint64 makerNoteOffset,
891 							   int64 offsetDelta,
892 							   uint64 minOffset,
893 							   uint64 maxOffset)
894 	{
895 
896 	uint8 firstBytes [16];
897 
898 	memset (firstBytes, 0, sizeof (firstBytes));
899 
900 	stream.SetReadPosition (makerNoteOffset);
901 
902 	stream.Get (firstBytes, (uint32) Min_uint64 (sizeof (firstBytes),
903 												 makerNoteCount));
904 
905 	// Epson MakerNote with header.
906 
907 	if (memcmp (firstBytes, "EPSON\000\001\000", 8) == 0)
908 		{
909 
910 		if (makerNoteCount > 8)
911 			{
912 
913 			ParseMakerNoteIFD (host,
914 							   stream,
915 							   makerNoteCount - 8,
916 				   	  		   makerNoteOffset + 8,
917 				   	  		   offsetDelta,
918 				   	  		   minOffset,
919 				   	  		   maxOffset,
920 				   	  		   tcEpsonMakerNote);
921 
922 			}
923 
924 		return;
925 
926 		}
927 
928 	// Fujifilm MakerNote.
929 
930 	if (memcmp (firstBytes, "FUJIFILM", 8) == 0)
931 		{
932 
933 		stream.SetReadPosition (makerNoteOffset + 8);
934 
935 		TempLittleEndian tempEndian (stream);
936 
937 		uint32 ifd_offset = stream.Get_uint32 ();
938 
939 		if (ifd_offset >= 12 && ifd_offset < makerNoteCount)
940 			{
941 
942 			ParseMakerNoteIFD (host,
943 							   stream,
944 							   makerNoteCount - ifd_offset,
945 							   makerNoteOffset + ifd_offset,
946 							   makerNoteOffset,
947 							   minOffset,
948 							   maxOffset,
949 							   tcFujiMakerNote);
950 
951 			}
952 
953 		return;
954 
955 		}
956 
957 	// Leica MakerNote for models that store entry offsets relative to the start of
958 	// the MakerNote (e.g., M9).
959 
960 	if ((memcmp (firstBytes, "LEICA\000\000\000", 8) == 0) ||
961 		(memcmp (firstBytes, "LEICA0\003\000",	  8) == 0) ||
962 		(memcmp (firstBytes, "LEICA\000\001\000", 8) == 0) ||
963 		(memcmp (firstBytes, "LEICA\000\005\000", 8) == 0))
964 		{
965 
966 		if (makerNoteCount > 8)
967 			{
968 
969 			ParseMakerNoteIFD (host,
970 							   stream,
971 							   makerNoteCount - 8,
972 							   makerNoteOffset + 8,
973 							   makerNoteOffset,
974 							   minOffset,
975 							   maxOffset,
976 							   tcLeicaMakerNote);
977 
978 			}
979 
980 		return;
981 
982 		}
983 
984 	// Leica MakerNote for models that store absolute entry offsets (i.e., relative
985 	// to the start of the file, e.g., S2).
986 
987 	if (memcmp (firstBytes, "LEICA\000\002\377", 8) == 0)
988 		{
989 
990 		if (makerNoteCount > 8)
991 			{
992 
993 			ParseMakerNoteIFD (host,
994 							   stream,
995 							   makerNoteCount - 8,
996 							   makerNoteOffset + 8,
997 							   offsetDelta,
998 							   minOffset,
999 							   maxOffset,
1000 							   tcLeicaMakerNote);
1001 
1002 			}
1003 
1004 		return;
1005 
1006 		}
1007 
1008 	// Nikon version 2 MakerNote with header.
1009 
1010 	if (memcmp (firstBytes, "Nikon\000\002", 7) == 0)
1011 		{
1012 
1013 		stream.SetReadPosition (makerNoteOffset + 10);
1014 
1015 		bool bigEndian = false;
1016 
1017 		uint16 endianMark = stream.Get_uint16 ();
1018 
1019 		if (endianMark == byteOrderMM)
1020 			{
1021 			bigEndian = true;
1022 			}
1023 
1024 		else if (endianMark != byteOrderII)
1025 			{
1026 			return;
1027 			}
1028 
1029 		TempBigEndian temp_endian (stream, bigEndian);
1030 
1031 		uint16 magic = stream.Get_uint16 ();
1032 
1033 		if (magic != 42)
1034 			{
1035 			return;
1036 			}
1037 
1038 		uint32 ifd_offset = stream.Get_uint32 ();
1039 
1040 		if (ifd_offset >= 8 && ifd_offset < makerNoteCount - 10)
1041 			{
1042 
1043 			ParseMakerNoteIFD (host,
1044 							   stream,
1045 							   makerNoteCount - 10 - ifd_offset,
1046 							   makerNoteOffset + 10 + ifd_offset,
1047 							   makerNoteOffset + 10,
1048 							   minOffset,
1049 							   maxOffset,
1050 							   tcNikonMakerNote);
1051 
1052 			}
1053 
1054 		return;
1055 
1056 		}
1057 
1058 	// Newer version of Olympus MakerNote with byte order mark.
1059 
1060 	if (memcmp (firstBytes, "OLYMPUS\000", 8) == 0)
1061 		{
1062 
1063 		stream.SetReadPosition (makerNoteOffset + 8);
1064 
1065 		bool bigEndian = false;
1066 
1067 		uint16 endianMark = stream.Get_uint16 ();
1068 
1069 		if (endianMark == byteOrderMM)
1070 			{
1071 			bigEndian = true;
1072 			}
1073 
1074 		else if (endianMark != byteOrderII)
1075 			{
1076 			return;
1077 			}
1078 
1079 		TempBigEndian temp_endian (stream, bigEndian);
1080 
1081 		uint16 version = stream.Get_uint16 ();
1082 
1083 		if (version != 3)
1084 			{
1085 			return;
1086 			}
1087 
1088 		if (makerNoteCount > 12)
1089 			{
1090 
1091 			ParseMakerNoteIFD (host,
1092 							   stream,
1093 							   makerNoteCount - 12,
1094 				   	  		   makerNoteOffset + 12,
1095 				   	  		   makerNoteOffset,
1096 				   	  		   minOffset,
1097 				   	  		   maxOffset,
1098 				   	  		   tcOlympusMakerNote);
1099 
1100 			}
1101 
1102 		return;
1103 
1104 		}
1105 
1106 	// Olympus MakerNote with header.
1107 
1108 	if (memcmp (firstBytes, "OLYMP", 5) == 0)
1109 		{
1110 
1111 		if (makerNoteCount > 8)
1112 			{
1113 
1114 			ParseMakerNoteIFD (host,
1115 							   stream,
1116 							   makerNoteCount - 8,
1117 				   	  		   makerNoteOffset + 8,
1118 				   	  		   offsetDelta,
1119 				   	  		   minOffset,
1120 				   	  		   maxOffset,
1121 				   	  		   tcOlympusMakerNote);
1122 
1123 			}
1124 
1125 		return;
1126 
1127 		}
1128 
1129 	// Panasonic MakerNote.
1130 
1131 	if (memcmp (firstBytes, "Panasonic\000\000\000", 12) == 0)
1132 		{
1133 
1134 		if (makerNoteCount > 12)
1135 			{
1136 
1137 			ParseMakerNoteIFD (host,
1138 							   stream,
1139 							   makerNoteCount - 12,
1140 							   makerNoteOffset + 12,
1141 							   offsetDelta,
1142 							   minOffset,
1143 							   maxOffset,
1144 							   tcPanasonicMakerNote);
1145 
1146 			}
1147 
1148 		return;
1149 
1150 		}
1151 
1152 	// Pentax MakerNote.
1153 
1154 	if (memcmp (firstBytes, "AOC", 4) == 0)
1155 		{
1156 
1157 		if (makerNoteCount > 6)
1158 			{
1159 
1160 			stream.SetReadPosition (makerNoteOffset + 4);
1161 
1162 			bool bigEndian = stream.BigEndian ();
1163 
1164 			uint16 endianMark = stream.Get_uint16 ();
1165 
1166 			if (endianMark == byteOrderMM)
1167 				{
1168 				bigEndian = true;
1169 				}
1170 
1171 			else if (endianMark == byteOrderII)
1172 				{
1173 				bigEndian = false;
1174 				}
1175 
1176 			TempBigEndian temp_endian (stream, bigEndian);
1177 
1178 			ParseMakerNoteIFD (host,
1179 							   stream,
1180 							   makerNoteCount - 6,
1181 							   makerNoteOffset + 6,
1182 							   offsetDelta,
1183 							   minOffset,
1184 							   maxOffset,
1185 							   tcPentaxMakerNote);
1186 
1187 			}
1188 
1189 		return;
1190 
1191 		}
1192 
1193 	// Ricoh MakerNote.
1194 
1195 	if (memcmp (firstBytes, "RICOH", 5) == 0 ||
1196 		memcmp (firstBytes, "Ricoh", 5) == 0)
1197 		{
1198 
1199 		if (makerNoteCount > 8)
1200 			{
1201 
1202 			TempBigEndian tempEndian (stream);
1203 
1204 			ParseMakerNoteIFD (host,
1205 							   stream,
1206 							   makerNoteCount - 8,
1207 				   	  		   makerNoteOffset + 8,
1208 				   	  		   offsetDelta,
1209 				   	  		   minOffset,
1210 				   	  		   maxOffset,
1211 				   	  		   tcRicohMakerNote);
1212 
1213 			}
1214 
1215 		return;
1216 
1217 		}
1218 
1219 	// Nikon MakerNote without header.
1220 
1221 	if (fExif->fMake.StartsWith ("NIKON"))
1222 		{
1223 
1224 		ParseMakerNoteIFD (host,
1225 						   stream,
1226 						   makerNoteCount,
1227 			   	  		   makerNoteOffset,
1228 			   	  		   offsetDelta,
1229 			   	  		   minOffset,
1230 			   	  		   maxOffset,
1231 			   	  		   tcNikonMakerNote);
1232 
1233 		return;
1234 
1235 		}
1236 
1237 	// Canon MakerNote.
1238 
1239 	if (fExif->fMake.StartsWith ("CANON"))
1240 		{
1241 
1242 		ParseMakerNoteIFD (host,
1243 						   stream,
1244 						   makerNoteCount,
1245 			   	  		   makerNoteOffset,
1246 			   	  		   offsetDelta,
1247 			   	  		   minOffset,
1248 			   	  		   maxOffset,
1249 			   	  		   tcCanonMakerNote);
1250 
1251 		return;
1252 
1253 		}
1254 
1255 	// Minolta MakerNote.
1256 
1257 	if (fExif->fMake.StartsWith ("MINOLTA"       ) ||
1258 		fExif->fMake.StartsWith ("KONICA MINOLTA"))
1259 		{
1260 
1261 		ParseMakerNoteIFD (host,
1262 						   stream,
1263 						   makerNoteCount,
1264 						   makerNoteOffset,
1265 						   offsetDelta,
1266 						   minOffset,
1267 						   maxOffset,
1268 						   tcMinoltaMakerNote);
1269 
1270 		return;
1271 
1272 		}
1273 
1274 	// Sony MakerNote.
1275 
1276 	if (fExif->fMake.StartsWith ("SONY"))
1277 		{
1278 
1279 		ParseMakerNoteIFD (host,
1280 						   stream,
1281 						   makerNoteCount,
1282 						   makerNoteOffset,
1283 						   offsetDelta,
1284 						   minOffset,
1285 						   maxOffset,
1286 						   tcSonyMakerNote);
1287 
1288 		return;
1289 
1290 		}
1291 
1292 	// Kodak MakerNote.
1293 
1294 	if (fExif->fMake.StartsWith ("EASTMAN KODAK"))
1295 		{
1296 
1297 		ParseMakerNoteIFD (host,
1298 						   stream,
1299 						   makerNoteCount,
1300 			   	  		   makerNoteOffset,
1301 			   	  		   offsetDelta,
1302 			   	  		   minOffset,
1303 			   	  		   maxOffset,
1304 			   	  		   tcKodakMakerNote);
1305 
1306 		return;
1307 
1308 		}
1309 
1310 	// Mamiya MakerNote.
1311 
1312 	if (fExif->fMake.StartsWith ("Mamiya"))
1313 		{
1314 
1315 		ParseMakerNoteIFD (host,
1316 						   stream,
1317 						   makerNoteCount,
1318 			   	  		   makerNoteOffset,
1319 			   	  		   offsetDelta,
1320 			   	  		   minOffset,
1321 			   	  		   maxOffset,
1322 			   	  		   tcMamiyaMakerNote);
1323 
1324 		// Mamiya uses a MakerNote chain.
1325 
1326 		while (fMakerNoteNextIFD)
1327 			{
1328 
1329 			ParseMakerNoteIFD (host,
1330 							   stream,
1331 							   makerNoteCount,
1332 							   offsetDelta + fMakerNoteNextIFD,
1333 							   offsetDelta,
1334 							   minOffset,
1335 							   maxOffset,
1336 							   tcMamiyaMakerNote);
1337 
1338 			}
1339 
1340 		return;
1341 
1342 		}
1343 
1344 	// Nikon MakerNote without header.
1345 
1346 	if (fExif->fMake.StartsWith ("Hasselblad"))
1347 		{
1348 
1349 		ParseMakerNoteIFD (host,
1350 						   stream,
1351 						   makerNoteCount,
1352 			   	  		   makerNoteOffset,
1353 			   	  		   offsetDelta,
1354 			   	  		   minOffset,
1355 			   	  		   maxOffset,
1356 			   	  		   tcHasselbladMakerNote);
1357 
1358 		return;
1359 
1360 		}
1361 
1362 	// Samsung MakerNote.
1363 
1364 	if (fExif->fMake.StartsWith ("Samsung"))
1365 		{
1366 
1367 		ParseMakerNoteIFD (host,
1368 						   stream,
1369 						   makerNoteCount,
1370 						   makerNoteOffset,
1371 						   makerNoteOffset,
1372 						   minOffset,
1373 						   maxOffset,
1374 						   tcSamsungMakerNote);
1375 
1376 		return;
1377 
1378 		}
1379 
1380 	// Casio MakerNote.
1381 
1382 	if (fExif->fMake.StartsWith ("CASIO COMPUTER") &&
1383 		memcmp (firstBytes, "QVC\000\000\000", 6) == 0)
1384 		{
1385 
1386 		ParseMakerNoteIFD (host,
1387 						   stream,
1388 						   makerNoteCount - 6,
1389 						   makerNoteOffset + 6,
1390 						   makerNoteOffset,
1391 						   minOffset,
1392 						   maxOffset,
1393 						   tcCasioMakerNote);
1394 
1395 		return;
1396 
1397 		}
1398 
1399 	}
1400 
1401 /*****************************************************************************/
1402 
1403 void dng_info::ParseSonyPrivateData (dng_host & /* host */,
1404 									 dng_stream & /* stream */,
1405 									 uint64 /* count */,
1406 									 uint64 /* oldOffset */,
1407 									 uint64 /* newOffset */)
1408 	{
1409 
1410 	// Sony private data is encrypted, sorry.
1411 
1412 	}
1413 
1414 /*****************************************************************************/
1415 
1416 void dng_info::ParseDNGPrivateData (dng_host &host,
1417 									dng_stream &stream)
1418 	{
1419 
1420 	if (fShared->fDNGPrivateDataCount < 2)
1421 		{
1422 		return;
1423 		}
1424 
1425 	// DNG private data should always start with a null-terminated
1426 	// company name, to define the format of the private data.
1427 
1428 	dng_string privateName;
1429 
1430 		{
1431 
1432 		char buffer [64];
1433 
1434 		stream.SetReadPosition (fShared->fDNGPrivateDataOffset);
1435 
1436 		uint32 readLength = Min_uint32 (fShared->fDNGPrivateDataCount,
1437 										sizeof (buffer) - 1);
1438 
1439 		stream.Get (buffer, readLength);
1440 
1441 		buffer [readLength] = 0;
1442 
1443 		privateName.Set (buffer);
1444 
1445 		}
1446 
1447 	// Pentax is storing their MakerNote in the DNGPrivateData data.
1448 
1449 	if (privateName.StartsWith ("PENTAX" ) ||
1450 		privateName.StartsWith ("SAMSUNG"))
1451 		{
1452 
1453 		#if qDNGValidate
1454 
1455 		if (gVerbose)
1456 			{
1457 			printf ("Parsing Pentax/Samsung DNGPrivateData\n\n");
1458 			}
1459 
1460 		#endif
1461 
1462 		stream.SetReadPosition (fShared->fDNGPrivateDataOffset + 8);
1463 
1464 		bool bigEndian = stream.BigEndian ();
1465 
1466 		uint16 endianMark = stream.Get_uint16 ();
1467 
1468 		if (endianMark == byteOrderMM)
1469 			{
1470 			bigEndian = true;
1471 			}
1472 
1473 		else if (endianMark == byteOrderII)
1474 			{
1475 			bigEndian = false;
1476 			}
1477 
1478 		TempBigEndian temp_endian (stream, bigEndian);
1479 
1480 		ParseMakerNoteIFD (host,
1481 						   stream,
1482 						   fShared->fDNGPrivateDataCount - 10,
1483 						   fShared->fDNGPrivateDataOffset + 10,
1484 						   fShared->fDNGPrivateDataOffset,
1485 						   fShared->fDNGPrivateDataOffset,
1486 						   fShared->fDNGPrivateDataOffset + fShared->fDNGPrivateDataCount,
1487 						   tcPentaxMakerNote);
1488 
1489 		return;
1490 
1491 		}
1492 
1493 	// Stop parsing if this is not an Adobe format block.
1494 
1495 	if (!privateName.Matches ("Adobe"))
1496 		{
1497 		return;
1498 		}
1499 
1500 	TempBigEndian temp_order (stream);
1501 
1502 	uint32 section_offset = 6;
1503 
1504 	while (SafeUint32Add(section_offset, 8) < fShared->fDNGPrivateDataCount)
1505 		{
1506 
1507 		stream.SetReadPosition (SafeUint64Add(fShared->fDNGPrivateDataOffset,
1508 												section_offset));
1509 
1510 		uint32 section_key   = stream.Get_uint32 ();
1511 		uint32 section_count = stream.Get_uint32 ();
1512 
1513 		if (section_key == DNG_CHAR4 ('M','a','k','N') && section_count > 6)
1514 			{
1515 
1516 			#if qDNGValidate
1517 
1518 			if (gVerbose)
1519 				{
1520 				printf ("Found MakerNote inside DNGPrivateData\n\n");
1521 				}
1522 
1523 			#endif
1524 
1525 			uint16 order_mark = stream.Get_uint16 ();
1526 			int64 old_offset = stream.Get_uint32 ();
1527 
1528 			uint32 tempSize = SafeUint32Sub(section_count, 6);
1529 
1530 			AutoPtr<dng_memory_block> tempBlock (host.Allocate (tempSize));
1531 
1532 			uint64 positionInOriginalFile = stream.PositionInOriginalFile();
1533 
1534 			stream.Get (tempBlock->Buffer (), tempSize);
1535 
1536 			dng_stream tempStream (tempBlock->Buffer (),
1537 								   tempSize,
1538 								   positionInOriginalFile);
1539 
1540 			tempStream.SetBigEndian (order_mark == byteOrderMM);
1541 
1542 			ParseMakerNote (host,
1543 							tempStream,
1544 							tempSize,
1545 							0,
1546 							0 - old_offset,
1547 							0,
1548 							tempSize);
1549 
1550 			}
1551 
1552 		else if (section_key == DNG_CHAR4 ('S','R','2',' ') && section_count > 6)
1553 			{
1554 
1555 			#if qDNGValidate
1556 
1557 			if (gVerbose)
1558 				{
1559 				printf ("Found Sony private data inside DNGPrivateData\n\n");
1560 				}
1561 
1562 			#endif
1563 
1564 			uint16 order_mark = stream.Get_uint16 ();
1565 			uint64 old_offset = stream.Get_uint32 ();
1566 
1567 			uint64 new_offset = fShared->fDNGPrivateDataOffset + section_offset + 14;
1568 
1569 			TempBigEndian sr2_order (stream, order_mark == byteOrderMM);
1570 
1571 			ParseSonyPrivateData (host,
1572 							  	  stream,
1573 								  section_count - 6,
1574 								  old_offset,
1575 								  new_offset);
1576 
1577 			}
1578 
1579 		else if (section_key == DNG_CHAR4 ('R','A','F',' ') && section_count > 4)
1580 			{
1581 
1582 			#if qDNGValidate
1583 
1584 			if (gVerbose)
1585 				{
1586 				printf ("Found Fuji RAF tags inside DNGPrivateData\n\n");
1587 				}
1588 
1589 			#endif
1590 
1591 			uint16 order_mark = stream.Get_uint16 ();
1592 
1593 			uint32 tagCount = stream.Get_uint32 ();
1594 
1595 			uint64 tagOffset = stream.Position ();
1596 
1597 			if (tagCount)
1598 				{
1599 
1600 				TempBigEndian raf_order (stream, order_mark == byteOrderMM);
1601 
1602 				ParseTag (host,
1603 						  stream,
1604 						  fExif.Get (),
1605 						  fShared.Get (),
1606 						  NULL,
1607 						  tcFujiRAF,
1608 						  tcFujiHeader,
1609 						  ttUndefined,
1610 						  tagCount,
1611 						  tagOffset,
1612 						  0);
1613 
1614 				stream.SetReadPosition (SafeUint64Add(tagOffset, tagCount));
1615 
1616 				}
1617 
1618 			tagCount = stream.Get_uint32 ();
1619 
1620 			tagOffset = stream.Position ();
1621 
1622 			if (tagCount)
1623 				{
1624 
1625 				TempBigEndian raf_order (stream, order_mark == byteOrderMM);
1626 
1627 				ParseTag (host,
1628 						  stream,
1629 						  fExif.Get (),
1630 						  fShared.Get (),
1631 						  NULL,
1632 						  tcFujiRAF,
1633 						  tcFujiRawInfo1,
1634 						  ttUndefined,
1635 						  tagCount,
1636 						  tagOffset,
1637 						  0);
1638 
1639 				stream.SetReadPosition (SafeUint64Add(tagOffset, tagCount));
1640 
1641 				}
1642 
1643 			tagCount = stream.Get_uint32 ();
1644 
1645 			tagOffset = stream.Position ();
1646 
1647 			if (tagCount)
1648 				{
1649 
1650 				TempBigEndian raf_order (stream, order_mark == byteOrderMM);
1651 
1652 				ParseTag (host,
1653 						  stream,
1654 						  fExif.Get (),
1655 						  fShared.Get (),
1656 						  NULL,
1657 						  tcFujiRAF,
1658 						  tcFujiRawInfo2,
1659 						  ttUndefined,
1660 						  tagCount,
1661 						  tagOffset,
1662 						  0);
1663 
1664 				stream.SetReadPosition (SafeUint64Add(tagOffset, tagCount));
1665 
1666 				}
1667 
1668 			}
1669 
1670 		else if (section_key == DNG_CHAR4 ('C','n','t','x') && section_count > 4)
1671 			{
1672 
1673 			#if qDNGValidate
1674 
1675 			if (gVerbose)
1676 				{
1677 				printf ("Found Contax Raw header inside DNGPrivateData\n\n");
1678 				}
1679 
1680 			#endif
1681 
1682 			uint16 order_mark = stream.Get_uint16 ();
1683 
1684 			uint32 tagCount  = stream.Get_uint32 ();
1685 
1686 			uint64 tagOffset = stream.Position ();
1687 
1688 			if (tagCount)
1689 				{
1690 
1691 				TempBigEndian contax_order (stream, order_mark == byteOrderMM);
1692 
1693 				ParseTag (host,
1694 						  stream,
1695 						  fExif.Get (),
1696 						  fShared.Get (),
1697 						  NULL,
1698 						  tcContaxRAW,
1699 						  tcContaxHeader,
1700 						  ttUndefined,
1701 						  tagCount,
1702 						  tagOffset,
1703 						  0);
1704 
1705 				}
1706 
1707 			}
1708 
1709 		else if (section_key == DNG_CHAR4 ('C','R','W',' ') && section_count > 4)
1710 			{
1711 
1712 			#if qDNGValidate
1713 
1714 			if (gVerbose)
1715 				{
1716 				printf ("Found Canon CRW tags inside DNGPrivateData\n\n");
1717 				}
1718 
1719 			#endif
1720 
1721 			uint16 order_mark = stream.Get_uint16 ();
1722 			uint32 entries    = stream.Get_uint16 ();
1723 
1724 			uint64 crwTagStart = stream.Position ();
1725 
1726 			for (uint32 parsePass = 1; parsePass <= 2; parsePass++)
1727 				{
1728 
1729 				stream.SetReadPosition (crwTagStart);
1730 
1731 				for (uint32 index = 0; index < entries; index++)
1732 					{
1733 
1734 					uint32 tagCode = stream.Get_uint16 ();
1735 
1736 					uint32 tagCount = stream.Get_uint32 ();
1737 
1738 					uint64 tagOffset = stream.Position ();
1739 
1740 					// We need to grab the model id tag first, and then all the
1741 					// other tags.
1742 
1743 					if ((parsePass == 1) == (tagCode == 0x5834))
1744 						{
1745 
1746 						TempBigEndian tag_order (stream, order_mark == byteOrderMM);
1747 
1748 						ParseTag (host,
1749 								  stream,
1750 								  fExif.Get (),
1751 								  fShared.Get (),
1752 								  NULL,
1753 								  tcCanonCRW,
1754 								  tagCode,
1755 								  ttUndefined,
1756 								  tagCount,
1757 								  tagOffset,
1758 								  0);
1759 
1760 						}
1761 
1762 					stream.SetReadPosition (tagOffset + tagCount);
1763 
1764 					}
1765 
1766 				}
1767 
1768 			}
1769 
1770 		else if (section_count > 4)
1771 			{
1772 
1773 			uint32 parentCode = 0;
1774 
1775 			bool code32  = false;
1776 			bool hasType = true;
1777 
1778 			switch (section_key)
1779 				{
1780 
1781 				case DNG_CHAR4 ('M','R','W',' '):
1782 					{
1783 					parentCode = tcMinoltaMRW;
1784 					code32     = true;
1785 					hasType    = false;
1786 					break;
1787 					}
1788 
1789 				case DNG_CHAR4 ('P','a','n','o'):
1790 					{
1791 					parentCode = tcPanasonicRAW;
1792 					break;
1793 					}
1794 
1795 				case DNG_CHAR4 ('L','e','a','f'):
1796 					{
1797 					parentCode = tcLeafMOS;
1798 					break;
1799 					}
1800 
1801 				case DNG_CHAR4 ('K','o','d','a'):
1802 					{
1803 					parentCode = tcKodakDCRPrivateIFD;
1804 					break;
1805 					}
1806 
1807 				case DNG_CHAR4 ('K','D','C',' '):
1808 					{
1809 					parentCode = tcKodakKDCPrivateIFD;
1810 					break;
1811 					}
1812 
1813 				default:
1814 					break;
1815 
1816 				}
1817 
1818 			if (parentCode)
1819 				{
1820 
1821 				#if qDNGValidate
1822 
1823 				if (gVerbose)
1824 					{
1825 					printf ("Found %s tags inside DNGPrivateData\n\n",
1826 							LookupParentCode (parentCode));
1827 					}
1828 
1829 				#endif
1830 
1831 				uint16 order_mark = stream.Get_uint16 ();
1832 				uint32 entries    = stream.Get_uint16 ();
1833 
1834 				for (uint32 index = 0; index < entries; index++)
1835 					{
1836 
1837 					uint32 tagCode = code32 ? stream.Get_uint32 ()
1838 											: stream.Get_uint16 ();
1839 
1840 					uint32 tagType  = hasType ? stream.Get_uint16 ()
1841 											  : ttUndefined;
1842 
1843 					uint32 tagCount = stream.Get_uint32 ();
1844 
1845 					uint32 tagSize = SafeUint32Mult(tagCount, TagTypeSize (tagType));
1846 
1847 					uint64 tagOffset = stream.Position ();
1848 
1849 					TempBigEndian tag_order (stream, order_mark == byteOrderMM);
1850 
1851 					ParseTag (host,
1852 							  stream,
1853 							  fExif.Get (),
1854 							  fShared.Get (),
1855 							  NULL,
1856 							  parentCode,
1857 							  tagCode,
1858 							  tagType,
1859 							  tagCount,
1860 							  tagOffset,
1861 							  0);
1862 
1863 					stream.SetReadPosition (SafeUint64Add(tagOffset, tagSize));
1864 
1865 					}
1866 
1867 				}
1868 
1869 			}
1870 
1871 		section_offset = SafeUint32Add(section_offset, 8);
1872 		section_offset = SafeUint32Add(section_offset, section_count);
1873 
1874 		if (section_offset & 1)
1875 			{
1876 			section_offset = SafeUint32Add(section_offset, 1);
1877 			}
1878 
1879 		}
1880 
1881 	}
1882 
1883 /*****************************************************************************/
1884 
1885 void dng_info::Parse (dng_host &host,
1886 					  dng_stream &stream)
1887 	{
1888 
1889 	fTIFFBlockOffset = stream.Position ();
1890 
1891 	fTIFFBlockOriginalOffset = stream.PositionInOriginalFile ();
1892 
1893 	// Check byte order indicator.
1894 
1895 	uint16 byteOrder = stream.Get_uint16 ();
1896 
1897 	if (byteOrder == byteOrderII)
1898 		{
1899 
1900 		fBigEndian = false;
1901 
1902 		#if qDNGValidate
1903 
1904 		if (gVerbose)
1905 			{
1906 			printf ("\nUses little-endian byte order\n");
1907 			}
1908 
1909 		#endif
1910 
1911 		stream.SetLittleEndian ();
1912 
1913 		}
1914 
1915 	else if (byteOrder == byteOrderMM)
1916 		{
1917 
1918 		fBigEndian = true;
1919 
1920 		#if qDNGValidate
1921 
1922 		if (gVerbose)
1923 			{
1924 			printf ("\nUses big-endian byte order\n");
1925 			}
1926 
1927 		#endif
1928 
1929 		stream.SetBigEndian ();
1930 
1931 		}
1932 
1933 	else
1934 		{
1935 
1936 		#if qDNGValidate
1937 
1938 		ReportError ("Unknown byte order");
1939 
1940 		#endif
1941 
1942 		ThrowBadFormat ();
1943 
1944 		}
1945 
1946 	// Check "magic number" indicator.
1947 
1948 	fMagic = stream.Get_uint16 ();
1949 
1950 	#if qDNGValidate
1951 
1952 	if (gVerbose)
1953 		{
1954 		printf ("Magic number = %u\n\n", (unsigned) fMagic);
1955 		}
1956 
1957 	#endif
1958 
1959 	ValidateMagic ();
1960 
1961 	// Parse IFD 0.
1962 
1963 	uint64 next_offset = stream.Get_uint32 ();
1964 
1965 	fExif.Reset (host.Make_dng_exif ());
1966 
1967 	fShared.Reset (host.Make_dng_shared ());
1968 
1969 	fIFD [0].Reset (host.Make_dng_ifd ());
1970 
1971 	ParseIFD (host,
1972 			  stream,
1973 			  fExif.Get (),
1974 			  fShared.Get (),
1975 			  fIFD [0].Get (),
1976 			  fTIFFBlockOffset + next_offset,
1977 			  fTIFFBlockOffset,
1978 			  0);
1979 
1980 	next_offset = fIFD [0]->fNextIFD;
1981 
1982 	fIFDCount = 1;
1983 
1984 	// Parse chained IFDs.
1985 
1986 	while (next_offset)
1987 		{
1988 
1989 		if (next_offset >= stream.Length ())
1990 			{
1991 
1992 			#if qDNGValidate
1993 
1994 				{
1995 
1996 				ReportWarning ("Chained IFD offset past end of stream");
1997 
1998 				}
1999 
2000 			#endif
2001 
2002 			break;
2003 
2004 			}
2005 
2006 		// Some TIFF file writers forget about the next IFD offset, so
2007 		// validate the IFD at that offset before parsing it.
2008 
2009 		if (!ValidateIFD (stream,
2010 						  fTIFFBlockOffset + next_offset,
2011 						  fTIFFBlockOffset))
2012 			{
2013 
2014 			#if qDNGValidate
2015 
2016 				{
2017 
2018 				ReportWarning ("Chained IFD is not valid");
2019 
2020 				}
2021 
2022 			#endif
2023 
2024 			break;
2025 
2026 			}
2027 
2028 		if (fChainedIFDCount == kMaxChainedIFDs)
2029 			{
2030 
2031 			#if qDNGValidate
2032 
2033 				{
2034 
2035 				ReportWarning ("Chained IFD count exceeds DNG SDK parsing limit");
2036 
2037 				}
2038 
2039 			#endif
2040 
2041 			break;
2042 
2043 			}
2044 
2045 		fChainedIFD [fChainedIFDCount].Reset (host.Make_dng_ifd ());
2046 
2047 		ParseIFD (host,
2048 				  stream,
2049 				  NULL,
2050 				  NULL,
2051 				  fChainedIFD [fChainedIFDCount].Get (),
2052 				  fTIFFBlockOffset + next_offset,
2053 				  fTIFFBlockOffset,
2054 				  tcFirstChainedIFD + fChainedIFDCount);
2055 
2056 		next_offset = fChainedIFD [fChainedIFDCount]->fNextIFD;
2057 
2058 		fChainedIFDCount++;
2059 
2060 		}
2061 
2062 	// Parse SubIFDs.
2063 
2064 	uint32 searchedIFDs = 0;
2065 
2066 	bool tooManySubIFDs = false;
2067 
2068 	while (searchedIFDs < fIFDCount && !tooManySubIFDs)
2069 		{
2070 
2071 		uint32 searchLimit = fIFDCount;
2072 
2073 		for (uint32 searchIndex = searchedIFDs;
2074 			 searchIndex < searchLimit && !tooManySubIFDs;
2075 			 searchIndex++)
2076 			{
2077 
2078 			for (uint32 subIndex = 0;
2079 			     subIndex < fIFD [searchIndex]->fSubIFDsCount;
2080 			     subIndex++)
2081 				{
2082 
2083 				if (fIFDCount == kMaxSubIFDs + 1)
2084 					{
2085 
2086 					tooManySubIFDs = true;
2087 
2088 					break;
2089 
2090 					}
2091 
2092 				stream.SetReadPosition (fIFD [searchIndex]->fSubIFDsOffset +
2093 							 			subIndex * 4);
2094 
2095 				uint32 sub_ifd_offset = stream.Get_uint32 ();
2096 
2097 				fIFD [fIFDCount].Reset (host.Make_dng_ifd ());
2098 
2099 				ParseIFD (host,
2100 						  stream,
2101 						  fExif.Get (),
2102 						  fShared.Get (),
2103 						  fIFD [fIFDCount].Get (),
2104 						  fTIFFBlockOffset + sub_ifd_offset,
2105 						  fTIFFBlockOffset,
2106 						  tcFirstSubIFD + fIFDCount - 1);
2107 
2108 				fIFDCount++;
2109 
2110 				}
2111 
2112 			searchedIFDs = searchLimit;
2113 
2114 			}
2115 
2116 		}
2117 
2118 	#if qDNGValidate
2119 
2120 		{
2121 
2122 		if (tooManySubIFDs)
2123 			{
2124 
2125 			ReportWarning ("SubIFD count exceeds DNG SDK parsing limit");
2126 
2127 			}
2128 
2129 		}
2130 
2131 	#endif
2132 
2133 	// Parse EXIF IFD.
2134 
2135 	if (fShared->fExifIFD)
2136 		{
2137 
2138 		ParseIFD (host,
2139 				  stream,
2140 				  fExif.Get (),
2141 				  fShared.Get (),
2142 				  NULL,
2143 				  fTIFFBlockOffset + fShared->fExifIFD,
2144 				  fTIFFBlockOffset,
2145 				  tcExifIFD);
2146 
2147 		}
2148 
2149 	// Parse GPS IFD.
2150 
2151 	if (fShared->fGPSInfo)
2152 		{
2153 
2154 		ParseIFD (host,
2155 				  stream,
2156 				  fExif.Get (),
2157 				  fShared.Get (),
2158 				  NULL,
2159 				  fTIFFBlockOffset + fShared->fGPSInfo,
2160 				  fTIFFBlockOffset,
2161 				  tcGPSInfo);
2162 
2163 		}
2164 
2165 	// Parse Interoperability IFD.
2166 
2167 	if (fShared->fInteroperabilityIFD)
2168 		{
2169 
2170 		// Some Kodak KDC files have bogus Interoperability IFDs, so
2171 		// validate the IFD before trying to parse it.
2172 
2173 		if (ValidateIFD (stream,
2174 						 fTIFFBlockOffset + fShared->fInteroperabilityIFD,
2175 						 fTIFFBlockOffset))
2176 			{
2177 
2178 			ParseIFD (host,
2179 					  stream,
2180 					  fExif.Get (),
2181 					  fShared.Get (),
2182 					  NULL,
2183 					  fTIFFBlockOffset + fShared->fInteroperabilityIFD,
2184 					  fTIFFBlockOffset,
2185 					  tcInteroperabilityIFD);
2186 
2187 			}
2188 
2189 		#if qDNGValidate
2190 
2191 		else
2192 			{
2193 
2194 			ReportWarning ("The Interoperability IFD is not a valid IFD");
2195 
2196 			}
2197 
2198 		#endif
2199 
2200 		}
2201 
2202 	// Parse Kodak DCR Private IFD.
2203 
2204 	if (fShared->fKodakDCRPrivateIFD)
2205 		{
2206 
2207 		ParseIFD (host,
2208 				  stream,
2209 				  fExif.Get (),
2210 				  fShared.Get (),
2211 				  NULL,
2212 				  fTIFFBlockOffset + fShared->fKodakDCRPrivateIFD,
2213 				  fTIFFBlockOffset,
2214 				  tcKodakDCRPrivateIFD);
2215 
2216 		}
2217 
2218 	// Parse Kodak KDC Private IFD.
2219 
2220 	if (fShared->fKodakKDCPrivateIFD)
2221 		{
2222 
2223 		ParseIFD (host,
2224 				  stream,
2225 				  fExif.Get (),
2226 				  fShared.Get (),
2227 				  NULL,
2228 				  fTIFFBlockOffset + fShared->fKodakKDCPrivateIFD,
2229 				  fTIFFBlockOffset,
2230 				  tcKodakKDCPrivateIFD);
2231 
2232 		}
2233 
2234 	// Parse MakerNote tag.
2235 
2236 	if (fShared->fMakerNoteCount)
2237 		{
2238 
2239 		ParseMakerNote (host,
2240 						stream,
2241 						(uint32) (fTIFFBlockOffset + fShared->fMakerNoteCount),
2242 						fShared->fMakerNoteOffset,
2243 						fTIFFBlockOffset,
2244 						0,
2245 						stream.Length ());
2246 
2247 		}
2248 
2249 	// Parse DNGPrivateData tag.
2250 
2251 	if (fShared->fDNGPrivateDataCount &&
2252 		fShared->fDNGVersion)
2253 		{
2254 
2255 		ParseDNGPrivateData (host, stream);
2256 
2257 		}
2258 
2259 	#if qDNGValidate
2260 
2261 	// If we are running dng_validate on stand-alone camera profile file,
2262 	// complete the validation of the profile.
2263 
2264 	if (fMagic == magicExtendedProfile)
2265 		{
2266 
2267 		dng_camera_profile_info &profileInfo = fShared->fCameraProfile;
2268 
2269 		dng_camera_profile profile;
2270 
2271 		profile.Parse (stream, profileInfo);
2272 
2273 		if (profileInfo.fColorPlanes < 3 || !profile.IsValid (profileInfo.fColorPlanes))
2274 			{
2275 
2276 			ReportError ("Invalid camera profile file");
2277 
2278 			}
2279 
2280 		}
2281 
2282 	#endif
2283 
2284 	}
2285 
2286 /*****************************************************************************/
2287 
2288 void dng_info::PostParse (dng_host &host)
2289 	{
2290 
2291 	uint32 index;
2292 
2293 	fExif->PostParse (host, *fShared.Get ());
2294 
2295 	fShared->PostParse (host, *fExif.Get ());
2296 
2297 	for (index = 0; index < fIFDCount; index++)
2298 		{
2299 
2300 		fIFD [index]->PostParse ();
2301 
2302 		}
2303 
2304 	for (index = 0; index < fChainedIFDCount; index++)
2305 		{
2306 
2307 		fChainedIFD [index]->PostParse ();
2308 
2309 		}
2310 
2311 	if (fShared->fDNGVersion != 0)
2312 		{
2313 
2314 		// Find main IFD.
2315 
2316 		fMainIndex = -1;
2317 
2318 		for (index = 0; index < fIFDCount; index++)
2319 			{
2320 
2321 			if (fIFD [index]->fUsesNewSubFileType &&
2322 				fIFD [index]->fNewSubFileType == sfMainImage)
2323 				{
2324 
2325 				if (fMainIndex == -1)
2326 					{
2327 
2328 					fMainIndex = index;
2329 
2330 					}
2331 
2332 				#if qDNGValidate
2333 
2334 				else
2335 					{
2336 
2337 					ReportError ("Multiple IFDs marked as main image");
2338 
2339 					}
2340 
2341 				#endif
2342 
2343 				}
2344 
2345 			else if (fIFD [index]->fNewSubFileType == sfPreviewImage ||
2346 					 fIFD [index]->fNewSubFileType == sfAltPreviewImage)
2347 				{
2348 
2349 				// Fill in default color space for DNG previews if not included.
2350 
2351 				if (fIFD [index]->fPreviewInfo.fColorSpace == previewColorSpace_MaxEnum)
2352 					{
2353 
2354 					if (fIFD [index]->fSamplesPerPixel == 1)
2355 						{
2356 
2357 						fIFD [index]->fPreviewInfo.fColorSpace = previewColorSpace_GrayGamma22;
2358 
2359 						}
2360 
2361 					else
2362 						{
2363 
2364 						fIFD [index]->fPreviewInfo.fColorSpace = previewColorSpace_sRGB;
2365 
2366 						}
2367 
2368 					}
2369 
2370 				}
2371 
2372 			}
2373 
2374 		// Deal with lossless JPEG bug in early DNG versions.
2375 
2376 		if (fShared->fDNGVersion < dngVersion_1_1_0_0)
2377 			{
2378 
2379 			if (fMainIndex != -1)
2380 				{
2381 
2382 				fIFD [fMainIndex]->fLosslessJPEGBug16 = true;
2383 
2384 				}
2385 
2386 			}
2387 
2388 		// Find mask index.
2389 
2390 		for (index = 0; index < fIFDCount; index++)
2391 			{
2392 
2393 			if (fIFD [index]->fNewSubFileType == sfTransparencyMask)
2394 				{
2395 
2396 				if (fMaskIndex == -1)
2397 					{
2398 
2399 					fMaskIndex = index;
2400 
2401 					}
2402 
2403 				#if qDNGValidate
2404 
2405 				else
2406 					{
2407 
2408 					ReportError ("Multiple IFDs marked as transparency mask image");
2409 
2410 					}
2411 
2412 				#endif
2413 
2414 				}
2415 
2416 			}
2417 
2418 		// Warn about Chained IFDs.
2419 
2420 		#if qDNGValidate
2421 
2422 		if (fChainedIFDCount > 0)
2423 			{
2424 
2425 			ReportWarning ("This file has Chained IFDs, which will be ignored by DNG readers");
2426 
2427 			}
2428 
2429 		#endif
2430 
2431 		}
2432 
2433 	}
2434 
2435 /*****************************************************************************/
2436 
2437 bool dng_info::IsValidDNG ()
2438 	{
2439 
2440 	// Check shared info.
2441 
2442 	if (!fShared->IsValidDNG ())
2443 		{
2444 
2445 		return false;
2446 
2447 		}
2448 
2449 	// Check TIFF magic number.
2450 
2451 	if (fMagic != 42)
2452 		{
2453 
2454 		#if qDNGValidate
2455 
2456 		ReportError ("Invalid TIFF magic number");
2457 
2458 		#endif
2459 
2460 		return false;
2461 
2462 		}
2463 
2464 	// Make sure we have a main image IFD.
2465 
2466 	if (fMainIndex == -1)
2467 		{
2468 
2469 		#if qDNGValidate
2470 
2471 		ReportError ("Unable to find main image IFD");
2472 
2473 		#endif
2474 
2475 		return false;
2476 
2477 		}
2478 
2479 	// Make sure is each IFD is valid.
2480 
2481 	for (uint32 index = 0; index < fIFDCount; index++)
2482 		{
2483 
2484 		uint32 parentCode = (index == 0 ? 0 : tcFirstSubIFD + index - 1);
2485 
2486 		if (!fIFD [index]->IsValidDNG (*fShared.Get (),
2487 								       parentCode))
2488 			{
2489 
2490 			// Only errors in the main and transparency mask IFDs are fatal to parsing.
2491 
2492 			if (index == (uint32) fMainIndex ||
2493 				index == (uint32) fMaskIndex)
2494 				{
2495 
2496 				return false;
2497 
2498 				}
2499 
2500 			}
2501 
2502 		}
2503 
2504 	return true;
2505 
2506 	}
2507 
2508 /*****************************************************************************/
2509