// // available at http://getid3.sourceforge.net // // or http://www.getid3.org // // // // FLV module by Seth Kaufman // // // // * version 0.1 (26 June 2005) // // // // minor modifications by James Heinrich // // * version 0.1.1 (15 July 2005) // // // // Support for On2 VP6 codec and meta information by // // Steve Webster // // * version 0.2 (22 February 2006) // // // // Modified to not read entire file into memory // // by James Heinrich // // * version 0.3 (15 June 2006) // // // ///////////////////////////////////////////////////////////////// // // // module.audio-video.flv.php // // module for analyzing Shockwave Flash Video files // // dependencies: NONE // // /// ///////////////////////////////////////////////////////////////// define('GETID3_FLV_TAG_AUDIO', 8); define('GETID3_FLV_TAG_VIDEO', 9); define('GETID3_FLV_TAG_META', 18); define('GETID3_FLV_VIDEO_H263', 2); define('GETID3_FLV_VIDEO_SCREEN', 3); define('GETID3_FLV_VIDEO_VP6', 4); class getid3_flv { function getid3_flv(&$fd, &$ThisFileInfo, $ReturnAllTagData=false) { fseek($fd, $ThisFileInfo['avdataoffset'], SEEK_SET); $FLVdataLength = $ThisFileInfo['avdataend'] - $ThisFileInfo['avdataoffset']; $FLVheader = fread($fd, 5); $ThisFileInfo['fileformat'] = 'flv'; $ThisFileInfo['flv']['header']['signature'] = substr($FLVheader, 0, 3); $ThisFileInfo['flv']['header']['version'] = getid3_lib::BigEndian2Int(substr($FLVheader, 3, 1)); $TypeFlags = getid3_lib::BigEndian2Int(substr($FLVheader, 4, 1)); if ($ThisFileInfo['flv']['header']['signature'] != 'FLV') { $ThisFileInfo['error'][] = 'Expecting "FLV" at offset '.$ThisFileInfo['avdataoffset'].', found "'.$ThisFileInfo['flv']['header']['signature'].'"'; unset($ThisFileInfo['flv']); unset($ThisFileInfo['fileformat']); return false; } $ThisFileInfo['flv']['header']['hasAudio'] = (bool) ($TypeFlags & 0x04); $ThisFileInfo['flv']['header']['hasVideo'] = (bool) ($TypeFlags & 0x01); $FrameSizeDataLength = getid3_lib::BigEndian2Int(fread($fd, 4)); $FLVheaderFrameLength = 9; if ($FrameSizeDataLength > $FLVheaderFrameLength) { fseek($fd, $FrameSizeDataLength - $FLVheaderFrameLength, SEEK_CUR); } $Duration = 0; while ((ftell($fd) + 1) < $ThisFileInfo['avdataend']) { //if (!$ThisFileInfo['flv']['header']['hasAudio'] || isset($ThisFileInfo['flv']['audio']['audioFormat'])) { // if (!$ThisFileInfo['flv']['header']['hasVideo'] || isset($ThisFileInfo['flv']['video']['videoCodec'])) { // break; // } //} $ThisTagHeader = fread($fd, 16); $PreviousTagLength = getid3_lib::BigEndian2Int(substr($ThisTagHeader, 0, 4)); $TagType = getid3_lib::BigEndian2Int(substr($ThisTagHeader, 4, 1)); $DataLength = getid3_lib::BigEndian2Int(substr($ThisTagHeader, 5, 3)); $Timestamp = getid3_lib::BigEndian2Int(substr($ThisTagHeader, 8, 3)); $LastHeaderByte = getid3_lib::BigEndian2Int(substr($ThisTagHeader, 15, 1)); $NextOffset = ftell($fd) - 1 + $DataLength; switch ($TagType) { case GETID3_FLV_TAG_AUDIO: if (!isset($ThisFileInfo['flv']['audio']['audioFormat'])) { $ThisFileInfo['flv']['audio']['audioFormat'] = $LastHeaderByte & 0x07; $ThisFileInfo['flv']['audio']['audioRate'] = ($LastHeaderByte & 0x30) / 0x10; $ThisFileInfo['flv']['audio']['audioSampleSize'] = ($LastHeaderByte & 0x40) / 0x40; $ThisFileInfo['flv']['audio']['audioType'] = ($LastHeaderByte & 0x80) / 0x80; } break; case GETID3_FLV_TAG_VIDEO: if (!isset($ThisFileInfo['flv']['video']['videoCodec'])) { $ThisFileInfo['flv']['video']['videoCodec'] = $LastHeaderByte & 0x07; $FLVvideoHeader = fread($fd, 11); if ($ThisFileInfo['flv']['video']['videoCodec'] != GETID3_FLV_VIDEO_VP6) { $PictureSizeType = (getid3_lib::BigEndian2Int(substr($FLVvideoHeader, 3, 2))) >> 7; $PictureSizeType = $PictureSizeType & 0x0007; $ThisFileInfo['flv']['header']['videoSizeType'] = $PictureSizeType; switch ($PictureSizeType) { case 0: $PictureSizeEnc = getid3_lib::BigEndian2Int(substr($FLVvideoHeader, 5, 2)); $PictureSizeEnc <<= 1; $ThisFileInfo['video']['resolution_x'] = ($PictureSizeEnc & 0xFF00) >> 8; $PictureSizeEnc = getid3_lib::BigEndian2Int(substr($FLVvideoHeader, 6, 2)); $PictureSizeEnc <<= 1; $ThisFileInfo['video']['resolution_y'] = ($PictureSizeEnc & 0xFF00) >> 8; break; case 1: $PictureSizeEnc = getid3_lib::BigEndian2Int(substr($FLVvideoHeader, 5, 4)); $PictureSizeEnc <<= 1; $ThisFileInfo['video']['resolution_x'] = ($PictureSizeEnc & 0xFFFF0000) >> 16; $PictureSizeEnc = getid3_lib::BigEndian2Int(substr($FLVvideoHeader, 7, 4)); $PictureSizeEnc <<= 1; $ThisFileInfo['video']['resolution_y'] = ($PictureSizeEnc & 0xFFFF0000) >> 16; break; case 2: $ThisFileInfo['video']['resolution_x'] = 352; $ThisFileInfo['video']['resolution_y'] = 288; break; case 3: $ThisFileInfo['video']['resolution_x'] = 176; $ThisFileInfo['video']['resolution_y'] = 144; break; case 4: $ThisFileInfo['video']['resolution_x'] = 128; $ThisFileInfo['video']['resolution_y'] = 96; break; case 5: $ThisFileInfo['video']['resolution_x'] = 320; $ThisFileInfo['video']['resolution_y'] = 240; break; case 6: $ThisFileInfo['video']['resolution_x'] = 160; $ThisFileInfo['video']['resolution_y'] = 120; break; default: $ThisFileInfo['video']['resolution_x'] = 0; $ThisFileInfo['video']['resolution_y'] = 0; break; } } } break; // Meta tag case GETID3_FLV_TAG_META: fseek($fd, -1, SEEK_CUR); $reader = new AMFReader(new AMFStream(fread($fd, $DataLength))); $eventName = $reader->readData(); $ThisFileInfo['meta'][$eventName] = $reader->readData(); unset($reader); $ThisFileInfo['video']['frame_rate'] = @$ThisFileInfo['meta']['onMetaData']['framerate']; $ThisFileInfo['video']['resolution_x'] = @$ThisFileInfo['meta']['onMetaData']['width']; $ThisFileInfo['video']['resolution_y'] = @$ThisFileInfo['meta']['onMetaData']['height']; break; default: // noop break; } if ($Timestamp > $Duration) { $Duration = $Timestamp; } fseek($fd, $NextOffset, SEEK_SET); } if ($ThisFileInfo['playtime_seconds'] = $Duration / 1000) { $ThisFileInfo['bitrate'] = ($ThisFileInfo['avdataend'] - $ThisFileInfo['avdataoffset']) / $ThisFileInfo['playtime_seconds']; } if ($ThisFileInfo['flv']['header']['hasAudio']) { $ThisFileInfo['audio']['codec'] = $this->FLVaudioFormat($ThisFileInfo['flv']['audio']['audioFormat']); $ThisFileInfo['audio']['sample_rate'] = $this->FLVaudioRate($ThisFileInfo['flv']['audio']['audioRate']); $ThisFileInfo['audio']['bits_per_sample'] = $this->FLVaudioBitDepth($ThisFileInfo['flv']['audio']['audioSampleSize']); $ThisFileInfo['audio']['channels'] = $ThisFileInfo['flv']['audio']['audioType'] + 1; // 0=mono,1=stereo $ThisFileInfo['audio']['lossless'] = ($ThisFileInfo['flv']['audio']['audioFormat'] ? false : true); // 0=uncompressed $ThisFileInfo['audio']['dataformat'] = 'flv'; } if (@$ThisFileInfo['flv']['header']['hasVideo']) { $ThisFileInfo['video']['codec'] = $this->FLVvideoCodec($ThisFileInfo['flv']['video']['videoCodec']); $ThisFileInfo['video']['dataformat'] = 'flv'; $ThisFileInfo['video']['lossless'] = false; } return true; } function FLVaudioFormat($id) { $FLVaudioFormat = array( 0 => 'uncompressed', 1 => 'ADPCM', 2 => 'mp3', 5 => 'Nellymoser 8kHz mono', 6 => 'Nellymoser', ); return (@$FLVaudioFormat[$id] ? @$FLVaudioFormat[$id] : false); } function FLVaudioRate($id) { $FLVaudioRate = array( 0 => 5500, 1 => 11025, 2 => 22050, 3 => 44100, ); return (@$FLVaudioRate[$id] ? @$FLVaudioRate[$id] : false); } function FLVaudioBitDepth($id) { $FLVaudioBitDepth = array( 0 => 8, 1 => 16, ); return (@$FLVaudioBitDepth[$id] ? @$FLVaudioBitDepth[$id] : false); } function FLVvideoCodec($id) { $FLVvideoCodec = array( GETID3_FLV_VIDEO_H263 => 'Sorenson H.263', GETID3_FLV_VIDEO_SCREEN => 'Screen video', GETID3_FLV_VIDEO_VP6 => 'On2 VP6', ); return (@$FLVvideoCodec[$id] ? @$FLVvideoCodec[$id] : false); } } class AMFStream { var $bytes; var $pos; function AMFStream(&$bytes) { $this->bytes =& $bytes; $this->pos = 0; } function readByte() { return getid3_lib::BigEndian2Int(substr($this->bytes, $this->pos++, 1)); } function readInt() { return ($this->readByte() << 8) + $this->readByte(); } function readLong() { return ($this->readByte() << 24) + ($this->readByte() << 16) + ($this->readByte() << 8) + $this->readByte(); } function readDouble() { return getid3_lib::BigEndian2Float($this->read(8)); } function readUTF() { $length = $this->readInt(); return $this->read($length); } function readLongUTF() { $length = $this->readLong(); return $this->read($length); } function read($length) { $val = substr($this->bytes, $this->pos, $length); $this->pos += $length; return $val; } function peekByte() { $pos = $this->pos; $val = $this->readByte(); $this->pos = $pos; return $val; } function peekInt() { $pos = $this->pos; $val = $this->readInt(); $this->pos = $pos; return $val; } function peekLong() { $pos = $this->pos; $val = $this->readLong(); $this->pos = $pos; return $val; } function peekDouble() { $pos = $this->pos; $val = $this->readDouble(); $this->pos = $pos; return $val; } function peekUTF() { $pos = $this->pos; $val = $this->readUTF(); $this->pos = $pos; return $val; } function peekLongUTF() { $pos = $this->pos; $val = $this->readLongUTF(); $this->pos = $pos; return $val; } } class AMFReader { var $stream; function AMFReader(&$stream) { $this->stream =& $stream; } function readData() { $value = null; $type = $this->stream->readByte(); switch($type) { // Double case 0: $value = $this->readDouble(); break; // Boolean case 1: $value = $this->readBoolean(); break; // String case 2: $value = $this->readString(); break; // Object case 3: $value = $this->readObject(); break; // null case 6: return null; break; // Mixed array case 8: $value = $this->readMixedArray(); break; // Array case 10: $value = $this->readArray(); break; // Date case 11: $value = $this->readDate(); break; // Long string case 13: $value = $this->readLongString(); break; // XML (handled as string) case 15: $value = $this->readXML(); break; // Typed object (handled as object) case 16: $value = $this->readTypedObject(); break; // Long string default: $value = '(unknown or unsupported data type)'; break; } return $value; } function readDouble() { return $this->stream->readDouble(); } function readBoolean() { return $this->stream->readByte() == 1; } function readString() { return $this->stream->readUTF(); } function readObject() { // Get highest numerical index - ignored $highestIndex = $this->stream->readLong(); $data = array(); while ($key = $this->stream->readUTF()) { // Mixed array record ends with empty string (0x00 0x00) and 0x09 if (($key == '') && ($this->stream->peekByte() == 0x09)) { // Consume byte $this->stream->readByte(); break; } $data[$key] = $this->readData(); } return $data; } function readMixedArray() { // Get highest numerical index - ignored $highestIndex = $this->stream->readLong(); $data = array(); while ($key = $this->stream->readUTF()) { // Mixed array record ends with empty string (0x00 0x00) and 0x09 if (($key == '') && ($this->stream->peekByte() == 0x09)) { // Consume byte $this->stream->readByte(); break; } if (is_numeric($key)) { $key = (float) $key; } $data[$key] = $this->readData(); } return $data; } function readArray() { $length = $this->stream->readLong(); $data = array(); for ($i = 0; $i < count($length); $i++) { $data[] = $this->readData(); } return $data; } function readDate() { $timestamp = $this->stream->readDouble(); $timezone = $this->stream->readInt(); return $timestamp; } function readLongString() { return $this->stream->readLongUTF(); } function readXML() { return $this->stream->readLongUTF(); } function readTypedObject() { $className = $this->stream->readUTF(); return $this->readObject(); } } ?>