ref: 750f45295075db5ec6a1ffda1263d429a0d32b84
dir: /midi-dobend/
midi: implement pitch wheel this is an aborted attempt to implement pitch bending (event 0xe0). this maintains a list of active notes, storing note number, amplitude and bent frequency; this greatly reduces cpu time spent in run(), also alleviating the issue of delays between writes to /dev/audio being too great... it blows chunks: - greatly complexifies midi note handling - pitch steps are discrete and too noticeable, actually worsening playback (blatant example: doom e2m2), but don't yet know where T % f >= f/2 is coming from or how to fix this → i don't yet know what i'm doing nor how to solve this correctly diff -r 9c4b9c67ff16 sys/src/games/midi.c --- a/sys/src/games/midi.c Thu Feb 16 20:11:20 2017 +0100 +++ b/sys/src/games/midi.c Wed Feb 22 01:34:39 2017 +0200 @@ -3,19 +3,29 @@ enum { SAMPLE = 44100 }; +typedef struct Note Note; +typedef struct Tracker Tracker; + +struct Note { + int nt; + Tracker *src; + int ch; + uchar a; + int f; + int ½f; + Note *n; +} nl[256], np; + struct Tracker { uchar *data; char ended; uvlong t; - uchar notes[16][128]; + double bend[16]; int cmd; } *tr; -typedef struct Tracker Tracker; - int fd, ofd, div, tempo = 500000, ntrack; uvlong T; -int freq[128]; void * emallocz(int size) @@ -108,32 +118,23 @@ } void -run(uvlong n) +run(uvlong dt) { - int samp, j, k, l, no[128]; + int samp, l; uchar *s; - int t, f; + int t; short u; - Tracker *x; - - samp = n - T; + Note *n; + + samp = dt - 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]; - } + for(n = np.n; n != &np; n = n->n) + if(T % n->f >= n->½f) + t += n->a; u = t*10; s[4 * l] = s[4 * l + 2] = u; s[4 * l + 1] = s[4 * l + 3] = u >> 8; @@ -144,11 +145,70 @@ } void +bend(Tracker *src, int ch, int b) +{ + int f; + double d; + Note *n; + + d = (b - 8192) / 4095.5; + d = pow(2, d / 12); + src->bend[ch] = d; + for(n = np.n; n != &np; n = n->n){ + if(n->src != src || n->ch != ch) + continue; + f = SAMPLE / (440 * d * pow(1.05946, n->nt - 69)); + n->f = f; + n->½f = f / 2; + } +} + +void +pop(Tracker *src, int ch, int nt) +{ + Note *n, *p; + + for(p = &np, n = np.n; n != &np; p = n, n = n->n) + if(n->src == src && n->ch == ch && n->nt == nt){ + p->n = n->n; + memset(n, 0, sizeof *n); + return; + } +} + +void +push(Tracker *src, int ch, int nt, int a) +{ + int f; + Note *n, *p; + + for(p = &np, n = np.n; n != &np; p = n, n = n->n) + if(n->src == src && n->ch == ch && n->nt == nt){ + n->a = a; + return; + } + for(n = nl; n < nl + nelem(nl); n++) + if(n->src == nil) + break; + if(n == nl + nelem(nl)) + return; + n->src = src; + n->ch = ch; + n->nt = nt; + n->a = a; + f = SAMPLE / (440 * src->bend[ch] * pow(1.05946, nt - 69)); + n->f = f; + n->½f = f / 2; + p->n = n; + n->n = &np; +} + +void readevent(Tracker *src) { uvlong l; int n,t; - + l = tconv(getvar(src)); run(src->t += l); t = get8(src); @@ -161,13 +221,12 @@ src->cmd = t; switch(t >> 4){ case 0x8: - n = get8(src); + pop(src, t & 15, get8(src)); get8(src); - src->notes[t & 15][n] = 0; break; case 0x9: n = get8(src); - src->notes[t & 15][n] = get8(src); + push(src, t & 15, n, get8(src)); break; case 0xB: get16(src); @@ -175,6 +234,11 @@ case 0xC: get8(src); break; + case 0xE: + n = get8(src) & 0x7f; + n = get8(src) << 7 & 0x3f80 | n; + bend(src, t & 15, n); + break; case 0xF: t = get8(src); n = get8(src); @@ -204,7 +268,7 @@ void main(int argc, char **argv) { - int i, size; + int i, j, size; uvlong t, mint; Tracker *x, *minx; @@ -228,9 +292,10 @@ size = get32(nil); tr[i].data = emallocz(size); read(fd, tr[i].data, size); + for(j = 0; j < 16; j++) + tr[i].bend[j] = 1.0; } - for(i = 0; i < 128; i++) - freq[i] = SAMPLE / (440 * pow(1.05946, i - 69)); + np.n = &np; for(;;){ minx = nil; mint = 0;