A JACK patch to fix some old issues, from author.

git-svn-id: https://svn.musicpd.org/mpd/trunk@5191 09075e82-0dd4-0310-85a5-a0d7c8717e4f
This commit is contained in:
Avuton Olrich 2006-12-30 00:14:45 +00:00
parent 878fc33ce9
commit 43e2981f20

View File

@ -32,64 +32,65 @@
#include <jack/types.h> #include <jack/types.h>
#include <jack/ringbuffer.h> #include <jack/ringbuffer.h>
/*#include "dmalloc.h"*/
#define MIN(a, b) ((a) < (b) ? (a) : (b)) #define MIN(a, b) ((a) < (b) ? (a) : (b))
char *name = "mpd";
char *output_ports[2] = {NULL, NULL};
int ringbuf_sz = 32768; int ringbuf_sz = 32768;
char *ports[2] = {NULL, NULL};
typedef struct _JackData { typedef struct _JackData {
jack_options_t options;
jack_port_t *ports[2]; jack_port_t *ports[2];
jack_client_t *client; jack_client_t *client;
jack_default_audio_sample_t **in;
jack_ringbuffer_t *ringbuffer[2]; jack_ringbuffer_t *ringbuffer[2];
jack_default_audio_sample_t *samples1;
jack_default_audio_sample_t *samples2;
int bps; int bps;
int shutdown; int shutdown;
int nports;
int our_xrun; int our_xrun;
} JackData; } JackData;
JackData *jd = NULL; /*JackData *jd = NULL;*/
static JackData *newJackData (void) static JackData *newJackData(void)
{ {
JackData *ret; JackData *ret;
ret = xcalloc(sizeof(JackData), 1);
ret = calloc (sizeof (JackData), 1);
ret->options = JackNullOption;
return ret; return ret;
} }
static void disconnect_jack (JackData *jd) static void jack_finishDriver(AudioOutput * audioOutput)
{
jack_deactivate (jd->client);
jack_client_close (jd->client);
ERROR ("disconnect_jack (pid=%d)\n", getpid ());
}
static void jack_finishDriver (AudioOutput * audioOutput)
{ {
JackData *jd = audioOutput->data; JackData *jd = audioOutput->data;
disconnect_jack (jd); jack_deactivate(jd->client);
jack_client_close(jd->client);
ERROR("disconnect_jack (pid=%d)\n", getpid ());
free (jd); if ( strcmp(name, "mpd") ) free(name);
free (ports[0]); if ( output_ports[0] ) free(output_ports[0]);
free (ports[1]); if ( output_ports[1] ) free(output_ports[1]);
jack_ringbuffer_free(jd->ringbuffer[0]);
jack_ringbuffer_free(jd->ringbuffer[1]);
free(jd->samples1);
free(jd->samples2);
free(jd);
} }
static int srate (jack_nframes_t rate, void *data) static int srate(jack_nframes_t rate, void *data)
{ {
JackData *jd = (JackData *) ((AudioOutput*) data)->data; JackData *jd = (JackData *) ((AudioOutput*) data)->data;
AudioFormat *audioFormat = &(((AudioOutput*) data)->outAudioFormat); AudioFormat *audioFormat = &(((AudioOutput*) data)->outAudioFormat);
audioFormat->sampleRate = (int) jack_get_sample_rate (jd->client); audioFormat->sampleRate = (int)jack_get_sample_rate(jd->client);
return 0; return 0;
} }
static int process (jack_nframes_t nframes, void *arg) static int process(jack_nframes_t nframes, void *arg)
{ {
size_t i; size_t i;
JackData *jd = (JackData *) arg; JackData *jd = (JackData *) arg;
@ -99,19 +100,19 @@ static int process (jack_nframes_t nframes, void *arg)
if ( nframes <= 0 ) if ( nframes <= 0 )
return 0; return 0;
out[0] = jack_port_get_buffer (jd->ports[0], nframes); out[0] = jack_port_get_buffer(jd->ports[0], nframes);
out[1] = jack_port_get_buffer (jd->ports[1], nframes); out[1] = jack_port_get_buffer(jd->ports[1], nframes);
avail_data = jack_ringbuffer_read_space (jd->ringbuffer[1]); avail_data = jack_ringbuffer_read_space(jd->ringbuffer[1]);
if ( avail_data > 0 ) { if ( avail_data > 0 ) {
avail_frames = avail_data / sizeof (jack_default_audio_sample_t); avail_frames = avail_data / sizeof(jack_default_audio_sample_t);
if (avail_frames > nframes) { if (avail_frames > nframes) {
avail_frames = nframes; avail_frames = nframes;
avail_data = nframes * sizeof (jack_default_audio_sample_t); avail_data = nframes * sizeof(jack_default_audio_sample_t);
} }
jack_ringbuffer_read (jd->ringbuffer[0], (char *)out[0], avail_data); jack_ringbuffer_read(jd->ringbuffer[0], (char *)out[0], avail_data);
jack_ringbuffer_read (jd->ringbuffer[1], (char *)out[1], avail_data); jack_ringbuffer_read(jd->ringbuffer[1], (char *)out[1], avail_data);
if (avail_frames < nframes) { if (avail_frames < nframes) {
jd->our_xrun = 1; jd->our_xrun = 1;
@ -121,148 +122,77 @@ static int process (jack_nframes_t nframes, void *arg)
} }
} else { } else {
//ERROR ("avail_data=%d, no play (pid=%d)!\n", avail_data, getpid ()); //ERROR ("avail_data=%d, no play (pid=%d)!\n", avail_data, getpid ());
for (i = 0; i < nframes; i++)
for (i = 0; i < nframes; i++) { out[0][i] = out[1][i] = 0.0;
out[0][i] = 0.0;
out[1][i] = 0.0;
}
} }
/*ERROR("process (pid=%d)\n", getpid());*/
return 0; return 0;
} }
static void shutdown_callback (void *arg) static void shutdown_callback(void *arg)
{ {
JackData *jd = (JackData *) arg; JackData *jd = (JackData *) arg;
jd->shutdown = 1; jd->shutdown = 1;
} }
static void set_audioformat (AudioOutput *audioOutput) static void set_audioformat(AudioOutput *audioOutput)
{ {
JackData *jd = audioOutput->data; JackData *jd = audioOutput->data;
AudioFormat *audioFormat = &audioOutput->outAudioFormat; AudioFormat *audioFormat = &audioOutput->outAudioFormat;
audioFormat->sampleRate = (int) jack_get_sample_rate (jd->client); audioFormat->sampleRate = (int) jack_get_sample_rate(jd->client);
ERROR ("samplerate = %d\n", audioFormat->sampleRate); ERROR ("samplerate = %d\n", audioFormat->sampleRate);
jd->nports = audioFormat->channels = 2; audioFormat->channels = 2;
audioFormat->bits = 16; audioFormat->bits = 16;
jd->bps = audioFormat->channels * jd->bps = audioFormat->channels *
audioFormat->channels * audioFormat->channels *
audioFormat->sampleRate; audioFormat->sampleRate;
} }
static int connect_jack (AudioOutput *audioOutput) static void error_callback(const char *msg)
{ {
JackData *jd = audioOutput->data; ERROR("jack: %s\n", msg);
const char **jports;
if ( (jd->client = jack_client_new ("mpd")) == NULL ) {
ERROR ("jack server not running?\n");
return -1;
}
jd->ringbuffer[0] = jack_ringbuffer_create (ringbuf_sz);
jd->ringbuffer[1] = jack_ringbuffer_create (ringbuf_sz);
jack_set_process_callback (jd->client, process, (void *)jd);
jack_set_sample_rate_callback (jd->client, (JackProcessCallback)srate,
(void *)audioOutput);
jack_on_shutdown (jd->client, shutdown_callback, (void *)jd);
if ( jack_activate (jd->client) ) {
ERROR ("cannot activate client");
jack_ringbuffer_free (jd->ringbuffer[0]);
jack_ringbuffer_free (jd->ringbuffer[1]);
return -1;
}
jd->ports[0] = jack_port_register (jd->client, "left",
JACK_DEFAULT_AUDIO_TYPE,
JackPortIsOutput, 0);
if ( !jd->ports[0] ) {
ERROR ("Cannot register output port.\n");
jack_ringbuffer_free (jd->ringbuffer[0]);
jack_ringbuffer_free (jd->ringbuffer[1]);
return -1;
}
jd->ports[1] = jack_port_register (jd->client, "right",
JACK_DEFAULT_AUDIO_TYPE,
JackPortIsOutput, 0);
if ( !jd->ports[1] ) {
ERROR ("Cannot register output port.\n");
jack_ringbuffer_free (jd->ringbuffer[0]);
jack_ringbuffer_free (jd->ringbuffer[1]);
return -1;
}
memset (jd->ringbuffer[0]->buf, 0, jd->ringbuffer[0]->size);
memset (jd->ringbuffer[1]->buf, 0, jd->ringbuffer[1]->size);
/* hay que buscar que hay */
if ( !ports[1] && (jports = jack_get_ports (jd->client, NULL, NULL,
JackPortIsPhysical|
JackPortIsInput)) ) {
ports[0] = (char *) jports[0];
ports[1] = (char *) ( jports[1] ? jports[1] : jports[0] );
ERROR ("jports: %s %s\n", ports[0], ports[1]);
free (jports);
}
if ( ports[1] ) {
if ( (jack_connect (jd->client, "mpd:left", ports[0])) != 0 ) {
ERROR ("%s is not a valid Jack Client / Port ", ports[0]);
jack_ringbuffer_free (jd->ringbuffer[0]);
jack_ringbuffer_free (jd->ringbuffer[1]);
return -1;
}
if ( (jack_connect (jd->client, "mpd:right", ports[1])) != 0 ) {
ERROR ("%s is not a valid Jack Client / Port ", ports[1]);
jack_ringbuffer_free (jd->ringbuffer[0]);
jack_ringbuffer_free (jd->ringbuffer[1]);
return -1;
}
}
ERROR ("connect_jack (pid=%d)\n", getpid ());
return 1;
} }
static int jack_initDriver (AudioOutput *audioOutput, ConfigParam *param) static int jack_initDriver(AudioOutput *audioOutput, ConfigParam *param)
{ {
BlockParam *bp; BlockParam *bp;
char *endptr; char *endptr;
int val; int val;
char *cp = NULL; char *cp = NULL;
if ( param ) { ERROR("jack_initDriver (pid=%d)\n", getpid());
bp = getBlockParam (param, "ports"); if ( ! param ) return 0;
if ( bp ) {
cp = strdup (bp->value);
ports[0] = strdup (strtok (cp, " ,"));
ports[1] = strdup (strtok (NULL, " ,"));
free (cp);
}
bp = getBlockParam (param, "ringbuffer_size"); if ( (bp = getBlockParam(param, "ports")) ) {
if ( bp ) { cp = strdup(bp->value);
errno = 0; ERROR("output_ports=%s\n", cp);
val = strtol (bp->value, &endptr, 10); output_ports[0] = strdup(strtok (cp, ","));
output_ports[1] = strdup(strtok (NULL, ","));
free(cp);
}
if ( errno == 0 && endptr != bp->value) { if ( (bp = getBlockParam(param, "ringbuffer_size")) ) {
ringbuf_sz = val; errno = 0;
ERROR ("ringbuffer_size=%d\n", ringbuf_sz); val = strtol(bp->value, &endptr, 10);
} else {
ERROR ("%s is not a number; ringbuf_size=%d\n", if ( errno == 0 && endptr != bp->value) {
bp->value, ringbuf_sz); ringbuf_sz = val < 32768 ? 32768 : val;
} ERROR("ringbuffer_size=%d\n", ringbuf_sz);
} else {
ERROR("%s is not a number; ringbuf_size=%d\n",
bp->value, ringbuf_sz);
} }
} }
ERROR ("jack_initDriver (pid=%d)\n", getpid ()); if ( (bp = getBlockParam(param, "name"))
&& (strcmp(bp->value, "mpd") != 0) ) {
name = strdup(bp->value);
ERROR("name=%s\n", name);
}
return 0; return 0;
} }
static int jack_testDefault(void) static int jack_testDefault(void)
@ -270,22 +200,104 @@ static int jack_testDefault(void)
return 0; return 0;
} }
static int jack_openDevice (AudioOutput *audioOutput) static int connect_jack(AudioOutput *audioOutput)
{ {
if ( !jd ) { JackData *jd = audioOutput->data;
jd = newJackData (); const char **jports;
audioOutput->data = jd; char *port_name;
if ( !connect_jack (audioOutput) ) { if ( (jd->client = jack_client_new(name)) == NULL ) {
free (jd); ERROR("jack server not running?\n");
return -1; return -1;
}
} }
set_audioformat (audioOutput); jack_set_error_function(error_callback);
jack_set_process_callback(jd->client, process, (void *)jd);
jack_set_sample_rate_callback(jd->client, (JackProcessCallback)srate,
(void *)audioOutput);
jack_on_shutdown(jd->client, shutdown_callback, (void *)jd);
if ( jack_activate(jd->client) ) {
ERROR("cannot activate client");
return -1;
}
jd->ports[0] = jack_port_register(jd->client, "left",
JACK_DEFAULT_AUDIO_TYPE,
JackPortIsOutput, 0);
if ( !jd->ports[0] ) {
ERROR("Cannot register left output port.\n");
return -1;
}
jd->ports[1] = jack_port_register(jd->client, "right",
JACK_DEFAULT_AUDIO_TYPE,
JackPortIsOutput, 0);
if ( !jd->ports[1] ) {
ERROR("Cannot register right output port.\n");
return -1;
}
/* hay que buscar que hay */
if ( !output_ports[1] && (jports = jack_get_ports(jd->client, NULL, NULL,
JackPortIsPhysical|
JackPortIsInput)) ) {
output_ports[0] = jports[0];
output_ports[1] = jports[1] ? jports[1] : jports[0];
ERROR("output_ports: %s %s\n", output_ports[0], output_ports[1]);
free(jports);
}
if ( output_ports[1] ) {
jd->ringbuffer[0] = jack_ringbuffer_create(ringbuf_sz);
jd->ringbuffer[1] = jack_ringbuffer_create(ringbuf_sz);
memset(jd->ringbuffer[0]->buf, 0, jd->ringbuffer[0]->size);
memset(jd->ringbuffer[1]->buf, 0, jd->ringbuffer[1]->size);
port_name = xmalloc(sizeof(char)*(7+strlen(name)));
sprintf(port_name, "%s:left", name);
if ( (jack_connect(jd->client, port_name, output_ports[0])) != 0 ) {
ERROR("%s is not a valid Jack Client / Port ", output_ports[0]);
jack_ringbuffer_free(jd->ringbuffer[0]);
jack_ringbuffer_free(jd->ringbuffer[1]);
free(port_name);
return -1;
}
sprintf(port_name, "%s:right", name);
if ( (jack_connect(jd->client, port_name, output_ports[1])) != 0 ) {
ERROR("%s is not a valid Jack Client / Port ", output_ports[1]);
jack_ringbuffer_free(jd->ringbuffer[0]);
jack_ringbuffer_free(jd->ringbuffer[1]);
free(port_name);
return -1;
}
free(port_name);
}
ERROR("connect_jack (pid=%d)\n", getpid());
return 1;
}
static int jack_openDevice(AudioOutput *audioOutput)
{
JackData *jd = audioOutput->data;
if ( !jd ) {
ERROR("connect!\n");
jd = newJackData();
audioOutput->data = jd;
if ( !connect_jack(audioOutput) ) {
jack_finishDriver(audioOutput);
return -1;
}
}
set_audioformat(audioOutput);
audioOutput->open = 1; audioOutput->open = 1;
ERROR ("jack_openDevice (pid=%d)!\n", getpid ()); ERROR("jack_openDevice (pid=%d)!\n", getpid ());
return 0; return 0;
} }
@ -293,7 +305,7 @@ static int jack_openDevice (AudioOutput *audioOutput)
static void jack_closeDevice(AudioOutput * audioOutput) static void jack_closeDevice(AudioOutput * audioOutput)
{ {
audioOutput->open = 0; audioOutput->open = 0;
ERROR ("jack_closeDevice (pid=%d)!\n", getpid ()); ERROR("jack_closeDevice (pid=%d)!\n", getpid());
} }
static void jack_dropBufferedAudio (AudioOutput * audioOutput) static void jack_dropBufferedAudio (AudioOutput * audioOutput)
@ -304,53 +316,60 @@ static void jack_dropBufferedAudio (AudioOutput * audioOutput)
static int jack_playAudio(AudioOutput * audioOutput, char *buff, int size) static int jack_playAudio(AudioOutput * audioOutput, char *buff, int size)
{ {
JackData *jd = audioOutput->data; JackData *jd = audioOutput->data;
size_t remain = size; size_t space;
size_t pos = 0; char *samples1;
char *samples2;
int i;
short *buffer = (short *) buff;
if ( jd->shutdown ) { if ( jd->shutdown ) {
ERROR ("Refusing to play, because there is no client thread.\n"); ERROR("Refusing to play, because there is no client thread.\n");
return 0; return 0;
} }
if ( jd->our_xrun ) { if ( jd->our_xrun ) {
ERROR ("xrun\n"); ERROR("xrun\n");
jd->our_xrun = 0; jd->our_xrun = 0;
} }
while (remain && !jd->shutdown) { /*ERROR("jack_playAudio: size=%d\n", size/4);*/
size_t space; /*ERROR("jack_playAudio - INICIO\n");*/
if ( (space = jack_ringbuffer_write_space (jd->ringbuffer[0])) if ( ! jd->samples1 ) {
> sizeof (jack_default_audio_sample_t) ) { ERROR("jd->samples1=xmalloc\n");
size_t to_write; jd->samples1 = (jack_default_audio_sample_t *)xmalloc(size);
}
if ( ! jd->samples2 ) {
ERROR("jd->samples2=xmalloc\n");
jd->samples2 = (jack_default_audio_sample_t *)xmalloc(size);
}
to_write = MIN (space, remain); /* primero convierto todo el buffer al formato que usa jack */
remain -= to_write; for (i=0; i<size/4; i++) {
to_write /= 4; *(jd->samples1 + i) = (jack_default_audio_sample_t) *(buffer++) / 32768;
//ERROR ("\t\tto_write=%d remain=%d (%d)\n", to_write, remain, to_write * 2 * 2); *(jd->samples2 + i) = (jack_default_audio_sample_t) *(buffer++) / 32768;
while (to_write--) { }
jack_default_audio_sample_t sample;
sample = *(short *)(buff + pos); samples1=(char *)jd->samples1;
sample /= 32768; samples2=(char *)jd->samples2;
pos += 2; while ( size && !jd->shutdown ) {
jack_ringbuffer_write (jd->ringbuffer[0], if ( (space = jack_ringbuffer_write_space(jd->ringbuffer[0])) >
(char *)&sample, sizeof(jack_default_audio_sample_t) ) {
sizeof (sample)); /*ERROR("\t size=%d space=%d\n", size, space);*/
space = MIN(space, size);
jack_ringbuffer_write(jd->ringbuffer[0],samples1,space);
sample = *(short *)(buff + pos); jack_ringbuffer_write(jd->ringbuffer[1],samples2,space);
sample /= 32768; size -= space;
pos += 2; samples1 += space;
jack_ringbuffer_write (jd->ringbuffer[1], samples2 += space;
(char *)&sample,
sizeof (sample));
}
} else { } else {
usleep (ringbuf_sz / (float)(jd->bps) * 100000.0); /*ERROR("\t space=%d\n", space);*/
usleep(ringbuf_sz/(float)(jd->bps) * 100000.0);
} }
} }
/*ERROR("jack_playAudio - FIN\n");*/
return 0; return 0;
} }