module.audio-video.mpeg.php 43 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682
  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.audio-video.mpeg.php //
  11. // module for analyzing MPEG files //
  12. // dependencies: module.audio.mp3.php //
  13. // ///
  14. /////////////////////////////////////////////////////////////////
  15. if (!defined('GETID3_INCLUDEPATH')) { // prevent path-exposing attacks that access modules directly on public webservers
  16. exit;
  17. }
  18. getid3_lib::IncludeDependency(GETID3_INCLUDEPATH.'module.audio.mp3.php', __FILE__, true);
  19. class getid3_mpeg extends getid3_handler
  20. {
  21. const START_CODE_BASE = "\x00\x00\x01";
  22. const VIDEO_PICTURE_START = "\x00\x00\x01\x00";
  23. const VIDEO_USER_DATA_START = "\x00\x00\x01\xB2";
  24. const VIDEO_SEQUENCE_HEADER = "\x00\x00\x01\xB3";
  25. const VIDEO_SEQUENCE_ERROR = "\x00\x00\x01\xB4";
  26. const VIDEO_EXTENSION_START = "\x00\x00\x01\xB5";
  27. const VIDEO_SEQUENCE_END = "\x00\x00\x01\xB7";
  28. const VIDEO_GROUP_START = "\x00\x00\x01\xB8";
  29. const AUDIO_START = "\x00\x00\x01\xC0";
  30. /**
  31. * @return bool
  32. */
  33. public function Analyze() {
  34. $info = &$this->getid3->info;
  35. $info['fileformat'] = 'mpeg';
  36. $this->fseek($info['avdataoffset']);
  37. $MPEGstreamData = $this->fread($this->getid3->option_fread_buffer_size);
  38. $MPEGstreamBaseOffset = 0; // how far are we from the beginning of the file data ($info['avdataoffset'])
  39. $MPEGstreamDataOffset = 0; // how far are we from the beginning of the buffer data (~32kB)
  40. $StartCodeValue = false;
  41. $prevStartCodeValue = false;
  42. $GOPcounter = -1;
  43. $FramesByGOP = array();
  44. $ParsedAVchannels = array();
  45. do {
  46. //echo $MPEGstreamDataOffset.' vs '.(strlen($MPEGstreamData) - 1024).'<Br>';
  47. if ($MPEGstreamDataOffset > (strlen($MPEGstreamData) - 16384)) {
  48. // buffer running low, get more data
  49. //echo 'reading more data<br>';
  50. $MPEGstreamData .= $this->fread($this->getid3->option_fread_buffer_size);
  51. if (strlen($MPEGstreamData) > $this->getid3->option_fread_buffer_size) {
  52. $MPEGstreamData = substr($MPEGstreamData, $MPEGstreamDataOffset);
  53. $MPEGstreamBaseOffset += $MPEGstreamDataOffset;
  54. $MPEGstreamDataOffset = 0;
  55. }
  56. }
  57. if (($StartCodeOffset = strpos($MPEGstreamData, self::START_CODE_BASE, $MPEGstreamDataOffset)) === false) {
  58. //echo 'no more start codes found.<br>';
  59. break;
  60. } else {
  61. $MPEGstreamDataOffset = $StartCodeOffset;
  62. $prevStartCodeValue = $StartCodeValue;
  63. $StartCodeValue = ord(substr($MPEGstreamData, $StartCodeOffset + 3, 1));
  64. //echo 'Found "'.strtoupper(dechex($StartCodeValue)).'" at offset '.($MPEGstreamBaseOffset + $StartCodeOffset).' ($MPEGstreamDataOffset = '.$MPEGstreamDataOffset.')<br>';
  65. }
  66. $MPEGstreamDataOffset += 4;
  67. switch ($StartCodeValue) {
  68. case 0x00: // picture_start_code
  69. if (!empty($info['mpeg']['video']['bitrate_mode']) && ($info['mpeg']['video']['bitrate_mode'] == 'vbr')) {
  70. $bitstream = getid3_lib::BigEndian2Bin(substr($MPEGstreamData, $StartCodeOffset + 4, 4));
  71. $bitstreamoffset = 0;
  72. $PictureHeader = array();
  73. $PictureHeader['temporal_reference'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 10); // 10-bit unsigned integer associated with each input picture. It is incremented by one, modulo 1024, for each input frame. When a frame is coded as two fields the temporal reference in the picture header of both fields is the same. Following a group start header the temporal reference of the earliest picture (in display order) shall be reset to zero.
  74. $PictureHeader['picture_coding_type'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 3); // 3 bits for picture_coding_type
  75. $PictureHeader['vbv_delay'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 16); // 16 bits for vbv_delay
  76. //... etc
  77. $FramesByGOP[$GOPcounter][] = $PictureHeader;
  78. }
  79. break;
  80. case 0xB3: // sequence_header_code
  81. // Note: purposely doing the less-pretty (and probably a bit slower) method of using string of bits rather than bitwise operations.
  82. // Mostly because PHP 32-bit doesn't handle unsigned integers well for bitwise operation.
  83. // Also the MPEG stream is designed as a bitstream and often doesn't align nicely with byte boundaries.
  84. $info['video']['codec'] = 'MPEG-1'; // will be updated if extension_start_code found
  85. $bitstream = getid3_lib::BigEndian2Bin(substr($MPEGstreamData, $StartCodeOffset + 4, 8));
  86. $bitstreamoffset = 0;
  87. $info['mpeg']['video']['raw']['horizontal_size_value'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 12); // 12 bits for horizontal frame size. Note: horizontal_size_extension, if present, will add 2 most-significant bits to this value
  88. $info['mpeg']['video']['raw']['vertical_size_value'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 12); // 12 bits for vertical frame size. Note: vertical_size_extension, if present, will add 2 most-significant bits to this value
  89. $info['mpeg']['video']['raw']['aspect_ratio_information'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 4); // 4 bits for aspect_ratio_information
  90. $info['mpeg']['video']['raw']['frame_rate_code'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 4); // 4 bits for Frame Rate id code
  91. $info['mpeg']['video']['raw']['bitrate'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 18); // 18 bits for bit_rate_value (18 set bits = VBR, otherwise bitrate = this value * 400)
  92. $marker_bit = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // The term "marker_bit" indicates a one bit field in which the value zero is forbidden. These marker bits are introduced at several points in the syntax to avoid start code emulation.
  93. $info['mpeg']['video']['raw']['vbv_buffer_size'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 10); // 10 bits vbv_buffer_size_value
  94. $info['mpeg']['video']['raw']['constrained_param_flag'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: constrained_param_flag
  95. $info['mpeg']['video']['raw']['load_intra_quantiser_matrix'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: load_intra_quantiser_matrix
  96. if ($info['mpeg']['video']['raw']['load_intra_quantiser_matrix']) {
  97. $bitstream .= getid3_lib::BigEndian2Bin(substr($MPEGstreamData, $StartCodeOffset + 12, 64));
  98. for ($i = 0; $i < 64; $i++) {
  99. $info['mpeg']['video']['raw']['intra_quantiser_matrix'][$i] = self::readBitsFromStream($bitstream, $bitstreamoffset, 8);
  100. }
  101. }
  102. $info['mpeg']['video']['raw']['load_non_intra_quantiser_matrix'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1);
  103. if ($info['mpeg']['video']['raw']['load_non_intra_quantiser_matrix']) {
  104. $bitstream .= getid3_lib::BigEndian2Bin(substr($MPEGstreamData, $StartCodeOffset + 12 + ($info['mpeg']['video']['raw']['load_intra_quantiser_matrix'] ? 64 : 0), 64));
  105. for ($i = 0; $i < 64; $i++) {
  106. $info['mpeg']['video']['raw']['non_intra_quantiser_matrix'][$i] = self::readBitsFromStream($bitstream, $bitstreamoffset, 8);
  107. }
  108. }
  109. $info['mpeg']['video']['pixel_aspect_ratio'] = self::videoAspectRatioLookup($info['mpeg']['video']['raw']['aspect_ratio_information']); // may be overridden later if file turns out to be MPEG-2
  110. $info['mpeg']['video']['pixel_aspect_ratio_text'] = self::videoAspectRatioTextLookup($info['mpeg']['video']['raw']['aspect_ratio_information']); // may be overridden later if file turns out to be MPEG-2
  111. $info['mpeg']['video']['frame_rate'] = self::videoFramerateLookup($info['mpeg']['video']['raw']['frame_rate_code']);
  112. if ($info['mpeg']['video']['raw']['bitrate'] == 0x3FFFF) { // 18 set bits = VBR
  113. //$this->warning('This version of getID3() ['.$this->getid3->version().'] cannot determine average bitrate of VBR MPEG video files');
  114. $info['mpeg']['video']['bitrate_mode'] = 'vbr';
  115. } else {
  116. $info['mpeg']['video']['bitrate'] = $info['mpeg']['video']['raw']['bitrate'] * 400;
  117. $info['mpeg']['video']['bitrate_mode'] = 'cbr';
  118. $info['video']['bitrate'] = $info['mpeg']['video']['bitrate'];
  119. }
  120. $info['video']['resolution_x'] = $info['mpeg']['video']['raw']['horizontal_size_value'];
  121. $info['video']['resolution_y'] = $info['mpeg']['video']['raw']['vertical_size_value'];
  122. $info['video']['frame_rate'] = $info['mpeg']['video']['frame_rate'];
  123. $info['video']['bitrate_mode'] = $info['mpeg']['video']['bitrate_mode'];
  124. $info['video']['pixel_aspect_ratio'] = $info['mpeg']['video']['pixel_aspect_ratio'];
  125. $info['video']['lossless'] = false;
  126. $info['video']['bits_per_sample'] = 24;
  127. break;
  128. case 0xB5: // extension_start_code
  129. $info['video']['codec'] = 'MPEG-2';
  130. $bitstream = getid3_lib::BigEndian2Bin(substr($MPEGstreamData, $StartCodeOffset + 4, 8)); // 48 bits for Sequence Extension ID; 61 bits for Sequence Display Extension ID; 59 bits for Sequence Scalable Extension ID
  131. $bitstreamoffset = 0;
  132. $info['mpeg']['video']['raw']['extension_start_code_identifier'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 4); // 4 bits for extension_start_code_identifier
  133. //echo $info['mpeg']['video']['raw']['extension_start_code_identifier'].'<br>';
  134. switch ($info['mpeg']['video']['raw']['extension_start_code_identifier']) {
  135. case 1: // 0001 Sequence Extension ID
  136. $info['mpeg']['video']['raw']['profile_and_level_indication'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 8); // 8 bits for profile_and_level_indication
  137. $info['mpeg']['video']['raw']['progressive_sequence'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: progressive_sequence
  138. $info['mpeg']['video']['raw']['chroma_format'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 2); // 2 bits for chroma_format
  139. $info['mpeg']['video']['raw']['horizontal_size_extension'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 2); // 2 bits for horizontal_size_extension
  140. $info['mpeg']['video']['raw']['vertical_size_extension'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 2); // 2 bits for vertical_size_extension
  141. $info['mpeg']['video']['raw']['bit_rate_extension'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 12); // 12 bits for bit_rate_extension
  142. $marker_bit = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // The term "marker_bit" indicates a one bit field in which the value zero is forbidden. These marker bits are introduced at several points in the syntax to avoid start code emulation.
  143. $info['mpeg']['video']['raw']['vbv_buffer_size_extension'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 8); // 8 bits for vbv_buffer_size_extension
  144. $info['mpeg']['video']['raw']['low_delay'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: low_delay
  145. $info['mpeg']['video']['raw']['frame_rate_extension_n'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 2); // 2 bits for frame_rate_extension_n
  146. $info['mpeg']['video']['raw']['frame_rate_extension_d'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 5); // 5 bits for frame_rate_extension_d
  147. $info['video']['resolution_x'] = ($info['mpeg']['video']['raw']['horizontal_size_extension'] << 12) | $info['mpeg']['video']['raw']['horizontal_size_value'];
  148. $info['video']['resolution_y'] = ($info['mpeg']['video']['raw']['vertical_size_extension'] << 12) | $info['mpeg']['video']['raw']['vertical_size_value'];
  149. $info['video']['interlaced'] = !$info['mpeg']['video']['raw']['progressive_sequence'];
  150. $info['mpeg']['video']['interlaced'] = !$info['mpeg']['video']['raw']['progressive_sequence'];
  151. $info['mpeg']['video']['chroma_format'] = self::chromaFormatTextLookup($info['mpeg']['video']['raw']['chroma_format']);
  152. if (isset($info['mpeg']['video']['raw']['aspect_ratio_information'])) {
  153. // MPEG-2 defines the aspect ratio flag differently from MPEG-1, but the MPEG-2 extension start code may occur after we've already looked up the aspect ratio assuming it was MPEG-1, so re-lookup assuming MPEG-2
  154. // This must be done after the extended size is known, so the display aspect ratios can be converted to pixel aspect ratios.
  155. $info['mpeg']['video']['pixel_aspect_ratio'] = self::videoAspectRatioLookup($info['mpeg']['video']['raw']['aspect_ratio_information'], 2, $info['video']['resolution_x'], $info['video']['resolution_y']);
  156. $info['mpeg']['video']['pixel_aspect_ratio_text'] = self::videoAspectRatioTextLookup($info['mpeg']['video']['raw']['aspect_ratio_information'], 2);
  157. $info['video']['pixel_aspect_ratio'] = $info['mpeg']['video']['pixel_aspect_ratio'];
  158. $info['video']['pixel_aspect_ratio_text'] = $info['mpeg']['video']['pixel_aspect_ratio_text'];
  159. }
  160. break;
  161. case 2: // 0010 Sequence Display Extension ID
  162. $info['mpeg']['video']['raw']['video_format'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 3); // 3 bits for video_format
  163. $info['mpeg']['video']['raw']['colour_description'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: colour_description
  164. if ($info['mpeg']['video']['raw']['colour_description']) {
  165. $info['mpeg']['video']['raw']['colour_primaries'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 8); // 8 bits for colour_primaries
  166. $info['mpeg']['video']['raw']['transfer_characteristics'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 8); // 8 bits for transfer_characteristics
  167. $info['mpeg']['video']['raw']['matrix_coefficients'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 8); // 8 bits for matrix_coefficients
  168. }
  169. $info['mpeg']['video']['raw']['display_horizontal_size'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 14); // 14 bits for display_horizontal_size
  170. $marker_bit = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // The term "marker_bit" indicates a one bit field in which the value zero is forbidden. These marker bits are introduced at several points in the syntax to avoid start code emulation.
  171. $info['mpeg']['video']['raw']['display_vertical_size'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 14); // 14 bits for display_vertical_size
  172. $info['mpeg']['video']['video_format'] = self::videoFormatTextLookup($info['mpeg']['video']['raw']['video_format']);
  173. break;
  174. case 3: // 0011 Quant Matrix Extension ID
  175. break;
  176. case 5: // 0101 Sequence Scalable Extension ID
  177. $info['mpeg']['video']['raw']['scalable_mode'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 2); // 2 bits for scalable_mode
  178. $info['mpeg']['video']['raw']['layer_id'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 4); // 4 bits for layer_id
  179. if ($info['mpeg']['video']['raw']['scalable_mode'] == 1) { // "spatial scalability"
  180. $info['mpeg']['video']['raw']['lower_layer_prediction_horizontal_size'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 14); // 14 bits for lower_layer_prediction_horizontal_size
  181. $marker_bit = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // The term "marker_bit" indicates a one bit field in which the value zero is forbidden. These marker bits are introduced at several points in the syntax to avoid start code emulation.
  182. $info['mpeg']['video']['raw']['lower_layer_prediction_vertical_size'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 14); // 14 bits for lower_layer_prediction_vertical_size
  183. $info['mpeg']['video']['raw']['horizontal_subsampling_factor_m'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 5); // 5 bits for horizontal_subsampling_factor_m
  184. $info['mpeg']['video']['raw']['horizontal_subsampling_factor_n'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 5); // 5 bits for horizontal_subsampling_factor_n
  185. $info['mpeg']['video']['raw']['vertical_subsampling_factor_m'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 5); // 5 bits for vertical_subsampling_factor_m
  186. $info['mpeg']['video']['raw']['vertical_subsampling_factor_n'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 5); // 5 bits for vertical_subsampling_factor_n
  187. } elseif ($info['mpeg']['video']['raw']['scalable_mode'] == 3) { // "temporal scalability"
  188. $info['mpeg']['video']['raw']['picture_mux_enable'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: picture_mux_enable
  189. if ($info['mpeg']['video']['raw']['picture_mux_enable']) {
  190. $info['mpeg']['video']['raw']['mux_to_progressive_sequence'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: mux_to_progressive_sequence
  191. }
  192. $info['mpeg']['video']['raw']['picture_mux_order'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 3); // 3 bits for picture_mux_order
  193. $info['mpeg']['video']['raw']['picture_mux_factor'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 3); // 3 bits for picture_mux_factor
  194. }
  195. $info['mpeg']['video']['scalable_mode'] = self::scalableModeTextLookup($info['mpeg']['video']['raw']['scalable_mode']);
  196. break;
  197. case 7: // 0111 Picture Display Extension ID
  198. break;
  199. case 8: // 1000 Picture Coding Extension ID
  200. $info['mpeg']['video']['raw']['f_code_00'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 4); // 4 bits for f_code[0][0] (forward horizontal)
  201. $info['mpeg']['video']['raw']['f_code_01'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 4); // 4 bits for f_code[0][1] (forward vertical)
  202. $info['mpeg']['video']['raw']['f_code_10'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 4); // 4 bits for f_code[1][0] (backward horizontal)
  203. $info['mpeg']['video']['raw']['f_code_11'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 4); // 4 bits for f_code[1][1] (backward vertical)
  204. $info['mpeg']['video']['raw']['intra_dc_precision'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 2); // 2 bits for intra_dc_precision
  205. $info['mpeg']['video']['raw']['picture_structure'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 2); // 2 bits for picture_structure
  206. $info['mpeg']['video']['raw']['top_field_first'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: top_field_first
  207. $info['mpeg']['video']['raw']['frame_pred_frame_dct'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: frame_pred_frame_dct
  208. $info['mpeg']['video']['raw']['concealment_motion_vectors'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: concealment_motion_vectors
  209. $info['mpeg']['video']['raw']['q_scale_type'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: q_scale_type
  210. $info['mpeg']['video']['raw']['intra_vlc_format'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: intra_vlc_format
  211. $info['mpeg']['video']['raw']['alternate_scan'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: alternate_scan
  212. $info['mpeg']['video']['raw']['repeat_first_field'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: repeat_first_field
  213. $info['mpeg']['video']['raw']['chroma_420_type'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: chroma_420_type
  214. $info['mpeg']['video']['raw']['progressive_frame'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: progressive_frame
  215. $info['mpeg']['video']['raw']['composite_display_flag'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: composite_display_flag
  216. if ($info['mpeg']['video']['raw']['composite_display_flag']) {
  217. $info['mpeg']['video']['raw']['v_axis'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: v_axis
  218. $info['mpeg']['video']['raw']['field_sequence'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 3); // 3 bits for field_sequence
  219. $info['mpeg']['video']['raw']['sub_carrier'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: sub_carrier
  220. $info['mpeg']['video']['raw']['burst_amplitude'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 7); // 7 bits for burst_amplitude
  221. $info['mpeg']['video']['raw']['sub_carrier_phase'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 8); // 8 bits for sub_carrier_phase
  222. }
  223. $info['mpeg']['video']['intra_dc_precision_bits'] = $info['mpeg']['video']['raw']['intra_dc_precision'] + 8;
  224. $info['mpeg']['video']['picture_structure'] = self::pictureStructureTextLookup($info['mpeg']['video']['raw']['picture_structure']);
  225. break;
  226. case 9: // 1001 Picture Spatial Scalable Extension ID
  227. break;
  228. case 10: // 1010 Picture Temporal Scalable Extension ID
  229. break;
  230. default:
  231. $this->warning('Unexpected $info[mpeg][video][raw][extension_start_code_identifier] value of '.$info['mpeg']['video']['raw']['extension_start_code_identifier']);
  232. break;
  233. }
  234. break;
  235. case 0xB8: // group_of_pictures_header
  236. $GOPcounter++;
  237. if (!empty($info['mpeg']['video']['bitrate_mode']) && ($info['mpeg']['video']['bitrate_mode'] == 'vbr')) {
  238. $bitstream = getid3_lib::BigEndian2Bin(substr($MPEGstreamData, $StartCodeOffset + 4, 4)); // 27 bits needed for group_of_pictures_header
  239. $bitstreamoffset = 0;
  240. $GOPheader = array();
  241. $GOPheader['byte_offset'] = $MPEGstreamBaseOffset + $StartCodeOffset;
  242. $GOPheader['drop_frame_flag'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: drop_frame_flag
  243. $GOPheader['time_code_hours'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 5); // 5 bits for time_code_hours
  244. $GOPheader['time_code_minutes'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 6); // 6 bits for time_code_minutes
  245. $marker_bit = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // The term "marker_bit" indicates a one bit field in which the value zero is forbidden. These marker bits are introduced at several points in the syntax to avoid start code emulation.
  246. $GOPheader['time_code_seconds'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 6); // 6 bits for time_code_seconds
  247. $GOPheader['time_code_pictures'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 6); // 6 bits for time_code_pictures
  248. $GOPheader['closed_gop'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: closed_gop
  249. $GOPheader['broken_link'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: broken_link
  250. $time_code_separator = ($GOPheader['drop_frame_flag'] ? ';' : ':'); // While non-drop time code is displayed with colons separating the digit pairs "HH:MM:SS:FF" drop frame is usually represented with a semi-colon (;) or period (.) as the divider between all the digit pairs "HH;MM;SS;FF", "HH.MM.SS.FF"
  251. $GOPheader['time_code'] = sprintf('%02d'.$time_code_separator.'%02d'.$time_code_separator.'%02d'.$time_code_separator.'%02d', $GOPheader['time_code_hours'], $GOPheader['time_code_minutes'], $GOPheader['time_code_seconds'], $GOPheader['time_code_pictures']);
  252. $info['mpeg']['group_of_pictures'][] = $GOPheader;
  253. }
  254. break;
  255. case 0xC0: // audio stream
  256. case 0xC1: // audio stream
  257. case 0xC2: // audio stream
  258. case 0xC3: // audio stream
  259. case 0xC4: // audio stream
  260. case 0xC5: // audio stream
  261. case 0xC6: // audio stream
  262. case 0xC7: // audio stream
  263. case 0xC8: // audio stream
  264. case 0xC9: // audio stream
  265. case 0xCA: // audio stream
  266. case 0xCB: // audio stream
  267. case 0xCC: // audio stream
  268. case 0xCD: // audio stream
  269. case 0xCE: // audio stream
  270. case 0xCF: // audio stream
  271. case 0xD0: // audio stream
  272. case 0xD1: // audio stream
  273. case 0xD2: // audio stream
  274. case 0xD3: // audio stream
  275. case 0xD4: // audio stream
  276. case 0xD5: // audio stream
  277. case 0xD6: // audio stream
  278. case 0xD7: // audio stream
  279. case 0xD8: // audio stream
  280. case 0xD9: // audio stream
  281. case 0xDA: // audio stream
  282. case 0xDB: // audio stream
  283. case 0xDC: // audio stream
  284. case 0xDD: // audio stream
  285. case 0xDE: // audio stream
  286. case 0xDF: // audio stream
  287. //case 0xE0: // video stream
  288. //case 0xE1: // video stream
  289. //case 0xE2: // video stream
  290. //case 0xE3: // video stream
  291. //case 0xE4: // video stream
  292. //case 0xE5: // video stream
  293. //case 0xE6: // video stream
  294. //case 0xE7: // video stream
  295. //case 0xE8: // video stream
  296. //case 0xE9: // video stream
  297. //case 0xEA: // video stream
  298. //case 0xEB: // video stream
  299. //case 0xEC: // video stream
  300. //case 0xED: // video stream
  301. //case 0xEE: // video stream
  302. //case 0xEF: // video stream
  303. if (isset($ParsedAVchannels[$StartCodeValue])) {
  304. break;
  305. }
  306. $ParsedAVchannels[$StartCodeValue] = $StartCodeValue;
  307. // http://en.wikipedia.org/wiki/Packetized_elementary_stream
  308. // http://dvd.sourceforge.net/dvdinfo/pes-hdr.html
  309. /*
  310. $PackedElementaryStream = array();
  311. if ($StartCodeValue >= 0xE0) {
  312. $PackedElementaryStream['stream_type'] = 'video';
  313. $PackedElementaryStream['stream_id'] = $StartCodeValue - 0xE0;
  314. } else {
  315. $PackedElementaryStream['stream_type'] = 'audio';
  316. $PackedElementaryStream['stream_id'] = $StartCodeValue - 0xC0;
  317. }
  318. $PackedElementaryStream['packet_length'] = getid3_lib::BigEndian2Int(substr($MPEGstreamData, $StartCodeOffset + 4, 2));
  319. $bitstream = getid3_lib::BigEndian2Bin(substr($MPEGstreamData, $StartCodeOffset + 6, 3)); // more may be needed below
  320. $bitstreamoffset = 0;
  321. $PackedElementaryStream['marker_bits'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 2); // 2 bits for marker_bits -- should be "10" = 2
  322. echo 'marker_bits = '.$PackedElementaryStream['marker_bits'].'<br>';
  323. $PackedElementaryStream['scrambling_control'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 2); // 2 bits for scrambling_control -- 00 implies not scrambled
  324. $PackedElementaryStream['priority'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: priority
  325. $PackedElementaryStream['data_alignment_indicator'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: data_alignment_indicator -- 1 indicates that the PES packet header is immediately followed by the video start code or audio syncword
  326. $PackedElementaryStream['copyright'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: copyright -- 1 implies copyrighted
  327. $PackedElementaryStream['original_or_copy'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: original_or_copy -- 1 implies original
  328. $PackedElementaryStream['pts_flag'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: pts_flag -- Presentation Time Stamp
  329. $PackedElementaryStream['dts_flag'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: dts_flag -- Decode Time Stamp
  330. $PackedElementaryStream['escr_flag'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: escr_flag -- Elementary Stream Clock Reference
  331. $PackedElementaryStream['es_rate_flag'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: es_rate_flag -- Elementary Stream [data] Rate
  332. $PackedElementaryStream['dsm_trick_mode_flag'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: dsm_trick_mode_flag -- DSM trick mode - not used by DVD
  333. $PackedElementaryStream['additional_copy_info_flag'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: additional_copy_info_flag
  334. $PackedElementaryStream['crc_flag'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: crc_flag
  335. $PackedElementaryStream['extension_flag'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 1); // 1 bit flag: extension_flag
  336. $PackedElementaryStream['pes_remain_header_length'] = self::readBitsFromStream($bitstream, $bitstreamoffset, 8); // 1 bit flag: priority
  337. $additional_header_bytes = 0;
  338. $additional_header_bytes += ($PackedElementaryStream['pts_flag'] ? 5 : 0);
  339. $additional_header_bytes += ($PackedElementaryStream['dts_flag'] ? 5 : 0);
  340. $additional_header_bytes += ($PackedElementaryStream['escr_flag'] ? 6 : 0);
  341. $additional_header_bytes += ($PackedElementaryStream['es_rate_flag'] ? 3 : 0);
  342. $additional_header_bytes += ($PackedElementaryStream['additional_copy_info_flag'] ? 1 : 0);
  343. $additional_header_bytes += ($PackedElementaryStream['crc_flag'] ? 2 : 0);
  344. $additional_header_bytes += ($PackedElementaryStream['extension_flag'] ? 1 : 0);
  345. $PackedElementaryStream['additional_header_bytes'] = $additional_header_bytes;
  346. $bitstream .= getid3_lib::BigEndian2Bin(substr($MPEGstreamData, $StartCodeOffset + 9, $additional_header_bytes));
  347. $info['mpeg']['packed_elementary_streams'][$PackedElementaryStream['stream_type']][$PackedElementaryStream['stream_id']][] = $PackedElementaryStream;
  348. */
  349. $getid3_temp = new getID3();
  350. $getid3_temp->openfile($this->getid3->filename);
  351. $getid3_temp->info = $info;
  352. $getid3_mp3 = new getid3_mp3($getid3_temp);
  353. for ($i = 0; $i <= 7; $i++) {
  354. // some files have the MPEG-audio header 8 bytes after the end of the $00 $00 $01 $C0 signature, some have it up to 13 bytes (or more?) after
  355. // I have no idea why or what the difference is, so this is a stupid hack.
  356. // If anybody has any better idea of what's going on, please let me know - info@getid3.org
  357. $getid3_temp->info = $info; // only overwrite real data if valid header found
  358. //echo 'audio at? '.($MPEGstreamBaseOffset + $StartCodeOffset + 4 + 8 + $i).'<br>';
  359. if ($getid3_mp3->decodeMPEGaudioHeader($MPEGstreamBaseOffset + $StartCodeOffset + 4 + 8 + $i, $getid3_temp->info, false)) {
  360. //echo 'yes!<br>';
  361. $info = $getid3_temp->info;
  362. $info['audio']['bitrate_mode'] = 'cbr';
  363. $info['audio']['lossless'] = false;
  364. break;
  365. }
  366. }
  367. unset($getid3_temp, $getid3_mp3);
  368. break;
  369. case 0xBC: // Program Stream Map
  370. case 0xBD: // Private stream 1 (non MPEG audio, subpictures)
  371. case 0xBE: // Padding stream
  372. case 0xBF: // Private stream 2 (navigation data)
  373. case 0xF0: // ECM stream
  374. case 0xF1: // EMM stream
  375. case 0xF2: // DSM-CC stream
  376. case 0xF3: // ISO/IEC_13522_stream
  377. case 0xF4: // ITU-I Rec. H.222.1 type A
  378. case 0xF5: // ITU-I Rec. H.222.1 type B
  379. case 0xF6: // ITU-I Rec. H.222.1 type C
  380. case 0xF7: // ITU-I Rec. H.222.1 type D
  381. case 0xF8: // ITU-I Rec. H.222.1 type E
  382. case 0xF9: // ancilliary stream
  383. case 0xFA: // ISO/IEC 14496-1 SL-packtized stream
  384. case 0xFB: // ISO/IEC 14496-1 FlexMux stream
  385. case 0xFC: // metadata stream
  386. case 0xFD: // extended stream ID
  387. case 0xFE: // reserved data stream
  388. case 0xFF: // program stream directory
  389. // ignore
  390. break;
  391. default:
  392. // ignore
  393. break;
  394. }
  395. } while (true);
  396. // // Temporary hack to account for interleaving overhead:
  397. // if (!empty($info['video']['bitrate']) && !empty($info['audio']['bitrate'])) {
  398. // $info['playtime_seconds'] = (($info['avdataend'] - $info['avdataoffset']) * 8) / ($info['video']['bitrate'] + $info['audio']['bitrate']);
  399. //
  400. // // Interleaved MPEG audio/video files have a certain amount of overhead that varies
  401. // // by both video and audio bitrates, and not in any sensible, linear/logarithmic pattern
  402. // // Use interpolated lookup tables to approximately guess how much is overhead, because
  403. // // playtime is calculated as filesize / total-bitrate
  404. // $info['playtime_seconds'] *= self::systemNonOverheadPercentage($info['video']['bitrate'], $info['audio']['bitrate']);
  405. //
  406. // //switch ($info['video']['bitrate']) {
  407. // // case('5000000'):
  408. // // $multiplier = 0.93292642112380355828048824319889;
  409. // // break;
  410. // // case('5500000'):
  411. // // $multiplier = 0.93582895375200989965359777343219;
  412. // // break;
  413. // // case('6000000'):
  414. // // $multiplier = 0.93796247714820932532911373859139;
  415. // // break;
  416. // // case('7000000'):
  417. // // $multiplier = 0.9413264083635103463010117778776;
  418. // // break;
  419. // // default:
  420. // // $multiplier = 1;
  421. // // break;
  422. // //}
  423. // //$info['playtime_seconds'] *= $multiplier;
  424. // //$this->warning('Interleaved MPEG audio/video playtime may be inaccurate. With current hack should be within a few seconds of accurate. Report to info@getid3.org if off by more than 10 seconds.');
  425. // if ($info['video']['bitrate'] < 50000) {
  426. // $this->warning('Interleaved MPEG audio/video playtime may be slightly inaccurate for video bitrates below 100kbps. Except in extreme low-bitrate situations, error should be less than 1%. Report to info@getid3.org if greater than this.');
  427. // }
  428. // }
  429. //
  430. /*
  431. $time_prev = 0;
  432. $byte_prev = 0;
  433. $vbr_bitrates = array();
  434. foreach ($info['mpeg']['group_of_pictures'] as $gopkey => $gopdata) {
  435. $time_this = ($gopdata['time_code_hours'] * 3600) + ($gopdata['time_code_minutes'] * 60) + $gopdata['time_code_seconds'] + ($gopdata['time_code_seconds'] / 30);
  436. $byte_this = $gopdata['byte_offset'];
  437. if ($gopkey > 0) {
  438. if ($time_this > $time_prev) {
  439. $bytedelta = $byte_this - $byte_prev;
  440. $timedelta = $time_this - $time_prev;
  441. $this_bitrate = ($bytedelta * 8) / $timedelta;
  442. echo $gopkey.': ('.number_format($time_prev, 2).'-'.number_format($time_this, 2).') '.number_format($bytedelta).' bytes over '.number_format($timedelta, 3).' seconds = '.number_format($this_bitrate / 1000, 2).'kbps<br>';
  443. $time_prev = $time_this;
  444. $byte_prev = $byte_this;
  445. $vbr_bitrates[] = $this_bitrate;
  446. }
  447. }
  448. }
  449. echo 'average_File_bitrate = '.number_format(array_sum($vbr_bitrates) / count($vbr_bitrates), 1).'<br>';
  450. */
  451. //echo '<pre>'.print_r($FramesByGOP, true).'</pre>';
  452. if (!empty($info['mpeg']['video']['bitrate_mode']) && ($info['mpeg']['video']['bitrate_mode'] == 'vbr')) {
  453. $last_GOP_id = max(array_keys($FramesByGOP));
  454. $frames_in_last_GOP = count($FramesByGOP[$last_GOP_id]);
  455. $gopdata = &$info['mpeg']['group_of_pictures'][$last_GOP_id];
  456. $info['playtime_seconds'] = ($gopdata['time_code_hours'] * 3600) + ($gopdata['time_code_minutes'] * 60) + $gopdata['time_code_seconds'] + (($gopdata['time_code_pictures'] + $frames_in_last_GOP + 1) / $info['mpeg']['video']['frame_rate']);
  457. if (!isset($info['video']['bitrate'])) {
  458. $overall_bitrate = ($info['avdataend'] - $info['avdataoffset']) * 8 / $info['playtime_seconds'];
  459. $info['video']['bitrate'] = $overall_bitrate - (isset($info['audio']['bitrate']) ? $info['audio']['bitrate'] : 0);
  460. }
  461. unset($info['mpeg']['group_of_pictures']);
  462. }
  463. return true;
  464. }
  465. /**
  466. * @param string $bitstream
  467. * @param int $bitstreamoffset
  468. * @param int $bits_to_read
  469. * @param bool $return_singlebit_as_boolean
  470. *
  471. * @return bool|int
  472. */
  473. private function readBitsFromStream(&$bitstream, &$bitstreamoffset, $bits_to_read, $return_singlebit_as_boolean=true) {
  474. $return = bindec(substr($bitstream, $bitstreamoffset, $bits_to_read));
  475. $bitstreamoffset += $bits_to_read;
  476. if (($bits_to_read == 1) && $return_singlebit_as_boolean) {
  477. $return = (bool) $return;
  478. }
  479. return $return;
  480. }
  481. /**
  482. * @param int $VideoBitrate
  483. * @param int $AudioBitrate
  484. *
  485. * @return float|int
  486. */
  487. public static function systemNonOverheadPercentage($VideoBitrate, $AudioBitrate) {
  488. $OverheadPercentage = 0;
  489. $AudioBitrate = max(min($AudioBitrate / 1000, 384), 32); // limit to range of 32kbps - 384kbps (should be only legal bitrates, but maybe VBR?)
  490. $VideoBitrate = max(min($VideoBitrate / 1000, 10000), 10); // limit to range of 10kbps - 10Mbps (beyond that curves flatten anyways, no big loss)
  491. //OMBB[audiobitrate] = array(video-10kbps, video-100kbps, video-1000kbps, video-10000kbps)
  492. $OverheadMultiplierByBitrate[32] = array(0, 0.9676287944368530, 0.9802276264360310, 0.9844916183244460, 0.9852821845179940);
  493. $OverheadMultiplierByBitrate[48] = array(0, 0.9779100089209830, 0.9787770035359320, 0.9846738664076130, 0.9852683013799960);
  494. $OverheadMultiplierByBitrate[56] = array(0, 0.9731249855367600, 0.9776624308938040, 0.9832606361852130, 0.9843922606633340);
  495. $OverheadMultiplierByBitrate[64] = array(0, 0.9755642683275760, 0.9795256705493390, 0.9836573009193170, 0.9851122539404470);
  496. $OverheadMultiplierByBitrate[96] = array(0, 0.9788025247497290, 0.9798553314148700, 0.9822956869792560, 0.9834815119124690);
  497. $OverheadMultiplierByBitrate[128] = array(0, 0.9816940050925480, 0.9821675936072120, 0.9829756927470870, 0.9839763420152050);
  498. $OverheadMultiplierByBitrate[160] = array(0, 0.9825894094561180, 0.9820913399073960, 0.9823907143253970, 0.9832821783651570);
  499. $OverheadMultiplierByBitrate[192] = array(0, 0.9832038474336260, 0.9825731694317960, 0.9821028622712400, 0.9828262076447620);
  500. $OverheadMultiplierByBitrate[224] = array(0, 0.9836516298538770, 0.9824718601823890, 0.9818302180625380, 0.9823735101626480);
  501. $OverheadMultiplierByBitrate[256] = array(0, 0.9845863022094920, 0.9837229411967540, 0.9824521662210830, 0.9828645172100790);
  502. $OverheadMultiplierByBitrate[320] = array(0, 0.9849565280263180, 0.9837683142805110, 0.9822885275960400, 0.9824424382727190);
  503. $OverheadMultiplierByBitrate[384] = array(0, 0.9856094774357600, 0.9844573394432720, 0.9825970399837330, 0.9824673808303890);
  504. $BitrateToUseMin = 32;
  505. $BitrateToUseMax = 32;
  506. $previousBitrate = 32;
  507. foreach ($OverheadMultiplierByBitrate as $key => $value) {
  508. if ($AudioBitrate >= $previousBitrate) {
  509. $BitrateToUseMin = $previousBitrate;
  510. }
  511. if ($AudioBitrate < $key) {
  512. $BitrateToUseMax = $key;
  513. break;
  514. }
  515. $previousBitrate = $key;
  516. }
  517. $FactorA = ($BitrateToUseMax - $AudioBitrate) / ($BitrateToUseMax - $BitrateToUseMin);
  518. $VideoBitrateLog10 = log10($VideoBitrate);
  519. $VideoFactorMin1 = $OverheadMultiplierByBitrate[$BitrateToUseMin][floor($VideoBitrateLog10)];
  520. $VideoFactorMin2 = $OverheadMultiplierByBitrate[$BitrateToUseMax][floor($VideoBitrateLog10)];
  521. $VideoFactorMax1 = $OverheadMultiplierByBitrate[$BitrateToUseMin][ceil($VideoBitrateLog10)];
  522. $VideoFactorMax2 = $OverheadMultiplierByBitrate[$BitrateToUseMax][ceil($VideoBitrateLog10)];
  523. $FactorV = $VideoBitrateLog10 - floor($VideoBitrateLog10);
  524. $OverheadPercentage = $VideoFactorMin1 * $FactorA * $FactorV;
  525. $OverheadPercentage += $VideoFactorMin2 * (1 - $FactorA) * $FactorV;
  526. $OverheadPercentage += $VideoFactorMax1 * $FactorA * (1 - $FactorV);
  527. $OverheadPercentage += $VideoFactorMax2 * (1 - $FactorA) * (1 - $FactorV);
  528. return $OverheadPercentage;
  529. }
  530. /**
  531. * @param int $rawframerate
  532. *
  533. * @return float
  534. */
  535. public static function videoFramerateLookup($rawframerate) {
  536. $lookup = array(0, 23.976, 24, 25, 29.97, 30, 50, 59.94, 60);
  537. return (float) (isset($lookup[$rawframerate]) ? $lookup[$rawframerate] : 0);
  538. }
  539. /**
  540. * @param int $rawaspectratio
  541. * @param int $mpeg_version
  542. * @param int $width
  543. * @param int $height
  544. *
  545. * @return float
  546. */
  547. public static function videoAspectRatioLookup($rawaspectratio, $mpeg_version=1, $width=0, $height=0) {
  548. $lookup = array(
  549. 1 => array(0, 1, 0.6735, 0.7031, 0.7615, 0.8055, 0.8437, 0.8935, 0.9157, 0.9815, 1.0255, 1.0695, 1.0950, 1.1575, 1.2015, 0),
  550. 2 => array(0, 1, 1.3333, 1.7778, 2.2100, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0),
  551. );
  552. $ratio = (float) (isset($lookup[$mpeg_version][$rawaspectratio]) ? $lookup[$mpeg_version][$rawaspectratio] : 0);
  553. if ($mpeg_version == 2 && $ratio != 1) {
  554. // Calculate pixel aspect ratio from MPEG-2 display aspect ratio
  555. $ratio = $ratio * $height / $width;
  556. }
  557. return $ratio;
  558. }
  559. /**
  560. * @param int $rawaspectratio
  561. * @param int $mpeg_version
  562. *
  563. * @return string
  564. */
  565. public static function videoAspectRatioTextLookup($rawaspectratio, $mpeg_version=1) {
  566. $lookup = array(
  567. 1 => array('forbidden', 'square pixels', '0.6735', '16:9, 625 line, PAL', '0.7615', '0.8055', '16:9, 525 line, NTSC', '0.8935', '4:3, 625 line, PAL, CCIR601', '0.9815', '1.0255', '1.0695', '4:3, 525 line, NTSC, CCIR601', '1.1575', '1.2015', 'reserved'),
  568. 2 => array('forbidden', 'square pixels', '4:3', '16:9', '2.21:1', 'reserved', 'reserved', 'reserved', 'reserved', 'reserved', 'reserved', 'reserved', 'reserved', 'reserved', 'reserved', 'reserved'), // http://dvd.sourceforge.net/dvdinfo/mpeghdrs.html
  569. );
  570. return (isset($lookup[$mpeg_version][$rawaspectratio]) ? $lookup[$mpeg_version][$rawaspectratio] : '');
  571. }
  572. /**
  573. * @param int $video_format
  574. *
  575. * @return string
  576. */
  577. public static function videoFormatTextLookup($video_format) {
  578. // ISO/IEC 13818-2, section 6.3.6, Table 6-6. Meaning of video_format
  579. $lookup = array('component', 'PAL', 'NTSC', 'SECAM', 'MAC', 'Unspecified video format', 'reserved(6)', 'reserved(7)');
  580. return (isset($lookup[$video_format]) ? $lookup[$video_format] : '');
  581. }
  582. /**
  583. * @param int $scalable_mode
  584. *
  585. * @return string
  586. */
  587. public static function scalableModeTextLookup($scalable_mode) {
  588. // ISO/IEC 13818-2, section 6.3.8, Table 6-10. Definition of scalable_mode
  589. $lookup = array('data partitioning', 'spatial scalability', 'SNR scalability', 'temporal scalability');
  590. return (isset($lookup[$scalable_mode]) ? $lookup[$scalable_mode] : '');
  591. }
  592. /**
  593. * @param int $picture_structure
  594. *
  595. * @return string
  596. */
  597. public static function pictureStructureTextLookup($picture_structure) {
  598. // ISO/IEC 13818-2, section 6.3.11, Table 6-14 Meaning of picture_structure
  599. $lookup = array('reserved', 'Top Field', 'Bottom Field', 'Frame picture');
  600. return (isset($lookup[$picture_structure]) ? $lookup[$picture_structure] : '');
  601. }
  602. /**
  603. * @param int $chroma_format
  604. *
  605. * @return string
  606. */
  607. public static function chromaFormatTextLookup($chroma_format) {
  608. // ISO/IEC 13818-2, section 6.3.11, Table 6-14 Meaning of picture_structure
  609. $lookup = array('reserved', '4:2:0', '4:2:2', '4:4:4');
  610. return (isset($lookup[$chroma_format]) ? $lookup[$chroma_format] : '');
  611. }
  612. }