Browse Source

Attempt to address interaction between jack buffer size and tempo.

tags/non-sequencer-v1.9.4
Jonathan Moore Liles 17 years ago
parent
commit
047a003df1
3 changed files with 49 additions and 28 deletions
  1. +17
    -9
      jack.C
  2. +31
    -18
      transport.C
  3. +1
    -1
      transport.H

+ 17
- 9
jack.C View File

@@ -48,7 +48,7 @@ int sample_rate;


const int MAX_PORT = 16; const int MAX_PORT = 16;


const int subticks_per_tick = 2048;
const int subticks_per_tick = 4096;


/* timers for notes on all channels and ports. When a note is played, /* timers for notes on all channels and ports. When a note is played,
* the respective value in this array is set to the note duraction in * the respective value in this array is set to the note duraction in
@@ -209,16 +209,16 @@ process ( jack_nframes_t nframes, void *arg )
static tick_t onph = 0; static tick_t onph = 0;
static int old_play_mode = PATTERN; static int old_play_mode = PATTERN;


::nframes = nframes;
static int not_dropped = 0;


// init all port buffers (maybe we should only do this as needed)
/* loop over stuff */
::nframes = nframes;


transport.nframes = nframes; transport.nframes = nframes;
transport.poll(); transport.poll();


tick_t ph = trunc( transport.ticks );
tick_t nph = trunc( transport.ticks + transport.ticks_per_period );
/* ph-nph is exclusive. It is important that in normal continuous playback each tick is covered exactly once! */
const tick_t ph = transport.ticks;
const tick_t nph = trunc( transport.ticks + transport.ticks_per_period );


if ( ! transport.valid ) if ( ! transport.valid )
goto schedule; goto schedule;
@@ -227,12 +227,20 @@ process ( jack_nframes_t nframes, void *arg )
goto schedule; goto schedule;


if ( ph != onph ) if ( ph != onph )
DWARNING( "dropped ticks" );
{
if ( onph > ph )
DWARNING( "duplicated %lu ticks (out of %d)", onph - ph, (int)(not_dropped * transport.ticks_per_period) );
else
DWARNING( "dropped %lu ticks (out of %d)", ph - onph, (int)(not_dropped * transport.ticks_per_period) );

not_dropped = 0;
}

++not_dropped;


onph = nph; onph = nph;


// MESSAGE( "tpp %f %f-%f", transport.ticks_per_period, ph, nph );
// MESSAGE( "tpp %f %lu-%lu", transport.ticks_per_period, ph, nph );
// DMESSAGE( "tpp %f %lu-%lu", transport.ticks_per_period, ph, nph );


