Без опису
Ви не можете вибрати більше 25 тем Теми мають розпочинатися з літери або цифри, можуть містити дефіси (-) і не повинні перевищувати 35 символів.

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516
  1. <?php
  2. /////////////////////////////////////////////////////////////////
  3. /// getID3() by James Heinrich <info@getid3.org> //
  4. // available at https://github.com/JamesHeinrich/getID3 //
  5. // or https://www.getid3.org //
  6. // or http://getid3.sourceforge.net //
  7. // see readme.txt for more details //
  8. /////////////////////////////////////////////////////////////////
  9. // //
  10. // module.archive.tiff.php //
  11. // module for analyzing TIFF files //
  12. // dependencies: NONE //
  13. // ///
  14. /////////////////////////////////////////////////////////////////
  15. if (!defined('GETID3_INCLUDEPATH')) { // prevent path-exposing attacks that access modules directly on public webservers
  16. exit;
  17. }
  18. class getid3_tiff extends getid3_handler
  19. {
  20. /**
  21. * @return bool
  22. */
  23. public function Analyze() {
  24. $info = &$this->getid3->info;
  25. $this->fseek($info['avdataoffset']);
  26. $TIFFheader = $this->fread(4);
  27. switch (substr($TIFFheader, 0, 2)) {
  28. case 'II':
  29. $info['tiff']['byte_order'] = 'Intel';
  30. break;
  31. case 'MM':
  32. $info['tiff']['byte_order'] = 'Motorola';
  33. break;
  34. default:
  35. $this->error('Invalid TIFF byte order identifier ('.substr($TIFFheader, 0, 2).') at offset '.$info['avdataoffset']);
  36. return false;
  37. }
  38. $info['fileformat'] = 'tiff';
  39. $info['video']['dataformat'] = 'tiff';
  40. $info['video']['lossless'] = true;
  41. $info['tiff']['ifd'] = array();
  42. $CurrentIFD = array();
  43. $FieldTypeByteLength = array(1=>1, 2=>1, 3=>2, 4=>4, 5=>8);
  44. $nextIFDoffset = $this->TIFFendian2Int($this->fread(4), $info['tiff']['byte_order']);
  45. while ($nextIFDoffset > 0) {
  46. $CurrentIFD['offset'] = $nextIFDoffset;
  47. $this->fseek($info['avdataoffset'] + $nextIFDoffset);
  48. $CurrentIFD['fieldcount'] = $this->TIFFendian2Int($this->fread(2), $info['tiff']['byte_order']);
  49. for ($i = 0; $i < $CurrentIFD['fieldcount']; $i++) {
  50. $CurrentIFD['fields'][$i]['raw']['tag'] = $this->TIFFendian2Int($this->fread(2), $info['tiff']['byte_order']);
  51. $CurrentIFD['fields'][$i]['raw']['type'] = $this->TIFFendian2Int($this->fread(2), $info['tiff']['byte_order']);
  52. $CurrentIFD['fields'][$i]['raw']['length'] = $this->TIFFendian2Int($this->fread(4), $info['tiff']['byte_order']);
  53. $CurrentIFD['fields'][$i]['raw']['valoff'] = $this->fread(4); // To save time and space the Value Offset contains the Value instead of pointing to the Value if and only if the Value fits into 4 bytes. If the Value is shorter than 4 bytes, it is left-justified within the 4-byte Value Offset, i.e., stored in the lowernumbered bytes. Whether the Value fits within 4 bytes is determined by the Type and Count of the field.
  54. $CurrentIFD['fields'][$i]['raw']['tag_name'] = $this->TIFFcommentName($CurrentIFD['fields'][$i]['raw']['tag']);
  55. switch ($CurrentIFD['fields'][$i]['raw']['type']) {
  56. case 1: // BYTE An 8-bit unsigned integer.
  57. if ($CurrentIFD['fields'][$i]['raw']['length'] <= 4) {
  58. $CurrentIFD['fields'][$i]['value'] = $this->TIFFendian2Int(substr($CurrentIFD['fields'][$i]['raw']['valoff'], 0, 1), $info['tiff']['byte_order']);
  59. } else {
  60. $CurrentIFD['fields'][$i]['offset'] = $this->TIFFendian2Int($CurrentIFD['fields'][$i]['raw']['valoff'], $info['tiff']['byte_order']);
  61. }
  62. break;
  63. case 2: // ASCII 8-bit bytes that store ASCII codes; the last byte must be null.
  64. if ($CurrentIFD['fields'][$i]['raw']['length'] <= 4) {
  65. $CurrentIFD['fields'][$i]['value'] = substr($CurrentIFD['fields'][$i]['raw']['valoff'], 3);
  66. } else {
  67. $CurrentIFD['fields'][$i]['offset'] = $this->TIFFendian2Int($CurrentIFD['fields'][$i]['raw']['valoff'], $info['tiff']['byte_order']);
  68. }
  69. break;
  70. case 3: // SHORT A 16-bit (2-byte) unsigned integer.
  71. if ($CurrentIFD['fields'][$i]['raw']['length'] <= 2) {
  72. $CurrentIFD['fields'][$i]['value'] = $this->TIFFendian2Int(substr($CurrentIFD['fields'][$i]['raw']['valoff'], 0, 2), $info['tiff']['byte_order']);
  73. } else {
  74. $CurrentIFD['fields'][$i]['offset'] = $this->TIFFendian2Int($CurrentIFD['fields'][$i]['raw']['valoff'], $info['tiff']['byte_order']);
  75. }
  76. break;
  77. case 4: // LONG A 32-bit (4-byte) unsigned integer.
  78. if ($CurrentIFD['fields'][$i]['raw']['length'] <= 4) {
  79. $CurrentIFD['fields'][$i]['value'] = $this->TIFFendian2Int($CurrentIFD['fields'][$i]['raw']['valoff'], $info['tiff']['byte_order']);
  80. } else {
  81. $CurrentIFD['fields'][$i]['offset'] = $this->TIFFendian2Int($CurrentIFD['fields'][$i]['raw']['valoff'], $info['tiff']['byte_order']);
  82. }
  83. break;
  84. case 5: // RATIONAL Two LONG_s: the first represents the numerator of a fraction, the second the denominator.
  85. case 7: // UNDEFINED An 8-bit byte that may contain anything, depending on the definition of the field.
  86. $CurrentIFD['fields'][$i]['offset'] = $this->TIFFendian2Int($CurrentIFD['fields'][$i]['raw']['valoff'], $info['tiff']['byte_order']);
  87. break;
  88. // Warning: It is possible that other TIFF field types will be added in the future. Readers should skip over fields containing an unexpected field type.
  89. // In TIFF 6.0, some new field types have been defined:
  90. // These new field types are also governed by the byte order (II or MM) in the TIFF header.
  91. case 6: // SBYTE An 8-bit signed (twos-complement) integer.
  92. case 8: // SSHORT A 16-bit (2-byte) signed (twos-complement) integer.
  93. case 9: // SLONG A 32-bit (4-byte) signed (twos-complement) integer.
  94. case 10: // SRATIONAL Two SLONGs: the first represents the numerator of a fraction, the second the denominator.
  95. case 11: // FLOAT Single precision (4-byte) IEEE format
  96. case 12: // DOUBLE Double precision (8-byte) IEEE format
  97. default:
  98. $this->warning('unhandled IFD field type '.$CurrentIFD['fields'][$i]['raw']['type'].' for IFD entry '.$i);
  99. break;
  100. }
  101. }
  102. $info['tiff']['ifd'][] = $CurrentIFD;
  103. $CurrentIFD = array();
  104. $nextIFDoffset = $this->TIFFendian2Int($this->fread(4), $info['tiff']['byte_order']);
  105. }
  106. foreach ($info['tiff']['ifd'] as $IFDid => $IFDarray) {
  107. foreach ($IFDarray['fields'] as $key => $fieldarray) {
  108. switch ($fieldarray['raw']['tag']) {
  109. case 256: // ImageWidth
  110. case 257: // ImageLength
  111. case 258: // BitsPerSample
  112. case 259: // Compression
  113. if (!isset($fieldarray['value'])) {
  114. $this->fseek($fieldarray['offset']);
  115. $info['tiff']['ifd'][$IFDid]['fields'][$key]['raw']['data'] = $this->fread($fieldarray['raw']['length'] * $FieldTypeByteLength[$fieldarray['raw']['type']]);
  116. }
  117. break;
  118. case 270: // ImageDescription
  119. case 271: // Make
  120. case 272: // Model
  121. case 305: // Software
  122. case 306: // DateTime
  123. case 315: // Artist
  124. case 316: // HostComputer
  125. if (isset($fieldarray['value'])) {
  126. $info['tiff']['ifd'][$IFDid]['fields'][$key]['raw']['data'] = $fieldarray['value'];
  127. } else {
  128. $this->fseek($fieldarray['offset']);
  129. $info['tiff']['ifd'][$IFDid]['fields'][$key]['raw']['data'] = $this->fread($fieldarray['raw']['length'] * $FieldTypeByteLength[$fieldarray['raw']['type']]);
  130. }
  131. break;
  132. case 700:
  133. $XMPmagic = '<?xpacket';
  134. $this->fseek($fieldarray['offset']);
  135. $xmpkey = (isset($info['tiff']['XMP']) ? count($info['tiff']['XMP']) : 0);
  136. $info['tiff']['XMP'][$xmpkey]['raw'] = $this->fread($fieldarray['raw']['length']);
  137. if (substr($info['tiff']['XMP'][$xmpkey]['raw'], 0, strlen($XMPmagic)) != $XMPmagic) {
  138. $this->warning('did not find expected XMP data at offset '.$fieldarray['offset']);
  139. unset($info['tiff']['XMP'][$xmpkey]['raw']);
  140. }
  141. break;
  142. }
  143. switch ($fieldarray['raw']['tag']) {
  144. case 256: // ImageWidth
  145. $info['video']['resolution_x'] = $fieldarray['value'];
  146. break;
  147. case 257: // ImageLength
  148. $info['video']['resolution_y'] = $fieldarray['value'];
  149. break;
  150. case 258: // BitsPerSample
  151. if (isset($fieldarray['value'])) {
  152. $info['video']['bits_per_sample'] = $fieldarray['value'];
  153. } else {
  154. $info['video']['bits_per_sample'] = 0;
  155. for ($i = 0; $i < $fieldarray['raw']['length']; $i++) {
  156. $info['video']['bits_per_sample'] += $this->TIFFendian2Int(substr($info['tiff']['ifd'][$IFDid]['fields'][$key]['raw']['data'], $i * $FieldTypeByteLength[$fieldarray['raw']['type']], $FieldTypeByteLength[$fieldarray['raw']['type']]), $info['tiff']['byte_order']);
  157. }
  158. }
  159. break;
  160. case 259: // Compression
  161. $info['video']['codec'] = $this->TIFFcompressionMethod($fieldarray['value']);
  162. break;
  163. case 270: // ImageDescription
  164. case 271: // Make
  165. case 272: // Model
  166. case 305: // Software
  167. case 306: // DateTime
  168. case 315: // Artist
  169. case 316: // HostComputer
  170. $TIFFcommentName = strtolower($fieldarray['raw']['tag_name']);
  171. if (isset($info['tiff']['comments'][$TIFFcommentName])) {
  172. $info['tiff']['comments'][$TIFFcommentName][] = $info['tiff']['ifd'][$IFDid]['fields'][$key]['raw']['data'];
  173. } else {
  174. $info['tiff']['comments'][$TIFFcommentName] = array($info['tiff']['ifd'][$IFDid]['fields'][$key]['raw']['data']);
  175. }
  176. break;
  177. default:
  178. break;
  179. }
  180. }
  181. }
  182. return true;
  183. }
  184. /**
  185. * @param string $bytestring
  186. * @param string $byteorder
  187. *
  188. * @return int|float|false
  189. */
  190. public function TIFFendian2Int($bytestring, $byteorder) {
  191. if ($byteorder == 'Intel') {
  192. return getid3_lib::LittleEndian2Int($bytestring);
  193. } elseif ($byteorder == 'Motorola') {
  194. return getid3_lib::BigEndian2Int($bytestring);
  195. }
  196. return false;
  197. }
  198. /**
  199. * @param int $id
  200. *
  201. * @return string
  202. */
  203. public function TIFFcompressionMethod($id) {
  204. // https://en.wikipedia.org/wiki/TIFF#TIFF_Compression_Tag
  205. static $TIFFcompressionMethod = array();
  206. if (empty($TIFFcompressionMethod)) {
  207. $TIFFcompressionMethod = array(
  208. 0x0001 => 'Uncompressed',
  209. 0x0002 => 'Huffman',
  210. 0x0003 => 'CCITT T.4',
  211. 0x0004 => 'CCITT T.6',
  212. 0x0005 => 'LZW',
  213. 0x0006 => 'JPEG-old',
  214. 0x0007 => 'JPEG',
  215. 0x0008 => 'deflate',
  216. 0x0009 => 'JBIG ITU-T T.85',
  217. 0x000A => 'JBIG ITU-T T.43',
  218. 0x7FFE => 'NeXT RLE 2-bit',
  219. 0x8005 => 'PackBits',
  220. 0x8029 => 'ThunderScan RLE 4-bit',
  221. 0x807F => 'RasterPadding',
  222. 0x8080 => 'RLE-LW',
  223. 0x8081 => 'RLE-CT',
  224. 0x8082 => 'RLE-BL',
  225. 0x80B2 => 'deflate-PK',
  226. 0x80B3 => 'Kodak-DCS',
  227. 0x8765 => 'JBIG',
  228. 0x8798 => 'JPEG2000',
  229. 0x8799 => 'Nikon NEF',
  230. 0x879B => 'JBIG2',
  231. );
  232. }
  233. return (isset($TIFFcompressionMethod[$id]) ? $TIFFcompressionMethod[$id] : 'unknown/invalid ('.$id.')');
  234. }
  235. /**
  236. * @param int $id
  237. *
  238. * @return string
  239. */
  240. public function TIFFcommentName($id) {
  241. // https://www.awaresystems.be/imaging/tiff/tifftags.html
  242. static $TIFFcommentName = array();
  243. if (empty($TIFFcommentName)) {
  244. $TIFFcommentName = array(
  245. 254 => 'NewSubfileType',
  246. 255 => 'SubfileType',
  247. 256 => 'ImageWidth',
  248. 257 => 'ImageLength',
  249. 258 => 'BitsPerSample',
  250. 259 => 'Compression',
  251. 262 => 'PhotometricInterpretation',
  252. 263 => 'Threshholding',
  253. 264 => 'CellWidth',
  254. 265 => 'CellLength',
  255. 266 => 'FillOrder',
  256. 269 => 'DocumentName',
  257. 270 => 'ImageDescription',
  258. 271 => 'Make',
  259. 272 => 'Model',
  260. 273 => 'StripOffsets',
  261. 274 => 'Orientation',
  262. 277 => 'SamplesPerPixel',
  263. 278 => 'RowsPerStrip',
  264. 279 => 'StripByteCounts',
  265. 280 => 'MinSampleValue',
  266. 281 => 'MaxSampleValue',
  267. 282 => 'XResolution',
  268. 283 => 'YResolution',
  269. 284 => 'PlanarConfiguration',
  270. 285 => 'PageName',
  271. 286 => 'XPosition',
  272. 287 => 'YPosition',
  273. 288 => 'FreeOffsets',
  274. 289 => 'FreeByteCounts',
  275. 290 => 'GrayResponseUnit',
  276. 291 => 'GrayResponseCurve',
  277. 292 => 'T4Options',
  278. 293 => 'T6Options',
  279. 296 => 'ResolutionUnit',
  280. 297 => 'PageNumber',
  281. 301 => 'TransferFunction',
  282. 305 => 'Software',
  283. 306 => 'DateTime',
  284. 315 => 'Artist',
  285. 316 => 'HostComputer',
  286. 317 => 'Predictor',
  287. 318 => 'WhitePoint',
  288. 319 => 'PrimaryChromaticities',
  289. 320 => 'ColorMap',
  290. 321 => 'HalftoneHints',
  291. 322 => 'TileWidth',
  292. 323 => 'TileLength',
  293. 324 => 'TileOffsets',
  294. 325 => 'TileByteCounts',
  295. 326 => 'BadFaxLines',
  296. 327 => 'CleanFaxData',
  297. 328 => 'ConsecutiveBadFaxLines',
  298. 330 => 'SubIFDs',
  299. 332 => 'InkSet',
  300. 333 => 'InkNames',
  301. 334 => 'NumberOfInks',
  302. 336 => 'DotRange',
  303. 337 => 'TargetPrinter',
  304. 338 => 'ExtraSamples',
  305. 339 => 'SampleFormat',
  306. 340 => 'SMinSampleValue',
  307. 341 => 'SMaxSampleValue',
  308. 342 => 'TransferRange',
  309. 343 => 'ClipPath',
  310. 344 => 'XClipPathUnits',
  311. 345 => 'YClipPathUnits',
  312. 346 => 'Indexed',
  313. 347 => 'JPEGTables',
  314. 351 => 'OPIProxy',
  315. 400 => 'GlobalParametersIFD',
  316. 401 => 'ProfileType',
  317. 402 => 'FaxProfile',
  318. 403 => 'CodingMethods',
  319. 404 => 'VersionYear',
  320. 405 => 'ModeNumber',
  321. 433 => 'Decode',
  322. 434 => 'DefaultImageColor',
  323. 512 => 'JPEGProc',
  324. 513 => 'JPEGInterchangeFormat',
  325. 514 => 'JPEGInterchangeFormatLngth',
  326. 515 => 'JPEGRestartInterval',
  327. 517 => 'JPEGLosslessPredictors',
  328. 518 => 'JPEGPointTransforms',
  329. 519 => 'JPEGQTables',
  330. 520 => 'JPEGDCTables',
  331. 521 => 'JPEGACTables',
  332. 529 => 'YCbCrCoefficients',
  333. 530 => 'YCbCrSubSampling',
  334. 531 => 'YCbCrPositioning',
  335. 532 => 'ReferenceBlackWhite',
  336. 559 => 'StripRowCounts',
  337. 700 => 'XMP',
  338. 32781 => 'ImageID',
  339. 33432 => 'Copyright',
  340. 34732 => 'ImageLayer',
  341. // Private Tags - https://www.awaresystems.be/imaging/tiff/tifftags/private.html
  342. 32932 => 'Wang Annotation', // Annotation data, as used in 'Imaging for Windows'.
  343. 33445 => 'MD FileTag', // Specifies the pixel data format encoding in the Molecular Dynamics GEL file format.
  344. 33446 => 'MD ScalePixel', // Specifies a scale factor in the Molecular Dynamics GEL file format.
  345. 33447 => 'MD ColorTable', // Used to specify the conversion from 16bit to 8bit in the Molecular Dynamics GEL file format.
  346. 33448 => 'MD LabName', // Name of the lab that scanned this file, as used in the Molecular Dynamics GEL file format.
  347. 33449 => 'MD SampleInfo', // Information about the sample, as used in the Molecular Dynamics GEL file format.
  348. 33450 => 'MD PrepDate', // Date the sample was prepared, as used in the Molecular Dynamics GEL file format.
  349. 33451 => 'MD PrepTime', // Time the sample was prepared, as used in the Molecular Dynamics GEL file format.
  350. 33452 => 'MD FileUnits', // Units for data in this file, as used in the Molecular Dynamics GEL file format.
  351. 33550 => 'ModelPixelScaleTag', // Used in interchangeable GeoTIFF files.
  352. 33723 => 'IPTC', // IPTC (International Press Telecommunications Council) metadata.
  353. 33918 => 'INGR Packet Data Tag', // Intergraph Application specific storage.
  354. 33919 => 'INGR Flag Registers', // Intergraph Application specific flags.
  355. 33920 => 'IrasB Transformation Matrix', // Originally part of Intergraph's GeoTIFF tags, but likely understood by IrasB only.
  356. 33922 => 'ModelTiepointTag', // Originally part of Intergraph's GeoTIFF tags, but now used in interchangeable GeoTIFF files.
  357. 34264 => 'ModelTransformationTag', // Used in interchangeable GeoTIFF files.
  358. 34377 => 'Photoshop', // Collection of Photoshop 'Image Resource Blocks'.
  359. 34665 => 'Exif IFD', // A pointer to the Exif IFD.
  360. 34675 => 'ICC Profile', // ICC profile data.
  361. 34735 => 'GeoKeyDirectoryTag', // Used in interchangeable GeoTIFF files.
  362. 34736 => 'GeoDoubleParamsTag', // Used in interchangeable GeoTIFF files.
  363. 34737 => 'GeoAsciiParamsTag', // Used in interchangeable GeoTIFF files.
  364. 34853 => 'GPS IFD', // A pointer to the Exif-related GPS Info IFD.
  365. 34908 => 'HylaFAX FaxRecvParams', // Used by HylaFAX.
  366. 34909 => 'HylaFAX FaxSubAddress', // Used by HylaFAX.
  367. 34910 => 'HylaFAX FaxRecvTime', // Used by HylaFAX.
  368. 37724 => 'ImageSourceData', // Used by Adobe Photoshop.
  369. 40965 => 'Interoperability IFD', // A pointer to the Exif-related Interoperability IFD.
  370. 42112 => 'GDAL_METADATA', // Used by the GDAL library, holds an XML list of name=value 'metadata' values about the image as a whole, and about specific samples.
  371. 42113 => 'GDAL_NODATA', // Used by the GDAL library, contains an ASCII encoded nodata or background pixel value.
  372. 50215 => 'Oce Scanjob Description', // Used in the Oce scanning process.
  373. 50216 => 'Oce Application Selector', // Used in the Oce scanning process.
  374. 50217 => 'Oce Identification Number', // Used in the Oce scanning process.
  375. 50218 => 'Oce ImageLogic Characteristics', // Used in the Oce scanning process.
  376. 50706 => 'DNGVersion', // Used in IFD 0 of DNG files.
  377. 50707 => 'DNGBackwardVersion', // Used in IFD 0 of DNG files.
  378. 50708 => 'UniqueCameraModel', // Used in IFD 0 of DNG files.
  379. 50709 => 'LocalizedCameraModel', // Used in IFD 0 of DNG files.
  380. 50710 => 'CFAPlaneColor', // Used in Raw IFD of DNG files.
  381. 50711 => 'CFALayout', // Used in Raw IFD of DNG files.
  382. 50712 => 'LinearizationTable', // Used in Raw IFD of DNG files.
  383. 50713 => 'BlackLevelRepeatDim', // Used in Raw IFD of DNG files.
  384. 50714 => 'BlackLevel', // Used in Raw IFD of DNG files.
  385. 50715 => 'BlackLevelDeltaH', // Used in Raw IFD of DNG files.
  386. 50716 => 'BlackLevelDeltaV', // Used in Raw IFD of DNG files.
  387. 50717 => 'WhiteLevel', // Used in Raw IFD of DNG files.
  388. 50718 => 'DefaultScale', // Used in Raw IFD of DNG files.
  389. 50719 => 'DefaultCropOrigin', // Used in Raw IFD of DNG files.
  390. 50720 => 'DefaultCropSize', // Used in Raw IFD of DNG files.
  391. 50721 => 'ColorMatrix1', // Used in IFD 0 of DNG files.
  392. 50722 => 'ColorMatrix2', // Used in IFD 0 of DNG files.
  393. 50723 => 'CameraCalibration1', // Used in IFD 0 of DNG files.
  394. 50724 => 'CameraCalibration2', // Used in IFD 0 of DNG files.
  395. 50725 => 'ReductionMatrix1', // Used in IFD 0 of DNG files.
  396. 50726 => 'ReductionMatrix2', // Used in IFD 0 of DNG files.
  397. 50727 => 'AnalogBalance', // Used in IFD 0 of DNG files.
  398. 50728 => 'AsShotNeutral', // Used in IFD 0 of DNG files.
  399. 50729 => 'AsShotWhiteXY', // Used in IFD 0 of DNG files.
  400. 50730 => 'BaselineExposure', // Used in IFD 0 of DNG files.
  401. 50731 => 'BaselineNoise', // Used in IFD 0 of DNG files.
  402. 50732 => 'BaselineSharpness', // Used in IFD 0 of DNG files.
  403. 50733 => 'BayerGreenSplit', // Used in Raw IFD of DNG files.
  404. 50734 => 'LinearResponseLimit', // Used in IFD 0 of DNG files.
  405. 50735 => 'CameraSerialNumber', // Used in IFD 0 of DNG files.
  406. 50736 => 'LensInfo', // Used in IFD 0 of DNG files.
  407. 50737 => 'ChromaBlurRadius', // Used in Raw IFD of DNG files.
  408. 50738 => 'AntiAliasStrength', // Used in Raw IFD of DNG files.
  409. 50740 => 'DNGPrivateData', // Used in IFD 0 of DNG files.
  410. 50741 => 'MakerNoteSafety', // Used in IFD 0 of DNG files.
  411. 50778 => 'CalibrationIlluminant1', // Used in IFD 0 of DNG files.
  412. 50779 => 'CalibrationIlluminant2', // Used in IFD 0 of DNG files.
  413. 50780 => 'BestQualityScale', // Used in Raw IFD of DNG files.
  414. 50784 => 'Alias Layer Metadata', // Alias Sketchbook Pro layer usage description.
  415. 50908 => 'TIFF_RSID', // This private tag is used in a GEOTIFF standard by DGIWG.
  416. 50909 => 'GEO_METADATA', // This private tag is used in a GEOTIFF standard by DGIWG.
  417. // EXIF tags - https://www.awaresystems.be/imaging/tiff/tifftags/privateifd/exif.html
  418. 33434 => 'ExposureTime', // Exposure time, given in seconds.
  419. 33437 => 'FNumber', // The F number.
  420. 34850 => 'ExposureProgram', // The class of the program used by the camera to set exposure when the picture is taken.
  421. 34852 => 'SpectralSensitivity', // Indicates the spectral sensitivity of each channel of the camera used.
  422. 34855 => 'ISOSpeedRatings', // Indicates the ISO Speed and ISO Latitude of the camera or input device as specified in ISO 12232.
  423. 34856 => 'OECF', // Indicates the Opto-Electric Conversion Function (OECF) specified in ISO 14524.
  424. 36864 => 'ExifVersion', // The version of the supported Exif standard.
  425. 36867 => 'DateTimeOriginal', // The date and time when the original image data was generated.
  426. 36868 => 'DateTimeDigitized', // The date and time when the image was stored as digital data.
  427. 37121 => 'ComponentsConfiguration', // Specific to compressed data; specifies the channels and complements PhotometricInterpretation
  428. 37122 => 'CompressedBitsPerPixel', // Specific to compressed data; states the compressed bits per pixel.
  429. 37377 => 'ShutterSpeedValue', // Shutter speed.
  430. 37378 => 'ApertureValue', // The lens aperture.
  431. 37379 => 'BrightnessValue', // The value of brightness.
  432. 37380 => 'ExposureBiasValue', // The exposure bias.
  433. 37381 => 'MaxApertureValue', // The smallest F number of the lens.
  434. 37382 => 'SubjectDistance', // The distance to the subject, given in meters.
  435. 37383 => 'MeteringMode', // The metering mode.
  436. 37384 => 'LightSource', // The kind of light source.
  437. 37385 => 'Flash', // Indicates the status of flash when the image was shot.
  438. 37386 => 'FocalLength', // The actual focal length of the lens, in mm.
  439. 37396 => 'SubjectArea', // Indicates the location and area of the main subject in the overall scene.
  440. 37500 => 'MakerNote', // Manufacturer specific information.
  441. 37510 => 'UserComment', // Keywords or comments on the image; complements ImageDescription.
  442. 37520 => 'SubsecTime', // A tag used to record fractions of seconds for the DateTime tag.
  443. 37521 => 'SubsecTimeOriginal', // A tag used to record fractions of seconds for the DateTimeOriginal tag.
  444. 37522 => 'SubsecTimeDigitized', // A tag used to record fractions of seconds for the DateTimeDigitized tag.
  445. 40960 => 'FlashpixVersion', // The Flashpix format version supported by a FPXR file.
  446. 40961 => 'ColorSpace', // The color space information tag is always recorded as the color space specifier.
  447. 40962 => 'PixelXDimension', // Specific to compressed data; the valid width of the meaningful image.
  448. 40963 => 'PixelYDimension', // Specific to compressed data; the valid height of the meaningful image.
  449. 40964 => 'RelatedSoundFile', // Used to record the name of an audio file related to the image data.
  450. 41483 => 'FlashEnergy', // Indicates the strobe energy at the time the image is captured, as measured in Beam Candle Power Seconds
  451. 41484 => 'SpatialFrequencyResponse', // Records the camera or input device spatial frequency table and SFR values in the direction of image width, image height, and diagonal direction, as specified in ISO 12233.
  452. 41486 => 'FocalPlaneXResolution', // Indicates the number of pixels in the image width (X) direction per FocalPlaneResolutionUnit on the camera focal plane.
  453. 41487 => 'FocalPlaneYResolution', // Indicates the number of pixels in the image height (Y) direction per FocalPlaneResolutionUnit on the camera focal plane.
  454. 41488 => 'FocalPlaneResolutionUnit', // Indicates the unit for measuring FocalPlaneXResolution and FocalPlaneYResolution.
  455. 41492 => 'SubjectLocation', // Indicates the location of the main subject in the scene.
  456. 41493 => 'ExposureIndex', // Indicates the exposure index selected on the camera or input device at the time the image is captured.
  457. 41495 => 'SensingMethod', // Indicates the image sensor type on the camera or input device.
  458. 41728 => 'FileSource', // Indicates the image source.
  459. 41729 => 'SceneType', // Indicates the type of scene.
  460. 41730 => 'CFAPattern', // Indicates the color filter array (CFA) geometric pattern of the image sensor when a one-chip color area sensor is used.
  461. 41985 => 'CustomRendered', // Indicates the use of special processing on image data, such as rendering geared to output.
  462. 41986 => 'ExposureMode', // Indicates the exposure mode set when the image was shot.
  463. 41987 => 'WhiteBalance', // Indicates the white balance mode set when the image was shot.
  464. 41988 => 'DigitalZoomRatio', // Indicates the digital zoom ratio when the image was shot.
  465. 41989 => 'FocalLengthIn35mmFilm', // Indicates the equivalent focal length assuming a 35mm film camera, in mm.
  466. 41990 => 'SceneCaptureType', // Indicates the type of scene that was shot.
  467. 41991 => 'GainControl', // Indicates the degree of overall image gain adjustment.
  468. 41992 => 'Contrast', // Indicates the direction of contrast processing applied by the camera when the image was shot.
  469. 41993 => 'Saturation', // Indicates the direction of saturation processing applied by the camera when the image was shot.
  470. 41994 => 'Sharpness', // Indicates the direction of sharpness processing applied by the camera when the image was shot.
  471. 41995 => 'DeviceSettingDescription', // This tag indicates information on the picture-taking conditions of a particular camera model.
  472. 41996 => 'SubjectDistanceRange', // Indicates the distance to the subject.
  473. 42016 => 'ImageUniqueID', // Indicates an identifier assigned uniquely to each image.
  474. );
  475. }
  476. return (isset($TIFFcommentName[$id]) ? $TIFFcommentName[$id] : 'unknown/invalid ('.$id.')');
  477. }
  478. }