aboutsummaryrefslogtreecommitdiff
path: root/cyclone/hammer/seq.c
diff options
context:
space:
mode:
Diffstat (limited to 'cyclone/hammer/seq.c')
-rw-r--r--cyclone/hammer/seq.c107
1 files changed, 88 insertions, 19 deletions
diff --git a/cyclone/hammer/seq.c b/cyclone/hammer/seq.c
index 503b205..a9d58b6 100644
--- a/cyclone/hammer/seq.c
+++ b/cyclone/hammer/seq.c
@@ -38,6 +38,7 @@ typedef struct _seq
int x_playhead;
float x_tempo;
double x_prevtime;
+ double x_clockdelay;
unsigned char x_status;
int x_evesize;
int x_expectedsize;
@@ -179,7 +180,7 @@ static void seq_stoprecording(t_seq *x)
x->x_isrecording = 0;
}
-static void seq_tick(t_seq *x)
+static void seq_clocktick(t_seq *x)
{
if (x->x_isplaying)
{
@@ -209,7 +210,14 @@ nextevent:
bp = ep->e_bytes;
goto nextevent;
}
- else clock_delay(x->x_clock, ep->e_delta);
+ else
+ {
+ x->x_clockdelay = ep->e_delta * x->x_tempo;
+ if (x->x_clockdelay < 0.)
+ x->x_clockdelay = 0.;
+ clock_delay(x->x_clock, x->x_clockdelay);
+ x->x_prevtime = clock_getlogicaltime();
+ }
}
else
{
@@ -220,14 +228,26 @@ nextevent:
}
}
+static void seq_tick(t_seq *x)
+{
+ /* FIXME */
+}
+
/* CHECKED running status not used in playback */
static void seq_dostart(t_seq *x, float tempo)
{
if (x->x_isplaying)
{
/* CHECKED tempo change */
+ double elapsed = clock_gettimesince(x->x_prevtime);
+ double left = x->x_clockdelay - elapsed;
+ if (left < 0)
+ left = 0;
+ else
+ left *= tempo / x->x_tempo;
+ clock_delay(x->x_clock, x->x_clockdelay = left);
+ x->x_prevtime = clock_getlogicaltime();
x->x_tempo = tempo;
- /* FIXME update the clock */
}
else
{
@@ -241,14 +261,18 @@ static void seq_dostart(t_seq *x, float tempo)
x->x_isplaying = 1;
/* playback data never sent within the scheduler event of
a start message (even for the first delta <= 0), LATER rethink */
- clock_delay(x->x_clock, x->x_sequence->e_delta);
+ x->x_clockdelay = x->x_sequence->e_delta * tempo;
+ if (x->x_clockdelay < 0.)
+ x->x_clockdelay = 0.;
+ clock_delay(x->x_clock, x->x_clockdelay);
+ x->x_prevtime = clock_getlogicaltime();
}
}
}
static void seq_bang(t_seq *x)
{
- seq_dostart(x, 1.0);
+ seq_dostart(x, 1.);
}
static void seq_float(t_seq *x, t_float f)
@@ -290,12 +314,10 @@ static void seq_list(t_seq *x, t_symbol *s, int ac, t_atom *av)
/* CHECKED anything else/more silently ignored */
}
-static void seq_record(t_seq *x)
+static void seq_dorecord(t_seq *x)
{
- /* CHECKED 'record' resets recording */
- if (x->x_isplaying) /* CHECKED 'record' stops playback */
+ if (x->x_isplaying) /* CHECKED 'record' and 'append' stops playback */
seq_stopplayback(x);
- seq_doclear(x, 0);
x->x_isrecording = 1;
x->x_prevtime = clock_getlogicaltime();
x->x_status = 0;
@@ -303,17 +325,34 @@ static void seq_record(t_seq *x)
x->x_expectedsize = -1; /* LATER rethink */
}
+static void seq_record(t_seq *x)
+{
+ /* CHECKED 'record' resets recording */
+ seq_doclear(x, 0);
+ seq_dorecord(x);
+}
+
static void seq_append(t_seq *x)
{
- if (x->x_isrecording)
- return; /* CHECKME 'append' does not reset recording */
- if (x->x_isplaying) /* CHECKME 'append' stops playback */
- seq_stopplayback(x);
- x->x_isrecording = 1;
- x->x_prevtime = clock_getlogicaltime();
- x->x_status = 0;
- x->x_evesize = 0;
- x->x_expectedsize = -1; /* LATER rethink */
+ /* CHECKED if isrecording, 'append' resets the timer */
+ seq_dorecord(x);
+}
+
+static void seq_start(t_seq *x, t_floatarg f)
+{
+ if (f < 0)
+ {
+ /* FIXME */
+ }
+ else
+ {
+ float tempo = (f == 0 ? 1. : 1024. / f);
+ if (tempo < 1e-20)
+ tempo = 1e-20;
+ else if (tempo > 1e20)
+ tempo = 1e20;
+ seq_dostart(x, tempo);
+ }
}
static void seq_stop(t_seq *x)
@@ -324,6 +363,27 @@ static void seq_stop(t_seq *x)
seq_stoprecording(x);
}
+/* CHECKED first delta time is set permanently (it is stored in a file) */
+static void seq_delay(t_seq *x, t_floatarg f)
+{
+ if (x->x_nevents)
+ /* CHECKED signed/unsigned bug, not emulated */
+ x->x_sequence->e_delta = (f > 0 ? f : 0);
+}
+
+/* CHECKED all delta times are set permanently (they are stored in a file) */
+static void seq_hook(t_seq *x, t_floatarg f)
+{
+ int nevents;
+ if (nevents = x->x_nevents)
+ {
+ t_seqevent *ev = x->x_sequence;
+ if (f < 0)
+ f = 0; /* CHECKED signed/unsigned bug, not emulated */
+ while (nevents--) ev++->e_delta *= f;
+ }
+}
+
static int seq_dogrowing(t_seq *x, int nevents)
{
if (nevents > x->x_size)
@@ -780,6 +840,7 @@ static void *seq_new(t_symbol *s)
x->x_canvas = canvas_getcurrent();
x->x_filehandle = hammerfile_new((t_pd *)x, 0,
seq_readhook, seq_writehook, 0);
+ x->x_tempo = 1.;
x->x_prevtime = 0;
x->x_size = SEQ_INISIZE;
x->x_nevents = 0;
@@ -792,7 +853,7 @@ static void *seq_new(t_symbol *s)
seq_doread(x, s, 1);
}
else x->x_defname = &s_;
- x->x_clock = clock_new(x, (t_method)seq_tick);
+ x->x_clock = clock_new(x, (t_method)seq_clocktick);
return (x);
}
@@ -813,8 +874,16 @@ void seq_setup(void)
gensym("record"), 0);
class_addmethod(seq_class, (t_method)seq_append,
gensym("append"), 0);
+ class_addmethod(seq_class, (t_method)seq_start,
+ gensym("start"), A_DEFFLOAT, 0);
class_addmethod(seq_class, (t_method)seq_stop,
gensym("stop"), 0);
+ class_addmethod(seq_class, (t_method)seq_tick,
+ gensym("tick"), 0);
+ class_addmethod(seq_class, (t_method)seq_delay,
+ gensym("delay"), A_FLOAT, 0); /* CHECKED arg obligatory */
+ class_addmethod(seq_class, (t_method)seq_hook,
+ gensym("hook"), A_FLOAT, 0); /* CHECKED arg obligatory */
class_addmethod(seq_class, (t_method)seq_read,
gensym("read"), A_DEFSYM, 0);
class_addmethod(seq_class, (t_method)seq_write,