switch ( old_play_mode ) switch ( old_play_mode )
{ {


+ 31
- 18
transport.C View File

@@ -37,20 +37,26 @@ static volatile bool _done;


/** callback for when we're Timebase Master, mostly taken from /** callback for when we're Timebase Master, mostly taken from
* transport.c in Jack's example clients. */ * transport.c in Jack's example clients. */
/* FIXME: there is a subtle interaction here between the tempo and
* JACK's buffer size. Inflating ticks_per_beat (as jack_transport
* does) diminishes the effect of this correlation, but does not
* eliminate it... This is caused by the accumulation of a precision
* error, and all timebase master routines I've examined appear to
* suffer from this same tempo distortion (and all use the magic
* number of 1920 ticks_per_beat in an attempt to reduce the magnitude
* of the error. Currently, we keep this behaviour. */
void void
Transport::timebase ( jack_transport_state_t state, jack_nframes_t nframes, jack_position_t *pos, int new_pos, void *arg ) Transport::timebase ( jack_transport_state_t state, jack_nframes_t nframes, jack_position_t *pos, int new_pos, void *arg )
{ {
pos->valid = JackPositionBBT;
pos->beats_per_bar = transport._master_beats_per_bar;
pos->ticks_per_beat = PPQN;

/* FIXME: WTF is this? Quarter note? */
pos->beat_type = transport._master_beat_type;

pos->beats_per_minute = transport._master_beats_per_minute;


if ( new_pos || ! _done ) if ( new_pos || ! _done )
{ {
pos->valid = JackPositionBBT;
pos->beats_per_bar = transport._master_beats_per_bar;
pos->ticks_per_beat = 1920.0; /* magic number means what? */
pos->beat_type = transport._master_beat_type;
pos->beats_per_minute = transport._master_beats_per_minute;

double wallclock = (double)pos->frame / (pos->frame_rate * 60); double wallclock = (double)pos->frame / (pos->frame_rate * 60);


unsigned long abs_tick = wallclock * pos->beats_per_minute * pos->ticks_per_beat; unsigned long abs_tick = wallclock * pos->beats_per_minute * pos->ticks_per_beat;
@@ -66,17 +72,19 @@ Transport::timebase ( jack_transport_state_t state, jack_nframes_t nframes, jack
} }
else else
{ {
// FIXME: use ticks_per_period here?
pos->tick += nframes * pos->ticks_per_beat * pos->beats_per_minute / (pos->frame_rate * 60); pos->tick += nframes * pos->ticks_per_beat * pos->beats_per_minute / (pos->frame_rate * 60);


while (pos->tick >= pos->ticks_per_beat) {
while ( pos->tick >= pos->ticks_per_beat )
{
pos->tick -= pos->ticks_per_beat; pos->tick -= pos->ticks_per_beat;
if (++pos->beat > pos->beats_per_bar) {

if ( ++pos->beat > pos->beats_per_bar )
{
pos->beat = 1; pos->beat = 1;

++pos->bar; ++pos->bar;
pos->bar_start_tick +=
pos->beats_per_bar
* pos->ticks_per_beat;

pos->bar_start_tick += pos->beats_per_bar * pos->ticks_per_beat;
} }
} }
} }
@@ -132,17 +140,22 @@ Transport::poll ( void )


/* FIXME: this only needs to be calculated if bpm or framerate changes */ /* FIXME: this only needs to be calculated if bpm or framerate changes */
{ {
double frames_per_beat = frame_rate * 60 / beats_per_minute;
const double frames_per_beat = frame_rate * 60 / beats_per_minute;


frames_per_tick = frames_per_beat / (double)PPQN; frames_per_tick = frames_per_beat / (double)PPQN;
ticks_per_period = nframes / frames_per_tick; ticks_per_period = nframes / frames_per_tick;
} }


tick_t abs_tick = (pos.bar * pos.beats_per_bar + pos.beat) * pos.ticks_per_beat + pos.tick; tick_t abs_tick = (pos.bar * pos.beats_per_bar + pos.beat) * pos.ticks_per_beat + pos.tick;
ticks = abs_tick * (PPQN / pos.ticks_per_beat);
// ticks = abs_tick / (pos.ticks_per_beat / PPQN);
// tick_t abs_tick = pos.bar_start_tick + (pos.beat * pos.ticks_per_beat) + pos.tick;

/* scale Jack's ticks to our ticks */

const double pulses_per_tick = PPQN / pos.ticks_per_beat;

ticks = abs_tick * pulses_per_tick;
tick = tick * pulses_per_tick;


tick = tick * (PPQN / pos.ticks_per_beat);
ticks_per_beat = PPQN; ticks_per_beat = PPQN;
} }




+ 1
- 1
transport.H View File

@@ -53,8 +53,8 @@ public:
playhead_t ticks; playhead_t ticks;


unsigned beats_per_bar; unsigned beats_per_bar;
unsigned ticks_per_beat;
unsigned beat_type; unsigned beat_type;
double ticks_per_beat;
double beats_per_minute; double beats_per_minute;


double ticks_per_period; double ticks_per_period;


Loading…
Cancel
Save