ref: affae6de24bf5bb8c5d9f49aa5e9c970c5f2a36b
dir: /examples/moddec.c/
#include <dumb.h>
#pragma lib "./libdumb.a$O"
#define MAX(a,b) ((a)>(b)?(a):(b))
#define MIN(a,b) ((a)<(b)?(a):(b))
#define SRATE 44100.0f
enum {
Nsamp = 4096,
};
static uchar b[Nsamp*2*2];
static void
usage(void)
{
fprint(2, "usage: %s [FILE]\n", argv0);
exits("usage");
}
void
main(int argc, char **argv)
{
DUMB_IT_SIGRENDERER *itren;
DUH_SIGRENDERER *ren;
sample_t **samp;
char *data, *t;
int n, sz, r;
double pos;
long nsamp;
DUH *f;
setfcr(getfcr() & ~(FPINVAL|FPOVFL));
pos = 0.0;
ARGBEGIN{
case 's':
pos = atof(EARGF(usage()));
break;
default:
usage();
}ARGEND;
if(argc != 0)
usage();
sz = 32768;
data = nil;
for(n = 0;; n += r){
if(sz-n < 65536){
sz *= 2;
if((data = realloc(data, sz)) == nil)
sysfatal("memory");
}
if((r = read(0, data+n, sz-n)) < 0)
sysfatal("%r");
if(r == 0)
break;
}
if((f = dumb_read_any(dumbfile_open_memory(data, sz), 1, 0)) == nil)
sysfatal("unknown/invalid mod");
if((t = (char*)duh_get_tag(f, "TITLE")) != nil && *t)
fprint(2, "%s\n", t);
ren = duh_start_sigrenderer(f, 0, 2, 0);
itren = duh_get_it_sigrenderer(ren);
dumb_it_set_loop_callback(itren, dumb_it_callback_terminate, nil);
dumb_it_set_xm_speed_zero_callback(itren, dumb_it_callback_terminate, nil);
dumb_it_set_resampling_quality(itren, DUMB_RQ_CUBIC);
if(pos > 0.0)
fprint(2, "time: %g\n", pos);
n = 0;
for(;;){
if(pos > 0.0){
pos -= (double)n / SRATE;
n = MIN(pos*SRATE, Nsamp);
if(n < 1)
pos = 0.0;
}
memset(b, 0, sizeof(b));
n = duh_render_int(ren, &samp, &nsamp, 16, 0, 1.0, 65536.0f/SRATE, MAX(n, Nsamp), b);
if(n <= 0)
break;
if(pos <= 0.0 && write(1, b, n*2*2) != n*2*2)
break;
}
exits(nil);
}