|
@@ -87,7 +87,8 @@ audio_driver *audio_driver_lookup(const char *name)
|
|
|
return NULL;
|
|
|
}
|
|
|
|
|
|
-static AudioState glob_audio_state;
|
|
|
+static QTAILQ_HEAD(AudioStateHead, AudioState) audio_states =
|
|
|
+ QTAILQ_HEAD_INITIALIZER(audio_states);
|
|
|
|
|
|
const struct mixeng_volume nominal_volume = {
|
|
|
.mute = 0,
|
|
@@ -1238,11 +1239,14 @@ static void audio_run_capture (AudioState *s)
|
|
|
|
|
|
void audio_run (const char *msg)
|
|
|
{
|
|
|
- AudioState *s = &glob_audio_state;
|
|
|
+ AudioState *s;
|
|
|
+
|
|
|
+ QTAILQ_FOREACH(s, &audio_states, list) {
|
|
|
+ audio_run_out(s);
|
|
|
+ audio_run_in(s);
|
|
|
+ audio_run_capture(s);
|
|
|
+ }
|
|
|
|
|
|
- audio_run_out (s);
|
|
|
- audio_run_in (s);
|
|
|
- audio_run_capture (s);
|
|
|
#ifdef DEBUG_POLL
|
|
|
{
|
|
|
static double prevtime;
|
|
@@ -1306,13 +1310,11 @@ bool audio_is_cleaning_up(void)
|
|
|
return is_cleaning_up;
|
|
|
}
|
|
|
|
|
|
-void audio_cleanup(void)
|
|
|
+static void free_audio_state(AudioState *s)
|
|
|
{
|
|
|
- AudioState *s = &glob_audio_state;
|
|
|
HWVoiceOut *hwo, *hwon;
|
|
|
HWVoiceIn *hwi, *hwin;
|
|
|
|
|
|
- is_cleaning_up = true;
|
|
|
QLIST_FOREACH_SAFE(hwo, &s->hw_head_out, entries, hwon) {
|
|
|
SWVoiceCap *sc;
|
|
|
|
|
@@ -1349,6 +1351,17 @@ void audio_cleanup(void)
|
|
|
qapi_free_Audiodev(s->dev);
|
|
|
s->dev = NULL;
|
|
|
}
|
|
|
+ g_free(s);
|
|
|
+}
|
|
|
+
|
|
|
+void audio_cleanup(void)
|
|
|
+{
|
|
|
+ is_cleaning_up = true;
|
|
|
+ while (!QTAILQ_EMPTY(&audio_states)) {
|
|
|
+ AudioState *s = QTAILQ_FIRST(&audio_states);
|
|
|
+ QTAILQ_REMOVE(&audio_states, s, list);
|
|
|
+ free_audio_state(s);
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
static const VMStateDescription vmstate_audio = {
|
|
@@ -1375,28 +1388,33 @@ static AudiodevListEntry *audiodev_find(
|
|
|
return NULL;
|
|
|
}
|
|
|
|
|
|
-static int audio_init(Audiodev *dev)
|
|
|
+/*
|
|
|
+ * if we have dev, this function was called because of an -audiodev argument =>
|
|
|
+ * initialize a new state with it
|
|
|
+ * if dev == NULL => legacy implicit initialization, return the already created
|
|
|
+ * state or create a new one
|
|
|
+ */
|
|
|
+static AudioState *audio_init(Audiodev *dev)
|
|
|
{
|
|
|
+ static bool atexit_registered;
|
|
|
size_t i;
|
|
|
int done = 0;
|
|
|
const char *drvname = NULL;
|
|
|
VMChangeStateEntry *e;
|
|
|
- AudioState *s = &glob_audio_state;
|
|
|
+ AudioState *s;
|
|
|
struct audio_driver *driver;
|
|
|
/* silence gcc warning about uninitialized variable */
|
|
|
AudiodevListHead head = QSIMPLEQ_HEAD_INITIALIZER(head);
|
|
|
|
|
|
- if (s->drv) {
|
|
|
- if (dev) {
|
|
|
- dolog("Cannot create more than one audio backend, sorry\n");
|
|
|
- qapi_free_Audiodev(dev);
|
|
|
- }
|
|
|
- return -1;
|
|
|
- }
|
|
|
-
|
|
|
if (dev) {
|
|
|
/* -audiodev option */
|
|
|
drvname = AudiodevDriver_str(dev->driver);
|
|
|
+ } else if (!QTAILQ_EMPTY(&audio_states)) {
|
|
|
+ /*
|
|
|
+ * todo: check for -audiodev once we have normal audiodev selection
|
|
|
+ * support
|
|
|
+ */
|
|
|
+ return QTAILQ_FIRST(&audio_states);
|
|
|
} else {
|
|
|
/* legacy implicit initialization */
|
|
|
head = audio_handle_legacy_opts();
|
|
@@ -1410,12 +1428,18 @@ static int audio_init(Audiodev *dev)
|
|
|
dev = QSIMPLEQ_FIRST(&head)->dev;
|
|
|
audio_validate_opts(dev, &error_abort);
|
|
|
}
|
|
|
+
|
|
|
+ s = g_malloc0(sizeof(AudioState));
|
|
|
s->dev = dev;
|
|
|
|
|
|
QLIST_INIT (&s->hw_head_out);
|
|
|
QLIST_INIT (&s->hw_head_in);
|
|
|
QLIST_INIT (&s->cap_head);
|
|
|
- atexit(audio_cleanup);
|
|
|
+ if (!atexit_registered) {
|
|
|
+ atexit(audio_cleanup);
|
|
|
+ atexit_registered = true;
|
|
|
+ }
|
|
|
+ QTAILQ_INSERT_TAIL(&audio_states, s, list);
|
|
|
|
|
|
s->ts = timer_new_ns(QEMU_CLOCK_VIRTUAL, audio_timer, s);
|
|
|
|
|
@@ -1480,7 +1504,7 @@ static int audio_init(Audiodev *dev)
|
|
|
|
|
|
QLIST_INIT (&s->card_head);
|
|
|
vmstate_register (NULL, 0, &vmstate_audio, s);
|
|
|
- return 0;
|
|
|
+ return s;
|
|
|
}
|
|
|
|
|
|
void audio_free_audiodev_list(AudiodevListHead *head)
|
|
@@ -1495,10 +1519,13 @@ void audio_free_audiodev_list(AudiodevListHead *head)
|
|
|
|
|
|
void AUD_register_card (const char *name, QEMUSoundCard *card)
|
|
|
{
|
|
|
- audio_init(NULL);
|
|
|
+ if (!card->state) {
|
|
|
+ card->state = audio_init(NULL);
|
|
|
+ }
|
|
|
+
|
|
|
card->name = g_strdup (name);
|
|
|
memset (&card->entries, 0, sizeof (card->entries));
|
|
|
- QLIST_INSERT_HEAD (&glob_audio_state.card_head, card, entries);
|
|
|
+ QLIST_INSERT_HEAD(&card->state->card_head, card, entries);
|
|
|
}
|
|
|
|
|
|
void AUD_remove_card (QEMUSoundCard *card)
|
|
@@ -1508,16 +1535,21 @@ void AUD_remove_card (QEMUSoundCard *card)
|
|
|
}
|
|
|
|
|
|
|
|
|
-CaptureVoiceOut *AUD_add_capture (
|
|
|
+CaptureVoiceOut *AUD_add_capture(
|
|
|
+ AudioState *s,
|
|
|
struct audsettings *as,
|
|
|
struct audio_capture_ops *ops,
|
|
|
void *cb_opaque
|
|
|
)
|
|
|
{
|
|
|
- AudioState *s = &glob_audio_state;
|
|
|
CaptureVoiceOut *cap;
|
|
|
struct capture_callback *cb;
|
|
|
|
|
|
+ if (!s) {
|
|
|
+ /* todo: remove when we have normal audiodev selection support */
|
|
|
+ s = audio_init(NULL);
|
|
|
+ }
|
|
|
+
|
|
|
if (audio_validate_settings (as)) {
|
|
|
dolog ("Invalid settings were passed when trying to add capture\n");
|
|
|
audio_print_settings (as);
|
|
@@ -1807,3 +1839,25 @@ int audio_buffer_bytes(AudiodevPerDirectionOptions *pdo,
|
|
|
return audio_buffer_samples(pdo, as, def_usecs) *
|
|
|
audioformat_bytes_per_sample(as->fmt);
|
|
|
}
|
|
|
+
|
|
|
+AudioState *audio_state_by_name(const char *name)
|
|
|
+{
|
|
|
+ AudioState *s;
|
|
|
+ QTAILQ_FOREACH(s, &audio_states, list) {
|
|
|
+ assert(s->dev);
|
|
|
+ if (strcmp(name, s->dev->id) == 0) {
|
|
|
+ return s;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return NULL;
|
|
|
+}
|
|
|
+
|
|
|
+const char *audio_get_id(QEMUSoundCard *card)
|
|
|
+{
|
|
|
+ if (card->state) {
|
|
|
+ assert(card->state->dev);
|
|
|
+ return card->state->dev->id;
|
|
|
+ } else {
|
|
|
+ return "";
|
|
|
+ }
|
|
|
+}
|