added midi player
This commit is contained in:
parent
2279c86091
commit
3b9b7a6ab8
2 changed files with 251 additions and 0 deletions
250
sys/src/games/midi.c
Normal file
250
sys/src/games/midi.c
Normal file
|
@ -0,0 +1,250 @@
|
|||
#include <u.h>
|
||||
#include <libc.h>
|
||||
|
||||
enum { SAMPLE = 44100 };
|
||||
|
||||
struct Tracker {
|
||||
uchar *data;
|
||||
char ended;
|
||||
uvlong t;
|
||||
uchar notes[16][128];
|
||||
int cmd;
|
||||
} *tr;
|
||||
|
||||
typedef struct Tracker Tracker;
|
||||
|
||||
int fd, ofd, div, tempo = 500000, ntrack;
|
||||
uvlong T;
|
||||
int freq[128];
|
||||
|
||||
void *
|
||||
emallocz(int size)
|
||||
{
|
||||
void *v;
|
||||
|
||||
v = malloc(size);
|
||||
if(v == nil)
|
||||
sysfatal("malloc: %r");
|
||||
memset(v, 0, size);
|
||||
return v;
|
||||
}
|
||||
|
||||
int
|
||||
get8(Tracker *src)
|
||||
{
|
||||
uchar c;
|
||||
|
||||
if(src == nil){
|
||||
if(read(fd, &c, 1) == 0)
|
||||
sysfatal("unexpected eof");
|
||||
return c;
|
||||
}
|
||||
return *src->data++;
|
||||
}
|
||||
|
||||
int
|
||||
get16(Tracker *src)
|
||||
{
|
||||
int x;
|
||||
|
||||
x = get8(src) << 8;
|
||||
return x | get8(src);
|
||||
}
|
||||
|
||||
int
|
||||
get32(Tracker *src)
|
||||
{
|
||||
int x;
|
||||
x = get16(src) << 16;
|
||||
return x | get16(src);
|
||||
}
|
||||
|
||||
int
|
||||
getvar(Tracker *src)
|
||||
{
|
||||
int k, x;
|
||||
|
||||
x = get8(src);
|
||||
k = x & 0x7F;
|
||||
while(x & 0x80){
|
||||
k <<= 7;
|
||||
x = get8(src);
|
||||
k |= x & 0x7F;
|
||||
}
|
||||
return k;
|
||||
}
|
||||
|
||||
int
|
||||
peekvar(Tracker *src)
|
||||
{
|
||||
uchar *p;
|
||||
int v;
|
||||
|
||||
p = src->data;
|
||||
v = getvar(src);
|
||||
src->data = p;
|
||||
return v;
|
||||
}
|
||||
|
||||
void
|
||||
skip(Tracker *src, int x)
|
||||
{
|
||||
if(x) do
|
||||
get8(src);
|
||||
while(--x);
|
||||
}
|
||||
|
||||
uvlong
|
||||
tconv(int n)
|
||||
{
|
||||
uvlong v;
|
||||
|
||||
v = n;
|
||||
v *= tempo;
|
||||
v *= SAMPLE;
|
||||
v /= div;
|
||||
v /= 1000000;
|
||||
return v;
|
||||
}
|
||||
|
||||
void
|
||||
run(uvlong n)
|
||||
{
|
||||
int samp, j, k, l, no[128];
|
||||
uchar *s;
|
||||
int t, f;
|
||||
short u;
|
||||
Tracker *x;
|
||||
|
||||
samp = n - T;
|
||||
if(samp <= 0)
|
||||
return;
|
||||
memset(no, 0, sizeof no);
|
||||
for(x = tr; x < tr + ntrack; x++){
|
||||
if(x->ended)
|
||||
continue;
|
||||
for(j = 0; j < 16; j++)
|
||||
for(k = 0; k < 128; k++)
|
||||
no[k] += x->notes[j][k];
|
||||
}
|
||||
s = emallocz(samp * 4);
|
||||
for(l = 0; l < samp; l++){
|
||||
t = 0;
|
||||
for(k = 0; k < 128; k++){
|
||||
f = (T % freq[k]) >= freq[k]/2 ? 1 : 0;
|
||||
t += f * no[k];
|
||||
}
|
||||
u = t*10;
|
||||
s[4 * l] = s[4 * l + 2] = u;
|
||||
s[4 * l + 1] = s[4 * l + 3] = u >> 8;
|
||||
T++;
|
||||
}
|
||||
write(ofd, s, samp * 4);
|
||||
free(s);
|
||||
}
|
||||
|
||||
void
|
||||
readevent(Tracker *src)
|
||||
{
|
||||
uvlong l;
|
||||
int n,t;
|
||||
|
||||
l = tconv(getvar(src));
|
||||
run(src->t += l);
|
||||
t = get8(src);
|
||||
if((t & 0x80) == 0){
|
||||
src->data--;
|
||||
t = src->cmd;
|
||||
if((t & 0x80) == 0)
|
||||
sysfatal("invalid midi");
|
||||
}else
|
||||
src->cmd = t;
|
||||
switch(t >> 4){
|
||||
case 0x8:
|
||||
n = get8(src);
|
||||
get8(src);
|
||||
src->notes[t & 15][n] = 0;
|
||||
break;
|
||||
case 0x9:
|
||||
n = get8(src);
|
||||
src->notes[t & 15][n] = get8(src);
|
||||
break;
|
||||
case 0xB:
|
||||
get16(src);
|
||||
break;
|
||||
case 0xC:
|
||||
get8(src);
|
||||
break;
|
||||
case 0xF:
|
||||
t = get8(src);
|
||||
n = get8(src);
|
||||
switch(t){
|
||||
case 0x2F:
|
||||
src->ended = 1;
|
||||
break;
|
||||
case 0x51:
|
||||
tempo = get16(src) << 8;
|
||||
tempo |= get8(src);
|
||||
break;
|
||||
case 5:
|
||||
write(1, src->data, n);
|
||||
skip(src, n);
|
||||
break;
|
||||
default:
|
||||
print("unknown meta event type %.2x\n", t);
|
||||
case 3: case 1: case 2: case 0x58: case 0x59: case 0x21:
|
||||
skip(src, n);
|
||||
}
|
||||
break;
|
||||
default:
|
||||
sysfatal("unknown event type %x", t>>4);
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
main(int argc, char **argv)
|
||||
{
|
||||
int i, size;
|
||||
uvlong t, mint;
|
||||
Tracker *x, *minx;
|
||||
|
||||
if(argc != 2)
|
||||
sysfatal("invalid arguments");
|
||||
fd = open(argv[1], OREAD);
|
||||
if(fd < 0)
|
||||
sysfatal("open: %r");
|
||||
ofd = open("/dev/audio", OWRITE);
|
||||
if(ofd < 0)
|
||||
sysfatal("ofd: %r");
|
||||
if(get32(nil) != 0x4D546864 || get32(nil) != 6)
|
||||
sysfatal("invalid file header");
|
||||
get16(nil);
|
||||
ntrack = get16(nil);
|
||||
div = get16(nil);
|
||||
tr = emallocz(ntrack * sizeof(*tr));
|
||||
for(i = 0; i < ntrack; i++){
|
||||
if(get32(nil) != 0x4D54726B)
|
||||
sysfatal("invalid track header");
|
||||
size = get32(nil);
|
||||
tr[i].data = emallocz(size);
|
||||
read(fd, tr[i].data, size);
|
||||
}
|
||||
for(i = 0; i < 128; i++)
|
||||
freq[i] = SAMPLE / (440 * pow(1.05946, i - 69));
|
||||
for(;;){
|
||||
minx = nil;
|
||||
mint = 0;
|
||||
for(x = tr; x < tr + ntrack; x++){
|
||||
if(x->ended)
|
||||
continue;
|
||||
t = tconv(peekvar(x)) + x->t;
|
||||
if(t < mint || minx == nil){
|
||||
mint = t;
|
||||
minx = x;
|
||||
}
|
||||
}
|
||||
if(minx == nil)
|
||||
exits(nil);
|
||||
readevent(minx);
|
||||
}
|
||||
}
|
|
@ -12,6 +12,7 @@ TARG=4s\
|
|||
glendy\
|
||||
packet\
|
||||
mandel\
|
||||
midi\
|
||||
|
||||
OFILES=
|
||||
HFILES=
|
||||
|
|
Loading…
Reference in a new issue