diff --git a/dub.json b/dub.json index 7abc030..8515b06 100644 --- a/dub.json +++ b/dub.json @@ -3,7 +3,7 @@ "targetType": "library", "importPaths": ["."], "sourceFiles": ["package.d"], - "description": "Subpackage collection for web, database, terminal ui, gui, scripting, and more with a commitment to long-term compatibility and stability.", + "description": "Subpackage collection for web, database, terminal ui, gui, scripting, and more with a commitment to long-term compatibility and stability. Use individual subpackages instead of the overall package to avoid bringing in things you don't need/want!", "authors": ["Adam D. Ruppe"], "license":"BSL-1.0", "dependencies": { @@ -27,6 +27,7 @@ ":terminal": "*", ":ttf": "*", ":color_base": "*", + "svg":"*", ":database_base": "*" }, "subPackages": [ diff --git a/joystick.d b/joystick.d index 15865f6..9fea479 100644 --- a/joystick.d +++ b/joystick.d @@ -91,6 +91,11 @@ XInput is only supported on newer operating systems (Vista I think), so I'm going to dynamically load it all and fallback on the old one if it fails. + + + + Other fancy joysticks work low level on linux at least but the high level api reduces them to boredom but like + hey the events are still there and it still basically works, you'd just have to give a custom mapping. */ module arsd.joystick; @@ -883,7 +888,7 @@ version(linux) { printf("\n"); while(true) { - int r = read(fd, &event, event.sizeof); + auto r = read(fd, &event, event.sizeof); assert(r == event.sizeof); // writef("\r%12s", event); @@ -891,7 +896,7 @@ version(linux) { axes[event.number] = event.value >> 12; } if(event.type & JS_EVENT_BUTTON) { - buttons[event.number] = event.value; + buttons[event.number] = cast(ubyte) event.value; } writef("\r%6s %1s", axes[0..8], buttons[0 .. 16]); stdout.flush(); diff --git a/jsvar.d b/jsvar.d index bd9b531..78697f6 100644 --- a/jsvar.d +++ b/jsvar.d @@ -1,4 +1,14 @@ /* + FIXME: + overloads can be done as an object representing the overload set + tat opCall does the dispatch. Then other overloads can actually + be added more sanely. + + FIXME: + instantiate template members when reflection with certain + arguments if marked right... + + FIXME: pointer to member functions can give a way to wrap things diff --git a/midi.d b/midi.d index e3becf8..15c9162 100644 --- a/midi.d +++ b/midi.d @@ -2,73 +2,689 @@ This file is a port of some old C code I had for reading and writing .mid files. Not much docs, but viewing the source may be helpful. I'll eventually refactor it into something more D-like + + History: + Written in C in August 2008 + + Minimally ported to D in September 2017 + + Updated May 2020 with significant changes. */ module arsd.midi; +import core.time; + +version(NewMidiDemo) +void main(string[] args) { + auto f = new MidiFile(); + + import std.file; + + //f.loadFromBytes(cast(ubyte[]) read("test.mid")); + f.loadFromBytes(cast(ubyte[]) read(args[1])); + + import arsd.simpleaudio; + import core.thread; + + auto o = MidiOutput(0); + setSigIntHandler(); + scope(exit) { + o.silenceAllNotes(); + o.reset(); + restoreSigIntHandler(); + } + + import std.stdio : writeln; + foreach(item; f.playbackStream) { + if(interrupted) return; + + Thread.sleep(item.wait); + if(!item.event.isMeta) + o.writeMidiMessage(item.event.status, item.event.data1, item.event.data2); + else + writeln(item); + } + + return; + + auto t = new MidiTrack(); + auto t2 = new MidiTrack(); + + f.tracks ~= t; + f.tracks ~= t2; + + t.events ~= MidiEvent(0, 0x90, C, 127); + t.events ~= MidiEvent(256, 0x90, C, 0); + t.events ~= MidiEvent(256, 0x90, D, 127); + t.events ~= MidiEvent(256, 0x90, D, 0); + t.events ~= MidiEvent(256, 0x90, E, 127); + t.events ~= MidiEvent(256, 0x90, E, 0); + t.events ~= MidiEvent(256, 0x90, F, 127); + t.events ~= MidiEvent(0, 0xff, 0x05, 0 /* unused */, ['h', 'a', 'm']); + t.events ~= MidiEvent(256, 0x90, F, 0); + + t2.events ~= MidiEvent(0, (MIDI_EVENT_PROGRAM_CHANGE << 4) | 0x01, 68); + t2.events ~= MidiEvent(128, 0x91, E, 127); + t2.events ~= MidiEvent(0, 0xff, 0x05, 0 /* unused */, ['a', 'd', 'r']); + t2.events ~= MidiEvent(1024, 0x91, E, 0); + + write("test.mid", f.toBytes()); +} + +@safe: + +class MidiFile { + /// + ubyte[] toBytes() { + MidiWriteBuffer buf; + + buf.write("MThd"); + buf.write4(6); + + buf.write2(format); + buf.write2(cast(ushort) tracks.length); + buf.write2(timing); + + foreach(track; tracks) { + auto data = track.toBytes(); + buf.write("MTrk"); + buf.write4(cast(int) data.length); + buf.write(data); + } + + return buf.bytes; + } + + /// + void loadFromBytes(ubyte[] bytes) { + // FIXME: actually read the riff header to skip properly + if(bytes.length && bytes[0] == 'R') + bytes = bytes[0x14 .. $]; + + MidiReadBuffer buf = MidiReadBuffer(bytes); + if(buf.readChars(4) != "MThd") + throw new Exception("not midi"); + if(buf.read4() != 6) + throw new Exception("idk what this even is"); + this.format = buf.read2(); + this.tracks = new MidiTrack[](buf.read2()); + this.timing = buf.read2(); + + foreach(ref track; tracks) { + track = new MidiTrack(); + track.loadFromBuffer(buf); + } + } + + // when I read, I plan to cut the end of track marker off. + + // 0 == combined into one track + // 1 == multiple tracks + // 2 == multiple one-track patterns + ushort format = 1; + + // FIXME + ushort timing = 0x80; // 128 ticks per quarter note + + MidiTrack[] tracks; + + /++ + Returns a forward range for playback. Each item is a command, which + is like the midi event but with some more annotations and control methods. + + Modifying this MidiFile object or any of its children during playback + may cause trouble. + + Note that you do not need to handle any meta events, it keeps the + tempo internally, but you can look at it if you like. + +/ + PlayStream playbackStream() { + return PlayStream(this); + } +} + +struct PlayStream { + static struct Event { + /// This is how long you wait until triggering this event. + /// Note it may be zero. + Duration wait; + + /// And this is the event. + MidiEvent event; + + string toString() { + return event.toString(); + } + + /// informational + MidiFile file; + /// ditto + MidiTrack track; + } + + PlayStream save() { + auto copy = this; + copy.trackPositions = this.trackPositions.dup; + return copy; + } + + MidiFile file; + this(MidiFile file) { + this.file = file; + this.trackPositions.length = file.tracks.length; + foreach(idx, ref tp; this.trackPositions) { + tp.remaining = file.tracks[idx].events[]; + tp.track = file.tracks[idx]; + } + + this.currentTrack = -1; + this.tempo = 500000; + popFront(); + } + + //@nogc: + + void popFront() { + done = true; + for(auto c = currentTrack + 1; c < trackPositions.length; c++) { + auto tp = trackPositions[c]; + + if(tp.remaining.length && tp.remaining[0].deltaTime == tp.clock) { + auto f = tp.remaining[0]; + trackPositions[c].remaining = tp.remaining[1 .. $]; + trackPositions[c].clock = 0; + if(tp.remaining.length == 0 || tp.remaining[0].deltaTime > 0) { + currentTrack += 1; + } + + pending = Event(0.seconds, f, file, tp.track); + processPending(); + done = false; + return; + } + } + + // if nothing happened there, time to advance the clock + int minWait = int.max; + int minWaitTrack = -1; + foreach(idx, track; trackPositions) { + if(track.remaining.length) { + auto dt = track.remaining[0].deltaTime - track.clock; + if(dt < minWait) { + minWait = dt; + minWaitTrack = cast(int) idx; + } + } + } + + if(minWaitTrack == -1) { + done = true; + return; + } + + foreach(ref tp; trackPositions) { + tp.clock += minWait; + } + + done = false; + + // file.timing, if high bit clear, is ticks per quarter note + // if high bit set... idk it is different. + // + // then the temp is microseconds per quarter note. + auto time = (minWait * tempo / file.timing).usecs; + + pending = Event(time, trackPositions[minWaitTrack].remaining[0], file, trackPositions[minWaitTrack].track); + processPending(); + trackPositions[minWaitTrack].remaining = trackPositions[minWaitTrack].remaining[1 .. $]; + trackPositions[minWaitTrack].clock = 0; + currentTrack = minWaitTrack; + + return; + } + + private struct TrackPosition { + MidiEvent[] remaining; + int clock; + MidiTrack track; + } + private TrackPosition[] trackPositions; + private int currentTrack; + + private void processPending() { + if(pending.event.status == 0xff && pending.event.data1 == MetaEvent.Tempo) { + this.tempo = 0; + foreach(i; pending.event.meta) { + this.tempo <<= 8; + this.tempo |= i; + } + } + } + + @property + Event front() { + return pending; + } + + private uint tempo; + private Event pending; + private bool done; + + @property + bool empty() { + return done; + } +} + +class MidiTrack { + ubyte[] toBytes() { + MidiWriteBuffer buf; + foreach(event; events) + event.writeToBuffer(buf); + + MidiEvent end; + end.status = 0xff; + end.data1 = 0x2f; + end.meta = null; + + end.writeToBuffer(buf); + + return buf.bytes; + } + + void loadFromBuffer(ref MidiReadBuffer buf) { + if(buf.readChars(4) != "MTrk") + throw new Exception("wtf no track header"); + + auto trackLength = buf.read4(); + auto begin = buf.bytes.length; + + ubyte runningStatus; + + while(buf.bytes.length) { + MidiEvent newEvent = MidiEvent.fromBuffer(buf, runningStatus); + if(newEvent.status == 0xff && newEvent.data1 == MetaEvent.EndOfTrack) { + break; + } + events ~= newEvent; + } + //assert(begin - trackLength == buf.bytes.length); + } + + MidiEvent[] events; + + override string toString() const { + string s; + foreach(event; events) + s ~= event.toString ~ "\n"; + return s; + } +} + +enum MetaEvent { + SequenceNumber = 0, + // these take a text param + Text = 1, + Copyright = 2, + Name = 3, + Instrument = 4, + Lyric = 5, + Marker = 6, + CuePoint = 7, + PatchName = 8, + DeviceName = 9, + + // no param + EndOfTrack = 0x2f, + + // different ones + Tempo = 0x51, // 3 bytes form big-endian micro-seconds per quarter note. 120 BPM default. + SMPTEOffset = 0x54, // 5 bytes. I don't get this one.... + TimeSignature = 0x58, // 4 bytes: numerator, denominator, clocks per click, 32nd notes per quarter note. (8 == quarter note gets the beat) + KeySignature = 0x59, // 2 bytes: first byte is signed offset from C in semitones, second byte is 0 for major, 1 for minor + + // arbitrary length custom param + Proprietary = 0x7f, + +} + +struct MidiEvent { + int deltaTime; + + ubyte status; + + ubyte data1; // if meta, this is the identifier + + //union { + //struct { + ubyte data2; + //} + + const(ubyte)[] meta; // iff status == 0xff + //} + + invariant () { + assert(status & 0x80); + assert(!(data1 & 0x80)); + assert(!(data2 & 0x80)); + assert(status == 0xff || meta is null); + } + + /// Convenience factories for various meta-events + static MidiEvent Text(string t) { return MidiEvent(0, 0xff, MetaEvent.Text, 0, cast(const(ubyte)[]) t); } + /// ditto + static MidiEvent Copyright(string t) { return MidiEvent(0, 0xff, MetaEvent.Copyright, 0, cast(const(ubyte)[]) t); } + /// ditto + static MidiEvent Name(string t) { return MidiEvent(0, 0xff, MetaEvent.Name, 0, cast(const(ubyte)[]) t); } + /// ditto + static MidiEvent Lyric(string t) { return MidiEvent(0, 0xff, MetaEvent.Lyric, 0, cast(const(ubyte)[]) t); } + /// ditto + static MidiEvent Marker(string t) { return MidiEvent(0, 0xff, MetaEvent.Marker, 0, cast(const(ubyte)[]) t); } + /// ditto + static MidiEvent CuePoint(string t) { return MidiEvent(0, 0xff, MetaEvent.CuePoint, 0, cast(const(ubyte)[]) t); } + + /// + bool isMeta() const { + return status == 0xff; + } + + /// + ubyte event() const { + return status >> 4; + } + + /// + ubyte channel() const { + return status & 0x0f; + } + + /// + string toString() const { + + static string tos(int a) { + char[16] buffer; + auto bufferPos = buffer.length; + do { + buffer[--bufferPos] = a % 10 + '0'; + a /= 10; + } while(a); + + return buffer[bufferPos .. $].idup; + } + + static string toh(ubyte b) { + char[2] buffer; + buffer[0] = (b >> 4) & 0x0f; + if(buffer[0] < 10) + buffer[0] += '0'; + else + buffer[0] += 'A' - 10; + buffer[1] = b & 0x0f; + if(buffer[1] < 10) + buffer[1] += '0'; + else + buffer[1] += 'A' - 10; + + return buffer.idup; + } + + string s; + s ~= tos(deltaTime); + s ~= ": "; + s ~= toh(status); + s ~= " "; + s ~= toh(data1); + s ~= " "; + if(isMeta) { + switch(data1) { + case MetaEvent.Text: + case MetaEvent.Copyright: + case MetaEvent.Name: + case MetaEvent.Instrument: + case MetaEvent.Lyric: + case MetaEvent.Marker: + case MetaEvent.CuePoint: + case MetaEvent.PatchName: + case MetaEvent.DeviceName: + s ~= cast(const(char)[]) meta; + break; + case MetaEvent.TimeSignature: + ubyte numerator = meta[0]; + ubyte denominator = meta[1]; + ubyte clocksPerClick = meta[2]; + ubyte notesPerQuarter = meta[3]; // 32nd notes / Q so 8 = quarter note gets the beat + + s ~= tos(numerator); + s ~= "/"; + s ~= tos(denominator); + s ~= " "; + s ~= tos(clocksPerClick); + s ~= " "; + s ~= tos(notesPerQuarter); + break; + case MetaEvent.KeySignature: + byte offset = meta[0]; + ubyte minor = meta[1]; + + if(offset < 0) { + s ~= "-"; + s ~= tos(-cast(int) offset); + } else { + s ~= tos(offset); + } + s ~= minor ? " minor" : " major"; + break; + // case MetaEvent.Tempo: + // could process this but idk if it needs to be shown + // break; + case MetaEvent.Proprietary: + foreach(m; meta) { + s ~= toh(m); + s ~= " "; + } + break; + default: + s ~= cast(const(char)[]) meta; + } + } else { + s ~= toh(data2); + } + + return s; + } + + static MidiEvent fromBuffer(ref MidiReadBuffer buf, ref ubyte runningStatus) { + MidiEvent event; + + start_over: + + event.deltaTime = buf.readv(); + + auto nb = buf.read1(); + + if(nb == 0xff) { + // meta... + event.status = 0xff; + event.data1 = buf.read1(); // the type + int len = buf.readv(); + auto meta = new ubyte[](len); + foreach(idx; 0 .. len) + meta[idx] = buf.read1(); + event.meta = meta; + } else if(nb >= 0xf0) { + // FIXME I'm just skipping this entirely but there might be value in here + nb = buf.read1(); + while(nb < 0xf0) + nb = buf.read1(); + goto start_over; + } else if(nb & 0b1000_0000) { + event.status = nb; + runningStatus = nb; + event.data1 = buf.read1(); + + if(event.event != MIDI_EVENT_CHANNEL_AFTERTOUCH && + event.event != MIDI_EVENT_PROGRAM_CHANGE) + { + event.data2 = buf.read1(); + } + } else { + event.status = runningStatus; + event.data1 = nb; + + if(event.event != MIDI_EVENT_CHANNEL_AFTERTOUCH && + event.event != MIDI_EVENT_PROGRAM_CHANGE) + { + event.data2 = buf.read1(); + } + } + + return event; + } + + void writeToBuffer(ref MidiWriteBuffer buf) const { + buf.writev(deltaTime); + buf.write1(status); + // FIXME: what about other sysex stuff? + if(meta) { + buf.write1(data1); + buf.writev(cast(int) meta.length); + buf.write(meta); + } else { + buf.write1(data1); + + if(event != MIDI_EVENT_CHANNEL_AFTERTOUCH && + event != MIDI_EVENT_PROGRAM_CHANGE) + { + buf.write1(data2); + } + } + } +} + +struct MidiReadBuffer { + ubyte[] bytes; + + char[] readChars(int len) { + auto c = bytes[0 .. len]; + bytes = bytes[len .. $]; + return cast(char[]) c; + } + ubyte[] readBytes(int len) { + auto c = bytes[0 .. len]; + bytes = bytes[len .. $]; + return c; + } + int read4() { + int i; + foreach(a; 0 .. 4) { + i <<= 8; + i |= bytes[0]; + bytes = bytes[1 .. $]; + } + return i; + } + ushort read2() { + ushort i; + foreach(a; 0 .. 2) { + i <<= 8; + i |= bytes[0]; + bytes = bytes[1 .. $]; + } + return i; + } + ubyte read1() { + auto b = bytes[0]; + bytes = bytes[1 .. $]; + return b; + } + int readv() { + int value = read1(); + ubyte c; + if(value & 0x80) { + value &= 0x7f; + do + value = (value << 7) | ((c = read1) & 0x7f); + while(c & 0x80); + } + return value; + } +} + +struct MidiWriteBuffer { + ubyte[] bytes; + + void write(const char[] a) { + bytes ~= a; + } + + void write(const ubyte[] a) { + bytes ~= a; + } + + void write4(int v) { + // big endian + bytes ~= (v >> 24) & 0xff; + bytes ~= (v >> 16) & 0xff; + bytes ~= (v >> 8) & 0xff; + bytes ~= v & 0xff; + } + + void write2(ushort v) { + // big endian + bytes ~= v >> 8; + bytes ~= v & 0xff; + } + + void write1(ubyte v) { + bytes ~= v; + } + + void writev(int v) { + // variable + uint buffer = v & 0x7f; + while((v >>= 7)) { + buffer <<= 8; + buffer |= ((v & 0x7f) | 0x80); + } + + while(true) { + bytes ~= buffer & 0xff; + if(buffer & 0x80) + buffer >>= 8; + else + break; + } + } +} import core.stdc.stdio; import core.stdc.stdlib; -/* NOTE: MIDI files are BIG ENDIAN! */ - -struct MidiChunk { - int timeStamp; // this is encoded my way. real file is msb == 1, more ubytes - ubyte event; // Event << 8 | channel is how it is actually stored - - // for channel events - ubyte channel; // see above - it is stored with event! - // channel == track btw - ubyte param1; // pitch (usually) - ubyte param2; // volume - not necessarily present - - ubyte status; // event << 4 | channel - - // for meta events (event = f, channel = f - ubyte type; - int length; // stored as variable length - ubyte* data; // only allocated if event == 255 - - MidiChunk* next; // next in the track - - // This stuff is just for playing help and such - // It is only set if you call recalculateMidiAbsolutes, and probably - // not maintained if you do edits - int track; - uint absoluteTime; - uint absoluteTimeInMilliSeconds; // for convenience - int absoluteWait; - MidiChunk* nextAbsolute; +int freq(int note){ + import std.math; + float r = note - 69; + r /= 12; + r = pow(2, r); + r*= 440; + return cast(int) r; } -/* - Meta event - timeStamp = 0 - event = 255 - channel = event - param1 param2 = not in gile - length = variable - data[length] -*/ - -struct MidiTrack { - // MTrk header - int lengthInBytes; - MidiChunk* chunks; // linked list - // the linked list should NOT hold the track ending chunk - // just hold a null instead -} - -struct Midi { - // headers go here - short type; - short numTracks; - short speed; - MidiTrack* tracks; /* Array of numTracks size */ - - // only set if you call recalculateMidiAbsolutes - MidiChunk* firstAbsolute; -} +enum A = 69; // 440 hz per midi spec +enum As = 70; +enum B = 71; +enum C = 72; // middle C + 1 octave +enum Cs = 73; +enum D = 74; +enum Ds = 75; +enum E = 76; +enum F = 77; +enum Fs = 78; +enum G = 79; +enum Gs = 80; +immutable string[] noteNames = [ // just do note % 12 to index this + "C", "C#", "D", "D#", "E", "F", "F#", "G", "G#", "A", "A#", "B" +]; enum MIDI_EVENT_NOTE_OFF = 0x08; enum MIDI_EVENT_NOTE_ON = 0x09; @@ -79,9 +695,34 @@ enum MIDI_EVENT_CHANNEL_AFTERTOUCH = 0x0d;// only one param enum MIDI_EVENT_PITCH_BEND = 0x0e; + /+ + 35 Acoustic Bass Drum 59 Ride Cymbal 2 + 36 Bass Drum 1 60 Hi Bongo + 37 Side Stick 61 Low Bongo + 38 Acoustic Snare 62 Mute Hi Conga + 39 Hand Clap 63 Open Hi Conga + 40 Electric Snare 64 Low Conga + 41 Low Floor Tom 65 High Timbale + 42 Closed Hi-Hat 66 Low Timbale + 43 High Floor Tom 67 High Agogo + 44 Pedal Hi-Hat 68 Low Agogo + 45 Low Tom 69 Cabasa + 46 Open Hi-Hat 70 Maracas + 47 Low-Mid Tom 71 Short Whistle + 48 Hi-Mid Tom 72 Long Whistle + 49 Crash Cymbal 1 73 Short Guiro + 50 High Tom 74 Long Guiro + 51 Ride Cymbal 1 75 Claves + 52 Chinese Cymbal 76 Hi Wood Block + 53 Ride Bell 77 Low Wood Block + 54 Tambourine 78 Mute Cuica + 55 Splash Cymbal 79 Open Cuica + 56 Cowbell 80 Mute Triangle + 57 Crash Cymbal 2 81 Open Triangle + 58 Vibraslap + +/ -/* -static char[][] instrumentNames = { +static immutable string[] instrumentNames = [ "", // 0 is nothing // Piano: "Acoustic Grand Piano", @@ -242,785 +883,11 @@ static char[][] instrumentNames = { "Helicopter", "Applause", "Gunshot" -}; -*/ - - -int addMidiTrack(Midi* mid){ - int trackNum; - MidiTrack* tracks; - tracks = cast(MidiTrack*) realloc(mid.tracks, MidiTrack.sizeof * (mid.numTracks + 1)); - if(tracks is null) - return -1; - - mid.tracks = tracks; - trackNum = mid.numTracks; - mid.numTracks++; - - mid.tracks[trackNum].lengthInBytes = 0; - mid.tracks[trackNum].chunks = null; - - return trackNum; -} - -int addMidiEvent(Midi* mid, int track, int deltatime, int event, int channel, int value1, int value2){ - int length = 2; - MidiChunk* c; - MidiChunk* current, previous; - if(track >= mid.numTracks) - return -1; - - c = cast(MidiChunk*) malloc(MidiChunk.sizeof); - if(c is null) - return -1; - - c.timeStamp = deltatime; - c.event = cast(ubyte) event; - c.channel = cast(ubyte) channel; - c.param1 = cast(ubyte) value1; - c.param2 = cast(ubyte) value2; - - c.status = cast(ubyte) ((event << 4) | channel); - - c.type = 0; - c.length = 0; - c.data = null; - - c.next = null; - - - previous = null; - current = mid.tracks[track].chunks; - while(current != null){ - previous = current; - current = current . next; - } - - if(previous){ - previous.next = c; - } else { - mid.tracks[track].chunks = c; - } - - length += getvldLength(deltatime); - if(event != MIDI_EVENT_CHANNEL_AFTERTOUCH && - event != MIDI_EVENT_PROGRAM_CHANGE) - length++; // param2 - mid.tracks[track].lengthInBytes += length; - - return 0; -} - -int addMidiMetaEvent(Midi* mid, int track, int dt, int type, int length, ubyte* data){ - int len = 2; - int a; - MidiChunk* c; - MidiChunk* current, previous; - - if(track >= mid.numTracks) - return -1; - - c = cast(MidiChunk*) malloc(MidiChunk.sizeof); - if(c == null) - return -1; - - c.timeStamp = dt; - c.event = 0xff; - c.channel = 0; - c.param1 = 0; - c.param2 = 0; - - c.type = cast(ubyte) type; - c.length = length; - // copy data in - c.data = cast(typeof(c.data)) malloc(length); - if(c.data == null){ - free(c); - return -1; - } - for(a = 0; a < length; a++) - c.data[a] = data[a]; - - - c.next = null; - - - previous = null; - current = mid.tracks[track].chunks; - while(current != null){ - previous = current; - current = current . next; - } - - if(previous){ - previous.next = c; - } else { - mid.tracks[track].chunks = c; - } - - len += getvldLength(dt); - len += length; - - mid.tracks[track].lengthInBytes += len; - - return 0; -} - -int createMidi(Midi** midWhere){ - Midi* mid; - - mid = cast(Midi*) malloc(Midi.sizeof); - if(mid == null) - return 1; - - mid.type = 1; - mid.numTracks = 0; - mid.speed = 0x80; // 128 ticks per quarter note - potential FIXME - mid.tracks = null; - - *midWhere = mid; - return 0; -} - -void freeChunkList(MidiChunk* c){ - if(c == null) - return; - freeChunkList(c.next); - if(c.event == 255) - free(c.data); - free(c); -} - -void freeMidi(Midi** mid){ - int a; - Midi* m = *mid; - - for(a = 0; a < m.numTracks; a++) - freeChunkList(m.tracks[a].chunks); - free(m.tracks); - free(m); - *mid = null; -} - -// FIXME: these fail on big endian machines -void write4(int v, FILE* fp){ - fputc(*(cast(ubyte*)&v + 3), fp); - fputc(*(cast(ubyte*)&v + 2), fp); - fputc(*(cast(ubyte*)&v + 1), fp); - fputc(*(cast(ubyte*)&v + 0), fp); -} - -void write2(short v, FILE* fp){ - fputc(*(cast(ubyte*)&v + 1), fp); - fputc(*(cast(ubyte*)&v + 0), fp); -} - -void writevld(uint v, FILE* fp){ - uint omg = v; - ubyte a; - ubyte[4] ubytes; - int c = 0; - more: - a = cast(ubyte) (omg&(~(1 << 7))); - omg >>= 7; - if(omg){ - ubytes[c++] = a; - goto more; - } - - ubytes[c] = a; - - for(; c >= 0; c--) - fputc(ubytes[c] | (c ? (1<<7):0), fp); -} - - -int read4(FILE* fp){ - int v; - - *(cast(ubyte*)&v + 3) = cast(ubyte) fgetc(fp); - *(cast(ubyte*)&v + 2) = cast(ubyte) fgetc(fp); - *(cast(ubyte*)&v + 1) = cast(ubyte) fgetc(fp); - *(cast(ubyte*)&v + 0) = cast(ubyte) fgetc(fp); - - return v; -} - -short read2(FILE* fp){ - short v; - *(cast(ubyte*)&v + 1) = cast(ubyte) fgetc(fp); - *(cast(ubyte*)&v + 0) = cast(ubyte) fgetc(fp); - - return v; -} - -uint readvld(FILE* fp){ - uint omg = 0; - ubyte a; - more: - a = cast(ubyte) fgetc(fp); - if(a & (1<<7)){ - a &= ~(1<<7); - omg <<= 7; - omg |= a; - goto more; - } - - omg <<= 7; - omg |= a; - - return omg; -} - - - -int getvldLength(uint v){ - int count = 0; - uint omg = v; - ubyte a; - more: - a = omg&((1 << 7)-1); // - omg >>= 7; - if(omg){ - a &= 1<<7; - count++; - goto more; - } - count++; - return count; -} - -// END: big endian fixme - -int loadMidi(Midi** midWhere, const char* filename){ - int error = 0; - FILE* fp; - Midi* mid = null; - int runningStatus; - int t, a; - int numtrk; - - int timestamp; - int event; - int channel; - int param1; - int type; - int length; - int param2; - ubyte* data; - - int done; - - fp = fopen(filename, "rb"); - if(fp == null){ - fprintf(stderr, "Cannot load file %s.\n", filename); - error = 1; - goto cleanup1; - } - - - if(fgetc(fp) != 'M') goto badfile; - if(fgetc(fp) != 'T') goto badfile; - if(fgetc(fp) != 'h') goto badfile; - if(fgetc(fp) != 'd') goto badfile; - if(read4(fp) != 6) goto badfile; - - if(createMidi(&mid) != 0){ - fprintf(stderr, "Could not allocate struct\n"); - error = 3; - goto cleanup3; - } - - mid.type = read2(fp); - numtrk = read2(fp); - mid.speed = read2(fp); - - for(t = 0; t < numtrk; t++){ - if(fgetc(fp) != 'M') goto badfile; - if(fgetc(fp) != 'T') goto badfile; - if(fgetc(fp) != 'r') goto badfile; - if(fgetc(fp) != 'k') goto badfile; - - if(addMidiTrack(mid) < 0){ - fprintf(stderr, "add midi track failed \n"); - error = 3; - goto cleanup3; - } - -// mid.tracks[t].lengthInBytes = read4(fp) - 4; - read4(fp); // ignoring it for now FIXME? - - done = 0; - do{ - timestamp = readvld(fp); - event = fgetc(fp); - if(event == 0xff){ - type = fgetc(fp); - length = readvld(fp); - - // potential optimization for malloc - if(length){ - data = cast(typeof(data)) malloc(length); - for(a = 0; a < length; a++) - data[a] = cast(ubyte) fgetc(fp); - } else - data = null; - - - if(type == 0x2f){ - done = 1; - } else { - // add the event to the list here - // FIXME: error check - addMidiMetaEvent(mid, t, timestamp, - type, length, data); - } - if(data) - free(data); - } else { - if(event < 0x80){ - param1 = event; - event = runningStatus; - } else { - runningStatus = event; - param1 = fgetc(fp); - } - - channel = event&0x0f; - event = event >> 4; - if(event != MIDI_EVENT_PROGRAM_CHANGE - && event != MIDI_EVENT_CHANNEL_AFTERTOUCH) - param2 = fgetc(fp); - - // add the event - // FIXME: error check - addMidiEvent(mid, t, timestamp, event, channel, param1, param2); - } - } while(!done); - } - - goto success; - badfile: - fprintf(stderr, "The file is not in the right format. %c\n", fgetc(fp)); - error = 2; - cleanup3: - if(mid != null) - freeMidi(&mid); - success: - fclose(fp); - *midWhere = mid; - cleanup1: - return error; -} - - - - - - -int saveMidi(Midi* mid, char* filename){ - int error = 0; - FILE* fp; - int t, a; - int runningStatus = -1; - int status; - - fp = fopen(filename, "wb"); - if(fp == null){ - fprintf(stderr, "Unable to open midi file (%s) for writing.\n", filename); - error = 1; - goto cleanup1; - } - - fputc('M', fp); - fputc('T', fp); - fputc('h', fp); - fputc('d', fp); - - write4(6, fp); - write2(mid.type, fp); - write2(mid.numTracks, fp); - write2(mid.speed, fp); - - for(t = 0; t < mid.numTracks; t++){ - fputc('M', fp); - fputc('T', fp); - fputc('r', fp); - fputc('k', fp); - - runningStatus = -1; - - write4(mid.tracks[t].lengthInBytes + 4, fp); - MidiChunk* current; - current = mid.tracks[t].chunks; - while(current != null){ - writevld(current.timeStamp, fp); - if(current.event == 0xff){ - fputc(current.event, fp); - fputc(current.type, fp); - writevld(current.length, fp); - for(a = 0; a < current.length; a++) - fputc(current.data[a], fp); - } else { - // FIXME: add support for writing running status - status = current.event << 4 | current.channel; - - // if(status != runningStatus){ - runningStatus = status; - fputc(status, fp); - // } - - fputc(current.param1, fp); - if(current.event != MIDI_EVENT_PROGRAM_CHANGE - &¤t.event != MIDI_EVENT_CHANNEL_AFTERTOUCH) - fputc(current.param2, fp); - } - current = current.next; - } - /* the end of track chunk */ - fputc(0, fp); - fputc(0xff, fp); - fputc(0x2f, fp); - fputc(0x00, fp); - } -/* cleanup2:*/ - fclose(fp); - cleanup1: - return error; -} - - -int removeMidiTrack(Midi* m, int track){ - int a; - if(track >= m.numTracks) - return -1; - - for(a = track; a < m.numTracks-1; a++){ - m.tracks[a] = m.tracks[a+1]; - } - - m.numTracks--; - - return 0; -} - -void printMidiEvent(MidiChunk* c){ - int e = c.event; - printf("%d %s %d %d\n", c.timeStamp, - e == MIDI_EVENT_NOTE_OFF ? "Note off".ptr - :e == MIDI_EVENT_NOTE_ON ? "Note on".ptr - :e == MIDI_EVENT_PROGRAM_CHANGE ? "Program change".ptr - :e == MIDI_EVENT_NOTE_AFTERTOUCH ? "Aftertouch".ptr - : "I dunno".ptr - , c.param1, c.param2); -} - -MidiChunk* getTrackNameChunk(Midi* m, int track){ - MidiChunk* c; - - if(track >= m.numTracks) - return null; - - c = m.tracks[track].chunks; - while(c){ - if(c.event == 0xff && c.type == 3) - return c; - - c = c.next; - } - - return c; -} - -int getMidiTempo(Midi* m){ - int a; - MidiChunk* c; - for(a = 0; a < m.numTracks; a++){ - c = m.tracks[a].chunks; - while(c){ - if(c.event == 0xff) - if(c.type == 0x51){ - int p = 0; - p |= cast(int)(c.data[0]) << 16; - p |= cast(int)(c.data[1]) << 8; - p |= cast(int)(c.data[2]) << 0; - - return 60000000 / p; - } - c = c.next; - } - } - - return 120; -} - -int getTempoFromTempoEvent(MidiChunk* c){ - int tempo = -1; - if(c.event == 0xff && c.type == 0x51){ - int p = 0; - p |= cast(int)(c.data[0]) << 16; - p |= cast(int)(c.data[1]) << 8; - p |= cast(int)(c.data[2]) << 0; - tempo = 60000000 / p; - } - return tempo; -} - -// returns milliseconds to wait given the params -int getMidiWaitTime(Midi* mid, int timeStamp, int tempo){ - return (timeStamp * 60000) / (tempo * mid.speed); -} - -// sets absolute values and links up, useful for playing or editing -// but remember you must recalculate them yourself if you change anything -// Returns the final absolute time in seconds -int recalculateMidiAbsolutes(Midi* mid){ - MidiChunk*[128] c; - int[128] trackWaits; - int playing; - int waited; - int minWait = 100000; - int a; - uint absoluteTime = 0; - int tempo = 120; - uint absoluteTimeInMilliSeconds = 0; - int t; - int timeOfLastEvent = 0; - - MidiChunk* absoulteCurrent; - - mid.firstAbsolute = null; - absoulteCurrent = null; - - playing = mid.numTracks; - for(a = 0; a < mid.numTracks; a++){ - c[a] = mid.tracks[a].chunks; - if(c[a]){ - trackWaits[a] = c[a].timeStamp; - if(trackWaits[a] < minWait) - minWait = trackWaits[a]; - } else - playing--; - } - - while(playing){ - waited = minWait; - minWait = 1000000; - absoluteTime += waited; - absoluteTimeInMilliSeconds += getMidiWaitTime(mid, waited, tempo); - for(a = 0; a < mid.numTracks; a++){ - if(!c[a]) - continue; - trackWaits[a] -= waited; - if(trackWaits[a] == 0){ - - t = getTempoFromTempoEvent(c[a]); - if(t != -1) - tempo = t; - - // append it to the list - if(absoulteCurrent == null){ - mid.firstAbsolute = c[a]; - absoulteCurrent = c[a]; - } else { - absoulteCurrent.nextAbsolute = c[a]; - absoulteCurrent = absoulteCurrent.nextAbsolute; - } - absoulteCurrent.nextAbsolute = null; - absoulteCurrent.absoluteTime = absoluteTime; - absoulteCurrent.absoluteTimeInMilliSeconds = absoluteTimeInMilliSeconds; - absoulteCurrent.track = a; - absoulteCurrent.absoluteWait = absoluteTime - timeOfLastEvent; - - timeOfLastEvent = absoluteTime; - c[a] = c[a].next; - if(c[a] == null){ - playing --; - trackWaits[a] = 1000000; - } - else - trackWaits[a] = c[a].timeStamp; - } - if(trackWaits[a] < minWait ) - minWait = trackWaits[a]; - } - } - - - return absoluteTimeInMilliSeconds / 1000; -} - -// returns approximate seconds -int getMidiLength(Midi* mid){ - return recalculateMidiAbsolutes(mid); -} - - - - - - -import arsd.simpleaudio; - -struct PlayingMidi { - ushort channelMask; /* The channels that will be played */ - int[128] playtracks; - - // Callbacks - // onPlayedNote. Args: this, note, midi ticks waited since last message - // This is intended for tablature creation - void function(void*, int, int) onPlayedNote; - // onMidiEvent. Args: this, event being executed - // This can be used to print it or whatever - // If you return 1, it skips the event. Return 0 for normal operation - int function(void*, MidiChunk*) onMidiEvent; - - Midi* mid; - MidiOutput* dev; - - int transpose; - float tempoMultiplier; - - /* This stuff is state for the midi in progress */ - int tempo; - - MidiChunk* current; - - int wait; -} - - -// the main loop for the first time -int resetPlayingMidi(PlayingMidi* pmid){ - pmid.current = pmid.mid.firstAbsolute; - pmid.tempo = 120; - pmid.wait = 0; - if(pmid.current) - return getMidiWaitTime(pmid.mid, pmid.current.absoluteWait, cast(int) (pmid.tempo * pmid.tempoMultiplier)); - return 0; -} - -void setPlayingMidiDefaults(PlayingMidi* pmid){ - int a; - pmid.channelMask =0xffff; - for(a = 0; a < 128; a++) - pmid.playtracks[a] = 1; - - pmid.onPlayedNote = null; - pmid.onMidiEvent = null; - - pmid.mid = null; - pmid.dev = null; - - pmid.transpose = 0; - pmid.tempoMultiplier = 1.0; - -} - - -void seekPlayingMidi(PlayingMidi* pmid, int sec){ - pmid.dev.silenceAllNotes(); - pmid.dev.reset(); - - pmid.current = pmid.mid.firstAbsolute; - while(pmid.current){ - if(pmid.current.absoluteTimeInMilliSeconds >= sec * 1000) - break; - pmid.current = pmid.current.next; - } -} - - -// This is the main loop. Returns how many milliseconds to wait before -// calling it again. If zero, then the song is over. -int advancePlayingMidi(PlayingMidi* pmid){ - MidiChunk* c; - if(pmid.current == null) - return 0; - more: - c = pmid.current; - pmid.wait += c.absoluteWait; - - if(pmid.onMidiEvent){ - if(pmid.onMidiEvent(pmid, c)) - goto skip; - } - - if(c.event != 0xff){ - if(pmid.playtracks[c.track]){ - if(pmid.channelMask & (1 << c.channel)){ - int note = c.param1; - if(c.event == MIDI_EVENT_NOTE_ON - || c.event == MIDI_EVENT_NOTE_AFTERTOUCH - || c.event == MIDI_EVENT_NOTE_OFF){ - note += pmid.transpose; - //skipCounter = SKIP_MAX; - } - - if(pmid.dev) - pmid.dev.writeMidiMessage(c.status, note, c.param2); - if(pmid.onPlayedNote) - if(c.event == MIDI_EVENT_NOTE_ON - && c.param2 != 0){ - pmid.onPlayedNote(pmid, - note, - (pmid.wait * 4) / (pmid.mid.speed)); - pmid.wait = 0; - } - } - } - } else { - if(c.type == 0x51) - pmid.tempo = getTempoFromTempoEvent(c); - } - - skip: - pmid.current = pmid.current.nextAbsolute; - if(pmid.current) - if(pmid.current.absoluteWait == 0) - goto more; - else - return getMidiWaitTime( - pmid.mid, - pmid.current.absoluteWait, - cast(int) (pmid.tempo * pmid.tempoMultiplier)); - else return 0; -} - - - +]; version(MidiDemo) { - - - -MidiOutput* globaldev; - -version(Windows) - import core.sys.windows.windows; -else { - import core.sys.posix.unistd; - void Sleep(int ms){ - usleep(ms*1000); - } - - import core.stdc.signal; - // FIXME: this sucks. - extern(C) - alias fuckyou = void function(int) @nogc nothrow @system; - extern(C) - void sigint(){ - if(globaldev){ - globaldev.silenceAllNotes(); - globaldev.reset(); - destroy(*globaldev); - } - exit(1); - } -} - enum SKIP_MAX = 3000; // allow no more than about 3 seconds of silence // if the -k option is set @@ -1033,44 +900,6 @@ void awesome(void* midiptr, int note, int wait) { // FIXME: add support for displaying lyrics extern(C) int main(int argc, char** argv){ - int a, b; - - PlayingMidi pmid; - - int tempo = 120; - Midi* mid; - MidiOutput midiout = MidiOutput(0); - MidiChunk*[128] c; - - int minWait = 10000, waited; - int playing; - - int wait = 0; - int num; - - char* filename = null; - - int verbose = 0; - float tempoMultiplier = 1; - int transpose = 0; - int displayinfo = 0; - int play = 1; - int tracing = 0; - int skip = 0; - int[128] playtracks; - int skipCounter = SKIP_MAX; - - ushort channelMask = 0xffff; - - int sleepTime = 0; - - version(Posix) { - signal(SIGINT, cast(fuckyou) &sigint); - } - - for(a = 0; a< 128; a++) - playtracks[a] = 1; - for(a = 1; a < argc; a++){ if(argv[a][0] == '-') @@ -1182,12 +1011,6 @@ extern(C) int main(int argc, char** argv){ return 1; } - - - - - - loadMidi(&mid, filename); if(mid == null){ printf("%s: unable to read file %s\n", argv[0], filename); @@ -1215,121 +1038,6 @@ extern(C) int main(int argc, char** argv){ return 0; } - - - if(play){ - globaldev = &midiout; - } else - globaldev = null; - - - recalculateMidiAbsolutes(mid); - setPlayingMidiDefaults(&pmid); - - if(tracing) - pmid.onPlayedNote = &awesome; - pmid.mid = mid; - pmid.dev = &midiout; - - for(a = 0; a < 127; a++) - pmid.playtracks[a] = playtracks[a]; - - pmid.channelMask = channelMask; - pmid.transpose = transpose; - pmid.tempoMultiplier = tempoMultiplier; - - - sleepTime = resetPlayingMidi(&pmid); - do { - //printf("%d\n", sleepTime); - if(play) { - if(skip && sleepTime > 1000) - sleepTime = 1000; - Sleep(sleepTime); - } - sleepTime = advancePlayingMidi(&pmid); - } while(sleepTime); - - -/* - playing = mid.numTracks; - - // prepare! - for(a = 0; a < mid.numTracks; a++){ - c[a] = mid.tracks[a].chunks; - if(c[a]){ - trackWaits[a] = c[a].timeStamp; - if(trackWaits[a] < minWait) - minWait = trackWaits[a]; - } else - playing--; - } - - while(playing){ - if(play && (!skip || skipCounter > 100)){ - Sleep(getMidiWaitTime(mid, minWait, (int)(tempo * tempoMultiplier))); - if(skip) - skipCounter -= getMidiWaitTime(mid, minWait, (int)(tempo*tempoMultiplier)); - } - waited = minWait; - minWait = 1000000; - wait += waited; - for(a = 0; a < mid.numTracks; a++){ - if(!c[a]) - continue; - trackWaits[a] -= waited; - if(trackWaits[a] == 0){ - if(c[a].event != 0xff){ - if(playtracks[a]){ - if(playchannels[c[a].channel]){ - int note = c[a].param1; - if(c[a].event == MIDI_EVENT_NOTE_ON - || c[a].event == MIDI_EVENT_NOTE_AFTERTOUCH - || c[a].event == MIDI_EVENT_NOTE_OFF){ - note += transpose; - skipCounter = SKIP_MAX; - } - - if(play) - writeMidiMessage(dev, c[a].status, note, c[a].param2); - if(tracing) - if(c[a].event == MIDI_EVENT_NOTE_ON - && c[a].param2 != 0){ - printf("%d %d ", - (wait * 4) / (mid.speed), - note); - fflush(stdout); - wait = 0; - } - } - } - // data output: - // waittime note - // waittime is in 1/16 notes - } else { - if(c[a].type == 0x51){ - tempo = getTempoFromTempoEvent(c[a]); - if(verbose) - printf("Tempo change: %d\n", tempo); - } - } - c[a] = c[a].next; - if(c[a] == null){ - playing --; - trackWaits[a] = 1000000; - } - else - trackWaits[a] = c[a].timeStamp; - } - - if(trackWaits[a] < minWait ) - minWait = trackWaits[a]; - } - } -*/ - - freeMidi(&mid); - return 0; } } diff --git a/mp3.d b/mp3.d new file mode 100644 index 0000000..4bc0872 --- /dev/null +++ b/mp3.d @@ -0,0 +1,3078 @@ +/* + * MPEG Audio Layer III decoder + * Copyright (c) 2001, 2002 Fabrice Bellard, + * (c) 2007 Martin J. Fiedler + * + * D conversion by Ketmar // Invisible Vector + * + * This file is a stripped-down version of the MPEG Audio decoder from + * the FFmpeg libavcodec library. + * + * FFmpeg and minimp3 are free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 2.1 of the License, or (at your option) any later version. + * + * FFmpeg and minimp3 are distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with FFmpeg; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA + */ +/++ + Port of ffmpeg's minimp3 lib to D. + + Authors: + Code originally by Fabrice Bellard and Martin J. Fiedler. + + Ported to D by ketmar. + + Hacked up by Adam. + License: + LGPL 2.1. ++/ +module arsd.mp3; + +/* code sample: + auto fi = File(args[1]); + + auto reader = delegate (void[] buf) { + auto rd = fi.rawRead(buf[]); + return cast(int)rd.length; + }; + + auto mp3 = new MP3Decoder(reader); + + if (!mp3.valid) { + writeln("invalid MP3 file!"); + return; + } + + writeln("sample rate: ", mp3.sampleRate); + writeln("channels : ", mp3.channels); + + auto fo = File("z00.raw", "w"); + while (mp3.valid) { + fo.rawWrite(mp3.frameSamples); + mp3.decodeNextFrame(reader); + } + fo.close(); +*/ + +/* determining mp3 duration with scanning: + auto fi = File(args.length > 1 ? args[1] : FileName); + + auto info = mp3Scan((void[] buf) { + auto rd = fi.rawRead(buf[]); + return cast(uint)rd.length; + }); + + if (!info.valid) { + writeln("invalid MP3 file!"); + } else { + writeln("sample rate: ", info.sampleRate); + writeln("channels : ", info.channels); + writeln("samples : ", info.samples); + auto seconds = info.samples/info.sampleRate; + writefln("time: %2s:%02s", seconds/60, seconds%60); + } +*/ + + +// ////////////////////////////////////////////////////////////////////////// // +alias MP3Decoder = MP3DecoderImpl!true; +alias MP3DecoderNoGC = MP3DecoderImpl!false; + + +// ////////////////////////////////////////////////////////////////////////// // +// see iv.mp3scan +/+ +struct MP3Info { + uint sampleRate; + ubyte channels; + ulong samples; + + @property bool valid () const pure nothrow @safe @nogc { return (sampleRate != 0); } +} + + +MP3Info mp3Scan(RDG) (scope RDG rdg) if (is(typeof({ + ubyte[2] buf; + int rd = rdg(buf[]); +}))) { + MP3Info info; + bool eofhit; + ubyte[4096] inbuf; + enum inbufsize = cast(uint)inbuf.length; + uint inbufpos, inbufused; + mp3_context_t* s = cast(mp3_context_t*)libc_calloc(mp3_context_t.sizeof, 1); + if (s is null) return info; + scope(exit) libc_free(s); + bool skipTagCheck; + int headersCount; + + void readMoreData () { + if (inbufused-inbufpos < 1441) { + import core.stdc.string : memmove; + auto left = inbufused-inbufpos; + if (inbufpos > 0) memmove(inbuf.ptr, inbuf.ptr+inbufpos, left); + inbufpos = 0; + inbufused = left; + // read more bytes + left = inbufsize-inbufused; + int rd = rdg(inbuf[inbufused..inbufused+left]); + if (rd <= 0) { + eofhit = true; + } else { + inbufused += rd; + } + } + } + + // now skip frames + while (!eofhit) { + readMoreData(); + if (eofhit && inbufused-inbufpos < 1024) break; + auto left = inbufused-inbufpos; + // check for tags + if (!skipTagCheck) { + skipTagCheck = true; + if (left >= 10) { + // check for ID3v2 + if (inbuf.ptr[0] == 'I' && inbuf.ptr[1] == 'D' && inbuf.ptr[2] == '3' && inbuf.ptr[3] != 0xff && inbuf.ptr[4] != 0xff && + ((inbuf.ptr[6]|inbuf.ptr[7]|inbuf.ptr[8]|inbuf.ptr[9])&0x80) == 0) { // see ID3v2 specs + // get tag size + uint sz = (inbuf.ptr[9]|(inbuf.ptr[8]<<7)|(inbuf.ptr[7]<<14)|(inbuf.ptr[6]<<21))+10; + // skip `sz` bytes, it's a tag + while (sz > 0 && !eofhit) { + readMoreData(); + left = inbufused-inbufpos; + if (left > sz) left = sz; + inbufpos += left; + sz -= left; + } + if (eofhit) break; + continue; + } + } + } else { + if (inbuf.ptr[0] == 'T' && inbuf.ptr[1] == 'A' && inbuf.ptr[2] == 'G') { + // this may be ID3v1, just skip 128 bytes + uint sz = 128; + while (sz > 0 && !eofhit) { + readMoreData(); + left = inbufused-inbufpos; + if (left > sz) left = sz; + inbufpos += left; + sz -= left; + } + if (eofhit) break; + continue; + } + } + int res = mp3_skip_frame(s, inbuf.ptr+inbufpos, left); + if (res < 0) { + // can't decode frame + if (inbufused-inbufpos < 1024) inbufpos = inbufused; else inbufpos += 1024; + } else { + if (headersCount < 6) ++headersCount; + if (!info.valid) { + if (s.sample_rate < 1024 || s.sample_rate > 96000) break; + if (s.nb_channels < 1 || s.nb_channels > 2) break; + info.sampleRate = s.sample_rate; + info.channels = cast(ubyte)s.nb_channels; + } + info.samples += s.sample_count; + inbufpos += res; + } + } + //{ import core.stdc.stdio : printf; printf("%d\n", headersCount); } + if (headersCount < 6) info = info.init; + return info; +} ++/ + + +// ////////////////////////////////////////////////////////////////////////// // +final class MP3DecoderImpl(bool allowGC) { +public: + enum MaxSamplesPerFrame = 1152*2; + + // read bytes into the buffer, return number of bytes read or 0 for EOF, -1 on error + // will never be called with empty buffer, or buffer more than 128KB + static if (allowGC) { + alias ReadBufFn = int delegate (void[] buf); + } else { + alias ReadBufFn = int delegate (void[] buf) nothrow @nogc; + } + +public: + static struct mp3_info_t { + int sample_rate; + int channels; + int audio_bytes; // generated amount of audio per frame + } + +private: + void* dec; + //ReadBufFn readBuf; + ubyte* inbuf; + uint inbufsize; // will allocate enough bytes for one frame + uint inbufpos, inbufused; + bool eofhit; + short[MaxSamplesPerFrame] samples; + uint scanLeft = 256*1024+16; // how much bytes we should scan (max ID3 size is 256KB) + + static if (allowGC) mixin(ObjectCodeMixin); else mixin("nothrow @nogc: "~ObjectCodeMixin); +} + +private enum ObjectCodeMixin = q{ +private: + uint ensureBytes (scope ReadBufFn readBuf, uint size) { + import core.stdc.string : memmove; + for (;;) { + assert(inbufused >= inbufpos); + uint left = inbufused-inbufpos; + if (left >= size) return size; + if (eofhit) return left; + if (left > 0) { + if (inbufpos > 0) memmove(inbuf, inbuf+inbufpos, left); + inbufused = left; + } else { + inbufused = 0; + } + inbufpos = 0; + //{ import std.conv : to; assert(size > inbufused, "size="~to!string(size)~"; inbufpos="~to!string(inbufpos)~"; inbufused="~to!string(inbufused)~"; inbufsize="~to!string(inbufsize)); } + assert(size > inbufused); + left = size-inbufused; + assert(left > 0); + if (inbufsize < inbufused+left) { + auto np = libc_realloc(inbuf, inbufused+left); + if (np is null) assert(0, "out of memory"); //FIXME + inbufsize = inbufused+left; + inbuf = cast(ubyte*)np; + } + auto rd = readBuf(inbuf[inbufused..inbufused+left]); + if (rd > left) assert(0, "mp3 reader returned too many bytes"); + if (rd <= 0) eofhit = true; else inbufused += rd; + } + } + + void removeBytes (uint size) { + if (size == 0) return; + if (size > inbufused-inbufpos) { + //assert(0, "the thing that should not be"); + // we will come here when we are scanning for MP3 frame and no more bytes left + eofhit = true; + inbufpos = inbufused; + } else { + inbufpos += size; + } + } + +private: + mp3_info_t info; + bool curFrameIsOk; + bool skipTagCheck; + +private: + bool decodeOneFrame (scope ReadBufFn readBuf, bool first=false) { + for (;;) { + if (!eofhit && inbufused-inbufpos < 1441) ensureBytes(readBuf, 64*1024); + int res, size = -1; + + // check for tags + if (!skipTagCheck) { + skipTagCheck = false; + if (inbufused-inbufpos >= 10) { + // check for ID3v2 + if (inbuf[inbufpos+0] == 'I' && inbuf[inbufpos+1] == 'D' && inbuf[inbufpos+2] == '3' && inbuf[inbufpos+3] != 0xff && inbuf[inbufpos+4] != 0xff && + ((inbuf[inbufpos+6]|inbuf[inbufpos+7]|inbuf[inbufpos+8]|inbuf[inbufpos+9])&0x80) == 0) { // see ID3v2 specs + // get tag size + uint sz = (inbuf[inbufpos+9]|(inbuf[inbufpos+8]<<7)|(inbuf[inbufpos+7]<<14)|(inbuf[inbufpos+6]<<21))+10; + // skip `sz` bytes, it's a tag + while (sz > 0 && !eofhit) { + ensureBytes(readBuf, 64*1024); + auto left = inbufused-inbufpos; + if (left > sz) left = sz; + removeBytes(left); + sz -= left; + } + if (eofhit) { curFrameIsOk = false; return false; } + continue; + } + } + } else { + if (inbuf[inbufpos+0] == 'T' && inbuf[inbufpos+1] == 'A' && inbuf[inbufpos+2] == 'G') { + // this may be ID3v1, just skip 128 bytes + uint sz = 128; + while (sz > 0 && !eofhit) { + ensureBytes(readBuf, 64*1024); + auto left = inbufused-inbufpos; + if (left > sz) left = sz; + removeBytes(left); + sz -= left; + } + if (eofhit) { curFrameIsOk = false; return false; } + continue; + } + } + + mp3_context_t* s = cast(mp3_context_t*)dec; + res = mp3_decode_frame(s, /*cast(int16_t*)out_*/samples.ptr, &size, inbuf+inbufpos, /*bytes*/inbufused-inbufpos); + if (res < 0) { + // can't decode frame + if (scanLeft >= 1024) { + scanLeft -= 1024; + removeBytes(1024); + continue; + } + curFrameIsOk = false; + return false; + } + info.audio_bytes = size; + if (first) { + info.sample_rate = s.sample_rate; + info.channels = s.nb_channels; + if ((info.sample_rate < 1024 || info.sample_rate > 96000) || + (info.channels < 1 || info.channels > 2) || + (info.audio_bytes < 2 || info.audio_bytes > MaxSamplesPerFrame*2 || info.audio_bytes%2 != 0)) + { + curFrameIsOk = false; + return false; + } + curFrameIsOk = true; + } else { + if ((s.sample_rate < 1024 || s.sample_rate > 96000) || + (s.nb_channels < 1 || s.nb_channels > 2) || + (size < 2 || size > MaxSamplesPerFrame*2 || size%2 != 0)) + { + curFrameIsOk = false; + } else { + curFrameIsOk = true; + } + } + if (curFrameIsOk) { + scanLeft = 256*1024+16; + removeBytes(s.frame_size); + return /*s.frame_size*/true; + } + if (scanLeft >= 1024) { + scanLeft -= 1024; + removeBytes(1024); + continue; + } + return false; + } + } + +public: + this (scope ReadBufFn reader) { + static if (allowGC) { + if (reader is null) throw new Exception("reader is null"); + } else { + if (reader is null) assert(0, "reader is null"); + } + //readBuf = reader; + dec = libc_calloc(mp3_context_t.sizeof, 1); + if (dec is null) assert(0, "out of memory"); // no, really! ;-) + //mp3_decode_init(cast(mp3_context_t*)dec); + if (!decodeOneFrame(reader, true)) close(); + } + + ~this () { close(); } + + void close () { + if (dec !is null) { libc_free(dec); dec = null; } + if (inbuf !is null) { libc_free(inbuf); inbuf = null; } + info.audio_bytes = 0; + } + + // restart decoding + void restart (scope ReadBufFn reader) { + inbufpos = inbufused = 0; + eofhit = false; + info.audio_bytes = 0; + scanLeft = 256*1024+16; + skipTagCheck = false; + if (!decodeOneFrame(reader, true)) close(); + } + + // empty read buffers and decode next frame; should be used to sync after seeking in input stream + void sync (scope ReadBufFn reader) { + inbufpos = inbufused = 0; + eofhit = false; + info.audio_bytes = 0; + scanLeft = 256*1024+16; + skipTagCheck = false; + if (!decodeOneFrame(reader)) close(); + } + + bool decodeNextFrame (scope ReadBufFn reader) { + if (!valid) return false; + static if (allowGC) scope(failure) close(); + if (reader is null) return false; + if (!decodeOneFrame(reader)) { + close(); + return false; + } + return true; + } + + @property bool valid () const pure nothrow @safe @nogc { return (dec !is null && curFrameIsOk); } + @property uint sampleRate () const pure nothrow @safe @nogc { return (valid ? info.sample_rate : 0); } + @property ubyte channels () const pure nothrow @safe @nogc { return (valid ? cast(ubyte)info.channels : 0); } + @property int samplesInFrame () const pure nothrow @safe @nogc { return (valid ? cast(ubyte)info.audio_bytes : 0); } + + @property short[] frameSamples () nothrow @nogc { + if (!valid) return null; + return samples[0..info.audio_bytes/2]; + } +}; + + +// ////////////////////////////////////////////////////////////////////////// // +private: +nothrow @nogc: +import core.stdc.stdlib : libc_calloc = calloc, libc_malloc = malloc, libc_realloc = realloc, libc_free = free; +import core.stdc.string : libc_memcpy = memcpy, libc_memset = memset, libc_memmove = memmove; + +import std.math : libc_pow = pow, libc_frexp = frexp, tan, M_PI = PI, sqrt, cos, sin; + +/* +void* libc_calloc (usize nmemb, usize count) { + import core.stdc.stdlib : calloc; + import core.stdc.stdio : printf; + printf("calloc(%zu, %zu)\n", nmemb, count); + return calloc(nmemb, count); +} + +void* libc_malloc (usize count) { + import core.stdc.stdlib : malloc; + import core.stdc.stdio : printf; + printf("malloc(%zu)\n", count+1024*1024); + return malloc(count); +} + +void* libc_realloc (void* ptr, usize count) { + import core.stdc.stdlib : realloc; + import core.stdc.stdio : printf; + printf("realloc(%p, %zu)\n", ptr, count); + return realloc(ptr, count+1024*1024); +} + +void libc_free (void* ptr) { + import core.stdc.stdlib : free; + import core.stdc.stdio : printf; + printf("free(%p)\n", ptr); + return free(ptr); +} +*/ + +enum MP3_FRAME_SIZE = 1152; +enum MP3_MAX_CODED_FRAME_SIZE = 1792; +enum MP3_MAX_CHANNELS = 2; +enum SBLIMIT = 32; + +enum MP3_STEREO = 0; +enum MP3_JSTEREO = 1; +enum MP3_DUAL = 2; +enum MP3_MONO = 3; + +enum SAME_HEADER_MASK = (0xffe00000 | (3 << 17) | (0xf << 12) | (3 << 10) | (3 << 19)); + +enum FRAC_BITS = 15; +enum WFRAC_BITS = 14; + +enum OUT_MAX = (32767); +enum OUT_MIN = (-32768); +enum OUT_SHIFT = (WFRAC_BITS + FRAC_BITS - 15); + +enum MODE_EXT_MS_STEREO = 2; +enum MODE_EXT_I_STEREO = 1; + +enum FRAC_ONE = (1 << FRAC_BITS); +//enum FIX(a) ((int)((a) * FRAC_ONE)) +enum FIXR(double a) = (cast(int)((a) * FRAC_ONE + 0.5)); +int FIXRx(double a) { static if (__VERSION__ > 2067) pragma(inline, true); return (cast(int)((a) * FRAC_ONE + 0.5)); } +//enum FRAC_RND(a) (((a) + (FRAC_ONE/2)) >> FRAC_BITS) +enum FIXHR(double a) = (cast(int)((a) * (1L<<32) + 0.5)); +int FIXHRx() (double a) { static if (__VERSION__ > 2067) pragma(inline, true); return (cast(int)((a) * (1L<<32) + 0.5)); } + +long MULL() (int a, int b) { static if (__VERSION__ > 2067) pragma(inline, true); return ((cast(long)(a) * cast(long)(b)) >> FRAC_BITS); } +long MULH() (int a, int b) { static if (__VERSION__ > 2067) pragma(inline, true); return ((cast(long)(a) * cast(long)(b)) >> 32); } +auto MULS(T) (T ra, T rb) { static if (__VERSION__ > 2067) pragma(inline, true); return ((ra) * (rb)); } + +enum ISQRT2 = FIXR!(0.70710678118654752440); + +enum HEADER_SIZE = 4; +enum BACKSTEP_SIZE = 512; +enum EXTRABYTES = 24; + + +// ////////////////////////////////////////////////////////////////////////// // +alias VLC_TYPE = short; +alias VT2 = VLC_TYPE[2]; + +alias int8_t = byte; +alias int16_t = short; +alias int32_t = int; +alias int64_t = long; + +alias uint8_t = ubyte; +alias uint16_t = ushort; +alias uint32_t = uint; +alias uint64_t = ulong; + +struct bitstream_t { + const(ubyte)* buffer, buffer_end; + int index; + int size_in_bits; +} + +struct vlc_t { + int bits; + //VLC_TYPE (*table)[2]; ///< code, bits + VT2* table; + int table_size, table_allocated; +} + +struct mp3_context_t { + uint8_t[2*BACKSTEP_SIZE+EXTRABYTES] last_buf; + int last_buf_size; + int frame_size; + uint32_t free_format_next_header; + int error_protection; + int sample_rate; + int sample_rate_index; + int bit_rate; + bitstream_t gb; + bitstream_t in_gb; + int nb_channels; + int sample_count; + int mode; + int mode_ext; + int lsf; + int16_t[512*2][MP3_MAX_CHANNELS] synth_buf; + int[MP3_MAX_CHANNELS] synth_buf_offset; + int32_t[SBLIMIT][36][MP3_MAX_CHANNELS] sb_samples; + int32_t[SBLIMIT*18][MP3_MAX_CHANNELS] mdct_buf; + int dither_state; + uint last_header; //&0xffff0c00u; +} + +struct granule_t { + uint8_t scfsi; + int part2_3_length; + int big_values; + int global_gain; + int scalefac_compress; + uint8_t block_type; + uint8_t switch_point; + int[3] table_select; + int[3] subblock_gain; + uint8_t scalefac_scale; + uint8_t count1table_select; + int[3] region_size; + int preflag; + int short_start, long_end; + uint8_t[40] scale_factors; + int32_t[SBLIMIT * 18] sb_hybrid; +} + +struct huff_table_t { + int xsize; + immutable(uint8_t)* bits; + immutable(uint16_t)* codes; +} + +__gshared vlc_t[16] huff_vlc; +__gshared vlc_t[2] huff_quad_vlc; +__gshared uint16_t[23][9] band_index_long; +enum TABLE_4_3_SIZE = (8191 + 16)*4; +__gshared int8_t* table_4_3_exp; +__gshared uint32_t* table_4_3_value; +__gshared uint32_t[512] exp_table; +__gshared uint32_t[16][512] expval_table; +__gshared int32_t[16][2] is_table; +__gshared int32_t[16][2][2] is_table_lsf; +__gshared int32_t[4][8] csa_table; +__gshared float[4][8] csa_table_float; +__gshared int32_t[36][8] mdct_win; +__gshared int16_t[512] window; + + +// ////////////////////////////////////////////////////////////////////////// // +static immutable uint16_t[15][2] mp3_bitrate_tab = [ + [0, 32, 40, 48, 56, 64, 80, 96, 112, 128, 160, 192, 224, 256, 320 ], + [0, 8, 16, 24, 32, 40, 48, 56, 64, 80, 96, 112, 128, 144, 160] +]; + +static immutable uint16_t[3] mp3_freq_tab = [ 44100, 48000, 32000 ]; + +static immutable int32_t[257] mp3_enwindow = [ + 0, -1, -1, -1, -1, -1, -1, -2, + -2, -2, -2, -3, -3, -4, -4, -5, + -5, -6, -7, -7, -8, -9, -10, -11, + -13, -14, -16, -17, -19, -21, -24, -26, + -29, -31, -35, -38, -41, -45, -49, -53, + -58, -63, -68, -73, -79, -85, -91, -97, + -104, -111, -117, -125, -132, -139, -147, -154, + -161, -169, -176, -183, -190, -196, -202, -208, + 213, 218, 222, 225, 227, 228, 228, 227, + 224, 221, 215, 208, 200, 189, 177, 163, + 146, 127, 106, 83, 57, 29, -2, -36, + -72, -111, -153, -197, -244, -294, -347, -401, + -459, -519, -581, -645, -711, -779, -848, -919, + -991, -1064, -1137, -1210, -1283, -1356, -1428, -1498, + -1567, -1634, -1698, -1759, -1817, -1870, -1919, -1962, + -2001, -2032, -2057, -2075, -2085, -2087, -2080, -2063, + 2037, 2000, 1952, 1893, 1822, 1739, 1644, 1535, + 1414, 1280, 1131, 970, 794, 605, 402, 185, + -45, -288, -545, -814, -1095, -1388, -1692, -2006, + -2330, -2663, -3004, -3351, -3705, -4063, -4425, -4788, + -5153, -5517, -5879, -6237, -6589, -6935, -7271, -7597, + -7910, -8209, -8491, -8755, -8998, -9219, -9416, -9585, + -9727, -9838, -9916, -9959, -9966, -9935, -9863, -9750, + -9592, -9389, -9139, -8840, -8492, -8092, -7640, -7134, + 6574, 5959, 5288, 4561, 3776, 2935, 2037, 1082, + 70, -998, -2122, -3300, -4533, -5818, -7154, -8540, + -9975,-11455,-12980,-14548,-16155,-17799,-19478,-21189, +-22929,-24694,-26482,-28289,-30112,-31947,-33791,-35640, +-37489,-39336,-41176,-43006,-44821,-46617,-48390,-50137, +-51853,-53534,-55178,-56778,-58333,-59838,-61289,-62684, +-64019,-65290,-66494,-67629,-68692,-69679,-70590,-71420, +-72169,-72835,-73415,-73908,-74313,-74630,-74856,-74992, + 75038, +]; + +static immutable uint8_t[16][2] slen_table = [ + [ 0, 0, 0, 0, 3, 1, 1, 1, 2, 2, 2, 3, 3, 3, 4, 4 ], + [ 0, 1, 2, 3, 0, 1, 2, 3, 1, 2, 3, 1, 2, 3, 2, 3 ], +]; + +static immutable uint8_t[4][3][6] lsf_nsf_table = [ + [ [ 6, 5, 5, 5 ], [ 9, 9, 9, 9 ], [ 6, 9, 9, 9 ] ], + [ [ 6, 5, 7, 3 ], [ 9, 9, 12, 6 ], [ 6, 9, 12, 6 ] ], + [ [ 11, 10, 0, 0 ], [ 18, 18, 0, 0 ], [ 15, 18, 0, 0 ] ], + [ [ 7, 7, 7, 0 ], [ 12, 12, 12, 0 ], [ 6, 15, 12, 0 ] ], + [ [ 6, 6, 6, 3 ], [ 12, 9, 9, 6 ], [ 6, 12, 9, 6 ] ], + [ [ 8, 8, 5, 0 ], [ 15, 12, 9, 0 ], [ 6, 18, 9, 0 ] ], +]; + +static immutable uint16_t[4] mp3_huffcodes_1 = [ 0x0001, 0x0001, 0x0001, 0x0000, ]; + +static immutable uint8_t[4] mp3_huffbits_1 = [ 1, 3, 2, 3, ]; + +static immutable uint16_t[9] mp3_huffcodes_2 = [ 0x0001, 0x0002, 0x0001, 0x0003, 0x0001, 0x0001, 0x0003, 0x0002, 0x0000, ]; + +static immutable uint8_t[9] mp3_huffbits_2 = [ 1, 3, 6, 3, 3, 5, 5, 5, 6, ]; + +static immutable uint16_t[9] mp3_huffcodes_3 = [ 0x0003, 0x0002, 0x0001, 0x0001, 0x0001, 0x0001, 0x0003, 0x0002, 0x0000, ]; + +static immutable uint8_t[9] mp3_huffbits_3 = [ 2, 2, 6, 3, 2, 5, 5, 5, 6, ]; + +static immutable uint16_t[16] mp3_huffcodes_5 = [ + 0x0001, 0x0002, 0x0006, 0x0005, 0x0003, 0x0001, 0x0004, 0x0004, + 0x0007, 0x0005, 0x0007, 0x0001, 0x0006, 0x0001, 0x0001, 0x0000, +]; + +static immutable uint8_t[16] mp3_huffbits_5 = [ + 1, 3, 6, 7, 3, 3, 6, 7, + 6, 6, 7, 8, 7, 6, 7, 8, +]; + +static immutable uint16_t[16] mp3_huffcodes_6 = [ + 0x0007, 0x0003, 0x0005, 0x0001, 0x0006, 0x0002, 0x0003, 0x0002, + 0x0005, 0x0004, 0x0004, 0x0001, 0x0003, 0x0003, 0x0002, 0x0000, +]; + +static immutable uint8_t[16] mp3_huffbits_6 = [ + 3, 3, 5, 7, 3, 2, 4, 5, + 4, 4, 5, 6, 6, 5, 6, 7, +]; + +static immutable uint16_t[36] mp3_huffcodes_7 = [ + 0x0001, 0x0002, 0x000a, 0x0013, 0x0010, 0x000a, 0x0003, 0x0003, + 0x0007, 0x000a, 0x0005, 0x0003, 0x000b, 0x0004, 0x000d, 0x0011, + 0x0008, 0x0004, 0x000c, 0x000b, 0x0012, 0x000f, 0x000b, 0x0002, + 0x0007, 0x0006, 0x0009, 0x000e, 0x0003, 0x0001, 0x0006, 0x0004, + 0x0005, 0x0003, 0x0002, 0x0000, +]; + +static immutable uint8_t[36] mp3_huffbits_7 = [ + 1, 3, 6, 8, 8, 9, 3, 4, + 6, 7, 7, 8, 6, 5, 7, 8, + 8, 9, 7, 7, 8, 9, 9, 9, + 7, 7, 8, 9, 9, 10, 8, 8, + 9, 10, 10, 10, +]; + +static immutable uint16_t[36] mp3_huffcodes_8 = [ + 0x0003, 0x0004, 0x0006, 0x0012, 0x000c, 0x0005, 0x0005, 0x0001, + 0x0002, 0x0010, 0x0009, 0x0003, 0x0007, 0x0003, 0x0005, 0x000e, + 0x0007, 0x0003, 0x0013, 0x0011, 0x000f, 0x000d, 0x000a, 0x0004, + 0x000d, 0x0005, 0x0008, 0x000b, 0x0005, 0x0001, 0x000c, 0x0004, + 0x0004, 0x0001, 0x0001, 0x0000, +]; + +static immutable uint8_t[36] mp3_huffbits_8 = [ + 2, 3, 6, 8, 8, 9, 3, 2, + 4, 8, 8, 8, 6, 4, 6, 8, + 8, 9, 8, 8, 8, 9, 9, 10, + 8, 7, 8, 9, 10, 10, 9, 8, + 9, 9, 11, 11, +]; + +static immutable uint16_t[36] mp3_huffcodes_9 = [ + 0x0007, 0x0005, 0x0009, 0x000e, 0x000f, 0x0007, 0x0006, 0x0004, + 0x0005, 0x0005, 0x0006, 0x0007, 0x0007, 0x0006, 0x0008, 0x0008, + 0x0008, 0x0005, 0x000f, 0x0006, 0x0009, 0x000a, 0x0005, 0x0001, + 0x000b, 0x0007, 0x0009, 0x0006, 0x0004, 0x0001, 0x000e, 0x0004, + 0x0006, 0x0002, 0x0006, 0x0000, +]; + +static immutable uint8_t[36] mp3_huffbits_9 = [ + 3, 3, 5, 6, 8, 9, 3, 3, + 4, 5, 6, 8, 4, 4, 5, 6, + 7, 8, 6, 5, 6, 7, 7, 8, + 7, 6, 7, 7, 8, 9, 8, 7, + 8, 8, 9, 9, +]; + +static immutable uint16_t[64] mp3_huffcodes_10 = [ + 0x0001, 0x0002, 0x000a, 0x0017, 0x0023, 0x001e, 0x000c, 0x0011, + 0x0003, 0x0003, 0x0008, 0x000c, 0x0012, 0x0015, 0x000c, 0x0007, + 0x000b, 0x0009, 0x000f, 0x0015, 0x0020, 0x0028, 0x0013, 0x0006, + 0x000e, 0x000d, 0x0016, 0x0022, 0x002e, 0x0017, 0x0012, 0x0007, + 0x0014, 0x0013, 0x0021, 0x002f, 0x001b, 0x0016, 0x0009, 0x0003, + 0x001f, 0x0016, 0x0029, 0x001a, 0x0015, 0x0014, 0x0005, 0x0003, + 0x000e, 0x000d, 0x000a, 0x000b, 0x0010, 0x0006, 0x0005, 0x0001, + 0x0009, 0x0008, 0x0007, 0x0008, 0x0004, 0x0004, 0x0002, 0x0000, +]; + +static immutable uint8_t[64] mp3_huffbits_10 = [ + 1, 3, 6, 8, 9, 9, 9, 10, + 3, 4, 6, 7, 8, 9, 8, 8, + 6, 6, 7, 8, 9, 10, 9, 9, + 7, 7, 8, 9, 10, 10, 9, 10, + 8, 8, 9, 10, 10, 10, 10, 10, + 9, 9, 10, 10, 11, 11, 10, 11, + 8, 8, 9, 10, 10, 10, 11, 11, + 9, 8, 9, 10, 10, 11, 11, 11, +]; + +static immutable uint16_t[64] mp3_huffcodes_11 = [ + 0x0003, 0x0004, 0x000a, 0x0018, 0x0022, 0x0021, 0x0015, 0x000f, + 0x0005, 0x0003, 0x0004, 0x000a, 0x0020, 0x0011, 0x000b, 0x000a, + 0x000b, 0x0007, 0x000d, 0x0012, 0x001e, 0x001f, 0x0014, 0x0005, + 0x0019, 0x000b, 0x0013, 0x003b, 0x001b, 0x0012, 0x000c, 0x0005, + 0x0023, 0x0021, 0x001f, 0x003a, 0x001e, 0x0010, 0x0007, 0x0005, + 0x001c, 0x001a, 0x0020, 0x0013, 0x0011, 0x000f, 0x0008, 0x000e, + 0x000e, 0x000c, 0x0009, 0x000d, 0x000e, 0x0009, 0x0004, 0x0001, + 0x000b, 0x0004, 0x0006, 0x0006, 0x0006, 0x0003, 0x0002, 0x0000, +]; + +static immutable uint8_t[64] mp3_huffbits_11 = [ + 2, 3, 5, 7, 8, 9, 8, 9, + 3, 3, 4, 6, 8, 8, 7, 8, + 5, 5, 6, 7, 8, 9, 8, 8, + 7, 6, 7, 9, 8, 10, 8, 9, + 8, 8, 8, 9, 9, 10, 9, 10, + 8, 8, 9, 10, 10, 11, 10, 11, + 8, 7, 7, 8, 9, 10, 10, 10, + 8, 7, 8, 9, 10, 10, 10, 10, +]; + +static immutable uint16_t[64] mp3_huffcodes_12 = [ + 0x0009, 0x0006, 0x0010, 0x0021, 0x0029, 0x0027, 0x0026, 0x001a, + 0x0007, 0x0005, 0x0006, 0x0009, 0x0017, 0x0010, 0x001a, 0x000b, + 0x0011, 0x0007, 0x000b, 0x000e, 0x0015, 0x001e, 0x000a, 0x0007, + 0x0011, 0x000a, 0x000f, 0x000c, 0x0012, 0x001c, 0x000e, 0x0005, + 0x0020, 0x000d, 0x0016, 0x0013, 0x0012, 0x0010, 0x0009, 0x0005, + 0x0028, 0x0011, 0x001f, 0x001d, 0x0011, 0x000d, 0x0004, 0x0002, + 0x001b, 0x000c, 0x000b, 0x000f, 0x000a, 0x0007, 0x0004, 0x0001, + 0x001b, 0x000c, 0x0008, 0x000c, 0x0006, 0x0003, 0x0001, 0x0000, +]; + +static immutable uint8_t[64] mp3_huffbits_12 = [ + 4, 3, 5, 7, 8, 9, 9, 9, + 3, 3, 4, 5, 7, 7, 8, 8, + 5, 4, 5, 6, 7, 8, 7, 8, + 6, 5, 6, 6, 7, 8, 8, 8, + 7, 6, 7, 7, 8, 8, 8, 9, + 8, 7, 8, 8, 8, 9, 8, 9, + 8, 7, 7, 8, 8, 9, 9, 10, + 9, 8, 8, 9, 9, 9, 9, 10, +]; + +static immutable uint16_t[256] mp3_huffcodes_13 = [ + 0x0001, 0x0005, 0x000e, 0x0015, 0x0022, 0x0033, 0x002e, 0x0047, + 0x002a, 0x0034, 0x0044, 0x0034, 0x0043, 0x002c, 0x002b, 0x0013, + 0x0003, 0x0004, 0x000c, 0x0013, 0x001f, 0x001a, 0x002c, 0x0021, + 0x001f, 0x0018, 0x0020, 0x0018, 0x001f, 0x0023, 0x0016, 0x000e, + 0x000f, 0x000d, 0x0017, 0x0024, 0x003b, 0x0031, 0x004d, 0x0041, + 0x001d, 0x0028, 0x001e, 0x0028, 0x001b, 0x0021, 0x002a, 0x0010, + 0x0016, 0x0014, 0x0025, 0x003d, 0x0038, 0x004f, 0x0049, 0x0040, + 0x002b, 0x004c, 0x0038, 0x0025, 0x001a, 0x001f, 0x0019, 0x000e, + 0x0023, 0x0010, 0x003c, 0x0039, 0x0061, 0x004b, 0x0072, 0x005b, + 0x0036, 0x0049, 0x0037, 0x0029, 0x0030, 0x0035, 0x0017, 0x0018, + 0x003a, 0x001b, 0x0032, 0x0060, 0x004c, 0x0046, 0x005d, 0x0054, + 0x004d, 0x003a, 0x004f, 0x001d, 0x004a, 0x0031, 0x0029, 0x0011, + 0x002f, 0x002d, 0x004e, 0x004a, 0x0073, 0x005e, 0x005a, 0x004f, + 0x0045, 0x0053, 0x0047, 0x0032, 0x003b, 0x0026, 0x0024, 0x000f, + 0x0048, 0x0022, 0x0038, 0x005f, 0x005c, 0x0055, 0x005b, 0x005a, + 0x0056, 0x0049, 0x004d, 0x0041, 0x0033, 0x002c, 0x002b, 0x002a, + 0x002b, 0x0014, 0x001e, 0x002c, 0x0037, 0x004e, 0x0048, 0x0057, + 0x004e, 0x003d, 0x002e, 0x0036, 0x0025, 0x001e, 0x0014, 0x0010, + 0x0035, 0x0019, 0x0029, 0x0025, 0x002c, 0x003b, 0x0036, 0x0051, + 0x0042, 0x004c, 0x0039, 0x0036, 0x0025, 0x0012, 0x0027, 0x000b, + 0x0023, 0x0021, 0x001f, 0x0039, 0x002a, 0x0052, 0x0048, 0x0050, + 0x002f, 0x003a, 0x0037, 0x0015, 0x0016, 0x001a, 0x0026, 0x0016, + 0x0035, 0x0019, 0x0017, 0x0026, 0x0046, 0x003c, 0x0033, 0x0024, + 0x0037, 0x001a, 0x0022, 0x0017, 0x001b, 0x000e, 0x0009, 0x0007, + 0x0022, 0x0020, 0x001c, 0x0027, 0x0031, 0x004b, 0x001e, 0x0034, + 0x0030, 0x0028, 0x0034, 0x001c, 0x0012, 0x0011, 0x0009, 0x0005, + 0x002d, 0x0015, 0x0022, 0x0040, 0x0038, 0x0032, 0x0031, 0x002d, + 0x001f, 0x0013, 0x000c, 0x000f, 0x000a, 0x0007, 0x0006, 0x0003, + 0x0030, 0x0017, 0x0014, 0x0027, 0x0024, 0x0023, 0x0035, 0x0015, + 0x0010, 0x0017, 0x000d, 0x000a, 0x0006, 0x0001, 0x0004, 0x0002, + 0x0010, 0x000f, 0x0011, 0x001b, 0x0019, 0x0014, 0x001d, 0x000b, + 0x0011, 0x000c, 0x0010, 0x0008, 0x0001, 0x0001, 0x0000, 0x0001, +]; + +static immutable uint8_t[256] mp3_huffbits_13 = [ + 1, 4, 6, 7, 8, 9, 9, 10, + 9, 10, 11, 11, 12, 12, 13, 13, + 3, 4, 6, 7, 8, 8, 9, 9, + 9, 9, 10, 10, 11, 12, 12, 12, + 6, 6, 7, 8, 9, 9, 10, 10, + 9, 10, 10, 11, 11, 12, 13, 13, + 7, 7, 8, 9, 9, 10, 10, 10, + 10, 11, 11, 11, 11, 12, 13, 13, + 8, 7, 9, 9, 10, 10, 11, 11, + 10, 11, 11, 12, 12, 13, 13, 14, + 9, 8, 9, 10, 10, 10, 11, 11, + 11, 11, 12, 11, 13, 13, 14, 14, + 9, 9, 10, 10, 11, 11, 11, 11, + 11, 12, 12, 12, 13, 13, 14, 14, + 10, 9, 10, 11, 11, 11, 12, 12, + 12, 12, 13, 13, 13, 14, 16, 16, + 9, 8, 9, 10, 10, 11, 11, 12, + 12, 12, 12, 13, 13, 14, 15, 15, + 10, 9, 10, 10, 11, 11, 11, 13, + 12, 13, 13, 14, 14, 14, 16, 15, + 10, 10, 10, 11, 11, 12, 12, 13, + 12, 13, 14, 13, 14, 15, 16, 17, + 11, 10, 10, 11, 12, 12, 12, 12, + 13, 13, 13, 14, 15, 15, 15, 16, + 11, 11, 11, 12, 12, 13, 12, 13, + 14, 14, 15, 15, 15, 16, 16, 16, + 12, 11, 12, 13, 13, 13, 14, 14, + 14, 14, 14, 15, 16, 15, 16, 16, + 13, 12, 12, 13, 13, 13, 15, 14, + 14, 17, 15, 15, 15, 17, 16, 16, + 12, 12, 13, 14, 14, 14, 15, 14, + 15, 15, 16, 16, 19, 18, 19, 16, +]; + +static immutable uint16_t[256] mp3_huffcodes_15 = [ + 0x0007, 0x000c, 0x0012, 0x0035, 0x002f, 0x004c, 0x007c, 0x006c, + 0x0059, 0x007b, 0x006c, 0x0077, 0x006b, 0x0051, 0x007a, 0x003f, + 0x000d, 0x0005, 0x0010, 0x001b, 0x002e, 0x0024, 0x003d, 0x0033, + 0x002a, 0x0046, 0x0034, 0x0053, 0x0041, 0x0029, 0x003b, 0x0024, + 0x0013, 0x0011, 0x000f, 0x0018, 0x0029, 0x0022, 0x003b, 0x0030, + 0x0028, 0x0040, 0x0032, 0x004e, 0x003e, 0x0050, 0x0038, 0x0021, + 0x001d, 0x001c, 0x0019, 0x002b, 0x0027, 0x003f, 0x0037, 0x005d, + 0x004c, 0x003b, 0x005d, 0x0048, 0x0036, 0x004b, 0x0032, 0x001d, + 0x0034, 0x0016, 0x002a, 0x0028, 0x0043, 0x0039, 0x005f, 0x004f, + 0x0048, 0x0039, 0x0059, 0x0045, 0x0031, 0x0042, 0x002e, 0x001b, + 0x004d, 0x0025, 0x0023, 0x0042, 0x003a, 0x0034, 0x005b, 0x004a, + 0x003e, 0x0030, 0x004f, 0x003f, 0x005a, 0x003e, 0x0028, 0x0026, + 0x007d, 0x0020, 0x003c, 0x0038, 0x0032, 0x005c, 0x004e, 0x0041, + 0x0037, 0x0057, 0x0047, 0x0033, 0x0049, 0x0033, 0x0046, 0x001e, + 0x006d, 0x0035, 0x0031, 0x005e, 0x0058, 0x004b, 0x0042, 0x007a, + 0x005b, 0x0049, 0x0038, 0x002a, 0x0040, 0x002c, 0x0015, 0x0019, + 0x005a, 0x002b, 0x0029, 0x004d, 0x0049, 0x003f, 0x0038, 0x005c, + 0x004d, 0x0042, 0x002f, 0x0043, 0x0030, 0x0035, 0x0024, 0x0014, + 0x0047, 0x0022, 0x0043, 0x003c, 0x003a, 0x0031, 0x0058, 0x004c, + 0x0043, 0x006a, 0x0047, 0x0036, 0x0026, 0x0027, 0x0017, 0x000f, + 0x006d, 0x0035, 0x0033, 0x002f, 0x005a, 0x0052, 0x003a, 0x0039, + 0x0030, 0x0048, 0x0039, 0x0029, 0x0017, 0x001b, 0x003e, 0x0009, + 0x0056, 0x002a, 0x0028, 0x0025, 0x0046, 0x0040, 0x0034, 0x002b, + 0x0046, 0x0037, 0x002a, 0x0019, 0x001d, 0x0012, 0x000b, 0x000b, + 0x0076, 0x0044, 0x001e, 0x0037, 0x0032, 0x002e, 0x004a, 0x0041, + 0x0031, 0x0027, 0x0018, 0x0010, 0x0016, 0x000d, 0x000e, 0x0007, + 0x005b, 0x002c, 0x0027, 0x0026, 0x0022, 0x003f, 0x0034, 0x002d, + 0x001f, 0x0034, 0x001c, 0x0013, 0x000e, 0x0008, 0x0009, 0x0003, + 0x007b, 0x003c, 0x003a, 0x0035, 0x002f, 0x002b, 0x0020, 0x0016, + 0x0025, 0x0018, 0x0011, 0x000c, 0x000f, 0x000a, 0x0002, 0x0001, + 0x0047, 0x0025, 0x0022, 0x001e, 0x001c, 0x0014, 0x0011, 0x001a, + 0x0015, 0x0010, 0x000a, 0x0006, 0x0008, 0x0006, 0x0002, 0x0000, +]; + +static immutable uint8_t[256] mp3_huffbits_15 = [ + 3, 4, 5, 7, 7, 8, 9, 9, + 9, 10, 10, 11, 11, 11, 12, 13, + 4, 3, 5, 6, 7, 7, 8, 8, + 8, 9, 9, 10, 10, 10, 11, 11, + 5, 5, 5, 6, 7, 7, 8, 8, + 8, 9, 9, 10, 10, 11, 11, 11, + 6, 6, 6, 7, 7, 8, 8, 9, + 9, 9, 10, 10, 10, 11, 11, 11, + 7, 6, 7, 7, 8, 8, 9, 9, + 9, 9, 10, 10, 10, 11, 11, 11, + 8, 7, 7, 8, 8, 8, 9, 9, + 9, 9, 10, 10, 11, 11, 11, 12, + 9, 7, 8, 8, 8, 9, 9, 9, + 9, 10, 10, 10, 11, 11, 12, 12, + 9, 8, 8, 9, 9, 9, 9, 10, + 10, 10, 10, 10, 11, 11, 11, 12, + 9, 8, 8, 9, 9, 9, 9, 10, + 10, 10, 10, 11, 11, 12, 12, 12, + 9, 8, 9, 9, 9, 9, 10, 10, + 10, 11, 11, 11, 11, 12, 12, 12, + 10, 9, 9, 9, 10, 10, 10, 10, + 10, 11, 11, 11, 11, 12, 13, 12, + 10, 9, 9, 9, 10, 10, 10, 10, + 11, 11, 11, 11, 12, 12, 12, 13, + 11, 10, 9, 10, 10, 10, 11, 11, + 11, 11, 11, 11, 12, 12, 13, 13, + 11, 10, 10, 10, 10, 11, 11, 11, + 11, 12, 12, 12, 12, 12, 13, 13, + 12, 11, 11, 11, 11, 11, 11, 11, + 12, 12, 12, 12, 13, 13, 12, 13, + 12, 11, 11, 11, 11, 11, 11, 12, + 12, 12, 12, 12, 13, 13, 13, 13, +]; + +static immutable uint16_t[256] mp3_huffcodes_16 = [ + 0x0001, 0x0005, 0x000e, 0x002c, 0x004a, 0x003f, 0x006e, 0x005d, + 0x00ac, 0x0095, 0x008a, 0x00f2, 0x00e1, 0x00c3, 0x0178, 0x0011, + 0x0003, 0x0004, 0x000c, 0x0014, 0x0023, 0x003e, 0x0035, 0x002f, + 0x0053, 0x004b, 0x0044, 0x0077, 0x00c9, 0x006b, 0x00cf, 0x0009, + 0x000f, 0x000d, 0x0017, 0x0026, 0x0043, 0x003a, 0x0067, 0x005a, + 0x00a1, 0x0048, 0x007f, 0x0075, 0x006e, 0x00d1, 0x00ce, 0x0010, + 0x002d, 0x0015, 0x0027, 0x0045, 0x0040, 0x0072, 0x0063, 0x0057, + 0x009e, 0x008c, 0x00fc, 0x00d4, 0x00c7, 0x0183, 0x016d, 0x001a, + 0x004b, 0x0024, 0x0044, 0x0041, 0x0073, 0x0065, 0x00b3, 0x00a4, + 0x009b, 0x0108, 0x00f6, 0x00e2, 0x018b, 0x017e, 0x016a, 0x0009, + 0x0042, 0x001e, 0x003b, 0x0038, 0x0066, 0x00b9, 0x00ad, 0x0109, + 0x008e, 0x00fd, 0x00e8, 0x0190, 0x0184, 0x017a, 0x01bd, 0x0010, + 0x006f, 0x0036, 0x0034, 0x0064, 0x00b8, 0x00b2, 0x00a0, 0x0085, + 0x0101, 0x00f4, 0x00e4, 0x00d9, 0x0181, 0x016e, 0x02cb, 0x000a, + 0x0062, 0x0030, 0x005b, 0x0058, 0x00a5, 0x009d, 0x0094, 0x0105, + 0x00f8, 0x0197, 0x018d, 0x0174, 0x017c, 0x0379, 0x0374, 0x0008, + 0x0055, 0x0054, 0x0051, 0x009f, 0x009c, 0x008f, 0x0104, 0x00f9, + 0x01ab, 0x0191, 0x0188, 0x017f, 0x02d7, 0x02c9, 0x02c4, 0x0007, + 0x009a, 0x004c, 0x0049, 0x008d, 0x0083, 0x0100, 0x00f5, 0x01aa, + 0x0196, 0x018a, 0x0180, 0x02df, 0x0167, 0x02c6, 0x0160, 0x000b, + 0x008b, 0x0081, 0x0043, 0x007d, 0x00f7, 0x00e9, 0x00e5, 0x00db, + 0x0189, 0x02e7, 0x02e1, 0x02d0, 0x0375, 0x0372, 0x01b7, 0x0004, + 0x00f3, 0x0078, 0x0076, 0x0073, 0x00e3, 0x00df, 0x018c, 0x02ea, + 0x02e6, 0x02e0, 0x02d1, 0x02c8, 0x02c2, 0x00df, 0x01b4, 0x0006, + 0x00ca, 0x00e0, 0x00de, 0x00da, 0x00d8, 0x0185, 0x0182, 0x017d, + 0x016c, 0x0378, 0x01bb, 0x02c3, 0x01b8, 0x01b5, 0x06c0, 0x0004, + 0x02eb, 0x00d3, 0x00d2, 0x00d0, 0x0172, 0x017b, 0x02de, 0x02d3, + 0x02ca, 0x06c7, 0x0373, 0x036d, 0x036c, 0x0d83, 0x0361, 0x0002, + 0x0179, 0x0171, 0x0066, 0x00bb, 0x02d6, 0x02d2, 0x0166, 0x02c7, + 0x02c5, 0x0362, 0x06c6, 0x0367, 0x0d82, 0x0366, 0x01b2, 0x0000, + 0x000c, 0x000a, 0x0007, 0x000b, 0x000a, 0x0011, 0x000b, 0x0009, + 0x000d, 0x000c, 0x000a, 0x0007, 0x0005, 0x0003, 0x0001, 0x0003, +]; + +static immutable uint8_t[256] mp3_huffbits_16 = [ + 1, 4, 6, 8, 9, 9, 10, 10, + 11, 11, 11, 12, 12, 12, 13, 9, + 3, 4, 6, 7, 8, 9, 9, 9, + 10, 10, 10, 11, 12, 11, 12, 8, + 6, 6, 7, 8, 9, 9, 10, 10, + 11, 10, 11, 11, 11, 12, 12, 9, + 8, 7, 8, 9, 9, 10, 10, 10, + 11, 11, 12, 12, 12, 13, 13, 10, + 9, 8, 9, 9, 10, 10, 11, 11, + 11, 12, 12, 12, 13, 13, 13, 9, + 9, 8, 9, 9, 10, 11, 11, 12, + 11, 12, 12, 13, 13, 13, 14, 10, + 10, 9, 9, 10, 11, 11, 11, 11, + 12, 12, 12, 12, 13, 13, 14, 10, + 10, 9, 10, 10, 11, 11, 11, 12, + 12, 13, 13, 13, 13, 15, 15, 10, + 10, 10, 10, 11, 11, 11, 12, 12, + 13, 13, 13, 13, 14, 14, 14, 10, + 11, 10, 10, 11, 11, 12, 12, 13, + 13, 13, 13, 14, 13, 14, 13, 11, + 11, 11, 10, 11, 12, 12, 12, 12, + 13, 14, 14, 14, 15, 15, 14, 10, + 12, 11, 11, 11, 12, 12, 13, 14, + 14, 14, 14, 14, 14, 13, 14, 11, + 12, 12, 12, 12, 12, 13, 13, 13, + 13, 15, 14, 14, 14, 14, 16, 11, + 14, 12, 12, 12, 13, 13, 14, 14, + 14, 16, 15, 15, 15, 17, 15, 11, + 13, 13, 11, 12, 14, 14, 13, 14, + 14, 15, 16, 15, 17, 15, 14, 11, + 9, 8, 8, 9, 9, 10, 10, 10, + 11, 11, 11, 11, 11, 11, 11, 8, +]; + +static immutable uint16_t[256] mp3_huffcodes_24 = [ + 0x000f, 0x000d, 0x002e, 0x0050, 0x0092, 0x0106, 0x00f8, 0x01b2, + 0x01aa, 0x029d, 0x028d, 0x0289, 0x026d, 0x0205, 0x0408, 0x0058, + 0x000e, 0x000c, 0x0015, 0x0026, 0x0047, 0x0082, 0x007a, 0x00d8, + 0x00d1, 0x00c6, 0x0147, 0x0159, 0x013f, 0x0129, 0x0117, 0x002a, + 0x002f, 0x0016, 0x0029, 0x004a, 0x0044, 0x0080, 0x0078, 0x00dd, + 0x00cf, 0x00c2, 0x00b6, 0x0154, 0x013b, 0x0127, 0x021d, 0x0012, + 0x0051, 0x0027, 0x004b, 0x0046, 0x0086, 0x007d, 0x0074, 0x00dc, + 0x00cc, 0x00be, 0x00b2, 0x0145, 0x0137, 0x0125, 0x010f, 0x0010, + 0x0093, 0x0048, 0x0045, 0x0087, 0x007f, 0x0076, 0x0070, 0x00d2, + 0x00c8, 0x00bc, 0x0160, 0x0143, 0x0132, 0x011d, 0x021c, 0x000e, + 0x0107, 0x0042, 0x0081, 0x007e, 0x0077, 0x0072, 0x00d6, 0x00ca, + 0x00c0, 0x00b4, 0x0155, 0x013d, 0x012d, 0x0119, 0x0106, 0x000c, + 0x00f9, 0x007b, 0x0079, 0x0075, 0x0071, 0x00d7, 0x00ce, 0x00c3, + 0x00b9, 0x015b, 0x014a, 0x0134, 0x0123, 0x0110, 0x0208, 0x000a, + 0x01b3, 0x0073, 0x006f, 0x006d, 0x00d3, 0x00cb, 0x00c4, 0x00bb, + 0x0161, 0x014c, 0x0139, 0x012a, 0x011b, 0x0213, 0x017d, 0x0011, + 0x01ab, 0x00d4, 0x00d0, 0x00cd, 0x00c9, 0x00c1, 0x00ba, 0x00b1, + 0x00a9, 0x0140, 0x012f, 0x011e, 0x010c, 0x0202, 0x0179, 0x0010, + 0x014f, 0x00c7, 0x00c5, 0x00bf, 0x00bd, 0x00b5, 0x00ae, 0x014d, + 0x0141, 0x0131, 0x0121, 0x0113, 0x0209, 0x017b, 0x0173, 0x000b, + 0x029c, 0x00b8, 0x00b7, 0x00b3, 0x00af, 0x0158, 0x014b, 0x013a, + 0x0130, 0x0122, 0x0115, 0x0212, 0x017f, 0x0175, 0x016e, 0x000a, + 0x028c, 0x015a, 0x00ab, 0x00a8, 0x00a4, 0x013e, 0x0135, 0x012b, + 0x011f, 0x0114, 0x0107, 0x0201, 0x0177, 0x0170, 0x016a, 0x0006, + 0x0288, 0x0142, 0x013c, 0x0138, 0x0133, 0x012e, 0x0124, 0x011c, + 0x010d, 0x0105, 0x0200, 0x0178, 0x0172, 0x016c, 0x0167, 0x0004, + 0x026c, 0x012c, 0x0128, 0x0126, 0x0120, 0x011a, 0x0111, 0x010a, + 0x0203, 0x017c, 0x0176, 0x0171, 0x016d, 0x0169, 0x0165, 0x0002, + 0x0409, 0x0118, 0x0116, 0x0112, 0x010b, 0x0108, 0x0103, 0x017e, + 0x017a, 0x0174, 0x016f, 0x016b, 0x0168, 0x0166, 0x0164, 0x0000, + 0x002b, 0x0014, 0x0013, 0x0011, 0x000f, 0x000d, 0x000b, 0x0009, + 0x0007, 0x0006, 0x0004, 0x0007, 0x0005, 0x0003, 0x0001, 0x0003, +]; + +static immutable uint8_t[256] mp3_huffbits_24 = [ + 4, 4, 6, 7, 8, 9, 9, 10, + 10, 11, 11, 11, 11, 11, 12, 9, + 4, 4, 5, 6, 7, 8, 8, 9, + 9, 9, 10, 10, 10, 10, 10, 8, + 6, 5, 6, 7, 7, 8, 8, 9, + 9, 9, 9, 10, 10, 10, 11, 7, + 7, 6, 7, 7, 8, 8, 8, 9, + 9, 9, 9, 10, 10, 10, 10, 7, + 8, 7, 7, 8, 8, 8, 8, 9, + 9, 9, 10, 10, 10, 10, 11, 7, + 9, 7, 8, 8, 8, 8, 9, 9, + 9, 9, 10, 10, 10, 10, 10, 7, + 9, 8, 8, 8, 8, 9, 9, 9, + 9, 10, 10, 10, 10, 10, 11, 7, + 10, 8, 8, 8, 9, 9, 9, 9, + 10, 10, 10, 10, 10, 11, 11, 8, + 10, 9, 9, 9, 9, 9, 9, 9, + 9, 10, 10, 10, 10, 11, 11, 8, + 10, 9, 9, 9, 9, 9, 9, 10, + 10, 10, 10, 10, 11, 11, 11, 8, + 11, 9, 9, 9, 9, 10, 10, 10, + 10, 10, 10, 11, 11, 11, 11, 8, + 11, 10, 9, 9, 9, 10, 10, 10, + 10, 10, 10, 11, 11, 11, 11, 8, + 11, 10, 10, 10, 10, 10, 10, 10, + 10, 10, 11, 11, 11, 11, 11, 8, + 11, 10, 10, 10, 10, 10, 10, 10, + 11, 11, 11, 11, 11, 11, 11, 8, + 12, 10, 10, 10, 10, 10, 10, 11, + 11, 11, 11, 11, 11, 11, 11, 8, + 8, 7, 7, 7, 7, 7, 7, 7, + 7, 7, 7, 8, 8, 8, 8, 4, +]; + +static immutable huff_table_t[16] mp3_huff_tables = [ +huff_table_t( 1, null, null ), +huff_table_t( 2, mp3_huffbits_1.ptr, mp3_huffcodes_1.ptr ), +huff_table_t( 3, mp3_huffbits_2.ptr, mp3_huffcodes_2.ptr ), +huff_table_t( 3, mp3_huffbits_3.ptr, mp3_huffcodes_3.ptr ), +huff_table_t( 4, mp3_huffbits_5.ptr, mp3_huffcodes_5.ptr ), +huff_table_t( 4, mp3_huffbits_6.ptr, mp3_huffcodes_6.ptr ), +huff_table_t( 6, mp3_huffbits_7.ptr, mp3_huffcodes_7.ptr ), +huff_table_t( 6, mp3_huffbits_8.ptr, mp3_huffcodes_8.ptr ), +huff_table_t( 6, mp3_huffbits_9.ptr, mp3_huffcodes_9.ptr ), +huff_table_t( 8, mp3_huffbits_10.ptr, mp3_huffcodes_10.ptr ), +huff_table_t( 8, mp3_huffbits_11.ptr, mp3_huffcodes_11.ptr ), +huff_table_t( 8, mp3_huffbits_12.ptr, mp3_huffcodes_12.ptr ), +huff_table_t( 16, mp3_huffbits_13.ptr, mp3_huffcodes_13.ptr ), +huff_table_t( 16, mp3_huffbits_15.ptr, mp3_huffcodes_15.ptr ), +huff_table_t( 16, mp3_huffbits_16.ptr, mp3_huffcodes_16.ptr ), +huff_table_t( 16, mp3_huffbits_24.ptr, mp3_huffcodes_24.ptr ), +]; + +static immutable uint8_t[2][32] mp3_huff_data = [ +[ 0, 0 ], +[ 1, 0 ], +[ 2, 0 ], +[ 3, 0 ], +[ 0, 0 ], +[ 4, 0 ], +[ 5, 0 ], +[ 6, 0 ], +[ 7, 0 ], +[ 8, 0 ], +[ 9, 0 ], +[ 10, 0 ], +[ 11, 0 ], +[ 12, 0 ], +[ 0, 0 ], +[ 13, 0 ], +[ 14, 1 ], +[ 14, 2 ], +[ 14, 3 ], +[ 14, 4 ], +[ 14, 6 ], +[ 14, 8 ], +[ 14, 10 ], +[ 14, 13 ], +[ 15, 4 ], +[ 15, 5 ], +[ 15, 6 ], +[ 15, 7 ], +[ 15, 8 ], +[ 15, 9 ], +[ 15, 11 ], +[ 15, 13 ], +]; + +static immutable uint8_t[16][2] mp3_quad_codes = [ + [ 1, 5, 4, 5, 6, 5, 4, 4, 7, 3, 6, 0, 7, 2, 3, 1, ], + [ 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0, ], +]; + +static immutable uint8_t[16][2] mp3_quad_bits = [ + [ 1, 4, 4, 5, 4, 6, 5, 6, 4, 5, 5, 6, 5, 6, 6, 6, ], + [ 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, ], +]; + +static immutable uint8_t[22][9] band_size_long = [ +[ 4, 4, 4, 4, 4, 4, 6, 6, 8, 8, 10, + 12, 16, 20, 24, 28, 34, 42, 50, 54, 76, 158, ], /* 44100 */ +[ 4, 4, 4, 4, 4, 4, 6, 6, 6, 8, 10, + 12, 16, 18, 22, 28, 34, 40, 46, 54, 54, 192, ], /* 48000 */ +[ 4, 4, 4, 4, 4, 4, 6, 6, 8, 10, 12, + 16, 20, 24, 30, 38, 46, 56, 68, 84, 102, 26, ], /* 32000 */ +[ 6, 6, 6, 6, 6, 6, 8, 10, 12, 14, 16, + 20, 24, 28, 32, 38, 46, 52, 60, 68, 58, 54, ], /* 22050 */ +[ 6, 6, 6, 6, 6, 6, 8, 10, 12, 14, 16, + 18, 22, 26, 32, 38, 46, 52, 64, 70, 76, 36, ], /* 24000 */ +[ 6, 6, 6, 6, 6, 6, 8, 10, 12, 14, 16, + 20, 24, 28, 32, 38, 46, 52, 60, 68, 58, 54, ], /* 16000 */ +[ 6, 6, 6, 6, 6, 6, 8, 10, 12, 14, 16, + 20, 24, 28, 32, 38, 46, 52, 60, 68, 58, 54, ], /* 11025 */ +[ 6, 6, 6, 6, 6, 6, 8, 10, 12, 14, 16, + 20, 24, 28, 32, 38, 46, 52, 60, 68, 58, 54, ], /* 12000 */ +[ 12, 12, 12, 12, 12, 12, 16, 20, 24, 28, 32, + 40, 48, 56, 64, 76, 90, 2, 2, 2, 2, 2, ], /* 8000 */ +]; + +static immutable uint8_t[13][9] band_size_short = [ +[ 4, 4, 4, 4, 6, 8, 10, 12, 14, 18, 22, 30, 56, ], /* 44100 */ +[ 4, 4, 4, 4, 6, 6, 10, 12, 14, 16, 20, 26, 66, ], /* 48000 */ +[ 4, 4, 4, 4, 6, 8, 12, 16, 20, 26, 34, 42, 12, ], /* 32000 */ +[ 4, 4, 4, 6, 6, 8, 10, 14, 18, 26, 32, 42, 18, ], /* 22050 */ +[ 4, 4, 4, 6, 8, 10, 12, 14, 18, 24, 32, 44, 12, ], /* 24000 */ +[ 4, 4, 4, 6, 8, 10, 12, 14, 18, 24, 30, 40, 18, ], /* 16000 */ +[ 4, 4, 4, 6, 8, 10, 12, 14, 18, 24, 30, 40, 18, ], /* 11025 */ +[ 4, 4, 4, 6, 8, 10, 12, 14, 18, 24, 30, 40, 18, ], /* 12000 */ +[ 8, 8, 8, 12, 16, 20, 24, 28, 36, 2, 2, 2, 26, ], /* 8000 */ +]; + +static immutable uint8_t[22][2] mp3_pretab = [ + [ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 ], + [ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 2, 2, 3, 3, 3, 2, 0 ], +]; + +static immutable float[8] ci_table = [ + -0.6f, -0.535f, -0.33f, -0.185f, -0.095f, -0.041f, -0.0142f, -0.0037f, +]; + +enum C1 = FIXHR!(0.98480775301220805936/2); +enum C2 = FIXHR!(0.93969262078590838405/2); +enum C3 = FIXHR!(0.86602540378443864676/2); +enum C4 = FIXHR!(0.76604444311897803520/2); +enum C5 = FIXHR!(0.64278760968653932632/2); +enum C6 = FIXHR!(0.5/2); +enum C7 = FIXHR!(0.34202014332566873304/2); +enum C8 = FIXHR!(0.17364817766693034885/2); + +static immutable int[9] icos36 = [ + FIXR!(0.50190991877167369479), + FIXR!(0.51763809020504152469), //0 + FIXR!(0.55168895948124587824), + FIXR!(0.61038729438072803416), + FIXR!(0.70710678118654752439), //1 + FIXR!(0.87172339781054900991), + FIXR!(1.18310079157624925896), + FIXR!(1.93185165257813657349), //2 + FIXR!(5.73685662283492756461), +]; + +static immutable int[9] icos36h = [ + FIXHR!(0.50190991877167369479/2), + FIXHR!(0.51763809020504152469/2), //0 + FIXHR!(0.55168895948124587824/2), + FIXHR!(0.61038729438072803416/2), + FIXHR!(0.70710678118654752439/2), //1 + FIXHR!(0.87172339781054900991/2), + FIXHR!(1.18310079157624925896/4), + FIXHR!(1.93185165257813657349/4), //2 + //FIXHR!(5.73685662283492756461), +]; + +//////////////////////////////////////////////////////////////////////////////// + +int unaligned32_be (const(uint8_t)* p) { + static if (__VERSION__ > 2067) pragma(inline, true); + return (((p[0]<<8) | p[1])<<16) | (p[2]<<8) | (p[3]); +} + +enum MIN_CACHE_BITS = 25; + +enum NEG_SSR32(string a, string s) = "((cast( int32_t)("~a~"))>>(32-("~s~")))"; +enum NEG_USR32(string a, string s) = "((cast(uint32_t)("~a~"))>>(32-("~s~")))"; + +enum OPEN_READER(string name, string gb) = + "int "~name~"_index = ("~gb~").index;\n"~ + "int "~name~"_cache = 0;\n"; + +enum CLOSE_READER(string name, string gb) = "("~gb~").index = "~name~"_index;"; + +enum UPDATE_CACHE(string name, string gb) = name~"_cache = unaligned32_be(&(("~gb~").buffer["~name~"_index>>3])) << ("~name~"_index&0x07);"; + +enum SKIP_CACHE(string name, string gb, string num) = name~"_cache <<= ("~num~");"; + +enum SKIP_COUNTER(string name, string gb, string num) = name~"_index += ("~num~");"; + +enum SKIP_BITS(string name, string gb, string num) = "{"~SKIP_CACHE!(name, gb, num)~SKIP_COUNTER!(name, gb, num)~"}"; + +enum LAST_SKIP_BITS(string name, string gb, string num) = SKIP_COUNTER!(name, gb, num); +enum LAST_SKIP_CACHE(string name, string gb, string num) = "{}"; + +enum SHOW_UBITS(string name, string gb, string num) = NEG_USR32!(name~"_cache", num); + +enum SHOW_SBITS(string name, string gb, string num) = NEG_SSR32(name~"_cache", num); + +enum GET_CACHE(string name, string gb) = "(cast(uint32_t)"~name~"_cache)"; + +int get_bits_count() (const(bitstream_t)* s) { static if (__VERSION__ > 2067) pragma(inline, true); return s.index; } + +void skip_bits_long (bitstream_t* s, int n) { static if (__VERSION__ > 2067) pragma(inline, true); s.index += n; } +//#define skip_bits skip_bits_long +alias skip_bits = skip_bits_long; + +void init_get_bits (bitstream_t* s, const(uint8_t)* buffer, int bit_size) { + int buffer_size = (bit_size+7)>>3; + if (buffer_size < 0 || bit_size < 0) { + buffer_size = bit_size = 0; + buffer = null; + } + s.buffer = buffer; + s.size_in_bits = bit_size; + s.buffer_end = buffer + buffer_size; + s.index = 0; +} + +uint get_bits (bitstream_t* s, int n){ + int tmp; + mixin(OPEN_READER!("re", "s")); + mixin(UPDATE_CACHE!("re", "s")); + tmp = mixin(SHOW_UBITS!("re", "s", "n")); + mixin(LAST_SKIP_BITS!("re", "s", "n")); + mixin(CLOSE_READER!("re", "s")); + return tmp; +} + +int get_bitsz (bitstream_t* s, int n) { + static if (__VERSION__ > 2067) pragma(inline, true); + return (n == 0 ? 0 : get_bits(s, n)); +} + +uint get_bits1 (bitstream_t* s) { + int index = s.index; + uint8_t result = s.buffer[index>>3]; + result <<= (index&0x07); + result >>= 8 - 1; + ++index; + s.index = index; + return result; +} + +void align_get_bits (bitstream_t* s) { + int n = (-get_bits_count(s)) & 7; + if (n) skip_bits(s, n); +} + +enum GET_DATA(string v, string table, string i, string wrap, string size) = +"{\n"~ +" const(uint8_t)* ptr = cast(const(uint8_t)*)"~table~"+"~i~"*"~wrap~";\n"~ +" switch ("~size~") {\n"~ +" case 1: "~v~" = *cast(const(uint8_t)*)ptr; break;\n"~ +" case 2: "~v~" = *cast(const(uint16_t)*)ptr; break;\n"~ +" default: "~v~" = *cast(const(uint32_t)*)ptr; break;\n"~ +" }\n"~ +"}\n"~ +""; + +int alloc_table (vlc_t* vlc, int size) { + int index; + index = vlc.table_size; + vlc.table_size += size; + if (vlc.table_size > vlc.table_allocated) { + vlc.table_allocated += (1 << vlc.bits); + vlc.table = cast(VT2*)libc_realloc(vlc.table, VT2.sizeof * vlc.table_allocated); + if (!vlc.table) return -1; + } + return index; +} + + +int build_table ( + vlc_t* vlc, int table_nb_bits, + int nb_codes, + const(void)* bits, int bits_wrap, int bits_size, + const(void)* codes, int codes_wrap, int codes_size, + uint32_t code_prefix, int n_prefix +) { + int i, j, k, n, table_size, table_index, nb, n1, index, code_prefix2; + uint32_t code; + //VLC_TYPE (*table)[2]; + VT2* table; + + table_size = 1 << table_nb_bits; + table_index = alloc_table(vlc, table_size); + if (table_index < 0) return -1; + table = &vlc.table[table_index]; + + for (i = 0; i < table_size; i++) { + table[i][1] = 0; //bits + table[i][0] = -1; //codes + } + + for (i = 0; i < nb_codes; i++) { + mixin(GET_DATA!("n", "bits", "i", "bits_wrap", "bits_size")); + mixin(GET_DATA!("code", "codes", "i", "codes_wrap", "codes_size")); + if (n <= 0) continue; + n -= n_prefix; + code_prefix2 = code >> n; + if (n > 0 && code_prefix2 == code_prefix) { + if (n <= table_nb_bits) { + j = (code << (table_nb_bits - n)) & (table_size - 1); + nb = 1 << (table_nb_bits - n); + for(k=0;k> n) & ((1 << table_nb_bits) - 1); + n1 = -cast(int)table[j][1]; //bits + if (n > n1) + n1 = n; + table[j][1] = cast(short)(-n1); //bits + } + } + } + for(i=0;i table_nb_bits) { + n = table_nb_bits; + table[i][1] = cast(short)(-n); //bits + } + index = build_table(vlc, n, nb_codes, + bits, bits_wrap, bits_size, + codes, codes_wrap, codes_size, + (code_prefix << table_nb_bits) | i, + n_prefix + table_nb_bits); + if (index < 0) + return -1; + table = &vlc.table[table_index]; + table[i][0] = cast(short)index; //code + } + } + return table_index; +} + +int init_vlc( + vlc_t *vlc, int nb_bits, int nb_codes, + const void *bits, int bits_wrap, int bits_size, + const void *codes, int codes_wrap, int codes_size +) { + vlc.bits = nb_bits; + if (build_table(vlc, nb_bits, nb_codes, + bits, bits_wrap, bits_size, + codes, codes_wrap, codes_size, + 0, 0) < 0) { + libc_free(vlc.table); + return -1; + } + return 0; +} + +enum GET_VLC(string code, string name, string gb, string table, string bits, string max_depth) = +"{\n"~ +" int n, index, nb_bits;\n"~ +"\n"~ +" index= "~SHOW_UBITS!(name, gb, bits)~";\n"~ +" "~code~" = "~table~"[index][0];\n"~ +" n = "~table~"[index][1];\n"~ +"\n"~ +" if ("~max_depth~" > 1 && n < 0){\n"~ +" "~LAST_SKIP_BITS!(name, gb, bits)~"\n"~ +" "~UPDATE_CACHE!(name, gb)~"\n"~ +"\n"~ +" nb_bits = -n;\n"~ +"\n"~ +" index= "~SHOW_UBITS!(name, gb, "nb_bits")~" + "~code~";\n"~ +" "~code~" = "~table~"[index][0];\n"~ +" n = "~table~"[index][1];\n"~ +" if ("~max_depth~" > 2 && n < 0){\n"~ +" "~LAST_SKIP_BITS!(name, gb, "nb_bits")~"\n"~ +" "~UPDATE_CACHE!(name, gb)~"\n"~ +"\n"~ +" nb_bits = -n;\n"~ +"\n"~ +" index= "~SHOW_UBITS!(name, gb, "nb_bits")~" + "~code~";\n"~ +" "~code~" = "~table~"[index][0];\n"~ +" n = "~table~"[index][1];\n"~ +" }\n"~ +" }\n"~ +" "~SKIP_BITS!(name, gb, "n")~"\n"~ +"}\n"~ +""; + +int get_vlc2(bitstream_t *s, VT2* table, int bits, int max_depth) { + int code; + + mixin(OPEN_READER!("re", "s")); + mixin(UPDATE_CACHE!("re", "s")); + + mixin(GET_VLC!("code", "re", "s", "table", "bits", "max_depth")); + + mixin(CLOSE_READER!("re", "s")); + return code; +} + +void switch_buffer (mp3_context_t *s, int *pos, int *end_pos, int *end_pos2) { + if(s.in_gb.buffer && *pos >= s.gb.size_in_bits){ + s.gb= s.in_gb; + s.in_gb.buffer=null; + skip_bits_long(&s.gb, *pos - *end_pos); + *end_pos2= + *end_pos= *end_pos2 + get_bits_count(&s.gb) - *pos; + *pos= get_bits_count(&s.gb); + } +} + + +// ////////////////////////////////////////////////////////////////////////// // +int mp3_check_header(uint32_t header){ + //pragma(inline, true); + /* header */ + if ((header & 0xffe00000) != 0xffe00000) return -1; + /* layer check */ + if ((header & (3<<17)) != (1 << 17)) return -1; + /* bit rate */ + if ((header & (0xf<<12)) == 0xf<<12) return -1; + /* frequency */ + if ((header & (3<<10)) == 3<<10) return -1; + return 0; +} + + +void lsf_sf_expand (int *slen, int sf, int n1, int n2, int n3) { + if (n3) { + slen[3] = sf % n3; + sf /= n3; + } else { + slen[3] = 0; + } + if (n2) { + slen[2] = sf % n2; + sf /= n2; + } else { + slen[2] = 0; + } + slen[1] = sf % n1; + sf /= n1; + slen[0] = sf; +} + +int l3_unscale(int value, int exponent) +{ + uint m; + int e; + + e = table_4_3_exp [4*value + (exponent&3)]; + m = table_4_3_value[4*value + (exponent&3)]; + e -= (exponent >> 2); + if (e > 31) + return 0; + m = (m + (1 << (e-1))) >> e; + + return m; +} + +int round_sample(int *sum) { + int sum1; + sum1 = (*sum) >> OUT_SHIFT; + *sum &= (1< OUT_MAX) + sum1 = OUT_MAX; + return sum1; +} + +void exponents_from_scale_factors (mp3_context_t *s, granule_t *g, int16_t *exponents) { + const(uint8_t)* bstab, pretab; + int len, i, j, k, l, v0, shift, gain; + int[3] gains; + int16_t *exp_ptr; + + exp_ptr = exponents; + gain = g.global_gain - 210; + shift = g.scalefac_scale + 1; + + bstab = band_size_long[s.sample_rate_index].ptr; + pretab = mp3_pretab[g.preflag].ptr; + for(i=0;i0;j--) + *exp_ptr++ = cast(short)v0; + } + + if (g.short_start < 13) { + bstab = band_size_short[s.sample_rate_index].ptr; + gains[0] = gain - (g.subblock_gain[0] << 3); + gains[1] = gain - (g.subblock_gain[1] << 3); + gains[2] = gain - (g.subblock_gain[2] << 3); + k = g.long_end; + for(i=g.short_start;i<13;i++) { + len = bstab[i]; + for(l=0;l<3;l++) { + v0 = gains[l] - (g.scale_factors[k++] << shift) + 400; + for(j=len;j>0;j--) + *exp_ptr++ = cast(short)v0; + } + } + } +} + +void reorder_block(mp3_context_t *s, granule_t *g) +{ + int i, j, len; + int32_t* ptr, dst, ptr1; + int32_t[576] tmp; + + if (g.block_type != 2) + return; + + if (g.switch_point) { + if (s.sample_rate_index != 8) { + ptr = g.sb_hybrid.ptr + 36; + } else { + ptr = g.sb_hybrid.ptr + 48; + } + } else { + ptr = g.sb_hybrid.ptr; + } + + for(i=g.short_start;i<13;i++) { + len = band_size_short[s.sample_rate_index][i]; + ptr1 = ptr; + dst = tmp.ptr; + for(j=len;j>0;j--) { + *dst++ = ptr[0*len]; + *dst++ = ptr[1*len]; + *dst++ = ptr[2*len]; + ptr++; + } + ptr+=2*len; + libc_memcpy(ptr1, tmp.ptr, len * 3 * (*ptr1).sizeof); + } +} + +void compute_antialias(mp3_context_t *s, granule_t *g) { + enum INT_AA(string j) = + "tmp0 = ptr[-1-"~j~"];\n"~ + "tmp1 = ptr[ "~j~"];\n"~ + "tmp2= cast(int)MULH(tmp0 + tmp1, csa[0+4*"~j~"]);\n"~ + "ptr[-1-"~j~"] = cast(int)(4*(tmp2 - MULH(tmp1, csa[2+4*"~j~"])));\n"~ + "ptr[ "~j~"] = cast(int)(4*(tmp2 + MULH(tmp0, csa[3+4*"~j~"])));\n"; + + int32_t* ptr, csa; + int n, i; + + /* we antialias only "long" bands */ + if (g.block_type == 2) { + if (!g.switch_point) + return; + /* XXX: check this for 8000Hz case */ + n = 1; + } else { + n = SBLIMIT - 1; + } + + ptr = g.sb_hybrid.ptr + 18; + for(i = n;i > 0;i--) { + int tmp0, tmp1, tmp2; + csa = &csa_table[0][0]; + + mixin(INT_AA!("0")); + mixin(INT_AA!("1")); + mixin(INT_AA!("2")); + mixin(INT_AA!("3")); + mixin(INT_AA!("4")); + mixin(INT_AA!("5")); + mixin(INT_AA!("6")); + mixin(INT_AA!("7")); + + ptr += 18; + } +} + +void compute_stereo (mp3_context_t *s, granule_t *g0, granule_t *g1) { + int i, j, k, l; + int32_t v1, v2; + int sf_max, tmp0, tmp1, sf, len, non_zero_found; + int32_t[16]* is_tab; + int32_t* tab0, tab1; + int[3] non_zero_found_short; + + if (s.mode_ext & MODE_EXT_I_STEREO) { + if (!s.lsf) { + is_tab = is_table.ptr; + sf_max = 7; + } else { + is_tab = is_table_lsf[g1.scalefac_compress & 1].ptr; + sf_max = 16; + } + + tab0 = g0.sb_hybrid.ptr + 576; + tab1 = g1.sb_hybrid.ptr + 576; + + non_zero_found_short[0] = 0; + non_zero_found_short[1] = 0; + non_zero_found_short[2] = 0; + k = (13 - g1.short_start) * 3 + g1.long_end - 3; + for(i = 12;i >= g1.short_start;i--) { + /* for last band, use previous scale factor */ + if (i != 11) + k -= 3; + len = band_size_short[s.sample_rate_index][i]; + for(l=2;l>=0;l--) { + tab0 -= len; + tab1 -= len; + if (!non_zero_found_short[l]) { + /* test if non zero band. if so, stop doing i-stereo */ + for(j=0;j= sf_max) + goto found1; + + v1 = is_tab[0][sf]; + v2 = is_tab[1][sf]; + for(j=0;j= 0;i--) { + len = band_size_long[s.sample_rate_index][i]; + tab0 -= len; + tab1 -= len; + /* test if non zero band. if so, stop doing i-stereo */ + if (!non_zero_found) { + for(j=0;j= sf_max) + goto found2; + v1 = is_tab[0][sf]; + v2 = is_tab[1][sf]; + for(j=0;j0;j--) { + int exponent, x, y, v; + int pos= get_bits_count(&s.gb); + + if (pos >= end_pos){ + switch_buffer(s, &pos, &end_pos, &end_pos2); + if(pos >= end_pos) + break; + } + y = get_vlc2(&s.gb, vlc.table, 7, 3); + + if(!y){ + g.sb_hybrid[s_index ] = + g.sb_hybrid[s_index+1] = 0; + s_index += 2; + continue; + } + + exponent= exponents[s_index]; + + if(y&16){ + x = y >> 5; + y = y & 0x0f; + if (x < 15){ + v = expval_table[ exponent ][ x ]; + }else{ + x += get_bitsz(&s.gb, linbits); + v = l3_unscale(x, exponent); + } + if (get_bits1(&s.gb)) + v = -v; + g.sb_hybrid[s_index] = v; + if (y < 15){ + v = expval_table[ exponent ][ y ]; + }else{ + y += get_bitsz(&s.gb, linbits); + v = l3_unscale(y, exponent); + } + if (get_bits1(&s.gb)) + v = -v; + g.sb_hybrid[s_index+1] = v; + }else{ + x = y >> 5; + y = y & 0x0f; + x += y; + if (x < 15){ + v = expval_table[ exponent ][ x ]; + }else{ + x += get_bitsz(&s.gb, linbits); + v = l3_unscale(x, exponent); + } + if (get_bits1(&s.gb)) + v = -v; + g.sb_hybrid[s_index+!!y] = v; + g.sb_hybrid[s_index+ !y] = 0; + } + s_index+=2; + } + } + + /* high frequencies */ + vlc = &huff_quad_vlc[g.count1table_select]; + last_pos=0; + while (s_index <= 572) { + int pos, code; + pos = get_bits_count(&s.gb); + if (pos >= end_pos) { + if (pos > end_pos2 && last_pos){ + /* some encoders generate an incorrect size for this + part. We must go back into the data */ + s_index -= 4; + skip_bits_long(&s.gb, last_pos - pos); + break; + } + switch_buffer(s, &pos, &end_pos, &end_pos2); + if(pos >= end_pos) + break; + } + last_pos= pos; + + code = get_vlc2(&s.gb, vlc.table, vlc.bits, 1); + g.sb_hybrid[s_index+0]= + g.sb_hybrid[s_index+1]= + g.sb_hybrid[s_index+2]= + g.sb_hybrid[s_index+3]= 0; + while(code){ + static immutable int[16] idxtab = [3,3,2,2,1,1,1,1,0,0,0,0,0,0,0,0]; + int v; + int pos_= s_index+idxtab[code]; + code ^= 8>>idxtab[code]; + v = exp_table[ exponents[pos_] ]; + if(get_bits1(&s.gb)) + v = -v; + g.sb_hybrid[pos_] = v; + } + s_index+=4; + } + /* + if (s_index >= g.sb_hybrid.length) { + import core.stdc.stdio : printf; + printf("s_index=%u; len=%u; len=%u\n", cast(uint)s_index, cast(uint)g.sb_hybrid.length, cast(uint)((g.sb_hybrid[0]).sizeof*(576 - s_index))); + assert(0); + } + */ + if ((g.sb_hybrid[0]).sizeof*(576 - s_index) > 0) { + libc_memset(&g.sb_hybrid[s_index], 0, (g.sb_hybrid[0]).sizeof*(576 - s_index)); + } + + /* skip extension bits */ + bits_left = end_pos2 - get_bits_count(&s.gb); + if (bits_left < 0) { + return -1; + } + skip_bits_long(&s.gb, bits_left); + + i= get_bits_count(&s.gb); + switch_buffer(s, &i, &end_pos, &end_pos2); + + return 0; +} + + +// ////////////////////////////////////////////////////////////////////////// // +void imdct12(int *out_, int *in_) +{ + int in0, in1, in2, in3, in4, in5, t1, t2; + + in0= in_[0*3]; + in1= in_[1*3] + in_[0*3]; + in2= in_[2*3] + in_[1*3]; + in3= in_[3*3] + in_[2*3]; + in4= in_[4*3] + in_[3*3]; + in5= in_[5*3] + in_[4*3]; + in5 += in3; + in3 += in1; + + in2= cast(int)MULH(2*in2, C3); + in3= cast(int)MULH(4*in3, C3); + + t1 = in0 - in4; + t2 = cast(int)MULH(2*(in1 - in5), icos36h[4]); + + out_[ 7]= + out_[10]= t1 + t2; + out_[ 1]= + out_[ 4]= t1 - t2; + + in0 += in4>>1; + in4 = in0 + in2; + in5 += 2*in1; + in1 = cast(int)MULH(in5 + in3, icos36h[1]); + out_[ 8]= + out_[ 9]= in4 + in1; + out_[ 2]= + out_[ 3]= in4 - in1; + + in0 -= in2; + in5 = cast(int)MULH(2*(in5 - in3), icos36h[7]); + out_[ 0]= + out_[ 5]= in0 - in5; + out_[ 6]= + out_[11]= in0 + in5; +} + +void imdct36(int *out_, int *buf, int *in_, int *win) +{ + int i, j, t0, t1, t2, t3, s0, s1, s2, s3; + int[18] tmp; + int* tmp1, in1; + + for(i=17;i>=1;i--) + in_[i] += in_[i-1]; + for(i=17;i>=3;i-=2) + in_[i] += in_[i-2]; + + for(j=0;j<2;j++) { + tmp1 = tmp.ptr + j; + in1 = in_ + j; + t2 = in1[2*4] + in1[2*8] - in1[2*2]; + + t3 = in1[2*0] + (in1[2*6]>>1); + t1 = in1[2*0] - in1[2*6]; + tmp1[ 6] = t1 - (t2>>1); + tmp1[16] = t1 + t2; + + t0 = cast(int)MULH(2*(in1[2*2] + in1[2*4]), C2); + t1 = cast(int)MULH( in1[2*4] - in1[2*8] , -2*C8); + t2 = cast(int)MULH(2*(in1[2*2] + in1[2*8]), -C4); + + tmp1[10] = t3 - t0 - t2; + tmp1[ 2] = t3 + t0 + t1; + tmp1[14] = t3 + t2 - t1; + + tmp1[ 4] = cast(int)MULH(2*(in1[2*5] + in1[2*7] - in1[2*1]), -C3); + t2 = cast(int)MULH(2*(in1[2*1] + in1[2*5]), C1); + t3 = cast(int)MULH( in1[2*5] - in1[2*7] , -2*C7); + t0 = cast(int)MULH(2*in1[2*3], C3); + + t1 = cast(int)MULH(2*(in1[2*1] + in1[2*7]), -C5); + + tmp1[ 0] = t2 + t3 + t0; + tmp1[12] = t2 + t1 - t0; + tmp1[ 8] = t3 - t1 - t0; + } + + i = 0; + for(j=0;j<4;j++) { + t0 = tmp[i]; + t1 = tmp[i + 2]; + s0 = t1 + t0; + s2 = t1 - t0; + + t2 = tmp[i + 1]; + t3 = tmp[i + 3]; + s1 = cast(int)MULH(2*(t3 + t2), icos36h[j]); + s3 = cast(int)MULL(t3 - t2, icos36[8 - j]); + + t0 = s0 + s1; + t1 = s0 - s1; + out_[(9 + j)*SBLIMIT] = cast(int)MULH(t1, win[9 + j]) + buf[9 + j]; + out_[(8 - j)*SBLIMIT] = cast(int)MULH(t1, win[8 - j]) + buf[8 - j]; + buf[9 + j] = cast(int)MULH(t0, win[18 + 9 + j]); + buf[8 - j] = cast(int)MULH(t0, win[18 + 8 - j]); + + t0 = s2 + s3; + t1 = s2 - s3; + out_[(9 + 8 - j)*SBLIMIT] = cast(int)MULH(t1, win[9 + 8 - j]) + buf[9 + 8 - j]; + out_[( j)*SBLIMIT] = cast(int)MULH(t1, win[ j]) + buf[ j]; + buf[9 + 8 - j] = cast(int)MULH(t0, win[18 + 9 + 8 - j]); + buf[ + j] = cast(int)MULH(t0, win[18 + j]); + i += 4; + } + + s0 = tmp[16]; + s1 = cast(int)MULH(2*tmp[17], icos36h[4]); + t0 = s0 + s1; + t1 = s0 - s1; + out_[(9 + 4)*SBLIMIT] = cast(int)MULH(t1, win[9 + 4]) + buf[9 + 4]; + out_[(8 - 4)*SBLIMIT] = cast(int)MULH(t1, win[8 - 4]) + buf[8 - 4]; + buf[9 + 4] = cast(int)MULH(t0, win[18 + 9 + 4]); + buf[8 - 4] = cast(int)MULH(t0, win[18 + 8 - 4]); +} + +void compute_imdct (mp3_context_t *s, granule_t *g, int32_t *sb_samples, int32_t *mdct_buf) { + int32_t* ptr, win, win1, buf, out_ptr, ptr1; + int32_t[12] out2; + int i, j, mdct_long_end, v, sblimit; + + /* find last non zero block */ + ptr = g.sb_hybrid.ptr + 576; + ptr1 = g.sb_hybrid.ptr + 2 * 18; + while (ptr >= ptr1) { + ptr -= 6; + v = ptr[0] | ptr[1] | ptr[2] | ptr[3] | ptr[4] | ptr[5]; + if (v != 0) + break; + } + sblimit = cast(int)((ptr - g.sb_hybrid.ptr) / 18) + 1; + + if (g.block_type == 2) { + /* XXX: check for 8000 Hz */ + if (g.switch_point) + mdct_long_end = 2; + else + mdct_long_end = 0; + } else { + mdct_long_end = sblimit; + } + + buf = mdct_buf; + ptr = g.sb_hybrid.ptr; + for(j=0;j 32767) + v = 32767; + else if (v < -32768) + v = -32768; + synth_buf[j] = cast(short)v; + } + /* copy to avoid wrap */ + libc_memcpy(synth_buf + 512, synth_buf, 32 * int16_t.sizeof); + + samples2 = samples + 31 * incr; + w = window; + w2 = window + 31; + + sum = *dither_state; + p = synth_buf + 16; + mixin(SUM8!("sum", "+=", "w", "p")); + p = synth_buf + 48; + mixin(SUM8!("sum", "-=", "w + 32", "p")); + *samples = cast(short)round_sample(&sum); + samples += incr; + w++; + + /* we calculate two samples at the same time to avoid one memory + access per two sample */ + for(j=1;j<16;j++) { + sum2 = 0; + p = synth_buf + 16 + j; + mixin(SUM8P2!("sum", "+=", "sum2", "-=", "w", "w2", "p")); + p = synth_buf + 48 - j; + mixin(SUM8P2!("sum", "-=", "sum2", "-=", "w + 32", "w2 + 32", "p")); + + *samples = cast(short)round_sample(&sum); + samples += incr; + sum += sum2; + *samples2 = cast(short)round_sample(&sum); + samples2 -= incr; + w++; + w2--; + } + + p = synth_buf + 32; + mixin(SUM8!("sum", "-=", "w + 32", "p")); + *samples = cast(short)round_sample(&sum); + *dither_state= sum; + + offset = (offset - 32) & 511; + *synth_buf_offset = offset; +} + + +// ////////////////////////////////////////////////////////////////////////// // +int decode_header(mp3_context_t *s, uint32_t header) { + static immutable short[4][4] sampleCount = [ + [0, 576, 1152, 384], // v2.5 + [0, 0, 0, 0], // reserved + [0, 576, 1152, 384], // v2 + [0, 1152, 1152, 384], // v1 + ]; + ubyte mpid = (header>>19)&0x03; + ubyte layer = (header>>17)&0x03; + + s.sample_count = sampleCount.ptr[mpid].ptr[layer]; + + int sample_rate, frame_size, mpeg25, padding; + int sample_rate_index, bitrate_index; + if (header & (1<<20)) { + s.lsf = (header & (1<<19)) ? 0 : 1; + mpeg25 = 0; + } else { + s.lsf = 1; + mpeg25 = 1; + } + + sample_rate_index = (header >> 10) & 3; + sample_rate = mp3_freq_tab[sample_rate_index] >> (s.lsf + mpeg25); + sample_rate_index += 3 * (s.lsf + mpeg25); + s.sample_rate_index = sample_rate_index; + s.error_protection = ((header >> 16) & 1) ^ 1; + s.sample_rate = sample_rate; + + bitrate_index = (header >> 12) & 0xf; + padding = (header >> 9) & 1; + s.mode = (header >> 6) & 3; + s.mode_ext = (header >> 4) & 3; + s.nb_channels = (s.mode == MP3_MONO) ? 1 : 2; + + if (bitrate_index != 0) { + frame_size = mp3_bitrate_tab[s.lsf][bitrate_index]; + s.bit_rate = frame_size * 1000; + s.frame_size = (frame_size * 144000) / (sample_rate << s.lsf) + padding; + } else { + /* if no frame size computed, signal it */ + return 1; + } + return 0; +} + +int mp_decode_layer3(mp3_context_t *s) { + int nb_granules, main_data_begin, private_bits; + int gr, ch, blocksplit_flag, i, j, k, n, bits_pos; + granule_t *g; + static granule_t[2][2] granules; + static int16_t[576] exponents; + const(uint8_t)* ptr; + + if (s.lsf) { + main_data_begin = get_bits(&s.gb, 8); + private_bits = get_bits(&s.gb, s.nb_channels); + nb_granules = 1; + } else { + main_data_begin = get_bits(&s.gb, 9); + if (s.nb_channels == 2) + private_bits = get_bits(&s.gb, 3); + else + private_bits = get_bits(&s.gb, 5); + nb_granules = 2; + for(ch=0;ch> 1; + l = region_address1 + region_address2 + 2; + /* should not overflow */ + if (l > 22) + l = 22; + g.region_size[1] = + band_index_long[s.sample_rate_index][l] >> 1; + } + /* convert region offsets to region sizes and truncate + size to big_values */ + g.region_size[2] = (576 / 2); + j = 0; + for(i=0;i<3;i++) { + k = g.region_size[i]; + if (g.big_values < k) k = g.big_values; + g.region_size[i] = k - j; + j = k; + } + + /* compute band indexes */ + if (g.block_type == 2) { + if (g.switch_point) { + /* if switched mode, we handle the 36 first samples as + long blocks. For 8000Hz, we handle the 48 first + exponents as long blocks (XXX: check this!) */ + if (s.sample_rate_index <= 2) + g.long_end = 8; + else if (s.sample_rate_index != 8) + g.long_end = 6; + else + g.long_end = 4; /* 8000 Hz */ + + g.short_start = 2 + (s.sample_rate_index != 8); + } else { + g.long_end = 0; + g.short_start = 0; + } + } else { + g.short_start = 13; + g.long_end = 22; + } + + g.preflag = 0; + if (!s.lsf) + g.preflag = get_bits(&s.gb, 1); + g.scalefac_scale = cast(ubyte)get_bits(&s.gb, 1); + g.count1table_select = cast(ubyte)get_bits(&s.gb, 1); + } + } + + ptr = s.gb.buffer + (get_bits_count(&s.gb)>>3); + /* now we get bits from the main_data_begin offset */ + if(main_data_begin > s.last_buf_size){ + s.last_buf_size= main_data_begin; + } + + libc_memcpy(s.last_buf.ptr + s.last_buf_size, ptr, EXTRABYTES); + s.in_gb= s.gb; + init_get_bits(&s.gb, s.last_buf.ptr + s.last_buf_size - main_data_begin, main_data_begin*8); + + for(gr=0;gr> k)) == 0) { + slen = (k < 2) ? slen1 : slen2; + if(slen){ + for(i=0;i>= 1; + if (sf < 180) { + lsf_sf_expand(slen.ptr, sf, 6, 6, 0); + tindex2 = 3; + } else if (sf < 244) { + lsf_sf_expand(slen.ptr, sf - 180, 4, 4, 0); + tindex2 = 4; + } else { + lsf_sf_expand(slen.ptr, sf - 244, 3, 0, 0); + tindex2 = 5; + } + } else { + /* normal case */ + if (sf < 400) { + lsf_sf_expand(slen.ptr, sf, 5, 4, 4); + tindex2 = 0; + } else if (sf < 500) { + lsf_sf_expand(slen.ptr, sf - 400, 5, 4, 0); + tindex2 = 1; + } else { + lsf_sf_expand(slen.ptr, sf - 500, 3, 0, 0); + tindex2 = 2; + g.preflag = 1; + } + } + + j = 0; + for(k=0;k<4;k++) { + n = lsf_nsf_table[tindex2][tindex][k]; + sl = slen[k]; + if(sl){ + for(i=0;i>3; + if(i >= 0 && i <= BACKSTEP_SIZE){ + libc_memmove(s.last_buf.ptr, s.gb.buffer + (get_bits_count(&s.gb)>>3), i); + s.last_buf_size=i; + } + s.gb= s.in_gb; + } + + align_get_bits(&s.gb); + i= (s.gb.size_in_bits - get_bits_count(&s.gb))>>3; + + if(i<0 || i > BACKSTEP_SIZE || nb_frames<0){ + i = buf_size - HEADER_SIZE; + if (BACKSTEP_SIZE < i) i = BACKSTEP_SIZE; + } + libc_memcpy(s.last_buf.ptr + s.last_buf_size, s.gb.buffer + buf_size - HEADER_SIZE - i, i); + s.last_buf_size += i; + + /* apply the synthesis filter */ + for(ch=0;ch> (16 - WFRAC_BITS); + } + window[i] = cast(short)v; + if ((i & 63) != 0) + v = -v; + if (i != 0) + window[512 - i] = cast(short)v; + } + + /* huffman decode tables */ + for(i=1;i<16;i++) { + const huff_table_t *h = &mp3_huff_tables[i]; + int xsize, x, y; + uint n; + uint8_t[512] tmp_bits; + uint16_t[512] tmp_codes; + + libc_memset(tmp_bits.ptr, 0, tmp_bits.sizeof); + libc_memset(tmp_codes.ptr, 0, tmp_codes.sizeof); + + xsize = h.xsize; + n = xsize * xsize; + + j = 0; + for(x=0;x>4); + double f= libc_pow(i&15, 4.0 / 3.0) * libc_pow(2, (exponent-400)*0.25 + FRAC_BITS + 5); + expval_table[exponent][i&15]= cast(uint)f; + if((i&15)==1) + exp_table[exponent]= cast(uint)f; + } + + for(i=0;i<7;i++) { + float f; + int v; + if (i != 6) { + f = tan(cast(double)i * M_PI / 12.0); + v = FIXRx(f / (1.0 + f)); + } else { + v = FIXR!(1.0); + } + is_table[0][i] = v; + is_table[1][6 - i] = v; + } + for(i=7;i<16;i++) + is_table[0][i] = is_table[1][i] = cast(int)0.0; + + for(i=0;i<16;i++) { + double f; + int e, k_; + + for(j=0;j<2;j++) { + e = -(j + 1) * ((i + 1) >> 1); + f = libc_pow(2.0, e / 4.0); + k_ = i & 1; + is_table_lsf[j][k_ ^ 1][i] = FIXRx(f); + is_table_lsf[j][k_][i] = FIXR!(1.0); + } + } + + for(i=0;i<8;i++) { + float ci, cs, ca; + ci = ci_table[i]; + cs = 1.0 / sqrt(1.0 + ci * ci); + ca = cs * ci; + csa_table[i][0] = FIXHRx(cs/4); + csa_table[i][1] = FIXHRx(ca/4); + csa_table[i][2] = FIXHRx(ca/4) + FIXHRx(cs/4); + csa_table[i][3] = FIXHRx(ca/4) - FIXHRx(cs/4); + csa_table_float[i][0] = cs; + csa_table_float[i][1] = ca; + csa_table_float[i][2] = ca + cs; + csa_table_float[i][3] = ca - cs; + } + + /* compute mdct windows */ + for(i=0;i<36;i++) { + for(j=0; j<4; j++){ + double d; + + if(j==2 && i%3 != 1) + continue; + + d= sin(M_PI * (i + 0.5) / 36.0); + if(j==1){ + if (i>=30) d= 0; + else if(i>=24) d= sin(M_PI * (i - 18 + 0.5) / 12.0); + else if(i>=18) d= 1; + }else if(j==3){ + if (i< 6) d= 0; + else if(i< 12) d= sin(M_PI * (i - 6 + 0.5) / 12.0); + else if(i< 18) d= 1; + } + d*= 0.5 / cos(M_PI*(2*i + 19)/72); + if(j==2) + mdct_win[j][i/3] = FIXHRx((d / (1<<5))); + else + mdct_win[j][i ] = FIXHRx((d / (1<<5))); + } + } + for(j=0;j<4;j++) { + for(i=0;i<36;i+=2) { + mdct_win[j + 4][i] = mdct_win[j][i]; + mdct_win[j + 4][i + 1] = -mdct_win[j][i + 1]; + } + } + //init = 1; + } + return 0; +} + +int mp3_decode_frame (mp3_context_t *s, int16_t *out_samples, int *data_size, uint8_t *buf, int buf_size) { + uint32_t header; + int out_size; + int extra_bytes = 0; + +retry: + if (buf_size < HEADER_SIZE) return -1; + + header = (buf[0]<<24)|(buf[1]<<16)|(buf[2]<<8)|buf[3]; + if (mp3_check_header(header) < 0){ + ++buf; + --buf_size; + ++extra_bytes; + goto retry; + } + + if (s.last_header && (header&0xffff0c00u) != s.last_header) { + ++buf; + --buf_size; + ++extra_bytes; + goto retry; + } + + if (decode_header(s, header) == 1) { + s.frame_size = -1; + return -1; + } + + if (s.frame_size<=0 || s.frame_size > buf_size) return -1; // incomplete frame + if (s.frame_size < buf_size) buf_size = s.frame_size; + + out_size = mp3_decode_main(s, out_samples, buf, buf_size); + if (out_size >= 0) { + *data_size = out_size; + s.last_header = header&0xffff0c00u; + } + // else: Error while decoding MPEG audio frame. + s.frame_size += extra_bytes; + return buf_size; +} + + +/+ +int mp3_skip_frame (mp3_context_t *s, uint8_t *buf, int buf_size) { + uint32_t header; + int out_size; + int extra_bytes = 0; + +retry: + if (buf_size < HEADER_SIZE) return -1; + + header = (buf[0] << 24) | (buf[1] << 16) | (buf[2] << 8) | buf[3]; + if (mp3_check_header(header) < 0) { + ++buf; + --buf_size; + ++extra_bytes; + goto retry; + } + + if (s.last_header && (header&0xffff0c00u) != s.last_header) { + ++buf; + --buf_size; + ++extra_bytes; + goto retry; + } + + if (decode_header(s, header) == 1) { + s.frame_size = -1; + return -1; + } + + if (s.frame_size <= 0 || s.frame_size > buf_size) return -1; // incomplete frame + if (s.frame_size < buf_size) buf_size = s.frame_size; + s.last_header = header&0xffff0c00u; + s.frame_size += extra_bytes; + return buf_size; +} ++/ diff --git a/script.d b/script.d index c836618..01d0eb5 100644 --- a/script.d +++ b/script.d @@ -1,4 +1,7 @@ /* + + FIXME: i kinda do want a catch type filter. + REPL plan: easy movement to/from a real editor can edit a specific function diff --git a/simpleaudio.d b/simpleaudio.d index 63e8aab..3d1380c 100644 --- a/simpleaudio.d +++ b/simpleaudio.d @@ -251,6 +251,15 @@ struct AudioOutputThread { else static assert(0); } + void playOgg()(string filename, bool loop = false) { + if(impl) + impl.playOgg(filename, loop); + } + void playMp3()(string filename) { + if(impl) + impl.playMp3(filename); + } + /// provides automatic [arsd.jsvar] script wrapping capability. Make sure the /// script also finishes before this goes out of scope or it may end up talking @@ -436,6 +445,55 @@ final class AudioPcmOutThreadImplementation : Thread { ); } + /// Requires mp3.d to be compiled in (module arsd.mp3) which is LGPL licensed. + void playMp3()(string filename) { + import arsd.mp3; + + import std.stdio; + auto fi = File(filename); + scope auto reader = delegate(void[] buf) { + return cast(int) fi.rawRead(buf[]).length; + }; + + auto mp3 = new MP3Decoder(reader); + if(!mp3.valid) + throw new Exception("no file"); + + auto next = mp3.frameSamples; + + addChannel( + delegate bool(short[] buffer) { + if(cast(int) buffer.length != buffer.length) + throw new Exception("eeeek"); + + more: + if(next.length >= buffer.length) { + buffer[] = next[0 .. buffer.length]; + next = next[buffer.length .. $]; + } else { + buffer[0 .. next.length] = next[]; + buffer = buffer[next.length .. $]; + + next = next[$..$]; + + if(buffer.length) { + if(mp3.valid) { + mp3.decodeNextFrame(reader); + next = mp3.frameSamples; + goto more; + } else { + buffer[] = 0; + return false; + } + } + } + + return true; + } + ); + } + + struct Sample { int operation; @@ -503,6 +561,28 @@ final class AudioPcmOutThreadImplementation : Thread { break; /+ case 2: // triangle wave + + short[] tone; + tone.length = 22050 * len / 1000; + + short valmax = cast(short) (cast(int) volume * short.max / 100); + int wavelength = 22050 / freq; + wavelength /= 2; + int da = valmax / wavelength; + int val = 0; + + for(int a = 0; a < tone.length; a++){ + tone[a] = cast(short) val; + val+= da; + if(da > 0 && val >= valmax) + da *= -1; + if(da < 0 && val <= -valmax) + da *= -1; + } + + data ~= tone; + + for(; i < sampleFinish; i++) { buffer[i] = val; // left and right do the same thing so we only count @@ -519,7 +599,41 @@ final class AudioPcmOutThreadImplementation : Thread { break; case 3: // sawtooth wave + short[] tone; + tone.length = 22050 * len / 1000; + + int valmax = volume * short.max / 100; + int wavelength = 22050 / freq; + int da = valmax / wavelength; + short val = 0; + + for(int a = 0; a < tone.length; a++){ + tone[a] = val; + val+= da; + if(val >= valmax) + val = 0; + } + + data ~= tone; case 4: // sine wave + short[] tone; + tone.length = 22050 * len / 1000; + + int valmax = volume * short.max / 100; + int val = 0; + + float i = 2*PI / (22050/freq); + + float f = 0; + for(int a = 0; a < tone.length; a++){ + tone[a] = cast(short) (valmax * sin(f)); + f += i; + if(f>= 2*PI) + f -= 2*PI; + } + + data ~= tone; + +/ case 5: // custom function val = currentSample.f(currentSample.x); @@ -1175,6 +1289,29 @@ struct MidiOutputThread { void popSong() {} } +version(Posix) { + import core.sys.posix.signal; + private sigaction_t oldSigIntr; + void setSigIntHandler() { + sigaction_t n; + n.sa_handler = &interruptSignalHandler; + n.sa_mask = cast(sigset_t) 0; + n.sa_flags = 0; + sigaction(SIGINT, &n, &oldSigIntr); + } + void restoreSigIntHandler() { + sigaction(SIGINT, &oldSigIntr, null); + } + + __gshared bool interrupted; + + private + extern(C) + void interruptSignalHandler(int sigNumber) nothrow { + interrupted = true; + } +} + /// Gives MIDI output access. struct MidiOutput { version(ALSA) { @@ -1210,6 +1347,8 @@ struct MidiOutput { static immutable ubyte[3] resetSequence = [0x0b << 4, 123, 0]; // send a controller event to reset it writeRawMessageData(resetSequence[]); + static immutable ubyte[1] resetCmd = [0xff]; + writeRawMessageData(resetCmd[]); // and flush it immediately snd_rawmidi_drain(handle); } else version(WinMM) { @@ -1248,6 +1387,8 @@ struct MidiOutput { /// Timing and sending sane data is your responsibility! /// The data should NOT include any timestamp bytes - each midi message should be 2 or 3 bytes. void writeRawMessageData(scope const(ubyte)[] data) { + if(data.length == 0) + return; version(ALSA) { ssize_t written; diff --git a/simpledisplay.d b/simpledisplay.d index 65c0715..111866f 100644 --- a/simpledisplay.d +++ b/simpledisplay.d @@ -9,7 +9,12 @@ * send messages without a recipient window * setTimeout * setInterval +*/ +/* + Classic games I want to add: + * my tetris clone + * pac man */ /* @@ -75,6 +80,9 @@ `-Lgdi32.lib -Luser32.lib` on the build command. If you want the Windows subsystem too, use `-L/subsystem:windows -L/entry:mainCRTStartup`. + If using ldc instead of dmd, use `-L/entry:wmainCRTstartup` instead of `mainCRTStartup`; + note the "w". + On Win32, you can pass `-L/subsystem:windows` if you don't want a console to be automatically allocated. @@ -6913,8 +6921,8 @@ struct ScreenPainter { } //this function draws a circle with the drawEllipse() function above, it requires the upper left point and the radius - void drawCircle(Point upperLeft, int radius) { - drawEllipse(upperLeft, Point(upperLeft.x + radius, upperLeft.y + radius)); + void drawCircle(Point upperLeft, int diameter) { + drawEllipse(upperLeft, Point(upperLeft.x + diameter, upperLeft.y + diameter)); } /// . @@ -8184,6 +8192,7 @@ version(Windows) { GetObject(i.handle, bm.sizeof, &bm); + // or should I AlphaBlend!??!?! BitBlt(hdc, x, y, w /* bm.bmWidth */, /*bm.bmHeight*/ h, hdcMem, ix, iy, SRCCOPY); SelectObject(hdcMem, hbmOld); @@ -8198,6 +8207,7 @@ version(Windows) { GetObject(s.handle, bm.sizeof, &bm); + // or should I AlphaBlend!??!?! BitBlt(hdc, x, y, bm.bmWidth, bm.bmHeight, hdcMem, 0, 0, SRCCOPY); SelectObject(hdcMem, hbmOld); diff --git a/terminal.d b/terminal.d index 7762cac..ce057f6 100644 --- a/terminal.d +++ b/terminal.d @@ -751,6 +751,8 @@ struct Terminal { // almost always if(t.indexOf("xterm") != -1) t = "xterm"; + if(t.indexOf("putty") != -1) + t = "xterm"; if(t.indexOf("tmux") != -1) t = "tmux"; if(t.indexOf("screen") != -1) diff --git a/wav.d b/wav.d new file mode 100644 index 0000000..9936c39 --- /dev/null +++ b/wav.d @@ -0,0 +1,421 @@ +/++ + Basic .wav file reading and writing. + + History: + Written May 15, 2020, but loosely based on code I wrote a + long time ago, at least August 2008 which is the oldest + file I have generated from the original code. + + The old code could only write files, the reading support + was all added in 2020. ++/ +module arsd.wav; + +import core.stdc.stdio; + +/++ + ++/ +struct WavWriter { + private FILE* fp; + + /++ + Opens the file with the given header params. + + Make sure you pass the correct params to header, except, + if you have a seekable stream, the data length can be zero + and it will be fixed when you close. If you have a non-seekable + stream though, you must give the size up front. + + If you need to go to memory, the best way is to just + append your data to your own buffer, then create a [WavFileHeader] + separately and prepend it. Wav files are simple, aside from + the header and maybe a terminating byte (which isn't really important + anyway), there's nothing special going on. + + Throws: Exception on error from [open]. + + --- + auto writer = WavWriter("myfile.wav", WavFileHeader(44100, 2, 16)); + writer.write(shortSamples); + --- + +/ + this(string filename, WavFileHeader header) { + this.header = header; + + if(!open(filename)) + throw new Exception("Couldn't open file for writing"); // FIXME: errno + } + + /++ + `WavWriter(WavFileHeader(44100, 2, 16));` + +/ + this(WavFileHeader header) @nogc nothrow { + this.header = header; + } + + /++ + Calls [close]. Errors are ignored. + +/ + ~this() @nogc { + close(); + } + + @disable this(this); + + private uint size; + private WavFileHeader header; + + @nogc: + + /++ + Returns: true on success, false on error. Check errno for details. + +/ + bool open(string filename) { + assert(fp is null); + assert(filename.length < 290); + + char[300] fn; + fn[0 .. filename.length] = filename[]; + fn[filename.length] = 0; + + fp = fopen(fn.ptr, "wb"); + if(fp is null) + return false; + if(fwrite(&header, header.sizeof, 1, fp) != 1) + return false; + + return true; + } + + /++ + Writes 8-bit samples to the file. You must have constructed the object with an 8 bit header. + + Returns: true on success, false on error. Check errno for details. + +/ + bool write(ubyte[] data) { + assert(header.bitsPerSample == 8); + if(fp is null) + return false; + if(fwrite(data.ptr, 1, data.length, fp) != data.length) + return false; + size += data.length; + return true; + } + + /++ + Writes 16-bit samples to the file. You must have constructed the object with 16 bit header. + + Returns: true on success, false on error. Check errno for details. + +/ + bool write(short[] data) { + assert(header.bitsPerSample == 16); + if(fp is null) + return false; + if(fwrite(data.ptr, 2, data.length, fp) != data.length) + return false; + size += data.length * 2; + return true; + } + + /++ + Returns: true on success, false on error. Check errno for details. + +/ + bool close() { + if(fp is null) + return true; + + // pad odd sized file as required by spec... + if(size & 1) { + fputc(0, fp); + } + + if(!header.dataLength) { + // put the length back at the beginning of the file + if(fseek(fp, 0, SEEK_SET) != 0) + return false; + auto n = header.withDataLengthInBytes(size); + if(fwrite(&n, 1, n.sizeof, fp) != 1) + return false; + } else { + assert(header.dataLength == size); + } + if(fclose(fp)) + return false; + fp = null; + return true; + } +} + +version(LittleEndian) {} else static assert(0, "just needs endian conversion coded in but i was lazy"); + +align(1) +/// +struct WavFileHeader { + align(1): + const ubyte[4] header = ['R', 'I', 'F', 'F']; + int topSize; // dataLength + 36 + const ubyte[4] type = ['W', 'A', 'V', 'E']; + const ubyte[4] fmtHeader = ['f', 'm', 't', ' ']; + const int fmtHeaderSize = 16; + const ushort audioFormat = 1; // PCM + + ushort numberOfChannels; + uint sampleRate; + + uint bytesPerSeconds; // bytesPerSampleTimesChannels * sampleRate + ushort bytesPerSampleTimesChannels; // bitsPerSample * channels / 8 + + ushort bitsPerSample; // 16 + + const ubyte[4] dataHeader = ['d', 'a', 't', 'a']; + uint dataLength; + // data follows. put a 0 at the end if dataLength is odd. + + /// + this(uint sampleRate, ushort numberOfChannels, ushort bitsPerSample, uint dataLengthInBytes = 0) @nogc pure @safe nothrow { + assert(bitsPerSample == 8 || bitsPerSample == 16); + + this.numberOfChannels = numberOfChannels; + this.sampleRate = sampleRate; + this.bitsPerSample = bitsPerSample; + + this.bytesPerSampleTimesChannels = cast(ushort) (numberOfChannels * bitsPerSample / 8); + this.bytesPerSeconds = this.bytesPerSampleTimesChannels * sampleRate; + + this.topSize = dataLengthInBytes + 36; + this.dataLength = dataLengthInBytes; + } + + /// + WavFileHeader withDataLengthInBytes(int dataLengthInBytes) const @nogc pure @safe nothrow { + return WavFileHeader(sampleRate, numberOfChannels, bitsPerSample, dataLengthInBytes); + } +} +static assert(WavFileHeader.sizeof == 44); + + +/++ + After construction, the parameters are set and you can set them. + After that, you process the samples range-style. + + It ignores chunks in the file that aren't the basic standard. + It throws exceptions if it isn't a bare-basic PCM wav file. + + See [wavReader] for the convenience constructors. + + Note that if you are reading a 16 bit file (`bitsPerSample == 16`), + you'll actually need to `cast(short[]) front`. + + --- + auto reader = wavReader(data[]); + foreach(chunk; reader) + play(chunk); + --- ++/ +struct WavReader(Range) { + const ushort numberOfChannels; + const int sampleRate; + const ushort bitsPerSample; + int dataLength; // don't modify plz + + private uint remainingDataLength; + + private Range underlying; + + private const(ubyte)[] frontBuffer; + + static if(is(Range == CFileChunks)) { + this(FILE* fp) { + underlying = CFileChunks(fp); + this(0); + } + } else { + this(Range r) { + this.underlying = r; + this(0); + } + } + + private this(int _initializationDummyVariable) { + this.frontBuffer = underlying.front; + + WavFileHeader header; + ubyte[] headerBytes = (cast(ubyte*) &header)[0 .. header.sizeof - 8]; + + if(this.frontBuffer.length >= headerBytes.length) { + headerBytes[] = this.frontBuffer[0 .. headerBytes.length]; + this.frontBuffer = this.frontBuffer[headerBytes.length .. $]; + } else { + throw new Exception("Probably not a wav file, or else pass bigger chunks please"); + } + + if(header.header != ['R', 'I', 'F', 'F']) + throw new Exception("Not a wav file; no RIFF header"); + if(header.type != ['W', 'A', 'V', 'E']) + throw new Exception("Not a wav file"); + // so technically the spec does NOT require fmt to be the first chunk.. + // but im gonna just be lazy + if(header.fmtHeader != ['f', 'm', 't', ' ']) + throw new Exception("Malformed or unsupported wav file"); + + if(header.fmtHeaderSize < 16) + throw new Exception("Unsupported wav format header"); + + auto additionalSkip = header.fmtHeaderSize - 16; + + if(header.audioFormat != 1) + throw new Exception("arsd.wav only supports the most basic wav files and this one has advanced encoding. try converting to a .mp3 file and use arsd.mp3."); + + this.numberOfChannels = header.numberOfChannels; + this.sampleRate = header.sampleRate; + this.bitsPerSample = header.bitsPerSample; + + if(header.bytesPerSampleTimesChannels != header.bitsPerSample * header.numberOfChannels / 8) + throw new Exception("Malformed wav file: header.bytesPerSampleTimesChannels didn't match"); + if(header.bytesPerSeconds != header.bytesPerSampleTimesChannels * header.sampleRate) + throw new Exception("Malformed wav file: header.bytesPerSeconds didn't match"); + + this.frontBuffer = this.frontBuffer[additionalSkip .. $]; + + static struct ChunkHeader { + align(1): + ubyte[4] type; + uint size; + } + static assert(ChunkHeader.sizeof == 8); + + ChunkHeader current; + ubyte[] chunkHeader = (cast(ubyte*) ¤t)[0 .. current.sizeof]; + + another_chunk: + + // now we're at the next chunk. want to skip until we hit data. + if(this.frontBuffer.length < chunkHeader.length) + throw new Exception("bug in arsd.wav the chunk isn't big enough to handle and im lazy. if you hit this send me your file plz"); + + chunkHeader[] = frontBuffer[0 .. chunkHeader.length]; + frontBuffer = frontBuffer[chunkHeader.length .. $]; + + if(current.type != ['d', 'a', 't', 'a']) { + // skip unsupported chunk... + drop_more: + if(frontBuffer.length > current.size) { + frontBuffer = frontBuffer[current.size .. $]; + } else { + current.size -= frontBuffer.length; + underlying.popFront(); + if(underlying.empty) { + throw new Exception("Ran out of data while trying to read wav chunks"); + } else { + frontBuffer = underlying.front; + goto drop_more; + } + } + goto another_chunk; + } else { + this.remainingDataLength = current.size; + } + + this.dataLength = this.remainingDataLength; + } + + @property const(ubyte)[] front() { + return frontBuffer; + } + + version(none) + void consumeBytes(size_t count) { + if(this.frontBuffer.length) + this.frontBuffer = this.frontBuffer[count .. $]; + } + + void popFront() { + remainingDataLength -= front.length; + + underlying.popFront(); + if(underlying.empty) + frontBuffer = null; + else + frontBuffer = underlying.front; + } + + @property bool empty() { + return remainingDataLength == 0 || this.underlying.empty; + } +} + +/++ + Convenience constructor for [WavReader] + + To read from a file, pass a filename, a FILE*, or a range that + reads chunks from a file. + + To read from a memory block, just pass it a `ubyte[]` slice. ++/ +WavReader!T wavReader(T)(T t) { + return WavReader!T(t); +} + +/// ditto +WavReader!DataBlock wavReader(const(ubyte)[] data) { + return WavReader!DataBlock(DataBlock(data)); +} + +struct DataBlock { + const(ubyte)[] front; + bool empty() { return front.length == 0; } + void popFront() { front = null; } +} + +/// Construct a [WavReader] from a filename. +WavReader!CFileChunks wavReader(string filename) { + assert(filename.length < 290); + + char[300] fn; + fn[0 .. filename.length] = filename[]; + fn[filename.length] = 0; + + auto fp = fopen(fn.ptr, "rb"); + if(fp is null) + throw new Exception("wav file unopenable"); // FIXME details + + return WavReader!CFileChunks(fp); +} + +struct CFileChunks { + FILE* fp; + this(FILE* fp) { + this.fp = fp; + buffer = new ubyte[](4096); + refcount = new int; + *refcount = 1; + popFront(); + } + this(this) { + if(refcount !is null) + (*refcount) += 1; + } + ~this() { + if(refcount is null) return; + (*refcount) -= 1; + if(*refcount == 0) { + fclose(fp); + } + } + + //ubyte[4096] buffer; + ubyte[] buffer; + int* refcount; + + ubyte[] front; + + void popFront() { + auto got = fread(buffer.ptr, 1, buffer.length, fp); + front = buffer[0 .. got]; + } + + bool empty() { + return front.length == 0 && (feof(fp) ? true : false); + } +}