Index: trunk/extensions/TimedMediaHandler/getid3/module.audio-video.matroska.php |
— | — | @@ -0,0 +1,1748 @@ |
| 2 | +<?php |
| 3 | +///////////////////////////////////////////////////////////////// |
| 4 | +/// getID3() by James Heinrich <info@getid3.org> // |
| 5 | +// available at http://getid3.sourceforge.net // |
| 6 | +// or http://www.getid3.org // |
| 7 | +///////////////////////////////////////////////////////////////// |
| 8 | +// See readme.txt for more details // |
| 9 | +///////////////////////////////////////////////////////////////// |
| 10 | +// // |
| 11 | +// module.audio-video.matriska.php // |
| 12 | +// module for analyzing Matroska containers // |
| 13 | +// dependencies: NONE // |
| 14 | +// /// |
| 15 | +///////////////////////////////////////////////////////////////// |
| 16 | + |
| 17 | + |
| 18 | +define('EBML_ID_CHAPTERS', 0x0043A770); // [10][43][A7][70] -- A system to define basic menus and partition data. For more detailed information, look at the Chapters Explanation. |
| 19 | +define('EBML_ID_SEEKHEAD', 0x014D9B74); // [11][4D][9B][74] -- Contains the position of other level 1 elements. |
| 20 | +define('EBML_ID_TAGS', 0x0254C367); // [12][54][C3][67] -- Element containing elements specific to Tracks/Chapters. A list of valid tags can be found here. |
| 21 | +define('EBML_ID_INFO', 0x0549A966); // [15][49][A9][66] -- Contains miscellaneous general information and statistics on the file. |
| 22 | +define('EBML_ID_TRACKS', 0x0654AE6B); // [16][54][AE][6B] -- A top-level block of information with many tracks described. |
| 23 | +define('EBML_ID_SEGMENT', 0x08538067); // [18][53][80][67] -- This element contains all other top-level (level 1) elements. Typically a Matroska file is composed of 1 segment. |
| 24 | +define('EBML_ID_ATTACHMENTS', 0x0941A469); // [19][41][A4][69] -- Contain attached files. |
| 25 | +define('EBML_ID_EBML', 0x0A45DFA3); // [1A][45][DF][A3] -- Set the EBML characteristics of the data to follow. Each EBML document has to start with this. |
| 26 | +define('EBML_ID_CUES', 0x0C53BB6B); // [1C][53][BB][6B] -- A top-level element to speed seeking access. All entries are local to the segment. |
| 27 | +define('EBML_ID_CLUSTER', 0x0F43B675); // [1F][43][B6][75] -- The lower level element containing the (monolithic) Block structure. |
| 28 | +define('EBML_ID_LANGUAGE', 0x02B59C); // [22][B5][9C] -- Specifies the language of the track in the Matroska languages form. |
| 29 | +define('EBML_ID_TRACKTIMECODESCALE', 0x03314F); // [23][31][4F] -- The scale to apply on this track to work at normal speed in relation with other tracks (mostly used to adjust video speed when the audio length differs). |
| 30 | +define('EBML_ID_DEFAULTDURATION', 0x03E383); // [23][E3][83] -- Number of nanoseconds (i.e. not scaled) per frame. |
| 31 | +define('EBML_ID_CODECNAME', 0x058688); // [25][86][88] -- A human-readable string specifying the codec. |
| 32 | +define('EBML_ID_CODECDOWNLOADURL', 0x06B240); // [26][B2][40] -- A URL to download about the codec used. |
| 33 | +define('EBML_ID_TIMECODESCALE', 0x0AD7B1); // [2A][D7][B1] -- Timecode scale in nanoseconds (1.000.000 means all timecodes in the segment are expressed in milliseconds). |
| 34 | +define('EBML_ID_COLOURSPACE', 0x0EB524); // [2E][B5][24] -- Same value as in AVI (32 bits). |
| 35 | +define('EBML_ID_GAMMAVALUE', 0x0FB523); // [2F][B5][23] -- Gamma Value. |
| 36 | +define('EBML_ID_CODECSETTINGS', 0x1A9697); // [3A][96][97] -- A string describing the encoding setting used. |
| 37 | +define('EBML_ID_CODECINFOURL', 0x1B4040); // [3B][40][40] -- A URL to find information about the codec used. |
| 38 | +define('EBML_ID_PREVFILENAME', 0x1C83AB); // [3C][83][AB] -- An escaped filename corresponding to the previous segment. |
| 39 | +define('EBML_ID_PREVUID', 0x1CB923); // [3C][B9][23] -- A unique ID to identify the previous chained segment (128 bits). |
| 40 | +define('EBML_ID_NEXTFILENAME', 0x1E83BB); // [3E][83][BB] -- An escaped filename corresponding to the next segment. |
| 41 | +define('EBML_ID_NEXTUID', 0x1EB923); // [3E][B9][23] -- A unique ID to identify the next chained segment (128 bits). |
| 42 | +define('EBML_ID_CONTENTCOMPALGO', 0x0254); // [42][54] -- The compression algorithm used. Algorithms that have been specified so far are: |
| 43 | +define('EBML_ID_CONTENTCOMPSETTINGS', 0x0255); // [42][55] -- Settings that might be needed by the decompressor. For Header Stripping (ContentCompAlgo=3), the bytes that were removed from the beggining of each frames of the track. |
| 44 | +define('EBML_ID_DOCTYPE', 0x0282); // [42][82] -- A string that describes the type of document that follows this EBML header ('matroska' in our case). |
| 45 | +define('EBML_ID_DOCTYPEREADVERSION', 0x0285); // [42][85] -- The minimum DocType version an interpreter has to support to read this file. |
| 46 | +define('EBML_ID_EBMLVERSION', 0x0286); // [42][86] -- The version of EBML parser used to create the file. |
| 47 | +define('EBML_ID_DOCTYPEVERSION', 0x0287); // [42][87] -- The version of DocType interpreter used to create the file. |
| 48 | +define('EBML_ID_EBMLMAXIDLENGTH', 0x02F2); // [42][F2] -- The maximum length of the IDs you'll find in this file (4 or less in Matroska). |
| 49 | +define('EBML_ID_EBMLMAXSIZELENGTH', 0x02F3); // [42][F3] -- The maximum length of the sizes you'll find in this file (8 or less in Matroska). This does not override the element size indicated at the beginning of an element. Elements that have an indicated size which is larger than what is allowed by EBMLMaxSizeLength shall be considered invalid. |
| 50 | +define('EBML_ID_EBMLREADVERSION', 0x02F7); // [42][F7] -- The minimum EBML version a parser has to support to read this file. |
| 51 | +define('EBML_ID_CHAPLANGUAGE', 0x037C); // [43][7C] -- The languages corresponding to the string, in the bibliographic ISO-639-2 form. |
| 52 | +define('EBML_ID_CHAPCOUNTRY', 0x037E); // [43][7E] -- The countries corresponding to the string, same 2 octets as in Internet domains. |
| 53 | +define('EBML_ID_SEGMENTFAMILY', 0x0444); // [44][44] -- A randomly generated unique ID that all segments related to each other must use (128 bits). |
| 54 | +define('EBML_ID_DATEUTC', 0x0461); // [44][61] -- Date of the origin of timecode (value 0), i.e. production date. |
| 55 | +define('EBML_ID_TAGLANGUAGE', 0x047A); // [44][7A] -- Specifies the language of the tag specified, in the Matroska languages form. |
| 56 | +define('EBML_ID_TAGDEFAULT', 0x0484); // [44][84] -- Indication to know if this is the default/original language to use for the given tag. |
| 57 | +define('EBML_ID_TAGBINARY', 0x0485); // [44][85] -- The values of the Tag if it is binary. Note that this cannot be used in the same SimpleTag as TagString. |
| 58 | +define('EBML_ID_TAGSTRING', 0x0487); // [44][87] -- The value of the Tag. |
| 59 | +define('EBML_ID_DURATION', 0x0489); // [44][89] -- Duration of the segment (based on TimecodeScale). |
| 60 | +define('EBML_ID_CHAPPROCESSPRIVATE', 0x050D); // [45][0D] -- Some optional data attached to the ChapProcessCodecID information. For ChapProcessCodecID = 1, it is the "DVD level" equivalent. |
| 61 | +define('EBML_ID_CHAPTERFLAGENABLED', 0x0598); // [45][98] -- Specify wether the chapter is enabled. It can be enabled/disabled by a Control Track. When disabled, the movie should skip all the content between the TimeStart and TimeEnd of this chapter. |
| 62 | +define('EBML_ID_TAGNAME', 0x05A3); // [45][A3] -- The name of the Tag that is going to be stored. |
| 63 | +define('EBML_ID_EDITIONENTRY', 0x05B9); // [45][B9] -- Contains all information about a segment edition. |
| 64 | +define('EBML_ID_EDITIONUID', 0x05BC); // [45][BC] -- A unique ID to identify the edition. It's useful for tagging an edition. |
| 65 | +define('EBML_ID_EDITIONFLAGHIDDEN', 0x05BD); // [45][BD] -- If an edition is hidden (1), it should not be available to the user interface (but still to Control Tracks). |
| 66 | +define('EBML_ID_EDITIONFLAGDEFAULT', 0x05DB); // [45][DB] -- If a flag is set (1) the edition should be used as the default one. |
| 67 | +define('EBML_ID_EDITIONFLAGORDERED', 0x05DD); // [45][DD] -- Specify if the chapters can be defined multiple times and the order to play them is enforced. |
| 68 | +define('EBML_ID_FILEDATA', 0x065C); // [46][5C] -- The data of the file. |
| 69 | +define('EBML_ID_FILEMIMETYPE', 0x0660); // [46][60] -- MIME type of the file. |
| 70 | +define('EBML_ID_FILENAME', 0x066E); // [46][6E] -- Filename of the attached file. |
| 71 | +define('EBML_ID_FILEREFERRAL', 0x0675); // [46][75] -- A binary value that a track/codec can refer to when the attachment is needed. |
| 72 | +define('EBML_ID_FILEDESCRIPTION', 0x067E); // [46][7E] -- A human-friendly name for the attached file. |
| 73 | +define('EBML_ID_FILEUID', 0x06AE); // [46][AE] -- Unique ID representing the file, as random as possible. |
| 74 | +define('EBML_ID_CONTENTENCALGO', 0x07E1); // [47][E1] -- The encryption algorithm used. The value '0' means that the contents have not been encrypted but only signed. Predefined values: |
| 75 | +define('EBML_ID_CONTENTENCKEYID', 0x07E2); // [47][E2] -- For public key algorithms this is the ID of the public key the the data was encrypted with. |
| 76 | +define('EBML_ID_CONTENTSIGNATURE', 0x07E3); // [47][E3] -- A cryptographic signature of the contents. |
| 77 | +define('EBML_ID_CONTENTSIGKEYID', 0x07E4); // [47][E4] -- This is the ID of the private key the data was signed with. |
| 78 | +define('EBML_ID_CONTENTSIGALGO', 0x07E5); // [47][E5] -- The algorithm used for the signature. A value of '0' means that the contents have not been signed but only encrypted. Predefined values: |
| 79 | +define('EBML_ID_CONTENTSIGHASHALGO', 0x07E6); // [47][E6] -- The hash algorithm used for the signature. A value of '0' means that the contents have not been signed but only encrypted. Predefined values: |
| 80 | +define('EBML_ID_MUXINGAPP', 0x0D80); // [4D][80] -- Muxing application or library ("libmatroska-0.4.3"). |
| 81 | +define('EBML_ID_SEEK', 0x0DBB); // [4D][BB] -- Contains a single seek entry to an EBML element. |
| 82 | +define('EBML_ID_CONTENTENCODINGORDER', 0x1031); // [50][31] -- Tells when this modification was used during encoding/muxing starting with 0 and counting upwards. The decoder/demuxer has to start with the highest order number it finds and work its way down. This value has to be unique over all ContentEncodingOrder elements in the segment. |
| 83 | +define('EBML_ID_CONTENTENCODINGSCOPE', 0x1032); // [50][32] -- A bit field that describes which elements have been modified in this way. Values (big endian) can be OR'ed. Possible values: |
| 84 | +define('EBML_ID_CONTENTENCODINGTYPE', 0x1033); // [50][33] -- A value describing what kind of transformation has been done. Possible values: |
| 85 | +define('EBML_ID_CONTENTCOMPRESSION', 0x1034); // [50][34] -- Settings describing the compression used. Must be present if the value of ContentEncodingType is 0 and absent otherwise. Each block must be decompressable even if no previous block is available in order not to prevent seeking. |
| 86 | +define('EBML_ID_CONTENTENCRYPTION', 0x1035); // [50][35] -- Settings describing the encryption used. Must be present if the value of ContentEncodingType is 1 and absent otherwise. |
| 87 | +define('EBML_ID_CUEREFNUMBER', 0x135F); // [53][5F] -- Number of the referenced Block of Track X in the specified Cluster. |
| 88 | +define('EBML_ID_NAME', 0x136E); // [53][6E] -- A human-readable track name. |
| 89 | +define('EBML_ID_CUEBLOCKNUMBER', 0x1378); // [53][78] -- Number of the Block in the specified Cluster. |
| 90 | +define('EBML_ID_TRACKOFFSET', 0x137F); // [53][7F] -- A value to add to the Block's Timecode. This can be used to adjust the playback offset of a track. |
| 91 | +define('EBML_ID_SEEKID', 0x13AB); // [53][AB] -- The binary ID corresponding to the element name. |
| 92 | +define('EBML_ID_SEEKPOSITION', 0x13AC); // [53][AC] -- The position of the element in the segment in octets (0 = first level 1 element). |
| 93 | +define('EBML_ID_STEREOMODE', 0x13B8); // [53][B8] -- Stereo-3D video mode on 2 bits (0: mono, 1: right eye, 2: left eye, 3: both eyes). |
| 94 | +define('EBML_ID_PIXELCROPBOTTOM', 0x14AA); // [54][AA] -- The number of video pixels to remove at the bottom of the image (for HDTV content). |
| 95 | +define('EBML_ID_DISPLAYWIDTH', 0x14B0); // [54][B0] -- Width of the video frames to display. |
| 96 | +define('EBML_ID_DISPLAYUNIT', 0x14B2); // [54][B2] -- Type of the unit for DisplayWidth/Height (0: pixels, 1: centimeters, 2: inches). |
| 97 | +define('EBML_ID_ASPECTRATIOTYPE', 0x14B3); // [54][B3] -- Specify the possible modifications to the aspect ratio (0: free resizing, 1: keep aspect ratio, 2: fixed). |
| 98 | +define('EBML_ID_DISPLAYHEIGHT', 0x14BA); // [54][BA] -- Height of the video frames to display. |
| 99 | +define('EBML_ID_PIXELCROPTOP', 0x14BB); // [54][BB] -- The number of video pixels to remove at the top of the image. |
| 100 | +define('EBML_ID_PIXELCROPLEFT', 0x14CC); // [54][CC] -- The number of video pixels to remove on the left of the image. |
| 101 | +define('EBML_ID_PIXELCROPRIGHT', 0x14DD); // [54][DD] -- The number of video pixels to remove on the right of the image. |
| 102 | +define('EBML_ID_FLAGFORCED', 0x15AA); // [55][AA] -- Set if that track MUST be used during playback. There can be many forced track for a kind (audio, video or subs), the player should select the one which language matches the user preference or the default + forced track. Overlay MAY happen between a forced and non-forced track of the same kind. |
| 103 | +define('EBML_ID_MAXBLOCKADDITIONID', 0x15EE); // [55][EE] -- The maximum value of BlockAddID. A value 0 means there is no BlockAdditions for this track. |
| 104 | +define('EBML_ID_WRITINGAPP', 0x1741); // [57][41] -- Writing application ("mkvmerge-0.3.3"). |
| 105 | +define('EBML_ID_CLUSTERSILENTTRACKS', 0x1854); // [58][54] -- The list of tracks that are not used in that part of the stream. It is useful when using overlay tracks on seeking. Then you should decide what track to use. |
| 106 | +define('EBML_ID_CLUSTERSILENTTRACKNUMBER', 0x18D7); // [58][D7] -- One of the track number that are not used from now on in the stream. It could change later if not specified as silent in a further Cluster. |
| 107 | +define('EBML_ID_ATTACHEDFILE', 0x21A7); // [61][A7] -- An attached file. |
| 108 | +define('EBML_ID_CONTENTENCODING', 0x2240); // [62][40] -- Settings for one content encoding like compression or encryption. |
| 109 | +define('EBML_ID_BITDEPTH', 0x2264); // [62][64] -- Bits per sample, mostly used for PCM. |
| 110 | +define('EBML_ID_CODECPRIVATE', 0x23A2); // [63][A2] -- Private data only known to the codec. |
| 111 | +define('EBML_ID_TARGETS', 0x23C0); // [63][C0] -- Contain all UIDs where the specified meta data apply. It is void to describe everything in the segment. |
| 112 | +define('EBML_ID_CHAPTERPHYSICALEQUIV', 0x23C3); // [63][C3] -- Specify the physical equivalent of this ChapterAtom like "DVD" (60) or "SIDE" (50), see complete list of values. |
| 113 | +define('EBML_ID_TAGCHAPTERUID', 0x23C4); // [63][C4] -- A unique ID to identify the Chapter(s) the tags belong to. If the value is 0 at this level, the tags apply to all chapters in the Segment. |
| 114 | +define('EBML_ID_TAGTRACKUID', 0x23C5); // [63][C5] -- A unique ID to identify the Track(s) the tags belong to. If the value is 0 at this level, the tags apply to all tracks in the Segment. |
| 115 | +define('EBML_ID_ATTACHMENTUID', 0x23C6); // [63][C6] -- A unique ID to identify the Attachment(s) the tags belong to. If the value is 0 at this level, the tags apply to all the attachments in the Segment. |
| 116 | +define('EBML_ID_TAGEDITIONUID', 0x23C9); // [63][C9] -- A unique ID to identify the EditionEntry(s) the tags belong to. If the value is 0 at this level, the tags apply to all editions in the Segment. |
| 117 | +define('EBML_ID_TARGETTYPE', 0x23CA); // [63][CA] -- An informational string that can be used to display the logical level of the target like "ALBUM", "TRACK", "MOVIE", "CHAPTER", etc (see TargetType). |
| 118 | +define('EBML_ID_TRACKTRANSLATE', 0x2624); // [66][24] -- The track identification for the given Chapter Codec. |
| 119 | +define('EBML_ID_TRACKTRANSLATETRACKID', 0x26A5); // [66][A5] -- The binary value used to represent this track in the chapter codec data. The format depends on the ChapProcessCodecID used. |
| 120 | +define('EBML_ID_TRACKTRANSLATECODEC', 0x26BF); // [66][BF] -- The chapter codec using this ID (0: Matroska Script, 1: DVD-menu). |
| 121 | +define('EBML_ID_TRACKTRANSLATEEDITIONUID', 0x26FC); // [66][FC] -- Specify an edition UID on which this translation applies. When not specified, it means for all editions found in the segment. |
| 122 | +define('EBML_ID_SIMPLETAG', 0x27C8); // [67][C8] -- Contains general information about the target. |
| 123 | +define('EBML_ID_TARGETTYPEVALUE', 0x28CA); // [68][CA] -- A number to indicate the logical level of the target (see TargetType). |
| 124 | +define('EBML_ID_CHAPPROCESSCOMMAND', 0x2911); // [69][11] -- Contains all the commands associated to the Atom. |
| 125 | +define('EBML_ID_CHAPPROCESSTIME', 0x2922); // [69][22] -- Defines when the process command should be handled (0: during the whole chapter, 1: before starting playback, 2: after playback of the chapter). |
| 126 | +define('EBML_ID_CHAPTERTRANSLATE', 0x2924); // [69][24] -- A tuple of corresponding ID used by chapter codecs to represent this segment. |
| 127 | +define('EBML_ID_CHAPPROCESSDATA', 0x2933); // [69][33] -- Contains the command information. The data should be interpreted depending on the ChapProcessCodecID value. For ChapProcessCodecID = 1, the data correspond to the binary DVD cell pre/post commands. |
| 128 | +define('EBML_ID_CHAPPROCESS', 0x2944); // [69][44] -- Contains all the commands associated to the Atom. |
| 129 | +define('EBML_ID_CHAPPROCESSCODECID', 0x2955); // [69][55] -- Contains the type of the codec used for the processing. A value of 0 means native Matroska processing (to be defined), a value of 1 means the DVD command set is used. More codec IDs can be added later. |
| 130 | +define('EBML_ID_CHAPTERTRANSLATEID', 0x29A5); // [69][A5] -- The binary value used to represent this segment in the chapter codec data. The format depends on the ChapProcessCodecID used. |
| 131 | +define('EBML_ID_CHAPTERTRANSLATECODEC', 0x29BF); // [69][BF] -- The chapter codec using this ID (0: Matroska Script, 1: DVD-menu). |
| 132 | +define('EBML_ID_CHAPTERTRANSLATEEDITIONUID', 0x29FC); // [69][FC] -- Specify an edition UID on which this correspondance applies. When not specified, it means for all editions found in the segment. |
| 133 | +define('EBML_ID_CONTENTENCODINGS', 0x2D80); // [6D][80] -- Settings for several content encoding mechanisms like compression or encryption. |
| 134 | +define('EBML_ID_MINCACHE', 0x2DE7); // [6D][E7] -- The minimum number of frames a player should be able to cache during playback. If set to 0, the reference pseudo-cache system is not used. |
| 135 | +define('EBML_ID_MAXCACHE', 0x2DF8); // [6D][F8] -- The maximum cache size required to store referenced frames in and the current frame. 0 means no cache is needed. |
| 136 | +define('EBML_ID_CHAPTERSEGMENTUID', 0x2E67); // [6E][67] -- A segment to play in place of this chapter. Edition ChapterSegmentEditionUID should be used for this segment, otherwise no edition is used. |
| 137 | +define('EBML_ID_CHAPTERSEGMENTEDITIONUID', 0x2EBC); // [6E][BC] -- The edition to play from the segment linked in ChapterSegmentUID. |
| 138 | +define('EBML_ID_TRACKOVERLAY', 0x2FAB); // [6F][AB] -- Specify that this track is an overlay track for the Track specified (in the u-integer). That means when this track has a gap (see SilentTracks) the overlay track should be used instead. The order of multiple TrackOverlay matters, the first one is the one that should be used. If not found it should be the second, etc. |
| 139 | +define('EBML_ID_TAG', 0x3373); // [73][73] -- Element containing elements specific to Tracks/Chapters. |
| 140 | +define('EBML_ID_SEGMENTFILENAME', 0x3384); // [73][84] -- A filename corresponding to this segment. |
| 141 | +define('EBML_ID_SEGMENTUID', 0x33A4); // [73][A4] -- A randomly generated unique ID to identify the current segment between many others (128 bits). |
| 142 | +define('EBML_ID_CHAPTERUID', 0x33C4); // [73][C4] -- A unique ID to identify the Chapter. |
| 143 | +define('EBML_ID_TRACKUID', 0x33C5); // [73][C5] -- A unique ID to identify the Track. This should be kept the same when making a direct stream copy of the Track to another file. |
| 144 | +define('EBML_ID_ATTACHMENTLINK', 0x3446); // [74][46] -- The UID of an attachment that is used by this codec. |
| 145 | +define('EBML_ID_CLUSTERBLOCKADDITIONS', 0x35A1); // [75][A1] -- Contain additional blocks to complete the main one. An EBML parser that has no knowledge of the Block structure could still see and use/skip these data. |
| 146 | +define('EBML_ID_CHANNELPOSITIONS', 0x347B); // [7D][7B] -- Table of horizontal angles for each successive channel, see appendix. |
| 147 | +define('EBML_ID_OUTPUTSAMPLINGFREQUENCY', 0x38B5); // [78][B5] -- Real output sampling frequency in Hz (used for SBR techniques). |
| 148 | +define('EBML_ID_TITLE', 0x3BA9); // [7B][A9] -- General name of the segment. |
| 149 | +define('EBML_ID_CHAPTERDISPLAY', 0x00); // [80] -- Contains all possible strings to use for the chapter display. |
| 150 | +define('EBML_ID_TRACKTYPE', 0x03); // [83] -- A set of track types coded on 8 bits (1: video, 2: audio, 3: complex, 0x10: logo, 0x11: subtitle, 0x12: buttons, 0x20: control). |
| 151 | +define('EBML_ID_CHAPSTRING', 0x05); // [85] -- Contains the string to use as the chapter atom. |
| 152 | +define('EBML_ID_CODECID', 0x06); // [86] -- An ID corresponding to the codec, see the codec page for more info. |
| 153 | +define('EBML_ID_FLAGDEFAULT', 0x08); // [88] -- Set if that track (audio, video or subs) SHOULD be used if no language found matches the user preference. |
| 154 | +define('EBML_ID_CHAPTERTRACKNUMBER', 0x09); // [89] -- UID of the Track to apply this chapter too. In the absense of a control track, choosing this chapter will select the listed Tracks and deselect unlisted tracks. Absense of this element indicates that the Chapter should be applied to any currently used Tracks. |
| 155 | +define('EBML_ID_CLUSTERSLICES', 0x0E); // [8E] -- Contains slices description. |
| 156 | +define('EBML_ID_CHAPTERTRACK', 0x0F); // [8F] -- List of tracks on which the chapter applies. If this element is not present, all tracks apply |
| 157 | +define('EBML_ID_CHAPTERTIMESTART', 0x11); // [91] -- Timecode of the start of Chapter (not scaled). |
| 158 | +define('EBML_ID_CHAPTERTIMEEND', 0x12); // [92] -- Timecode of the end of Chapter (timecode excluded, not scaled). |
| 159 | +define('EBML_ID_CUEREFTIME', 0x16); // [96] -- Timecode of the referenced Block. |
| 160 | +define('EBML_ID_CUEREFCLUSTER', 0x17); // [97] -- Position of the Cluster containing the referenced Block. |
| 161 | +define('EBML_ID_CHAPTERFLAGHIDDEN', 0x18); // [98] -- If a chapter is hidden (1), it should not be available to the user interface (but still to Control Tracks). |
| 162 | +define('EBML_ID_FLAGINTERLACED', 0x1A); // [9A] -- Set if the video is interlaced. |
| 163 | +define('EBML_ID_CLUSTERBLOCKDURATION', 0x1B); // [9B] -- The duration of the Block (based on TimecodeScale). This element is mandatory when DefaultDuration is set for the track. When not written and with no DefaultDuration, the value is assumed to be the difference between the timecode of this Block and the timecode of the next Block in "display" order (not coding order). This element can be useful at the end of a Track (as there is not other Block available), or when there is a break in a track like for subtitle tracks. |
| 164 | +define('EBML_ID_FLAGLACING', 0x1C); // [9C] -- Set if the track may contain blocks using lacing. |
| 165 | +define('EBML_ID_CHANNELS', 0x1F); // [9F] -- Numbers of channels in the track. |
| 166 | +define('EBML_ID_CLUSTERBLOCKGROUP', 0x20); // [A0] -- Basic container of information containing a single Block or BlockVirtual, and information specific to that Block/VirtualBlock. |
| 167 | +define('EBML_ID_CLUSTERBLOCK', 0x21); // [A1] -- Block containing the actual data to be rendered and a timecode relative to the Cluster Timecode. |
| 168 | +define('EBML_ID_CLUSTERBLOCKVIRTUAL', 0x22); // [A2] -- A Block with no data. It must be stored in the stream at the place the real Block should be in display order. |
| 169 | +define('EBML_ID_CLUSTERSIMPLEBLOCK', 0x23); // [A3] -- Similar to Block but without all the extra information, mostly used to reduced overhead when no extra feature is needed. |
| 170 | +define('EBML_ID_CLUSTERCODECSTATE', 0x24); // [A4] -- The new codec state to use. Data interpretation is private to the codec. This information should always be referenced by a seek entry. |
| 171 | +define('EBML_ID_CLUSTERBLOCKADDITIONAL', 0x25); // [A5] -- Interpreted by the codec as it wishes (using the BlockAddID). |
| 172 | +define('EBML_ID_CLUSTERBLOCKMORE', 0x26); // [A6] -- Contain the BlockAdditional and some parameters. |
| 173 | +define('EBML_ID_CLUSTERPOSITION', 0x27); // [A7] -- Position of the Cluster in the segment (0 in live broadcast streams). It might help to resynchronise offset on damaged streams. |
| 174 | +define('EBML_ID_CODECDECODEALL', 0x2A); // [AA] -- The codec can decode potentially damaged data. |
| 175 | +define('EBML_ID_CLUSTERPREVSIZE', 0x2B); // [AB] -- Size of the previous Cluster, in octets. Can be useful for backward playing. |
| 176 | +define('EBML_ID_TRACKENTRY', 0x2E); // [AE] -- Describes a track with all elements. |
| 177 | +define('EBML_ID_CLUSTERENCRYPTEDBLOCK', 0x2F); // [AF] -- Similar to SimpleBlock but the data inside the Block are Transformed (encrypt and/or signed). |
| 178 | +define('EBML_ID_PIXELWIDTH', 0x30); // [B0] -- Width of the encoded video frames in pixels. |
| 179 | +define('EBML_ID_CUETIME', 0x33); // [B3] -- Absolute timecode according to the segment time base. |
| 180 | +define('EBML_ID_SAMPLINGFREQUENCY', 0x35); // [B5] -- Sampling frequency in Hz. |
| 181 | +define('EBML_ID_CHAPTERATOM', 0x36); // [B6] -- Contains the atom information to use as the chapter atom (apply to all tracks). |
| 182 | +define('EBML_ID_CUETRACKPOSITIONS', 0x37); // [B7] -- Contain positions for different tracks corresponding to the timecode. |
| 183 | +define('EBML_ID_FLAGENABLED', 0x39); // [B9] -- Set if the track is used. |
| 184 | +define('EBML_ID_PIXELHEIGHT', 0x3A); // [BA] -- Height of the encoded video frames in pixels. |
| 185 | +define('EBML_ID_CUEPOINT', 0x3B); // [BB] -- Contains all information relative to a seek point in the segment. |
| 186 | +define('EBML_ID_CRC32', 0x3F); // [BF] -- The CRC is computed on all the data of the Master element it's in, regardless of its position. It's recommended to put the CRC value at the beggining of the Master element for easier reading. All level 1 elements should include a CRC-32. |
| 187 | +define('EBML_ID_CLUSTERBLOCKADDITIONID', 0x4B); // [CB] -- The ID of the BlockAdditional element (0 is the main Block). |
| 188 | +define('EBML_ID_CLUSTERLACENUMBER', 0x4C); // [CC] -- The reverse number of the frame in the lace (0 is the last frame, 1 is the next to last, etc). While there are a few files in the wild with this element, it is no longer in use and has been deprecated. Being able to interpret this element is not required for playback. |
| 189 | +define('EBML_ID_CLUSTERFRAMENUMBER', 0x4D); // [CD] -- The number of the frame to generate from this lace with this delay (allow you to generate many frames from the same Block/Frame). |
| 190 | +define('EBML_ID_CLUSTERDELAY', 0x4E); // [CE] -- The (scaled) delay to apply to the element. |
| 191 | +define('EBML_ID_CLUSTERDURATION', 0x4F); // [CF] -- The (scaled) duration to apply to the element. |
| 192 | +define('EBML_ID_TRACKNUMBER', 0x57); // [D7] -- The track number as used in the Block Header (using more than 127 tracks is not encouraged, though the design allows an unlimited number). |
| 193 | +define('EBML_ID_CUEREFERENCE', 0x5B); // [DB] -- The Clusters containing the required referenced Blocks. |
| 194 | +define('EBML_ID_VIDEO', 0x60); // [E0] -- Video settings. |
| 195 | +define('EBML_ID_AUDIO', 0x61); // [E1] -- Audio settings. |
| 196 | +define('EBML_ID_CLUSTERTIMESLICE', 0x68); // [E8] -- Contains extra time information about the data contained in the Block. While there are a few files in the wild with this element, it is no longer in use and has been deprecated. Being able to interpret this element is not required for playback. |
| 197 | +define('EBML_ID_CUECODECSTATE', 0x6A); // [EA] -- The position of the Codec State corresponding to this Cue element. 0 means that the data is taken from the initial Track Entry. |
| 198 | +define('EBML_ID_CUEREFCODECSTATE', 0x6B); // [EB] -- The position of the Codec State corresponding to this referenced element. 0 means that the data is taken from the initial Track Entry. |
| 199 | +define('EBML_ID_VOID', 0x6C); // [EC] -- Used to void damaged data, to avoid unexpected behaviors when using damaged data. The content is discarded. Also used to reserve space in a sub-element for later use. |
| 200 | +define('EBML_ID_CLUSTERTIMECODE', 0x67); // [E7] -- Absolute timecode of the cluster (based on TimecodeScale). |
| 201 | +define('EBML_ID_CLUSTERBLOCKADDID', 0x6E); // [EE] -- An ID to identify the BlockAdditional level. |
| 202 | +define('EBML_ID_CUECLUSTERPOSITION', 0x71); // [F1] -- The position of the Cluster containing the required Block. |
| 203 | +define('EBML_ID_CUETRACK', 0x77); // [F7] -- The track for which a position is given. |
| 204 | +define('EBML_ID_CLUSTERREFERENCEPRIORITY', 0x7A); // [FA] -- This frame is referenced and has the specified cache priority. In cache only a frame of the same or higher priority can replace this frame. A value of 0 means the frame is not referenced. |
| 205 | +define('EBML_ID_CLUSTERREFERENCEBLOCK', 0x7B); // [FB] -- Timecode of another frame used as a reference (ie: B or P frame). The timecode is relative to the block it's attached to. |
| 206 | +define('EBML_ID_CLUSTERREFERENCEVIRTUAL', 0x7D); // [FD] -- Relative position of the data that should be in position of the virtual block. |
| 207 | + |
| 208 | + |
| 209 | +class getid3_matroska |
| 210 | +{ |
| 211 | + var $read_buffer_size = 32768; // size of read buffer, 32kB is default |
| 212 | + var $hide_clusters = true; // if true, do not return information about CLUSTER chunks, since there's a lot of them and they're not usually useful |
| 213 | + var $warnings = array(); |
| 214 | + |
| 215 | + function getid3_matroska(&$fd, &$ThisFileInfo) { |
| 216 | + |
| 217 | + // http://www.matroska.org/technical/specs/index.html#EBMLBasics |
| 218 | + $offset = $ThisFileInfo['avdataoffset']; |
| 219 | + $EBMLdata = ''; |
| 220 | + $EBMLdata_offset = $offset; |
| 221 | + |
| 222 | + if ($ThisFileInfo['avdataend'] > 2147483648) { |
| 223 | + $this->warnings[] = 'This version of getID3() may or may not correctly handle Matroska files larger than 2GB'; |
| 224 | + } |
| 225 | + |
| 226 | + while ($offset < $ThisFileInfo['avdataend']) { |
| 227 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 228 | + |
| 229 | + $top_element_offset = $offset; |
| 230 | + $top_element_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 231 | + $top_element_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 232 | + if ($top_element_length === false) { |
| 233 | + $this->warnings[] = 'invalid chunk length at '.$top_element_offset; |
| 234 | + $offset = pow(2, 63); |
| 235 | + break; |
| 236 | + } |
| 237 | + $top_element_endoffset = $offset + $top_element_length; |
| 238 | + switch ($top_element_id) { |
| 239 | + case EBML_ID_EBML: |
| 240 | + $ThisFileInfo['fileformat'] = 'matroska'; |
| 241 | + $ThisFileInfo['matroska']['header']['offset'] = $top_element_offset; |
| 242 | + $ThisFileInfo['matroska']['header']['length'] = $top_element_length; |
| 243 | + |
| 244 | + while ($offset < $top_element_endoffset) { |
| 245 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 246 | + $element_data = array(); |
| 247 | + $element_data_offset = $offset; |
| 248 | + $element_data['id'] = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 249 | + $element_data['id_name'] = $this->EBMLidName($element_data['id']); |
| 250 | + $element_data['length'] = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 251 | + $end_offset = $offset + $element_data['length']; |
| 252 | + |
| 253 | + switch ($element_data['id']) { |
| 254 | + case EBML_ID_VOID: // padding, ignore |
| 255 | + break; |
| 256 | + case EBML_ID_EBMLVERSION: |
| 257 | + case EBML_ID_EBMLREADVERSION: |
| 258 | + case EBML_ID_EBMLMAXIDLENGTH: |
| 259 | + case EBML_ID_EBMLMAXSIZELENGTH: |
| 260 | + case EBML_ID_DOCTYPEVERSION: |
| 261 | + case EBML_ID_DOCTYPEREADVERSION: |
| 262 | + $element_data['data'] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $element_data['length'])); |
| 263 | + break; |
| 264 | + case EBML_ID_DOCTYPE: |
| 265 | + $element_data['data'] = trim(substr($EBMLdata, $offset - $EBMLdata_offset, $element_data['length']), "\x00"); |
| 266 | + break; |
| 267 | + default: |
| 268 | + $this->warnings[] = 'Unhandled track.video element ['.basename(__FILE__).':'.__LINE__.'] ('.$element_data['id'].'::'.$element_data['id_name'].') at '.$element_data_offset; |
| 269 | + break; |
| 270 | + } |
| 271 | + $offset = $end_offset; |
| 272 | + $ThisFileInfo['matroska']['header']['elements'][] = $element_data; |
| 273 | + } |
| 274 | + break; |
| 275 | + |
| 276 | + |
| 277 | + case EBML_ID_SEGMENT: |
| 278 | + $ThisFileInfo['matroska']['segment'][0]['offset'] = $top_element_offset; |
| 279 | + $ThisFileInfo['matroska']['segment'][0]['length'] = $top_element_length; |
| 280 | + |
| 281 | + $segment_key = -1; |
| 282 | + while ($offset < $ThisFileInfo['avdataend']) { |
| 283 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 284 | + |
| 285 | + $element_data = array(); |
| 286 | + $element_data['offset'] = $offset; |
| 287 | + $element_data['id'] = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 288 | + $element_data['id_name'] = $this->EBMLidName($element_data['id']); |
| 289 | + $element_data['length'] = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 290 | + if ($element_data['length'] === false) { |
| 291 | + $this->warnings[] = 'invalid chunk length at '.$element_data['offset']; |
| 292 | + //$offset = pow(2, 63); |
| 293 | + $offset = $ThisFileInfo['avdataend']; |
| 294 | + break; |
| 295 | + } |
| 296 | + $element_end = $offset + $element_data['length']; |
| 297 | + switch ($element_data['id']) { |
| 298 | + //case EBML_ID_CLUSTER: |
| 299 | + // // too many cluster entries, probably not useful |
| 300 | + // break; |
| 301 | + case false: |
| 302 | + $this->warnings[] = 'invalid ID at '.$element_data['offset']; |
| 303 | + $offset = $element_end; |
| 304 | + continue 3; |
| 305 | + default: |
| 306 | + $ThisFileInfo['matroska']['segments'][] = $element_data; |
| 307 | + break; |
| 308 | + } |
| 309 | + $segment_key++; |
| 310 | + |
| 311 | + switch ($element_data['id']) { |
| 312 | + case EBML_ID_SEEKHEAD: // Contains the position of other level 1 elements |
| 313 | + while ($offset < $element_end) { |
| 314 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 315 | + $seek_entry = array(); |
| 316 | + $seek_entry['offset'] = $offset; |
| 317 | + $seek_entry['id'] = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 318 | + $seek_entry['id_name'] = $this->EBMLidName($seek_entry['id']); |
| 319 | + $seek_entry['length'] = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 320 | + $seek_end_offset = $offset + $seek_entry['length']; |
| 321 | + switch ($seek_entry['id']) { |
| 322 | + case EBML_ID_SEEK: // Contains a single seek entry to an EBML element |
| 323 | + while ($offset < $seek_end_offset) { |
| 324 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 325 | + $id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 326 | + $length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 327 | + $value = substr($EBMLdata, $offset - $EBMLdata_offset, $length); |
| 328 | + $offset += $length; |
| 329 | + switch ($id) { |
| 330 | + case EBML_ID_SEEKID: |
| 331 | + $dummy = 0; |
| 332 | + $seek_entry['target_id'] = $this->readEBMLint($value, $dummy); |
| 333 | + $seek_entry['target_name'] = $this->EBMLidName($seek_entry['target_id']); |
| 334 | + break; |
| 335 | + case EBML_ID_SEEKPOSITION: |
| 336 | + $seek_entry['target_offset'] = $element_data['offset'] + getid3_lib::BigEndian2Int($value); |
| 337 | + break; |
| 338 | + default: |
| 339 | + $ThisFileInfo['error'][] = 'Unhandled segment ['.basename(__FILE__).':'.__LINE__.'] ('.$id.') at '.$offset; |
| 340 | + break; |
| 341 | + } |
| 342 | + } |
| 343 | + $ThisFileInfo['matroska']['seek'][] = $seek_entry; |
| 344 | + //switch ($seek_entry['target_id']) { |
| 345 | + // case EBML_ID_CLUSTER: |
| 346 | + // // too many cluster seek points, probably not useful |
| 347 | + // break; |
| 348 | + // default: |
| 349 | + // $ThisFileInfo['matroska']['seek'][] = $seek_entry; |
| 350 | + // break; |
| 351 | + //} |
| 352 | + break; |
| 353 | + default: |
| 354 | + $this->warnings[] = 'Unhandled seekhead element ['.basename(__FILE__).':'.__LINE__.'] ('.$seek_entry['id'].') at '.$offset; |
| 355 | + break; |
| 356 | + } |
| 357 | + $offset = $seek_end_offset; |
| 358 | + } |
| 359 | + break; |
| 360 | + |
| 361 | + case EBML_ID_TRACKS: // information about all tracks in segment |
| 362 | + $ThisFileInfo['matroska']['tracks'] = $element_data; |
| 363 | + while ($offset < $element_end) { |
| 364 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 365 | + $track_entry = array(); |
| 366 | + $track_entry['offset'] = $offset; |
| 367 | + $track_entry['id'] = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 368 | + $track_entry['id_name'] = $this->EBMLidName($track_entry['id']); |
| 369 | + $track_entry['length'] = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 370 | + $track_entry_endoffset = $offset + $track_entry['length']; |
| 371 | + switch ($track_entry['id']) { |
| 372 | + case EBML_ID_TRACKENTRY: //subelements: Describes a track with all elements. |
| 373 | + while ($offset < $track_entry_endoffset) { |
| 374 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 375 | + $subelement_offset = $offset; |
| 376 | + $subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 377 | + $subelement_idname = $this->EBMLidName($subelement_id); |
| 378 | + $subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 379 | + $subelement_end = $offset + $subelement_length; |
| 380 | + switch ($subelement_id) { |
| 381 | + case EBML_ID_TRACKNUMBER: |
| 382 | + case EBML_ID_TRACKUID: |
| 383 | + case EBML_ID_TRACKTYPE: |
| 384 | + case EBML_ID_MINCACHE: |
| 385 | + case EBML_ID_MAXCACHE: |
| 386 | + case EBML_ID_MAXBLOCKADDITIONID: |
| 387 | + case EBML_ID_DEFAULTDURATION: // nanoseconds per frame |
| 388 | + $track_entry[$subelement_idname] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $subelement_length)); |
| 389 | + break; |
| 390 | + |
| 391 | + case EBML_ID_TRACKTIMECODESCALE: |
| 392 | + $track_entry[$subelement_idname] = getid3_lib::BigEndian2Float(substr($EBMLdata, $offset - $EBMLdata_offset, $subelement_length)); |
| 393 | + break; |
| 394 | + |
| 395 | + case EBML_ID_CODECID: |
| 396 | + case EBML_ID_LANGUAGE: |
| 397 | + case EBML_ID_NAME: |
| 398 | + case EBML_ID_CODECNAME: |
| 399 | + case EBML_ID_CODECPRIVATE: |
| 400 | + $track_entry[$subelement_idname] = trim(substr($EBMLdata, $offset - $EBMLdata_offset, $subelement_length), "\x00"); |
| 401 | + break; |
| 402 | + |
| 403 | + case EBML_ID_FLAGENABLED: |
| 404 | + case EBML_ID_FLAGDEFAULT: |
| 405 | + case EBML_ID_FLAGFORCED: |
| 406 | + case EBML_ID_FLAGLACING: |
| 407 | + case EBML_ID_CODECDECODEALL: |
| 408 | + $track_entry[$subelement_idname] = (bool) getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $subelement_length)); |
| 409 | + break; |
| 410 | + |
| 411 | + case EBML_ID_VIDEO: |
| 412 | + while ($offset < $subelement_end) { |
| 413 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 414 | + $sub_subelement_offset = $offset; |
| 415 | + $sub_subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 416 | + $sub_subelement_idname = $this->EBMLidName($sub_subelement_id); |
| 417 | + $sub_subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 418 | + $sub_subelement_end = $offset + $sub_subelement_length; |
| 419 | + switch ($sub_subelement_id) { |
| 420 | + case EBML_ID_PIXELWIDTH: |
| 421 | + case EBML_ID_PIXELHEIGHT: |
| 422 | + case EBML_ID_STEREOMODE: |
| 423 | + case EBML_ID_PIXELCROPBOTTOM: |
| 424 | + case EBML_ID_PIXELCROPTOP: |
| 425 | + case EBML_ID_PIXELCROPLEFT: |
| 426 | + case EBML_ID_PIXELCROPRIGHT: |
| 427 | + case EBML_ID_DISPLAYWIDTH: |
| 428 | + case EBML_ID_DISPLAYHEIGHT: |
| 429 | + case EBML_ID_DISPLAYUNIT: |
| 430 | + case EBML_ID_ASPECTRATIOTYPE: |
| 431 | + $track_entry[$sub_subelement_idname] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_subelement_length)); |
| 432 | + break; |
| 433 | + case EBML_ID_FLAGINTERLACED: |
| 434 | + $track_entry[$sub_subelement_idname] = (bool) getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_subelement_length)); |
| 435 | + break; |
| 436 | + case EBML_ID_GAMMAVALUE: |
| 437 | + $track_entry[$sub_subelement_idname] = getid3_lib::BigEndian2Float(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_subelement_length)); |
| 438 | + break; |
| 439 | + case EBML_ID_COLOURSPACE: |
| 440 | + $track_entry[$sub_subelement_idname] = trim(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_subelement_length), "\x00"); |
| 441 | + break; |
| 442 | + default: |
| 443 | + $this->warnings[] = 'Unhandled track.video element ['.basename(__FILE__).':'.__LINE__.'] ('.$sub_subelement_id.'::'.$sub_subelement_idname.') at '.$sub_subelement_offset; |
| 444 | + break; |
| 445 | + } |
| 446 | + $offset = $sub_subelement_end; |
| 447 | + } |
| 448 | + |
| 449 | + if (isset($track_entry[$this->EBMLidName(EBML_ID_CODECID)]) && ($track_entry[$this->EBMLidName(EBML_ID_CODECID)] == 'V_MS/VFW/FOURCC') && isset($track_entry[$this->EBMLidName(EBML_ID_CODECPRIVATE)])) { |
| 450 | + if (getid3_lib::IncludeDependency(GETID3_INCLUDEPATH.'module.audio-video.riff.php', __FILE__, false)) { |
| 451 | + $track_entry['codec_private_parsed'] = getid3_riff::ParseBITMAPINFOHEADER($track_entry[$this->EBMLidName(EBML_ID_CODECPRIVATE)]); |
| 452 | + } else { |
| 453 | + $this->warnings[] = 'Unable to parse codec private data ['.basename(__FILE__).':'.__LINE__.'] because cannot include "module.audio-video.riff.php"'; |
| 454 | + } |
| 455 | + } |
| 456 | + break; |
| 457 | + |
| 458 | + case EBML_ID_AUDIO: |
| 459 | + while ($offset < $subelement_end) { |
| 460 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 461 | + $sub_subelement_offset = $offset; |
| 462 | + $sub_subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 463 | + $sub_subelement_idname = $this->EBMLidName($sub_subelement_id); |
| 464 | + $sub_subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 465 | + $sub_subelement_end = $offset + $sub_subelement_length; |
| 466 | + switch ($sub_subelement_id) { |
| 467 | + case EBML_ID_CHANNELS: |
| 468 | + case EBML_ID_BITDEPTH: |
| 469 | + $track_entry[$sub_subelement_idname] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_subelement_length)); |
| 470 | + break; |
| 471 | + case EBML_ID_SAMPLINGFREQUENCY: |
| 472 | + case EBML_ID_OUTPUTSAMPLINGFREQUENCY: |
| 473 | + $track_entry[$sub_subelement_idname] = getid3_lib::BigEndian2Float(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_subelement_length)); |
| 474 | + break; |
| 475 | + case EBML_ID_CHANNELPOSITIONS: |
| 476 | + $track_entry[$sub_subelement_idname] = trim(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_subelement_length), "\x00"); |
| 477 | + break; |
| 478 | + default: |
| 479 | + $this->warnings[] = 'Unhandled track.video element ['.basename(__FILE__).':'.__LINE__.'] ('.$sub_subelement_id.'::'.$sub_subelement_idname.') at '.$sub_subelement_offset; |
| 480 | + break; |
| 481 | + } |
| 482 | + $offset = $sub_subelement_end; |
| 483 | + } |
| 484 | + break; |
| 485 | + |
| 486 | + case EBML_ID_CONTENTENCODINGS: |
| 487 | + while ($offset < $subelement_end) { |
| 488 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 489 | + $sub_subelement_offset = $offset; |
| 490 | + $sub_subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 491 | + $sub_subelement_idname = $this->EBMLidName($sub_subelement_id); |
| 492 | + $sub_subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 493 | + $sub_subelement_end = $offset + $sub_subelement_length; |
| 494 | + switch ($sub_subelement_id) { |
| 495 | + case EBML_ID_CONTENTENCODING: |
| 496 | + while ($offset < $sub_subelement_end) { |
| 497 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 498 | + $sub_sub_subelement_offset = $offset; |
| 499 | + $sub_sub_subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 500 | + $sub_sub_subelement_idname = $this->EBMLidName($sub_sub_subelement_id); |
| 501 | + $sub_sub_subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 502 | + $sub_sub_subelement_end = $offset + $sub_sub_subelement_length; |
| 503 | + switch ($sub_sub_subelement_id) { |
| 504 | + case EBML_ID_CONTENTENCODINGORDER: |
| 505 | + case EBML_ID_CONTENTENCODINGSCOPE: |
| 506 | + case EBML_ID_CONTENTENCODINGTYPE: |
| 507 | + $track_entry[$sub_subelement_idname][$sub_sub_subelement_idname] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_sub_subelement_length)); |
| 508 | + break; |
| 509 | + case EBML_ID_CONTENTCOMPRESSION: |
| 510 | + while ($offset < $sub_sub_subelement_end) { |
| 511 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 512 | + $sub_sub_sub_subelement_offset = $offset; |
| 513 | + $sub_sub_sub_subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 514 | + $sub_sub_sub_subelement_idname = $this->EBMLidName($sub_sub_subelement_id); |
| 515 | + $sub_sub_sub_subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 516 | + $sub_sub_sub_subelement_end = $offset + $sub_sub_sub_subelement_length; |
| 517 | + switch ($sub_sub_sub_subelement_id) { |
| 518 | + case EBML_ID_CONTENTCOMPALGO: |
| 519 | + $track_entry[$sub_subelement_idname][$sub_sub_subelement_idname][$sub_sub_sub_subelement_idname] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_sub_sub_subelement_length)); |
| 520 | + break; |
| 521 | + case EBML_ID_CONTENTCOMPSETTINGS: |
| 522 | + $track_entry[$sub_subelement_idname][$sub_sub_subelement_idname][$sub_sub_sub_subelement_idname] = substr($EBMLdata, $offset - $EBMLdata_offset, $sub_sub_sub_subelement_length); |
| 523 | + break; |
| 524 | + default: |
| 525 | + $this->warnings[] = 'Unhandled track.contentencodings.contentencoding.contentcompression element ['.basename(__FILE__).':'.__LINE__.'] ('.$subelement_id.'::'.$subelement_idname.' ['.$subelement_length.' bytes]) at '.$subelement_offset; |
| 526 | + break; |
| 527 | + } |
| 528 | + $offset = $sub_sub_sub_subelement_end; |
| 529 | + } |
| 530 | + break; |
| 531 | + |
| 532 | + case EBML_ID_CONTENTENCRYPTION: |
| 533 | + while ($offset < $sub_sub_subelement_end) { |
| 534 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 535 | + $sub_sub_sub_subelement_offset = $offset; |
| 536 | + $sub_sub_sub_subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 537 | + $sub_sub_sub_subelement_idname = $this->EBMLidName($sub_sub_subelement_id); |
| 538 | + $sub_sub_sub_subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 539 | + $sub_sub_sub_subelement_end = $offset + $sub_sub_sub_subelement_length; |
| 540 | + switch ($sub_sub_sub_subelement_id) { |
| 541 | + case EBML_ID_CONTENTENCALGO: |
| 542 | + case EBML_ID_CONTENTSIGALGO: |
| 543 | + case EBML_ID_CONTENTSIGHASHALGO: |
| 544 | + $track_entry[$sub_subelement_idname][$sub_sub_subelement_idname][$sub_sub_sub_subelement_idname] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_sub_sub_subelement_length)); |
| 545 | + break; |
| 546 | + case EBML_ID_CONTENTENCKEYID: |
| 547 | + case EBML_ID_CONTENTSIGNATURE: |
| 548 | + case EBML_ID_CONTENTSIGKEYID: |
| 549 | + $track_entry[$sub_subelement_idname][$sub_sub_subelement_idname][$sub_sub_sub_subelement_idname] = substr($EBMLdata, $offset - $EBMLdata_offset, $sub_sub_sub_subelement_length); |
| 550 | + break; |
| 551 | + default: |
| 552 | + $this->warnings[] = 'Unhandled track.contentencodings.contentencoding.contentcompression element ['.basename(__FILE__).':'.__LINE__.'] ('.$subelement_id.'::'.$subelement_idname.' ['.$subelement_length.' bytes]) at '.$subelement_offset; |
| 553 | + break; |
| 554 | + } |
| 555 | + $offset = $sub_sub_sub_subelement_end; |
| 556 | + } |
| 557 | + break; |
| 558 | + |
| 559 | + default: |
| 560 | + $this->warnings[] = 'Unhandled track.contentencodings.contentencoding element ['.basename(__FILE__).':'.__LINE__.'] ('.$subelement_id.'::'.$subelement_idname.' ['.$subelement_length.' bytes]) at '.$subelement_offset; |
| 561 | + break; |
| 562 | + } |
| 563 | + $offset = $sub_sub_subelement_end; |
| 564 | + } |
| 565 | + break; |
| 566 | + default: |
| 567 | + $this->warnings[] = 'Unhandled track.contentencodings element ['.basename(__FILE__).':'.__LINE__.'] ('.$subelement_id.'::'.$subelement_idname.' ['.$subelement_length.' bytes]) at '.$subelement_offset; |
| 568 | + break; |
| 569 | + } |
| 570 | + $offset = $sub_subelement_end; |
| 571 | + } |
| 572 | + break; |
| 573 | + |
| 574 | + default: |
| 575 | + $this->warnings[] = 'Unhandled track element ['.basename(__FILE__).':'.__LINE__.'] ('.$subelement_id.'::'.$subelement_idname.' ['.$subelement_length.' bytes]) at '.$subelement_offset; |
| 576 | + break; |
| 577 | + } |
| 578 | + $offset = $subelement_end; |
| 579 | + } |
| 580 | + break; |
| 581 | + default: |
| 582 | + $this->warnings[] = 'Unhandled track element ['.basename(__FILE__).':'.__LINE__.'] ('.$track_entry['id'].'::'.$track_entry['id_name'].') at '.$track_entry['offset']; |
| 583 | + $offset = $track_entry_endoffset; |
| 584 | + break; |
| 585 | + } |
| 586 | + $ThisFileInfo['matroska']['tracks']['tracks'][] = $track_entry; |
| 587 | + } |
| 588 | + break; |
| 589 | + |
| 590 | + case EBML_ID_INFO: // Contains the position of other level 1 elements |
| 591 | + $info_entry = array(); |
| 592 | + while ($offset < $element_end) { |
| 593 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 594 | + $subelement_offset = $offset; |
| 595 | + $subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 596 | + $subelement_idname = $this->EBMLidName($subelement_id); |
| 597 | + $subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 598 | + $subelement_end = $offset + $subelement_length; |
| 599 | + switch ($subelement_id) { |
| 600 | + case EBML_ID_CHAPTERTRANSLATEEDITIONUID: |
| 601 | + case EBML_ID_CHAPTERTRANSLATECODEC: |
| 602 | + case EBML_ID_TIMECODESCALE: |
| 603 | + $info_entry[$subelement_idname] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $subelement_length)); |
| 604 | + break; |
| 605 | + case EBML_ID_DURATION: |
| 606 | + $info_entry[$subelement_idname] = getid3_lib::BigEndian2Float(substr($EBMLdata, $offset - $EBMLdata_offset, $subelement_length)); |
| 607 | + break; |
| 608 | + case EBML_ID_DATEUTC: |
| 609 | + $info_entry[$subelement_idname] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $subelement_length)); |
| 610 | + $info_entry[$subelement_idname.'_unix'] = $this->EBMLdate2unix($info_entry[$subelement_idname]); |
| 611 | + break; |
| 612 | + case EBML_ID_SEGMENTUID: |
| 613 | + case EBML_ID_PREVUID: |
| 614 | + case EBML_ID_NEXTUID: |
| 615 | + case EBML_ID_SEGMENTFAMILY: |
| 616 | + case EBML_ID_CHAPTERTRANSLATEID: |
| 617 | + $info_entry[$subelement_idname] = trim(substr($EBMLdata, $offset - $EBMLdata_offset, $subelement_length), "\x00"); |
| 618 | + break; |
| 619 | + case EBML_ID_SEGMENTFILENAME: |
| 620 | + case EBML_ID_PREVFILENAME: |
| 621 | + case EBML_ID_NEXTFILENAME: |
| 622 | + case EBML_ID_TITLE: |
| 623 | + case EBML_ID_MUXINGAPP: |
| 624 | + case EBML_ID_WRITINGAPP: |
| 625 | + $info_entry[$subelement_idname] = trim(substr($EBMLdata, $offset - $EBMLdata_offset, $subelement_length), "\x00"); |
| 626 | + break; |
| 627 | + default: |
| 628 | + $this->warnings[] = 'Unhandled info element ['.basename(__FILE__).':'.__LINE__.'] ('.$subelement_id.'::'.$subelement_idname.' ['.$subelement_length.' bytes]) at '.$subelement_offset; |
| 629 | + break; |
| 630 | + } |
| 631 | + $offset = $subelement_end; |
| 632 | + } |
| 633 | + $ThisFileInfo['matroska']['info'][] = $info_entry; |
| 634 | + break; |
| 635 | + |
| 636 | + case EBML_ID_CUES: |
| 637 | + $cues_entry = array(); |
| 638 | + while ($offset < $element_end) { |
| 639 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 640 | + $subelement_offset = $offset; |
| 641 | + $subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 642 | + $subelement_idname = $this->EBMLidName($subelement_id); |
| 643 | + $subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 644 | + $subelement_end = $offset + $subelement_length; |
| 645 | + switch ($subelement_id) { |
| 646 | + case EBML_ID_CUEPOINT: |
| 647 | + $cuepoint_entry = array(); |
| 648 | + while ($offset < $subelement_end) { |
| 649 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 650 | + $sub_subelement_offset = $offset; |
| 651 | + $sub_subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 652 | + $sub_subelement_idname = $this->EBMLidName($sub_subelement_id); |
| 653 | + $sub_subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 654 | + $sub_subelement_end = $offset + $sub_subelement_length; |
| 655 | + switch ($sub_subelement_id) { |
| 656 | + case EBML_ID_CUETRACKPOSITIONS: |
| 657 | + while ($offset < $sub_subelement_end) { |
| 658 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 659 | + $sub_sub_subelement_offset = $offset; |
| 660 | + $sub_sub_subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 661 | + $sub_sub_subelement_idname = $this->EBMLidName($sub_sub_subelement_id); |
| 662 | + $sub_sub_subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 663 | + $sub_sub_subelement_end = $offset + $sub_sub_subelement_length; |
| 664 | + switch ($sub_sub_subelement_id) { |
| 665 | + case EBML_ID_CUETRACK: |
| 666 | + $cuepoint_entry[$sub_sub_subelement_idname] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_sub_subelement_length)); |
| 667 | + break; |
| 668 | + default: |
| 669 | + $this->warnings[] = 'Unhandled cues.cuepoint.cuetrackpositions element ['.basename(__FILE__).':'.__LINE__.'] ('.$sub_sub_subelement_id.'::'.$sub_sub_subelement_idname.') at '.$sub_sub_subelement_offset; |
| 670 | + break; |
| 671 | + } |
| 672 | + $offset = $sub_subelement_end; |
| 673 | + } |
| 674 | + break; |
| 675 | + case EBML_ID_CUETIME: |
| 676 | + $cuepoint_entry[$subelement_idname] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_subelement_length)); |
| 677 | + break; |
| 678 | + default: |
| 679 | + $this->warnings[] = 'Unhandled cues.cuepoint element ['.basename(__FILE__).':'.__LINE__.'] ('.$sub_subelement_id.'::'.$sub_subelement_idname.') at '.$sub_subelement_offset; |
| 680 | + break; |
| 681 | + } |
| 682 | + $offset = $sub_subelement_end; |
| 683 | + } |
| 684 | + $cues_entry[] = $cuepoint_entry; |
| 685 | + $offset = $sub_subelement_end; |
| 686 | + break; |
| 687 | + default: |
| 688 | + $this->warnings[] = 'Unhandled cues element ['.basename(__FILE__).':'.__LINE__.'] ('.$subelement_id.'::'.$subelement_idname.' ['.$subelement_length.' bytes]) at '.$subelement_offset; |
| 689 | + break; |
| 690 | + } |
| 691 | + $offset = $subelement_end; |
| 692 | + } |
| 693 | + $ThisFileInfo['matroska']['cues'] = $cues_entry; |
| 694 | + break; |
| 695 | + |
| 696 | + case EBML_ID_TAGS: |
| 697 | + $tags_entry = array(); |
| 698 | + while ($offset < $element_end) { |
| 699 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 700 | + $subelement_offset = $offset; |
| 701 | + $subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 702 | + $subelement_idname = $this->EBMLidName($subelement_id); |
| 703 | + $subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 704 | + $subelement_end = $offset + $subelement_length; |
| 705 | + switch ($subelement_id) { |
| 706 | + case EBML_ID_WRITINGAPP: |
| 707 | + $tags_entry[$subelement_idname] = substr($EBMLdata, $offset - $EBMLdata_offset, $subelement_length); |
| 708 | + break; |
| 709 | + case EBML_ID_TAG: |
| 710 | + $tag_entry = array(); |
| 711 | + while ($offset < $subelement_end) { |
| 712 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 713 | + $sub_subelement_offset = $offset; |
| 714 | + $sub_subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 715 | + $sub_subelement_idname = $this->EBMLidName($sub_subelement_id); |
| 716 | + $sub_subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 717 | + $sub_subelement_end = $offset + $sub_subelement_length; |
| 718 | + switch ($sub_subelement_id) { |
| 719 | + case EBML_ID_TARGETS: |
| 720 | + $targets_entry = array(); |
| 721 | + while ($offset < $sub_subelement_end) { |
| 722 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 723 | + $sub_sub_subelement_offset = $offset; |
| 724 | + $sub_sub_subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 725 | + $sub_sub_subelement_idname = $this->EBMLidName($sub_sub_subelement_id); |
| 726 | + $sub_sub_subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 727 | + $sub_sub_subelement_end = $offset + $sub_sub_subelement_length; |
| 728 | + switch ($sub_sub_subelement_id) { |
| 729 | + case EBML_ID_TARGETTYPEVALUE: |
| 730 | + case EBML_ID_EDITIONUID: |
| 731 | + case EBML_ID_CHAPTERUID: |
| 732 | + case EBML_ID_ATTACHMENTUID: |
| 733 | + case EBML_ID_TAGTRACKUID: |
| 734 | + case EBML_ID_TAGCHAPTERUID: |
| 735 | + $targets_entry[$sub_sub_subelement_idname] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_sub_subelement_length)); |
| 736 | + break; |
| 737 | + default: |
| 738 | + $this->warnings[] = 'Unhandled tag.targets element ['.basename(__FILE__).':'.__LINE__.'] ('.$sub_sub_subelement_id.'::'.$sub_sub_subelement_idname.') at '.$sub_sub_subelement_offset; |
| 739 | + break; |
| 740 | + } |
| 741 | + $offset = $sub_sub_subelement_end; |
| 742 | + } |
| 743 | + $tag_entry[$sub_subelement_idname][] = $targets_entry; |
| 744 | + break; |
| 745 | + case EBML_ID_SIMPLETAG: |
| 746 | + $simpletag_entry = array(); |
| 747 | + while ($offset < $sub_subelement_end) { |
| 748 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 749 | + $sub_sub_subelement_offset = $offset; |
| 750 | + $sub_sub_subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 751 | + $sub_sub_subelement_idname = $this->EBMLidName($sub_sub_subelement_id); |
| 752 | + $sub_sub_subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 753 | + $sub_sub_subelement_end = $offset + $sub_sub_subelement_length; |
| 754 | + switch ($sub_sub_subelement_id) { |
| 755 | + case EBML_ID_TAGNAME: |
| 756 | + case EBML_ID_TAGLANGUAGE: |
| 757 | + case EBML_ID_TAGSTRING: |
| 758 | + case EBML_ID_TAGBINARY: |
| 759 | + $simpletag_entry[$sub_sub_subelement_idname] = substr($EBMLdata, $offset - $EBMLdata_offset, $sub_sub_subelement_length); |
| 760 | + break; |
| 761 | + case EBML_ID_TAGDEFAULT: |
| 762 | + $simpletag_entry[$sub_sub_subelement_idname] = (bool) getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_sub_subelement_length)); |
| 763 | + break; |
| 764 | + default: |
| 765 | + $this->warnings[] = 'Unhandled tag.simpletag element ['.basename(__FILE__).':'.__LINE__.'] ('.$sub_sub_subelement_id.'::'.$sub_sub_subelement_idname.') at '.$sub_sub_subelement_offset; |
| 766 | + break; |
| 767 | + } |
| 768 | + $offset = $sub_sub_subelement_end; |
| 769 | + } |
| 770 | + $tag_entry[$sub_subelement_idname][] = $simpletag_entry; |
| 771 | + break; |
| 772 | + case EBML_ID_TARGETTYPE: |
| 773 | + $tag_entry[$sub_subelement_idname] = substr($EBMLdata, $offset - $EBMLdata_offset, $sub_subelement_length); |
| 774 | + break; |
| 775 | + case EBML_ID_TRACKUID: |
| 776 | + $tag_entry[$sub_subelement_idname] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_subelement_length)); |
| 777 | + break; |
| 778 | + default: |
| 779 | + $this->warnings[] = 'Unhandled tags.tag element ['.basename(__FILE__).':'.__LINE__.'] ('.$sub_subelement_id.'::'.$sub_subelement_idname.') at '.$sub_subelement_offset; |
| 780 | + break; |
| 781 | + } |
| 782 | + $offset = $sub_subelement_end; |
| 783 | + } |
| 784 | + $tags_entry['tags'][] = $tag_entry; |
| 785 | + $offset = $sub_subelement_end; |
| 786 | + break; |
| 787 | + default: |
| 788 | + $this->warnings[] = 'Unhandled tags element ['.basename(__FILE__).':'.__LINE__.'] ('.$subelement_id.'::'.$subelement_idname.' ['.$subelement_length.' bytes]) at '.$subelement_offset; |
| 789 | + break; |
| 790 | + } |
| 791 | + $offset = $subelement_end; |
| 792 | + } |
| 793 | + $ThisFileInfo['matroska']['tags'] = $tags_entry; |
| 794 | + break; |
| 795 | + |
| 796 | + |
| 797 | + case EBML_ID_ATTACHMENTS: |
| 798 | + while ($offset < $element_end) { |
| 799 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 800 | + $subelement_offset = $offset; |
| 801 | + $subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 802 | + $subelement_idname = $this->EBMLidName($subelement_id); |
| 803 | + $subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 804 | + $subelement_end = $offset + $subelement_length; |
| 805 | + switch ($subelement_id) { |
| 806 | + case EBML_ID_ATTACHEDFILE: |
| 807 | + $attachedfile_entry = array(); |
| 808 | + while ($offset < $subelement_end) { |
| 809 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 810 | + $sub_subelement_offset = $offset; |
| 811 | + $sub_subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 812 | + $sub_subelement_idname = $this->EBMLidName($sub_subelement_id); |
| 813 | + $sub_subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 814 | + $sub_subelement_end = $offset + $sub_subelement_length; |
| 815 | + switch ($sub_subelement_id) { |
| 816 | + case EBML_ID_FILEDESCRIPTION: |
| 817 | + case EBML_ID_FILENAME: |
| 818 | + case EBML_ID_FILEMIMETYPE: |
| 819 | + $attachedfile_entry[$sub_subelement_idname] = substr($EBMLdata, $offset - $EBMLdata_offset, $sub_subelement_length); |
| 820 | + break; |
| 821 | + |
| 822 | + case EBML_ID_FILEDATA: |
| 823 | + $attachedfile_entry['data_offset'] = $offset; |
| 824 | + $attachedfile_entry['data_length'] = $sub_subelement_length; |
| 825 | + if ($sub_subelement_length < 1024) { |
| 826 | + $attachedfile_entry[$sub_subelement_idname] = substr($EBMLdata, $offset - $EBMLdata_offset, $sub_subelement_length); |
| 827 | + } |
| 828 | + break; |
| 829 | + |
| 830 | + case EBML_ID_FILEUID: |
| 831 | + $attachedfile_entry[$sub_subelement_idname] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_subelement_length)); |
| 832 | + break; |
| 833 | + |
| 834 | + default: |
| 835 | + $this->warnings[] = 'Unhandled attachment.attachedfile element ['.basename(__FILE__).':'.__LINE__.'] ('.$sub_subelement_id.'::'.$sub_subelement_idname.') at '.$sub_subelement_offset; |
| 836 | + break; |
| 837 | + } |
| 838 | + $offset = $sub_subelement_end; |
| 839 | + } |
| 840 | + $ThisFileInfo['matroska']['attachments'][] = $attachedfile_entry; |
| 841 | + $offset = $sub_subelement_end; |
| 842 | + break; |
| 843 | + default: |
| 844 | + $this->warnings[] = 'Unhandled tags element ['.basename(__FILE__).':'.__LINE__.'] ('.$subelement_id.'::'.$subelement_idname.' ['.$subelement_length.' bytes]) at '.$subelement_offset; |
| 845 | + break; |
| 846 | + } |
| 847 | + $offset = $subelement_end; |
| 848 | + } |
| 849 | + break; |
| 850 | + |
| 851 | + |
| 852 | + case EBML_ID_CHAPTERS: // not important to us, contains mostly actual audio/video data, ignore |
| 853 | + while ($offset < $element_end) { |
| 854 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 855 | + $subelement_offset = $offset; |
| 856 | + $subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 857 | + $subelement_idname = $this->EBMLidName($subelement_id); |
| 858 | + $subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 859 | + $subelement_end = $offset + $subelement_length; |
| 860 | + switch ($subelement_id) { |
| 861 | + case EBML_ID_EDITIONENTRY: |
| 862 | + $editionentry_entry = array(); |
| 863 | + while ($offset < $subelement_end) { |
| 864 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 865 | + $sub_subelement_offset = $offset; |
| 866 | + $sub_subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 867 | + $sub_subelement_idname = $this->EBMLidName($sub_subelement_id); |
| 868 | + $sub_subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 869 | + $sub_subelement_end = $offset + $sub_subelement_length; |
| 870 | + switch ($sub_subelement_id) { |
| 871 | + case EBML_ID_EDITIONUID: |
| 872 | + $editionentry_entry[$sub_subelement_idname] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_subelement_length)); |
| 873 | + break; |
| 874 | + case EBML_ID_EDITIONFLAGHIDDEN: |
| 875 | + case EBML_ID_EDITIONFLAGDEFAULT: |
| 876 | + case EBML_ID_EDITIONFLAGORDERED: |
| 877 | + $editionentry_entry[$sub_subelement_idname] = (bool) getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_subelement_length)); |
| 878 | + break; |
| 879 | + case EBML_ID_CHAPTERATOM: |
| 880 | + $chapteratom_entry = array(); |
| 881 | + while ($offset < $sub_subelement_end) { |
| 882 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 883 | + $sub_sub_subelement_offset = $offset; |
| 884 | + $sub_sub_subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 885 | + $sub_sub_subelement_idname = $this->EBMLidName($sub_sub_subelement_id); |
| 886 | + $sub_sub_subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 887 | + $sub_sub_subelement_end = $offset + $sub_sub_subelement_length; |
| 888 | + switch ($sub_sub_subelement_id) { |
| 889 | + case EBML_ID_CHAPTERSEGMENTUID: |
| 890 | + case EBML_ID_CHAPTERSEGMENTEDITIONUID: |
| 891 | + $chapteratom_entry[$sub_sub_subelement_idname] = substr($EBMLdata, $offset - $EBMLdata_offset, $sub_sub_subelement_length); |
| 892 | + break; |
| 893 | + case EBML_ID_CHAPTERFLAGENABLED: |
| 894 | + case EBML_ID_CHAPTERFLAGHIDDEN: |
| 895 | + $chapteratom_entry[$sub_sub_subelement_idname] = (bool) getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_sub_subelement_length)); |
| 896 | + break; |
| 897 | + case EBML_ID_CHAPTERUID: |
| 898 | + case EBML_ID_CHAPTERTIMESTART: |
| 899 | + case EBML_ID_CHAPTERTIMEEND: |
| 900 | + $chapteratom_entry[$sub_sub_subelement_idname] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_sub_subelement_length)); |
| 901 | + break; |
| 902 | + case EBML_ID_CHAPTERTRACK: |
| 903 | + $chaptertrack_entry = array(); |
| 904 | + while ($offset < $sub_sub_subelement_end) { |
| 905 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 906 | + $sub_sub_sub_subelement_offset = $offset; |
| 907 | + $sub_sub_sub_subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 908 | + $sub_sub_sub_subelement_idname = $this->EBMLidName($sub_sub_subelement_id); |
| 909 | + $sub_sub_sub_subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 910 | + $sub_sub_sub_subelement_end = $offset + $sub_sub_sub_subelement_length; |
| 911 | + switch ($sub_sub_sub_subelement_id) { |
| 912 | + case EBML_ID_CHAPTERTRACKNUMBER: |
| 913 | + $chaptertrack_entry[$sub_sub_sub_subelement_idname] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_sub_sub_subelement_length)); |
| 914 | + break; |
| 915 | + default: |
| 916 | + $this->warnings[] = 'Unhandled chapters.editionentry.chapteratom.chaptertrack element ['.basename(__FILE__).':'.__LINE__.'] ('.$sub_sub_sub_subelement_id.'::'.$sub_sub_sub_subelement_idname.') at '.$sub_sub_sub_subelement_offset; |
| 917 | + break; |
| 918 | + } |
| 919 | + $offset = $sub_sub_sub_subelement_end; |
| 920 | + } |
| 921 | + $chapteratom_entry[$sub_sub_subelement_idname][] = $chaptertrack_entry; |
| 922 | + break; |
| 923 | + case EBML_ID_CHAPTERDISPLAY: |
| 924 | + $chapterdisplay_entry = array(); |
| 925 | + while ($offset < $sub_sub_subelement_end) { |
| 926 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 927 | + $sub_sub_sub_subelement_offset = $offset; |
| 928 | + $sub_sub_sub_subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 929 | + $sub_sub_sub_subelement_idname = $this->EBMLidName($sub_sub_sub_subelement_id); |
| 930 | + $sub_sub_sub_subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 931 | + $sub_sub_sub_subelement_end = $offset + $sub_sub_sub_subelement_length; |
| 932 | + switch ($sub_sub_sub_subelement_id) { |
| 933 | + case EBML_ID_CHAPSTRING: |
| 934 | + case EBML_ID_CHAPLANGUAGE: |
| 935 | + case EBML_ID_CHAPCOUNTRY: |
| 936 | + $chapterdisplay_entry[$sub_sub_sub_subelement_idname] = substr($EBMLdata, $offset - $EBMLdata_offset, $sub_sub_sub_subelement_length); |
| 937 | + break; |
| 938 | + default: |
| 939 | + $this->warnings[] = 'Unhandled chapters.editionentry.chapteratom.chapterdisplay element ['.basename(__FILE__).':'.__LINE__.'] ('.$sub_sub_sub_subelement_id.'::'.$sub_sub_sub_subelement_idname.') at '.$sub_sub_sub_subelement_offset; |
| 940 | + break; |
| 941 | + } |
| 942 | + $offset = $sub_sub_sub_subelement_end; |
| 943 | + } |
| 944 | + $chapteratom_entry[$sub_sub_subelement_idname][] = $chapterdisplay_entry; |
| 945 | + break; |
| 946 | + default: |
| 947 | + $this->warnings[] = 'Unhandled chapters.editionentry.chapteratom element ['.basename(__FILE__).':'.__LINE__.'] ('.$sub_sub_subelement_id.'::'.$sub_sub_subelement_idname.') at '.$sub_sub_subelement_offset; |
| 948 | + break; |
| 949 | + } |
| 950 | + $offset = $sub_sub_subelement_end; |
| 951 | + } |
| 952 | + $editionentry_entry[$sub_subelement_idname][] = $chapteratom_entry; |
| 953 | + break; |
| 954 | + default: |
| 955 | + $this->warnings[] = 'Unhandled chapters.editionentry element ['.basename(__FILE__).':'.__LINE__.'] ('.$sub_subelement_id.'::'.$sub_subelement_idname.') at '.$sub_subelement_offset; |
| 956 | + break; |
| 957 | + } |
| 958 | + $offset = $sub_subelement_end; |
| 959 | + } |
| 960 | + $ThisFileInfo['matroska']['chapters'][] = $editionentry_entry; |
| 961 | + $offset = $sub_subelement_end; |
| 962 | + break; |
| 963 | + default: |
| 964 | + $this->warnings[] = 'Unhandled chapters element ['.basename(__FILE__).':'.__LINE__.'] ('.$subelement_id.'::'.$subelement_idname.' ['.$subelement_length.' bytes]) at '.$subelement_offset; |
| 965 | + break; |
| 966 | + } |
| 967 | + $offset = $subelement_end; |
| 968 | + } |
| 969 | + break; |
| 970 | + |
| 971 | + |
| 972 | + case EBML_ID_VOID: // padding, ignore |
| 973 | + $void_entry = array(); |
| 974 | + $void_entry['offset'] = $offset; |
| 975 | + $ThisFileInfo['matroska']['void'][] = $void_entry; |
| 976 | + break; |
| 977 | + |
| 978 | + case EBML_ID_CLUSTER: // not important to us, contains mostly actual audio/video data, ignore |
| 979 | + $cluster_entry = array(); |
| 980 | + while ($offset < $element_end) { |
| 981 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 982 | + $subelement_offset = $offset; |
| 983 | + $subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 984 | + $subelement_idname = $this->EBMLidName($subelement_id); |
| 985 | + $subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 986 | + $subelement_end = $offset + $subelement_length; |
| 987 | + switch ($subelement_id) { |
| 988 | + case EBML_ID_CLUSTERTIMECODE: |
| 989 | + case EBML_ID_CLUSTERPOSITION: |
| 990 | + case EBML_ID_CLUSTERPREVSIZE: |
| 991 | + $cluster_entry[$subelement_idname] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $subelement_length)); |
| 992 | + break; |
| 993 | + |
| 994 | + case EBML_ID_CLUSTERSILENTTRACKS: |
| 995 | + $cluster_silent_tracks = array(); |
| 996 | + while ($offset < $subelement_end) { |
| 997 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 998 | + $sub_subelement_offset = $offset; |
| 999 | + $sub_subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 1000 | + $sub_subelement_idname = $this->EBMLidName($sub_subelement_id); |
| 1001 | + $sub_subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 1002 | + $sub_subelement_end = $offset + $sub_subelement_length; |
| 1003 | + switch ($sub_subelement_id) { |
| 1004 | + case EBML_ID_CLUSTERSILENTTRACKNUMBER: |
| 1005 | + $cluster_silent_tracks[] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_subelement_length)); |
| 1006 | + break; |
| 1007 | + default: |
| 1008 | + $this->warnings[] = 'Unhandled clusters.silenttracks element ['.basename(__FILE__).':'.__LINE__.'] ('.$sub_subelement_id.'::'.$sub_subelement_idname.') at '.$sub_subelement_offset; |
| 1009 | + break; |
| 1010 | + } |
| 1011 | + $offset = $sub_subelement_end; |
| 1012 | + } |
| 1013 | + $cluster_entry[$subelement_idname][] = $cluster_silent_tracks; |
| 1014 | + $offset = $sub_subelement_end; |
| 1015 | + break; |
| 1016 | + |
| 1017 | + case EBML_ID_CLUSTERBLOCKGROUP: |
| 1018 | + $cluster_block_group = array('offset'=>$offset); |
| 1019 | + while ($offset < $subelement_end) { |
| 1020 | + $this->EnsureBufferHasEnoughData($fd, $EBMLdata, $offset, $EBMLdata_offset); |
| 1021 | + $sub_subelement_offset = $offset; |
| 1022 | + $sub_subelement_id = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 1023 | + $sub_subelement_idname = $this->EBMLidName($sub_subelement_id); |
| 1024 | + $sub_subelement_length = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 1025 | + $sub_subelement_end = $offset + $sub_subelement_length; |
| 1026 | + switch ($sub_subelement_id) { |
| 1027 | + case EBML_ID_CLUSTERBLOCK: |
| 1028 | + $cluster_block_data = array(); |
| 1029 | + $cluster_block_data['tracknumber'] = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 1030 | + $cluster_block_data['timecode'] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset, 2)); |
| 1031 | + $offset += 2; |
| 1032 | + // unsure whether this is 1 octect or 2 octets? (http://matroska.org/technical/specs/index.html#block_structure) |
| 1033 | + $cluster_block_data['flags_raw'] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset, 1)); |
| 1034 | + $offset += 1; |
| 1035 | + //$cluster_block_data['flags']['reserved1'] = (($cluster_block_data['flags_raw'] & 0xF0) >> 4); |
| 1036 | + $cluster_block_data['flags']['invisible'] = (bool) (($cluster_block_data['flags_raw'] & 0x08) >> 3); |
| 1037 | + $cluster_block_data['flags']['lacing'] = (($cluster_block_data['flags_raw'] & 0x06) >> 1); |
| 1038 | + //$cluster_block_data['flags']['reserved2'] = (($cluster_block_data['flags_raw'] & 0x01) >> 0); |
| 1039 | + $cluster_block_data['flags']['lacing_type'] = $this->MatroskaBlockLacingType($cluster_block_data['flags']['lacing']); |
| 1040 | + if ($cluster_block_data['flags']['lacing'] != 0) { |
| 1041 | + $cluster_block_data['lace_frames'] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset, 1)); // Number of frames in the lace-1 (uint8) |
| 1042 | + $offset += 1; |
| 1043 | + if ($cluster_block_data['flags']['lacing'] != 2) { |
| 1044 | + $cluster_block_data['lace_frames'] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset, 1)); // Lace-coded size of each frame of the lace, except for the last one (multiple uint8). *This is not used with Fixed-size lacing as it is calculated automatically from (total size of lace) / (number of frames in lace). |
| 1045 | + $offset += 1; |
| 1046 | + } |
| 1047 | + } |
| 1048 | + if (!isset($ThisFileInfo['matroska']['track_data_offsets'][$cluster_block_data['tracknumber']])) { |
| 1049 | + $ThisFileInfo['matroska']['track_data_offsets'][$cluster_block_data['tracknumber']] = $offset; |
| 1050 | + } |
| 1051 | + $cluster_block_group[$sub_subelement_idname] = $cluster_block_data; |
| 1052 | + break; |
| 1053 | + |
| 1054 | + case EBML_ID_CLUSTERREFERENCEPRIORITY: // unsigned-int |
| 1055 | + case EBML_ID_CLUSTERBLOCKDURATION: // unsigned-int |
| 1056 | + $cluster_block_group[$sub_subelement_idname] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_subelement_length)); |
| 1057 | + break; |
| 1058 | + |
| 1059 | + case EBML_ID_CLUSTERREFERENCEBLOCK: // signed-int |
| 1060 | + $cluster_block_group[$sub_subelement_idname] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset - $EBMLdata_offset, $sub_subelement_length), false, true); |
| 1061 | + break; |
| 1062 | + |
| 1063 | + default: |
| 1064 | + $this->warnings[] = 'Unhandled clusters.blockgroup element ['.basename(__FILE__).':'.__LINE__.'] ('.$sub_subelement_id.'::'.$sub_subelement_idname.') at '.$sub_subelement_offset; |
| 1065 | + break; |
| 1066 | + } |
| 1067 | + $offset = $sub_subelement_end; |
| 1068 | + } |
| 1069 | + $cluster_entry[$subelement_idname][] = $cluster_block_group; |
| 1070 | + $offset = $sub_subelement_end; |
| 1071 | + break; |
| 1072 | + |
| 1073 | + case EBML_ID_CLUSTERSIMPLEBLOCK: |
| 1074 | + // http://www.matroska.org/technical/specs/index.html#simpleblock_structure |
| 1075 | + $cluster_block_data = array(); |
| 1076 | + $cluster_block_data['tracknumber'] = $this->readEBMLint($EBMLdata, $offset, $EBMLdata_offset); |
| 1077 | + $cluster_block_data['timecode'] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset, 2)); |
| 1078 | + $offset += 2; |
| 1079 | + $cluster_block_data['flags_raw'] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset, 1)); |
| 1080 | + $offset += 1; |
| 1081 | + $cluster_block_data['flags']['keyframe'] = (($cluster_block_data['flags_raw'] & 0x80) >> 7); |
| 1082 | + $cluster_block_data['flags']['reserved1'] = (($cluster_block_data['flags_raw'] & 0x70) >> 4); |
| 1083 | + $cluster_block_data['flags']['invisible'] = (($cluster_block_data['flags_raw'] & 0x08) >> 3); |
| 1084 | + $cluster_block_data['flags']['lacing'] = (($cluster_block_data['flags_raw'] & 0x06) >> 1); // 00=no lacing; 01=Xiph lacing; 11=EBML lacing; 10=fixed-size lacing |
| 1085 | + $cluster_block_data['flags']['discardable'] = (($cluster_block_data['flags_raw'] & 0x01)); |
| 1086 | + |
| 1087 | + if ($cluster_block_data['flags']['lacing'] > 0) { |
| 1088 | + $cluster_block_data['lace_frames'] = 1 + getid3_lib::BigEndian2Int(substr($EBMLdata, $offset, 1)); |
| 1089 | + $offset += 1; |
| 1090 | + if ($cluster_block_data['flags']['lacing'] != 0x02) { |
| 1091 | + // *This is not used with Fixed-size lacing as it is calculated automatically from (total size of lace) / (number of frames in lace). |
| 1092 | + $cluster_block_data['lace_frame_size'] = getid3_lib::BigEndian2Int(substr($EBMLdata, $offset, 1)); |
| 1093 | + $offset += 1; |
| 1094 | + } |
| 1095 | + } |
| 1096 | + |
| 1097 | + if (!isset($ThisFileInfo['matroska']['track_data_offsets'][$cluster_block_data['tracknumber']])) { |
| 1098 | + $ThisFileInfo['matroska']['track_data_offsets'][$cluster_block_data['tracknumber']] = $offset; |
| 1099 | + } |
| 1100 | + $cluster_block_group[$sub_subelement_idname] = $cluster_block_data; |
| 1101 | + break; |
| 1102 | + |
| 1103 | + default: |
| 1104 | + $this->warnings[] = 'Unhandled cluster element ['.basename(__FILE__).':'.__LINE__.'] ('.$subelement_id.'::'.$subelement_idname.' ['.$subelement_length.' bytes]) at '.$subelement_offset; |
| 1105 | + break; |
| 1106 | + } |
| 1107 | + $offset = $subelement_end; |
| 1108 | + } |
| 1109 | + $ThisFileInfo['matroska']['cluster'][] = $cluster_entry; |
| 1110 | + |
| 1111 | + // check to see if all the data we need exists already, if so, break out of the loop |
| 1112 | + if (isset($ThisFileInfo['matroska']['info']) && is_array($ThisFileInfo['matroska']['info'])) { |
| 1113 | + if (isset($ThisFileInfo['matroska']['tracks']['tracks']) && is_array($ThisFileInfo['matroska']['tracks']['tracks'])) { |
| 1114 | + break 2; |
| 1115 | + } |
| 1116 | + } |
| 1117 | + break; |
| 1118 | + |
| 1119 | + default: |
| 1120 | + if ($element_data['id_name'] == dechex($element_data['id'])) { |
| 1121 | + $ThisFileInfo['error'][] = 'Unhandled segment ['.basename(__FILE__).':'.__LINE__.'] ('.$element_data['id'].') at '.$element_data_offset; |
| 1122 | + } else { |
| 1123 | + $this->warnings[] = 'Unhandled segment ['.basename(__FILE__).':'.__LINE__.'] ('.$element_data['id'].'::'.$element_data['id_name'].') at '.$element_data['offset']; |
| 1124 | + } |
| 1125 | + break; |
| 1126 | + } |
| 1127 | + $offset = $element_end; |
| 1128 | + } |
| 1129 | + break; |
| 1130 | + |
| 1131 | + |
| 1132 | + default: |
| 1133 | + $ThisFileInfo['error'][] = 'Unhandled chunk ['.basename(__FILE__).':'.__LINE__.'] ('.$top_element_id.') at '.$offset; |
| 1134 | + break; |
| 1135 | + } |
| 1136 | + $offset = $top_element_endoffset; |
| 1137 | + } |
| 1138 | + |
| 1139 | + |
| 1140 | + |
| 1141 | + if (isset($ThisFileInfo['matroska']['info']) && is_array($ThisFileInfo['matroska']['info'])) { |
| 1142 | + foreach ($ThisFileInfo['matroska']['info'] as $key => $infoarray) { |
| 1143 | + if (isset($infoarray['Duration'])) { |
| 1144 | + // TimecodeScale is how many nanoseconds each Duration unit is |
| 1145 | + $ThisFileInfo['playtime_seconds'] = $infoarray['Duration'] * ((isset($infoarray['TimecodeScale']) ? $infoarray['TimecodeScale'] : 1000000) / 1000000000); |
| 1146 | + break; |
| 1147 | + } |
| 1148 | + } |
| 1149 | + } |
| 1150 | + if (isset($ThisFileInfo['matroska']['tracks']['tracks']) && is_array($ThisFileInfo['matroska']['tracks']['tracks'])) { |
| 1151 | + foreach ($ThisFileInfo['matroska']['tracks']['tracks'] as $key => $trackarray) { |
| 1152 | + $track_info = array(); |
| 1153 | + switch (isset($trackarray['TrackType']) ? $trackarray['TrackType'] : '') { |
| 1154 | + case 1: // Video |
| 1155 | + if (!empty($trackarray['PixelWidth'])) { $track_info['resolution_x'] = $trackarray['PixelWidth']; } |
| 1156 | + if (!empty($trackarray['PixelHeight'])) { $track_info['resolution_y'] = $trackarray['PixelHeight']; } |
| 1157 | + if (!empty($trackarray['DisplayWidth'])) { $track_info['display_x'] = $trackarray['DisplayWidth']; } |
| 1158 | + if (!empty($trackarray['DisplayHeight'])) { $track_info['display_y'] = $trackarray['DisplayHeight']; } |
| 1159 | + if (!empty($trackarray['DefaultDuration'])) { $track_info['frame_rate'] = round(1000000000 / $trackarray['DefaultDuration'], 3); } |
| 1160 | + if (!empty($trackarray['CodecID'])) { $track_info['dataformat'] = $this->MatroskaCodecIDtoCommonName($trackarray['CodecID']); } |
| 1161 | + if (!empty($trackarray['codec_private_parsed']['fourcc'])) { $track_info['fourcc'] = $trackarray['codec_private_parsed']['fourcc']; } |
| 1162 | + $ThisFileInfo['video']['streams'][] = $track_info; |
| 1163 | + if (isset($track_info['resolution_x']) && empty($ThisFileInfo['video']['resolution_x'])) { |
| 1164 | + foreach ($track_info as $key => $value) { |
| 1165 | + $ThisFileInfo['video'][$key] = $value; |
| 1166 | + } |
| 1167 | + } |
| 1168 | + break; |
| 1169 | + case 2: // Audio |
| 1170 | + if (!empty($trackarray['CodecID'])) { $track_info['dataformat'] = $this->MatroskaCodecIDtoCommonName($trackarray['CodecID']); } |
| 1171 | + if (!empty($trackarray['SamplingFrequency'])) { $track_info['sample_rate'] = $trackarray['SamplingFrequency']; } |
| 1172 | + if (!empty($trackarray['Channels'])) { $track_info['channels'] = $trackarray['Channels']; } |
| 1173 | + if (!empty($trackarray['BitDepth'])) { $track_info['bits_per_sample'] = $trackarray['BitDepth']; } |
| 1174 | + switch (isset($trackarray[$this->EBMLidName(EBML_ID_CODECID)]) ? $trackarray[$this->EBMLidName(EBML_ID_CODECID)] : '') { |
| 1175 | + case 'A_PCM/INT/LIT': |
| 1176 | + case 'A_PCM/INT/BIG': |
| 1177 | + $track_info['bitrate'] = $trackarray['SamplingFrequency'] * $trackarray['Channels'] * $trackarray['BitDepth']; |
| 1178 | + break; |
| 1179 | + |
| 1180 | + case 'A_AC3': |
| 1181 | + if (getid3_lib::IncludeDependency(GETID3_INCLUDEPATH.'module.audio.ac3.php', __FILE__, false)) { |
| 1182 | + if (isset($ThisFileInfo['matroska']['track_data_offsets'][$trackarray['TrackNumber']])) { |
| 1183 | + $ac3_thisfileinfo = array('avdataoffset'=>$ThisFileInfo['matroska']['track_data_offsets'][$trackarray['TrackNumber']]); |
| 1184 | + $getid3_ac3 = new getid3_ac3($fd, $ac3_thisfileinfo); |
| 1185 | + $ThisFileInfo['matroska']['track_codec_parsed'][$trackarray['TrackNumber']] = $ac3_thisfileinfo; |
| 1186 | + if (!empty($ac3_thisfileinfo['error'])) { |
| 1187 | + foreach ($ac3_thisfileinfo['error'] as $newerror) { |
| 1188 | + $this->warnings[] = 'getid3_ac3() says: ['.$newerror.']'; |
| 1189 | + } |
| 1190 | + } |
| 1191 | + if (!empty($ac3_thisfileinfo['warning'])) { |
| 1192 | + foreach ($ac3_thisfileinfo['warning'] as $newerror) { |
| 1193 | + $this->warnings[] = 'getid3_ac3() says: ['.$newerror.']'; |
| 1194 | + } |
| 1195 | + } |
| 1196 | + if (isset($ac3_thisfileinfo['audio']) && is_array($ac3_thisfileinfo['audio'])) { |
| 1197 | + foreach ($ac3_thisfileinfo['audio'] as $key => $value) { |
| 1198 | + $track_info[$key] = $value; |
| 1199 | + } |
| 1200 | + } |
| 1201 | + unset($ac3_thisfileinfo); |
| 1202 | + unset($getid3_ac3); |
| 1203 | + } else { |
| 1204 | + $this->warnings[] = 'Unable to parse audio data ['.basename(__FILE__).':'.__LINE__.'] because $ThisFileInfo[matroska][track_data_offsets]['.$trackarray['TrackNumber'].'] not set'; |
| 1205 | + } |
| 1206 | + } else { |
| 1207 | + $this->warnings[] = 'Unable to parse audio data ['.basename(__FILE__).':'.__LINE__.'] because cannot include "module.audio.ac3.php"'; |
| 1208 | + } |
| 1209 | + break; |
| 1210 | + |
| 1211 | + case 'A_DTS': |
| 1212 | + $dts_offset = $ThisFileInfo['matroska']['track_data_offsets'][$trackarray['TrackNumber']]; |
| 1213 | + // this is a NASTY hack, but sometimes audio data is off by a byte or two and not sure why, email info@getid3.org if you can explain better |
| 1214 | + fseek($fd, $dts_offset, SEEK_SET); |
| 1215 | + $magic_test = fread($fd, 8); |
| 1216 | + for ($i = 0; $i < 4; $i++) { |
| 1217 | + // look to see if DTS "magic" is here, if so adjust offset by that many bytes |
| 1218 | + if (substr($magic_test, $i, 4) == "\x7F\xFE\x80\x01") { |
| 1219 | + $dts_offset += $i; |
| 1220 | + break; |
| 1221 | + } |
| 1222 | + } |
| 1223 | + if (getid3_lib::IncludeDependency(GETID3_INCLUDEPATH.'module.audio.dts.php', __FILE__, false)) { |
| 1224 | + $dts_thisfileinfo = array('avdataoffset'=>$dts_offset); |
| 1225 | + $getid3_dts = new getid3_dts($fd, $dts_thisfileinfo); |
| 1226 | + $ThisFileInfo['matroska']['track_codec_parsed'][$trackarray['TrackNumber']] = $dts_thisfileinfo; |
| 1227 | + if (!empty($dts_thisfileinfo['error'])) { |
| 1228 | + foreach ($dts_thisfileinfo['error'] as $newerror) { |
| 1229 | + $this->warnings[] = 'getid3_dts() says: ['.$newerror.']'; |
| 1230 | + } |
| 1231 | + } |
| 1232 | + if (!empty($dts_thisfileinfo['warning'])) { |
| 1233 | + foreach ($dts_thisfileinfo['warning'] as $newerror) { |
| 1234 | + $this->warnings[] = 'getid3_dts() says: ['.$newerror.']'; |
| 1235 | + } |
| 1236 | + } |
| 1237 | + if (isset($dts_thisfileinfo['audio']) && is_array($dts_thisfileinfo['audio'])) { |
| 1238 | + foreach ($dts_thisfileinfo['audio'] as $key => $value) { |
| 1239 | + $track_info[$key] = $value; |
| 1240 | + } |
| 1241 | + } |
| 1242 | + unset($dts_thisfileinfo); |
| 1243 | + unset($getid3_dts); |
| 1244 | + } else { |
| 1245 | + $this->warnings[] = 'Unable to parse audio data ['.basename(__FILE__).':'.__LINE__.'] because cannot include "module.audio.dts.php"'; |
| 1246 | + } |
| 1247 | + break; |
| 1248 | + |
| 1249 | + case 'A_AAC': |
| 1250 | +$this->warnings[] = 'This version of getID3() [v'.GETID3_VERSION.'] has problems parsing AAC audio in Matroska containers ['.basename(__FILE__).':'.__LINE__.']'; |
| 1251 | + if (getid3_lib::IncludeDependency(GETID3_INCLUDEPATH.'module.audio.aac.php', __FILE__, false)) { |
| 1252 | + // $aac_thisfileinfo = array('avdataoffset'=>$ThisFileInfo['matroska']['track_data_offsets'][$trackarray['TrackNumber']]); |
| 1253 | + // $getid3_aac = new getid3_aac($fd, $aac_thisfileinfo); |
| 1254 | + // $ThisFileInfo['matroska']['track_codec_parsed'][$trackarray['TrackNumber']] = $aac_thisfileinfo; |
| 1255 | + // if (isset($aac_thisfileinfo['audio']) && is_array($aac_thisfileinfo['audio'])) { |
| 1256 | + // foreach ($aac_thisfileinfo['audio'] as $key => $value) { |
| 1257 | + // $track_info[$key] = $value; |
| 1258 | + // } |
| 1259 | + // } |
| 1260 | + // unset($aac_thisfileinfo); |
| 1261 | + // unset($getid3_aac); |
| 1262 | + } else { |
| 1263 | + $this->warnings[] = 'Unable to parse audio data ['.basename(__FILE__).':'.__LINE__.'] because cannot include "module.audio.aac.php"'; |
| 1264 | + } |
| 1265 | + break; |
| 1266 | + |
| 1267 | + case 'A_MPEG/L3': |
| 1268 | + if (getid3_lib::IncludeDependency(GETID3_INCLUDEPATH.'module.audio.mp3.php', __FILE__, false)) { |
| 1269 | + $mp3_thisfileinfo = array( |
| 1270 | + 'avdataoffset' => $ThisFileInfo['matroska']['track_data_offsets'][$trackarray['TrackNumber']], |
| 1271 | + 'avdataend' => $ThisFileInfo['matroska']['track_data_offsets'][$trackarray['TrackNumber']] + 1024, |
| 1272 | + ); |
| 1273 | + $getid3_mp3 = new getid3_mp3($fd, $mp3_thisfileinfo); |
| 1274 | + $getid3_mp3->allow_bruteforce = true; |
| 1275 | + //getid3_mp3::getOnlyMPEGaudioInfo($fd, $mp3_thisfileinfo, $offset, false); |
| 1276 | + $ThisFileInfo['matroska']['track_codec_parsed'][$trackarray['TrackNumber']] = $mp3_thisfileinfo; |
| 1277 | + if (!empty($mp3_thisfileinfo['error'])) { |
| 1278 | + foreach ($mp3_thisfileinfo['error'] as $newerror) { |
| 1279 | + $this->warnings[] = 'getid3_mp3() says: ['.$newerror.']'; |
| 1280 | + } |
| 1281 | + } |
| 1282 | + if (!empty($mp3_thisfileinfo['warning'])) { |
| 1283 | + foreach ($mp3_thisfileinfo['warning'] as $newerror) { |
| 1284 | + $this->warnings[] = 'getid3_mp3() says: ['.$newerror.']'; |
| 1285 | + } |
| 1286 | + } |
| 1287 | + if (isset($mp3_thisfileinfo['audio']) && is_array($mp3_thisfileinfo['audio'])) { |
| 1288 | + foreach ($mp3_thisfileinfo['audio'] as $key => $value) { |
| 1289 | + $track_info[$key] = $value; |
| 1290 | + } |
| 1291 | + } |
| 1292 | + unset($mp3_thisfileinfo); |
| 1293 | + unset($getid3_mp3); |
| 1294 | + } else { |
| 1295 | + $this->warnings[] = 'Unable to parse audio data ['.basename(__FILE__).':'.__LINE__.'] because cannot include "module.audio.mp3.php"'; |
| 1296 | + } |
| 1297 | + break; |
| 1298 | + |
| 1299 | + case 'A_VORBIS': |
| 1300 | + if (isset($trackarray['CodecPrivate'])) { |
| 1301 | + // this is a NASTY hack, email info@getid3.org if you have a better idea how to get this info out |
| 1302 | + $found_vorbis = false; |
| 1303 | + for ($vorbis_offset = 1; $vorbis_offset < 16; $vorbis_offset++) { |
| 1304 | + if (substr($trackarray['CodecPrivate'], $vorbis_offset, 6) == 'vorbis') { |
| 1305 | + $vorbis_offset--; |
| 1306 | + $found_vorbis = true; |
| 1307 | + break; |
| 1308 | + } |
| 1309 | + } |
| 1310 | + if ($found_vorbis) { |
| 1311 | + if (getid3_lib::IncludeDependency(GETID3_INCLUDEPATH.'module.audio.ogg.php', __FILE__, false)) { |
| 1312 | + $vorbis_fileinfo = array(); |
| 1313 | + $oggpageinfo['page_seqno'] = 0; |
| 1314 | + getid3_ogg::ParseVorbisPageHeader($trackarray['CodecPrivate'], $vorbis_offset, $vorbis_fileinfo, $oggpageinfo); |
| 1315 | + $ThisFileInfo['matroska']['track_codec_parsed'][$trackarray['TrackNumber']] = $vorbis_fileinfo; |
| 1316 | + if (!empty($vorbis_fileinfo['error'])) { |
| 1317 | + foreach ($vorbis_fileinfo['error'] as $newerror) { |
| 1318 | + $this->warnings[] = 'getid3_ogg() says: ['.$newerror.']'; |
| 1319 | + } |
| 1320 | + } |
| 1321 | + if (!empty($vorbis_fileinfo['warning'])) { |
| 1322 | + foreach ($vorbis_fileinfo['warning'] as $newerror) { |
| 1323 | + $this->warnings[] = 'getid3_ogg() says: ['.$newerror.']'; |
| 1324 | + } |
| 1325 | + } |
| 1326 | + if (isset($vorbis_fileinfo['audio']) && is_array($vorbis_fileinfo['audio'])) { |
| 1327 | + foreach ($vorbis_fileinfo['audio'] as $key => $value) { |
| 1328 | + $track_info[$key] = $value; |
| 1329 | + } |
| 1330 | + } |
| 1331 | + if (!empty($vorbis_fileinfo['ogg']['bitrate_average'])) { |
| 1332 | + $track_info['bitrate'] = $vorbis_fileinfo['ogg']['bitrate_average']; |
| 1333 | + } elseif (!empty($vorbis_fileinfo['ogg']['bitrate_nominal'])) { |
| 1334 | + $track_info['bitrate'] = $vorbis_fileinfo['ogg']['bitrate_nominal']; |
| 1335 | + } |
| 1336 | + unset($vorbis_fileinfo); |
| 1337 | + unset($oggpageinfo); |
| 1338 | + } else { |
| 1339 | + $this->warnings[] = 'Unable to parse audio data ['.basename(__FILE__).':'.__LINE__.'] because cannot include "module.audio.ogg.php"'; |
| 1340 | + } |
| 1341 | + } else { |
| 1342 | + } |
| 1343 | + } else { |
| 1344 | + } |
| 1345 | + break; |
| 1346 | + |
| 1347 | + default: |
| 1348 | + $this->warnings[] = 'Unhandled audio type "'.(isset($trackarray[$this->EBMLidName(EBML_ID_CODECID)]) ? $trackarray[$this->EBMLidName(EBML_ID_CODECID)] : '').'"'; |
| 1349 | + break; |
| 1350 | + } |
| 1351 | + |
| 1352 | + |
| 1353 | + $ThisFileInfo['audio']['streams'][] = $track_info; |
| 1354 | + if (isset($track_info['dataformat']) && empty($ThisFileInfo['audio']['dataformat'])) { |
| 1355 | + foreach ($track_info as $key => $value) { |
| 1356 | + $ThisFileInfo['audio'][$key] = $value; |
| 1357 | + } |
| 1358 | + } |
| 1359 | + break; |
| 1360 | + default: |
| 1361 | + // ignore, do nothing |
| 1362 | + break; |
| 1363 | + } |
| 1364 | + } |
| 1365 | + } |
| 1366 | + |
| 1367 | + if ($this->hide_clusters) { |
| 1368 | + // too much data returned that is usually not useful |
| 1369 | + if (isset($ThisFileInfo['matroska']['segments']) && is_array($ThisFileInfo['matroska']['segments'])) { |
| 1370 | + foreach ($ThisFileInfo['matroska']['segments'] as $key => $segmentsarray) { |
| 1371 | + if ($segmentsarray['id'] == EBML_ID_CLUSTER) { |
| 1372 | + unset($ThisFileInfo['matroska']['segments'][$key]); |
| 1373 | + } |
| 1374 | + } |
| 1375 | + } |
| 1376 | + if (isset($ThisFileInfo['matroska']['seek']) && is_array($ThisFileInfo['matroska']['seek'])) { |
| 1377 | + foreach ($ThisFileInfo['matroska']['seek'] as $key => $seekarray) { |
| 1378 | + if ($seekarray['target_id'] == EBML_ID_CLUSTER) { |
| 1379 | + unset($ThisFileInfo['matroska']['seek'][$key]); |
| 1380 | + } |
| 1381 | + } |
| 1382 | + } |
| 1383 | + //unset($ThisFileInfo['matroska']['cluster']); |
| 1384 | + //unset($ThisFileInfo['matroska']['track_data_offsets']); |
| 1385 | + } |
| 1386 | + |
| 1387 | + if (!empty($ThisFileInfo['video']['streams'])) { |
| 1388 | + $ThisFileInfo['mime_type'] = 'video/x-matroska'; |
| 1389 | + } elseif (!empty($ThisFileInfo['video']['streams'])) { |
| 1390 | + $ThisFileInfo['mime_type'] = 'audio/x-matroska'; |
| 1391 | + } elseif (isset($ThisFileInfo['mime_type'])) { |
| 1392 | + unset($ThisFileInfo['mime_type']); |
| 1393 | + } |
| 1394 | + |
| 1395 | + foreach ($this->warnings as $key => $value) { |
| 1396 | + $ThisFileInfo['warning'][] = $value; |
| 1397 | + } |
| 1398 | + |
| 1399 | + return true; |
| 1400 | + } |
| 1401 | + |
| 1402 | + |
| 1403 | +/////////////////////////////////////// |
| 1404 | + |
| 1405 | + |
| 1406 | + function EnsureBufferHasEnoughData(&$fd, &$EBMLdata, &$offset, &$EBMLdata_offset) { |
| 1407 | + $min_data = 1024; |
| 1408 | + if ($offset > 2147450880) { // 2^31 - 2^15 (2G-32k) |
| 1409 | + $offset = pow(2,63); |
| 1410 | + return false; |
| 1411 | + } elseif (($offset - $EBMLdata_offset) >= (strlen($EBMLdata) - $min_data)) { |
| 1412 | + fseek($fd, $offset, SEEK_SET); |
| 1413 | + $EBMLdata_offset = ftell($fd); |
| 1414 | + $EBMLdata = fread($fd, $this->read_buffer_size); |
| 1415 | + } |
| 1416 | + return true; |
| 1417 | + } |
| 1418 | + |
| 1419 | + function readEBMLint(&$string, &$offset, $dataoffset=0) { |
| 1420 | + $actual_offset = $offset - $dataoffset; |
| 1421 | + if ($offset > 2147450880) { // 2^31 - 2^15 (2G-32k) |
| 1422 | + $this->warnings[] = 'aborting readEBMLint() because $offset larger than 2GB'; |
| 1423 | + return false; |
| 1424 | + } elseif ($actual_offset >= strlen($string)) { |
| 1425 | + $this->warnings[] = '$actual_offset > $string in readEBMLint($string['.strlen($string).'], '.$offset.', '.$dataoffset.')'; |
| 1426 | + return false; |
| 1427 | + } elseif ($actual_offset < 0) { |
| 1428 | + $this->warnings[] = '$actual_offset < 0 in readEBMLint($string['.strlen($string).'], '.$offset.', '.$dataoffset.')'; |
| 1429 | + return false; |
| 1430 | + } |
| 1431 | + $first_byte_int = ord($string{$actual_offset}); |
| 1432 | + if (0x80 & $first_byte_int) { |
| 1433 | + $length = 1; |
| 1434 | + } elseif (0x40 & $first_byte_int) { |
| 1435 | + $length = 2; |
| 1436 | + } elseif (0x20 & $first_byte_int) { |
| 1437 | + $length = 3; |
| 1438 | + } elseif (0x10 & $first_byte_int) { |
| 1439 | + $length = 4; |
| 1440 | + } elseif (0x08 & $first_byte_int) { |
| 1441 | + $length = 5; |
| 1442 | + } elseif (0x04 & $first_byte_int) { |
| 1443 | + $length = 6; |
| 1444 | + } elseif (0x02 & $first_byte_int) { |
| 1445 | + $length = 7; |
| 1446 | + } elseif (0x01 & $first_byte_int) { |
| 1447 | + $length = 8; |
| 1448 | + } else { |
| 1449 | + $offset = pow(2,63); // abort processing, skip to end of file |
| 1450 | + $this->warnings[] = 'invalid EBML integer (leading 0x00) at '.$offset; |
| 1451 | + return false; |
| 1452 | + } |
| 1453 | + $int_value = $this->EBML2Int(substr($string, $actual_offset, $length)); |
| 1454 | + $offset += $length; |
| 1455 | + return $int_value; |
| 1456 | + } |
| 1457 | + |
| 1458 | + function EBML2Int($EBMLstring) { |
| 1459 | + // http://matroska.org/specs/ |
| 1460 | + |
| 1461 | + // Element ID coded with an UTF-8 like system: |
| 1462 | + // 1xxx xxxx - Class A IDs (2^7 -2 possible values) (base 0x8X) |
| 1463 | + // 01xx xxxx xxxx xxxx - Class B IDs (2^14-2 possible values) (base 0x4X 0xXX) |
| 1464 | + // 001x xxxx xxxx xxxx xxxx xxxx - Class C IDs (2^21-2 possible values) (base 0x2X 0xXX 0xXX) |
| 1465 | + // 0001 xxxx xxxx xxxx xxxx xxxx xxxx xxxx - Class D IDs (2^28-2 possible values) (base 0x1X 0xXX 0xXX 0xXX) |
| 1466 | + // Values with all x at 0 and 1 are reserved (hence the -2). |
| 1467 | + |
| 1468 | + // Data size, in octets, is also coded with an UTF-8 like system : |
| 1469 | + // 1xxx xxxx - value 0 to 2^7-2 |
| 1470 | + // 01xx xxxx xxxx xxxx - value 0 to 2^14-2 |
| 1471 | + // 001x xxxx xxxx xxxx xxxx xxxx - value 0 to 2^21-2 |
| 1472 | + // 0001 xxxx xxxx xxxx xxxx xxxx xxxx xxxx - value 0 to 2^28-2 |
| 1473 | + // 0000 1xxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx - value 0 to 2^35-2 |
| 1474 | + // 0000 01xx xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx - value 0 to 2^42-2 |
| 1475 | + // 0000 001x xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx - value 0 to 2^49-2 |
| 1476 | + // 0000 0001 xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx - value 0 to 2^56-2 |
| 1477 | + |
| 1478 | + $first_byte_int = ord($EBMLstring{0}); |
| 1479 | + if (0x80 & $first_byte_int) { |
| 1480 | + $EBMLstring{0} = chr($first_byte_int & 0x7F); |
| 1481 | + } elseif (0x40 & $first_byte_int) { |
| 1482 | + $EBMLstring{0} = chr($first_byte_int & 0x3F); |
| 1483 | + } elseif (0x20 & $first_byte_int) { |
| 1484 | + $EBMLstring{0} = chr($first_byte_int & 0x1F); |
| 1485 | + } elseif (0x10 & $first_byte_int) { |
| 1486 | + $EBMLstring{0} = chr($first_byte_int & 0x0F); |
| 1487 | + } elseif (0x08 & $first_byte_int) { |
| 1488 | + $EBMLstring{0} = chr($first_byte_int & 0x07); |
| 1489 | + } elseif (0x04 & $first_byte_int) { |
| 1490 | + $EBMLstring{0} = chr($first_byte_int & 0x03); |
| 1491 | + } elseif (0x02 & $first_byte_int) { |
| 1492 | + $EBMLstring{0} = chr($first_byte_int & 0x01); |
| 1493 | + } elseif (0x01 & $first_byte_int) { |
| 1494 | + $EBMLstring{0} = chr($first_byte_int & 0x00); |
| 1495 | + } else { |
| 1496 | + return false; |
| 1497 | + } |
| 1498 | + return getid3_lib::BigEndian2Int($EBMLstring); |
| 1499 | + } |
| 1500 | + |
| 1501 | + |
| 1502 | + function EBMLdate2unix($EBMLdatestamp) { |
| 1503 | + // Date - signed 8 octets integer in nanoseconds with 0 indicating the precise beginning of the millennium (at 2001-01-01T00:00:00,000000000 UTC) |
| 1504 | + // 978307200 == mktime(0, 0, 0, 1, 1, 2001) == January 1, 2001 12:00:00am UTC |
| 1505 | + return round(($EBMLdatestamp / 1000000000) + 978307200); |
| 1506 | + } |
| 1507 | + |
| 1508 | + |
| 1509 | + function MatroskaBlockLacingType($lacingtype) { |
| 1510 | + // http://matroska.org/technical/specs/index.html#block_structure |
| 1511 | + static $MatroskaBlockLacingType = array(); |
| 1512 | + if (empty($MatroskaBlockLacingType)) { |
| 1513 | + $MatroskaBlockLacingType[0x00] = 'no lacing'; |
| 1514 | + $MatroskaBlockLacingType[0x01] = 'Xiph lacing'; |
| 1515 | + $MatroskaBlockLacingType[0x02] = 'fixed-size lacing'; |
| 1516 | + $MatroskaBlockLacingType[0x03] = 'EBML lacing'; |
| 1517 | + } |
| 1518 | + return (isset($MatroskaBlockLacingType[$lacingtype]) ? $MatroskaBlockLacingType[$lacingtype] : $lacingtype); |
| 1519 | + } |
| 1520 | + |
| 1521 | + function MatroskaCodecIDtoCommonName($codecid) { |
| 1522 | + // http://www.matroska.org/technical/specs/codecid/index.html |
| 1523 | + static $MatroskaCodecIDlist = array(); |
| 1524 | + if (empty($MatroskaCodecIDlist)) { |
| 1525 | + $MatroskaCodecIDlist['A_AAC'] = 'aac'; |
| 1526 | + $MatroskaCodecIDlist['A_AAC/MPEG2/LC'] = 'aac'; |
| 1527 | + $MatroskaCodecIDlist['A_AC3'] = 'ac3'; |
| 1528 | + $MatroskaCodecIDlist['A_DTS'] = 'dts'; |
| 1529 | + $MatroskaCodecIDlist['A_FLAC'] = 'flac'; |
| 1530 | + $MatroskaCodecIDlist['A_MPEG/L1'] = 'mp1'; |
| 1531 | + $MatroskaCodecIDlist['A_MPEG/L2'] = 'mp2'; |
| 1532 | + $MatroskaCodecIDlist['A_MPEG/L3'] = 'mp3'; |
| 1533 | + $MatroskaCodecIDlist['A_PCM/INT/LIT'] = 'pcm'; // PCM Integer Little Endian |
| 1534 | + $MatroskaCodecIDlist['A_PCM/INT/BIG'] = 'pcm'; // PCM Integer Big Endian |
| 1535 | + $MatroskaCodecIDlist['A_QUICKTIME/QDMC'] = 'quicktime'; // Quicktime: QDesign Music |
| 1536 | + $MatroskaCodecIDlist['A_QUICKTIME/QDM2'] = 'quicktime'; // Quicktime: QDesign Music v2 |
| 1537 | + $MatroskaCodecIDlist['A_VORBIS'] = 'vorbis'; |
| 1538 | + $MatroskaCodecIDlist['V_MPEG1'] = 'mpeg'; |
| 1539 | + $MatroskaCodecIDlist['V_THEORA'] = 'theora'; |
| 1540 | + $MatroskaCodecIDlist['V_REAL/RV40'] = 'real'; |
| 1541 | + $MatroskaCodecIDlist['V_REAL/RV10'] = 'real'; |
| 1542 | + $MatroskaCodecIDlist['V_REAL/RV20'] = 'real'; |
| 1543 | + $MatroskaCodecIDlist['V_REAL/RV30'] = 'real'; |
| 1544 | + $MatroskaCodecIDlist['V_QUICKTIME'] = 'quicktime'; // Quicktime |
| 1545 | + $MatroskaCodecIDlist['V_MPEG4/ISO/AP'] = 'mpeg4'; |
| 1546 | + $MatroskaCodecIDlist['V_MPEG4/ISO/ASP'] = 'mpeg4'; |
| 1547 | + $MatroskaCodecIDlist['V_MPEG4/ISO/AVC'] = 'h264'; |
| 1548 | + $MatroskaCodecIDlist['V_MPEG4/ISO/SP'] = 'mpeg4'; |
| 1549 | + } |
| 1550 | + return (isset($MatroskaCodecIDlist[$codecid]) ? $MatroskaCodecIDlist[$codecid] : $codecid); |
| 1551 | + } |
| 1552 | + |
| 1553 | + function EBMLidName($value) { |
| 1554 | + static $EBMLidList = array(); |
| 1555 | + if (empty($EBMLidList)) { |
| 1556 | + $EBMLidList[EBML_ID_ASPECTRATIOTYPE] = 'AspectRatioType'; |
| 1557 | + $EBMLidList[EBML_ID_ATTACHEDFILE] = 'AttachedFile'; |
| 1558 | + $EBMLidList[EBML_ID_ATTACHMENTLINK] = 'AttachmentLink'; |
| 1559 | + $EBMLidList[EBML_ID_ATTACHMENTS] = 'Attachments'; |
| 1560 | + $EBMLidList[EBML_ID_ATTACHMENTUID] = 'AttachmentUID'; |
| 1561 | + $EBMLidList[EBML_ID_AUDIO] = 'Audio'; |
| 1562 | + $EBMLidList[EBML_ID_BITDEPTH] = 'BitDepth'; |
| 1563 | + $EBMLidList[EBML_ID_CHANNELPOSITIONS] = 'ChannelPositions'; |
| 1564 | + $EBMLidList[EBML_ID_CHANNELS] = 'Channels'; |
| 1565 | + $EBMLidList[EBML_ID_CHAPCOUNTRY] = 'ChapCountry'; |
| 1566 | + $EBMLidList[EBML_ID_CHAPLANGUAGE] = 'ChapLanguage'; |
| 1567 | + $EBMLidList[EBML_ID_CHAPPROCESS] = 'ChapProcess'; |
| 1568 | + $EBMLidList[EBML_ID_CHAPPROCESSCODECID] = 'ChapProcessCodecID'; |
| 1569 | + $EBMLidList[EBML_ID_CHAPPROCESSCOMMAND] = 'ChapProcessCommand'; |
| 1570 | + $EBMLidList[EBML_ID_CHAPPROCESSDATA] = 'ChapProcessData'; |
| 1571 | + $EBMLidList[EBML_ID_CHAPPROCESSPRIVATE] = 'ChapProcessPrivate'; |
| 1572 | + $EBMLidList[EBML_ID_CHAPPROCESSTIME] = 'ChapProcessTime'; |
| 1573 | + $EBMLidList[EBML_ID_CHAPSTRING] = 'ChapString'; |
| 1574 | + $EBMLidList[EBML_ID_CHAPTERATOM] = 'ChapterAtom'; |
| 1575 | + $EBMLidList[EBML_ID_CHAPTERDISPLAY] = 'ChapterDisplay'; |
| 1576 | + $EBMLidList[EBML_ID_CHAPTERFLAGENABLED] = 'ChapterFlagEnabled'; |
| 1577 | + $EBMLidList[EBML_ID_CHAPTERFLAGHIDDEN] = 'ChapterFlagHidden'; |
| 1578 | + $EBMLidList[EBML_ID_CHAPTERPHYSICALEQUIV] = 'ChapterPhysicalEquiv'; |
| 1579 | + $EBMLidList[EBML_ID_CHAPTERS] = 'Chapters'; |
| 1580 | + $EBMLidList[EBML_ID_CHAPTERSEGMENTEDITIONUID] = 'ChapterSegmentEditionUID'; |
| 1581 | + $EBMLidList[EBML_ID_CHAPTERSEGMENTUID] = 'ChapterSegmentUID'; |
| 1582 | + $EBMLidList[EBML_ID_CHAPTERTIMEEND] = 'ChapterTimeEnd'; |
| 1583 | + $EBMLidList[EBML_ID_CHAPTERTIMESTART] = 'ChapterTimeStart'; |
| 1584 | + $EBMLidList[EBML_ID_CHAPTERTRACK] = 'ChapterTrack'; |
| 1585 | + $EBMLidList[EBML_ID_CHAPTERTRACKNUMBER] = 'ChapterTrackNumber'; |
| 1586 | + $EBMLidList[EBML_ID_CHAPTERTRANSLATE] = 'ChapterTranslate'; |
| 1587 | + $EBMLidList[EBML_ID_CHAPTERTRANSLATECODEC] = 'ChapterTranslateCodec'; |
| 1588 | + $EBMLidList[EBML_ID_CHAPTERTRANSLATEEDITIONUID] = 'ChapterTranslateEditionUID'; |
| 1589 | + $EBMLidList[EBML_ID_CHAPTERTRANSLATEID] = 'ChapterTranslateID'; |
| 1590 | + $EBMLidList[EBML_ID_CHAPTERUID] = 'ChapterUID'; |
| 1591 | + $EBMLidList[EBML_ID_CLUSTER] = 'Cluster'; |
| 1592 | + $EBMLidList[EBML_ID_CLUSTERBLOCK] = 'ClusterBlock'; |
| 1593 | + $EBMLidList[EBML_ID_CLUSTERBLOCKADDID] = 'ClusterBlockAddID'; |
| 1594 | + $EBMLidList[EBML_ID_CLUSTERBLOCKADDITIONAL] = 'ClusterBlockAdditional'; |
| 1595 | + $EBMLidList[EBML_ID_CLUSTERBLOCKADDITIONID] = 'ClusterBlockAdditionID'; |
| 1596 | + $EBMLidList[EBML_ID_CLUSTERBLOCKADDITIONS] = 'ClusterBlockAdditions'; |
| 1597 | + $EBMLidList[EBML_ID_CLUSTERBLOCKDURATION] = 'ClusterBlockDuration'; |
| 1598 | + $EBMLidList[EBML_ID_CLUSTERBLOCKGROUP] = 'ClusterBlockGroup'; |
| 1599 | + $EBMLidList[EBML_ID_CLUSTERBLOCKMORE] = 'ClusterBlockMore'; |
| 1600 | + $EBMLidList[EBML_ID_CLUSTERBLOCKVIRTUAL] = 'ClusterBlockVirtual'; |
| 1601 | + $EBMLidList[EBML_ID_CLUSTERCODECSTATE] = 'ClusterCodecState'; |
| 1602 | + $EBMLidList[EBML_ID_CLUSTERDELAY] = 'ClusterDelay'; |
| 1603 | + $EBMLidList[EBML_ID_CLUSTERDURATION] = 'ClusterDuration'; |
| 1604 | + $EBMLidList[EBML_ID_CLUSTERENCRYPTEDBLOCK] = 'ClusterEncryptedBlock'; |
| 1605 | + $EBMLidList[EBML_ID_CLUSTERFRAMENUMBER] = 'ClusterFrameNumber'; |
| 1606 | + $EBMLidList[EBML_ID_CLUSTERLACENUMBER] = 'ClusterLaceNumber'; |
| 1607 | + $EBMLidList[EBML_ID_CLUSTERPOSITION] = 'ClusterPosition'; |
| 1608 | + $EBMLidList[EBML_ID_CLUSTERPREVSIZE] = 'ClusterPrevSize'; |
| 1609 | + $EBMLidList[EBML_ID_CLUSTERREFERENCEBLOCK] = 'ClusterReferenceBlock'; |
| 1610 | + $EBMLidList[EBML_ID_CLUSTERREFERENCEPRIORITY] = 'ClusterReferencePriority'; |
| 1611 | + $EBMLidList[EBML_ID_CLUSTERREFERENCEVIRTUAL] = 'ClusterReferenceVirtual'; |
| 1612 | + $EBMLidList[EBML_ID_CLUSTERSILENTTRACKNUMBER] = 'ClusterSilentTrackNumber'; |
| 1613 | + $EBMLidList[EBML_ID_CLUSTERSILENTTRACKS] = 'ClusterSilentTracks'; |
| 1614 | + $EBMLidList[EBML_ID_CLUSTERSIMPLEBLOCK] = 'ClusterSimpleBlock'; |
| 1615 | + $EBMLidList[EBML_ID_CLUSTERTIMECODE] = 'ClusterTimecode'; |
| 1616 | + $EBMLidList[EBML_ID_CLUSTERTIMESLICE] = 'ClusterTimeSlice'; |
| 1617 | + $EBMLidList[EBML_ID_CODECDECODEALL] = 'CodecDecodeAll'; |
| 1618 | + $EBMLidList[EBML_ID_CODECDOWNLOADURL] = 'CodecDownloadURL'; |
| 1619 | + $EBMLidList[EBML_ID_CODECID] = 'CodecID'; |
| 1620 | + $EBMLidList[EBML_ID_CODECINFOURL] = 'CodecInfoURL'; |
| 1621 | + $EBMLidList[EBML_ID_CODECNAME] = 'CodecName'; |
| 1622 | + $EBMLidList[EBML_ID_CODECPRIVATE] = 'CodecPrivate'; |
| 1623 | + $EBMLidList[EBML_ID_CODECSETTINGS] = 'CodecSettings'; |
| 1624 | + $EBMLidList[EBML_ID_COLOURSPACE] = 'ColourSpace'; |
| 1625 | + $EBMLidList[EBML_ID_CONTENTCOMPALGO] = 'ContentCompAlgo'; |
| 1626 | + $EBMLidList[EBML_ID_CONTENTCOMPRESSION] = 'ContentCompression'; |
| 1627 | + $EBMLidList[EBML_ID_CONTENTCOMPSETTINGS] = 'ContentCompSettings'; |
| 1628 | + $EBMLidList[EBML_ID_CONTENTENCALGO] = 'ContentEncAlgo'; |
| 1629 | + $EBMLidList[EBML_ID_CONTENTENCKEYID] = 'ContentEncKeyID'; |
| 1630 | + $EBMLidList[EBML_ID_CONTENTENCODING] = 'ContentEncoding'; |
| 1631 | + $EBMLidList[EBML_ID_CONTENTENCODINGORDER] = 'ContentEncodingOrder'; |
| 1632 | + $EBMLidList[EBML_ID_CONTENTENCODINGS] = 'ContentEncodings'; |
| 1633 | + $EBMLidList[EBML_ID_CONTENTENCODINGSCOPE] = 'ContentEncodingScope'; |
| 1634 | + $EBMLidList[EBML_ID_CONTENTENCODINGTYPE] = 'ContentEncodingType'; |
| 1635 | + $EBMLidList[EBML_ID_CONTENTENCRYPTION] = 'ContentEncryption'; |
| 1636 | + $EBMLidList[EBML_ID_CONTENTSIGALGO] = 'ContentSigAlgo'; |
| 1637 | + $EBMLidList[EBML_ID_CONTENTSIGHASHALGO] = 'ContentSigHashAlgo'; |
| 1638 | + $EBMLidList[EBML_ID_CONTENTSIGKEYID] = 'ContentSigKeyID'; |
| 1639 | + $EBMLidList[EBML_ID_CONTENTSIGNATURE] = 'ContentSignature'; |
| 1640 | + $EBMLidList[EBML_ID_CRC32] = 'CRC32'; |
| 1641 | + $EBMLidList[EBML_ID_CUEBLOCKNUMBER] = 'CueBlockNumber'; |
| 1642 | + $EBMLidList[EBML_ID_CUECLUSTERPOSITION] = 'CueClusterPosition'; |
| 1643 | + $EBMLidList[EBML_ID_CUECODECSTATE] = 'CueCodecState'; |
| 1644 | + $EBMLidList[EBML_ID_CUEPOINT] = 'CuePoint'; |
| 1645 | + $EBMLidList[EBML_ID_CUEREFCLUSTER] = 'CueRefCluster'; |
| 1646 | + $EBMLidList[EBML_ID_CUEREFCODECSTATE] = 'CueRefCodecState'; |
| 1647 | + $EBMLidList[EBML_ID_CUEREFERENCE] = 'CueReference'; |
| 1648 | + $EBMLidList[EBML_ID_CUEREFNUMBER] = 'CueRefNumber'; |
| 1649 | + $EBMLidList[EBML_ID_CUEREFTIME] = 'CueRefTime'; |
| 1650 | + $EBMLidList[EBML_ID_CUES] = 'Cues'; |
| 1651 | + $EBMLidList[EBML_ID_CUETIME] = 'CueTime'; |
| 1652 | + $EBMLidList[EBML_ID_CUETRACK] = 'CueTrack'; |
| 1653 | + $EBMLidList[EBML_ID_CUETRACKPOSITIONS] = 'CueTrackPositions'; |
| 1654 | + $EBMLidList[EBML_ID_DATEUTC] = 'DateUTC'; |
| 1655 | + $EBMLidList[EBML_ID_DEFAULTDURATION] = 'DefaultDuration'; |
| 1656 | + $EBMLidList[EBML_ID_DISPLAYHEIGHT] = 'DisplayHeight'; |
| 1657 | + $EBMLidList[EBML_ID_DISPLAYUNIT] = 'DisplayUnit'; |
| 1658 | + $EBMLidList[EBML_ID_DISPLAYWIDTH] = 'DisplayWidth'; |
| 1659 | + $EBMLidList[EBML_ID_DOCTYPE] = 'DocType'; |
| 1660 | + $EBMLidList[EBML_ID_DOCTYPEREADVERSION] = 'DocTypeReadVersion'; |
| 1661 | + $EBMLidList[EBML_ID_DOCTYPEVERSION] = 'DocTypeVersion'; |
| 1662 | + $EBMLidList[EBML_ID_DURATION] = 'Duration'; |
| 1663 | + $EBMLidList[EBML_ID_EBMLMAXIDLENGTH] = 'EBMLMaxIDLength'; |
| 1664 | + $EBMLidList[EBML_ID_EBMLMAXSIZELENGTH] = 'EBMLMaxSizeLength'; |
| 1665 | + $EBMLidList[EBML_ID_EBMLREADVERSION] = 'EBMLReadVersion'; |
| 1666 | + $EBMLidList[EBML_ID_EBMLVERSION] = 'EBMLVersion'; |
| 1667 | + $EBMLidList[EBML_ID_EDITIONENTRY] = 'EditionEntry'; |
| 1668 | + $EBMLidList[EBML_ID_EDITIONFLAGDEFAULT] = 'EditionFlagDefault'; |
| 1669 | + $EBMLidList[EBML_ID_EDITIONFLAGHIDDEN] = 'EditionFlagHidden'; |
| 1670 | + $EBMLidList[EBML_ID_EDITIONFLAGORDERED] = 'EditionFlagOrdered'; |
| 1671 | + $EBMLidList[EBML_ID_EDITIONUID] = 'EditionUID'; |
| 1672 | + $EBMLidList[EBML_ID_FILEDATA] = 'FileData'; |
| 1673 | + $EBMLidList[EBML_ID_FILEDESCRIPTION] = 'FileDescription'; |
| 1674 | + $EBMLidList[EBML_ID_FILEMIMETYPE] = 'FileMimeType'; |
| 1675 | + $EBMLidList[EBML_ID_FILENAME] = 'FileName'; |
| 1676 | + $EBMLidList[EBML_ID_FILEREFERRAL] = 'FileReferral'; |
| 1677 | + $EBMLidList[EBML_ID_FILEUID] = 'FileUID'; |
| 1678 | + $EBMLidList[EBML_ID_FLAGDEFAULT] = 'FlagDefault'; |
| 1679 | + $EBMLidList[EBML_ID_FLAGENABLED] = 'FlagEnabled'; |
| 1680 | + $EBMLidList[EBML_ID_FLAGFORCED] = 'FlagForced'; |
| 1681 | + $EBMLidList[EBML_ID_FLAGINTERLACED] = 'FlagInterlaced'; |
| 1682 | + $EBMLidList[EBML_ID_FLAGLACING] = 'FlagLacing'; |
| 1683 | + $EBMLidList[EBML_ID_GAMMAVALUE] = 'GammaValue'; |
| 1684 | + $EBMLidList[EBML_ID_INFO] = 'Info'; |
| 1685 | + $EBMLidList[EBML_ID_LANGUAGE] = 'Language'; |
| 1686 | + $EBMLidList[EBML_ID_MAXBLOCKADDITIONID] = 'MaxBlockAdditionID'; |
| 1687 | + $EBMLidList[EBML_ID_MAXCACHE] = 'MaxCache'; |
| 1688 | + $EBMLidList[EBML_ID_MINCACHE] = 'MinCache'; |
| 1689 | + $EBMLidList[EBML_ID_MUXINGAPP] = 'MuxingApp'; |
| 1690 | + $EBMLidList[EBML_ID_NAME] = 'Name'; |
| 1691 | + $EBMLidList[EBML_ID_NEXTFILENAME] = 'NextFilename'; |
| 1692 | + $EBMLidList[EBML_ID_NEXTUID] = 'NextUID'; |
| 1693 | + $EBMLidList[EBML_ID_OUTPUTSAMPLINGFREQUENCY] = 'OutputSamplingFrequency'; |
| 1694 | + $EBMLidList[EBML_ID_PIXELCROPBOTTOM] = 'PixelCropBottom'; |
| 1695 | + $EBMLidList[EBML_ID_PIXELCROPLEFT] = 'PixelCropLeft'; |
| 1696 | + $EBMLidList[EBML_ID_PIXELCROPRIGHT] = 'PixelCropRight'; |
| 1697 | + $EBMLidList[EBML_ID_PIXELCROPTOP] = 'PixelCropTop'; |
| 1698 | + $EBMLidList[EBML_ID_PIXELHEIGHT] = 'PixelHeight'; |
| 1699 | + $EBMLidList[EBML_ID_PIXELWIDTH] = 'PixelWidth'; |
| 1700 | + $EBMLidList[EBML_ID_PREVFILENAME] = 'PrevFilename'; |
| 1701 | + $EBMLidList[EBML_ID_PREVUID] = 'PrevUID'; |
| 1702 | + $EBMLidList[EBML_ID_SAMPLINGFREQUENCY] = 'SamplingFrequency'; |
| 1703 | + $EBMLidList[EBML_ID_SEEK] = 'Seek'; |
| 1704 | + $EBMLidList[EBML_ID_SEEKHEAD] = 'SeekHead'; |
| 1705 | + $EBMLidList[EBML_ID_SEEKID] = 'SeekID'; |
| 1706 | + $EBMLidList[EBML_ID_SEEKPOSITION] = 'SeekPosition'; |
| 1707 | + $EBMLidList[EBML_ID_SEGMENTFAMILY] = 'SegmentFamily'; |
| 1708 | + $EBMLidList[EBML_ID_SEGMENTFILENAME] = 'SegmentFilename'; |
| 1709 | + $EBMLidList[EBML_ID_SEGMENTUID] = 'SegmentUID'; |
| 1710 | + $EBMLidList[EBML_ID_SIMPLETAG] = 'SimpleTag'; |
| 1711 | + $EBMLidList[EBML_ID_CLUSTERSLICES] = 'ClusterSlices'; |
| 1712 | + $EBMLidList[EBML_ID_STEREOMODE] = 'StereoMode'; |
| 1713 | + $EBMLidList[EBML_ID_TAG] = 'Tag'; |
| 1714 | + $EBMLidList[EBML_ID_TAGBINARY] = 'TagBinary'; |
| 1715 | + $EBMLidList[EBML_ID_TAGCHAPTERUID] = 'TagChapterUID'; |
| 1716 | + $EBMLidList[EBML_ID_TAGDEFAULT] = 'TagDefault'; |
| 1717 | + $EBMLidList[EBML_ID_TAGEDITIONUID] = 'TagEditionUID'; |
| 1718 | + $EBMLidList[EBML_ID_TAGLANGUAGE] = 'TagLanguage'; |
| 1719 | + $EBMLidList[EBML_ID_TAGNAME] = 'TagName'; |
| 1720 | + $EBMLidList[EBML_ID_TAGTRACKUID] = 'TagTrackUID'; |
| 1721 | + $EBMLidList[EBML_ID_TAGS] = 'Tags'; |
| 1722 | + $EBMLidList[EBML_ID_TAGSTRING] = 'TagString'; |
| 1723 | + $EBMLidList[EBML_ID_TARGETS] = 'Targets'; |
| 1724 | + $EBMLidList[EBML_ID_TARGETTYPE] = 'TargetType'; |
| 1725 | + $EBMLidList[EBML_ID_TARGETTYPEVALUE] = 'TargetTypeValue'; |
| 1726 | + $EBMLidList[EBML_ID_TIMECODESCALE] = 'TimecodeScale'; |
| 1727 | + $EBMLidList[EBML_ID_TITLE] = 'Title'; |
| 1728 | + $EBMLidList[EBML_ID_TRACKENTRY] = 'TrackEntry'; |
| 1729 | + $EBMLidList[EBML_ID_TRACKNUMBER] = 'TrackNumber'; |
| 1730 | + $EBMLidList[EBML_ID_TRACKOFFSET] = 'TrackOffset'; |
| 1731 | + $EBMLidList[EBML_ID_TRACKOVERLAY] = 'TrackOverlay'; |
| 1732 | + $EBMLidList[EBML_ID_TRACKS] = 'Tracks'; |
| 1733 | + $EBMLidList[EBML_ID_TRACKTIMECODESCALE] = 'TrackTimecodeScale'; |
| 1734 | + $EBMLidList[EBML_ID_TRACKTRANSLATE] = 'TrackTranslate'; |
| 1735 | + $EBMLidList[EBML_ID_TRACKTRANSLATECODEC] = 'TrackTranslateCodec'; |
| 1736 | + $EBMLidList[EBML_ID_TRACKTRANSLATEEDITIONUID] = 'TrackTranslateEditionUID'; |
| 1737 | + $EBMLidList[EBML_ID_TRACKTRANSLATETRACKID] = 'TrackTranslateTrackID'; |
| 1738 | + $EBMLidList[EBML_ID_TRACKTYPE] = 'TrackType'; |
| 1739 | + $EBMLidList[EBML_ID_TRACKUID] = 'TrackUID'; |
| 1740 | + $EBMLidList[EBML_ID_VIDEO] = 'Video'; |
| 1741 | + $EBMLidList[EBML_ID_VOID] = 'Void'; |
| 1742 | + $EBMLidList[EBML_ID_WRITINGAPP] = 'WritingApp'; |
| 1743 | + } |
| 1744 | + return (isset($EBMLidList[$value]) ? $EBMLidList[$value] : dechex($value)); |
| 1745 | + } |
| 1746 | + |
| 1747 | +} |
| 1748 | + |
| 1749 | +?> |
\ No newline at end of file |
Index: trunk/extensions/TimedMediaHandler/getid3/module.audio.flac.php |
— | — | @@ -0,0 +1,403 @@ |
| 2 | +<?php |
| 3 | +///////////////////////////////////////////////////////////////// |
| 4 | +/// getID3() by James Heinrich <info@getid3.org> // |
| 5 | +// available at http://getid3.sourceforge.net // |
| 6 | +// or http://www.getid3.org // |
| 7 | +///////////////////////////////////////////////////////////////// |
| 8 | +// See readme.txt for more details // |
| 9 | +///////////////////////////////////////////////////////////////// |
| 10 | +// // |
| 11 | +// module.audio.flac.php // |
| 12 | +// module for analyzing FLAC and OggFLAC audio files // |
| 13 | +// dependencies: module.audio.ogg.php // |
| 14 | +// /// |
| 15 | +///////////////////////////////////////////////////////////////// |
| 16 | + |
| 17 | + |
| 18 | +getid3_lib::IncludeDependency(GETID3_INCLUDEPATH.'module.audio.ogg.php', __FILE__, true); |
| 19 | + |
| 20 | +class getid3_flac |
| 21 | +{ |
| 22 | + |
| 23 | + function getid3_flac(&$fd, &$ThisFileInfo) { |
| 24 | + // http://flac.sourceforge.net/format.html |
| 25 | + |
| 26 | + fseek($fd, $ThisFileInfo['avdataoffset'], SEEK_SET); |
| 27 | + $StreamMarker = fread($fd, 4); |
| 28 | + if ($StreamMarker != 'fLaC') { |
| 29 | + $ThisFileInfo['error'][] = 'Expecting "fLaC" at offset '.$ThisFileInfo['avdataoffset'].', found "'.$StreamMarker.'"'; |
| 30 | + return false; |
| 31 | + } |
| 32 | + $ThisFileInfo['fileformat'] = 'flac'; |
| 33 | + $ThisFileInfo['audio']['dataformat'] = 'flac'; |
| 34 | + $ThisFileInfo['audio']['bitrate_mode'] = 'vbr'; |
| 35 | + $ThisFileInfo['audio']['lossless'] = true; |
| 36 | + |
| 37 | + return getid3_flac::FLACparseMETAdata($fd, $ThisFileInfo); |
| 38 | + } |
| 39 | + |
| 40 | + |
| 41 | + static function FLACparseMETAdata(&$fd, &$ThisFileInfo) { |
| 42 | + |
| 43 | + do { |
| 44 | + $METAdataBlockOffset = ftell($fd); |
| 45 | + $METAdataBlockHeader = fread($fd, 4); |
| 46 | + $METAdataLastBlockFlag = (bool) (getid3_lib::BigEndian2Int(substr($METAdataBlockHeader, 0, 1)) & 0x80); |
| 47 | + $METAdataBlockType = getid3_lib::BigEndian2Int(substr($METAdataBlockHeader, 0, 1)) & 0x7F; |
| 48 | + $METAdataBlockLength = getid3_lib::BigEndian2Int(substr($METAdataBlockHeader, 1, 3)); |
| 49 | + $METAdataBlockTypeText = getid3_flac::FLACmetaBlockTypeLookup($METAdataBlockType); |
| 50 | + |
| 51 | + if ($METAdataBlockLength < 0) { |
| 52 | + $ThisFileInfo['error'][] = 'corrupt or invalid METADATA_BLOCK_HEADER.BLOCK_TYPE ('.$METAdataBlockType.') at offset '.$METAdataBlockOffset; |
| 53 | + break; |
| 54 | + } |
| 55 | + |
| 56 | + $ThisFileInfo['flac'][$METAdataBlockTypeText]['raw'] = array(); |
| 57 | + $ThisFileInfo_flac_METAdataBlockTypeText_raw = &$ThisFileInfo['flac'][$METAdataBlockTypeText]['raw']; |
| 58 | + |
| 59 | + $ThisFileInfo_flac_METAdataBlockTypeText_raw['offset'] = $METAdataBlockOffset; |
| 60 | + $ThisFileInfo_flac_METAdataBlockTypeText_raw['last_meta_block'] = $METAdataLastBlockFlag; |
| 61 | + $ThisFileInfo_flac_METAdataBlockTypeText_raw['block_type'] = $METAdataBlockType; |
| 62 | + $ThisFileInfo_flac_METAdataBlockTypeText_raw['block_type_text'] = $METAdataBlockTypeText; |
| 63 | + $ThisFileInfo_flac_METAdataBlockTypeText_raw['block_length'] = $METAdataBlockLength; |
| 64 | + ob_start(); |
| 65 | + $ThisFileInfo_flac_METAdataBlockTypeText_raw['block_data'] = fread($fd, $METAdataBlockLength); |
| 66 | + $errormessage = ob_get_contents(); |
| 67 | + ob_end_clean(); |
| 68 | + $ThisFileInfo['avdataoffset'] = ftell($fd); |
| 69 | + |
| 70 | + switch ($METAdataBlockTypeText) { |
| 71 | + case 'STREAMINFO': // 0x00 |
| 72 | + if (!getid3_flac::FLACparseSTREAMINFO($ThisFileInfo_flac_METAdataBlockTypeText_raw['block_data'], $ThisFileInfo)) { |
| 73 | + return false; |
| 74 | + } |
| 75 | + break; |
| 76 | + |
| 77 | + case 'PADDING': // 0x01 |
| 78 | + // ignore |
| 79 | + break; |
| 80 | + |
| 81 | + case 'APPLICATION': // 0x02 |
| 82 | + if (!getid3_flac::FLACparseAPPLICATION($ThisFileInfo_flac_METAdataBlockTypeText_raw['block_data'], $ThisFileInfo)) { |
| 83 | + return false; |
| 84 | + } |
| 85 | + break; |
| 86 | + |
| 87 | + case 'SEEKTABLE': // 0x03 |
| 88 | + if (!getid3_flac::FLACparseSEEKTABLE($ThisFileInfo_flac_METAdataBlockTypeText_raw['block_data'], $ThisFileInfo)) { |
| 89 | + return false; |
| 90 | + } |
| 91 | + break; |
| 92 | + |
| 93 | + case 'VORBIS_COMMENT': // 0x04 |
| 94 | + $OldOffset = ftell($fd); |
| 95 | + fseek($fd, 0 - $METAdataBlockLength, SEEK_CUR); |
| 96 | + getid3_ogg::ParseVorbisCommentsFilepointer($fd, $ThisFileInfo); |
| 97 | + fseek($fd, $OldOffset, SEEK_SET); |
| 98 | + break; |
| 99 | + |
| 100 | + case 'CUESHEET': // 0x05 |
| 101 | + if (!getid3_flac::FLACparseCUESHEET($ThisFileInfo_flac_METAdataBlockTypeText_raw['block_data'], $ThisFileInfo)) { |
| 102 | + return false; |
| 103 | + } |
| 104 | + break; |
| 105 | + |
| 106 | + case 'PICTURE': // 0x06 |
| 107 | + if (!getid3_flac::FLACparsePICTURE($ThisFileInfo_flac_METAdataBlockTypeText_raw['block_data'], $ThisFileInfo)) { |
| 108 | + return false; |
| 109 | + } |
| 110 | + break; |
| 111 | + |
| 112 | + default: |
| 113 | + $ThisFileInfo['warning'][] = 'Unhandled METADATA_BLOCK_HEADER.BLOCK_TYPE ('.$METAdataBlockType.') at offset '.$METAdataBlockOffset; |
| 114 | + break; |
| 115 | + } |
| 116 | + |
| 117 | + } while ($METAdataLastBlockFlag === false); |
| 118 | + |
| 119 | + |
| 120 | + if (isset($ThisFileInfo['flac']['STREAMINFO'])) { |
| 121 | + $ThisFileInfo['flac']['compressed_audio_bytes'] = $ThisFileInfo['avdataend'] - $ThisFileInfo['avdataoffset']; |
| 122 | + $ThisFileInfo['flac']['uncompressed_audio_bytes'] = $ThisFileInfo['flac']['STREAMINFO']['samples_stream'] * $ThisFileInfo['flac']['STREAMINFO']['channels'] * ($ThisFileInfo['flac']['STREAMINFO']['bits_per_sample'] / 8); |
| 123 | + if ($ThisFileInfo['flac']['uncompressed_audio_bytes'] == 0) { |
| 124 | + $ThisFileInfo['error'][] = 'Corrupt FLAC file: uncompressed_audio_bytes == zero'; |
| 125 | + return false; |
| 126 | + } |
| 127 | + $ThisFileInfo['flac']['compression_ratio'] = $ThisFileInfo['flac']['compressed_audio_bytes'] / $ThisFileInfo['flac']['uncompressed_audio_bytes']; |
| 128 | + } |
| 129 | + |
| 130 | + // set md5_data_source - built into flac 0.5+ |
| 131 | + if (isset($ThisFileInfo['flac']['STREAMINFO']['audio_signature'])) { |
| 132 | + |
| 133 | + if ($ThisFileInfo['flac']['STREAMINFO']['audio_signature'] === str_repeat("\x00", 16)) { |
| 134 | + |
| 135 | + $ThisFileInfo['warning'][] = 'FLAC STREAMINFO.audio_signature is null (known issue with libOggFLAC)'; |
| 136 | + |
| 137 | + } else { |
| 138 | + |
| 139 | + $ThisFileInfo['md5_data_source'] = ''; |
| 140 | + $md5 = $ThisFileInfo['flac']['STREAMINFO']['audio_signature']; |
| 141 | + for ($i = 0; $i < strlen($md5); $i++) { |
| 142 | + $ThisFileInfo['md5_data_source'] .= str_pad(dechex(ord($md5{$i})), 2, '00', STR_PAD_LEFT); |
| 143 | + } |
| 144 | + if (!preg_match('/^[0-9a-f]{32}$/', $ThisFileInfo['md5_data_source'])) { |
| 145 | + unset($ThisFileInfo['md5_data_source']); |
| 146 | + } |
| 147 | + |
| 148 | + } |
| 149 | + |
| 150 | + } |
| 151 | + |
| 152 | + $ThisFileInfo['audio']['bits_per_sample'] = $ThisFileInfo['flac']['STREAMINFO']['bits_per_sample']; |
| 153 | + if ($ThisFileInfo['audio']['bits_per_sample'] == 8) { |
| 154 | + // special case |
| 155 | + // must invert sign bit on all data bytes before MD5'ing to match FLAC's calculated value |
| 156 | + // MD5sum calculates on unsigned bytes, but FLAC calculated MD5 on 8-bit audio data as signed |
| 157 | + $ThisFileInfo['warning'][] = 'FLAC calculates MD5 data strangely on 8-bit audio, so the stored md5_data_source value will not match the decoded WAV file'; |
| 158 | + } |
| 159 | + if (!empty($ThisFileInfo['ogg']['vendor'])) { |
| 160 | + $ThisFileInfo['audio']['encoder'] = $ThisFileInfo['ogg']['vendor']; |
| 161 | + } |
| 162 | + |
| 163 | + return true; |
| 164 | + } |
| 165 | + |
| 166 | + static function FLACmetaBlockTypeLookup($blocktype) { |
| 167 | + static $FLACmetaBlockTypeLookup = array(); |
| 168 | + if (empty($FLACmetaBlockTypeLookup)) { |
| 169 | + $FLACmetaBlockTypeLookup[0] = 'STREAMINFO'; |
| 170 | + $FLACmetaBlockTypeLookup[1] = 'PADDING'; |
| 171 | + $FLACmetaBlockTypeLookup[2] = 'APPLICATION'; |
| 172 | + $FLACmetaBlockTypeLookup[3] = 'SEEKTABLE'; |
| 173 | + $FLACmetaBlockTypeLookup[4] = 'VORBIS_COMMENT'; |
| 174 | + $FLACmetaBlockTypeLookup[5] = 'CUESHEET'; |
| 175 | + $FLACmetaBlockTypeLookup[6] = 'PICTURE'; |
| 176 | + } |
| 177 | + return (isset($FLACmetaBlockTypeLookup[$blocktype]) ? $FLACmetaBlockTypeLookup[$blocktype] : 'reserved'); |
| 178 | + } |
| 179 | + |
| 180 | + static function FLACapplicationIDLookup($applicationid) { |
| 181 | + static $FLACapplicationIDLookup = array(); |
| 182 | + if (empty($FLACapplicationIDLookup)) { |
| 183 | + // http://flac.sourceforge.net/id.html |
| 184 | + $FLACapplicationIDLookup[0x46746F6C] = 'flac-tools'; // 'Ftol' |
| 185 | + $FLACapplicationIDLookup[0x46746F6C] = 'Sound Font FLAC'; // 'SFFL' |
| 186 | + } |
| 187 | + return (isset($FLACapplicationIDLookup[$applicationid]) ? $FLACapplicationIDLookup[$applicationid] : 'reserved'); |
| 188 | + } |
| 189 | + |
| 190 | + static function FLACpictureTypeLookup($type_id) { |
| 191 | + static $lookup = array ( |
| 192 | + 0 => 'Other', |
| 193 | + 1 => '32x32 pixels \'file icon\' (PNG only)', |
| 194 | + 2 => 'Other file icon', |
| 195 | + 3 => 'Cover (front)', |
| 196 | + 4 => 'Cover (back)', |
| 197 | + 5 => 'Leaflet page', |
| 198 | + 6 => 'Media (e.g. label side of CD)', |
| 199 | + 7 => 'Lead artist/lead performer/soloist', |
| 200 | + 8 => 'Artist/performer', |
| 201 | + 9 => 'Conductor', |
| 202 | + 10 => 'Band/Orchestra', |
| 203 | + 11 => 'Composer', |
| 204 | + 12 => 'Lyricist/text writer', |
| 205 | + 13 => 'Recording Location', |
| 206 | + 14 => 'During recording', |
| 207 | + 15 => 'During performance', |
| 208 | + 16 => 'Movie/video screen capture', |
| 209 | + 17 => 'A bright coloured fish', |
| 210 | + 18 => 'Illustration', |
| 211 | + 19 => 'Band/artist logotype', |
| 212 | + 20 => 'Publisher/Studio logotype', |
| 213 | + ); |
| 214 | + return (isset($lookup[$type_id]) ? $lookup[$type_id] : 'reserved'); |
| 215 | + } |
| 216 | + |
| 217 | + static function FLACparseSTREAMINFO($METAdataBlockData, &$ThisFileInfo) { |
| 218 | + $offset = 0; |
| 219 | + $ThisFileInfo['flac']['STREAMINFO']['min_block_size'] = getid3_lib::BigEndian2Int(substr($METAdataBlockData, $offset, 2)); |
| 220 | + $offset += 2; |
| 221 | + $ThisFileInfo['flac']['STREAMINFO']['max_block_size'] = getid3_lib::BigEndian2Int(substr($METAdataBlockData, $offset, 2)); |
| 222 | + $offset += 2; |
| 223 | + $ThisFileInfo['flac']['STREAMINFO']['min_frame_size'] = getid3_lib::BigEndian2Int(substr($METAdataBlockData, $offset, 3)); |
| 224 | + $offset += 3; |
| 225 | + $ThisFileInfo['flac']['STREAMINFO']['max_frame_size'] = getid3_lib::BigEndian2Int(substr($METAdataBlockData, $offset, 3)); |
| 226 | + $offset += 3; |
| 227 | + |
| 228 | + $SampleRateChannelsSampleBitsStreamSamples = getid3_lib::BigEndian2Bin(substr($METAdataBlockData, $offset, 8)); |
| 229 | + $ThisFileInfo['flac']['STREAMINFO']['sample_rate'] = getid3_lib::Bin2Dec(substr($SampleRateChannelsSampleBitsStreamSamples, 0, 20)); |
| 230 | + $ThisFileInfo['flac']['STREAMINFO']['channels'] = getid3_lib::Bin2Dec(substr($SampleRateChannelsSampleBitsStreamSamples, 20, 3)) + 1; |
| 231 | + $ThisFileInfo['flac']['STREAMINFO']['bits_per_sample'] = getid3_lib::Bin2Dec(substr($SampleRateChannelsSampleBitsStreamSamples, 23, 5)) + 1; |
| 232 | + $ThisFileInfo['flac']['STREAMINFO']['samples_stream'] = getid3_lib::Bin2Dec(substr($SampleRateChannelsSampleBitsStreamSamples, 28, 36)); |
| 233 | + $offset += 8; |
| 234 | + |
| 235 | + $ThisFileInfo['flac']['STREAMINFO']['audio_signature'] = substr($METAdataBlockData, $offset, 16); |
| 236 | + $offset += 16; |
| 237 | + |
| 238 | + if (!empty($ThisFileInfo['flac']['STREAMINFO']['sample_rate'])) { |
| 239 | + |
| 240 | + $ThisFileInfo['audio']['bitrate_mode'] = 'vbr'; |
| 241 | + $ThisFileInfo['audio']['sample_rate'] = $ThisFileInfo['flac']['STREAMINFO']['sample_rate']; |
| 242 | + $ThisFileInfo['audio']['channels'] = $ThisFileInfo['flac']['STREAMINFO']['channels']; |
| 243 | + $ThisFileInfo['audio']['bits_per_sample'] = $ThisFileInfo['flac']['STREAMINFO']['bits_per_sample']; |
| 244 | + $ThisFileInfo['playtime_seconds'] = $ThisFileInfo['flac']['STREAMINFO']['samples_stream'] / $ThisFileInfo['flac']['STREAMINFO']['sample_rate']; |
| 245 | + if ($ThisFileInfo['playtime_seconds'] > 0) { |
| 246 | + $ThisFileInfo['audio']['bitrate'] = (($ThisFileInfo['avdataend'] - $ThisFileInfo['avdataoffset']) * 8) / $ThisFileInfo['playtime_seconds']; |
| 247 | + } |
| 248 | + |
| 249 | + } else { |
| 250 | + |
| 251 | + $ThisFileInfo['error'][] = 'Corrupt METAdata block: STREAMINFO'; |
| 252 | + return false; |
| 253 | + |
| 254 | + } |
| 255 | + |
| 256 | + unset($ThisFileInfo['flac']['STREAMINFO']['raw']); |
| 257 | + |
| 258 | + return true; |
| 259 | + } |
| 260 | + |
| 261 | + |
| 262 | + static function FLACparseAPPLICATION($METAdataBlockData, &$ThisFileInfo) { |
| 263 | + $offset = 0; |
| 264 | + $ApplicationID = getid3_lib::BigEndian2Int(substr($METAdataBlockData, $offset, 4)); |
| 265 | + $offset += 4; |
| 266 | + $ThisFileInfo['flac']['APPLICATION'][$ApplicationID]['name'] = getid3_flac::FLACapplicationIDLookup($ApplicationID); |
| 267 | + $ThisFileInfo['flac']['APPLICATION'][$ApplicationID]['data'] = substr($METAdataBlockData, $offset); |
| 268 | + $offset = $METAdataBlockLength; |
| 269 | + |
| 270 | + unset($ThisFileInfo['flac']['APPLICATION']['raw']); |
| 271 | + |
| 272 | + return true; |
| 273 | + } |
| 274 | + |
| 275 | + |
| 276 | + static function FLACparseSEEKTABLE($METAdataBlockData, &$ThisFileInfo) { |
| 277 | + $offset = 0; |
| 278 | + $METAdataBlockLength = strlen($METAdataBlockData); |
| 279 | + $placeholderpattern = str_repeat("\xFF", 8); |
| 280 | + while ($offset < $METAdataBlockLength) { |
| 281 | + $SampleNumberString = substr($METAdataBlockData, $offset, 8); |
| 282 | + $offset += 8; |
| 283 | + if ($SampleNumberString == $placeholderpattern) { |
| 284 | + |
| 285 | + // placeholder point |
| 286 | + getid3_lib::safe_inc($ThisFileInfo['flac']['SEEKTABLE']['placeholders'], 1); |
| 287 | + $offset += 10; |
| 288 | + |
| 289 | + } else { |
| 290 | + |
| 291 | + $SampleNumber = getid3_lib::BigEndian2Int($SampleNumberString); |
| 292 | + $ThisFileInfo['flac']['SEEKTABLE'][$SampleNumber]['offset'] = getid3_lib::BigEndian2Int(substr($METAdataBlockData, $offset, 8)); |
| 293 | + $offset += 8; |
| 294 | + $ThisFileInfo['flac']['SEEKTABLE'][$SampleNumber]['samples'] = getid3_lib::BigEndian2Int(substr($METAdataBlockData, $offset, 2)); |
| 295 | + $offset += 2; |
| 296 | + |
| 297 | + } |
| 298 | + } |
| 299 | + |
| 300 | + unset($ThisFileInfo['flac']['SEEKTABLE']['raw']); |
| 301 | + |
| 302 | + return true; |
| 303 | + } |
| 304 | + |
| 305 | + static function FLACparseCUESHEET($METAdataBlockData, &$ThisFileInfo) { |
| 306 | + $offset = 0; |
| 307 | + $ThisFileInfo['flac']['CUESHEET']['media_catalog_number'] = trim(substr($METAdataBlockData, $offset, 128), "\0"); |
| 308 | + $offset += 128; |
| 309 | + $ThisFileInfo['flac']['CUESHEET']['lead_in_samples'] = getid3_lib::BigEndian2Int(substr($METAdataBlockData, $offset, 8)); |
| 310 | + $offset += 8; |
| 311 | + $ThisFileInfo['flac']['CUESHEET']['flags']['is_cd'] = (bool) (getid3_lib::BigEndian2Int(substr($METAdataBlockData, $offset, 1)) & 0x80); |
| 312 | + $offset += 1; |
| 313 | + |
| 314 | + $offset += 258; // reserved |
| 315 | + |
| 316 | + $ThisFileInfo['flac']['CUESHEET']['number_tracks'] = getid3_lib::BigEndian2Int(substr($METAdataBlockData, $offset, 1)); |
| 317 | + $offset += 1; |
| 318 | + |
| 319 | + for ($track = 0; $track < $ThisFileInfo['flac']['CUESHEET']['number_tracks']; $track++) { |
| 320 | + $TrackSampleOffset = getid3_lib::BigEndian2Int(substr($METAdataBlockData, $offset, 8)); |
| 321 | + $offset += 8; |
| 322 | + $TrackNumber = getid3_lib::BigEndian2Int(substr($METAdataBlockData, $offset, 1)); |
| 323 | + $offset += 1; |
| 324 | + |
| 325 | + $ThisFileInfo['flac']['CUESHEET']['tracks'][$TrackNumber]['sample_offset'] = $TrackSampleOffset; |
| 326 | + |
| 327 | + $ThisFileInfo['flac']['CUESHEET']['tracks'][$TrackNumber]['isrc'] = substr($METAdataBlockData, $offset, 12); |
| 328 | + $offset += 12; |
| 329 | + |
| 330 | + $TrackFlagsRaw = getid3_lib::BigEndian2Int(substr($METAdataBlockData, $offset, 1)); |
| 331 | + $offset += 1; |
| 332 | + $ThisFileInfo['flac']['CUESHEET']['tracks'][$TrackNumber]['flags']['is_audio'] = (bool) ($TrackFlagsRaw & 0x80); |
| 333 | + $ThisFileInfo['flac']['CUESHEET']['tracks'][$TrackNumber]['flags']['pre_emphasis'] = (bool) ($TrackFlagsRaw & 0x40); |
| 334 | + |
| 335 | + $offset += 13; // reserved |
| 336 | + |
| 337 | + $ThisFileInfo['flac']['CUESHEET']['tracks'][$TrackNumber]['index_points'] = getid3_lib::BigEndian2Int(substr($METAdataBlockData, $offset, 1)); |
| 338 | + $offset += 1; |
| 339 | + |
| 340 | + for ($index = 0; $index < $ThisFileInfo['flac']['CUESHEET']['tracks'][$TrackNumber]['index_points']; $index++) { |
| 341 | + $IndexSampleOffset = getid3_lib::BigEndian2Int(substr($METAdataBlockData, $offset, 8)); |
| 342 | + $offset += 8; |
| 343 | + $IndexNumber = getid3_lib::BigEndian2Int(substr($METAdataBlockData, $offset, 1)); |
| 344 | + $offset += 1; |
| 345 | + |
| 346 | + $offset += 3; // reserved |
| 347 | + |
| 348 | + $ThisFileInfo['flac']['CUESHEET']['tracks'][$TrackNumber]['indexes'][$IndexNumber] = $IndexSampleOffset; |
| 349 | + } |
| 350 | + } |
| 351 | + |
| 352 | + unset($ThisFileInfo['flac']['CUESHEET']['raw']); |
| 353 | + |
| 354 | + return true; |
| 355 | + } |
| 356 | + |
| 357 | + |
| 358 | + static function FLACparsePICTURE($meta_data_block_data, &$ThisFileInfo) { |
| 359 | + $picture = &$ThisFileInfo['flac']['PICTURE'][sizeof($ThisFileInfo['flac']['PICTURE']) - 1]; |
| 360 | + |
| 361 | + $offset = 0; |
| 362 | + |
| 363 | + $picture['typeid'] = getid3_lib::BigEndian2Int(substr($meta_data_block_data, $offset, 4)); |
| 364 | + $picture['type'] = getid3_flac::FLACpictureTypeLookup($picture['typeid']); |
| 365 | + $offset += 4; |
| 366 | + |
| 367 | + $length = getid3_lib::BigEndian2Int(substr($meta_data_block_data, $offset, 4)); |
| 368 | + $offset += 4; |
| 369 | + |
| 370 | + $picture['mime_type'] = substr($meta_data_block_data, $offset, $length); |
| 371 | + $offset += $length; |
| 372 | + |
| 373 | + $length = getid3_lib::BigEndian2Int(substr($meta_data_block_data, $offset, 4)); |
| 374 | + $offset += 4; |
| 375 | + |
| 376 | + $picture['description'] = substr($meta_data_block_data, $offset, $length); |
| 377 | + $offset += $length; |
| 378 | + |
| 379 | + $picture['width'] = getid3_lib::BigEndian2Int(substr($meta_data_block_data, $offset, 4)); |
| 380 | + $offset += 4; |
| 381 | + |
| 382 | + $picture['height'] = getid3_lib::BigEndian2Int(substr($meta_data_block_data, $offset, 4)); |
| 383 | + $offset += 4; |
| 384 | + |
| 385 | + $picture['color_depth'] = getid3_lib::BigEndian2Int(substr($meta_data_block_data, $offset, 4)); |
| 386 | + $offset += 4; |
| 387 | + |
| 388 | + $picture['colors_indexed'] = getid3_lib::BigEndian2Int(substr($meta_data_block_data, $offset, 4)); |
| 389 | + $offset += 4; |
| 390 | + |
| 391 | + $length = getid3_lib::BigEndian2Int(substr($meta_data_block_data, $offset, 4)); |
| 392 | + $offset += 4; |
| 393 | + |
| 394 | + $picture['image_data'] = substr($meta_data_block_data, $offset, $length); |
| 395 | + $offset += $length; |
| 396 | + $picture['data_length'] = strlen($picture['image_data']); |
| 397 | + |
| 398 | + unset($ThisFileInfo['flac']['PICTURE']['raw']); |
| 399 | + |
| 400 | + return true; |
| 401 | + } |
| 402 | +} |
| 403 | + |
| 404 | +?> |
\ No newline at end of file |
Index: trunk/extensions/TimedMediaHandler/getid3/license.txt |
— | — | @@ -0,0 +1,340 @@ |
| 2 | + GNU GENERAL PUBLIC LICENSE |
| 3 | + Version 2, June 1991 |
| 4 | + |
| 5 | + Copyright (C) 1989, 1991 Free Software Foundation, Inc. |
| 6 | + 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
| 7 | + Everyone is permitted to copy and distribute verbatim copies |
| 8 | + of this license document, but changing it is not allowed. |
| 9 | + |
| 10 | + Preamble |
| 11 | + |
| 12 | + The licenses for most software are designed to take away your |
| 13 | +freedom to share and change it. By contrast, the GNU General Public |
| 14 | +License is intended to guarantee your freedom to share and change free |
| 15 | +software--to make sure the software is free for all its users. This |
| 16 | +General Public License applies to most of the Free Software |
| 17 | +Foundation's software and to any other program whose authors commit to |
| 18 | +using it. (Some other Free Software Foundation software is covered by |
| 19 | +the GNU Library General Public License instead.) You can apply it to |
| 20 | +your programs, too. |
| 21 | + |
| 22 | + When we speak of free software, we are referring to freedom, not |
| 23 | +price. Our General Public Licenses are designed to make sure that you |
| 24 | +have the freedom to distribute copies of free software (and charge for |
| 25 | +this service if you wish), that you receive source code or can get it |
| 26 | +if you want it, that you can change the software or use pieces of it |
| 27 | +in new free programs; and that you know you can do these things. |
| 28 | + |
| 29 | + To protect your rights, we need to make restrictions that forbid |
| 30 | +anyone to deny you these rights or to ask you to surrender the rights. |
| 31 | +These restrictions translate to certain responsibilities for you if you |
| 32 | +distribute copies of the software, or if you modify it. |
| 33 | + |
| 34 | + For example, if you distribute copies of such a program, whether |
| 35 | +gratis or for a fee, you must give the recipients all the rights that |
| 36 | +you have. You must make sure that they, too, receive or can get the |
| 37 | +source code. And you must show them these terms so they know their |
| 38 | +rights. |
| 39 | + |
| 40 | + We protect your rights with two steps: (1) copyright the software, and |
| 41 | +(2) offer you this license which gives you legal permission to copy, |
| 42 | +distribute and/or modify the software. |
| 43 | + |
| 44 | + Also, for each author's protection and ours, we want to make certain |
| 45 | +that everyone understands that there is no warranty for this free |
| 46 | +software. If the software is modified by someone else and passed on, we |
| 47 | +want its recipients to know that what they have is not the original, so |
| 48 | +that any problems introduced by others will not reflect on the original |
| 49 | +authors' reputations. |
| 50 | + |
| 51 | + Finally, any free program is threatened constantly by software |
| 52 | +patents. We wish to avoid the danger that redistributors of a free |
| 53 | +program will individually obtain patent licenses, in effect making the |
| 54 | +program proprietary. To prevent this, we have made it clear that any |
| 55 | +patent must be licensed for everyone's free use or not licensed at all. |
| 56 | + |
| 57 | + The precise terms and conditions for copying, distribution and |
| 58 | +modification follow. |
| 59 | + |
| 60 | + GNU GENERAL PUBLIC LICENSE |
| 61 | + TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION |
| 62 | + |
| 63 | + 0. This License applies to any program or other work which contains |
| 64 | +a notice placed by the copyright holder saying it may be distributed |
| 65 | +under the terms of this General Public License. The "Program", below, |
| 66 | +refers to any such program or work, and a "work based on the Program" |
| 67 | +means either the Program or any derivative work under copyright law: |
| 68 | +that is to say, a work containing the Program or a portion of it, |
| 69 | +either verbatim or with modifications and/or translated into another |
| 70 | +language. (Hereinafter, translation is included without limitation in |
| 71 | +the term "modification".) Each licensee is addressed as "you". |
| 72 | + |
| 73 | +Activities other than copying, distribution and modification are not |
| 74 | +covered by this License; they are outside its scope. The act of |
| 75 | +running the Program is not restricted, and the output from the Program |
| 76 | +is covered only if its contents constitute a work based on the |
| 77 | +Program (independent of having been made by running the Program). |
| 78 | +Whether that is true depends on what the Program does. |
| 79 | + |
| 80 | + 1. You may copy and distribute verbatim copies of the Program's |
| 81 | +source code as you receive it, in any medium, provided that you |
| 82 | +conspicuously and appropriately publish on each copy an appropriate |
| 83 | +copyright notice and disclaimer of warranty; keep intact all the |
| 84 | +notices that refer to this License and to the absence of any warranty; |
| 85 | +and give any other recipients of the Program a copy of this License |
| 86 | +along with the Program. |
| 87 | + |
| 88 | +You may charge a fee for the physical act of transferring a copy, and |
| 89 | +you may at your option offer warranty protection in exchange for a fee. |
| 90 | + |
| 91 | + 2. You may modify your copy or copies of the Program or any portion |
| 92 | +of it, thus forming a work based on the Program, and copy and |
| 93 | +distribute such modifications or work under the terms of Section 1 |
| 94 | +above, provided that you also meet all of these conditions: |
| 95 | + |
| 96 | + a) You must cause the modified files to carry prominent notices |
| 97 | + stating that you changed the files and the date of any change. |
| 98 | + |
| 99 | + b) You must cause any work that you distribute or publish, that in |
| 100 | + whole or in part contains or is derived from the Program or any |
| 101 | + part thereof, to be licensed as a whole at no charge to all third |
| 102 | + parties under the terms of this License. |
| 103 | + |
| 104 | + c) If the modified program normally reads commands interactively |
| 105 | + when run, you must cause it, when started running for such |
| 106 | + interactive use in the most ordinary way, to print or display an |
| 107 | + announcement including an appropriate copyright notice and a |
| 108 | + notice that there is no warranty (or else, saying that you provide |
| 109 | + a warranty) and that users may redistribute the program under |
| 110 | + these conditions, and telling the user how to view a copy of this |
| 111 | + License. (Exception: if the Program itself is interactive but |
| 112 | + does not normally print such an announcement, your work based on |
| 113 | + the Program is not required to print an announcement.) |
| 114 | + |
| 115 | +These requirements apply to the modified work as a whole. If |
| 116 | +identifiable sections of that work are not derived from the Program, |
| 117 | +and can be reasonably considered independent and separate works in |
| 118 | +themselves, then this License, and its terms, do not apply to those |
| 119 | +sections when you distribute them as separate works. But when you |
| 120 | +distribute the same sections as part of a whole which is a work based |
| 121 | +on the Program, the distribution of the whole must be on the terms of |
| 122 | +this License, whose permissions for other licensees extend to the |
| 123 | +entire whole, and thus to each and every part regardless of who wrote it. |
| 124 | + |
| 125 | +Thus, it is not the intent of this section to claim rights or contest |
| 126 | +your rights to work written entirely by you; rather, the intent is to |
| 127 | +exercise the right to control the distribution of derivative or |
| 128 | +collective works based on the Program. |
| 129 | + |
| 130 | +In addition, mere aggregation of another work not based on the Program |
| 131 | +with the Program (or with a work based on the Program) on a volume of |
| 132 | +a storage or distribution medium does not bring the other work under |
| 133 | +the scope of this License. |
| 134 | + |
| 135 | + 3. You may copy and distribute the Program (or a work based on it, |
| 136 | +under Section 2) in object code or executable form under the terms of |
| 137 | +Sections 1 and 2 above provided that you also do one of the following: |
| 138 | + |
| 139 | + a) Accompany it with the complete corresponding machine-readable |
| 140 | + source code, which must be distributed under the terms of Sections |
| 141 | + 1 and 2 above on a medium customarily used for software interchange; or, |
| 142 | + |
| 143 | + b) Accompany it with a written offer, valid for at least three |
| 144 | + years, to give any third party, for a charge no more than your |
| 145 | + cost of physically performing source distribution, a complete |
| 146 | + machine-readable copy of the corresponding source code, to be |
| 147 | + distributed under the terms of Sections 1 and 2 above on a medium |
| 148 | + customarily used for software interchange; or, |
| 149 | + |
| 150 | + c) Accompany it with the information you received as to the offer |
| 151 | + to distribute corresponding source code. (This alternative is |
| 152 | + allowed only for noncommercial distribution and only if you |
| 153 | + received the program in object code or executable form with such |
| 154 | + an offer, in accord with Subsection b above.) |
| 155 | + |
| 156 | +The source code for a work means the preferred form of the work for |
| 157 | +making modifications to it. For an executable work, complete source |
| 158 | +code means all the source code for all modules it contains, plus any |
| 159 | +associated interface definition files, plus the scripts used to |
| 160 | +control compilation and installation of the executable. However, as a |
| 161 | +special exception, the source code distributed need not include |
| 162 | +anything that is normally distributed (in either source or binary |
| 163 | +form) with the major components (compiler, kernel, and so on) of the |
| 164 | +operating system on which the executable runs, unless that component |
| 165 | +itself accompanies the executable. |
| 166 | + |
| 167 | +If distribution of executable or object code is made by offering |
| 168 | +access to copy from a designated place, then offering equivalent |
| 169 | +access to copy the source code from the same place counts as |
| 170 | +distribution of the source code, even though third parties are not |
| 171 | +compelled to copy the source along with the object code. |
| 172 | + |
| 173 | + 4. You may not copy, modify, sublicense, or distribute the Program |
| 174 | +except as expressly provided under this License. Any attempt |
| 175 | +otherwise to copy, modify, sublicense or distribute the Program is |
| 176 | +void, and will automatically terminate your rights under this License. |
| 177 | +However, parties who have received copies, or rights, from you under |
| 178 | +this License will not have their licenses terminated so long as such |
| 179 | +parties remain in full compliance. |
| 180 | + |
| 181 | + 5. You are not required to accept this License, since you have not |
| 182 | +signed it. However, nothing else grants you permission to modify or |
| 183 | +distribute the Program or its derivative works. These actions are |
| 184 | +prohibited by law if you do not accept this License. Therefore, by |
| 185 | +modifying or distributing the Program (or any work based on the |
| 186 | +Program), you indicate your acceptance of this License to do so, and |
| 187 | +all its terms and conditions for copying, distributing or modifying |
| 188 | +the Program or works based on it. |
| 189 | + |
| 190 | + 6. Each time you redistribute the Program (or any work based on the |
| 191 | +Program), the recipient automatically receives a license from the |
| 192 | +original licensor to copy, distribute or modify the Program subject to |
| 193 | +these terms and conditions. You may not impose any further |
| 194 | +restrictions on the recipients' exercise of the rights granted herein. |
| 195 | +You are not responsible for enforcing compliance by third parties to |
| 196 | +this License. |
| 197 | + |
| 198 | + 7. If, as a consequence of a court judgment or allegation of patent |
| 199 | +infringement or for any other reason (not limited to patent issues), |
| 200 | +conditions are imposed on you (whether by court order, agreement or |
| 201 | +otherwise) that contradict the conditions of this License, they do not |
| 202 | +excuse you from the conditions of this License. If you cannot |
| 203 | +distribute so as to satisfy simultaneously your obligations under this |
| 204 | +License and any other pertinent obligations, then as a consequence you |
| 205 | +may not distribute the Program at all. For example, if a patent |
| 206 | +license would not permit royalty-free redistribution of the Program by |
| 207 | +all those who receive copies directly or indirectly through you, then |
| 208 | +the only way you could satisfy both it and this License would be to |
| 209 | +refrain entirely from distribution of the Program. |
| 210 | + |
| 211 | +If any portion of this section is held invalid or unenforceable under |
| 212 | +any particular circumstance, the balance of the section is intended to |
| 213 | +apply and the section as a whole is intended to apply in other |
| 214 | +circumstances. |
| 215 | + |
| 216 | +It is not the purpose of this section to induce you to infringe any |
| 217 | +patents or other property right claims or to contest validity of any |
| 218 | +such claims; this section has the sole purpose of protecting the |
| 219 | +integrity of the free software distribution system, which is |
| 220 | +implemented by public license practices. Many people have made |
| 221 | +generous contributions to the wide range of software distributed |
| 222 | +through that system in reliance on consistent application of that |
| 223 | +system; it is up to the author/donor to decide if he or she is willing |
| 224 | +to distribute software through any other system and a licensee cannot |
| 225 | +impose that choice. |
| 226 | + |
| 227 | +This section is intended to make thoroughly clear what is believed to |
| 228 | +be a consequence of the rest of this License. |
| 229 | + |
| 230 | + 8. If the distribution and/or use of the Program is restricted in |
| 231 | +certain countries either by patents or by copyrighted interfaces, the |
| 232 | +original copyright holder who places the Program under this License |
| 233 | +may add an explicit geographical distribution limitation excluding |
| 234 | +those countries, so that distribution is permitted only in or among |
| 235 | +countries not thus excluded. In such case, this License incorporates |
| 236 | +the limitation as if written in the body of this License. |
| 237 | + |
| 238 | + 9. The Free Software Foundation may publish revised and/or new versions |
| 239 | +of the General Public License from time to time. Such new versions will |
| 240 | +be similar in spirit to the present version, but may differ in detail to |
| 241 | +address new problems or concerns. |
| 242 | + |
| 243 | +Each version is given a distinguishing version number. If the Program |
| 244 | +specifies a version number of this License which applies to it and "any |
| 245 | +later version", you have the option of following the terms and conditions |
| 246 | +either of that version or of any later version published by the Free |
| 247 | +Software Foundation. If the Program does not specify a version number of |
| 248 | +this License, you may choose any version ever published by the Free Software |
| 249 | +Foundation. |
| 250 | + |
| 251 | + 10. If you wish to incorporate parts of the Program into other free |
| 252 | +programs whose distribution conditions are different, write to the author |
| 253 | +to ask for permission. For software which is copyrighted by the Free |
| 254 | +Software Foundation, write to the Free Software Foundation; we sometimes |
| 255 | +make exceptions for this. Our decision will be guided by the two goals |
| 256 | +of preserving the free status of all derivatives of our free software and |
| 257 | +of promoting the sharing and reuse of software generally. |
| 258 | + |
| 259 | + NO WARRANTY |
| 260 | + |
| 261 | + 11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY |
| 262 | +FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN |
| 263 | +OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES |
| 264 | +PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED |
| 265 | +OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF |
| 266 | +MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS |
| 267 | +TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE |
| 268 | +PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, |
| 269 | +REPAIR OR CORRECTION. |
| 270 | + |
| 271 | + 12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING |
| 272 | +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR |
| 273 | +REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, |
| 274 | +INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING |
| 275 | +OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED |
| 276 | +TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY |
| 277 | +YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER |
| 278 | +PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE |
| 279 | +POSSIBILITY OF SUCH DAMAGES. |
| 280 | + |
| 281 | + END OF TERMS AND CONDITIONS |
| 282 | + |
| 283 | + How to Apply These Terms to Your New Programs |
| 284 | + |
| 285 | + If you develop a new program, and you want it to be of the greatest |
| 286 | +possible use to the public, the best way to achieve this is to make it |
| 287 | +free software which everyone can redistribute and change under these terms. |
| 288 | + |
| 289 | + To do so, attach the following notices to the program. It is safest |
| 290 | +to attach them to the start of each source file to most effectively |
| 291 | +convey the exclusion of warranty; and each file should have at least |
| 292 | +the "copyright" line and a pointer to where the full notice is found. |
| 293 | + |
| 294 | + <one line to give the program's name and a brief idea of what it does.> |
| 295 | + Copyright (C) <year> <name of author> |
| 296 | + |
| 297 | + This program is free software; you can redistribute it and/or modify |
| 298 | + it under the terms of the GNU General Public License as published by |
| 299 | + the Free Software Foundation; either version 2 of the License, or |
| 300 | + (at your option) any later version. |
| 301 | + |
| 302 | + This program is distributed in the hope that it will be useful, |
| 303 | + but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 304 | + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| 305 | + GNU General Public License for more details. |
| 306 | + |
| 307 | + You should have received a copy of the GNU General Public License |
| 308 | + along with this program; if not, write to the Free Software |
| 309 | + Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
| 310 | + |
| 311 | + |
| 312 | +Also add information on how to contact you by electronic and paper mail. |
| 313 | + |
| 314 | +If the program is interactive, make it output a short notice like this |
| 315 | +when it starts in an interactive mode: |
| 316 | + |
| 317 | + Gnomovision version 69, Copyright (C) year name of author |
| 318 | + Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'. |
| 319 | + This is free software, and you are welcome to redistribute it |
| 320 | + under certain conditions; type `show c' for details. |
| 321 | + |
| 322 | +The hypothetical commands `show w' and `show c' should show the appropriate |
| 323 | +parts of the General Public License. Of course, the commands you use may |
| 324 | +be called something other than `show w' and `show c'; they could even be |
| 325 | +mouse-clicks or menu items--whatever suits your program. |
| 326 | + |
| 327 | +You should also get your employer (if you work as a programmer) or your |
| 328 | +school, if any, to sign a "copyright disclaimer" for the program, if |
| 329 | +necessary. Here is a sample; alter the names: |
| 330 | + |
| 331 | + Yoyodyne, Inc., hereby disclaims all copyright interest in the program |
| 332 | + `Gnomovision' (which makes passes at compilers) written by James Hacker. |
| 333 | + |
| 334 | + <signature of Ty Coon>, 1 April 1989 |
| 335 | + Ty Coon, President of Vice |
| 336 | + |
| 337 | +This General Public License does not permit incorporating your program into |
| 338 | +proprietary programs. If your program is a subroutine library, you may |
| 339 | +consider it more useful to permit linking proprietary applications with the |
| 340 | +library. If this is what you want to do, use the GNU Library General |
| 341 | +Public License instead of this License. |
Property changes on: trunk/extensions/TimedMediaHandler/getid3/license.txt |
___________________________________________________________________ |
Added: svn:mime-type |
1 | 342 | + text/plain |
Index: trunk/extensions/TimedMediaHandler/getid3/getid3.php |
— | — | @@ -0,0 +1,1501 @@ |
| 2 | +<?php |
| 3 | +///////////////////////////////////////////////////////////////// |
| 4 | +/// getID3() by James Heinrich <info@getid3.org> // |
| 5 | +// available at http://getid3.sourceforge.net // |
| 6 | +// or http://www.getid3.org // |
| 7 | +///////////////////////////////////////////////////////////////// |
| 8 | +// // |
| 9 | +// Please see readme.txt for more information // |
| 10 | +// /// |
| 11 | +///////////////////////////////////////////////////////////////// |
| 12 | + |
| 13 | +// Defines |
| 14 | +define('GETID3_VERSION', '1.8.4-20110203'); |
| 15 | +define('GETID3_FREAD_BUFFER_SIZE', 16384); // read buffer size in bytes |
| 16 | + |
| 17 | + |
| 18 | +// attempt to define temp dir as something flexible but reliable |
| 19 | +$temp_dir = ini_get('upload_tmp_dir'); |
| 20 | +if ($temp_dir && (!is_dir($temp_dir) || !is_readable($temp_dir))) { |
| 21 | + $temp_dir = ''; |
| 22 | +} |
| 23 | +if (!$temp_dir && function_exists('sys_get_temp_dir')) { |
| 24 | + // PHP v5.2.1+ |
| 25 | + // sys_get_temp_dir() may give inaccessible temp dir, e.g. with open_basedir on virtual hosts |
| 26 | + $temp_dir = sys_get_temp_dir(); |
| 27 | +} |
| 28 | +$temp_dir = realpath($temp_dir); |
| 29 | +$open_basedir = ini_get('open_basedir'); |
| 30 | +if ($open_basedir) { |
| 31 | + // e.g. "/var/www/vhosts/getid3.org/httpdocs/:/tmp/" |
| 32 | + $temp_dir = str_replace(array('/', '\\'), DIRECTORY_SEPARATOR, $temp_dir); |
| 33 | + $open_basedir = str_replace(array('/', '\\'), DIRECTORY_SEPARATOR, $open_basedir); |
| 34 | + if (substr($temp_dir, -1, 1) != DIRECTORY_SEPARATOR) { |
| 35 | + $temp_dir .= DIRECTORY_SEPARATOR; |
| 36 | + } |
| 37 | + $found_valid_tempdir = false; |
| 38 | + $open_basedirs = explode(':', $open_basedir); |
| 39 | + foreach ($open_basedirs as $basedir) { |
| 40 | + if (substr($basedir, -1, 1) != DIRECTORY_SEPARATOR) { |
| 41 | + $basedir .= DIRECTORY_SEPARATOR; |
| 42 | + } |
| 43 | + if (preg_match('#^'.preg_quote($basedir).'#', $temp_dir)) { |
| 44 | + $found_valid_tempdir = true; |
| 45 | + break; |
| 46 | + } |
| 47 | + } |
| 48 | + if (!$found_valid_tempdir) { |
| 49 | + $temp_dir = ''; |
| 50 | + } |
| 51 | + unset($open_basedirs, $found_valid_tempdir, $basedir); |
| 52 | +} |
| 53 | +if (!$temp_dir) { |
| 54 | + $temp_dir = '*'; // invalid directory name should force tempnam() to use system default temp dir |
| 55 | +} |
| 56 | +// $temp_dir = '/something/else/'; // feel free to override temp dir here if it works better for your system |
| 57 | +define('GETID3_TEMP_DIR', $temp_dir); |
| 58 | +unset($open_basedir, $temp_dir); |
| 59 | + |
| 60 | + |
| 61 | + |
| 62 | +class getID3 |
| 63 | +{ |
| 64 | + // public: Settings |
| 65 | + var $encoding = 'ISO-8859-1'; // CASE SENSITIVE! - i.e. (must be supported by iconv()) |
| 66 | + // Examples: ISO-8859-1 UTF-8 UTF-16 UTF-16BE |
| 67 | + |
| 68 | + var $encoding_id3v1 = 'ISO-8859-1'; // Should always be 'ISO-8859-1', but some tags may be written in other encodings such as 'EUC-CN' |
| 69 | + |
| 70 | + var $tempdir = GETID3_TEMP_DIR; |
| 71 | + |
| 72 | + // public: Optional tag checks - disable for speed. |
| 73 | + var $option_tag_id3v1 = true; // Read and process ID3v1 tags |
| 74 | + var $option_tag_id3v2 = true; // Read and process ID3v2 tags |
| 75 | + var $option_tag_lyrics3 = true; // Read and process Lyrics3 tags |
| 76 | + var $option_tag_apetag = true; // Read and process APE tags |
| 77 | + var $option_tags_process = true; // Copy tags to root key 'tags' and encode to $this->encoding |
| 78 | + var $option_tags_html = true; // Copy tags to root key 'tags_html' properly translated from various encodings to HTML entities |
| 79 | + |
| 80 | + // public: Optional tag/comment calucations |
| 81 | + var $option_extra_info = true; // Calculate additional info such as bitrate, channelmode etc |
| 82 | + |
| 83 | + // public: Optional calculations |
| 84 | + var $option_md5_data = false; // Get MD5 sum of data part - slow |
| 85 | + var $option_md5_data_source = false; // Use MD5 of source file if availble - only FLAC and OptimFROG |
| 86 | + var $option_sha1_data = false; // Get SHA1 sum of data part - slow |
| 87 | + var $option_max_2gb_check = true; // Check whether file is larger than 2 Gb and thus not supported by PHP |
| 88 | + |
| 89 | + // private |
| 90 | + var $filename; |
| 91 | + |
| 92 | + |
| 93 | + // public: constructor |
| 94 | + function getID3() |
| 95 | + { |
| 96 | + |
| 97 | + $this->startup_error = ''; |
| 98 | + $this->startup_warning = ''; |
| 99 | + |
| 100 | + // Check for PHP version >= 4.2.0 |
| 101 | + if (PHP_VERSION < '4.2.0') { // version_compare not available before PHP v4.1.0, do not use for initial version check |
| 102 | + $this->startup_error .= 'getID3() requires PHP v4.2.0 or higher - you are running v'.PHP_VERSION; |
| 103 | + } |
| 104 | + if (version_compare(PHP_VERSION, '5.0.0', '<')) { |
| 105 | + $this->startup_warning .= 'getID3() v1.8+ recommends PHP v5.0.0 or higher - you are running v'.PHP_VERSION; |
| 106 | + } |
| 107 | + |
| 108 | + // Check memory |
| 109 | + $memory_limit = ini_get('memory_limit'); |
| 110 | + if (preg_match('#([0-9]+)M#i', $memory_limit, $matches)) { |
| 111 | + // could be stored as "16M" rather than 16777216 for example |
| 112 | + $memory_limit = $matches[1] * 1048576; |
| 113 | + } elseif (preg_match('#([0-9]+)G#i', $memory_limit, $matches)) { // The 'G' modifier is available since PHP 5.1.0 |
| 114 | + // could be stored as "2G" rather than 2147483648 for example |
| 115 | + $memory_limit = $matches[1] * 1073741824; |
| 116 | + } |
| 117 | + if ($memory_limit <= 0) { |
| 118 | + // memory limits probably disabled |
| 119 | + } elseif ($memory_limit <= 3145728) { |
| 120 | + $this->startup_error .= 'PHP has less than 3MB available memory and will very likely run out. Increase memory_limit in php.ini'; |
| 121 | + } elseif ($memory_limit <= 12582912) { |
| 122 | + $this->startup_warning .= 'PHP has less than 12MB available memory and might run out if all modules are loaded. Increase memory_limit in php.ini'; |
| 123 | + } |
| 124 | + |
| 125 | + // Check safe_mode off |
| 126 | + if (preg_match('#(1|ON)#i', ini_get('safe_mode'))) { |
| 127 | + $this->warning('WARNING: Safe mode is on, shorten support disabled, md5data/sha1data for ogg vorbis disabled, ogg vorbos/flac tag writing disabled.'); |
| 128 | + } |
| 129 | + |
| 130 | + if (intval(ini_get('mbstring.func_overload')) > 0) { |
| 131 | + $this->warning('WARNING: php.ini contains "mbstring.func_overload = '.ini_get('mbstring.func_overload').'", this may break things.'); |
| 132 | + } |
| 133 | + |
| 134 | + // Check timezone config setting |
| 135 | + if (!ini_get('date.timezone')) { |
| 136 | + if (function_exists('date_default_timezone_set')) { |
| 137 | + $this->warning('php.ini should have "date.timezone" set, but it does not. Setting timezone to "America/New_York"'); |
| 138 | + date_default_timezone_set('America/New_York'); |
| 139 | + } else { |
| 140 | + $this->warning('php.ini should have "date.timezone" set, but it does not.'); |
| 141 | + } |
| 142 | + } |
| 143 | + |
| 144 | + // Check for magic_quotes_runtime |
| 145 | + if (function_exists('get_magic_quotes_runtime')) { |
| 146 | + if (get_magic_quotes_runtime()) { |
| 147 | + return $this->startup_error('magic_quotes_runtime must be disabled before running getID3(). Surround getid3 block by set_magic_quotes_runtime(0) and set_magic_quotes_runtime(1).'); |
| 148 | + } |
| 149 | + } |
| 150 | + |
| 151 | + // Check for magic_quotes_gpc |
| 152 | + if (function_exists('magic_quotes_gpc')) { |
| 153 | + if (get_magic_quotes_gpc()) { |
| 154 | + return $this->startup_error('magic_quotes_gpc must be disabled before running getID3(). Surround getid3 block by set_magic_quotes_gpc(0) and set_magic_quotes_gpc(1).'); |
| 155 | + } |
| 156 | + } |
| 157 | + |
| 158 | + // define a constant rather than looking up every time it is needed |
| 159 | + if (!defined('GETID3_OS_ISWINDOWS')) { |
| 160 | + if (strtoupper(substr(PHP_OS, 0, 3)) == 'WIN') { |
| 161 | + define('GETID3_OS_ISWINDOWS', true); |
| 162 | + } else { |
| 163 | + define('GETID3_OS_ISWINDOWS', false); |
| 164 | + } |
| 165 | + } |
| 166 | + |
| 167 | + // Get base path of getID3() - ONCE |
| 168 | + if (!defined('GETID3_INCLUDEPATH')) { |
| 169 | + foreach (get_included_files() as $key => $val) { |
| 170 | + if (basename($val) == 'getid3.php') { |
| 171 | + define('GETID3_INCLUDEPATH', dirname($val).DIRECTORY_SEPARATOR); |
| 172 | + break; |
| 173 | + } |
| 174 | + } |
| 175 | + } |
| 176 | + |
| 177 | + // Load support library |
| 178 | + if (!include_once(GETID3_INCLUDEPATH.'getid3.lib.php')) { |
| 179 | + $this->startup_error .= 'getid3.lib.php is missing or corrupt'; |
| 180 | + } |
| 181 | + |
| 182 | + |
| 183 | + // Needed for Windows only: |
| 184 | + // Define locations of helper applications for Shorten, VorbisComment, MetaFLAC |
| 185 | + // as well as other helper functions such as head, tail, md5sum, etc |
| 186 | + // This path cannot contain spaces, but the below code will attempt to get the |
| 187 | + // 8.3-equivalent path automatically |
| 188 | + // IMPORTANT: This path must include the trailing slash |
| 189 | + if (GETID3_OS_ISWINDOWS && !defined('GETID3_HELPERAPPSDIR')) { |
| 190 | + |
| 191 | + $helperappsdir = GETID3_INCLUDEPATH.'..'.DIRECTORY_SEPARATOR.'helperapps'; // must not have any space in this path |
| 192 | + |
| 193 | + if (!is_dir($helperappsdir)) { |
| 194 | + $this->startup_error .= '"'.$helperappsdir.'" cannot be defined as GETID3_HELPERAPPSDIR because it does not exist'; |
| 195 | + } elseif (strpos(realpath($helperappsdir), ' ') !== false) { |
| 196 | + $DirPieces = explode(DIRECTORY_SEPARATOR, realpath($helperappsdir)); |
| 197 | + $path_so_far = array(); |
| 198 | + foreach ($DirPieces as $key => $value) { |
| 199 | + if (strpos($value, ' ') !== false) { |
| 200 | + if (!empty($path_so_far)) { |
| 201 | + $commandline = 'dir /x '.escapeshellarg(implode(DIRECTORY_SEPARATOR, $path_so_far)); |
| 202 | + $dir_listing = `$commandline`; |
| 203 | + $lines = explode("\n", $dir_listing); |
| 204 | + foreach ($lines as $line) { |
| 205 | + $line = trim($line); |
| 206 | + if (preg_match('#^([0-9/]{10}) +([0-9:]{4,5}( [AP]M)?) +(<DIR>|[0-9,]+) +([^ ]{0,11}) +(.+)$#', $line, $matches)) { |
| 207 | + list($dummy, $date, $time, $ampm, $filesize, $shortname, $filename) = $matches; |
| 208 | + if ((strtoupper($filesize) == '<DIR>') && (strtolower($filename) == strtolower($value))) { |
| 209 | + $value = $shortname; |
| 210 | + } |
| 211 | + } |
| 212 | + } |
| 213 | + } else { |
| 214 | + $this->startup_error .= 'GETID3_HELPERAPPSDIR must not have any spaces in it - use 8dot3 naming convention if neccesary. You can run "dir /x" from the commandline to see the correct 8.3-style names.'; |
| 215 | + } |
| 216 | + } |
| 217 | + $path_so_far[] = $value; |
| 218 | + } |
| 219 | + $helperappsdir = implode(DIRECTORY_SEPARATOR, $path_so_far); |
| 220 | + } |
| 221 | + define('GETID3_HELPERAPPSDIR', $helperappsdir.DIRECTORY_SEPARATOR); |
| 222 | + } |
| 223 | + |
| 224 | + } |
| 225 | + |
| 226 | + |
| 227 | + // public: setOption |
| 228 | + function setOption($optArray) { |
| 229 | + if (!is_array($optArray) || empty($optArray)) { |
| 230 | + return false; |
| 231 | + } |
| 232 | + foreach ($optArray as $opt => $val) { |
| 233 | + //if (isset($this, $opt) === false) { |
| 234 | + if (isset($this->$opt) === false) { |
| 235 | + continue; |
| 236 | + } |
| 237 | + $this->$opt = $val; |
| 238 | + } |
| 239 | + return true; |
| 240 | + } |
| 241 | + |
| 242 | + |
| 243 | + // public: analyze file - replaces GetAllFileInfo() and GetTagOnly() |
| 244 | + function analyze($filename) { |
| 245 | + if (!empty($this->startup_error)) { |
| 246 | + return $this->error($this->startup_error); |
| 247 | + } |
| 248 | + if (!empty($this->startup_warning)) { |
| 249 | + $this->warning($this->startup_warning); |
| 250 | + } |
| 251 | + |
| 252 | + // init result array and set parameters |
| 253 | + $this->info = array(); |
| 254 | + $this->info['GETID3_VERSION'] = GETID3_VERSION; |
| 255 | + |
| 256 | + // Check encoding/iconv support |
| 257 | + if (!function_exists('iconv') && !in_array($this->encoding, array('ISO-8859-1', 'UTF-8', 'UTF-16LE', 'UTF-16BE', 'UTF-16'))) { |
| 258 | + $errormessage = 'iconv() support is needed for encodings other than ISO-8859-1, UTF-8, UTF-16LE, UTF16-BE, UTF-16. '; |
| 259 | + if (GETID3_OS_ISWINDOWS) { |
| 260 | + $errormessage .= 'PHP does not have iconv() support. Please enable php_iconv.dll in php.ini, and copy iconv.dll from c:/php/dlls to c:/windows/system32'; |
| 261 | + } else { |
| 262 | + $errormessage .= 'PHP is not compiled with iconv() support. Please recompile with the --with-iconv switch'; |
| 263 | + } |
| 264 | + return $this->error($errormessage); |
| 265 | + } |
| 266 | + |
| 267 | + // remote files not supported |
| 268 | + if (preg_match('/^(ht|f)tp:\/\//', $filename)) { |
| 269 | + return $this->error('Remote files are not supported in this version of getID3() - please copy the file locally first'); |
| 270 | + } |
| 271 | + |
| 272 | + $filename = str_replace('/', DIRECTORY_SEPARATOR, $filename); |
| 273 | + $filename = preg_replace('#(.+)'.preg_quote(DIRECTORY_SEPARATOR).'{2,}#U', '\1'.DIRECTORY_SEPARATOR, $filename); |
| 274 | + |
| 275 | + // open local file |
| 276 | + if (file_exists($filename)) { |
| 277 | + ob_start(); |
| 278 | + if ($fp = fopen($filename, 'rb')) { |
| 279 | + // great |
| 280 | + ob_end_clean(); |
| 281 | + } else { |
| 282 | + $fopen_error = ob_get_contents(); |
| 283 | + ob_end_clean(); |
| 284 | + return $this->error('Could not open file "'.$filename.'" (fopen says: '.$fopen_error.')'); |
| 285 | + } |
| 286 | + } else { |
| 287 | + return $this->error('Could not open file "'.$filename.'" (does not exist)'); |
| 288 | + } |
| 289 | + |
| 290 | + // set parameters |
| 291 | + $this->info['filesize'] = filesize($filename); |
| 292 | + |
| 293 | + // option_max_2gb_check |
| 294 | + if ($this->option_max_2gb_check) { |
| 295 | + // PHP doesn't support integers larger than 31-bit (~2GB) |
| 296 | + // filesize() simply returns (filesize % (pow(2, 32)), no matter the actual filesize |
| 297 | + // ftell() returns 0 if seeking to the end is beyond the range of unsigned integer |
| 298 | + fseek($fp, 0, SEEK_END); |
| 299 | + if ((($this->info['filesize'] != 0) && (ftell($fp) == 0)) || |
| 300 | + ($this->info['filesize'] < 0) || |
| 301 | + (ftell($fp) < 0)) { |
| 302 | + $real_filesize = false; |
| 303 | + if (GETID3_OS_ISWINDOWS) { |
| 304 | + $commandline = 'dir /-C "'.str_replace('/', DIRECTORY_SEPARATOR, $filename).'"'; |
| 305 | + $dir_output = `$commandline`; |
| 306 | + if (preg_match('#1 File\(s\)[ ]+([0-9]+) bytes#i', $dir_output, $matches)) { |
| 307 | + $real_filesize = (float) $matches[1]; |
| 308 | + } |
| 309 | + } else { |
| 310 | + $commandline = 'ls -o -g -G --time-style=long-iso '.escapeshellarg($filename); |
| 311 | + $dir_output = `$commandline`; |
| 312 | + if (preg_match('#([0-9]+) ([0-9]{4}-[0-9]{2}\-[0-9]{2} [0-9]{2}:[0-9]{2}) '.str_replace('#', '\\#', preg_quote($filename)).'$#', $dir_output, $matches)) { |
| 313 | + $real_filesize = (float) $matches[1]; |
| 314 | + } |
| 315 | + } |
| 316 | + if ($real_filesize === false) { |
| 317 | + unset($this->info['filesize']); |
| 318 | + fclose($fp); |
| 319 | + return $this->error('File is most likely larger than 2GB and is not supported by PHP'); |
| 320 | + } elseif ($real_filesize < pow(2, 31)) { |
| 321 | + unset($this->info['filesize']); |
| 322 | + fclose($fp); |
| 323 | + return $this->error('PHP seems to think the file is larger than 2GB, but filesystem reports it as '.number_format($real_filesize, 3).'GB, please report to info@getid3.org'); |
| 324 | + } |
| 325 | + $this->info['filesize'] = $real_filesize; |
| 326 | + $this->error('File is larger than 2GB (filesystem reports it as '.number_format($real_filesize, 3).'GB) and is not properly supported by PHP.'); |
| 327 | + } |
| 328 | + } |
| 329 | + |
| 330 | + // set more parameters |
| 331 | + $this->info['avdataoffset'] = 0; |
| 332 | + $this->info['avdataend'] = $this->info['filesize']; |
| 333 | + $this->info['fileformat'] = ''; // filled in later |
| 334 | + $this->info['audio']['dataformat'] = ''; // filled in later, unset if not used |
| 335 | + $this->info['video']['dataformat'] = ''; // filled in later, unset if not used |
| 336 | + $this->info['tags'] = array(); // filled in later, unset if not used |
| 337 | + $this->info['error'] = array(); // filled in later, unset if not used |
| 338 | + $this->info['warning'] = array(); // filled in later, unset if not used |
| 339 | + $this->info['comments'] = array(); // filled in later, unset if not used |
| 340 | + $this->info['encoding'] = $this->encoding; // required by id3v2 and iso modules - can be unset at the end if desired |
| 341 | + |
| 342 | + // set redundant parameters - might be needed in some include file |
| 343 | + $this->info['filename'] = basename($filename); |
| 344 | + $this->info['filepath'] = str_replace('\\', '/', realpath(dirname($filename))); |
| 345 | + $this->info['filenamepath'] = $this->info['filepath'].'/'.$this->info['filename']; |
| 346 | + |
| 347 | + |
| 348 | + // handle ID3v2 tag - done first - already at beginning of file |
| 349 | + // ID3v2 detection (even if not parsing) is always done otherwise fileformat is much harder to detect |
| 350 | + if ($this->option_tag_id3v2) { |
| 351 | + |
| 352 | + $GETID3_ERRORARRAY = &$this->info['warning']; |
| 353 | + if (getid3_lib::IncludeDependency(GETID3_INCLUDEPATH.'module.tag.id3v2.php', __FILE__, false)) { |
| 354 | + $tag = new getid3_id3v2($fp, $this->info); |
| 355 | + unset($tag); |
| 356 | + } |
| 357 | + |
| 358 | + } else { |
| 359 | + |
| 360 | + fseek($fp, 0, SEEK_SET); |
| 361 | + $header = fread($fp, 10); |
| 362 | + if ((substr($header, 0, 3) == 'ID3') && (strlen($header) == 10)) { |
| 363 | + $this->info['id3v2']['header'] = true; |
| 364 | + $this->info['id3v2']['majorversion'] = ord($header{3}); |
| 365 | + $this->info['id3v2']['minorversion'] = ord($header{4}); |
| 366 | + $this->info['id3v2']['headerlength'] = getid3_lib::BigEndian2Int(substr($header, 6, 4), 1) + 10; // length of ID3v2 tag in 10-byte header doesn't include 10-byte header length |
| 367 | + |
| 368 | + $this->info['id3v2']['tag_offset_start'] = 0; |
| 369 | + $this->info['id3v2']['tag_offset_end'] = $this->info['id3v2']['tag_offset_start'] + $this->info['id3v2']['headerlength']; |
| 370 | + $this->info['avdataoffset'] = $this->info['id3v2']['tag_offset_end']; |
| 371 | + } |
| 372 | + |
| 373 | + } |
| 374 | + |
| 375 | + |
| 376 | + // handle ID3v1 tag |
| 377 | + if ($this->option_tag_id3v1) { |
| 378 | + if (!file_exists(GETID3_INCLUDEPATH.'module.tag.id3v1.php') || !include_once(GETID3_INCLUDEPATH.'module.tag.id3v1.php')) { |
| 379 | + return $this->error('module.tag.id3v1.php is missing - you may disable option_tag_id3v1.'); |
| 380 | + } |
| 381 | + $tag = new getid3_id3v1($fp, $this->info); |
| 382 | + unset($tag); |
| 383 | + } |
| 384 | + |
| 385 | + // handle APE tag |
| 386 | + if ($this->option_tag_apetag) { |
| 387 | + if (!file_exists(GETID3_INCLUDEPATH.'module.tag.apetag.php') || !include_once(GETID3_INCLUDEPATH.'module.tag.apetag.php')) { |
| 388 | + return $this->error('module.tag.apetag.php is missing - you may disable option_tag_apetag.'); |
| 389 | + } |
| 390 | + $tag = new getid3_apetag($fp, $this->info); |
| 391 | + unset($tag); |
| 392 | + } |
| 393 | + |
| 394 | + // handle lyrics3 tag |
| 395 | + if ($this->option_tag_lyrics3) { |
| 396 | + if (!file_exists(GETID3_INCLUDEPATH.'module.tag.lyrics3.php') || !include_once(GETID3_INCLUDEPATH.'module.tag.lyrics3.php')) { |
| 397 | + return $this->error('module.tag.lyrics3.php is missing - you may disable option_tag_lyrics3.'); |
| 398 | + } |
| 399 | + $tag = new getid3_lyrics3($fp, $this->info); |
| 400 | + unset($tag); |
| 401 | + } |
| 402 | + |
| 403 | + // read 32 kb file data |
| 404 | + fseek($fp, $this->info['avdataoffset'], SEEK_SET); |
| 405 | + $formattest = fread($fp, 32774); |
| 406 | + |
| 407 | + // determine format |
| 408 | + $determined_format = $this->GetFileFormat($formattest, $filename); |
| 409 | + |
| 410 | + // unable to determine file format |
| 411 | + if (!$determined_format) { |
| 412 | + fclose($fp); |
| 413 | + return $this->error('unable to determine file format'); |
| 414 | + } |
| 415 | + |
| 416 | + // check for illegal ID3 tags |
| 417 | + if (isset($determined_format['fail_id3']) && (in_array('id3v1', $this->info['tags']) || in_array('id3v2', $this->info['tags']))) { |
| 418 | + if ($determined_format['fail_id3'] === 'ERROR') { |
| 419 | + fclose($fp); |
| 420 | + return $this->error('ID3 tags not allowed on this file type.'); |
| 421 | + } elseif ($determined_format['fail_id3'] === 'WARNING') { |
| 422 | + $this->info['warning'][] = 'ID3 tags not allowed on this file type.'; |
| 423 | + } |
| 424 | + } |
| 425 | + |
| 426 | + // check for illegal APE tags |
| 427 | + if (isset($determined_format['fail_ape']) && in_array('ape', $this->info['tags'])) { |
| 428 | + if ($determined_format['fail_ape'] === 'ERROR') { |
| 429 | + fclose($fp); |
| 430 | + return $this->error('APE tags not allowed on this file type.'); |
| 431 | + } elseif ($determined_format['fail_ape'] === 'WARNING') { |
| 432 | + $this->info['warning'][] = 'APE tags not allowed on this file type.'; |
| 433 | + } |
| 434 | + } |
| 435 | + |
| 436 | + // set mime type |
| 437 | + $this->info['mime_type'] = $determined_format['mime_type']; |
| 438 | + |
| 439 | + // supported format signature pattern detected, but module deleted |
| 440 | + if (!file_exists(GETID3_INCLUDEPATH.$determined_format['include'])) { |
| 441 | + fclose($fp); |
| 442 | + return $this->error('Format not supported, module "'.$determined_format['include'].'" was removed.'); |
| 443 | + } |
| 444 | + |
| 445 | + // module requires iconv support |
| 446 | + if (!function_exists('iconv') && !empty($determined_format['iconv_req'])) { |
| 447 | + return $this->error('iconv support is required for this module ('.$determined_format['include'].').'); |
| 448 | + } |
| 449 | + |
| 450 | + // include module |
| 451 | + include_once(GETID3_INCLUDEPATH.$determined_format['include']); |
| 452 | + |
| 453 | + // instantiate module class |
| 454 | + $class_name = 'getid3_'.$determined_format['module']; |
| 455 | + if (!class_exists($class_name)) { |
| 456 | + return $this->error('Format not supported, module "'.$determined_format['include'].'" is corrupt.'); |
| 457 | + } |
| 458 | + if (isset($determined_format['option'])) { |
| 459 | + $class = new $class_name($fp, $this->info, $determined_format['option']); |
| 460 | + } else { |
| 461 | + $class = new $class_name($fp, $this->info); |
| 462 | + } |
| 463 | + unset($class); |
| 464 | + |
| 465 | + // close file |
| 466 | + fclose($fp); |
| 467 | + |
| 468 | + // process all tags - copy to 'tags' and convert charsets |
| 469 | + if ($this->option_tags_process) { |
| 470 | + $this->HandleAllTags(); |
| 471 | + } |
| 472 | + |
| 473 | + // perform more calculations |
| 474 | + if ($this->option_extra_info) { |
| 475 | + $this->ChannelsBitratePlaytimeCalculations(); |
| 476 | + $this->CalculateCompressionRatioVideo(); |
| 477 | + $this->CalculateCompressionRatioAudio(); |
| 478 | + $this->CalculateReplayGain(); |
| 479 | + $this->ProcessAudioStreams(); |
| 480 | + } |
| 481 | + |
| 482 | + // get the MD5 sum of the audio/video portion of the file - without ID3/APE/Lyrics3/etc header/footer tags |
| 483 | + if ($this->option_md5_data) { |
| 484 | + // do not cald md5_data if md5_data_source is present - set by flac only - future MPC/SV8 too |
| 485 | + if (!$this->option_md5_data_source || empty($this->info['md5_data_source'])) { |
| 486 | + $this->getHashdata('md5'); |
| 487 | + } |
| 488 | + } |
| 489 | + |
| 490 | + // get the SHA1 sum of the audio/video portion of the file - without ID3/APE/Lyrics3/etc header/footer tags |
| 491 | + if ($this->option_sha1_data) { |
| 492 | + $this->getHashdata('sha1'); |
| 493 | + } |
| 494 | + |
| 495 | + // remove undesired keys |
| 496 | + $this->CleanUp(); |
| 497 | + |
| 498 | + // return info array |
| 499 | + return $this->info; |
| 500 | + } |
| 501 | + |
| 502 | + |
| 503 | + // private: error handling |
| 504 | + function error($message) { |
| 505 | + |
| 506 | + $this->CleanUp(); |
| 507 | + |
| 508 | + $this->info['error'][] = $message; |
| 509 | + return $this->info; |
| 510 | + } |
| 511 | + |
| 512 | + |
| 513 | + // private: warning handling |
| 514 | + function warning($message) { |
| 515 | + $this->info['warning'][] = $message; |
| 516 | + return true; |
| 517 | + } |
| 518 | + |
| 519 | + |
| 520 | + // private: CleanUp |
| 521 | + function CleanUp() { |
| 522 | + |
| 523 | + // remove possible empty keys |
| 524 | + $AVpossibleEmptyKeys = array('dataformat', 'bits_per_sample', 'encoder_options', 'streams', 'bitrate'); |
| 525 | + foreach ($AVpossibleEmptyKeys as $dummy => $key) { |
| 526 | + if (empty($this->info['audio'][$key]) && isset($this->info['audio'][$key])) { |
| 527 | + unset($this->info['audio'][$key]); |
| 528 | + } |
| 529 | + if (empty($this->info['video'][$key]) && isset($this->info['video'][$key])) { |
| 530 | + unset($this->info['video'][$key]); |
| 531 | + } |
| 532 | + } |
| 533 | + |
| 534 | + // remove empty root keys |
| 535 | + if (!empty($this->info)) { |
| 536 | + foreach ($this->info as $key => $value) { |
| 537 | + if (empty($this->info[$key]) && ($this->info[$key] !== 0) && ($this->info[$key] !== '0')) { |
| 538 | + unset($this->info[$key]); |
| 539 | + } |
| 540 | + } |
| 541 | + } |
| 542 | + |
| 543 | + // remove meaningless entries from unknown-format files |
| 544 | + if (empty($this->info['fileformat'])) { |
| 545 | + if (isset($this->info['avdataoffset'])) { |
| 546 | + unset($this->info['avdataoffset']); |
| 547 | + } |
| 548 | + if (isset($this->info['avdataend'])) { |
| 549 | + unset($this->info['avdataend']); |
| 550 | + } |
| 551 | + } |
| 552 | + } |
| 553 | + |
| 554 | + |
| 555 | + // return array containing information about all supported formats |
| 556 | + function GetFileFormatArray() { |
| 557 | + static $format_info = array(); |
| 558 | + if (empty($format_info)) { |
| 559 | + $format_info = array( |
| 560 | + |
| 561 | + // Audio formats |
| 562 | + |
| 563 | + // AC-3 - audio - Dolby AC-3 / Dolby Digital |
| 564 | + 'ac3' => array( |
| 565 | + 'pattern' => '^\x0B\x77', |
| 566 | + 'group' => 'audio', |
| 567 | + 'module' => 'ac3', |
| 568 | + 'mime_type' => 'audio/ac3', |
| 569 | + ), |
| 570 | + |
| 571 | + // AAC - audio - Advanced Audio Coding (AAC) - ADIF format |
| 572 | + 'adif' => array( |
| 573 | + 'pattern' => '^ADIF', |
| 574 | + 'group' => 'audio', |
| 575 | + 'module' => 'aac', |
| 576 | + 'option' => 'adif', |
| 577 | + 'mime_type' => 'application/octet-stream', |
| 578 | + 'fail_ape' => 'WARNING', |
| 579 | + ), |
| 580 | + |
| 581 | + |
| 582 | + // AA - audio - Audible Audiobook |
| 583 | + 'adts' => array( |
| 584 | + 'pattern' => '^.{4}\x57\x90\x75\x36', |
| 585 | + 'group' => 'audio', |
| 586 | + 'module' => 'aa', |
| 587 | + 'mime_type' => 'audio/audible ', |
| 588 | + ), |
| 589 | + |
| 590 | + // AAC - audio - Advanced Audio Coding (AAC) - ADTS format (very similar to MP3) |
| 591 | + 'adts' => array( |
| 592 | + 'pattern' => '^\xFF[\xF0-\xF1\xF8-\xF9]', |
| 593 | + 'group' => 'audio', |
| 594 | + 'module' => 'aac', |
| 595 | + 'option' => 'adts', |
| 596 | + 'mime_type' => 'application/octet-stream', |
| 597 | + 'fail_ape' => 'WARNING', |
| 598 | + ), |
| 599 | + |
| 600 | + |
| 601 | + // AU - audio - NeXT/Sun AUdio (AU) |
| 602 | + 'au' => array( |
| 603 | + 'pattern' => '^\.snd', |
| 604 | + 'group' => 'audio', |
| 605 | + 'module' => 'au', |
| 606 | + 'mime_type' => 'audio/basic', |
| 607 | + ), |
| 608 | + |
| 609 | + // AVR - audio - Audio Visual Research |
| 610 | + 'avr' => array( |
| 611 | + 'pattern' => '^2BIT', |
| 612 | + 'group' => 'audio', |
| 613 | + 'module' => 'avr', |
| 614 | + 'mime_type' => 'application/octet-stream', |
| 615 | + ), |
| 616 | + |
| 617 | + // BONK - audio - Bonk v0.9+ |
| 618 | + 'bonk' => array( |
| 619 | + 'pattern' => '^\x00(BONK|INFO|META| ID3)', |
| 620 | + 'group' => 'audio', |
| 621 | + 'module' => 'bonk', |
| 622 | + 'mime_type' => 'audio/xmms-bonk', |
| 623 | + ), |
| 624 | + |
| 625 | + // DSS - audio - Digital Speech Standard |
| 626 | + 'dss' => array( |
| 627 | + 'pattern' => '^[\x02]dss', |
| 628 | + 'group' => 'audio', |
| 629 | + 'module' => 'dss', |
| 630 | + 'mime_type' => 'application/octet-stream', |
| 631 | + ), |
| 632 | + |
| 633 | + // DTS - audio - Dolby Theatre System |
| 634 | + 'dts' => array( |
| 635 | + 'pattern' => '^\x7F\xFE\x80\x01', |
| 636 | + 'group' => 'audio', |
| 637 | + 'module' => 'dts', |
| 638 | + 'mime_type' => 'audio/dts', |
| 639 | + ), |
| 640 | + |
| 641 | + // FLAC - audio - Free Lossless Audio Codec |
| 642 | + 'flac' => array( |
| 643 | + 'pattern' => '^fLaC', |
| 644 | + 'group' => 'audio', |
| 645 | + 'module' => 'flac', |
| 646 | + 'mime_type' => 'audio/x-flac', |
| 647 | + ), |
| 648 | + |
| 649 | + // LA - audio - Lossless Audio (LA) |
| 650 | + 'la' => array( |
| 651 | + 'pattern' => '^LA0[2-4]', |
| 652 | + 'group' => 'audio', |
| 653 | + 'module' => 'la', |
| 654 | + 'mime_type' => 'application/octet-stream', |
| 655 | + ), |
| 656 | + |
| 657 | + // LPAC - audio - Lossless Predictive Audio Compression (LPAC) |
| 658 | + 'lpac' => array( |
| 659 | + 'pattern' => '^LPAC', |
| 660 | + 'group' => 'audio', |
| 661 | + 'module' => 'lpac', |
| 662 | + 'mime_type' => 'application/octet-stream', |
| 663 | + ), |
| 664 | + |
| 665 | + // MIDI - audio - MIDI (Musical Instrument Digital Interface) |
| 666 | + 'midi' => array( |
| 667 | + 'pattern' => '^MThd', |
| 668 | + 'group' => 'audio', |
| 669 | + 'module' => 'midi', |
| 670 | + 'mime_type' => 'audio/midi', |
| 671 | + ), |
| 672 | + |
| 673 | + // MAC - audio - Monkey's Audio Compressor |
| 674 | + 'mac' => array( |
| 675 | + 'pattern' => '^MAC ', |
| 676 | + 'group' => 'audio', |
| 677 | + 'module' => 'monkey', |
| 678 | + 'mime_type' => 'application/octet-stream', |
| 679 | + ), |
| 680 | + |
| 681 | +// has been known to produce false matches in random files (e.g. JPEGs), leave out until more precise matching available |
| 682 | +// // MOD - audio - MODule (assorted sub-formats) |
| 683 | +// 'mod' => array( |
| 684 | +// 'pattern' => '^.{1080}(M\\.K\\.|M!K!|FLT4|FLT8|[5-9]CHN|[1-3][0-9]CH)', |
| 685 | +// 'group' => 'audio', |
| 686 | +// 'module' => 'mod', |
| 687 | +// 'option' => 'mod', |
| 688 | +// 'mime_type' => 'audio/mod', |
| 689 | +// ), |
| 690 | + |
| 691 | + // MOD - audio - MODule (Impulse Tracker) |
| 692 | + 'it' => array( |
| 693 | + 'pattern' => '^IMPM', |
| 694 | + 'group' => 'audio', |
| 695 | + 'module' => 'mod', |
| 696 | + 'option' => 'it', |
| 697 | + 'mime_type' => 'audio/it', |
| 698 | + ), |
| 699 | + |
| 700 | + // MOD - audio - MODule (eXtended Module, various sub-formats) |
| 701 | + 'xm' => array( |
| 702 | + 'pattern' => '^Extended Module', |
| 703 | + 'group' => 'audio', |
| 704 | + 'module' => 'mod', |
| 705 | + 'option' => 'xm', |
| 706 | + 'mime_type' => 'audio/xm', |
| 707 | + ), |
| 708 | + |
| 709 | + // MOD - audio - MODule (ScreamTracker) |
| 710 | + 's3m' => array( |
| 711 | + 'pattern' => '^.{44}SCRM', |
| 712 | + 'group' => 'audio', |
| 713 | + 'module' => 'mod', |
| 714 | + 'option' => 's3m', |
| 715 | + 'mime_type' => 'audio/s3m', |
| 716 | + ), |
| 717 | + |
| 718 | + // MPC - audio - Musepack / MPEGplus |
| 719 | + 'mpc' => array( |
| 720 | + 'pattern' => '^(MPCK|MP\+|[\x00\x01\x10\x11\x40\x41\x50\x51\x80\x81\x90\x91\xC0\xC1\xD0\xD1][\x20-37][\x00\x20\x40\x60\x80\xA0\xC0\xE0])', |
| 721 | + 'group' => 'audio', |
| 722 | + 'module' => 'mpc', |
| 723 | + 'mime_type' => 'audio/x-musepack', |
| 724 | + ), |
| 725 | + |
| 726 | + // MP3 - audio - MPEG-audio Layer 3 (very similar to AAC-ADTS) |
| 727 | + 'mp3' => array( |
| 728 | + 'pattern' => '^\xFF[\xE2-\xE7\xF2-\xF7\xFA-\xFF][\x00-\x0B\x10-\x1B\x20-\x2B\x30-\x3B\x40-\x4B\x50-\x5B\x60-\x6B\x70-\x7B\x80-\x8B\x90-\x9B\xA0-\xAB\xB0-\xBB\xC0-\xCB\xD0-\xDB\xE0-\xEB\xF0-\xFB]', |
| 729 | + 'group' => 'audio', |
| 730 | + 'module' => 'mp3', |
| 731 | + 'mime_type' => 'audio/mpeg', |
| 732 | + ), |
| 733 | + |
| 734 | + // OFR - audio - OptimFROG |
| 735 | + 'ofr' => array( |
| 736 | + 'pattern' => '^(\*RIFF|OFR)', |
| 737 | + 'group' => 'audio', |
| 738 | + 'module' => 'optimfrog', |
| 739 | + 'mime_type' => 'application/octet-stream', |
| 740 | + ), |
| 741 | + |
| 742 | + // RKAU - audio - RKive AUdio compressor |
| 743 | + 'rkau' => array( |
| 744 | + 'pattern' => '^RKA', |
| 745 | + 'group' => 'audio', |
| 746 | + 'module' => 'rkau', |
| 747 | + 'mime_type' => 'application/octet-stream', |
| 748 | + ), |
| 749 | + |
| 750 | + // SHN - audio - Shorten |
| 751 | + 'shn' => array( |
| 752 | + 'pattern' => '^ajkg', |
| 753 | + 'group' => 'audio', |
| 754 | + 'module' => 'shorten', |
| 755 | + 'mime_type' => 'audio/xmms-shn', |
| 756 | + 'fail_id3' => 'ERROR', |
| 757 | + 'fail_ape' => 'ERROR', |
| 758 | + ), |
| 759 | + |
| 760 | + // TTA - audio - TTA Lossless Audio Compressor (http://tta.corecodec.org) |
| 761 | + 'tta' => array( |
| 762 | + 'pattern' => '^TTA', // could also be '^TTA(\x01|\x02|\x03|2|1)' |
| 763 | + 'group' => 'audio', |
| 764 | + 'module' => 'tta', |
| 765 | + 'mime_type' => 'application/octet-stream', |
| 766 | + ), |
| 767 | + |
| 768 | + // VOC - audio - Creative Voice (VOC) |
| 769 | + 'voc' => array( |
| 770 | + 'pattern' => '^Creative Voice File', |
| 771 | + 'group' => 'audio', |
| 772 | + 'module' => 'voc', |
| 773 | + 'mime_type' => 'audio/voc', |
| 774 | + ), |
| 775 | + |
| 776 | + // VQF - audio - transform-domain weighted interleave Vector Quantization Format (VQF) |
| 777 | + 'vqf' => array( |
| 778 | + 'pattern' => '^TWIN', |
| 779 | + 'group' => 'audio', |
| 780 | + 'module' => 'vqf', |
| 781 | + 'mime_type' => 'application/octet-stream', |
| 782 | + ), |
| 783 | + |
| 784 | + // WV - audio - WavPack (v4.0+) |
| 785 | + 'wv' => array( |
| 786 | + 'pattern' => '^wvpk', |
| 787 | + 'group' => 'audio', |
| 788 | + 'module' => 'wavpack', |
| 789 | + 'mime_type' => 'application/octet-stream', |
| 790 | + ), |
| 791 | + |
| 792 | + |
| 793 | + // Audio-Video formats |
| 794 | + |
| 795 | + // ASF - audio/video - Advanced Streaming Format, Windows Media Video, Windows Media Audio |
| 796 | + 'asf' => array( |
| 797 | + 'pattern' => '^\x30\x26\xB2\x75\x8E\x66\xCF\x11\xA6\xD9\x00\xAA\x00\x62\xCE\x6C', |
| 798 | + 'group' => 'audio-video', |
| 799 | + 'module' => 'asf', |
| 800 | + 'mime_type' => 'video/x-ms-asf', |
| 801 | + 'iconv_req' => false, |
| 802 | + ), |
| 803 | + |
| 804 | + // BINK - audio/video - Bink / Smacker |
| 805 | + 'bink' => array( |
| 806 | + 'pattern' => '^(BIK|SMK)', |
| 807 | + 'group' => 'audio-video', |
| 808 | + 'module' => 'bink', |
| 809 | + 'mime_type' => 'application/octet-stream', |
| 810 | + ), |
| 811 | + |
| 812 | + // FLV - audio/video - FLash Video |
| 813 | + 'flv' => array( |
| 814 | + 'pattern' => '^FLV\x01', |
| 815 | + 'group' => 'audio-video', |
| 816 | + 'module' => 'flv', |
| 817 | + 'mime_type' => 'video/x-flv', |
| 818 | + ), |
| 819 | + |
| 820 | + // MKAV - audio/video - Mastroka |
| 821 | + 'matroska' => array( |
| 822 | + 'pattern' => '^\x1A\x45\xDF\xA3', |
| 823 | + 'group' => 'audio-video', |
| 824 | + 'module' => 'matroska', |
| 825 | + 'mime_type' => 'video/x-matroska', // may also be audio/x-matroska |
| 826 | + ), |
| 827 | + |
| 828 | + // MPEG - audio/video - MPEG (Moving Pictures Experts Group) |
| 829 | + 'mpeg' => array( |
| 830 | + 'pattern' => '^\x00\x00\x01(\xBA|\xB3)', |
| 831 | + 'group' => 'audio-video', |
| 832 | + 'module' => 'mpeg', |
| 833 | + 'mime_type' => 'video/mpeg', |
| 834 | + ), |
| 835 | + |
| 836 | + // NSV - audio/video - Nullsoft Streaming Video (NSV) |
| 837 | + 'nsv' => array( |
| 838 | + 'pattern' => '^NSV[sf]', |
| 839 | + 'group' => 'audio-video', |
| 840 | + 'module' => 'nsv', |
| 841 | + 'mime_type' => 'application/octet-stream', |
| 842 | + ), |
| 843 | + |
| 844 | + // Ogg - audio/video - Ogg (Ogg-Vorbis, Ogg-FLAC, Speex, Ogg-Theora(*), Ogg-Tarkin(*)) |
| 845 | + 'ogg' => array( |
| 846 | + 'pattern' => '^OggS', |
| 847 | + 'group' => 'audio', |
| 848 | + 'module' => 'ogg', |
| 849 | + 'mime_type' => 'application/ogg', |
| 850 | + 'fail_id3' => 'WARNING', |
| 851 | + 'fail_ape' => 'WARNING', |
| 852 | + ), |
| 853 | + |
| 854 | + // QT - audio/video - Quicktime |
| 855 | + 'quicktime' => array( |
| 856 | + 'pattern' => '^.{4}(cmov|free|ftyp|mdat|moov|pnot|skip|wide)', |
| 857 | + 'group' => 'audio-video', |
| 858 | + 'module' => 'quicktime', |
| 859 | + 'mime_type' => 'video/quicktime', |
| 860 | + ), |
| 861 | + |
| 862 | + // RIFF - audio/video - Resource Interchange File Format (RIFF) / WAV / AVI / CD-audio / SDSS = renamed variant used by SmartSound QuickTracks (www.smartsound.com) / FORM = Audio Interchange File Format (AIFF) |
| 863 | + 'riff' => array( |
| 864 | + 'pattern' => '^(RIFF|SDSS|FORM)', |
| 865 | + 'group' => 'audio-video', |
| 866 | + 'module' => 'riff', |
| 867 | + 'mime_type' => 'audio/x-wave', |
| 868 | + 'fail_ape' => 'WARNING', |
| 869 | + ), |
| 870 | + |
| 871 | + // Real - audio/video - RealAudio, RealVideo |
| 872 | + 'real' => array( |
| 873 | + 'pattern' => '^(\\.RMF|\\.ra)', |
| 874 | + 'group' => 'audio-video', |
| 875 | + 'module' => 'real', |
| 876 | + 'mime_type' => 'audio/x-realaudio', |
| 877 | + ), |
| 878 | + |
| 879 | + // SWF - audio/video - ShockWave Flash |
| 880 | + 'swf' => array( |
| 881 | + 'pattern' => '^(F|C)WS', |
| 882 | + 'group' => 'audio-video', |
| 883 | + 'module' => 'swf', |
| 884 | + 'mime_type' => 'application/x-shockwave-flash', |
| 885 | + ), |
| 886 | + |
| 887 | + |
| 888 | + // Still-Image formats |
| 889 | + |
| 890 | + // BMP - still image - Bitmap (Windows, OS/2; uncompressed, RLE8, RLE4) |
| 891 | + 'bmp' => array( |
| 892 | + 'pattern' => '^BM', |
| 893 | + 'group' => 'graphic', |
| 894 | + 'module' => 'bmp', |
| 895 | + 'mime_type' => 'image/bmp', |
| 896 | + 'fail_id3' => 'ERROR', |
| 897 | + 'fail_ape' => 'ERROR', |
| 898 | + ), |
| 899 | + |
| 900 | + // GIF - still image - Graphics Interchange Format |
| 901 | + 'gif' => array( |
| 902 | + 'pattern' => '^GIF', |
| 903 | + 'group' => 'graphic', |
| 904 | + 'module' => 'gif', |
| 905 | + 'mime_type' => 'image/gif', |
| 906 | + 'fail_id3' => 'ERROR', |
| 907 | + 'fail_ape' => 'ERROR', |
| 908 | + ), |
| 909 | + |
| 910 | + // JPEG - still image - Joint Photographic Experts Group (JPEG) |
| 911 | + 'jpg' => array( |
| 912 | + 'pattern' => '^\xFF\xD8\xFF', |
| 913 | + 'group' => 'graphic', |
| 914 | + 'module' => 'jpg', |
| 915 | + 'mime_type' => 'image/jpeg', |
| 916 | + 'fail_id3' => 'ERROR', |
| 917 | + 'fail_ape' => 'ERROR', |
| 918 | + ), |
| 919 | + |
| 920 | + // PCD - still image - Kodak Photo CD |
| 921 | + 'pcd' => array( |
| 922 | + 'pattern' => '^.{2048}PCD_IPI\x00', |
| 923 | + 'group' => 'graphic', |
| 924 | + 'module' => 'pcd', |
| 925 | + 'mime_type' => 'image/x-photo-cd', |
| 926 | + 'fail_id3' => 'ERROR', |
| 927 | + 'fail_ape' => 'ERROR', |
| 928 | + ), |
| 929 | + |
| 930 | + |
| 931 | + // PNG - still image - Portable Network Graphics (PNG) |
| 932 | + 'png' => array( |
| 933 | + 'pattern' => '^\x89\x50\x4E\x47\x0D\x0A\x1A\x0A', |
| 934 | + 'group' => 'graphic', |
| 935 | + 'module' => 'png', |
| 936 | + 'mime_type' => 'image/png', |
| 937 | + 'fail_id3' => 'ERROR', |
| 938 | + 'fail_ape' => 'ERROR', |
| 939 | + ), |
| 940 | + |
| 941 | + |
| 942 | + // SVG - still image - Scalable Vector Graphics (SVG) |
| 943 | + 'svg' => array( |
| 944 | + 'pattern' => '(<!DOCTYPE svg PUBLIC |xmlns="http:\/\/www\.w3\.org\/2000\/svg")', |
| 945 | + 'group' => 'graphic', |
| 946 | + 'module' => 'svg', |
| 947 | + 'mime_type' => 'image/svg+xml', |
| 948 | + 'fail_id3' => 'ERROR', |
| 949 | + 'fail_ape' => 'ERROR', |
| 950 | + ), |
| 951 | + |
| 952 | + |
| 953 | + // TIFF - still image - Tagged Information File Format (TIFF) |
| 954 | + 'tiff' => array( |
| 955 | + 'pattern' => '^(II\x2A\x00|MM\x00\x2A)', |
| 956 | + 'group' => 'graphic', |
| 957 | + 'module' => 'tiff', |
| 958 | + 'mime_type' => 'image/tiff', |
| 959 | + 'fail_id3' => 'ERROR', |
| 960 | + 'fail_ape' => 'ERROR', |
| 961 | + ), |
| 962 | + |
| 963 | + |
| 964 | + // Data formats |
| 965 | + |
| 966 | + // ISO - data - International Standards Organization (ISO) CD-ROM Image |
| 967 | + 'iso' => array( |
| 968 | + 'pattern' => '^.{32769}CD001', |
| 969 | + 'group' => 'misc', |
| 970 | + 'module' => 'iso', |
| 971 | + 'mime_type' => 'application/octet-stream', |
| 972 | + 'fail_id3' => 'ERROR', |
| 973 | + 'fail_ape' => 'ERROR', |
| 974 | + 'iconv_req' => false, |
| 975 | + ), |
| 976 | + |
| 977 | + // RAR - data - RAR compressed data |
| 978 | + 'rar' => array( |
| 979 | + 'pattern' => '^Rar\!', |
| 980 | + 'group' => 'archive', |
| 981 | + 'module' => 'rar', |
| 982 | + 'mime_type' => 'application/octet-stream', |
| 983 | + 'fail_id3' => 'ERROR', |
| 984 | + 'fail_ape' => 'ERROR', |
| 985 | + ), |
| 986 | + |
| 987 | + // SZIP - audio/data - SZIP compressed data |
| 988 | + 'szip' => array( |
| 989 | + 'pattern' => '^SZ\x0A\x04', |
| 990 | + 'group' => 'archive', |
| 991 | + 'module' => 'szip', |
| 992 | + 'mime_type' => 'application/octet-stream', |
| 993 | + 'fail_id3' => 'ERROR', |
| 994 | + 'fail_ape' => 'ERROR', |
| 995 | + ), |
| 996 | + |
| 997 | + // TAR - data - TAR compressed data |
| 998 | + 'tar' => array( |
| 999 | + 'pattern' => '^.{100}[0-9\x20]{7}\x00[0-9\x20]{7}\x00[0-9\x20]{7}\x00[0-9\x20\x00]{12}[0-9\x20\x00]{12}', |
| 1000 | + 'group' => 'archive', |
| 1001 | + 'module' => 'tar', |
| 1002 | + 'mime_type' => 'application/x-tar', |
| 1003 | + 'fail_id3' => 'ERROR', |
| 1004 | + 'fail_ape' => 'ERROR', |
| 1005 | + ), |
| 1006 | + |
| 1007 | + // GZIP - data - GZIP compressed data |
| 1008 | + 'gz' => array( |
| 1009 | + 'pattern' => '^\x1F\x8B\x08', |
| 1010 | + 'group' => 'archive', |
| 1011 | + 'module' => 'gzip', |
| 1012 | + 'mime_type' => 'application/x-gzip', |
| 1013 | + 'fail_id3' => 'ERROR', |
| 1014 | + 'fail_ape' => 'ERROR', |
| 1015 | + ), |
| 1016 | + |
| 1017 | + // ZIP - data - ZIP compressed data |
| 1018 | + 'zip' => array( |
| 1019 | + 'pattern' => '^PK\x03\x04', |
| 1020 | + 'group' => 'archive', |
| 1021 | + 'module' => 'zip', |
| 1022 | + 'mime_type' => 'application/zip', |
| 1023 | + 'fail_id3' => 'ERROR', |
| 1024 | + 'fail_ape' => 'ERROR', |
| 1025 | + ), |
| 1026 | + |
| 1027 | + |
| 1028 | + // Misc other formats |
| 1029 | + |
| 1030 | + // PAR2 - data - Parity Volume Set Specification 2.0 |
| 1031 | + 'par2' => array ( |
| 1032 | + 'pattern' => '^PAR2\x00PKT', |
| 1033 | + 'group' => 'misc', |
| 1034 | + 'module' => 'par2', |
| 1035 | + 'mime_type' => 'application/octet-stream', |
| 1036 | + 'fail_id3' => 'ERROR', |
| 1037 | + 'fail_ape' => 'ERROR', |
| 1038 | + ), |
| 1039 | + |
| 1040 | + // PDF - data - Portable Document Format |
| 1041 | + 'pdf' => array( |
| 1042 | + 'pattern' => '^\x25PDF', |
| 1043 | + 'group' => 'misc', |
| 1044 | + 'module' => 'pdf', |
| 1045 | + 'mime_type' => 'application/pdf', |
| 1046 | + 'fail_id3' => 'ERROR', |
| 1047 | + 'fail_ape' => 'ERROR', |
| 1048 | + ), |
| 1049 | + |
| 1050 | + // MSOFFICE - data - ZIP compressed data |
| 1051 | + 'msoffice' => array( |
| 1052 | + 'pattern' => '^\xD0\xCF\x11\xE0', // D0CF11E == DOCFILE == Microsoft Office Document |
| 1053 | + 'group' => 'misc', |
| 1054 | + 'module' => 'msoffice', |
| 1055 | + 'mime_type' => 'application/octet-stream', |
| 1056 | + 'fail_id3' => 'ERROR', |
| 1057 | + 'fail_ape' => 'ERROR', |
| 1058 | + ), |
| 1059 | + |
| 1060 | + // CUE - data - CUEsheet (index to single-file disc images) |
| 1061 | + 'cue' => array( |
| 1062 | + 'pattern' => '', // empty pattern means cannot be automatically detected, will fall through all other formats and match based on filename and very basic file contents |
| 1063 | + 'group' => 'misc', |
| 1064 | + 'module' => 'cue', |
| 1065 | + 'mime_type' => 'application/octet-stream', |
| 1066 | + ), |
| 1067 | + |
| 1068 | + ); |
| 1069 | + } |
| 1070 | + |
| 1071 | + return $format_info; |
| 1072 | + } |
| 1073 | + |
| 1074 | + |
| 1075 | + |
| 1076 | + function GetFileFormat(&$filedata, $filename='') { |
| 1077 | + // this function will determine the format of a file based on usually |
| 1078 | + // the first 2-4 bytes of the file (8 bytes for PNG, 16 bytes for JPG, |
| 1079 | + // and in the case of ISO CD image, 6 bytes offset 32kb from the start |
| 1080 | + // of the file). |
| 1081 | + |
| 1082 | + // Identify file format - loop through $format_info and detect with reg expr |
| 1083 | + foreach ($this->GetFileFormatArray() as $format_name => $info) { |
| 1084 | + // The /s switch on preg_match() forces preg_match() NOT to treat |
| 1085 | + // newline (0x0A) characters as special chars but do a binary match |
| 1086 | + if (!empty($info['pattern']) && preg_match('#'.$info['pattern'].'#s', $filedata)) { |
| 1087 | + $info['include'] = 'module.'.$info['group'].'.'.$info['module'].'.php'; |
| 1088 | + return $info; |
| 1089 | + } |
| 1090 | + } |
| 1091 | + |
| 1092 | + |
| 1093 | + if (preg_match('#\.mp[123a]$#i', $filename)) { |
| 1094 | + // Too many mp3 encoders on the market put gabage in front of mpeg files |
| 1095 | + // use assume format on these if format detection failed |
| 1096 | + $GetFileFormatArray = $this->GetFileFormatArray(); |
| 1097 | + $info = $GetFileFormatArray['mp3']; |
| 1098 | + $info['include'] = 'module.'.$info['group'].'.'.$info['module'].'.php'; |
| 1099 | + return $info; |
| 1100 | + } elseif (preg_match('/\.cue$/i', $filename) && preg_match('#FILE "[^"]+" (BINARY|MOTOROLA|AIFF|WAVE|MP3)#', $filedata)) { |
| 1101 | + // there's not really a useful consistent "magic" at the beginning of .cue files to identify them |
| 1102 | + // so until I think of something better, just go by filename if all other format checks fail |
| 1103 | + // and verify there's at least one instance of "TRACK xx AUDIO" in the file |
| 1104 | + $GetFileFormatArray = $this->GetFileFormatArray(); |
| 1105 | + $info = $GetFileFormatArray['cue']; |
| 1106 | + $info['include'] = 'module.'.$info['group'].'.'.$info['module'].'.php'; |
| 1107 | + return $info; |
| 1108 | + } |
| 1109 | + |
| 1110 | + return false; |
| 1111 | + } |
| 1112 | + |
| 1113 | + |
| 1114 | + // converts array to $encoding charset from $this->encoding |
| 1115 | + function CharConvert(&$array, $encoding) { |
| 1116 | + |
| 1117 | + // identical encoding - end here |
| 1118 | + if ($encoding == $this->encoding) { |
| 1119 | + return; |
| 1120 | + } |
| 1121 | + |
| 1122 | + // loop thru array |
| 1123 | + foreach ($array as $key => $value) { |
| 1124 | + |
| 1125 | + // go recursive |
| 1126 | + if (is_array($value)) { |
| 1127 | + $this->CharConvert($array[$key], $encoding); |
| 1128 | + } |
| 1129 | + |
| 1130 | + // convert string |
| 1131 | + elseif (is_string($value)) { |
| 1132 | + $array[$key] = trim(getid3_lib::iconv_fallback($encoding, $this->encoding, $value)); |
| 1133 | + } |
| 1134 | + } |
| 1135 | + } |
| 1136 | + |
| 1137 | + |
| 1138 | + function HandleAllTags() { |
| 1139 | + |
| 1140 | + // key name => array (tag name, character encoding) |
| 1141 | + static $tags; |
| 1142 | + if (empty($tags)) { |
| 1143 | + $tags = array( |
| 1144 | + 'asf' => array('asf' , 'UTF-16LE'), |
| 1145 | + 'midi' => array('midi' , 'ISO-8859-1'), |
| 1146 | + 'nsv' => array('nsv' , 'ISO-8859-1'), |
| 1147 | + 'ogg' => array('vorbiscomment' , 'UTF-8'), |
| 1148 | + 'png' => array('png' , 'UTF-8'), |
| 1149 | + 'tiff' => array('tiff' , 'ISO-8859-1'), |
| 1150 | + 'quicktime' => array('quicktime' , 'UTF-8'), |
| 1151 | + 'real' => array('real' , 'ISO-8859-1'), |
| 1152 | + 'vqf' => array('vqf' , 'ISO-8859-1'), |
| 1153 | + 'zip' => array('zip' , 'ISO-8859-1'), |
| 1154 | + 'riff' => array('riff' , 'ISO-8859-1'), |
| 1155 | + 'lyrics3' => array('lyrics3' , 'ISO-8859-1'), |
| 1156 | + 'id3v1' => array('id3v1' , $this->encoding_id3v1), |
| 1157 | + 'id3v2' => array('id3v2' , 'UTF-8'), // not according to the specs (every frame can have a different encoding), but getID3() force-converts all encodings to UTF-8 |
| 1158 | + 'ape' => array('ape' , 'UTF-8'), |
| 1159 | + 'cue' => array('cue' , 'ISO-8859-1'), |
| 1160 | + ); |
| 1161 | + } |
| 1162 | + |
| 1163 | + // loop thru comments array |
| 1164 | + foreach ($tags as $comment_name => $tagname_encoding_array) { |
| 1165 | + list($tag_name, $encoding) = $tagname_encoding_array; |
| 1166 | + |
| 1167 | + // fill in default encoding type if not already present |
| 1168 | + if (isset($this->info[$comment_name]) && !isset($this->info[$comment_name]['encoding'])) { |
| 1169 | + $this->info[$comment_name]['encoding'] = $encoding; |
| 1170 | + } |
| 1171 | + |
| 1172 | + // copy comments if key name set |
| 1173 | + if (!empty($this->info[$comment_name]['comments'])) { |
| 1174 | + |
| 1175 | + foreach ($this->info[$comment_name]['comments'] as $tag_key => $valuearray) { |
| 1176 | + foreach ($valuearray as $key => $value) { |
| 1177 | + if (strlen(trim($value)) > 0) { |
| 1178 | + $this->info['tags'][trim($tag_name)][trim($tag_key)][] = $value; // do not trim!! Unicode characters will get mangled if trailing nulls are removed! |
| 1179 | + } |
| 1180 | + } |
| 1181 | + } |
| 1182 | + |
| 1183 | + if (!isset($this->info['tags'][$tag_name])) { |
| 1184 | + // comments are set but contain nothing but empty strings, so skip |
| 1185 | + continue; |
| 1186 | + } |
| 1187 | + |
| 1188 | + if ($this->option_tags_html) { |
| 1189 | + foreach ($this->info['tags'][$tag_name] as $tag_key => $valuearray) { |
| 1190 | + foreach ($valuearray as $key => $value) { |
| 1191 | + if (is_string($value)) { |
| 1192 | + //$this->info['tags_html'][$tag_name][$tag_key][$key] = getid3_lib::MultiByteCharString2HTML($value, $encoding); |
| 1193 | + $this->info['tags_html'][$tag_name][$tag_key][$key] = str_replace('�', '', getid3_lib::MultiByteCharString2HTML($value, $encoding)); |
| 1194 | + } else { |
| 1195 | + $this->info['tags_html'][$tag_name][$tag_key][$key] = $value; |
| 1196 | + } |
| 1197 | + } |
| 1198 | + } |
| 1199 | + } |
| 1200 | + |
| 1201 | + $this->CharConvert($this->info['tags'][$tag_name], $encoding); // only copy gets converted! |
| 1202 | + } |
| 1203 | + |
| 1204 | + } |
| 1205 | + return true; |
| 1206 | + } |
| 1207 | + |
| 1208 | + |
| 1209 | + function getHashdata($algorithm) { |
| 1210 | + switch ($algorithm) { |
| 1211 | + case 'md5': |
| 1212 | + case 'sha1': |
| 1213 | + break; |
| 1214 | + |
| 1215 | + default: |
| 1216 | + return $this->error('bad algorithm "'.$algorithm.'" in getHashdata()'); |
| 1217 | + break; |
| 1218 | + } |
| 1219 | + |
| 1220 | + if (!empty($this->info['fileformat']) && !empty($this->info['dataformat']) && ($this->info['fileformat'] == 'ogg') && ($this->info['audio']['dataformat'] == 'vorbis')) { |
| 1221 | + |
| 1222 | + // We cannot get an identical md5_data value for Ogg files where the comments |
| 1223 | + // span more than 1 Ogg page (compared to the same audio data with smaller |
| 1224 | + // comments) using the normal getID3() method of MD5'ing the data between the |
| 1225 | + // end of the comments and the end of the file (minus any trailing tags), |
| 1226 | + // because the page sequence numbers of the pages that the audio data is on |
| 1227 | + // do not match. Under normal circumstances, where comments are smaller than |
| 1228 | + // the nominal 4-8kB page size, then this is not a problem, but if there are |
| 1229 | + // very large comments, the only way around it is to strip off the comment |
| 1230 | + // tags with vorbiscomment and MD5 that file. |
| 1231 | + // This procedure must be applied to ALL Ogg files, not just the ones with |
| 1232 | + // comments larger than 1 page, because the below method simply MD5's the |
| 1233 | + // whole file with the comments stripped, not just the portion after the |
| 1234 | + // comments block (which is the standard getID3() method. |
| 1235 | + |
| 1236 | + // The above-mentioned problem of comments spanning multiple pages and changing |
| 1237 | + // page sequence numbers likely happens for OggSpeex and OggFLAC as well, but |
| 1238 | + // currently vorbiscomment only works on OggVorbis files. |
| 1239 | + |
| 1240 | + if (preg_match('#(1|ON)#i', ini_get('safe_mode'))) { |
| 1241 | + |
| 1242 | + $this->info['warning'][] = 'Failed making system call to vorbiscomment.exe - '.$algorithm.'_data is incorrect - error returned: PHP running in Safe Mode (backtick operator not available)'; |
| 1243 | + $this->info[$algorithm.'_data'] = false; |
| 1244 | + |
| 1245 | + } else { |
| 1246 | + |
| 1247 | + // Prevent user from aborting script |
| 1248 | + $old_abort = ignore_user_abort(true); |
| 1249 | + |
| 1250 | + // Create empty file |
| 1251 | + $empty = tempnam(GETID3_TEMP_DIR, 'getID3'); |
| 1252 | + touch($empty); |
| 1253 | + |
| 1254 | + // Use vorbiscomment to make temp file without comments |
| 1255 | + $temp = tempnam(GETID3_TEMP_DIR, 'getID3'); |
| 1256 | + $file = $this->info['filenamepath']; |
| 1257 | + |
| 1258 | + if (GETID3_OS_ISWINDOWS) { |
| 1259 | + |
| 1260 | + if (file_exists(GETID3_HELPERAPPSDIR.'vorbiscomment.exe')) { |
| 1261 | + |
| 1262 | + $commandline = '"'.GETID3_HELPERAPPSDIR.'vorbiscomment.exe" -w -c "'.$empty.'" "'.$file.'" "'.$temp.'"'; |
| 1263 | + $VorbisCommentError = `$commandline`; |
| 1264 | + |
| 1265 | + } else { |
| 1266 | + |
| 1267 | + $VorbisCommentError = 'vorbiscomment.exe not found in '.GETID3_HELPERAPPSDIR; |
| 1268 | + |
| 1269 | + } |
| 1270 | + |
| 1271 | + } else { |
| 1272 | + |
| 1273 | + $commandline = 'vorbiscomment -w -c "'.$empty.'" "'.$file.'" "'.$temp.'" 2>&1'; |
| 1274 | + $commandline = 'vorbiscomment -w -c '.escapeshellarg($empty).' '.escapeshellarg($file).' '.escapeshellarg($temp).' 2>&1'; |
| 1275 | + $VorbisCommentError = `$commandline`; |
| 1276 | + |
| 1277 | + } |
| 1278 | + |
| 1279 | + if (!empty($VorbisCommentError)) { |
| 1280 | + |
| 1281 | + $this->info['warning'][] = 'Failed making system call to vorbiscomment(.exe) - '.$algorithm.'_data will be incorrect. If vorbiscomment is unavailable, please download from http://www.vorbis.com/download.psp and put in the getID3() directory. Error returned: '.$VorbisCommentError; |
| 1282 | + $this->info[$algorithm.'_data'] = false; |
| 1283 | + |
| 1284 | + } else { |
| 1285 | + |
| 1286 | + // Get hash of newly created file |
| 1287 | + switch ($algorithm) { |
| 1288 | + case 'md5': |
| 1289 | + $this->info[$algorithm.'_data'] = getid3_lib::md5_file($temp); |
| 1290 | + break; |
| 1291 | + |
| 1292 | + case 'sha1': |
| 1293 | + $this->info[$algorithm.'_data'] = getid3_lib::sha1_file($temp); |
| 1294 | + break; |
| 1295 | + } |
| 1296 | + } |
| 1297 | + |
| 1298 | + // Clean up |
| 1299 | + unlink($empty); |
| 1300 | + unlink($temp); |
| 1301 | + |
| 1302 | + // Reset abort setting |
| 1303 | + ignore_user_abort($old_abort); |
| 1304 | + |
| 1305 | + } |
| 1306 | + |
| 1307 | + } else { |
| 1308 | + |
| 1309 | + if (!empty($this->info['avdataoffset']) || (isset($this->info['avdataend']) && ($this->info['avdataend'] < $this->info['filesize']))) { |
| 1310 | + |
| 1311 | + // get hash from part of file |
| 1312 | + $this->info[$algorithm.'_data'] = getid3_lib::hash_data($this->info['filenamepath'], $this->info['avdataoffset'], $this->info['avdataend'], $algorithm); |
| 1313 | + |
| 1314 | + } else { |
| 1315 | + |
| 1316 | + // get hash from whole file |
| 1317 | + switch ($algorithm) { |
| 1318 | + case 'md5': |
| 1319 | + $this->info[$algorithm.'_data'] = getid3_lib::md5_file($this->info['filenamepath']); |
| 1320 | + break; |
| 1321 | + |
| 1322 | + case 'sha1': |
| 1323 | + $this->info[$algorithm.'_data'] = getid3_lib::sha1_file($this->info['filenamepath']); |
| 1324 | + break; |
| 1325 | + } |
| 1326 | + } |
| 1327 | + |
| 1328 | + } |
| 1329 | + return true; |
| 1330 | + } |
| 1331 | + |
| 1332 | + |
| 1333 | + function ChannelsBitratePlaytimeCalculations() { |
| 1334 | + |
| 1335 | + // set channelmode on audio |
| 1336 | + if (!isset($this->info['audio']['channels'])) { |
| 1337 | + // ignore |
| 1338 | + } elseif ($this->info['audio']['channels'] == 1) { |
| 1339 | + $this->info['audio']['channelmode'] = 'mono'; |
| 1340 | + } elseif ($this->info['audio']['channels'] == 2) { |
| 1341 | + $this->info['audio']['channelmode'] = 'stereo'; |
| 1342 | + } |
| 1343 | + |
| 1344 | + // Calculate combined bitrate - audio + video |
| 1345 | + $CombinedBitrate = 0; |
| 1346 | + $CombinedBitrate += (isset($this->info['audio']['bitrate']) ? $this->info['audio']['bitrate'] : 0); |
| 1347 | + $CombinedBitrate += (isset($this->info['video']['bitrate']) ? $this->info['video']['bitrate'] : 0); |
| 1348 | + if (($CombinedBitrate > 0) && empty($this->info['bitrate'])) { |
| 1349 | + $this->info['bitrate'] = $CombinedBitrate; |
| 1350 | + } |
| 1351 | + //if ((isset($this->info['video']) && !isset($this->info['video']['bitrate'])) || (isset($this->info['audio']) && !isset($this->info['audio']['bitrate']))) { |
| 1352 | + // // for example, VBR MPEG video files cannot determine video bitrate: |
| 1353 | + // // should not set overall bitrate and playtime from audio bitrate only |
| 1354 | + // unset($this->info['bitrate']); |
| 1355 | + //} |
| 1356 | + |
| 1357 | + // video bitrate undetermined, but calculable |
| 1358 | + if (isset($this->info['video']['dataformat']) && $this->info['video']['dataformat'] && (!isset($this->info['video']['bitrate']) || ($this->info['video']['bitrate'] == 0))) { |
| 1359 | + // if video bitrate not set |
| 1360 | + if (isset($this->info['audio']['bitrate']) && ($this->info['audio']['bitrate'] > 0) && ($this->info['audio']['bitrate'] == $this->info['bitrate'])) { |
| 1361 | + // AND if audio bitrate is set to same as overall bitrate |
| 1362 | + if (isset($this->info['playtime_seconds']) && ($this->info['playtime_seconds'] > 0)) { |
| 1363 | + // AND if playtime is set |
| 1364 | + if (isset($this->info['avdataend']) && isset($this->info['avdataoffset'])) { |
| 1365 | + // AND if AV data offset start/end is known |
| 1366 | + // THEN we can calculate the video bitrate |
| 1367 | + $this->info['bitrate'] = round((($this->info['avdataend'] - $this->info['avdataoffset']) * 8) / $this->info['playtime_seconds']); |
| 1368 | + $this->info['video']['bitrate'] = $this->info['bitrate'] - $this->info['audio']['bitrate']; |
| 1369 | + } |
| 1370 | + } |
| 1371 | + } |
| 1372 | + } |
| 1373 | + |
| 1374 | + if ((!isset($this->info['playtime_seconds']) || ($this->info['playtime_seconds'] <= 0)) && !empty($this->info['bitrate'])) { |
| 1375 | + $this->info['playtime_seconds'] = (($this->info['avdataend'] - $this->info['avdataoffset']) * 8) / $this->info['bitrate']; |
| 1376 | + } |
| 1377 | + |
| 1378 | + if (!isset($this->info['bitrate']) && !empty($this->info['playtime_seconds'])) { |
| 1379 | + $this->info['bitrate'] = (($this->info['avdataend'] - $this->info['avdataoffset']) * 8) / $this->info['playtime_seconds']; |
| 1380 | + } |
| 1381 | + if (isset($this->info['bitrate']) && empty($this->info['audio']['bitrate']) && empty($this->info['video']['bitrate'])) { |
| 1382 | + if (isset($this->info['audio']['dataformat']) && empty($this->info['video']['resolution_x'])) { |
| 1383 | + // audio only |
| 1384 | + $this->info['audio']['bitrate'] = $this->info['bitrate']; |
| 1385 | + } elseif (isset($this->info['video']['resolution_x']) && empty($this->info['audio']['dataformat'])) { |
| 1386 | + // video only |
| 1387 | + $this->info['video']['bitrate'] = $this->info['bitrate']; |
| 1388 | + } |
| 1389 | + } |
| 1390 | + |
| 1391 | + // Set playtime string |
| 1392 | + if (!empty($this->info['playtime_seconds']) && empty($this->info['playtime_string'])) { |
| 1393 | + $this->info['playtime_string'] = getid3_lib::PlaytimeString($this->info['playtime_seconds']); |
| 1394 | + } |
| 1395 | + } |
| 1396 | + |
| 1397 | + |
| 1398 | + function CalculateCompressionRatioVideo() { |
| 1399 | + if (empty($this->info['video'])) { |
| 1400 | + return false; |
| 1401 | + } |
| 1402 | + if (empty($this->info['video']['resolution_x']) || empty($this->info['video']['resolution_y'])) { |
| 1403 | + return false; |
| 1404 | + } |
| 1405 | + if (empty($this->info['video']['bits_per_sample'])) { |
| 1406 | + return false; |
| 1407 | + } |
| 1408 | + |
| 1409 | + switch ($this->info['video']['dataformat']) { |
| 1410 | + case 'bmp': |
| 1411 | + case 'gif': |
| 1412 | + case 'jpeg': |
| 1413 | + case 'jpg': |
| 1414 | + case 'png': |
| 1415 | + case 'tiff': |
| 1416 | + $FrameRate = 1; |
| 1417 | + $PlaytimeSeconds = 1; |
| 1418 | + $BitrateCompressed = $this->info['filesize'] * 8; |
| 1419 | + break; |
| 1420 | + |
| 1421 | + default: |
| 1422 | + if (!empty($this->info['video']['frame_rate'])) { |
| 1423 | + $FrameRate = $this->info['video']['frame_rate']; |
| 1424 | + } else { |
| 1425 | + return false; |
| 1426 | + } |
| 1427 | + if (!empty($this->info['playtime_seconds'])) { |
| 1428 | + $PlaytimeSeconds = $this->info['playtime_seconds']; |
| 1429 | + } else { |
| 1430 | + return false; |
| 1431 | + } |
| 1432 | + if (!empty($this->info['video']['bitrate'])) { |
| 1433 | + $BitrateCompressed = $this->info['video']['bitrate']; |
| 1434 | + } else { |
| 1435 | + return false; |
| 1436 | + } |
| 1437 | + break; |
| 1438 | + } |
| 1439 | + $BitrateUncompressed = $this->info['video']['resolution_x'] * $this->info['video']['resolution_y'] * $this->info['video']['bits_per_sample'] * $FrameRate; |
| 1440 | + |
| 1441 | + $this->info['video']['compression_ratio'] = $BitrateCompressed / $BitrateUncompressed; |
| 1442 | + return true; |
| 1443 | + } |
| 1444 | + |
| 1445 | + |
| 1446 | + function CalculateCompressionRatioAudio() { |
| 1447 | + if (empty($this->info['audio']['bitrate']) || empty($this->info['audio']['channels']) || empty($this->info['audio']['sample_rate'])) { |
| 1448 | + return false; |
| 1449 | + } |
| 1450 | + $this->info['audio']['compression_ratio'] = $this->info['audio']['bitrate'] / ($this->info['audio']['channels'] * $this->info['audio']['sample_rate'] * (!empty($this->info['audio']['bits_per_sample']) ? $this->info['audio']['bits_per_sample'] : 16)); |
| 1451 | + |
| 1452 | + if (!empty($this->info['audio']['streams'])) { |
| 1453 | + foreach ($this->info['audio']['streams'] as $streamnumber => $streamdata) { |
| 1454 | + if (!empty($streamdata['bitrate']) && !empty($streamdata['channels']) && !empty($streamdata['sample_rate'])) { |
| 1455 | + $this->info['audio']['streams'][$streamnumber]['compression_ratio'] = $streamdata['bitrate'] / ($streamdata['channels'] * $streamdata['sample_rate'] * (!empty($streamdata['bits_per_sample']) ? $streamdata['bits_per_sample'] : 16)); |
| 1456 | + } |
| 1457 | + } |
| 1458 | + } |
| 1459 | + return true; |
| 1460 | + } |
| 1461 | + |
| 1462 | + |
| 1463 | + function CalculateReplayGain() { |
| 1464 | + if (isset($this->info['replay_gain'])) { |
| 1465 | + $this->info['replay_gain']['reference_volume'] = 89; |
| 1466 | + if (isset($this->info['replay_gain']['track']['adjustment'])) { |
| 1467 | + $this->info['replay_gain']['track']['volume'] = $this->info['replay_gain']['reference_volume'] - $this->info['replay_gain']['track']['adjustment']; |
| 1468 | + } |
| 1469 | + if (isset($this->info['replay_gain']['album']['adjustment'])) { |
| 1470 | + $this->info['replay_gain']['album']['volume'] = $this->info['replay_gain']['reference_volume'] - $this->info['replay_gain']['album']['adjustment']; |
| 1471 | + } |
| 1472 | + |
| 1473 | + if (isset($this->info['replay_gain']['track']['peak'])) { |
| 1474 | + $this->info['replay_gain']['track']['max_noclip_gain'] = 0 - getid3_lib::RGADamplitude2dB($this->info['replay_gain']['track']['peak']); |
| 1475 | + } |
| 1476 | + if (isset($this->info['replay_gain']['album']['peak'])) { |
| 1477 | + $this->info['replay_gain']['album']['max_noclip_gain'] = 0 - getid3_lib::RGADamplitude2dB($this->info['replay_gain']['album']['peak']); |
| 1478 | + } |
| 1479 | + } |
| 1480 | + return true; |
| 1481 | + } |
| 1482 | + |
| 1483 | + function ProcessAudioStreams() { |
| 1484 | + if (!empty($this->info['audio']['bitrate']) || !empty($this->info['audio']['channels']) || !empty($this->info['audio']['sample_rate'])) { |
| 1485 | + if (!isset($this->info['audio']['streams'])) { |
| 1486 | + foreach ($this->info['audio'] as $key => $value) { |
| 1487 | + if ($key != 'streams') { |
| 1488 | + $this->info['audio']['streams'][0][$key] = $value; |
| 1489 | + } |
| 1490 | + } |
| 1491 | + } |
| 1492 | + } |
| 1493 | + return true; |
| 1494 | + } |
| 1495 | + |
| 1496 | + function getid3_tempnam() { |
| 1497 | + return tempnam($this->tempdir, 'gI3'); |
| 1498 | + } |
| 1499 | + |
| 1500 | +} |
| 1501 | + |
| 1502 | +?> |
\ No newline at end of file |
Index: trunk/extensions/TimedMediaHandler/getid3/module.audio.ogg.php |
— | — | @@ -0,0 +1,565 @@ |
| 2 | +<?php |
| 3 | +///////////////////////////////////////////////////////////////// |
| 4 | +/// getID3() by James Heinrich <info@getid3.org> // |
| 5 | +// available at http://getid3.sourceforge.net // |
| 6 | +// or http://www.getid3.org // |
| 7 | +///////////////////////////////////////////////////////////////// |
| 8 | +// See readme.txt for more details // |
| 9 | +///////////////////////////////////////////////////////////////// |
| 10 | +// // |
| 11 | +// module.audio.ogg.php // |
| 12 | +// module for analyzing Ogg Vorbis, OggFLAC and Speex files // |
| 13 | +// dependencies: module.audio.flac.php // |
| 14 | +// /// |
| 15 | +///////////////////////////////////////////////////////////////// |
| 16 | + |
| 17 | +getid3_lib::IncludeDependency(GETID3_INCLUDEPATH.'module.audio.flac.php', __FILE__, true); |
| 18 | + |
| 19 | +class getid3_ogg |
| 20 | +{ |
| 21 | + |
| 22 | + function getid3_ogg(&$fd, &$ThisFileInfo) { |
| 23 | + |
| 24 | + $ThisFileInfo['fileformat'] = 'ogg'; |
| 25 | + |
| 26 | + // Warn about illegal tags - only vorbiscomments are allowed |
| 27 | + if (isset($ThisFileInfo['id3v2'])) { |
| 28 | + $ThisFileInfo['warning'][] = 'Illegal ID3v2 tag present.'; |
| 29 | + } |
| 30 | + if (isset($ThisFileInfo['id3v1'])) { |
| 31 | + $ThisFileInfo['warning'][] = 'Illegal ID3v1 tag present.'; |
| 32 | + } |
| 33 | + if (isset($ThisFileInfo['ape'])) { |
| 34 | + $ThisFileInfo['warning'][] = 'Illegal APE tag present.'; |
| 35 | + } |
| 36 | + |
| 37 | + |
| 38 | + // Page 1 - Stream Header |
| 39 | + |
| 40 | + fseek($fd, $ThisFileInfo['avdataoffset'], SEEK_SET); |
| 41 | + |
| 42 | + $oggpageinfo = getid3_ogg::ParseOggPageHeader($fd); |
| 43 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']] = $oggpageinfo; |
| 44 | + |
| 45 | + if (ftell($fd) >= GETID3_FREAD_BUFFER_SIZE) { |
| 46 | + $ThisFileInfo['error'][] = 'Could not find start of Ogg page in the first '.GETID3_FREAD_BUFFER_SIZE.' bytes (this might not be an Ogg-Vorbis file?)'; |
| 47 | + unset($ThisFileInfo['fileformat']); |
| 48 | + unset($ThisFileInfo['ogg']); |
| 49 | + return false; |
| 50 | + } |
| 51 | + |
| 52 | + $filedata = fread($fd, $oggpageinfo['page_length']); |
| 53 | + $filedataoffset = 0; |
| 54 | + |
| 55 | + if (substr($filedata, 0, 4) == 'fLaC') { |
| 56 | + |
| 57 | + $ThisFileInfo['audio']['dataformat'] = 'flac'; |
| 58 | + $ThisFileInfo['audio']['bitrate_mode'] = 'vbr'; |
| 59 | + $ThisFileInfo['audio']['lossless'] = true; |
| 60 | + |
| 61 | + } elseif (substr($filedata, 1, 6) == 'vorbis') { |
| 62 | + |
| 63 | + $this->ParseVorbisPageHeader($filedata, $filedataoffset, $ThisFileInfo, $oggpageinfo); |
| 64 | + |
| 65 | + } elseif (substr($filedata, 0, 8) == 'Speex ') { |
| 66 | + |
| 67 | + // http://www.speex.org/manual/node10.html |
| 68 | + |
| 69 | + $ThisFileInfo['audio']['dataformat'] = 'speex'; |
| 70 | + $ThisFileInfo['mime_type'] = 'audio/speex'; |
| 71 | + $ThisFileInfo['audio']['bitrate_mode'] = 'abr'; |
| 72 | + $ThisFileInfo['audio']['lossless'] = false; |
| 73 | + |
| 74 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['speex_string'] = substr($filedata, $filedataoffset, 8); // hard-coded to 'Speex ' |
| 75 | + $filedataoffset += 8; |
| 76 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['speex_version'] = substr($filedata, $filedataoffset, 20); |
| 77 | + $filedataoffset += 20; |
| 78 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['speex_version_id'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 4)); |
| 79 | + $filedataoffset += 4; |
| 80 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['header_size'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 4)); |
| 81 | + $filedataoffset += 4; |
| 82 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['rate'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 4)); |
| 83 | + $filedataoffset += 4; |
| 84 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['mode'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 4)); |
| 85 | + $filedataoffset += 4; |
| 86 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['mode_bitstream_version'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 4)); |
| 87 | + $filedataoffset += 4; |
| 88 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['nb_channels'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 4)); |
| 89 | + $filedataoffset += 4; |
| 90 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['bitrate'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 4)); |
| 91 | + $filedataoffset += 4; |
| 92 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['framesize'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 4)); |
| 93 | + $filedataoffset += 4; |
| 94 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['vbr'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 4)); |
| 95 | + $filedataoffset += 4; |
| 96 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['frames_per_packet'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 4)); |
| 97 | + $filedataoffset += 4; |
| 98 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['extra_headers'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 4)); |
| 99 | + $filedataoffset += 4; |
| 100 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['reserved1'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 4)); |
| 101 | + $filedataoffset += 4; |
| 102 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['reserved2'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 4)); |
| 103 | + $filedataoffset += 4; |
| 104 | + |
| 105 | + $ThisFileInfo['speex']['speex_version'] = trim($ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['speex_version']); |
| 106 | + $ThisFileInfo['speex']['sample_rate'] = $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['rate']; |
| 107 | + $ThisFileInfo['speex']['channels'] = $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['nb_channels']; |
| 108 | + $ThisFileInfo['speex']['vbr'] = (bool) $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['vbr']; |
| 109 | + $ThisFileInfo['speex']['band_type'] = getid3_ogg::SpeexBandModeLookup($ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['mode']); |
| 110 | + |
| 111 | + $ThisFileInfo['audio']['sample_rate'] = $ThisFileInfo['speex']['sample_rate']; |
| 112 | + $ThisFileInfo['audio']['channels'] = $ThisFileInfo['speex']['channels']; |
| 113 | + if ($ThisFileInfo['speex']['vbr']) { |
| 114 | + $ThisFileInfo['audio']['bitrate_mode'] = 'vbr'; |
| 115 | + } |
| 116 | + |
| 117 | + } else { |
| 118 | + |
| 119 | + $ThisFileInfo['error'][] = 'Expecting either "Speex " or "vorbis" identifier strings, found neither'; |
| 120 | + unset($ThisFileInfo['ogg']); |
| 121 | + unset($ThisFileInfo['mime_type']); |
| 122 | + return false; |
| 123 | + |
| 124 | + } |
| 125 | + |
| 126 | + |
| 127 | + // Page 2 - Comment Header |
| 128 | + |
| 129 | + $oggpageinfo = getid3_ogg::ParseOggPageHeader($fd); |
| 130 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']] = $oggpageinfo; |
| 131 | + |
| 132 | + switch ($ThisFileInfo['audio']['dataformat']) { |
| 133 | + |
| 134 | + case 'vorbis': |
| 135 | + $filedata = fread($fd, $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['page_length']); |
| 136 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['packet_type'] = getid3_lib::LittleEndian2Int(substr($filedata, 0, 1)); |
| 137 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['stream_type'] = substr($filedata, 1, 6); // hard-coded to 'vorbis' |
| 138 | + |
| 139 | + getid3_ogg::ParseVorbisCommentsFilepointer($fd, $ThisFileInfo); |
| 140 | + break; |
| 141 | + |
| 142 | + case 'flac': |
| 143 | + if (!getid3_flac::FLACparseMETAdata($fd, $ThisFileInfo)) { |
| 144 | + $ThisFileInfo['error'][] = 'Failed to parse FLAC headers'; |
| 145 | + return false; |
| 146 | + } |
| 147 | + break; |
| 148 | + |
| 149 | + case 'speex': |
| 150 | + fseek($fd, $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['page_length'], SEEK_CUR); |
| 151 | + getid3_ogg::ParseVorbisCommentsFilepointer($fd, $ThisFileInfo); |
| 152 | + break; |
| 153 | + |
| 154 | + } |
| 155 | + |
| 156 | + |
| 157 | + |
| 158 | + // Last Page - Number of Samples |
| 159 | + |
| 160 | + if ($ThisFileInfo['avdataend'] >= pow(2, 31)) { |
| 161 | + |
| 162 | + $ThisFileInfo['warning'][] = 'Unable to parse Ogg end chunk file (PHP does not support file operations beyond 2GB)'; |
| 163 | + |
| 164 | + } else { |
| 165 | + |
| 166 | + fseek($fd, max($ThisFileInfo['avdataend'] - GETID3_FREAD_BUFFER_SIZE, 0), SEEK_SET); |
| 167 | + $LastChunkOfOgg = strrev(fread($fd, GETID3_FREAD_BUFFER_SIZE)); |
| 168 | + if ($LastOggSpostion = strpos($LastChunkOfOgg, 'SggO')) { |
| 169 | + fseek($fd, $ThisFileInfo['avdataend'] - ($LastOggSpostion + strlen('SggO')), SEEK_SET); |
| 170 | + $ThisFileInfo['avdataend'] = ftell($fd); |
| 171 | + $ThisFileInfo['ogg']['pageheader']['eos'] = getid3_ogg::ParseOggPageHeader($fd); |
| 172 | + $ThisFileInfo['ogg']['samples'] = $ThisFileInfo['ogg']['pageheader']['eos']['pcm_abs_position']; |
| 173 | + if ($ThisFileInfo['ogg']['samples'] == 0) { |
| 174 | + $ThisFileInfo['error'][] = 'Corrupt Ogg file: eos.number of samples == zero'; |
| 175 | + return false; |
| 176 | + } |
| 177 | + $ThisFileInfo['ogg']['bitrate_average'] = (($ThisFileInfo['avdataend'] - $ThisFileInfo['avdataoffset']) * 8) / ($ThisFileInfo['ogg']['samples'] / $ThisFileInfo['audio']['sample_rate']); |
| 178 | + } |
| 179 | + |
| 180 | + } |
| 181 | + |
| 182 | + if (!empty($ThisFileInfo['ogg']['bitrate_average'])) { |
| 183 | + $ThisFileInfo['audio']['bitrate'] = $ThisFileInfo['ogg']['bitrate_average']; |
| 184 | + } elseif (!empty($ThisFileInfo['ogg']['bitrate_nominal'])) { |
| 185 | + $ThisFileInfo['audio']['bitrate'] = $ThisFileInfo['ogg']['bitrate_nominal']; |
| 186 | + } elseif (!empty($ThisFileInfo['ogg']['bitrate_min']) && !empty($ThisFileInfo['ogg']['bitrate_max'])) { |
| 187 | + $ThisFileInfo['audio']['bitrate'] = ($ThisFileInfo['ogg']['bitrate_min'] + $ThisFileInfo['ogg']['bitrate_max']) / 2; |
| 188 | + } |
| 189 | + if (isset($ThisFileInfo['audio']['bitrate']) && !isset($ThisFileInfo['playtime_seconds'])) { |
| 190 | + if ($ThisFileInfo['audio']['bitrate'] == 0) { |
| 191 | + $ThisFileInfo['error'][] = 'Corrupt Ogg file: bitrate_audio == zero'; |
| 192 | + return false; |
| 193 | + } |
| 194 | + $ThisFileInfo['playtime_seconds'] = (float) ((($ThisFileInfo['avdataend'] - $ThisFileInfo['avdataoffset']) * 8) / $ThisFileInfo['audio']['bitrate']); |
| 195 | + } |
| 196 | + |
| 197 | + if (isset($ThisFileInfo['ogg']['vendor'])) { |
| 198 | + $ThisFileInfo['audio']['encoder'] = preg_replace('/^Encoded with /', '', $ThisFileInfo['ogg']['vendor']); |
| 199 | + |
| 200 | + // Vorbis only |
| 201 | + if ($ThisFileInfo['audio']['dataformat'] == 'vorbis') { |
| 202 | + |
| 203 | + // Vorbis 1.0 starts with Xiph.Org |
| 204 | + if (preg_match('/^Xiph.Org/', $ThisFileInfo['audio']['encoder'])) { |
| 205 | + |
| 206 | + if ($ThisFileInfo['audio']['bitrate_mode'] == 'abr') { |
| 207 | + |
| 208 | + // Set -b 128 on abr files |
| 209 | + $ThisFileInfo['audio']['encoder_options'] = '-b '.round($ThisFileInfo['ogg']['bitrate_nominal'] / 1000); |
| 210 | + |
| 211 | + } elseif (($ThisFileInfo['audio']['bitrate_mode'] == 'vbr') && ($ThisFileInfo['audio']['channels'] == 2) && ($ThisFileInfo['audio']['sample_rate'] >= 44100) && ($ThisFileInfo['audio']['sample_rate'] <= 48000)) { |
| 212 | + // Set -q N on vbr files |
| 213 | + $ThisFileInfo['audio']['encoder_options'] = '-q '.$this->get_quality_from_nominal_bitrate($ThisFileInfo['ogg']['bitrate_nominal']); |
| 214 | + |
| 215 | + } |
| 216 | + } |
| 217 | + |
| 218 | + if (empty($ThisFileInfo['audio']['encoder_options']) && !empty($ThisFileInfo['ogg']['bitrate_nominal'])) { |
| 219 | + $ThisFileInfo['audio']['encoder_options'] = 'Nominal bitrate: '.intval(round($ThisFileInfo['ogg']['bitrate_nominal'] / 1000)).'kbps'; |
| 220 | + } |
| 221 | + } |
| 222 | + } |
| 223 | + |
| 224 | + return true; |
| 225 | + } |
| 226 | + |
| 227 | + static function ParseVorbisPageHeader(&$filedata, &$filedataoffset, &$ThisFileInfo, &$oggpageinfo) { |
| 228 | + $ThisFileInfo['audio']['dataformat'] = 'vorbis'; |
| 229 | + $ThisFileInfo['audio']['lossless'] = false; |
| 230 | + |
| 231 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['packet_type'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 1)); |
| 232 | + $filedataoffset += 1; |
| 233 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['stream_type'] = substr($filedata, $filedataoffset, 6); // hard-coded to 'vorbis' |
| 234 | + $filedataoffset += 6; |
| 235 | + $ThisFileInfo['ogg']['bitstreamversion'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 4)); |
| 236 | + $filedataoffset += 4; |
| 237 | + $ThisFileInfo['ogg']['numberofchannels'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 1)); |
| 238 | + $filedataoffset += 1; |
| 239 | + $ThisFileInfo['audio']['channels'] = $ThisFileInfo['ogg']['numberofchannels']; |
| 240 | + $ThisFileInfo['ogg']['samplerate'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 4)); |
| 241 | + $filedataoffset += 4; |
| 242 | + if ($ThisFileInfo['ogg']['samplerate'] == 0) { |
| 243 | + $ThisFileInfo['error'][] = 'Corrupt Ogg file: sample rate == zero'; |
| 244 | + return false; |
| 245 | + } |
| 246 | + $ThisFileInfo['audio']['sample_rate'] = $ThisFileInfo['ogg']['samplerate']; |
| 247 | + $ThisFileInfo['ogg']['samples'] = 0; // filled in later |
| 248 | + $ThisFileInfo['ogg']['bitrate_average'] = 0; // filled in later |
| 249 | + $ThisFileInfo['ogg']['bitrate_max'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 4)); |
| 250 | + $filedataoffset += 4; |
| 251 | + $ThisFileInfo['ogg']['bitrate_nominal'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 4)); |
| 252 | + $filedataoffset += 4; |
| 253 | + $ThisFileInfo['ogg']['bitrate_min'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 4)); |
| 254 | + $filedataoffset += 4; |
| 255 | + $ThisFileInfo['ogg']['blocksize_small'] = pow(2, getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 1)) & 0x0F); |
| 256 | + $ThisFileInfo['ogg']['blocksize_large'] = pow(2, (getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 1)) & 0xF0) >> 4); |
| 257 | + $ThisFileInfo['ogg']['stop_bit'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 1)); // must be 1, marks end of packet |
| 258 | + |
| 259 | + $ThisFileInfo['audio']['bitrate_mode'] = 'vbr'; // overridden if actually abr |
| 260 | + if ($ThisFileInfo['ogg']['bitrate_max'] == 0xFFFFFFFF) { |
| 261 | + unset($ThisFileInfo['ogg']['bitrate_max']); |
| 262 | + $ThisFileInfo['audio']['bitrate_mode'] = 'abr'; |
| 263 | + } |
| 264 | + if ($ThisFileInfo['ogg']['bitrate_nominal'] == 0xFFFFFFFF) { |
| 265 | + unset($ThisFileInfo['ogg']['bitrate_nominal']); |
| 266 | + } |
| 267 | + if ($ThisFileInfo['ogg']['bitrate_min'] == 0xFFFFFFFF) { |
| 268 | + unset($ThisFileInfo['ogg']['bitrate_min']); |
| 269 | + $ThisFileInfo['audio']['bitrate_mode'] = 'abr'; |
| 270 | + } |
| 271 | + return true; |
| 272 | + } |
| 273 | + |
| 274 | + static function ParseOggPageHeader(&$fd) { |
| 275 | + // http://xiph.org/ogg/vorbis/doc/framing.html |
| 276 | + $oggheader['page_start_offset'] = ftell($fd); // where we started from in the file |
| 277 | + |
| 278 | + $filedata = fread($fd, GETID3_FREAD_BUFFER_SIZE); |
| 279 | + $filedataoffset = 0; |
| 280 | + while ((substr($filedata, $filedataoffset++, 4) != 'OggS')) { |
| 281 | + if ((ftell($fd) - $oggheader['page_start_offset']) >= GETID3_FREAD_BUFFER_SIZE) { |
| 282 | + // should be found before here |
| 283 | + return false; |
| 284 | + } |
| 285 | + if ((($filedataoffset + 28) > strlen($filedata)) || (strlen($filedata) < 28)) { |
| 286 | + if (feof($fd) || (($filedata .= fread($fd, GETID3_FREAD_BUFFER_SIZE)) === false)) { |
| 287 | + // get some more data, unless eof, in which case fail |
| 288 | + return false; |
| 289 | + } |
| 290 | + } |
| 291 | + } |
| 292 | + $filedataoffset += strlen('OggS') - 1; // page, delimited by 'OggS' |
| 293 | + |
| 294 | + $oggheader['stream_structver'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 1)); |
| 295 | + $filedataoffset += 1; |
| 296 | + $oggheader['flags_raw'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 1)); |
| 297 | + $filedataoffset += 1; |
| 298 | + $oggheader['flags']['fresh'] = (bool) ($oggheader['flags_raw'] & 0x01); // fresh packet |
| 299 | + $oggheader['flags']['bos'] = (bool) ($oggheader['flags_raw'] & 0x02); // first page of logical bitstream (bos) |
| 300 | + $oggheader['flags']['eos'] = (bool) ($oggheader['flags_raw'] & 0x04); // last page of logical bitstream (eos) |
| 301 | + |
| 302 | + $oggheader['pcm_abs_position'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 8)); |
| 303 | + $filedataoffset += 8; |
| 304 | + $oggheader['stream_serialno'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 4)); |
| 305 | + $filedataoffset += 4; |
| 306 | + $oggheader['page_seqno'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 4)); |
| 307 | + $filedataoffset += 4; |
| 308 | + $oggheader['page_checksum'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 4)); |
| 309 | + $filedataoffset += 4; |
| 310 | + $oggheader['page_segments'] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 1)); |
| 311 | + $filedataoffset += 1; |
| 312 | + $oggheader['page_length'] = 0; |
| 313 | + for ($i = 0; $i < $oggheader['page_segments']; $i++) { |
| 314 | + $oggheader['segment_table'][$i] = getid3_lib::LittleEndian2Int(substr($filedata, $filedataoffset, 1)); |
| 315 | + $filedataoffset += 1; |
| 316 | + $oggheader['page_length'] += $oggheader['segment_table'][$i]; |
| 317 | + } |
| 318 | + $oggheader['header_end_offset'] = $oggheader['page_start_offset'] + $filedataoffset; |
| 319 | + $oggheader['page_end_offset'] = $oggheader['header_end_offset'] + $oggheader['page_length']; |
| 320 | + fseek($fd, $oggheader['header_end_offset'], SEEK_SET); |
| 321 | + |
| 322 | + return $oggheader; |
| 323 | + } |
| 324 | + |
| 325 | + |
| 326 | + static function ParseVorbisCommentsFilepointer(&$fd, &$ThisFileInfo) { |
| 327 | + |
| 328 | + $OriginalOffset = ftell($fd); |
| 329 | + $CommentStartOffset = $OriginalOffset; |
| 330 | + $commentdataoffset = 0; |
| 331 | + $VorbisCommentPage = 1; |
| 332 | + |
| 333 | + switch ($ThisFileInfo['audio']['dataformat']) { |
| 334 | + case 'vorbis': |
| 335 | + $CommentStartOffset = $ThisFileInfo['ogg']['pageheader'][$VorbisCommentPage]['page_start_offset']; // Second Ogg page, after header block |
| 336 | + fseek($fd, $CommentStartOffset, SEEK_SET); |
| 337 | + $commentdataoffset = 27 + $ThisFileInfo['ogg']['pageheader'][$VorbisCommentPage]['page_segments']; |
| 338 | + $commentdata = fread($fd, getid3_ogg::OggPageSegmentLength($ThisFileInfo['ogg']['pageheader'][$VorbisCommentPage], 1) + $commentdataoffset); |
| 339 | + |
| 340 | + $commentdataoffset += (strlen('vorbis') + 1); |
| 341 | + break; |
| 342 | + |
| 343 | + case 'flac': |
| 344 | + fseek($fd, $ThisFileInfo['flac']['VORBIS_COMMENT']['raw']['offset'] + 4, SEEK_SET); |
| 345 | + $commentdata = fread($fd, $ThisFileInfo['flac']['VORBIS_COMMENT']['raw']['block_length']); |
| 346 | + break; |
| 347 | + |
| 348 | + case 'speex': |
| 349 | + $CommentStartOffset = $ThisFileInfo['ogg']['pageheader'][$VorbisCommentPage]['page_start_offset']; // Second Ogg page, after header block |
| 350 | + fseek($fd, $CommentStartOffset, SEEK_SET); |
| 351 | + $commentdataoffset = 27 + $ThisFileInfo['ogg']['pageheader'][$VorbisCommentPage]['page_segments']; |
| 352 | + $commentdata = fread($fd, getid3_ogg::OggPageSegmentLength($ThisFileInfo['ogg']['pageheader'][$VorbisCommentPage], 1) + $commentdataoffset); |
| 353 | + break; |
| 354 | + |
| 355 | + default: |
| 356 | + return false; |
| 357 | + break; |
| 358 | + } |
| 359 | + |
| 360 | + $VendorSize = getid3_lib::LittleEndian2Int(substr($commentdata, $commentdataoffset, 4)); |
| 361 | + $commentdataoffset += 4; |
| 362 | + |
| 363 | + $ThisFileInfo['ogg']['vendor'] = substr($commentdata, $commentdataoffset, $VendorSize); |
| 364 | + $commentdataoffset += $VendorSize; |
| 365 | + |
| 366 | + $CommentsCount = getid3_lib::LittleEndian2Int(substr($commentdata, $commentdataoffset, 4)); |
| 367 | + $commentdataoffset += 4; |
| 368 | + $ThisFileInfo['avdataoffset'] = $CommentStartOffset + $commentdataoffset; |
| 369 | + |
| 370 | + $basicfields = array('TITLE', 'ARTIST', 'ALBUM', 'TRACKNUMBER', 'GENRE', 'DATE', 'DESCRIPTION', 'COMMENT'); |
| 371 | + for ($i = 0; $i < $CommentsCount; $i++) { |
| 372 | + |
| 373 | + $ThisFileInfo['ogg']['comments_raw'][$i]['dataoffset'] = $CommentStartOffset + $commentdataoffset; |
| 374 | + |
| 375 | + if (ftell($fd) < ($ThisFileInfo['ogg']['comments_raw'][$i]['dataoffset'] + 4)) { |
| 376 | + $VorbisCommentPage++; |
| 377 | + |
| 378 | + $oggpageinfo = getid3_ogg::ParseOggPageHeader($fd); |
| 379 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']] = $oggpageinfo; |
| 380 | + |
| 381 | + // First, save what we haven't read yet |
| 382 | + $AsYetUnusedData = substr($commentdata, $commentdataoffset); |
| 383 | + |
| 384 | + // Then take that data off the end |
| 385 | + $commentdata = substr($commentdata, 0, $commentdataoffset); |
| 386 | + |
| 387 | + // Add [headerlength] bytes of dummy data for the Ogg Page Header, just to keep absolute offsets correct |
| 388 | + $commentdata .= str_repeat("\x00", 27 + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['page_segments']); |
| 389 | + $commentdataoffset += (27 + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['page_segments']); |
| 390 | + |
| 391 | + // Finally, stick the unused data back on the end |
| 392 | + $commentdata .= $AsYetUnusedData; |
| 393 | + |
| 394 | + //$commentdata .= fread($fd, $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['page_length']); |
| 395 | + $commentdata .= fread($fd, getid3_ogg::OggPageSegmentLength($ThisFileInfo['ogg']['pageheader'][$VorbisCommentPage], 1)); |
| 396 | + |
| 397 | + } |
| 398 | + $ThisFileInfo['ogg']['comments_raw'][$i]['size'] = getid3_lib::LittleEndian2Int(substr($commentdata, $commentdataoffset, 4)); |
| 399 | + |
| 400 | + // replace avdataoffset with position just after the last vorbiscomment |
| 401 | + $ThisFileInfo['avdataoffset'] = $ThisFileInfo['ogg']['comments_raw'][$i]['dataoffset'] + $ThisFileInfo['ogg']['comments_raw'][$i]['size'] + 4; |
| 402 | + |
| 403 | + $commentdataoffset += 4; |
| 404 | + while ((strlen($commentdata) - $commentdataoffset) < $ThisFileInfo['ogg']['comments_raw'][$i]['size']) { |
| 405 | + if (($ThisFileInfo['ogg']['comments_raw'][$i]['size'] > $ThisFileInfo['avdataend']) || ($ThisFileInfo['ogg']['comments_raw'][$i]['size'] < 0)) { |
| 406 | + $ThisFileInfo['warning'][] = 'Invalid Ogg comment size (comment #'.$i.', claims to be '.number_format($ThisFileInfo['ogg']['comments_raw'][$i]['size']).' bytes) - aborting reading comments'; |
| 407 | + break 2; |
| 408 | + } |
| 409 | + |
| 410 | + $VorbisCommentPage++; |
| 411 | + |
| 412 | + $oggpageinfo = getid3_ogg::ParseOggPageHeader($fd); |
| 413 | + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']] = $oggpageinfo; |
| 414 | + |
| 415 | + // First, save what we haven't read yet |
| 416 | + $AsYetUnusedData = substr($commentdata, $commentdataoffset); |
| 417 | + |
| 418 | + // Then take that data off the end |
| 419 | + $commentdata = substr($commentdata, 0, $commentdataoffset); |
| 420 | + |
| 421 | + // Add [headerlength] bytes of dummy data for the Ogg Page Header, just to keep absolute offsets correct |
| 422 | + $commentdata .= str_repeat("\x00", 27 + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['page_segments']); |
| 423 | + $commentdataoffset += (27 + $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['page_segments']); |
| 424 | + |
| 425 | + // Finally, stick the unused data back on the end |
| 426 | + $commentdata .= $AsYetUnusedData; |
| 427 | + |
| 428 | + //$commentdata .= fread($fd, $ThisFileInfo['ogg']['pageheader'][$oggpageinfo['page_seqno']]['page_length']); |
| 429 | + if (!isset($ThisFileInfo['ogg']['pageheader'][$VorbisCommentPage])) { |
| 430 | + $ThisFileInfo['warning'][] = 'undefined Vorbis Comment page "'.$VorbisCommentPage.'" at offset '.ftell($fd); |
| 431 | + break; |
| 432 | + } |
| 433 | + $readlength = getid3_ogg::OggPageSegmentLength($ThisFileInfo['ogg']['pageheader'][$VorbisCommentPage], 1); |
| 434 | + if ($readlength <= 0) { |
| 435 | + $ThisFileInfo['warning'][] = 'invalid length Vorbis Comment page "'.$VorbisCommentPage.'" at offset '.ftell($fd); |
| 436 | + break; |
| 437 | + } |
| 438 | + $commentdata .= fread($fd, $readlength); |
| 439 | + |
| 440 | + //$filebaseoffset += $oggpageinfo['header_end_offset'] - $oggpageinfo['page_start_offset']; |
| 441 | + } |
| 442 | + $commentstring = substr($commentdata, $commentdataoffset, $ThisFileInfo['ogg']['comments_raw'][$i]['size']); |
| 443 | + $commentdataoffset += $ThisFileInfo['ogg']['comments_raw'][$i]['size']; |
| 444 | + |
| 445 | + if (!$commentstring) { |
| 446 | + |
| 447 | + // no comment? |
| 448 | + $ThisFileInfo['warning'][] = 'Blank Ogg comment ['.$i.']'; |
| 449 | + |
| 450 | + } elseif (strstr($commentstring, '=')) { |
| 451 | + |
| 452 | + $commentexploded = explode('=', $commentstring, 2); |
| 453 | + $ThisFileInfo['ogg']['comments_raw'][$i]['key'] = strtoupper($commentexploded[0]); |
| 454 | + $ThisFileInfo['ogg']['comments_raw'][$i]['value'] = (isset($commentexploded[1]) ? $commentexploded[1] : ''); |
| 455 | + $ThisFileInfo['ogg']['comments_raw'][$i]['data'] = base64_decode($ThisFileInfo['ogg']['comments_raw'][$i]['value']); |
| 456 | + |
| 457 | + $ThisFileInfo['ogg']['comments'][strtolower($ThisFileInfo['ogg']['comments_raw'][$i]['key'])][] = $ThisFileInfo['ogg']['comments_raw'][$i]['value']; |
| 458 | + |
| 459 | + $imageinfo = array(); |
| 460 | + $imagechunkcheck = getid3_lib::GetDataImageSize($ThisFileInfo['ogg']['comments_raw'][$i]['data'], $imageinfo); |
| 461 | + $ThisFileInfo['ogg']['comments_raw'][$i]['image_mime'] = getid3_lib::image_type_to_mime_type($imagechunkcheck[2]); |
| 462 | + if (!$ThisFileInfo['ogg']['comments_raw'][$i]['image_mime'] || ($ThisFileInfo['ogg']['comments_raw'][$i]['image_mime'] == 'application/octet-stream')) { |
| 463 | + unset($ThisFileInfo['ogg']['comments_raw'][$i]['image_mime']); |
| 464 | + unset($ThisFileInfo['ogg']['comments_raw'][$i]['data']); |
| 465 | + } |
| 466 | + |
| 467 | + } else { |
| 468 | + |
| 469 | + $ThisFileInfo['warning'][] = '[known problem with CDex >= v1.40, < v1.50b7] Invalid Ogg comment name/value pair ['.$i.']: '.$commentstring; |
| 470 | + |
| 471 | + } |
| 472 | + } |
| 473 | + |
| 474 | + |
| 475 | + // Replay Gain Adjustment |
| 476 | + // http://privatewww.essex.ac.uk/~djmrob/replaygain/ |
| 477 | + if (isset($ThisFileInfo['ogg']['comments']) && is_array($ThisFileInfo['ogg']['comments'])) { |
| 478 | + foreach ($ThisFileInfo['ogg']['comments'] as $index => $commentvalue) { |
| 479 | + switch ($index) { |
| 480 | + case 'rg_audiophile': |
| 481 | + case 'replaygain_album_gain': |
| 482 | + $ThisFileInfo['replay_gain']['album']['adjustment'] = (double) $commentvalue[0]; |
| 483 | + unset($ThisFileInfo['ogg']['comments'][$index]); |
| 484 | + break; |
| 485 | + |
| 486 | + case 'rg_radio': |
| 487 | + case 'replaygain_track_gain': |
| 488 | + $ThisFileInfo['replay_gain']['track']['adjustment'] = (double) $commentvalue[0]; |
| 489 | + unset($ThisFileInfo['ogg']['comments'][$index]); |
| 490 | + break; |
| 491 | + |
| 492 | + case 'replaygain_album_peak': |
| 493 | + $ThisFileInfo['replay_gain']['album']['peak'] = (double) $commentvalue[0]; |
| 494 | + unset($ThisFileInfo['ogg']['comments'][$index]); |
| 495 | + break; |
| 496 | + |
| 497 | + case 'rg_peak': |
| 498 | + case 'replaygain_track_peak': |
| 499 | + $ThisFileInfo['replay_gain']['track']['peak'] = (double) $commentvalue[0]; |
| 500 | + unset($ThisFileInfo['ogg']['comments'][$index]); |
| 501 | + break; |
| 502 | + |
| 503 | + |
| 504 | + default: |
| 505 | + // do nothing |
| 506 | + break; |
| 507 | + } |
| 508 | + } |
| 509 | + } |
| 510 | + |
| 511 | + fseek($fd, $OriginalOffset, SEEK_SET); |
| 512 | + |
| 513 | + return true; |
| 514 | + } |
| 515 | + |
| 516 | + static function SpeexBandModeLookup($mode) { |
| 517 | + static $SpeexBandModeLookup = array(); |
| 518 | + if (empty($SpeexBandModeLookup)) { |
| 519 | + $SpeexBandModeLookup[0] = 'narrow'; |
| 520 | + $SpeexBandModeLookup[1] = 'wide'; |
| 521 | + $SpeexBandModeLookup[2] = 'ultra-wide'; |
| 522 | + } |
| 523 | + return (isset($SpeexBandModeLookup[$mode]) ? $SpeexBandModeLookup[$mode] : null); |
| 524 | + } |
| 525 | + |
| 526 | + |
| 527 | + static function OggPageSegmentLength($OggInfoArray, $SegmentNumber=1) { |
| 528 | + for ($i = 0; $i < $SegmentNumber; $i++) { |
| 529 | + $segmentlength = 0; |
| 530 | + foreach ($OggInfoArray['segment_table'] as $key => $value) { |
| 531 | + $segmentlength += $value; |
| 532 | + if ($value < 255) { |
| 533 | + break; |
| 534 | + } |
| 535 | + } |
| 536 | + } |
| 537 | + return $segmentlength; |
| 538 | + } |
| 539 | + |
| 540 | + |
| 541 | + static function get_quality_from_nominal_bitrate($nominal_bitrate) { |
| 542 | + |
| 543 | + // decrease precision |
| 544 | + $nominal_bitrate = $nominal_bitrate / 1000; |
| 545 | + |
| 546 | + if ($nominal_bitrate < 128) { |
| 547 | + // q-1 to q4 |
| 548 | + $qval = ($nominal_bitrate - 64) / 16; |
| 549 | + } elseif ($nominal_bitrate < 256) { |
| 550 | + // q4 to q8 |
| 551 | + $qval = $nominal_bitrate / 32; |
| 552 | + } elseif ($nominal_bitrate < 320) { |
| 553 | + // q8 to q9 |
| 554 | + $qval = ($nominal_bitrate + 256) / 64; |
| 555 | + } else { |
| 556 | + // q9 to q10 |
| 557 | + $qval = ($nominal_bitrate + 1300) / 180; |
| 558 | + } |
| 559 | + //return $qval; // 5.031324 |
| 560 | + //return intval($qval); // 5 |
| 561 | + return round($qval, 1); // 5 or 4.9 |
| 562 | + } |
| 563 | + |
| 564 | +} |
| 565 | + |
| 566 | +?> |
\ No newline at end of file |
Index: trunk/extensions/TimedMediaHandler/getid3/getid3.lib.php |
— | — | @@ -0,0 +1,1340 @@ |
| 2 | +<?php |
| 3 | +///////////////////////////////////////////////////////////////// |
| 4 | +/// getID3() by James Heinrich <info@getid3.org> // |
| 5 | +// available at http://getid3.sourceforge.net // |
| 6 | +// or http://www.getid3.org // |
| 7 | +///////////////////////////////////////////////////////////////// |
| 8 | +// // |
| 9 | +// getid3.lib.php - part of getID3() // |
| 10 | +// See readme.txt for more details // |
| 11 | +// /// |
| 12 | +///////////////////////////////////////////////////////////////// |
| 13 | + |
| 14 | + |
| 15 | +class getid3_lib |
| 16 | +{ |
| 17 | + |
| 18 | + static function PrintHexBytes($string, $hex=true, $spaces=true, $htmlsafe=true) { |
| 19 | + $returnstring = ''; |
| 20 | + for ($i = 0; $i < strlen($string); $i++) { |
| 21 | + if ($hex) { |
| 22 | + $returnstring .= str_pad(dechex(ord($string{$i})), 2, '0', STR_PAD_LEFT); |
| 23 | + } else { |
| 24 | + $returnstring .= ' '.(preg_match("#[\x20-\x7E]#", $string{$i}) ? $string{$i} : '€'); |
| 25 | + } |
| 26 | + if ($spaces) { |
| 27 | + $returnstring .= ' '; |
| 28 | + } |
| 29 | + } |
| 30 | + if ($htmlsafe) { |
| 31 | + $returnstring = htmlentities($returnstring); |
| 32 | + } |
| 33 | + return $returnstring; |
| 34 | + } |
| 35 | + |
| 36 | + static function trunc($floatnumber) { |
| 37 | + // truncates a floating-point number at the decimal point |
| 38 | + // returns int (if possible, otherwise float) |
| 39 | + if ($floatnumber >= 1) { |
| 40 | + $truncatednumber = floor($floatnumber); |
| 41 | + } elseif ($floatnumber <= -1) { |
| 42 | + $truncatednumber = ceil($floatnumber); |
| 43 | + } else { |
| 44 | + $truncatednumber = 0; |
| 45 | + } |
| 46 | + if ($truncatednumber <= 1073741824) { // 2^30 |
| 47 | + $truncatednumber = (int) $truncatednumber; |
| 48 | + } |
| 49 | + return $truncatednumber; |
| 50 | + } |
| 51 | + |
| 52 | + |
| 53 | + static function safe_inc(&$variable, $increment=1) { |
| 54 | + if (isset($variable)) { |
| 55 | + $variable += $increment; |
| 56 | + } else { |
| 57 | + $variable = $increment; |
| 58 | + } |
| 59 | + return true; |
| 60 | + } |
| 61 | + |
| 62 | + static function CastAsInt($floatnum) { |
| 63 | + // convert to float if not already |
| 64 | + $floatnum = (float) $floatnum; |
| 65 | + |
| 66 | + // convert a float to type int, only if possible |
| 67 | + if (getid3_lib::trunc($floatnum) == $floatnum) { |
| 68 | + // it's not floating point |
| 69 | + if ($floatnum <= 2147483647) { // 2^31 |
| 70 | + // it's within int range |
| 71 | + $floatnum = (int) $floatnum; |
| 72 | + } |
| 73 | + } |
| 74 | + return $floatnum; |
| 75 | + } |
| 76 | + |
| 77 | + |
| 78 | + static function DecimalizeFraction($fraction) { |
| 79 | + list($numerator, $denominator) = explode('/', $fraction); |
| 80 | + return $numerator / ($denominator ? $denominator : 1); |
| 81 | + } |
| 82 | + |
| 83 | + |
| 84 | + static function DecimalBinary2Float($binarynumerator) { |
| 85 | + $numerator = getid3_lib::Bin2Dec($binarynumerator); |
| 86 | + $denominator = getid3_lib::Bin2Dec('1'.str_repeat('0', strlen($binarynumerator))); |
| 87 | + return ($numerator / $denominator); |
| 88 | + } |
| 89 | + |
| 90 | + |
| 91 | + static function NormalizeBinaryPoint($binarypointnumber, $maxbits=52) { |
| 92 | + // http://www.scri.fsu.edu/~jac/MAD3401/Backgrnd/binary.html |
| 93 | + if (strpos($binarypointnumber, '.') === false) { |
| 94 | + $binarypointnumber = '0.'.$binarypointnumber; |
| 95 | + } elseif ($binarypointnumber{0} == '.') { |
| 96 | + $binarypointnumber = '0'.$binarypointnumber; |
| 97 | + } |
| 98 | + $exponent = 0; |
| 99 | + while (($binarypointnumber{0} != '1') || (substr($binarypointnumber, 1, 1) != '.')) { |
| 100 | + if (substr($binarypointnumber, 1, 1) == '.') { |
| 101 | + $exponent--; |
| 102 | + $binarypointnumber = substr($binarypointnumber, 2, 1).'.'.substr($binarypointnumber, 3); |
| 103 | + } else { |
| 104 | + $pointpos = strpos($binarypointnumber, '.'); |
| 105 | + $exponent += ($pointpos - 1); |
| 106 | + $binarypointnumber = str_replace('.', '', $binarypointnumber); |
| 107 | + $binarypointnumber = $binarypointnumber{0}.'.'.substr($binarypointnumber, 1); |
| 108 | + } |
| 109 | + } |
| 110 | + $binarypointnumber = str_pad(substr($binarypointnumber, 0, $maxbits + 2), $maxbits + 2, '0', STR_PAD_RIGHT); |
| 111 | + return array('normalized'=>$binarypointnumber, 'exponent'=>(int) $exponent); |
| 112 | + } |
| 113 | + |
| 114 | + |
| 115 | + static function Float2BinaryDecimal($floatvalue) { |
| 116 | + // http://www.scri.fsu.edu/~jac/MAD3401/Backgrnd/binary.html |
| 117 | + $maxbits = 128; // to how many bits of precision should the calculations be taken? |
| 118 | + $intpart = getid3_lib::trunc($floatvalue); |
| 119 | + $floatpart = abs($floatvalue - $intpart); |
| 120 | + $pointbitstring = ''; |
| 121 | + while (($floatpart != 0) && (strlen($pointbitstring) < $maxbits)) { |
| 122 | + $floatpart *= 2; |
| 123 | + $pointbitstring .= (string) getid3_lib::trunc($floatpart); |
| 124 | + $floatpart -= getid3_lib::trunc($floatpart); |
| 125 | + } |
| 126 | + $binarypointnumber = decbin($intpart).'.'.$pointbitstring; |
| 127 | + return $binarypointnumber; |
| 128 | + } |
| 129 | + |
| 130 | + |
| 131 | + static function Float2String($floatvalue, $bits) { |
| 132 | + // http://www.scri.fsu.edu/~jac/MAD3401/Backgrnd/ieee-expl.html |
| 133 | + switch ($bits) { |
| 134 | + case 32: |
| 135 | + $exponentbits = 8; |
| 136 | + $fractionbits = 23; |
| 137 | + break; |
| 138 | + |
| 139 | + case 64: |
| 140 | + $exponentbits = 11; |
| 141 | + $fractionbits = 52; |
| 142 | + break; |
| 143 | + |
| 144 | + default: |
| 145 | + return false; |
| 146 | + break; |
| 147 | + } |
| 148 | + if ($floatvalue >= 0) { |
| 149 | + $signbit = '0'; |
| 150 | + } else { |
| 151 | + $signbit = '1'; |
| 152 | + } |
| 153 | + $normalizedbinary = getid3_lib::NormalizeBinaryPoint(getid3_lib::Float2BinaryDecimal($floatvalue), $fractionbits); |
| 154 | + $biasedexponent = pow(2, $exponentbits - 1) - 1 + $normalizedbinary['exponent']; // (127 or 1023) +/- exponent |
| 155 | + $exponentbitstring = str_pad(decbin($biasedexponent), $exponentbits, '0', STR_PAD_LEFT); |
| 156 | + $fractionbitstring = str_pad(substr($normalizedbinary['normalized'], 2), $fractionbits, '0', STR_PAD_RIGHT); |
| 157 | + |
| 158 | + return getid3_lib::BigEndian2String(getid3_lib::Bin2Dec($signbit.$exponentbitstring.$fractionbitstring), $bits % 8, false); |
| 159 | + } |
| 160 | + |
| 161 | + |
| 162 | + static function LittleEndian2Float($byteword) { |
| 163 | + return getid3_lib::BigEndian2Float(strrev($byteword)); |
| 164 | + } |
| 165 | + |
| 166 | + |
| 167 | + static function BigEndian2Float($byteword) { |
| 168 | + // ANSI/IEEE Standard 754-1985, Standard for Binary Floating Point Arithmetic |
| 169 | + // http://www.psc.edu/general/software/packages/ieee/ieee.html |
| 170 | + // http://www.scri.fsu.edu/~jac/MAD3401/Backgrnd/ieee.html |
| 171 | + |
| 172 | + $bitword = getid3_lib::BigEndian2Bin($byteword); |
| 173 | + if (!$bitword) { |
| 174 | + return 0; |
| 175 | + } |
| 176 | + $signbit = $bitword{0}; |
| 177 | + |
| 178 | + switch (strlen($byteword) * 8) { |
| 179 | + case 32: |
| 180 | + $exponentbits = 8; |
| 181 | + $fractionbits = 23; |
| 182 | + break; |
| 183 | + |
| 184 | + case 64: |
| 185 | + $exponentbits = 11; |
| 186 | + $fractionbits = 52; |
| 187 | + break; |
| 188 | + |
| 189 | + case 80: |
| 190 | + // 80-bit Apple SANE format |
| 191 | + // http://www.mactech.com/articles/mactech/Vol.06/06.01/SANENormalized/ |
| 192 | + $exponentstring = substr($bitword, 1, 15); |
| 193 | + $isnormalized = intval($bitword{16}); |
| 194 | + $fractionstring = substr($bitword, 17, 63); |
| 195 | + $exponent = pow(2, getid3_lib::Bin2Dec($exponentstring) - 16383); |
| 196 | + $fraction = $isnormalized + getid3_lib::DecimalBinary2Float($fractionstring); |
| 197 | + $floatvalue = $exponent * $fraction; |
| 198 | + if ($signbit == '1') { |
| 199 | + $floatvalue *= -1; |
| 200 | + } |
| 201 | + return $floatvalue; |
| 202 | + break; |
| 203 | + |
| 204 | + default: |
| 205 | + return false; |
| 206 | + break; |
| 207 | + } |
| 208 | + $exponentstring = substr($bitword, 1, $exponentbits); |
| 209 | + $fractionstring = substr($bitword, $exponentbits + 1, $fractionbits); |
| 210 | + $exponent = getid3_lib::Bin2Dec($exponentstring); |
| 211 | + $fraction = getid3_lib::Bin2Dec($fractionstring); |
| 212 | + |
| 213 | + if (($exponent == (pow(2, $exponentbits) - 1)) && ($fraction != 0)) { |
| 214 | + // Not a Number |
| 215 | + $floatvalue = false; |
| 216 | + } elseif (($exponent == (pow(2, $exponentbits) - 1)) && ($fraction == 0)) { |
| 217 | + if ($signbit == '1') { |
| 218 | + $floatvalue = '-infinity'; |
| 219 | + } else { |
| 220 | + $floatvalue = '+infinity'; |
| 221 | + } |
| 222 | + } elseif (($exponent == 0) && ($fraction == 0)) { |
| 223 | + if ($signbit == '1') { |
| 224 | + $floatvalue = -0; |
| 225 | + } else { |
| 226 | + $floatvalue = 0; |
| 227 | + } |
| 228 | + $floatvalue = ($signbit ? 0 : -0); |
| 229 | + } elseif (($exponent == 0) && ($fraction != 0)) { |
| 230 | + // These are 'unnormalized' values |
| 231 | + $floatvalue = pow(2, (-1 * (pow(2, $exponentbits - 1) - 2))) * getid3_lib::DecimalBinary2Float($fractionstring); |
| 232 | + if ($signbit == '1') { |
| 233 | + $floatvalue *= -1; |
| 234 | + } |
| 235 | + } elseif ($exponent != 0) { |
| 236 | + $floatvalue = pow(2, ($exponent - (pow(2, $exponentbits - 1) - 1))) * (1 + getid3_lib::DecimalBinary2Float($fractionstring)); |
| 237 | + if ($signbit == '1') { |
| 238 | + $floatvalue *= -1; |
| 239 | + } |
| 240 | + } |
| 241 | + return (float) $floatvalue; |
| 242 | + } |
| 243 | + |
| 244 | + |
| 245 | + static function BigEndian2Int($byteword, $synchsafe=false, $signed=false) { |
| 246 | + $intvalue = 0; |
| 247 | + $bytewordlen = strlen($byteword); |
| 248 | + if ($bytewordlen == 0) { |
| 249 | + return false; |
| 250 | + } |
| 251 | + for ($i = 0; $i < $bytewordlen; $i++) { |
| 252 | + if ($synchsafe) { // disregard MSB, effectively 7-bit bytes |
| 253 | + $intvalue = $intvalue | (ord($byteword{$i}) & 0x7F) << (($bytewordlen - 1 - $i) * 7); |
| 254 | + } else { |
| 255 | + $intvalue += ord($byteword{$i}) * pow(256, ($bytewordlen - 1 - $i)); |
| 256 | + } |
| 257 | + } |
| 258 | + if ($signed && !$synchsafe) { |
| 259 | + // synchsafe ints are not allowed to be signed |
| 260 | + switch ($bytewordlen) { |
| 261 | + case 1: |
| 262 | + case 2: |
| 263 | + case 3: |
| 264 | + case 4: |
| 265 | + $signMaskBit = 0x80 << (8 * ($bytewordlen - 1)); |
| 266 | + if ($intvalue & $signMaskBit) { |
| 267 | + $intvalue = 0 - ($intvalue & ($signMaskBit - 1)); |
| 268 | + } |
| 269 | + break; |
| 270 | + |
| 271 | + default: |
| 272 | + die('ERROR: Cannot have signed integers larger than 32-bits ('.strlen($byteword).') in getid3_lib::BigEndian2Int()'); |
| 273 | + break; |
| 274 | + } |
| 275 | + } |
| 276 | + return getid3_lib::CastAsInt($intvalue); |
| 277 | + } |
| 278 | + |
| 279 | + |
| 280 | + static function LittleEndian2Int($byteword, $signed=false) { |
| 281 | + return getid3_lib::BigEndian2Int(strrev($byteword), false, $signed); |
| 282 | + } |
| 283 | + |
| 284 | + |
| 285 | + static function BigEndian2Bin($byteword) { |
| 286 | + $binvalue = ''; |
| 287 | + $bytewordlen = strlen($byteword); |
| 288 | + for ($i = 0; $i < $bytewordlen; $i++) { |
| 289 | + $binvalue .= str_pad(decbin(ord($byteword{$i})), 8, '0', STR_PAD_LEFT); |
| 290 | + } |
| 291 | + return $binvalue; |
| 292 | + } |
| 293 | + |
| 294 | + |
| 295 | + static function BigEndian2String($number, $minbytes=1, $synchsafe=false, $signed=false) { |
| 296 | + if ($number < 0) { |
| 297 | + return false; |
| 298 | + } |
| 299 | + $maskbyte = (($synchsafe || $signed) ? 0x7F : 0xFF); |
| 300 | + $intstring = ''; |
| 301 | + if ($signed) { |
| 302 | + if ($minbytes > 4) { |
| 303 | + die('ERROR: Cannot have signed integers larger than 32-bits in getid3_lib::BigEndian2String()'); |
| 304 | + } |
| 305 | + $number = $number & (0x80 << (8 * ($minbytes - 1))); |
| 306 | + } |
| 307 | + while ($number != 0) { |
| 308 | + $quotient = ($number / ($maskbyte + 1)); |
| 309 | + $intstring = chr(ceil(($quotient - floor($quotient)) * $maskbyte)).$intstring; |
| 310 | + $number = floor($quotient); |
| 311 | + } |
| 312 | + return str_pad($intstring, $minbytes, "\x00", STR_PAD_LEFT); |
| 313 | + } |
| 314 | + |
| 315 | + |
| 316 | + static function Dec2Bin($number) { |
| 317 | + while ($number >= 256) { |
| 318 | + $bytes[] = (($number / 256) - (floor($number / 256))) * 256; |
| 319 | + $number = floor($number / 256); |
| 320 | + } |
| 321 | + $bytes[] = $number; |
| 322 | + $binstring = ''; |
| 323 | + for ($i = 0; $i < count($bytes); $i++) { |
| 324 | + $binstring = (($i == count($bytes) - 1) ? decbin($bytes[$i]) : str_pad(decbin($bytes[$i]), 8, '0', STR_PAD_LEFT)).$binstring; |
| 325 | + } |
| 326 | + return $binstring; |
| 327 | + } |
| 328 | + |
| 329 | + |
| 330 | + static function Bin2Dec($binstring, $signed=false) { |
| 331 | + $signmult = 1; |
| 332 | + if ($signed) { |
| 333 | + if ($binstring{0} == '1') { |
| 334 | + $signmult = -1; |
| 335 | + } |
| 336 | + $binstring = substr($binstring, 1); |
| 337 | + } |
| 338 | + $decvalue = 0; |
| 339 | + for ($i = 0; $i < strlen($binstring); $i++) { |
| 340 | + $decvalue += ((int) substr($binstring, strlen($binstring) - $i - 1, 1)) * pow(2, $i); |
| 341 | + } |
| 342 | + return getid3_lib::CastAsInt($decvalue * $signmult); |
| 343 | + } |
| 344 | + |
| 345 | + |
| 346 | + static function Bin2String($binstring) { |
| 347 | + // return 'hi' for input of '0110100001101001' |
| 348 | + $string = ''; |
| 349 | + $binstringreversed = strrev($binstring); |
| 350 | + for ($i = 0; $i < strlen($binstringreversed); $i += 8) { |
| 351 | + $string = chr(getid3_lib::Bin2Dec(strrev(substr($binstringreversed, $i, 8)))).$string; |
| 352 | + } |
| 353 | + return $string; |
| 354 | + } |
| 355 | + |
| 356 | + |
| 357 | + static function LittleEndian2String($number, $minbytes=1, $synchsafe=false) { |
| 358 | + $intstring = ''; |
| 359 | + while ($number > 0) { |
| 360 | + if ($synchsafe) { |
| 361 | + $intstring = $intstring.chr($number & 127); |
| 362 | + $number >>= 7; |
| 363 | + } else { |
| 364 | + $intstring = $intstring.chr($number & 255); |
| 365 | + $number >>= 8; |
| 366 | + } |
| 367 | + } |
| 368 | + return str_pad($intstring, $minbytes, "\x00", STR_PAD_RIGHT); |
| 369 | + } |
| 370 | + |
| 371 | + |
| 372 | + static function array_merge_clobber($array1, $array2) { |
| 373 | + // written by kcØhireability*com |
| 374 | + // taken from http://www.php.net/manual/en/function.array-merge-recursive.php |
| 375 | + if (!is_array($array1) || !is_array($array2)) { |
| 376 | + return false; |
| 377 | + } |
| 378 | + $newarray = $array1; |
| 379 | + foreach ($array2 as $key => $val) { |
| 380 | + if (is_array($val) && isset($newarray[$key]) && is_array($newarray[$key])) { |
| 381 | + $newarray[$key] = getid3_lib::array_merge_clobber($newarray[$key], $val); |
| 382 | + } else { |
| 383 | + $newarray[$key] = $val; |
| 384 | + } |
| 385 | + } |
| 386 | + return $newarray; |
| 387 | + } |
| 388 | + |
| 389 | + |
| 390 | + static function array_merge_noclobber($array1, $array2) { |
| 391 | + if (!is_array($array1) || !is_array($array2)) { |
| 392 | + return false; |
| 393 | + } |
| 394 | + $newarray = $array1; |
| 395 | + foreach ($array2 as $key => $val) { |
| 396 | + if (is_array($val) && isset($newarray[$key]) && is_array($newarray[$key])) { |
| 397 | + $newarray[$key] = getid3_lib::array_merge_noclobber($newarray[$key], $val); |
| 398 | + } elseif (!isset($newarray[$key])) { |
| 399 | + $newarray[$key] = $val; |
| 400 | + } |
| 401 | + } |
| 402 | + return $newarray; |
| 403 | + } |
| 404 | + |
| 405 | + |
| 406 | + static function fileextension($filename, $numextensions=1) { |
| 407 | + if (strstr($filename, '.')) { |
| 408 | + $reversedfilename = strrev($filename); |
| 409 | + $offset = 0; |
| 410 | + for ($i = 0; $i < $numextensions; $i++) { |
| 411 | + $offset = strpos($reversedfilename, '.', $offset + 1); |
| 412 | + if ($offset === false) { |
| 413 | + return ''; |
| 414 | + } |
| 415 | + } |
| 416 | + return strrev(substr($reversedfilename, 0, $offset)); |
| 417 | + } |
| 418 | + return ''; |
| 419 | + } |
| 420 | + |
| 421 | + |
| 422 | + static function PlaytimeString($playtimeseconds) { |
| 423 | + $sign = (($playtimeseconds < 0) ? '-' : ''); |
| 424 | + $playtimeseconds = abs($playtimeseconds); |
| 425 | + $contentseconds = round((($playtimeseconds / 60) - floor($playtimeseconds / 60)) * 60); |
| 426 | + $contentminutes = floor($playtimeseconds / 60); |
| 427 | + if ($contentseconds >= 60) { |
| 428 | + $contentseconds -= 60; |
| 429 | + $contentminutes++; |
| 430 | + } |
| 431 | + return $sign.intval($contentminutes).':'.str_pad($contentseconds, 2, 0, STR_PAD_LEFT); |
| 432 | + } |
| 433 | + |
| 434 | + |
| 435 | + static function image_type_to_mime_type($imagetypeid) { |
| 436 | + // only available in PHP v4.3.0+ |
| 437 | + static $image_type_to_mime_type = array(); |
| 438 | + if (empty($image_type_to_mime_type)) { |
| 439 | + $image_type_to_mime_type[1] = 'image/gif'; // GIF |
| 440 | + $image_type_to_mime_type[2] = 'image/jpeg'; // JPEG |
| 441 | + $image_type_to_mime_type[3] = 'image/png'; // PNG |
| 442 | + $image_type_to_mime_type[4] = 'application/x-shockwave-flash'; // Flash |
| 443 | + $image_type_to_mime_type[5] = 'image/psd'; // PSD |
| 444 | + $image_type_to_mime_type[6] = 'image/bmp'; // BMP |
| 445 | + $image_type_to_mime_type[7] = 'image/tiff'; // TIFF: little-endian (Intel) |
| 446 | + $image_type_to_mime_type[8] = 'image/tiff'; // TIFF: big-endian (Motorola) |
| 447 | + //$image_type_to_mime_type[9] = 'image/jpc'; // JPC |
| 448 | + //$image_type_to_mime_type[10] = 'image/jp2'; // JPC |
| 449 | + //$image_type_to_mime_type[11] = 'image/jpx'; // JPC |
| 450 | + //$image_type_to_mime_type[12] = 'image/jb2'; // JPC |
| 451 | + $image_type_to_mime_type[13] = 'application/x-shockwave-flash'; // Shockwave |
| 452 | + $image_type_to_mime_type[14] = 'image/iff'; // IFF |
| 453 | + } |
| 454 | + return (isset($image_type_to_mime_type[$imagetypeid]) ? $image_type_to_mime_type[$imagetypeid] : 'application/octet-stream'); |
| 455 | + } |
| 456 | + |
| 457 | + |
| 458 | + static function DateMac2Unix($macdate) { |
| 459 | + // Macintosh timestamp: seconds since 00:00h January 1, 1904 |
| 460 | + // UNIX timestamp: seconds since 00:00h January 1, 1970 |
| 461 | + return getid3_lib::CastAsInt($macdate - 2082844800); |
| 462 | + } |
| 463 | + |
| 464 | + |
| 465 | + static function FixedPoint8_8($rawdata) { |
| 466 | + return getid3_lib::BigEndian2Int(substr($rawdata, 0, 1)) + (float) (getid3_lib::BigEndian2Int(substr($rawdata, 1, 1)) / pow(2, 8)); |
| 467 | + } |
| 468 | + |
| 469 | + |
| 470 | + static function FixedPoint16_16($rawdata) { |
| 471 | + return getid3_lib::BigEndian2Int(substr($rawdata, 0, 2)) + (float) (getid3_lib::BigEndian2Int(substr($rawdata, 2, 2)) / pow(2, 16)); |
| 472 | + } |
| 473 | + |
| 474 | + |
| 475 | + static function FixedPoint2_30($rawdata) { |
| 476 | + $binarystring = getid3_lib::BigEndian2Bin($rawdata); |
| 477 | + return getid3_lib::Bin2Dec(substr($binarystring, 0, 2)) + (float) (getid3_lib::Bin2Dec(substr($binarystring, 2, 30)) / 1073741824); |
| 478 | + } |
| 479 | + |
| 480 | + |
| 481 | + static function CreateDeepArray($ArrayPath, $Separator, $Value) { |
| 482 | + // assigns $Value to a nested array path: |
| 483 | + // $foo = getid3_lib::CreateDeepArray('/path/to/my', '/', 'file.txt') |
| 484 | + // is the same as: |
| 485 | + // $foo = array('path'=>array('to'=>'array('my'=>array('file.txt')))); |
| 486 | + // or |
| 487 | + // $foo['path']['to']['my'] = 'file.txt'; |
| 488 | + while ($ArrayPath && ($ArrayPath{0} == $Separator)) { |
| 489 | + $ArrayPath = substr($ArrayPath, 1); |
| 490 | + } |
| 491 | + if (($pos = strpos($ArrayPath, $Separator)) !== false) { |
| 492 | + $ReturnedArray[substr($ArrayPath, 0, $pos)] = getid3_lib::CreateDeepArray(substr($ArrayPath, $pos + 1), $Separator, $Value); |
| 493 | + } else { |
| 494 | + $ReturnedArray[$ArrayPath] = $Value; |
| 495 | + } |
| 496 | + return $ReturnedArray; |
| 497 | + } |
| 498 | + |
| 499 | + static function array_max($arraydata, $returnkey=false) { |
| 500 | + $maxvalue = false; |
| 501 | + $maxkey = false; |
| 502 | + foreach ($arraydata as $key => $value) { |
| 503 | + if (!is_array($value)) { |
| 504 | + if ($value > $maxvalue) { |
| 505 | + $maxvalue = $value; |
| 506 | + $maxkey = $key; |
| 507 | + } |
| 508 | + } |
| 509 | + } |
| 510 | + return ($returnkey ? $maxkey : $maxvalue); |
| 511 | + } |
| 512 | + |
| 513 | + static function array_min($arraydata, $returnkey=false) { |
| 514 | + $minvalue = false; |
| 515 | + $minkey = false; |
| 516 | + foreach ($arraydata as $key => $value) { |
| 517 | + if (!is_array($value)) { |
| 518 | + if ($value > $minvalue) { |
| 519 | + $minvalue = $value; |
| 520 | + $minkey = $key; |
| 521 | + } |
| 522 | + } |
| 523 | + } |
| 524 | + return ($returnkey ? $minkey : $minvalue); |
| 525 | + } |
| 526 | + |
| 527 | + |
| 528 | + static function md5_file($file) { |
| 529 | + |
| 530 | + // md5_file() exists in PHP 4.2.0+. |
| 531 | + if (function_exists('md5_file')) { |
| 532 | + return md5_file($file); |
| 533 | + } |
| 534 | + |
| 535 | + if (GETID3_OS_ISWINDOWS) { |
| 536 | + |
| 537 | + $RequiredFiles = array('cygwin1.dll', 'md5sum.exe'); |
| 538 | + foreach ($RequiredFiles as $required_file) { |
| 539 | + if (!is_readable(GETID3_HELPERAPPSDIR.$required_file)) { |
| 540 | + die(implode(' and ', $RequiredFiles).' are required in '.GETID3_HELPERAPPSDIR.' for getid3_lib::md5_file() to function under Windows in PHP < v4.2.0'); |
| 541 | + } |
| 542 | + } |
| 543 | + $commandline = GETID3_HELPERAPPSDIR.'md5sum.exe "'.str_replace('/', DIRECTORY_SEPARATOR, $file).'"'; |
| 544 | + if (preg_match("#^[\\]?([0-9a-f]{32})#", strtolower(`$commandline`), $r)) { |
| 545 | + return $r[1]; |
| 546 | + } |
| 547 | + |
| 548 | + } else { |
| 549 | + |
| 550 | + // The following works under UNIX only |
| 551 | + $file = str_replace('`', '\\`', $file); |
| 552 | + if (preg_match("#^([0-9a-f]{32})[ \t\n\r]#i", `md5sum "$file"`, $r)) { |
| 553 | + return $r[1]; |
| 554 | + } |
| 555 | + |
| 556 | + } |
| 557 | + return false; |
| 558 | + } |
| 559 | + |
| 560 | + |
| 561 | + static function sha1_file($file) { |
| 562 | + |
| 563 | + // sha1_file() exists in PHP 4.3.0+. |
| 564 | + if (function_exists('sha1_file')) { |
| 565 | + return sha1_file($file); |
| 566 | + } |
| 567 | + |
| 568 | + $file = str_replace('`', '\\`', $file); |
| 569 | + |
| 570 | + if (GETID3_OS_ISWINDOWS) { |
| 571 | + |
| 572 | + $RequiredFiles = array('cygwin1.dll', 'sha1sum.exe'); |
| 573 | + foreach ($RequiredFiles as $required_file) { |
| 574 | + if (!is_readable(GETID3_HELPERAPPSDIR.$required_file)) { |
| 575 | + die(implode(' and ', $RequiredFiles).' are required in '.GETID3_HELPERAPPSDIR.' for getid3_lib::sha1_file() to function under Windows in PHP < v4.3.0'); |
| 576 | + } |
| 577 | + } |
| 578 | + $commandline = GETID3_HELPERAPPSDIR.'sha1sum.exe "'.str_replace('/', DIRECTORY_SEPARATOR, $file).'"'; |
| 579 | + if (preg_match("#^sha1=([0-9a-f]{40})#", strtolower(`$commandline`), $r)) { |
| 580 | + return $r[1]; |
| 581 | + } |
| 582 | + |
| 583 | + } else { |
| 584 | + |
| 585 | + $commandline = 'sha1sum '.escapeshellarg($file).''; |
| 586 | + if (preg_match("#^([0-9a-f]{40})[ \t\n\r]#", strtolower(`$commandline`), $r)) { |
| 587 | + return $r[1]; |
| 588 | + } |
| 589 | + |
| 590 | + } |
| 591 | + |
| 592 | + return false; |
| 593 | + } |
| 594 | + |
| 595 | + |
| 596 | + // Allan Hansen <ahØartemis*dk> |
| 597 | + // getid3_lib::md5_data() - returns md5sum for a file from startuing position to absolute end position |
| 598 | + static function hash_data($file, $offset, $end, $algorithm) { |
| 599 | + if ($end >= pow(2, 31)) { |
| 600 | + return false; |
| 601 | + } |
| 602 | + |
| 603 | + switch ($algorithm) { |
| 604 | + case 'md5': |
| 605 | + $hash_function = 'md5_file'; |
| 606 | + $unix_call = 'md5sum'; |
| 607 | + $windows_call = 'md5sum.exe'; |
| 608 | + $hash_length = 32; |
| 609 | + break; |
| 610 | + |
| 611 | + case 'sha1': |
| 612 | + $hash_function = 'sha1_file'; |
| 613 | + $unix_call = 'sha1sum'; |
| 614 | + $windows_call = 'sha1sum.exe'; |
| 615 | + $hash_length = 40; |
| 616 | + break; |
| 617 | + |
| 618 | + default: |
| 619 | + die('Invalid algorithm ('.$algorithm.') in getid3_lib::hash_data()'); |
| 620 | + break; |
| 621 | + } |
| 622 | + $size = $end - $offset; |
| 623 | + while (true) { |
| 624 | + if (GETID3_OS_ISWINDOWS) { |
| 625 | + |
| 626 | + // It seems that sha1sum.exe for Windows only works on physical files, does not accept piped data |
| 627 | + // Fall back to create-temp-file method: |
| 628 | + if ($algorithm == 'sha1') { |
| 629 | + break; |
| 630 | + } |
| 631 | + |
| 632 | + $RequiredFiles = array('cygwin1.dll', 'head.exe', 'tail.exe', $windows_call); |
| 633 | + foreach ($RequiredFiles as $required_file) { |
| 634 | + if (!is_readable(GETID3_HELPERAPPSDIR.$required_file)) { |
| 635 | + // helper apps not available - fall back to old method |
| 636 | + break; |
| 637 | + } |
| 638 | + } |
| 639 | + $commandline = GETID3_HELPERAPPSDIR.'head.exe -c '.$end.' "'.escapeshellarg(str_replace('/', DIRECTORY_SEPARATOR, $file)).'" | '; |
| 640 | + $commandline .= GETID3_HELPERAPPSDIR.'tail.exe -c '.$size.' | '; |
| 641 | + $commandline .= GETID3_HELPERAPPSDIR.$windows_call; |
| 642 | + |
| 643 | + } else { |
| 644 | + |
| 645 | + $commandline = 'head -c'.$end.' '.escapeshellarg($file).' | '; |
| 646 | + $commandline .= 'tail -c'.$size.' | '; |
| 647 | + $commandline .= $unix_call; |
| 648 | + |
| 649 | + } |
| 650 | + if (preg_match('#(1|ON)#i', ini_get('safe_mode'))) { |
| 651 | + $ThisFileInfo['warning'][] = 'PHP running in Safe Mode - backtick operator not available, using slower non-system-call '.$algorithm.' algorithm'; |
| 652 | + break; |
| 653 | + } |
| 654 | + return substr(`$commandline`, 0, $hash_length); |
| 655 | + } |
| 656 | + |
| 657 | + // try to create a temporary file in the system temp directory - invalid dirname should force to system temp dir |
| 658 | + if (($data_filename = tempnam(GETID3_TEMP_DIR, 'getID3')) === false) { |
| 659 | + // can't find anywhere to create a temp file, just die |
| 660 | + return false; |
| 661 | + } |
| 662 | + |
| 663 | + // Init |
| 664 | + $result = false; |
| 665 | + |
| 666 | + // copy parts of file |
| 667 | + ob_start(); |
| 668 | + if ($fp = fopen($file, 'rb')) { |
| 669 | + ob_end_clean(); |
| 670 | + ob_start(); |
| 671 | + if ($fp_data = fopen($data_filename, 'wb')) { |
| 672 | + fseek($fp, $offset, SEEK_SET); |
| 673 | + $byteslefttowrite = $end - $offset; |
| 674 | + while (($byteslefttowrite > 0) && ($buffer = fread($fp, GETID3_FREAD_BUFFER_SIZE))) { |
| 675 | + $byteswritten = fwrite($fp_data, $buffer, $byteslefttowrite); |
| 676 | + $byteslefttowrite -= $byteswritten; |
| 677 | + } |
| 678 | + fclose($fp_data); |
| 679 | + $result = getid3_lib::$hash_function($data_filename); |
| 680 | + } else { |
| 681 | + $errormessage = ob_get_contents(); |
| 682 | + ob_end_clean(); |
| 683 | + } |
| 684 | + fclose($fp); |
| 685 | + } else { |
| 686 | + $errormessage = ob_get_contents(); |
| 687 | + ob_end_clean(); |
| 688 | + } |
| 689 | + unlink($data_filename); |
| 690 | + return $result; |
| 691 | + } |
| 692 | + |
| 693 | + |
| 694 | + static function iconv_fallback_int_utf8($charval) { |
| 695 | + if ($charval < 128) { |
| 696 | + // 0bbbbbbb |
| 697 | + $newcharstring = chr($charval); |
| 698 | + } elseif ($charval < 2048) { |
| 699 | + // 110bbbbb 10bbbbbb |
| 700 | + $newcharstring = chr(($charval >> 6) | 0xC0); |
| 701 | + $newcharstring .= chr(($charval & 0x3F) | 0x80); |
| 702 | + } elseif ($charval < 65536) { |
| 703 | + // 1110bbbb 10bbbbbb 10bbbbbb |
| 704 | + $newcharstring = chr(($charval >> 12) | 0xE0); |
| 705 | + $newcharstring .= chr(($charval >> 6) | 0xC0); |
| 706 | + $newcharstring .= chr(($charval & 0x3F) | 0x80); |
| 707 | + } else { |
| 708 | + // 11110bbb 10bbbbbb 10bbbbbb 10bbbbbb |
| 709 | + $newcharstring = chr(($charval >> 18) | 0xF0); |
| 710 | + $newcharstring .= chr(($charval >> 12) | 0xC0); |
| 711 | + $newcharstring .= chr(($charval >> 6) | 0xC0); |
| 712 | + $newcharstring .= chr(($charval & 0x3F) | 0x80); |
| 713 | + } |
| 714 | + return $newcharstring; |
| 715 | + } |
| 716 | + |
| 717 | + // ISO-8859-1 => UTF-8 |
| 718 | + static function iconv_fallback_iso88591_utf8($string, $bom=false) { |
| 719 | + if (function_exists('utf8_encode')) { |
| 720 | + return utf8_encode($string); |
| 721 | + } |
| 722 | + // utf8_encode() unavailable, use getID3()'s iconv_fallback() conversions (possibly PHP is compiled without XML support) |
| 723 | + $newcharstring = ''; |
| 724 | + if ($bom) { |
| 725 | + $newcharstring .= "\xEF\xBB\xBF"; |
| 726 | + } |
| 727 | + for ($i = 0; $i < strlen($string); $i++) { |
| 728 | + $charval = ord($string{$i}); |
| 729 | + $newcharstring .= getid3_lib::iconv_fallback_int_utf8($charval); |
| 730 | + } |
| 731 | + return $newcharstring; |
| 732 | + } |
| 733 | + |
| 734 | + // ISO-8859-1 => UTF-16BE |
| 735 | + static function iconv_fallback_iso88591_utf16be($string, $bom=false) { |
| 736 | + $newcharstring = ''; |
| 737 | + if ($bom) { |
| 738 | + $newcharstring .= "\xFE\xFF"; |
| 739 | + } |
| 740 | + for ($i = 0; $i < strlen($string); $i++) { |
| 741 | + $newcharstring .= "\x00".$string{$i}; |
| 742 | + } |
| 743 | + return $newcharstring; |
| 744 | + } |
| 745 | + |
| 746 | + // ISO-8859-1 => UTF-16LE |
| 747 | + static function iconv_fallback_iso88591_utf16le($string, $bom=false) { |
| 748 | + $newcharstring = ''; |
| 749 | + if ($bom) { |
| 750 | + $newcharstring .= "\xFF\xFE"; |
| 751 | + } |
| 752 | + for ($i = 0; $i < strlen($string); $i++) { |
| 753 | + $newcharstring .= $string{$i}."\x00"; |
| 754 | + } |
| 755 | + return $newcharstring; |
| 756 | + } |
| 757 | + |
| 758 | + // ISO-8859-1 => UTF-16LE (BOM) |
| 759 | + static function iconv_fallback_iso88591_utf16($string) { |
| 760 | + return getid3_lib::iconv_fallback_iso88591_utf16le($string, true); |
| 761 | + } |
| 762 | + |
| 763 | + // UTF-8 => ISO-8859-1 |
| 764 | + static function iconv_fallback_utf8_iso88591($string) { |
| 765 | + if (function_exists('utf8_decode')) { |
| 766 | + return utf8_decode($string); |
| 767 | + } |
| 768 | + // utf8_decode() unavailable, use getID3()'s iconv_fallback() conversions (possibly PHP is compiled without XML support) |
| 769 | + $newcharstring = ''; |
| 770 | + $offset = 0; |
| 771 | + $stringlength = strlen($string); |
| 772 | + while ($offset < $stringlength) { |
| 773 | + if ((ord($string{$offset}) | 0x07) == 0xF7) { |
| 774 | + // 11110bbb 10bbbbbb 10bbbbbb 10bbbbbb |
| 775 | + $charval = ((ord($string{($offset + 0)}) & 0x07) << 18) & |
| 776 | + ((ord($string{($offset + 1)}) & 0x3F) << 12) & |
| 777 | + ((ord($string{($offset + 2)}) & 0x3F) << 6) & |
| 778 | + (ord($string{($offset + 3)}) & 0x3F); |
| 779 | + $offset += 4; |
| 780 | + } elseif ((ord($string{$offset}) | 0x0F) == 0xEF) { |
| 781 | + // 1110bbbb 10bbbbbb 10bbbbbb |
| 782 | + $charval = ((ord($string{($offset + 0)}) & 0x0F) << 12) & |
| 783 | + ((ord($string{($offset + 1)}) & 0x3F) << 6) & |
| 784 | + (ord($string{($offset + 2)}) & 0x3F); |
| 785 | + $offset += 3; |
| 786 | + } elseif ((ord($string{$offset}) | 0x1F) == 0xDF) { |
| 787 | + // 110bbbbb 10bbbbbb |
| 788 | + $charval = ((ord($string{($offset + 0)}) & 0x1F) << 6) & |
| 789 | + (ord($string{($offset + 1)}) & 0x3F); |
| 790 | + $offset += 2; |
| 791 | + } elseif ((ord($string{$offset}) | 0x7F) == 0x7F) { |
| 792 | + // 0bbbbbbb |
| 793 | + $charval = ord($string{$offset}); |
| 794 | + $offset += 1; |
| 795 | + } else { |
| 796 | + // error? throw some kind of warning here? |
| 797 | + $charval = false; |
| 798 | + $offset += 1; |
| 799 | + } |
| 800 | + if ($charval !== false) { |
| 801 | + $newcharstring .= (($charval < 256) ? chr($charval) : '?'); |
| 802 | + } |
| 803 | + } |
| 804 | + return $newcharstring; |
| 805 | + } |
| 806 | + |
| 807 | + // UTF-8 => UTF-16BE |
| 808 | + static function iconv_fallback_utf8_utf16be($string, $bom=false) { |
| 809 | + $newcharstring = ''; |
| 810 | + if ($bom) { |
| 811 | + $newcharstring .= "\xFE\xFF"; |
| 812 | + } |
| 813 | + $offset = 0; |
| 814 | + $stringlength = strlen($string); |
| 815 | + while ($offset < $stringlength) { |
| 816 | + if ((ord($string{$offset}) | 0x07) == 0xF7) { |
| 817 | + // 11110bbb 10bbbbbb 10bbbbbb 10bbbbbb |
| 818 | + $charval = ((ord($string{($offset + 0)}) & 0x07) << 18) & |
| 819 | + ((ord($string{($offset + 1)}) & 0x3F) << 12) & |
| 820 | + ((ord($string{($offset + 2)}) & 0x3F) << 6) & |
| 821 | + (ord($string{($offset + 3)}) & 0x3F); |
| 822 | + $offset += 4; |
| 823 | + } elseif ((ord($string{$offset}) | 0x0F) == 0xEF) { |
| 824 | + // 1110bbbb 10bbbbbb 10bbbbbb |
| 825 | + $charval = ((ord($string{($offset + 0)}) & 0x0F) << 12) & |
| 826 | + ((ord($string{($offset + 1)}) & 0x3F) << 6) & |
| 827 | + (ord($string{($offset + 2)}) & 0x3F); |
| 828 | + $offset += 3; |
| 829 | + } elseif ((ord($string{$offset}) | 0x1F) == 0xDF) { |
| 830 | + // 110bbbbb 10bbbbbb |
| 831 | + $charval = ((ord($string{($offset + 0)}) & 0x1F) << 6) & |
| 832 | + (ord($string{($offset + 1)}) & 0x3F); |
| 833 | + $offset += 2; |
| 834 | + } elseif ((ord($string{$offset}) | 0x7F) == 0x7F) { |
| 835 | + // 0bbbbbbb |
| 836 | + $charval = ord($string{$offset}); |
| 837 | + $offset += 1; |
| 838 | + } else { |
| 839 | + // error? throw some kind of warning here? |
| 840 | + $charval = false; |
| 841 | + $offset += 1; |
| 842 | + } |
| 843 | + if ($charval !== false) { |
| 844 | + $newcharstring .= (($charval < 65536) ? getid3_lib::BigEndian2String($charval, 2) : "\x00".'?'); |
| 845 | + } |
| 846 | + } |
| 847 | + return $newcharstring; |
| 848 | + } |
| 849 | + |
| 850 | + // UTF-8 => UTF-16LE |
| 851 | + static function iconv_fallback_utf8_utf16le($string, $bom=false) { |
| 852 | + $newcharstring = ''; |
| 853 | + if ($bom) { |
| 854 | + $newcharstring .= "\xFF\xFE"; |
| 855 | + } |
| 856 | + $offset = 0; |
| 857 | + $stringlength = strlen($string); |
| 858 | + while ($offset < $stringlength) { |
| 859 | + if ((ord($string{$offset}) | 0x07) == 0xF7) { |
| 860 | + // 11110bbb 10bbbbbb 10bbbbbb 10bbbbbb |
| 861 | + $charval = ((ord($string{($offset + 0)}) & 0x07) << 18) & |
| 862 | + ((ord($string{($offset + 1)}) & 0x3F) << 12) & |
| 863 | + ((ord($string{($offset + 2)}) & 0x3F) << 6) & |
| 864 | + (ord($string{($offset + 3)}) & 0x3F); |
| 865 | + $offset += 4; |
| 866 | + } elseif ((ord($string{$offset}) | 0x0F) == 0xEF) { |
| 867 | + // 1110bbbb 10bbbbbb 10bbbbbb |
| 868 | + $charval = ((ord($string{($offset + 0)}) & 0x0F) << 12) & |
| 869 | + ((ord($string{($offset + 1)}) & 0x3F) << 6) & |
| 870 | + (ord($string{($offset + 2)}) & 0x3F); |
| 871 | + $offset += 3; |
| 872 | + } elseif ((ord($string{$offset}) | 0x1F) == 0xDF) { |
| 873 | + // 110bbbbb 10bbbbbb |
| 874 | + $charval = ((ord($string{($offset + 0)}) & 0x1F) << 6) & |
| 875 | + (ord($string{($offset + 1)}) & 0x3F); |
| 876 | + $offset += 2; |
| 877 | + } elseif ((ord($string{$offset}) | 0x7F) == 0x7F) { |
| 878 | + // 0bbbbbbb |
| 879 | + $charval = ord($string{$offset}); |
| 880 | + $offset += 1; |
| 881 | + } else { |
| 882 | + // error? maybe throw some warning here? |
| 883 | + $charval = false; |
| 884 | + $offset += 1; |
| 885 | + } |
| 886 | + if ($charval !== false) { |
| 887 | + $newcharstring .= (($charval < 65536) ? getid3_lib::LittleEndian2String($charval, 2) : '?'."\x00"); |
| 888 | + } |
| 889 | + } |
| 890 | + return $newcharstring; |
| 891 | + } |
| 892 | + |
| 893 | + // UTF-8 => UTF-16LE (BOM) |
| 894 | + static function iconv_fallback_utf8_utf16($string) { |
| 895 | + return getid3_lib::iconv_fallback_utf8_utf16le($string, true); |
| 896 | + } |
| 897 | + |
| 898 | + // UTF-16BE => UTF-8 |
| 899 | + static function iconv_fallback_utf16be_utf8($string) { |
| 900 | + if (substr($string, 0, 2) == "\xFE\xFF") { |
| 901 | + // strip BOM |
| 902 | + $string = substr($string, 2); |
| 903 | + } |
| 904 | + $newcharstring = ''; |
| 905 | + for ($i = 0; $i < strlen($string); $i += 2) { |
| 906 | + $charval = getid3_lib::BigEndian2Int(substr($string, $i, 2)); |
| 907 | + $newcharstring .= getid3_lib::iconv_fallback_int_utf8($charval); |
| 908 | + } |
| 909 | + return $newcharstring; |
| 910 | + } |
| 911 | + |
| 912 | + // UTF-16LE => UTF-8 |
| 913 | + static function iconv_fallback_utf16le_utf8($string) { |
| 914 | + if (substr($string, 0, 2) == "\xFF\xFE") { |
| 915 | + // strip BOM |
| 916 | + $string = substr($string, 2); |
| 917 | + } |
| 918 | + $newcharstring = ''; |
| 919 | + for ($i = 0; $i < strlen($string); $i += 2) { |
| 920 | + $charval = getid3_lib::LittleEndian2Int(substr($string, $i, 2)); |
| 921 | + $newcharstring .= getid3_lib::iconv_fallback_int_utf8($charval); |
| 922 | + } |
| 923 | + return $newcharstring; |
| 924 | + } |
| 925 | + |
| 926 | + // UTF-16BE => ISO-8859-1 |
| 927 | + static function iconv_fallback_utf16be_iso88591($string) { |
| 928 | + if (substr($string, 0, 2) == "\xFE\xFF") { |
| 929 | + // strip BOM |
| 930 | + $string = substr($string, 2); |
| 931 | + } |
| 932 | + $newcharstring = ''; |
| 933 | + for ($i = 0; $i < strlen($string); $i += 2) { |
| 934 | + $charval = getid3_lib::BigEndian2Int(substr($string, $i, 2)); |
| 935 | + $newcharstring .= (($charval < 256) ? chr($charval) : '?'); |
| 936 | + } |
| 937 | + return $newcharstring; |
| 938 | + } |
| 939 | + |
| 940 | + // UTF-16LE => ISO-8859-1 |
| 941 | + static function iconv_fallback_utf16le_iso88591($string) { |
| 942 | + if (substr($string, 0, 2) == "\xFF\xFE") { |
| 943 | + // strip BOM |
| 944 | + $string = substr($string, 2); |
| 945 | + } |
| 946 | + $newcharstring = ''; |
| 947 | + for ($i = 0; $i < strlen($string); $i += 2) { |
| 948 | + $charval = getid3_lib::LittleEndian2Int(substr($string, $i, 2)); |
| 949 | + $newcharstring .= (($charval < 256) ? chr($charval) : '?'); |
| 950 | + } |
| 951 | + return $newcharstring; |
| 952 | + } |
| 953 | + |
| 954 | + // UTF-16 (BOM) => ISO-8859-1 |
| 955 | + static function iconv_fallback_utf16_iso88591($string) { |
| 956 | + $bom = substr($string, 0, 2); |
| 957 | + if ($bom == "\xFE\xFF") { |
| 958 | + return getid3_lib::iconv_fallback_utf16be_iso88591(substr($string, 2)); |
| 959 | + } elseif ($bom == "\xFF\xFE") { |
| 960 | + return getid3_lib::iconv_fallback_utf16le_iso88591(substr($string, 2)); |
| 961 | + } |
| 962 | + return $string; |
| 963 | + } |
| 964 | + |
| 965 | + // UTF-16 (BOM) => UTF-8 |
| 966 | + static function iconv_fallback_utf16_utf8($string) { |
| 967 | + $bom = substr($string, 0, 2); |
| 968 | + if ($bom == "\xFE\xFF") { |
| 969 | + return getid3_lib::iconv_fallback_utf16be_utf8(substr($string, 2)); |
| 970 | + } elseif ($bom == "\xFF\xFE") { |
| 971 | + return getid3_lib::iconv_fallback_utf16le_utf8(substr($string, 2)); |
| 972 | + } |
| 973 | + return $string; |
| 974 | + } |
| 975 | + |
| 976 | + static function iconv_fallback($in_charset, $out_charset, $string) { |
| 977 | + |
| 978 | + if ($in_charset == $out_charset) { |
| 979 | + return $string; |
| 980 | + } |
| 981 | + |
| 982 | + // iconv() availble |
| 983 | + if (function_exists('iconv')) { |
| 984 | + |
| 985 | + ob_start(); |
| 986 | + if ($converted_string = iconv($in_charset, $out_charset.'//TRANSLIT', $string)) { |
| 987 | + ob_end_clean(); |
| 988 | + switch ($out_charset) { |
| 989 | + case 'ISO-8859-1': |
| 990 | + $converted_string = rtrim($converted_string, "\x00"); |
| 991 | + break; |
| 992 | + } |
| 993 | + return $converted_string; |
| 994 | + } else { |
| 995 | + $errormessage = ob_get_contents(); |
| 996 | + ob_end_clean(); |
| 997 | + } |
| 998 | + |
| 999 | + // iconv() may sometimes fail with "illegal character in input string" error message |
| 1000 | + // and return an empty string, but returning the unconverted string is more useful |
| 1001 | + return $string; |
| 1002 | + } |
| 1003 | + |
| 1004 | + |
| 1005 | + // iconv() not available |
| 1006 | + static $ConversionFunctionList = array(); |
| 1007 | + if (empty($ConversionFunctionList)) { |
| 1008 | + $ConversionFunctionList['ISO-8859-1']['UTF-8'] = 'iconv_fallback_iso88591_utf8'; |
| 1009 | + $ConversionFunctionList['ISO-8859-1']['UTF-16'] = 'iconv_fallback_iso88591_utf16'; |
| 1010 | + $ConversionFunctionList['ISO-8859-1']['UTF-16BE'] = 'iconv_fallback_iso88591_utf16be'; |
| 1011 | + $ConversionFunctionList['ISO-8859-1']['UTF-16LE'] = 'iconv_fallback_iso88591_utf16le'; |
| 1012 | + $ConversionFunctionList['UTF-8']['ISO-8859-1'] = 'iconv_fallback_utf8_iso88591'; |
| 1013 | + $ConversionFunctionList['UTF-8']['UTF-16'] = 'iconv_fallback_utf8_utf16'; |
| 1014 | + $ConversionFunctionList['UTF-8']['UTF-16BE'] = 'iconv_fallback_utf8_utf16be'; |
| 1015 | + $ConversionFunctionList['UTF-8']['UTF-16LE'] = 'iconv_fallback_utf8_utf16le'; |
| 1016 | + $ConversionFunctionList['UTF-16']['ISO-8859-1'] = 'iconv_fallback_utf16_iso88591'; |
| 1017 | + $ConversionFunctionList['UTF-16']['UTF-8'] = 'iconv_fallback_utf16_utf8'; |
| 1018 | + $ConversionFunctionList['UTF-16LE']['ISO-8859-1'] = 'iconv_fallback_utf16le_iso88591'; |
| 1019 | + $ConversionFunctionList['UTF-16LE']['UTF-8'] = 'iconv_fallback_utf16le_utf8'; |
| 1020 | + $ConversionFunctionList['UTF-16BE']['ISO-8859-1'] = 'iconv_fallback_utf16be_iso88591'; |
| 1021 | + $ConversionFunctionList['UTF-16BE']['UTF-8'] = 'iconv_fallback_utf16be_utf8'; |
| 1022 | + } |
| 1023 | + if (isset($ConversionFunctionList[strtoupper($in_charset)][strtoupper($out_charset)])) { |
| 1024 | + $ConversionFunction = $ConversionFunctionList[strtoupper($in_charset)][strtoupper($out_charset)]; |
| 1025 | + return getid3_lib::$ConversionFunction($string); |
| 1026 | + } |
| 1027 | + die('PHP does not have iconv() support - cannot convert from '.$in_charset.' to '.$out_charset); |
| 1028 | + } |
| 1029 | + |
| 1030 | + |
| 1031 | + static function MultiByteCharString2HTML($string, $charset='ISO-8859-1') { |
| 1032 | + $HTMLstring = ''; |
| 1033 | + |
| 1034 | + switch ($charset) { |
| 1035 | + case 'ISO-8859-1': |
| 1036 | + case 'ISO8859-1': |
| 1037 | + case 'ISO-8859-15': |
| 1038 | + case 'ISO8859-15': |
| 1039 | + case 'cp866': |
| 1040 | + case 'ibm866': |
| 1041 | + case '866': |
| 1042 | + case 'cp1251': |
| 1043 | + case 'Windows-1251': |
| 1044 | + case 'win-1251': |
| 1045 | + case '1251': |
| 1046 | + case 'cp1252': |
| 1047 | + case 'Windows-1252': |
| 1048 | + case '1252': |
| 1049 | + case 'KOI8-R': |
| 1050 | + case 'koi8-ru': |
| 1051 | + case 'koi8r': |
| 1052 | + case 'BIG5': |
| 1053 | + case '950': |
| 1054 | + case 'GB2312': |
| 1055 | + case '936': |
| 1056 | + case 'BIG5-HKSCS': |
| 1057 | + case 'Shift_JIS': |
| 1058 | + case 'SJIS': |
| 1059 | + case '932': |
| 1060 | + case 'EUC-JP': |
| 1061 | + case 'EUCJP': |
| 1062 | + $HTMLstring = htmlentities($string, ENT_COMPAT, $charset); |
| 1063 | + break; |
| 1064 | + |
| 1065 | + case 'UTF-8': |
| 1066 | + $strlen = strlen($string); |
| 1067 | + for ($i = 0; $i < $strlen; $i++) { |
| 1068 | + $char_ord_val = ord($string{$i}); |
| 1069 | + $charval = 0; |
| 1070 | + if ($char_ord_val < 0x80) { |
| 1071 | + $charval = $char_ord_val; |
| 1072 | + } elseif ((($char_ord_val & 0xF0) >> 4) == 0x0F && $i+3 < $strlen) { |
| 1073 | + $charval = (($char_ord_val & 0x07) << 18); |
| 1074 | + $charval += ((ord($string{++$i}) & 0x3F) << 12); |
| 1075 | + $charval += ((ord($string{++$i}) & 0x3F) << 6); |
| 1076 | + $charval += (ord($string{++$i}) & 0x3F); |
| 1077 | + } elseif ((($char_ord_val & 0xE0) >> 5) == 0x07 && $i+2 < $strlen) { |
| 1078 | + $charval = (($char_ord_val & 0x0F) << 12); |
| 1079 | + $charval += ((ord($string{++$i}) & 0x3F) << 6); |
| 1080 | + $charval += (ord($string{++$i}) & 0x3F); |
| 1081 | + } elseif ((($char_ord_val & 0xC0) >> 6) == 0x03 && $i+1 < $strlen) { |
| 1082 | + $charval = (($char_ord_val & 0x1F) << 6); |
| 1083 | + $charval += (ord($string{++$i}) & 0x3F); |
| 1084 | + } |
| 1085 | + if (($charval >= 32) && ($charval <= 127)) { |
| 1086 | + $HTMLstring .= htmlentities(chr($charval)); |
| 1087 | + } else { |
| 1088 | + $HTMLstring .= '&#'.$charval.';'; |
| 1089 | + } |
| 1090 | + } |
| 1091 | + break; |
| 1092 | + |
| 1093 | + case 'UTF-16LE': |
| 1094 | + for ($i = 0; $i < strlen($string); $i += 2) { |
| 1095 | + $charval = getid3_lib::LittleEndian2Int(substr($string, $i, 2)); |
| 1096 | + if (($charval >= 32) && ($charval <= 127)) { |
| 1097 | + $HTMLstring .= chr($charval); |
| 1098 | + } else { |
| 1099 | + $HTMLstring .= '&#'.$charval.';'; |
| 1100 | + } |
| 1101 | + } |
| 1102 | + break; |
| 1103 | + |
| 1104 | + case 'UTF-16BE': |
| 1105 | + for ($i = 0; $i < strlen($string); $i += 2) { |
| 1106 | + $charval = getid3_lib::BigEndian2Int(substr($string, $i, 2)); |
| 1107 | + if (($charval >= 32) && ($charval <= 127)) { |
| 1108 | + $HTMLstring .= chr($charval); |
| 1109 | + } else { |
| 1110 | + $HTMLstring .= '&#'.$charval.';'; |
| 1111 | + } |
| 1112 | + } |
| 1113 | + break; |
| 1114 | + |
| 1115 | + default: |
| 1116 | + $HTMLstring = 'ERROR: Character set "'.$charset.'" not supported in MultiByteCharString2HTML()'; |
| 1117 | + break; |
| 1118 | + } |
| 1119 | + return $HTMLstring; |
| 1120 | + } |
| 1121 | + |
| 1122 | + |
| 1123 | + |
| 1124 | + static function RGADnameLookup($namecode) { |
| 1125 | + static $RGADname = array(); |
| 1126 | + if (empty($RGADname)) { |
| 1127 | + $RGADname[0] = 'not set'; |
| 1128 | + $RGADname[1] = 'Track Gain Adjustment'; |
| 1129 | + $RGADname[2] = 'Album Gain Adjustment'; |
| 1130 | + } |
| 1131 | + |
| 1132 | + return (isset($RGADname[$namecode]) ? $RGADname[$namecode] : ''); |
| 1133 | + } |
| 1134 | + |
| 1135 | + |
| 1136 | + static function RGADoriginatorLookup($originatorcode) { |
| 1137 | + static $RGADoriginator = array(); |
| 1138 | + if (empty($RGADoriginator)) { |
| 1139 | + $RGADoriginator[0] = 'unspecified'; |
| 1140 | + $RGADoriginator[1] = 'pre-set by artist/producer/mastering engineer'; |
| 1141 | + $RGADoriginator[2] = 'set by user'; |
| 1142 | + $RGADoriginator[3] = 'determined automatically'; |
| 1143 | + } |
| 1144 | + |
| 1145 | + return (isset($RGADoriginator[$originatorcode]) ? $RGADoriginator[$originatorcode] : ''); |
| 1146 | + } |
| 1147 | + |
| 1148 | + |
| 1149 | + static function RGADadjustmentLookup($rawadjustment, $signbit) { |
| 1150 | + $adjustment = $rawadjustment / 10; |
| 1151 | + if ($signbit == 1) { |
| 1152 | + $adjustment *= -1; |
| 1153 | + } |
| 1154 | + return (float) $adjustment; |
| 1155 | + } |
| 1156 | + |
| 1157 | + |
| 1158 | + static function RGADgainString($namecode, $originatorcode, $replaygain) { |
| 1159 | + if ($replaygain < 0) { |
| 1160 | + $signbit = '1'; |
| 1161 | + } else { |
| 1162 | + $signbit = '0'; |
| 1163 | + } |
| 1164 | + $storedreplaygain = intval(round($replaygain * 10)); |
| 1165 | + $gainstring = str_pad(decbin($namecode), 3, '0', STR_PAD_LEFT); |
| 1166 | + $gainstring .= str_pad(decbin($originatorcode), 3, '0', STR_PAD_LEFT); |
| 1167 | + $gainstring .= $signbit; |
| 1168 | + $gainstring .= str_pad(decbin($storedreplaygain), 9, '0', STR_PAD_LEFT); |
| 1169 | + |
| 1170 | + return $gainstring; |
| 1171 | + } |
| 1172 | + |
| 1173 | + static function RGADamplitude2dB($amplitude) { |
| 1174 | + return 20 * log10($amplitude); |
| 1175 | + } |
| 1176 | + |
| 1177 | + |
| 1178 | + static function GetDataImageSize($imgData, &$imageinfo) { |
| 1179 | + $GetDataImageSize = false; |
| 1180 | + if ($tempfilename = tempnam(GETID3_TEMP_DIR, 'getID3')) { |
| 1181 | + ob_start(); |
| 1182 | + if ($tmp = fopen($tempfilename, 'wb')) { |
| 1183 | + ob_end_clean(); |
| 1184 | + fwrite($tmp, $imgData); |
| 1185 | + fclose($tmp); |
| 1186 | + ob_start(); |
| 1187 | + $GetDataImageSize = GetImageSize($tempfilename, $imageinfo); |
| 1188 | + $errormessage = ob_get_contents(); |
| 1189 | + ob_end_clean(); |
| 1190 | + } else { |
| 1191 | + $errormessage = ob_get_contents(); |
| 1192 | + ob_end_clean(); |
| 1193 | + } |
| 1194 | + unlink($tempfilename); |
| 1195 | + } |
| 1196 | + return $GetDataImageSize; |
| 1197 | + } |
| 1198 | + |
| 1199 | + static function ImageTypesLookup($imagetypeid) { |
| 1200 | + static $ImageTypesLookup = array(); |
| 1201 | + if (empty($ImageTypesLookup)) { |
| 1202 | + $ImageTypesLookup[1] = 'gif'; |
| 1203 | + $ImageTypesLookup[2] = 'jpeg'; |
| 1204 | + $ImageTypesLookup[3] = 'png'; |
| 1205 | + $ImageTypesLookup[4] = 'swf'; |
| 1206 | + $ImageTypesLookup[5] = 'psd'; |
| 1207 | + $ImageTypesLookup[6] = 'bmp'; |
| 1208 | + $ImageTypesLookup[7] = 'tiff (little-endian)'; |
| 1209 | + $ImageTypesLookup[8] = 'tiff (big-endian)'; |
| 1210 | + $ImageTypesLookup[9] = 'jpc'; |
| 1211 | + $ImageTypesLookup[10] = 'jp2'; |
| 1212 | + $ImageTypesLookup[11] = 'jpx'; |
| 1213 | + $ImageTypesLookup[12] = 'jb2'; |
| 1214 | + $ImageTypesLookup[13] = 'swc'; |
| 1215 | + $ImageTypesLookup[14] = 'iff'; |
| 1216 | + } |
| 1217 | + return (isset($ImageTypesLookup[$imagetypeid]) ? $ImageTypesLookup[$imagetypeid] : ''); |
| 1218 | + } |
| 1219 | + |
| 1220 | + static function CopyTagsToComments(&$ThisFileInfo) { |
| 1221 | + |
| 1222 | + // Copy all entries from ['tags'] into common ['comments'] |
| 1223 | + if (!empty($ThisFileInfo['tags'])) { |
| 1224 | + foreach ($ThisFileInfo['tags'] as $tagtype => $tagarray) { |
| 1225 | + foreach ($tagarray as $tagname => $tagdata) { |
| 1226 | + foreach ($tagdata as $key => $value) { |
| 1227 | + if (!empty($value)) { |
| 1228 | + if (empty($ThisFileInfo['comments'][$tagname])) { |
| 1229 | + |
| 1230 | + // fall through and append value |
| 1231 | + |
| 1232 | + } elseif ($tagtype == 'id3v1') { |
| 1233 | + |
| 1234 | + $newvaluelength = strlen(trim($value)); |
| 1235 | + foreach ($ThisFileInfo['comments'][$tagname] as $existingkey => $existingvalue) { |
| 1236 | + $oldvaluelength = strlen(trim($existingvalue)); |
| 1237 | + if (($newvaluelength <= $oldvaluelength) && (substr($existingvalue, 0, $newvaluelength) == trim($value))) { |
| 1238 | + // new value is identical but shorter-than (or equal-length to) one already in comments - skip |
| 1239 | + break 2; |
| 1240 | + } |
| 1241 | + } |
| 1242 | + |
| 1243 | + } else { |
| 1244 | + |
| 1245 | + $newvaluelength = strlen(trim($value)); |
| 1246 | + foreach ($ThisFileInfo['comments'][$tagname] as $existingkey => $existingvalue) { |
| 1247 | + $oldvaluelength = strlen(trim($existingvalue)); |
| 1248 | + if (($newvaluelength > $oldvaluelength) && (substr(trim($value), 0, strlen($existingvalue)) == $existingvalue)) { |
| 1249 | + $ThisFileInfo['comments'][$tagname][$existingkey] = trim($value); |
| 1250 | + break 2; |
| 1251 | + } |
| 1252 | + } |
| 1253 | + |
| 1254 | + } |
| 1255 | + if (empty($ThisFileInfo['comments'][$tagname]) || !in_array(trim($value), $ThisFileInfo['comments'][$tagname])) { |
| 1256 | + $ThisFileInfo['comments'][$tagname][] = trim($value); |
| 1257 | + } |
| 1258 | + } |
| 1259 | + } |
| 1260 | + } |
| 1261 | + } |
| 1262 | + |
| 1263 | + // Copy to ['comments_html'] |
| 1264 | + foreach ($ThisFileInfo['comments'] as $field => $values) { |
| 1265 | + foreach ($values as $index => $value) { |
| 1266 | + $ThisFileInfo['comments_html'][$field][$index] = str_replace('�', '', getid3_lib::MultiByteCharString2HTML($value, $ThisFileInfo['encoding'])); |
| 1267 | + } |
| 1268 | + } |
| 1269 | + } |
| 1270 | + } |
| 1271 | + |
| 1272 | + |
| 1273 | + static function EmbeddedLookup($key, $begin, $end, $file, $name) { |
| 1274 | + |
| 1275 | + // Cached |
| 1276 | + static $cache; |
| 1277 | + if (isset($cache[$file][$name])) { |
| 1278 | + return (isset($cache[$file][$name][$key]) ? $cache[$file][$name][$key] : ''); |
| 1279 | + } |
| 1280 | + |
| 1281 | + // Init |
| 1282 | + $keylength = strlen($key); |
| 1283 | + $line_count = $end - $begin - 7; |
| 1284 | + |
| 1285 | + // Open php file |
| 1286 | + $fp = fopen($file, 'r'); |
| 1287 | + |
| 1288 | + // Discard $begin lines |
| 1289 | + for ($i = 0; $i < ($begin + 3); $i++) { |
| 1290 | + fgets($fp, 1024); |
| 1291 | + } |
| 1292 | + |
| 1293 | + // Loop thru line |
| 1294 | + while (0 < $line_count--) { |
| 1295 | + |
| 1296 | + // Read line |
| 1297 | + $line = ltrim(fgets($fp, 1024), "\t "); |
| 1298 | + |
| 1299 | + // METHOD A: only cache the matching key - less memory but slower on next lookup of not-previously-looked-up key |
| 1300 | + //$keycheck = substr($line, 0, $keylength); |
| 1301 | + //if ($key == $keycheck) { |
| 1302 | + // $cache[$file][$name][$keycheck] = substr($line, $keylength + 1); |
| 1303 | + // break; |
| 1304 | + //} |
| 1305 | + |
| 1306 | + // METHOD B: cache all keys in this lookup - more memory but faster on next lookup of not-previously-looked-up key |
| 1307 | + //$cache[$file][$name][substr($line, 0, $keylength)] = trim(substr($line, $keylength + 1)); |
| 1308 | + $explodedLine = explode("\t", $line, 2); |
| 1309 | + $ThisKey = (isset($explodedLine[0]) ? $explodedLine[0] : ''); |
| 1310 | + $ThisValue = (isset($explodedLine[1]) ? $explodedLine[1] : ''); |
| 1311 | + $cache[$file][$name][$ThisKey] = trim($ThisValue); |
| 1312 | + } |
| 1313 | + |
| 1314 | + // Close and return |
| 1315 | + fclose($fp); |
| 1316 | + return (isset($cache[$file][$name][$key]) ? $cache[$file][$name][$key] : ''); |
| 1317 | + } |
| 1318 | + |
| 1319 | + static function IncludeDependency($filename, $sourcefile, $DieOnFailure=false) { |
| 1320 | + global $GETID3_ERRORARRAY; |
| 1321 | + |
| 1322 | + if (file_exists($filename)) { |
| 1323 | + if (include_once($filename)) { |
| 1324 | + return true; |
| 1325 | + } else { |
| 1326 | + $diemessage = basename($sourcefile).' depends on '.$filename.', which has errors'; |
| 1327 | + } |
| 1328 | + } else { |
| 1329 | + $diemessage = basename($sourcefile).' depends on '.$filename.', which is missing'; |
| 1330 | + } |
| 1331 | + if ($DieOnFailure) { |
| 1332 | + die($diemessage); |
| 1333 | + } else { |
| 1334 | + $GETID3_ERRORARRAY[] = $diemessage; |
| 1335 | + } |
| 1336 | + return false; |
| 1337 | + } |
| 1338 | + |
| 1339 | +} |
| 1340 | + |
| 1341 | +?> |
\ No newline at end of file |
Index: trunk/extensions/TimedMediaHandler/getid3/readme.txt |
— | — | @@ -0,0 +1,574 @@ |
| 2 | +///////////////////////////////////////////////////////////////// |
| 3 | +/// getID3() by James Heinrich <info@getid3.org> // |
| 4 | +// available at http://getid3.sourceforge.net // |
| 5 | +// or http://www.getid3.org // |
| 6 | +///////////////////////////////////////////////////////////////// |
| 7 | +// // |
| 8 | +// changelog.txt - part of getID3() // |
| 9 | +// See readme.txt for more details // |
| 10 | +// /// |
| 11 | +///////////////////////////////////////////////////////////////// |
| 12 | + |
| 13 | + This code is released under the GNU GPL: |
| 14 | + http://www.gnu.org/copyleft/gpl.html |
| 15 | + |
| 16 | + +---------------------------------------------+ |
| 17 | + | If you do use this code somewhere, send me | |
| 18 | + | an email and tell me how/where you used it. | |
| 19 | + | | |
| 20 | + | If you want to donate, there is a link on | |
| 21 | + | http://www.getid3.org for PayPal donations. | |
| 22 | + +---------------------------------------------+ |
| 23 | + |
| 24 | + |
| 25 | + |
| 26 | +Quick Start |
| 27 | +=========================================================================== |
| 28 | + |
| 29 | +Q: How can I check that getID3() works on my server/files? |
| 30 | +A: Unzip getID3() to a directory, then access /demos/demo.browse.php |
| 31 | + |
| 32 | + |
| 33 | + |
| 34 | +Support |
| 35 | +=========================================================================== |
| 36 | + |
| 37 | +Q: I have a question, or I found a bug. What do I do? |
| 38 | +A: The preferred method of support requests and/or bug reports is the |
| 39 | + forum at http://support.getid3.org/ |
| 40 | + |
| 41 | + |
| 42 | + |
| 43 | +Sourceforge Notification |
| 44 | +=========================================================================== |
| 45 | + |
| 46 | +It's highly recommended that you sign up for notification from |
| 47 | +Sourceforge for when new versions are released. Please visit: |
| 48 | +http://sourceforge.net/project/showfiles.php?group_id=55859 |
| 49 | +and click the little "monitor package" icon/link. If you're |
| 50 | +previously signed up for the mailing list, be aware that it has |
| 51 | +been discontinued, only the automated Sourceforge notification |
| 52 | +will be used from now on. |
| 53 | + |
| 54 | + |
| 55 | + |
| 56 | +What does getID3() do? |
| 57 | +=========================================================================== |
| 58 | + |
| 59 | +Reads & parses (to varying degrees): |
| 60 | + � tags: |
| 61 | + * APE (v1 and v2) |
| 62 | + * ID3v1 (& ID3v1.1) |
| 63 | + * ID3v2 (v2.4, v2.3, v2.2) |
| 64 | + * Lyrics3 (v1 & v2) |
| 65 | + |
| 66 | + � audio-lossy: |
| 67 | + * MP3/MP2/MP1 |
| 68 | + * MPC / Musepack |
| 69 | + * Ogg (Vorbis, OggFLAC, Speex) |
| 70 | + * AC3 |
| 71 | + * DTS |
| 72 | + * RealAudio |
| 73 | + * Speex |
| 74 | + * DSS |
| 75 | + * VQF |
| 76 | + |
| 77 | + � audio-lossless: |
| 78 | + * AIFF |
| 79 | + * AU |
| 80 | + * Bonk |
| 81 | + * CD-audio (*.cda) |
| 82 | + * FLAC |
| 83 | + * LA (Lossless Audio) |
| 84 | + * LiteWave |
| 85 | + * LPAC |
| 86 | + * MIDI |
| 87 | + * Monkey's Audio |
| 88 | + * OptimFROG |
| 89 | + * RKAU |
| 90 | + * Shorten |
| 91 | + * TTA |
| 92 | + * VOC |
| 93 | + * WAV (RIFF) |
| 94 | + * WavPack |
| 95 | + |
| 96 | + � audio-video: |
| 97 | + * ASF: ASF, Windows Media Audio (WMA), Windows Media Video (WMV) |
| 98 | + * AVI (RIFF) |
| 99 | + * Flash |
| 100 | + * Matroska (MKV) |
| 101 | + * MPEG-1 / MPEG-2 |
| 102 | + * NSV (Nullsoft Streaming Video) |
| 103 | + * Quicktime |
| 104 | + * RealVideo |
| 105 | + |
| 106 | + � still image: |
| 107 | + * BMP |
| 108 | + * GIF |
| 109 | + * JPEG |
| 110 | + * PNG |
| 111 | + * TIFF |
| 112 | + * SWF (Flash) |
| 113 | + * PhotoCD |
| 114 | + |
| 115 | + � data: |
| 116 | + * ISO-9660 CD-ROM image (directory structure) |
| 117 | + * SZIP (limited support) |
| 118 | + * ZIP (directory structure) |
| 119 | + * TAR |
| 120 | + * CUE |
| 121 | + |
| 122 | + |
| 123 | +Writes: |
| 124 | + * ID3v1 (& ID3v1.1) |
| 125 | + * ID3v2 (v2.3 & v2.4) |
| 126 | + * VorbisComment on OggVorbis |
| 127 | + * VorbisComment on FLAC (not OggFLAC) |
| 128 | + * APE v2 |
| 129 | + * Lyrics3 (delete only) |
| 130 | + |
| 131 | + |
| 132 | + |
| 133 | +Requirements |
| 134 | +=========================================================================== |
| 135 | + |
| 136 | +* PHP 4.2.0 (or higher) for getID3() 1.7.x (and earlier) |
| 137 | +* PHP 5.3.0 (or higher) for getID3() 1.8.x (and up) |
| 138 | +* PHP 5.0.0 (or higher) for getID3() 2.0.x (and up) |
| 139 | +* at least 4MB memory for PHP. 8MB is highly recommended. |
| 140 | + 12MB is required with all modules loaded. |
| 141 | + |
| 142 | + |
| 143 | + |
| 144 | +Usage |
| 145 | +=========================================================================== |
| 146 | + |
| 147 | +See /demos/demo.basic.php for a very basic use of getID3() with no |
| 148 | +fancy output, just scanning one file. |
| 149 | + |
| 150 | +See structure.txt for the returned data structure. |
| 151 | + |
| 152 | +*> For an example of a complete directory-browsing, <* |
| 153 | +*> file-scanning implementation of getID3(), please run <* |
| 154 | +*> /demos/demo.browse.php <* |
| 155 | + |
| 156 | +See /demos/demo.mysql.php for a sample recursive scanning code that |
| 157 | +scans every file in a given directory, and all sub-directories, stores |
| 158 | +the results in a database and allows various analysis / maintenance |
| 159 | +operations |
| 160 | + |
| 161 | +To analyze remote files over HTTP or FTP you need to copy the file |
| 162 | +locally first before running getID3(). Your code would look something |
| 163 | +like this: |
| 164 | + |
| 165 | +// Copy remote file locally to scan with getID3() |
| 166 | +$remotefilename = 'http://www.example.com/filename.mp3'; |
| 167 | +if ($fp_remote = fopen($remotefilename, 'rb')) { |
| 168 | + $localtempfilename = tempnam('/tmp', 'getID3'); |
| 169 | + if ($fp_local = fopen($localtempfilename, 'wb')) { |
| 170 | + while ($buffer = fread($fp_remote, 8192)) { |
| 171 | + fwrite($fp_local, $buffer); |
| 172 | + } |
| 173 | + fclose($fp_local); |
| 174 | + |
| 175 | + // Initialize getID3 engine |
| 176 | + $getID3 = new getID3; |
| 177 | + |
| 178 | + $ThisFileInfo = $getID3->analyze($filename); |
| 179 | + |
| 180 | + // Delete temporary file |
| 181 | + unlink($localtempfilename); |
| 182 | + } |
| 183 | + fclose($fp_remote); |
| 184 | +} |
| 185 | + |
| 186 | + |
| 187 | +See /demos/demo.write.php for how to write tags. |
| 188 | + |
| 189 | + |
| 190 | + |
| 191 | +What does the returned data structure look like? |
| 192 | +=========================================================================== |
| 193 | + |
| 194 | +See structure.txt |
| 195 | + |
| 196 | +It is recommended that you look at the output of |
| 197 | +/demos/demo.browse.php scanning the file(s) you're interested in to |
| 198 | +confirm what data is actually returned for any particular filetype in |
| 199 | +general, and your files in particular, as the actual data returned |
| 200 | +may vary considerably depending on what information is available in |
| 201 | +the file itself. |
| 202 | + |
| 203 | + |
| 204 | + |
| 205 | +Notes |
| 206 | +=========================================================================== |
| 207 | + |
| 208 | +getID3() 1.x: |
| 209 | +If the format parser encounters a critical problem, it will return |
| 210 | +something in $fileinfo['error'], describing the encountered error. If |
| 211 | +a less critical error or notice is generated it will appear in |
| 212 | +$fileinfo['warning']. Both keys may contain more than one warning or |
| 213 | +error. If something is returned in ['error'] then the file was not |
| 214 | +correctly parsed and returned data may or may not be correct and/or |
| 215 | +complete. If something is returned in ['warning'] (and not ['error']) |
| 216 | +then the data that is returned is OK - usually getID3() is reporting |
| 217 | +errors in the file that have been worked around due to known bugs in |
| 218 | +other programs. Some warnings may indicate that the data that is |
| 219 | +returned is OK but that some data could not be extracted due to |
| 220 | +errors in the file. |
| 221 | + |
| 222 | +getID3() 2.x: |
| 223 | +See above except errors are thrown (so you will only get one error). |
| 224 | + |
| 225 | + |
| 226 | + |
| 227 | +Disclaimer |
| 228 | +=========================================================================== |
| 229 | + |
| 230 | +getID3() has been tested on many systems, on many types of files, |
| 231 | +under many operating systems, and is generally believe to be stable |
| 232 | +and safe. That being said, there is still the chance there is an |
| 233 | +undiscovered and/or unfixed bug that may potentially corrupt your |
| 234 | +file, especially within the writing functions. By using getID3() you |
| 235 | +agree that it's not my fault if any of your files are corrupted. |
| 236 | +In fact, I'm not liable for anything :) |
| 237 | + |
| 238 | + |
| 239 | + |
| 240 | +License |
| 241 | +=========================================================================== |
| 242 | + |
| 243 | +GNU General Public License - see license.txt |
| 244 | + |
| 245 | +This program is free software; you can redistribute it and/or |
| 246 | +modify it under the terms of the GNU General Public License |
| 247 | +as published by the Free Software Foundation; either version 2 |
| 248 | +of the License, or (at your option) any later version. |
| 249 | + |
| 250 | +This program is distributed in the hope that it will be useful, |
| 251 | +but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 252 | +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| 253 | +GNU General Public License for more details. |
| 254 | + |
| 255 | +You should have received a copy of the GNU General Public License |
| 256 | +along with this program; if not, write to: |
| 257 | +Free Software Foundation, Inc. |
| 258 | +59 Temple Place - Suite 330 |
| 259 | +Boston, MA 02111-1307, USA. |
| 260 | + |
| 261 | +FAQ: |
| 262 | +Q: Can I use getID3() in my program? Do I need a commercial license? |
| 263 | +A: You're generally free to use getID3 however you see fit. The only |
| 264 | + case in which you would require a commercial license is if you're |
| 265 | + selling your closed-source program that integrates getID3. If you |
| 266 | + sell your program including a copy of getID3, that's fine as long |
| 267 | + as you include a copy of the sourcecode when you sell it. Or you |
| 268 | + can distribute your code without getID3 and say "download it from |
| 269 | + getid3.sourceforge.net" |
| 270 | + |
| 271 | + |
| 272 | + |
| 273 | +Future Plans |
| 274 | +=========================================================================== |
| 275 | + |
| 276 | +* Writing support for Real |
| 277 | +* Better support for MP4 container format |
| 278 | +* Scan for appended ID3v2 tag at end of file per ID3v2.4 specs (Section 5.0) |
| 279 | +* Support for JPEG-2000 (http://www.morgan-multimedia.com/jpeg2000_overview.htm) |
| 280 | +* Support for MOD (mod/stm/s3m/it/xm/mtm/ult/669) |
| 281 | +* Support for ACE (thanks Vince) |
| 282 | +* Support for Ogg other than Vorbis, Speex and OggFlac (ie. Ogg+Xvid) |
| 283 | +* Ability to create Xing/LAME VBR header for VBR MP3s that are missing VBR header |
| 284 | +* Ability to "clean" ID3v2 padding (replace invalid padding with valid padding) |
| 285 | +* Warn if MP3s change version mid-stream (in full-scan mode) |
| 286 | +* check for corrupt/broken mid-file MP3 streams in histogram scan |
| 287 | +* Support for lossless-compression formats |
| 288 | + (http://www.firstpr.com.au/audiocomp/lossless/#Links) |
| 289 | + (http://compression.ca/act-sound.html) |
| 290 | + (http://web.inter.nl.net/users/hvdh/lossless/lossless.htm) |
| 291 | +* Support for RIFF-INFO chunks |
| 292 | + * http://lotto.st-andrews.ac.uk/~njh/tag_interchange.html |
| 293 | + (thanks Nick Humfrey <njh�surgeradio*co*uk>) |
| 294 | + * http://abcavi.narod.ru/sof/abcavi/infotags.htm |
| 295 | + (thanks Kibi) |
| 296 | +* Better support for Bink video |
| 297 | +* http://www.hr/josip/DSP/AudioFile2.html |
| 298 | +* http://www.pcisys.net/~melanson/codecs/ |
| 299 | +* Detect mp3PRO |
| 300 | +* Support for PSD |
| 301 | +* Support for JPC |
| 302 | +* Support for JP2 |
| 303 | +* Support for JPX |
| 304 | +* Support for JB2 |
| 305 | +* Support for IFF |
| 306 | +* Support for ICO |
| 307 | +* Support for ANI |
| 308 | +* Support for EXE (comments, author, etc) (thanks p*quaedackers�planet*nl) |
| 309 | +* Support for DVD-IFO (region, subtitles, aspect ratio, etc) |
| 310 | + (thanks p*quaedackers�planet*nl) |
| 311 | +* More complete support for SWF - parsing encapsulated MP3 and/or JPEG content |
| 312 | + (thanks n8n8�yahoo*com) |
| 313 | +* Support for a2b |
| 314 | +* Optional scan-through-frames for AVI verification |
| 315 | + (thanks rockcohen�massive-interactive*nl) |
| 316 | +* Support for TTF (thanks info�butterflyx*com) |
| 317 | +* Support for DSS (http://www.getid3.org/phpBB3/viewtopic.php?t=171) |
| 318 | +* Support for SMAF (http://smaf-yamaha.com/what/demo.html) |
| 319 | + http://www.getid3.org/phpBB3/viewtopic.php?t=182 |
| 320 | +* Support for AMR (http://www.getid3.org/phpBB3/viewtopic.php?t=195) |
| 321 | +* Support for 3gpp (http://www.getid3.org/phpBB3/viewtopic.php?t=195) |
| 322 | +* Support for ID4 (http://www.wackysoft.cjb.net grizlyY2K�hotmail*com) |
| 323 | +* Parse XML data returned in Ogg comments |
| 324 | +* Parse XML data from Quicktime SMIL metafiles (klausrath�mac*com) |
| 325 | +* ID3v2 genre string creator function |
| 326 | +* More complete parsing of JPG |
| 327 | +* Support for all old-style ASF packets |
| 328 | +* ASF/WMA/WMV tag writing |
| 329 | +* Parse declared T??? ID3v2 text information frames, where appropriate |
| 330 | + (thanks Christian Fritz for the idea) |
| 331 | +* Recognize encoder: |
| 332 | + http://www.guerillasoft.com/EncSpot2/index.html |
| 333 | + http://ff123.net/identify.html |
| 334 | + http://www.hydrogenaudio.org/?act=ST&f=16&t=9414 |
| 335 | + http://www.hydrogenaudio.org/?showtopic=11785 |
| 336 | +* Support for other OS/2 bitmap structures: Bitmap Array('BA'), |
| 337 | + Color Icon('CI'), Color Pointer('CP'), Icon('IC'), Pointer ('PT') |
| 338 | + http://netghost.narod.ru/gff/graphics/summary/os2bmp.htm |
| 339 | +* Support for WavPack RAW mode |
| 340 | +* ASF/WMA/WMV data packet parsing |
| 341 | +* ID3v2FrameFlagsLookupTagAlter() |
| 342 | +* ID3v2FrameFlagsLookupFileAlter() |
| 343 | +* obey ID3v2 tag alter/preserve/discard rules |
| 344 | +* http://www.geocities.com/SiliconValley/Sector/9654/Softdoc/Illyrium/Aolyr.htm |
| 345 | +* proper checking for LINK/LNK frame validity in ID3v2 writing |
| 346 | +* proper checking for ASPI-TLEN frame validity in ID3v2 writing |
| 347 | +* proper checking for COMR frame validity in ID3v2 writing |
| 348 | +* http://www.geocities.co.jp/SiliconValley-Oakland/3664/index.html |
| 349 | +* decode GEOB ID3v2 structure as encoded by RealJukebox, |
| 350 | + decode NCON ID3v2 structure as encoded by MusicMatch |
| 351 | + (probably won't happen - the formats are proprietary) |
| 352 | + |
| 353 | + |
| 354 | + |
| 355 | +Known Bugs/Issues in getID3() that may be fixed eventually |
| 356 | +=========================================================================== |
| 357 | + |
| 358 | +* Cannot determine bitrate for MPEG video with VBR video data |
| 359 | + (need documentation) |
| 360 | +* Interlace/progressive cannot be determined for MPEG video |
| 361 | + (need documentation) |
| 362 | +* MIDI playtime is sometimes inaccurate |
| 363 | +* AAC-RAW mode files cannot be identified |
| 364 | +* WavPack-RAW mode files cannot be identified |
| 365 | +* mp4 files report lots of "Unknown QuickTime atom type" |
| 366 | + (need documentation) |
| 367 | +* Encrypted ASF/WMA/WMV files warn about "unhandled GUID |
| 368 | + ASF_Content_Encryption_Object" |
| 369 | +* Bitrate split between audio and video cannot be calculated for |
| 370 | + NSV, only the total bitrate. (need documentation) |
| 371 | +* All Ogg formats (Vorbis, OggFLAC, Speex) are affected by the |
| 372 | + problem of large VorbisComments spanning multiple Ogg pages, but |
| 373 | + but only OggVorbis files can be processed with vorbiscomment. |
| 374 | +* The version of "head" supplied with Mac OS 10.2.8 (maybe other |
| 375 | + versions too) does only understands a single option (-n) and |
| 376 | + therefore fails. getID3 ignores this and returns wrong md5_data. |
| 377 | + |
| 378 | + |
| 379 | + |
| 380 | +Known Bugs/Issues in getID3() that cannot be fixed |
| 381 | +-------------------------------------------------- |
| 382 | + |
| 383 | +* Files larger than 2GB cannot always be parsed fully by getID3() |
| 384 | + due to limitations in the PHP filesystem functions. |
| 385 | + NOTE: Since v1.7.8b3 there is partial support for larger-than- |
| 386 | + 2GB files, most of which will parse OK, as long as no critical |
| 387 | + data is located beyond the 2GB offset. |
| 388 | + Known will-work: |
| 389 | + * ZIP (format doesn't support files >2GB) |
| 390 | + * FLAC (current encoders don't support files >2GB) |
| 391 | + Known will-not-work: |
| 392 | + * ID3v1 tags (always located at end-of-file) |
| 393 | + * Lyrics3 tags (always located at end-of-file) |
| 394 | + * APE tags (always located at end-of-file) |
| 395 | + Maybe-will-work: |
| 396 | + * Quicktime (will work if needed metadata is before 2GB offset, |
| 397 | + that is if the file has been hinted/optimized for streaming) |
| 398 | + * RIFF.WAV (should work fine, but gives warnings about not being |
| 399 | + able to parse all chunks) |
| 400 | + * RIFF.AVI (playtime will probably be wrong, is only based on |
| 401 | + "movi" chunk that fits in the first 2GB, should issue error |
| 402 | + to show that playtime is incorrect. Other data should be mostly |
| 403 | + correct, assuming that data is constant throughout the file) |
| 404 | + |
| 405 | + |
| 406 | + |
| 407 | +Known Bugs/Issues in other programs |
| 408 | +----------------------------------- |
| 409 | + |
| 410 | +* Windows Media Player (up to v11) and iTunes (up to v10+) do |
| 411 | + not correctly handle ID3v2.3 tags with UTF-16BE+BOM |
| 412 | + encoding (they assume the data is UTF-16LE+BOM and either |
| 413 | + crash (WMP) or output Asian character set (iTunes) |
| 414 | +* Winamp (up to v2.80 at least) does not support ID3v2.4 tags, |
| 415 | + only ID3v2.3 |
| 416 | + see: http://forums.winamp.com/showthread.php?postid=387524 |
| 417 | +* Some versions of Helium2 (www.helium2.com) do not write |
| 418 | + ID3v2.4-compliant Frame Sizes, even though the tag is marked |
| 419 | + as ID3v2.4) (detected by getID3()) |
| 420 | +* MP3ext V3.3.17 places a non-compliant padding string at the end |
| 421 | + of the ID3v2 header. This is supposedly fixed in v3.4b21 but |
| 422 | + only if you manually add a registry key. This fix is not yet |
| 423 | + confirmed. (detected by getID3()) |
| 424 | +* CDex v1.40 (fixed by v1.50b7) writes non-compliant Ogg comment |
| 425 | + strings, supposed to be in the format "NAME=value" but actually |
| 426 | + written just "value" (detected by getID3()) |
| 427 | +* Oggenc 0.9-rc3 flags the encoded file as ABR whether it's |
| 428 | + actually ABR or VBR. |
| 429 | +* iTunes (versions "X v2.0.3", "v3.0.1" are known-guilty, probably |
| 430 | + other versions are too) writes ID3v2.3 comment tags using a |
| 431 | + frame name 'COM ' which is not valid for ID3v2.3+ (it's an |
| 432 | + ID3v2.2-style frame name) (detected by getID3()) |
| 433 | +* MP2enc does not encode mono CBR MP2 files properly (half speed |
| 434 | + sound and double playtime) |
| 435 | +* MP2enc does not encode mono VBR MP2 files properly (actually |
| 436 | + encoded as stereo) |
| 437 | +* tooLAME does not encode mono VBR MP2 files properly (actually |
| 438 | + encoded as stereo) |
| 439 | +* AACenc encodes files in VBR mode (actually ABR) even if CBR is |
| 440 | + specified |
| 441 | +* AAC/ADIF - bitrate_mode = cbr for vbr files |
| 442 | +* LAME 3.90-3.92 prepends one frame of null data (space for the |
| 443 | + LAME/VBR header, but it never gets written) when encoding in CBR |
| 444 | + mode with the DLL |
| 445 | +* Ahead Nero encodes TwinVQF with a DSIZ value (which is supposed |
| 446 | + to be the filesize in bytes) of "0" for TwinVQF v1.0 and "1" for |
| 447 | + TwinVQF v2.0 (detected by getID3()) |
| 448 | +* Ahead Nero encodes TwinVQF files 1 second shorter than they |
| 449 | + should be |
| 450 | +* AAC-ADTS files are always actually encoded VBR, even if CBR mode |
| 451 | + is specified (the CBR-mode switches on the encoder enable ABR |
| 452 | + mode, not CBR as such, but it's not possible to tell the |
| 453 | + difference between such ABR files and true VBR) |
| 454 | +* STREAMINFO.audio_signature in OggFLAC is always null. "The reason |
| 455 | + it's like that is because there is no seeking support in |
| 456 | + libOggFLAC yet, so it has no way to go back and write the |
| 457 | + computed sum after encoding. Seeking support in Ogg FLAC is the |
| 458 | + #1 item for the next release." - Josh Coalson (FLAC developer) |
| 459 | + NOTE: getID3() will calculate md5_data in a method similar to |
| 460 | + other file formats, but that value cannot be compared to the |
| 461 | + md5_data value from FLAC data in a FLAC file format. |
| 462 | +* STREAMINFO.audio_signature is not calculated in FLAC v0.3.0 & |
| 463 | + v0.4.0 - getID3() will calculate md5_data in a method similar to |
| 464 | + other file formats, but that value cannot be compared to the |
| 465 | + md5_data value from FLAC v0.5.0+ |
| 466 | +* RioPort (various versions including 2.0 and 3.11) tags ID3v2 with |
| 467 | + a WCOM frame that has no data portion |
| 468 | +* Earlier versions of Coolplayer adds illegal ID3 tags to Ogg Vorbis |
| 469 | + files, thus making them corrupt. |
| 470 | +* Meracl ID3 Tag Writer v1.3.4 (and older) incorrectly truncates the |
| 471 | + last byte of data from an MP3 file when appending a new ID3v1 tag. |
| 472 | + (detected by getID3()) |
| 473 | +* Lossless-Audio files encoded with and without the -noseek switch |
| 474 | + do actually differ internally and therefore cannot match md5_data |
| 475 | +* iTunes has been known to append a new ID3v1 tag on the end of an |
| 476 | + existing ID3v1 tag when ID3v2 tag is also present |
| 477 | + (detected by getID3()) |
| 478 | + |
| 479 | + |
| 480 | + |
| 481 | + |
| 482 | +Reference material: |
| 483 | +=========================================================================== |
| 484 | + |
| 485 | +[www.id3.org material now mirrored at http://id3lib.sourceforge.net/id3/] |
| 486 | +* http://www.id3.org/id3v2.4.0-structure.txt |
| 487 | +* http://www.id3.org/id3v2.4.0-frames.txt |
| 488 | +* http://www.id3.org/id3v2.4.0-changes.txt |
| 489 | +* http://www.id3.org/id3v2.3.0.txt |
| 490 | +* http://www.id3.org/id3v2-00.txt |
| 491 | +* http://www.id3.org/mp3frame.html |
| 492 | +* http://minnie.tuhs.org/pipermail/mp3encoder/2001-January/001800.html <mathewhendry@hotmail.com> |
| 493 | +* http://www.dv.co.yu/mpgscript/mpeghdr.htm |
| 494 | +* http://www.mp3-tech.org/programmer/frame_header.html |
| 495 | +* http://users.belgacom.net/gc247244/extra/tag.html |
| 496 | +* http://gabriel.mp3-tech.org/mp3infotag.html |
| 497 | +* http://www.id3.org/iso4217.html |
| 498 | +* http://www.unicode.org/Public/MAPPINGS/ISO8859/8859-1.TXT |
| 499 | +* http://www.xiph.org/ogg/vorbis/doc/framing.html |
| 500 | +* http://www.xiph.org/ogg/vorbis/doc/v-comment.html |
| 501 | +* http://leknor.com/code/php/class.ogg.php.txt |
| 502 | +* http://www.id3.org/iso639-2.html |
| 503 | +* http://www.id3.org/lyrics3.html |
| 504 | +* http://www.id3.org/lyrics3200.html |
| 505 | +* http://www.psc.edu/general/software/packages/ieee/ieee.html |
| 506 | +* http://www.scri.fsu.edu/~jac/MAD3401/Backgrnd/ieee-expl.html |
| 507 | +* http://www.scri.fsu.edu/~jac/MAD3401/Backgrnd/binary.html |
| 508 | +* http://www.jmcgowan.com/avi.html |
| 509 | +* http://www.wotsit.org/ |
| 510 | +* http://www.herdsoft.com/ti/davincie/davp3xo2.htm |
| 511 | +* http://www.mathdogs.com/vorbis-illuminated/bitstream-appendix.html |
| 512 | +* "Standard MIDI File Format" by Dustin Caldwell (from www.wotsit.org) |
| 513 | +* http://midistudio.com/Help/GMSpecs_Patches.htm |
| 514 | +* http://www.xiph.org/archives/vorbis/200109/0459.html |
| 515 | +* http://www.replaygain.org/ |
| 516 | +* http://www.lossless-audio.com/ |
| 517 | +* http://download.microsoft.com/download/winmediatech40/Doc/1.0/WIN98MeXP/EN-US/ASF_Specification_v.1.0.exe |
| 518 | +* http://mediaxw.sourceforge.net/files/doc/Active%20Streaming%20Format%20(ASF)%201.0%20Specification.pdf |
| 519 | +* http://www.uni-jena.de/~pfk/mpp/sv8/ |
| 520 | +* http://jfaul.de/atl/ |
| 521 | +* http://www.uni-jena.de/~pfk/mpp/ |
| 522 | +* http://www.libpng.org/pub/png/spec/png-1.2-pdg.html |
| 523 | +* http://www.real.com/devzone/library/creating/rmsdk/doc/rmff.htm |
| 524 | +* http://www.fastgraph.com/help/bmp_os2_header_format.html |
| 525 | +* http://netghost.narod.ru/gff/graphics/summary/os2bmp.htm |
| 526 | +* http://flac.sourceforge.net/format.html |
| 527 | +* http://www.research.att.com/projects/mpegaudio/mpeg2.html |
| 528 | +* http://www.audiocoding.com/wiki/index.php?page=AAC |
| 529 | +* http://libmpeg.org/mpeg4/doc/w2203tfs.pdf |
| 530 | +* http://www.geocities.com/xhelmboyx/quicktime/formats/qtm-layout.txt |
| 531 | +* http://developer.apple.com/techpubs/quicktime/qtdevdocs/RM/frameset.htm |
| 532 | +* http://www.nullsoft.com/nsv/ |
| 533 | +* http://www.wotsit.org/download.asp?f=iso9660 |
| 534 | +* http://sandbox.mc.edu/~bennet/cs110/tc/tctod.html |
| 535 | +* http://www.cdroller.com/htm/readdata.html |
| 536 | +* http://www.speex.org/manual/node10.html |
| 537 | +* http://www.harmony-central.com/Computer/Programming/aiff-file-format.doc |
| 538 | +* http://www.faqs.org/rfcs/rfc2361.html |
| 539 | +* http://ghido.shelter.ro/ |
| 540 | +* http://www.ebu.ch/tech_t3285.pdf |
| 541 | +* http://www.sr.se/utveckling/tu/bwf |
| 542 | +* http://ftp.aessc.org/pub/aes46-2002.pdf |
| 543 | +* http://cartchunk.org:8080/ |
| 544 | +* http://www.broadcastpapers.com/radio/cartchunk01.htm |
| 545 | +* http://www.hr/josip/DSP/AudioFile2.html |
| 546 | +* http://home.attbi.com/~chris.bagwell/AudioFormats-11.html |
| 547 | +* http://www.pure-mac.com/extkey.html |
| 548 | +* http://cesnet.dl.sourceforge.net/sourceforge/bonkenc/bonk-binary-format-0.9.txt |
| 549 | +* http://www.headbands.com/gspot/ |
| 550 | +* http://www.openswf.org/spec/SWFfileformat.html |
| 551 | +* http://j-faul.virtualave.net/ |
| 552 | +* http://www.btinternet.com/~AnthonyJ/Atari/programming/avr_format.html |
| 553 | +* http://cui.unige.ch/OSG/info/AudioFormats/ap11.html |
| 554 | +* http://sswf.sourceforge.net/SWFalexref.html |
| 555 | +* http://www.geocities.com/xhelmboyx/quicktime/formats/qti-layout.txt |
| 556 | +* http://www-lehre.informatik.uni-osnabrueck.de/~fbstark/diplom/docs/swf/Flash_Uncovered.htm |
| 557 | +* http://developer.apple.com/quicktime/icefloe/dispatch012.html |
| 558 | +* http://www.csdn.net/Dev/Format/graphics/PCD.htm |
| 559 | +* http://tta.iszf.irk.ru/ |
| 560 | +* http://www.atsc.org/standards/a_52a.pdf |
| 561 | +* http://www.alanwood.net/unicode/ |
| 562 | +* http://www.freelists.org/archives/matroska-devel/07-2003/msg00010.html |
| 563 | +* http://www.its.msstate.edu/net/real/reports/config/tags.stats |
| 564 | +* http://homepages.slingshot.co.nz/~helmboy/quicktime/formats/qtm-layout.txt |
| 565 | +* http://brennan.young.net/Comp/LiveStage/things.html |
| 566 | +* http://www.multiweb.cz/twoinches/MP3inside.htm |
| 567 | +* http://www.geocities.co.jp/SiliconValley-Oakland/3664/alittle.html#GenreExtended |
| 568 | +* http://www.mactech.com/articles/mactech/Vol.06/06.01/SANENormalized/ |
| 569 | +* http://www.unicode.org/unicode/faq/utf_bom.html |
| 570 | +* http://tta.corecodec.org/?menu=format |
| 571 | +* http://www.scvi.net/nsvformat.htm |
| 572 | +* http://pda.etsi.org/pda/queryform.asp |
| 573 | +* http://cpansearch.perl.org/src/RGIBSON/Audio-DSS-0.02/lib/Audio/DSS.pm |
| 574 | +* http://trac.musepack.net/trac/wiki/SV8Specification |
| 575 | +* http://wyday.com/cuesharp/specification.php |
\ No newline at end of file |
Property changes on: trunk/extensions/TimedMediaHandler/getid3/readme.txt |
___________________________________________________________________ |
Added: svn:mime-type |
1 | 576 | + text/plain |
Index: trunk/extensions/TimedMediaHandler/handlers/OggHandler.php |
— | — | @@ -0,0 +1,113 @@ |
| 2 | +<?php |
| 3 | +/** |
| 4 | + * ogg handler |
| 5 | + */ |
| 6 | +class OggHandler extends TimedMediaHandler { |
| 7 | + const METADATA_VERSION = 2; |
| 8 | + |
| 9 | + function getMetadata( $image, $path ) { |
| 10 | + $metadata = array( 'version' => self::METADATA_VERSION ); |
| 11 | + |
| 12 | + if ( !class_exists( 'File_Ogg' ) ) { |
| 13 | + require( 'File/Ogg.php' ); |
| 14 | + } |
| 15 | + try { |
| 16 | + $f = new File_Ogg( $path ); |
| 17 | + $streams = array(); |
| 18 | + foreach ( $f->listStreams() as $streamType => $streamIDs ) { |
| 19 | + foreach ( $streamIDs as $streamID ) { |
| 20 | + $stream = $f->getStream( $streamID ); |
| 21 | + $streams[$streamID] = array( |
| 22 | + 'serial' => $stream->getSerial(), |
| 23 | + 'group' => $stream->getGroup(), |
| 24 | + 'type' => $stream->getType(), |
| 25 | + 'vendor' => $stream->getVendor(), |
| 26 | + 'length' => $stream->getLength(), |
| 27 | + 'size' => $stream->getSize(), |
| 28 | + 'header' => $stream->getHeader(), |
| 29 | + 'comments' => $stream->getComments() |
| 30 | + ); |
| 31 | + } |
| 32 | + } |
| 33 | + $metadata['streams'] = $streams; |
| 34 | + $metadata['length'] = $f->getLength(); |
| 35 | + // Get the offset of the file (in cases where the file is a segment copy) |
| 36 | + $metadata['offset'] = $f->getStartOffset(); |
| 37 | + } catch ( PEAR_Exception $e ) { |
| 38 | + // File not found, invalid stream, etc. |
| 39 | + $metadata['error'] = array( |
| 40 | + 'message' => $e->getMessage(), |
| 41 | + 'code' => $e->getCode() |
| 42 | + ); |
| 43 | + } |
| 44 | + return serialize( $metadata ); |
| 45 | + } |
| 46 | + /** |
| 47 | + * Get the "media size" |
| 48 | + * |
| 49 | + */ |
| 50 | + function getImageSize( $file, $path, $metadata = false ) { |
| 51 | + global $wgMediaVideoTypes; |
| 52 | + // Just return the size of the first video stream |
| 53 | + if ( $metadata === false ) { |
| 54 | + $metadata = $file->getMetadata(); |
| 55 | + } |
| 56 | + $metadata = $this->unpackMetadata( $metadata ); |
| 57 | + if ( isset( $metadata['error'] ) || !isset( $metadata['streams'] ) ) { |
| 58 | + return false; |
| 59 | + } |
| 60 | + foreach ( $metadata['streams'] as $stream ) { |
| 61 | + if ( in_array( $stream['type'], $wgMediaVideoTypes ) ) { |
| 62 | + return array( |
| 63 | + $stream['header']['PICW'], |
| 64 | + $stream['header']['PICH'] |
| 65 | + ); |
| 66 | + } |
| 67 | + } |
| 68 | + return array( false, false ); |
| 69 | + } |
| 70 | + |
| 71 | + function unpackMetadata( $metadata ) { |
| 72 | + $unser = @unserialize( $metadata ); |
| 73 | + if ( isset( $unser['version'] ) && $unser['version'] == self::METADATA_VERSION ) { |
| 74 | + return $unser; |
| 75 | + } else { |
| 76 | + return false; |
| 77 | + } |
| 78 | + } |
| 79 | + |
| 80 | + function getMetadataType( $image = '' ) { |
| 81 | + return 'ogg'; |
| 82 | + } |
| 83 | + |
| 84 | + function getStreamTypes( $file ) { |
| 85 | + $streamTypes = ''; |
| 86 | + $metadata = $this->unpackMetadata( $file->getMetadata() ); |
| 87 | + if ( !$metadata || isset( $metadata['error'] ) ) { |
| 88 | + return false; |
| 89 | + } |
| 90 | + foreach ( $metadata['streams'] as $stream ) { |
| 91 | + $streamTypes[$stream['type']] = true; |
| 92 | + } |
| 93 | + return array_keys( $streamTypes ); |
| 94 | + } |
| 95 | + |
| 96 | + function getOffset( $file ){ |
| 97 | + $metadata = $this->unpackMetadata( $file->getMetadata() ); |
| 98 | + if ( !$metadata || isset( $metadata['error'] ) || !isset( $metadata['offset']) ) { |
| 99 | + return 0; |
| 100 | + } else { |
| 101 | + return $metadata['offset']; |
| 102 | + } |
| 103 | + } |
| 104 | + |
| 105 | + function getLength( $file ) { |
| 106 | + $metadata = $this->unpackMetadata( $file->getMetadata() ); |
| 107 | + if ( !$metadata || isset( $metadata['error'] ) ) { |
| 108 | + return 0; |
| 109 | + } else { |
| 110 | + return $metadata['length']; |
| 111 | + } |
| 112 | + } |
| 113 | + |
| 114 | +} |
\ No newline at end of file |
Index: trunk/extensions/TimedMediaHandler/handlers/WebMHandler.php |
— | — | @@ -0,0 +1,91 @@ |
| 2 | +<?php |
| 3 | +/** |
| 4 | + * WebM handler |
| 5 | + */ |
| 6 | +class WebMHandler extends TimedMediaHandler { |
| 7 | + const METADATA_VERSION = 1; |
| 8 | + |
| 9 | + function getMetadata( $image, $path ) { |
| 10 | + $metadata = array( 'version' => self::METADATA_VERSION ); |
| 11 | + |
| 12 | + $getID3 = new getID3(); |
| 13 | + |
| 14 | + // Don't grab stuff we don't use: |
| 15 | + $getID3->option_tag_id3v1 = false; // Read and process ID3v1 tags |
| 16 | + $getID3->option_tag_id3v2 = false; // Read and process ID3v2 tags |
| 17 | + $getID3->option_tag_lyrics3 = false; // Read and process Lyrics3 tags |
| 18 | + $getID3->option_tag_apetag = false; // Read and process APE tags |
| 19 | + $getID3->option_tags_process = false; // Copy tags to root key 'tags' and encode to $this->encoding |
| 20 | + $getID3->option_tags_html = false; // Copy tags to root key 'tags_html' properly translated from various encodings to HTML entities |
| 21 | + |
| 22 | + // Analyze file and store returned data in $ThisFileInfo |
| 23 | + $id3 = $getID3->analyze( $path ); |
| 24 | + // Unset some parts of id3 that are too detailed and matroska specific: |
| 25 | + unset( $id3['matroska'] ); |
| 26 | + // remove file paths |
| 27 | + unset( $id3['filename'] ); |
| 28 | + unset( $id3['filepath'] ); |
| 29 | + unset( $id3['filenamepath']); |
| 30 | + return serialize( $id3 ); |
| 31 | + } |
| 32 | + |
| 33 | + /** |
| 34 | + * Get the "media size" |
| 35 | + * |
| 36 | + */ |
| 37 | + function getImageSize( $file, $path, $metadata = false ) { |
| 38 | + global $wgMediaVideoTypes; |
| 39 | + // Just return the size of the first video stream |
| 40 | + if ( $metadata === false ) { |
| 41 | + $metadata = $file->getMetadata(); |
| 42 | + } |
| 43 | + $metadata = $this->unpackMetadata( $metadata ); |
| 44 | + if ( isset( $metadata['error'] ) || !isset( $metadata['streams'] ) ) { |
| 45 | + return false; |
| 46 | + } |
| 47 | + foreach ( $metadata['video'] as $stream ) { |
| 48 | + return array( |
| 49 | + $stream['resolution_x'], |
| 50 | + $stream['resolution_y'] |
| 51 | + ); |
| 52 | + } |
| 53 | + return array( false, false ); |
| 54 | + } |
| 55 | + |
| 56 | + function unpackMetadata( $metadata ) { |
| 57 | + $unser = @unserialize( $metadata ); |
| 58 | + if ( isset( $unser['version'] ) && $unser['version'] == self::METADATA_VERSION ) { |
| 59 | + return $unser; |
| 60 | + } else { |
| 61 | + return false; |
| 62 | + } |
| 63 | + } |
| 64 | + |
| 65 | + function getMetadataType( $image = '' ) { |
| 66 | + return 'webm'; |
| 67 | + } |
| 68 | + |
| 69 | + function getStreamTypes( $file ) { |
| 70 | + $streamTypes = ''; |
| 71 | + $metadata = self::unpackMetadata( $file->getMetadata() ); |
| 72 | + if ( !$metadata || isset( $metadata['error'] ) ) { |
| 73 | + return false; |
| 74 | + } |
| 75 | + print_r( $metadata ); |
| 76 | + die(); |
| 77 | + foreach ( $metadata['streams'] as $stream ) { |
| 78 | + $streamTypes[$stream['type']] = true; |
| 79 | + } |
| 80 | + return array_keys( $streamTypes ); |
| 81 | + } |
| 82 | + |
| 83 | + function getLength( $file ) { |
| 84 | + $metadata = $this->unpackMetadata( $file->getMetadata() ); |
| 85 | + if ( !$metadata || isset( $metadata['error'] ) ) { |
| 86 | + return 0; |
| 87 | + } else { |
| 88 | + return $metadata['playtime_seconds']; |
| 89 | + } |
| 90 | + } |
| 91 | + |
| 92 | +} |
\ No newline at end of file |
Index: trunk/extensions/TimedMediaHandler/TimedMediaHandler_body.php |
— | — | @@ -3,7 +3,6 @@ |
4 | 4 | // TODO: Fix core printable stylesheet. Descendant selectors suck. |
5 | 5 | |
6 | 6 | class TimedMediaHandler extends MediaHandler { |
7 | | - const OGG_METADATA_VERSION = 2; |
8 | 7 | |
9 | 8 | static $magicDone = false; |
10 | 9 | |
— | — | @@ -11,6 +10,10 @@ |
12 | 11 | return true; |
13 | 12 | } |
14 | 13 | |
| 14 | + function getImageSize( $file, $path, $metadata = false ) { |
| 15 | + /* override by handler */ |
| 16 | + } |
| 17 | + |
15 | 18 | /** |
16 | 19 | * Get the list of supported wikitext embed params |
17 | 20 | */ |
— | — | @@ -85,8 +88,6 @@ |
86 | 89 | /** |
87 | 90 | * Utility functions |
88 | 91 | */ |
89 | | - |
90 | | - |
91 | 92 | public static function parseTimeString( $seekString, $length = false ) { |
92 | 93 | $parts = explode( ':', $seekString ); |
93 | 94 | $time = 0; |
— | — | @@ -105,87 +106,8 @@ |
106 | 107 | $time = $length - 1; |
107 | 108 | } |
108 | 109 | return $time; |
109 | | - } |
110 | | - /** |
111 | | - * Get the "media size" |
112 | | - * |
113 | | - */ |
114 | | - function getImageSize( $file, $path, $metadata = false ) { |
115 | | - global $wgMediaVideoTypes; |
116 | | - // Just return the size of the first video stream |
117 | | - if ( $metadata === false ) { |
118 | | - $metadata = $file->getMetadata(); |
119 | | - } |
120 | | - $metadata = $this->unpackMetadata( $metadata ); |
121 | | - if ( isset( $metadata['error'] ) || !isset( $metadata['streams'] ) ) { |
122 | | - return false; |
123 | | - } |
124 | | - foreach ( $metadata['streams'] as $stream ) { |
125 | | - if ( in_array( $stream['type'], $wgMediaVideoTypes ) ) { |
126 | | - return array( |
127 | | - $stream['header']['PICW'], |
128 | | - $stream['header']['PICH'] |
129 | | - ); |
130 | | - } |
131 | | - } |
132 | | - return array( false, false ); |
133 | | - } |
| 110 | + } |
134 | 111 | |
135 | | - function getMetadata( $image, $path ) { |
136 | | - // Get the $image type: |
137 | | - print_r( $image ); |
138 | | - die(); |
139 | | - |
140 | | - $metadata = array( 'version' => self::OGG_METADATA_VERSION ); |
141 | | - |
142 | | - if ( !class_exists( 'File_Ogg' ) ) { |
143 | | - require( 'File/Ogg.php' ); |
144 | | - } |
145 | | - try { |
146 | | - $f = new File_Ogg( $path ); |
147 | | - $streams = array(); |
148 | | - foreach ( $f->listStreams() as $streamType => $streamIDs ) { |
149 | | - foreach ( $streamIDs as $streamID ) { |
150 | | - $stream = $f->getStream( $streamID ); |
151 | | - $streams[$streamID] = array( |
152 | | - 'serial' => $stream->getSerial(), |
153 | | - 'group' => $stream->getGroup(), |
154 | | - 'type' => $stream->getType(), |
155 | | - 'vendor' => $stream->getVendor(), |
156 | | - 'length' => $stream->getLength(), |
157 | | - 'size' => $stream->getSize(), |
158 | | - 'header' => $stream->getHeader(), |
159 | | - 'comments' => $stream->getComments() |
160 | | - ); |
161 | | - } |
162 | | - } |
163 | | - $metadata['streams'] = $streams; |
164 | | - $metadata['length'] = $f->getLength(); |
165 | | - // Get the offset of the file (in cases where the file is a segment copy) |
166 | | - $metadata['offset'] = $f->getStartOffset(); |
167 | | - } catch ( PEAR_Exception $e ) { |
168 | | - // File not found, invalid stream, etc. |
169 | | - $metadata['error'] = array( |
170 | | - 'message' => $e->getMessage(), |
171 | | - 'code' => $e->getCode() |
172 | | - ); |
173 | | - } |
174 | | - return serialize( $metadata ); |
175 | | - } |
176 | | - |
177 | | - function unpackMetadata( $metadata ) { |
178 | | - $unser = @unserialize( $metadata ); |
179 | | - if ( isset( $unser['version'] ) && $unser['version'] == self::OGG_METADATA_VERSION ) { |
180 | | - return $unser; |
181 | | - } else { |
182 | | - return false; |
183 | | - } |
184 | | - } |
185 | | - |
186 | | - function getMetadataType( $image ) { |
187 | | - return 'ogg'; |
188 | | - } |
189 | | - |
190 | 112 | function isMetadataValid( $image, $metadata ) { |
191 | 113 | return $this->unpackMetadata( $metadata ) !== false; |
192 | 114 | } |
— | — | @@ -221,49 +143,25 @@ |
222 | 144 | } |
223 | 145 | |
224 | 146 | // Generate thumb: |
225 | | - $thumbStatus = TimedMediaThumbnail::gennerateThumb( $file, $dstPath, $params, $width, $height ); |
| 147 | + $thumbStatus = TimedMediaThumbnail::get( $file, $dstPath, $params, $width, $height ); |
226 | 148 | if( $thumbStatus !== true ){ |
227 | 149 | return $thumbStatus; |
228 | 150 | } |
229 | 151 | |
230 | 152 | return new TimedMediaTransformOutput( $baseConfig ); |
231 | 153 | } |
232 | | - |
| 154 | + |
233 | 155 | function canRender( $file ) { return true; } |
234 | 156 | function mustRender( $file ) { return true; } |
235 | 157 | |
236 | | - function getLength( $file ) { |
237 | | - $metadata = $this->unpackMetadata( $file->getMetadata() ); |
238 | | - if ( !$metadata || isset( $metadata['error'] ) ) { |
239 | | - return 0; |
240 | | - } else { |
241 | | - return $metadata['length']; |
242 | | - } |
243 | | - } |
| 158 | + // Get a stream offset time |
244 | 159 | function getOffset( $file ){ |
245 | | - $metadata = $this->unpackMetadata( $file->getMetadata() ); |
246 | | - if ( !$metadata || isset( $metadata['error'] ) || !isset( $metadata['offset']) ) { |
247 | | - return 0; |
248 | | - } else { |
249 | | - return $metadata['offset']; |
250 | | - } |
| 160 | + return 0; |
251 | 161 | } |
252 | 162 | |
253 | | - function getStreamTypes( $file ) { |
254 | | - $streamTypes = ''; |
255 | | - $metadata = $this->unpackMetadata( $file->getMetadata() ); |
256 | | - if ( !$metadata || isset( $metadata['error'] ) ) { |
257 | | - return false; |
258 | | - } |
259 | | - foreach ( $metadata['streams'] as $stream ) { |
260 | | - $streamTypes[$stream['type']] = true; |
261 | | - } |
262 | | - return array_keys( $streamTypes ); |
263 | | - } |
264 | | - |
265 | 163 | function getShortDesc( $file ) { |
266 | 164 | global $wgLang, $wgMediaAudioTypes, $wgMediaVideoTypes; |
267 | | - wfLoadExtensionMessages( 'TimedMediaHandler' ); |
| 165 | + |
268 | 166 | $streamTypes = $this->getStreamTypes( $file ); |
269 | 167 | if ( !$streamTypes ) { |
270 | 168 | return parent::getShortDesc( $file ); |
— | — | @@ -282,7 +180,7 @@ |
283 | 181 | |
284 | 182 | function getLongDesc( $file ) { |
285 | 183 | global $wgLang, $wgMediaVideoTypes, $wgMediaAudioTypes; |
286 | | - wfLoadExtensionMessages( 'TimedMediaHandler' ); |
| 184 | + |
287 | 185 | $streamTypes = $this->getStreamTypes( $file ); |
288 | 186 | if ( !$streamTypes ) { |
289 | 187 | $unpacked = $this->unpackMetadata( $file->getMetadata() ); |
— | — | @@ -321,7 +219,7 @@ |
322 | 220 | |
323 | 221 | function getDimensionsString( $file ) { |
324 | 222 | global $wgLang; |
325 | | - wfLoadExtensionMessages( 'TimedMediaHandler' ); |
| 223 | + |
326 | 224 | if ( $file->getWidth() ) { |
327 | 225 | return wfMsg( 'video-dims', $wgLang->formatTimePeriod( $this->getLength( $file ) ), |
328 | 226 | $wgLang->formatNum( $file->getWidth() ), |
— | — | @@ -331,15 +229,6 @@ |
332 | 230 | } |
333 | 231 | } |
334 | 232 | |
335 | | - static function getMyScriptPath() { |
336 | | - global $wgScriptPath; |
337 | | - return "$wgScriptPath/extensions/TimedMediaHandler"; |
338 | | - } |
339 | | - |
340 | | - function setHeaders( $out ) { |
341 | | - |
342 | | - } |
343 | | - |
344 | 233 | function parserTransformHook( $parser, $file ) { |
345 | 234 | if ( isset( $parser->mOutput->hasOggTransform ) ) { |
346 | 235 | return; |
— | — | @@ -354,4 +243,8 @@ |
355 | 244 | $instance->setHeaders( $outputPage ); |
356 | 245 | } |
357 | 246 | } |
| 247 | +} |
| 248 | +// Setup named Timed Media handlers |
| 249 | +class TimedMediaHandlerOgg extends TimedMediaHandler { |
| 250 | + |
358 | 251 | } |
\ No newline at end of file |
Index: trunk/extensions/TimedMediaHandler/TimedMediaHandler.php |
— | — | @@ -8,9 +8,6 @@ |
9 | 9 | // Set up the timed media handler dir: |
10 | 10 | $timedMediaDir = dirname(__FILE__); |
11 | 11 | |
12 | | -$wgMediaHandlers['application/ogg'] = 'TimedMediaHandler'; |
13 | | -$wgMediaHandlers['application/webm'] = 'TimedMediaHandler'; |
14 | | - |
15 | 12 | if ( !in_array( 'ogg', $wgFileExtensions ) ) { |
16 | 13 | $wgFileExtensions[] = 'ogg'; |
17 | 14 | } |
— | — | @@ -23,18 +20,26 @@ |
24 | 21 | if ( !in_array( 'webm', $wgFileExtensions ) ) { |
25 | 22 | $wgFileExtensions[] = 'webm'; |
26 | 23 | } |
27 | | - |
| 24 | +// Pear based OGG parsing: |
28 | 25 | ini_set( 'include_path', |
29 | 26 | "$timedMediaDir/PEAR/File_Ogg" . |
30 | 27 | PATH_SEPARATOR . |
31 | 28 | ini_get( 'include_path' ) ); |
32 | 29 | |
| 30 | +// Getid3 WebM parsing: |
| 31 | +$wgAutoloadClasses['getID3' ] = "$timedMediaDir/getid3/getid3.php"; |
33 | 32 | |
34 | 33 | // Timed Media Handler AutoLoad Classes: |
35 | 34 | $wgAutoloadClasses['TimedMediaHandler'] = "$timedMediaDir/TimedMediaHandler_body.php"; |
36 | 35 | $wgAutoloadClasses['TimedMediaHandlerHooks'] = "$timedMediaDir/TimedMediaHandler.hooks.php"; |
37 | 36 | $wgAutoloadClasses['TimedMediaTransformOutput'] = "$timedMediaDir/TimedMediaTransformOutput.php"; |
38 | | -$wgAutoloadClasses['TimedMediaIframeOutput' ] = "$timedMediaDir/TimedMediaIframeOutput.php"; |
| 37 | +$wgAutoloadClasses['TimedMediaIframeOutput'] = "$timedMediaDir/TimedMediaIframeOutput.php"; |
| 38 | +$wgAutoloadClasses['TimedMediaThumbnail'] = "$timedMediaDir/TimedMediaThumbnail.php"; |
| 39 | + |
| 40 | + |
| 41 | +$wgAutoloadClasses['OggHandler'] = "$timedMediaDir/handlers/OggHandler.php"; |
| 42 | +$wgAutoloadClasses['WebMHandler'] = "$timedMediaDir/handlers/WebMHandler.php"; |
| 43 | + |
39 | 44 | $wgAutoloadClasses['WebVideoTranscode'] = "$timedMediaDir/WebVideoTranscode/WebVideoTranscode.php"; |
40 | 45 | $wgAutoloadClasses['WebVideoTranscodeJob'] = "$timedMediaDir/WebVideoTranscode/WebVideoTranscodeJob.php"; |
41 | 46 | |
— | — | @@ -46,6 +51,17 @@ |
47 | 52 | $wgExtensionMessagesFiles['TimedMediaHandler'] = "$timedMediaDir/TimedMediaHandler.i18n.php"; |
48 | 53 | $wgExtensionMessagesFiles['TimedMediaHandlerMagic'] = "$timedMediaDir/TimedMediaHandler.i18n.magic.php"; |
49 | 54 | |
| 55 | +// Setup globals |
| 56 | + |
| 57 | +/** |
| 58 | + * Setup a metadata cache :( |
| 59 | + * |
| 60 | + * Its very costly to generate metadata! I am not sure who or why the file repos don't get |
| 61 | + * instantiated with a path, and then could lazy init things like other normal objects and |
| 62 | + * have a local cache of their metadata! |
| 63 | + */ |
| 64 | +$wgMediaHandlerMetadataCache = array(); |
| 65 | + |
50 | 66 | // Register all Timed Media Handler hooks: |
51 | 67 | TimedMediaHandlerHooks::register(); |
52 | 68 | |
— | — | @@ -62,7 +78,7 @@ |
63 | 79 | /******************* CONFIGURATION STARTS HERE **********************/ |
64 | 80 | |
65 | 81 | // Set the supported ogg codecs: |
66 | | -$wgMediaVideoTypes = array( 'Theora', 'WebM' ); |
| 82 | +$wgMediaVideoTypes = array( 'Theora', 'Vp8' ); |
67 | 83 | $wgMediaAudioTypes = array( 'Vorbis', 'Speex', 'FLAC' ); |
68 | 84 | |
69 | 85 | // Default skin for mwEmbed player ( class attribute of video tag ) |
Index: trunk/extensions/TimedMediaHandler/WebVideoTranscode/WebVideoTranscode.php |
— | — | @@ -181,10 +181,10 @@ |
182 | 182 | continue; |
183 | 183 | } |
184 | 184 | // TranscodeKey not found ( check if the file is in progress ) ( tmp transcode location ) |
185 | | - //if( is_file( self::getEncodeTargetFilePath( $file, $transcodeKey ) ){ |
| 185 | + if( is_file( self::getTargetEncodePath( $file, $transcodeKey ) ) ) { |
186 | 186 | // file in progress / in queue |
187 | 187 | // XXX Note we could check date and flag as failure somewhere |
188 | | - //} else { |
| 188 | + } else { |
189 | 189 | // no in-progress file add to job queue and touch the target |
190 | 190 | $job = new WebVideoTranscodeJob( $file->getTitle(), array( |
191 | 191 | 'transcodeMode' => 'derivative', |
— | — | @@ -192,10 +192,13 @@ |
193 | 193 | ) ); |
194 | 194 | $jobId = $job->insert(); |
195 | 195 | if( $jobId ){ |
196 | | - // If the job was inserted: |
197 | | - touch( self::getTargetEncodePath( $file, $transcodeKey ) ); |
| 196 | + // Make the thumb target directory: |
| 197 | + wfMkdirParents( dirname( self::getTargetEncodePath( $file, $transcodeKey ) ) ); |
| 198 | + // If the job was inserted touch the file ( so we don't add the job again ) |
| 199 | + // ONCE REAADY UNCOMMENT HERE: |
| 200 | + //touch( self::getTargetEncodePath( $file, $transcodeKey ) ); |
198 | 201 | } |
199 | | - //} |
| 202 | + } |
200 | 203 | } |
201 | 204 | } |
202 | 205 | return $sources; |
Index: trunk/extensions/TimedMediaHandler/TimedMediaHandler.hooks.php |
— | — | @@ -10,7 +10,11 @@ |
11 | 11 | class TimedMediaHandlerHooks { |
12 | 12 | // Register TimedMediaHandler Hooks |
13 | 13 | static function register(){ |
14 | | - global $wgParserOutputHooks, $wgHooks, $wgJobClasses, $wgJobExplitRequestTypes; |
| 14 | + global $wgParserOutputHooks, $wgHooks, $wgJobClasses, $wgJobExplitRequestTypes, $wgMediaHandlers; |
| 15 | + |
| 16 | + // Setup media Handlers: |
| 17 | + $wgMediaHandlers['application/ogg'] = 'OggHandler'; |
| 18 | + $wgMediaHandlers['video/webm'] = 'WebMHandler'; |
15 | 19 | |
16 | 20 | // Parser hook for TimedMediaHandler output |
17 | 21 | $wgParserOutputHooks['TimedMediaHandler'] = array( 'TimedMediaHandler', 'outputHook' ); |
— | — | @@ -28,7 +32,7 @@ |
29 | 33 | ); |
30 | 34 | |
31 | 35 | /** |
32 | | - * Add support for the "timedText" NameSpace |
| 36 | + * Add support for the "TimedText" NameSpace |
33 | 37 | */ |
34 | 38 | global $wgExtraNamespaces; |
35 | 39 | $timedTextNS = null; |
Index: trunk/extensions/TimedMediaHandler/TimedMediaThumbnail.php |
— | — | @@ -1,39 +1,72 @@ |
2 | 2 | <?php |
3 | 3 | class TimedMediaThumbnail { |
4 | | - function get($file, $dstPath, $params, $width, $height){ |
| 4 | + |
| 5 | + function get( $file, $dstPath, $params, $width, $height){ |
5 | 6 | global $wgFFmpegLocation, $wgOggThumbLocation; |
6 | | - |
7 | | - $length = $this->getLength( $file ); |
8 | | - $thumbtime = false; |
9 | | - if ( isset( $params['thumbtime'] ) ) { |
10 | | - $thumbtime = TimedMediaHandler::parseTimeString( $params['thumbtime'], $length ); |
11 | | - } |
12 | | - if ( $thumbtime === false ) { |
13 | | - // If start time param isset use that for the thumb: |
14 | | - if( isset( $params['start'] ) ){ |
15 | | - $thumbtime = TimedMediaHandler::parseTimeString( $params['start'], $length ); |
16 | | - }else{ |
17 | | - # Seek to midpoint by default, it tends to be more interesting than the start |
18 | | - $thumbtime = $length / 2; |
19 | | - } |
20 | | - } |
| 7 | + $thumbtime = self::getThumbTime($file, $params ); |
| 8 | + |
21 | 9 | wfMkdirParents( dirname( $dstPath ) ); |
22 | 10 | |
23 | 11 | wfDebug( "Creating video thumbnail at $dstPath\n" ); |
24 | | - |
25 | | - // First check for oggThumb |
26 | | - if( $wgOggThumbLocation && is_file( $wgOggThumbLocation ) ){ |
27 | | - $cmd = wfEscapeShellArg( $wgOggThumbLocation ) . |
28 | | - ' -t '. intval( $thumbtime ) . ' ' . |
29 | | - ' -n ' . wfEscapeShellArg( $dstPath ) . ' ' . |
30 | | - ' ' . wfEscapeShellArg( $file->getPath() ) . ' 2>&1'; |
31 | | - $returnText = wfShellExec( $cmd, $retval ); |
32 | | - //check if it was successful or if we should try ffmpeg: |
33 | | - if ( !$this->removeBadFile( $dstPath, $retval ) ) { |
34 | | - return true; |
35 | | - } |
| 12 | + |
| 13 | + $width = ( isset( $params['width'] ) |
| 14 | + && |
| 15 | + $params['width'] > 0 |
| 16 | + && |
| 17 | + $params['width'] < $file->getWidth() |
| 18 | + ) ? $params['width'] : $file->getWidth(); |
| 19 | + |
| 20 | + $height = ( isset( $params['height'] ) |
| 21 | + && |
| 22 | + $params['height'] > 0 |
| 23 | + && |
| 24 | + $params['height'] < $file->getHeight() |
| 25 | + ) ? $params['height'] : $file->getHeight(); |
| 26 | + |
| 27 | + // If ogg try OggThumb: |
| 28 | + if( self::tryOggThumb($file, $dstPath, $width, $height, $thumbtime ) ){ |
| 29 | + return true; |
36 | 30 | } |
37 | | - |
| 31 | + // Else try and return the ffmpeg thumbnail attempt: |
| 32 | + return self::tryFfmpegThumb($file, $dstPath, $width, $height, $thumbtime ); |
| 33 | + } |
| 34 | + /** |
| 35 | + * Try to render a thumbnail using oggThumb: |
| 36 | + * |
| 37 | + * @param $file {Object} File object |
| 38 | + * @param $dstPath {string} Destination path for the rendered thumbnail |
| 39 | + * @param $dstPath {array} Thumb rendering parameters ( like size and time ) |
| 40 | + */ |
| 41 | + function tryOggThumb($file, $dstPath, $width, $height, $thumbtime ){ |
| 42 | + global $wgOggThumbLocation; |
| 43 | + |
| 44 | + // Check for ogg format file and $wgOggThumbLocation |
| 45 | + if( !$file->getHandler()->getMetadataType() == 'ogg' |
| 46 | + || !$wgOggThumbLocation |
| 47 | + || !is_file( $wgOggThumbLocation ) |
| 48 | + ){ |
| 49 | + return false; |
| 50 | + } |
| 51 | + |
| 52 | + $cmd = wfEscapeShellArg( $wgOggThumbLocation ) . |
| 53 | + ' -t '. intval( $thumbtime ) . ' ' . |
| 54 | + ' -n ' . wfEscapeShellArg( $dstPath ) . ' ' . |
| 55 | + ' ' . wfEscapeShellArg( $file->getPath() ) . ' 2>&1'; |
| 56 | + $returnText = wfShellExec( $cmd, $retval ); |
| 57 | + |
| 58 | + // Check if it was successful |
| 59 | + if ( !$file->getHandler()->removeBadFile( $dstPath, $retval ) ) { |
| 60 | + return true; |
| 61 | + } |
| 62 | + return false; |
| 63 | + } |
| 64 | + |
| 65 | + function tryFfmpegThumb($file, $dstPath, $width, $height, $thumbtime ){ |
| 66 | + global $wgFFmpegLocation; |
| 67 | + if( !$wgFFmpegLocation || !is_file( $wgFFmpegLocation ) ){ |
| 68 | + return false; |
| 69 | + } |
| 70 | + |
38 | 71 | $cmd = wfEscapeShellArg( $wgFFmpegLocation ) . |
39 | 72 | ' -ss ' . intval( $thumbtime ) . ' ' . |
40 | 73 | ' -i ' . wfEscapeShellArg( $file->getPath() ) . |
— | — | @@ -44,45 +77,41 @@ |
45 | 78 | |
46 | 79 | $retval = 0; |
47 | 80 | $returnText = wfShellExec( $cmd, $retval ); |
48 | | - |
49 | | - if ( $this->removeBadFile( $dstPath, $retval ) || $retval ) { |
50 | | - #re-attempt encode command on frame time 1 and with mapping (special case for chopped oggs) |
51 | | - $cmd = wfEscapeShellArg( $wgFFmpegLocation ) . |
52 | | - ' -map 0:1 '. |
53 | | - ' -ss 1 ' . |
54 | | - ' -i ' . wfEscapeShellArg( $file->getPath() ) . |
55 | | - ' -f mjpeg -an -vframes 1 ' . |
56 | | - wfEscapeShellArg( $dstPath ) . ' 2>&1'; |
57 | | - $retval = 0; |
58 | | - $returnText = wfShellExec( $cmd, $retval ); |
| 81 | + |
| 82 | + // Check if it was successful |
| 83 | + if ( !$file->getHandler()->removeBadFile( $dstPath, $retval ) ) { |
| 84 | + return true; |
59 | 85 | } |
60 | | - |
61 | | - if ( $this->removeBadFile( $dstPath, $retval ) || $retval ) { |
62 | | - #No mapping, time zero. A last ditch attempt. |
63 | | - $cmd = wfEscapeShellArg( $wgFFmpegLocation ) . |
64 | | - ' -ss 0 ' . |
65 | | - ' -i ' . wfEscapeShellArg( $file->getPath() ) . |
66 | | - ' -f mjpeg -an -vframes 1 ' . |
67 | | - wfEscapeShellArg( $dstPath ) . ' 2>&1'; |
68 | | - |
69 | | - $retval = 0; |
70 | | - $returnText = wfShellExec( $cmd, $retval ); |
71 | | - // If still bad return error: |
72 | | - if ( $this->removeBadFile( $dstPath, $retval ) || $retval ) { |
73 | | - // Filter nonsense |
74 | | - $lines = explode( "\n", str_replace( "\r\n", "\n", $returnText ) ); |
75 | | - if ( substr( $lines[0], 0, 6 ) == 'FFmpeg' ) { |
76 | | - for ( $i = 1; $i < count( $lines ); $i++ ) { |
77 | | - if ( substr( $lines[$i], 0, 2 ) != ' ' ) { |
78 | | - break; |
79 | | - } |
80 | | - } |
81 | | - $lines = array_slice( $lines, $i ); |
| 86 | + // Filter nonsense |
| 87 | + $lines = explode( "\n", str_replace( "\r\n", "\n", $returnText ) ); |
| 88 | + if ( substr( $lines[0], 0, 6 ) == 'FFmpeg' ) { |
| 89 | + for ( $i = 1; $i < count( $lines ); $i++ ) { |
| 90 | + if ( substr( $lines[$i], 0, 2 ) != ' ' ) { |
| 91 | + break; |
82 | 92 | } |
83 | | - // Return error box |
84 | | - return new MediaTransformError( 'thumbnail_error', $width, $height, implode( "\n", $lines ) ); |
85 | 93 | } |
| 94 | + $lines = array_slice( $lines, $i ); |
86 | 95 | } |
87 | | - //if we did not return an error return true to continue media thum display |
88 | | - return true; |
89 | | - } |
\ No newline at end of file |
| 96 | + // Return error box |
| 97 | + return new MediaTransformError( 'thumbnail_error', $width, $height, implode( "\n", $lines ) ); |
| 98 | + } |
| 99 | + |
| 100 | + function getThumbTime( $file, $params ){ |
| 101 | + |
| 102 | + $length = $file->getLength(); |
| 103 | + $thumbtime = false; |
| 104 | + if ( isset( $params['thumbtime'] ) ) { |
| 105 | + $thumbtime = TimedMediaHandler::parseTimeString( $params['thumbtime'], $length ); |
| 106 | + } |
| 107 | + if ( $thumbtime === false ) { |
| 108 | + // If start time param isset use that for the thumb: |
| 109 | + if( isset( $params['start'] ) ){ |
| 110 | + $thumbtime = TimedMediaHandler::parseTimeString( $params['start'], $length ); |
| 111 | + }else{ |
| 112 | + # Seek to midpoint by default, it tends to be more interesting than the start |
| 113 | + $thumbtime = $length / 2; |
| 114 | + } |
| 115 | + } |
| 116 | + return $thumbtime; |
| 117 | + } |
| 118 | +} |
\ No newline at end of file |
Index: trunk/extensions/TimedMediaHandler/README |
— | — | @@ -12,9 +12,10 @@ |
13 | 13 | |
14 | 14 | To install this extension, add the following to the end of your LocalSettings.php: |
15 | 15 | |
16 | | - // TimedMediaHandler is dependent on mwEmbedSupport |
17 | | - require( "$IP/extensions/ |
18 | | - |
| 16 | + // You need mwEmbedSupport ( if not already added ) |
| 17 | + require( "$IP/extensions/MwEmbedSupport/MwEmbedSupport.php" ); |
| 18 | + |
| 19 | + // TimedMediaHandler |
19 | 20 | require( "$IP/extensions/TimedMediaHandler/TimedMediaHandler.php" ); |
20 | 21 | |
21 | 22 | oggThumb |