File indexing completed on 2024-12-22 05:33:12

0001 <?php
0002 /////////////////////////////////////////////////////////////////
0003 /// getID3() by James Heinrich <info@getid3.org>               //
0004 //  available at http://getid3.sourceforge.net                 //
0005 //            or http://www.getid3.org                         //
0006 //          also https://github.com/JamesHeinrich/getID3       //
0007 /////////////////////////////////////////////////////////////////
0008 // See readme.txt for more details                             //
0009 /////////////////////////////////////////////////////////////////
0010 //                                                             //
0011 // module.audio.monkey.php                                     //
0012 // module for analyzing Monkey's Audio files                   //
0013 // dependencies: NONE                                          //
0014 //                                                            ///
0015 /////////////////////////////////////////////////////////////////
0016 
0017 
0018 class getid3_monkey extends getid3_handler
0019 {
0020 
0021   public function Analyze() {
0022     $info = &$this->getid3->info;
0023 
0024     // based loosely on code from TMonkey by Jurgen Faul <jfaulØgmx*de>
0025     // http://jfaul.de/atl  or  http://j-faul.virtualave.net/atl/atl.html
0026 
0027     $info['fileformat']            = 'mac';
0028     $info['audio']['dataformat']   = 'mac';
0029     $info['audio']['bitrate_mode'] = 'vbr';
0030     $info['audio']['lossless']     = true;
0031 
0032     $info['monkeys_audio']['raw'] = array();
0033     $thisfile_monkeysaudio                = &$info['monkeys_audio'];
0034     $thisfile_monkeysaudio_raw            = &$thisfile_monkeysaudio['raw'];
0035 
0036     $this->fseek($info['avdataoffset']);
0037     $MACheaderData = $this->fread(74);
0038 
0039     $thisfile_monkeysaudio_raw['magic'] = substr($MACheaderData, 0, 4);
0040     $magic = 'MAC ';
0041     if ($thisfile_monkeysaudio_raw['magic'] != $magic) {
0042       $info['error'][] = 'Expecting "'.getid3_lib::PrintHexBytes($magic).'" at offset '.$info['avdataoffset'].', found "'.getid3_lib::PrintHexBytes($thisfile_monkeysaudio_raw['magic']).'"';
0043       unset($info['fileformat']);
0044       return false;
0045     }
0046     $thisfile_monkeysaudio_raw['nVersion']             = getid3_lib::LittleEndian2Int(substr($MACheaderData, 4, 2)); // appears to be uint32 in 3.98+
0047 
0048     if ($thisfile_monkeysaudio_raw['nVersion'] < 3980) {
0049       $thisfile_monkeysaudio_raw['nCompressionLevel']    = getid3_lib::LittleEndian2Int(substr($MACheaderData, 6, 2));
0050       $thisfile_monkeysaudio_raw['nFormatFlags']         = getid3_lib::LittleEndian2Int(substr($MACheaderData, 8, 2));
0051       $thisfile_monkeysaudio_raw['nChannels']            = getid3_lib::LittleEndian2Int(substr($MACheaderData, 10, 2));
0052       $thisfile_monkeysaudio_raw['nSampleRate']          = getid3_lib::LittleEndian2Int(substr($MACheaderData, 12, 4));
0053       $thisfile_monkeysaudio_raw['nHeaderDataBytes']     = getid3_lib::LittleEndian2Int(substr($MACheaderData, 16, 4));
0054       $thisfile_monkeysaudio_raw['nWAVTerminatingBytes'] = getid3_lib::LittleEndian2Int(substr($MACheaderData, 20, 4));
0055       $thisfile_monkeysaudio_raw['nTotalFrames']         = getid3_lib::LittleEndian2Int(substr($MACheaderData, 24, 4));
0056       $thisfile_monkeysaudio_raw['nFinalFrameSamples']   = getid3_lib::LittleEndian2Int(substr($MACheaderData, 28, 4));
0057       $thisfile_monkeysaudio_raw['nPeakLevel']           = getid3_lib::LittleEndian2Int(substr($MACheaderData, 32, 4));
0058       $thisfile_monkeysaudio_raw['nSeekElements']        = getid3_lib::LittleEndian2Int(substr($MACheaderData, 38, 2));
0059       $offset = 8;
0060     } else {
0061       $offset = 8;
0062       // APE_DESCRIPTOR
0063       $thisfile_monkeysaudio_raw['nDescriptorBytes']       = getid3_lib::LittleEndian2Int(substr($MACheaderData, $offset,  4));
0064       $offset += 4;
0065       $thisfile_monkeysaudio_raw['nHeaderBytes']           = getid3_lib::LittleEndian2Int(substr($MACheaderData, $offset,  4));
0066       $offset += 4;
0067       $thisfile_monkeysaudio_raw['nSeekTableBytes']        = getid3_lib::LittleEndian2Int(substr($MACheaderData, $offset,  4));
0068       $offset += 4;
0069       $thisfile_monkeysaudio_raw['nHeaderDataBytes']       = getid3_lib::LittleEndian2Int(substr($MACheaderData, $offset,  4));
0070       $offset += 4;
0071       $thisfile_monkeysaudio_raw['nAPEFrameDataBytes']     = getid3_lib::LittleEndian2Int(substr($MACheaderData, $offset,  4));
0072       $offset += 4;
0073       $thisfile_monkeysaudio_raw['nAPEFrameDataBytesHigh'] = getid3_lib::LittleEndian2Int(substr($MACheaderData, $offset,  4));
0074       $offset += 4;
0075       $thisfile_monkeysaudio_raw['nTerminatingDataBytes']  = getid3_lib::LittleEndian2Int(substr($MACheaderData, $offset,  4));
0076       $offset += 4;
0077       $thisfile_monkeysaudio_raw['cFileMD5']               =                              substr($MACheaderData, $offset, 16);
0078       $offset += 16;
0079 
0080       // APE_HEADER
0081       $thisfile_monkeysaudio_raw['nCompressionLevel']    = getid3_lib::LittleEndian2Int(substr($MACheaderData, $offset, 2));
0082       $offset += 2;
0083       $thisfile_monkeysaudio_raw['nFormatFlags']         = getid3_lib::LittleEndian2Int(substr($MACheaderData, $offset, 2));
0084       $offset += 2;
0085       $thisfile_monkeysaudio_raw['nBlocksPerFrame']      = getid3_lib::LittleEndian2Int(substr($MACheaderData, $offset, 4));
0086       $offset += 4;
0087       $thisfile_monkeysaudio_raw['nFinalFrameBlocks']    = getid3_lib::LittleEndian2Int(substr($MACheaderData, $offset, 4));
0088       $offset += 4;
0089       $thisfile_monkeysaudio_raw['nTotalFrames']         = getid3_lib::LittleEndian2Int(substr($MACheaderData, $offset, 4));
0090       $offset += 4;
0091       $thisfile_monkeysaudio_raw['nBitsPerSample']       = getid3_lib::LittleEndian2Int(substr($MACheaderData, $offset, 2));
0092       $offset += 2;
0093       $thisfile_monkeysaudio_raw['nChannels']            = getid3_lib::LittleEndian2Int(substr($MACheaderData, $offset, 2));
0094       $offset += 2;
0095       $thisfile_monkeysaudio_raw['nSampleRate']          = getid3_lib::LittleEndian2Int(substr($MACheaderData, $offset, 4));
0096       $offset += 4;
0097     }
0098 
0099     $thisfile_monkeysaudio['flags']['8-bit']         = (bool) ($thisfile_monkeysaudio_raw['nFormatFlags'] & 0x0001);
0100     $thisfile_monkeysaudio['flags']['crc-32']        = (bool) ($thisfile_monkeysaudio_raw['nFormatFlags'] & 0x0002);
0101     $thisfile_monkeysaudio['flags']['peak_level']    = (bool) ($thisfile_monkeysaudio_raw['nFormatFlags'] & 0x0004);
0102     $thisfile_monkeysaudio['flags']['24-bit']        = (bool) ($thisfile_monkeysaudio_raw['nFormatFlags'] & 0x0008);
0103     $thisfile_monkeysaudio['flags']['seek_elements'] = (bool) ($thisfile_monkeysaudio_raw['nFormatFlags'] & 0x0010);
0104     $thisfile_monkeysaudio['flags']['no_wav_header'] = (bool) ($thisfile_monkeysaudio_raw['nFormatFlags'] & 0x0020);
0105     $thisfile_monkeysaudio['version']                = $thisfile_monkeysaudio_raw['nVersion'] / 1000;
0106     $thisfile_monkeysaudio['compression']            = $this->MonkeyCompressionLevelNameLookup($thisfile_monkeysaudio_raw['nCompressionLevel']);
0107     if ($thisfile_monkeysaudio_raw['nVersion'] < 3980) {
0108       $thisfile_monkeysaudio['samples_per_frame']      = $this->MonkeySamplesPerFrame($thisfile_monkeysaudio_raw['nVersion'], $thisfile_monkeysaudio_raw['nCompressionLevel']);
0109     }
0110     $thisfile_monkeysaudio['bits_per_sample']        = ($thisfile_monkeysaudio['flags']['24-bit'] ? 24 : ($thisfile_monkeysaudio['flags']['8-bit'] ? 8 : 16));
0111     $thisfile_monkeysaudio['channels']               = $thisfile_monkeysaudio_raw['nChannels'];
0112     $info['audio']['channels']               = $thisfile_monkeysaudio['channels'];
0113     $thisfile_monkeysaudio['sample_rate']            = $thisfile_monkeysaudio_raw['nSampleRate'];
0114     if ($thisfile_monkeysaudio['sample_rate'] == 0) {
0115       $info['error'][] = 'Corrupt MAC file: frequency == zero';
0116       return false;
0117     }
0118     $info['audio']['sample_rate']            = $thisfile_monkeysaudio['sample_rate'];
0119     if ($thisfile_monkeysaudio['flags']['peak_level']) {
0120       $thisfile_monkeysaudio['peak_level']         = $thisfile_monkeysaudio_raw['nPeakLevel'];
0121       $thisfile_monkeysaudio['peak_ratio']         = $thisfile_monkeysaudio['peak_level'] / pow(2, $thisfile_monkeysaudio['bits_per_sample'] - 1);
0122     }
0123     if ($thisfile_monkeysaudio_raw['nVersion'] >= 3980) {
0124       $thisfile_monkeysaudio['samples']            = (($thisfile_monkeysaudio_raw['nTotalFrames'] - 1) * $thisfile_monkeysaudio_raw['nBlocksPerFrame']) + $thisfile_monkeysaudio_raw['nFinalFrameBlocks'];
0125     } else {
0126       $thisfile_monkeysaudio['samples']            = (($thisfile_monkeysaudio_raw['nTotalFrames'] - 1) * $thisfile_monkeysaudio['samples_per_frame']) + $thisfile_monkeysaudio_raw['nFinalFrameSamples'];
0127     }
0128     $thisfile_monkeysaudio['playtime']               = $thisfile_monkeysaudio['samples'] / $thisfile_monkeysaudio['sample_rate'];
0129     if ($thisfile_monkeysaudio['playtime'] == 0) {
0130       $info['error'][] = 'Corrupt MAC file: playtime == zero';
0131       return false;
0132     }
0133     $info['playtime_seconds']                = $thisfile_monkeysaudio['playtime'];
0134     $thisfile_monkeysaudio['compressed_size']        = $info['avdataend'] - $info['avdataoffset'];
0135     $thisfile_monkeysaudio['uncompressed_size']      = $thisfile_monkeysaudio['samples'] * $thisfile_monkeysaudio['channels'] * ($thisfile_monkeysaudio['bits_per_sample'] / 8);
0136     if ($thisfile_monkeysaudio['uncompressed_size'] == 0) {
0137       $info['error'][] = 'Corrupt MAC file: uncompressed_size == zero';
0138       return false;
0139     }
0140     $thisfile_monkeysaudio['compression_ratio']      = $thisfile_monkeysaudio['compressed_size'] / ($thisfile_monkeysaudio['uncompressed_size'] + $thisfile_monkeysaudio_raw['nHeaderDataBytes']);
0141     $thisfile_monkeysaudio['bitrate']                = (($thisfile_monkeysaudio['samples'] * $thisfile_monkeysaudio['channels'] * $thisfile_monkeysaudio['bits_per_sample']) / $thisfile_monkeysaudio['playtime']) * $thisfile_monkeysaudio['compression_ratio'];
0142     $info['audio']['bitrate']                = $thisfile_monkeysaudio['bitrate'];
0143 
0144     // add size of MAC header to avdataoffset
0145     if ($thisfile_monkeysaudio_raw['nVersion'] >= 3980) {
0146       $info['avdataoffset'] += $thisfile_monkeysaudio_raw['nDescriptorBytes'];
0147       $info['avdataoffset'] += $thisfile_monkeysaudio_raw['nHeaderBytes'];
0148       $info['avdataoffset'] += $thisfile_monkeysaudio_raw['nSeekTableBytes'];
0149       $info['avdataoffset'] += $thisfile_monkeysaudio_raw['nHeaderDataBytes'];
0150 
0151       $info['avdataend'] -= $thisfile_monkeysaudio_raw['nTerminatingDataBytes'];
0152     } else {
0153       $info['avdataoffset'] += $offset;
0154     }
0155 
0156     if ($thisfile_monkeysaudio_raw['nVersion'] >= 3980) {
0157       if ($thisfile_monkeysaudio_raw['cFileMD5'] === str_repeat("\x00", 16)) {
0158         //$info['warning'][] = 'cFileMD5 is null';
0159       } else {
0160         $info['md5_data_source'] = '';
0161         $md5 = $thisfile_monkeysaudio_raw['cFileMD5'];
0162         for ($i = 0; $i < strlen($md5); $i++) {
0163           $info['md5_data_source'] .= str_pad(dechex(ord($md5{$i})), 2, '00', STR_PAD_LEFT);
0164         }
0165         if (!preg_match('/^[0-9a-f]{32}$/', $info['md5_data_source'])) {
0166           unset($info['md5_data_source']);
0167         }
0168       }
0169     }
0170 
0171 
0172 
0173     $info['audio']['bits_per_sample'] = $thisfile_monkeysaudio['bits_per_sample'];
0174     $info['audio']['encoder']         = 'MAC v'.number_format($thisfile_monkeysaudio['version'], 2);
0175     $info['audio']['encoder_options'] = ucfirst($thisfile_monkeysaudio['compression']).' compression';
0176 
0177     return true;
0178   }
0179 
0180   public function MonkeyCompressionLevelNameLookup($compressionlevel) {
0181     static $MonkeyCompressionLevelNameLookup = array(
0182       0     => 'unknown',
0183       1000  => 'fast',
0184       2000  => 'normal',
0185       3000  => 'high',
0186       4000  => 'extra-high',
0187       5000  => 'insane'
0188     );
0189     return (isset($MonkeyCompressionLevelNameLookup[$compressionlevel]) ? $MonkeyCompressionLevelNameLookup[$compressionlevel] : 'invalid');
0190   }
0191 
0192   public function MonkeySamplesPerFrame($versionid, $compressionlevel) {
0193     if ($versionid >= 3950) {
0194       return 73728 * 4;
0195     } elseif ($versionid >= 3900) {
0196       return 73728;
0197     } elseif (($versionid >= 3800) && ($compressionlevel == 4000)) {
0198       return 73728;
0199     } else {
0200       return 9216;
0201     }
0202   }
0203 
0204 }