This repository has been archived on 2025-09-15. You can view files and clone it. You cannot open issues or pull requests or push a commit.
Files
alsa-scarlett-gui/src/alsa.c
Geoffrey D. Bennett 47034d7901 Remove widgets from struct alsa_elem and add data to callbacks
Rather than having widget/widget2/widget_callback fields in the struct
alsa_elem, have a list of callbacks and allow private data to be
passed to callbacks.
2024-02-05 20:04:58 +10:30

660 lines
17 KiB
C

// SPDX-FileCopyrightText: 2022 Geoffrey D. Bennett <g@b4.vu>
// SPDX-License-Identifier: GPL-3.0-or-later
#include <sys/inotify.h>
#include "alsa.h"
#include "stringhelper.h"
#include "window-iface.h"
#define MAX_TLV_RANGE_SIZE 256
// names for the port categories
const char *port_category_names[PC_COUNT] = {
"Hardware Outputs",
"Mixer Inputs",
"DSP Inputs",
"PCM Inputs"
};
// global array of cards
GArray *alsa_cards;
// static fd and wd for ALSA inotify
static int inotify_fd, inotify_wd;
// forward declaration
static void alsa_elem_change(struct alsa_elem *elem);
void fatal_alsa_error(const char *msg, int err) {
fprintf(stderr, "%s: %s\n", msg, snd_strerror(err));
exit(1);
}
//
// functions to locate elements or get information about them
//
// return the element with the exact matching name
struct alsa_elem *get_elem_by_name(GArray *elems, char *name) {
for (int i = 0; i < elems->len; i++) {
struct alsa_elem *elem = &g_array_index(elems, struct alsa_elem, i);
if (!elem->card)
continue;
if (strcmp(elem->name, name) == 0)
return elem;
}
return NULL;
}
// return the first element with a name starting with the given prefix
struct alsa_elem *get_elem_by_prefix(GArray *elems, char *prefix) {
int prefix_len = strlen(prefix);
for (int i = 0; i < elems->len; i++) {
struct alsa_elem *elem = &g_array_index(elems, struct alsa_elem, i);
if (!elem->card)
continue;
if (strncmp(elem->name, prefix, prefix_len) == 0)
return elem;
}
return NULL;
}
// find the maximum number in the matching elements
// search by element name prefix and substring
// e.g. get_max_elem_by_name(elems, "Line", "Pad Capture Switch")
// will return 8 when the last pad capture switch is
// "Line In 8 Pad Capture Switch"
int get_max_elem_by_name(GArray *elems, char *prefix, char *needle) {
int max = 0;
int l = strlen(prefix);
for (int i = 0; i < elems->len; i++) {
struct alsa_elem *elem = &g_array_index(elems, struct alsa_elem, i);
int num;
if (!elem->card)
continue;
if (strncmp(elem->name, prefix, l) != 0)
continue;
if (!strstr(elem->name, needle))
continue;
num = get_num_from_string(elem->name);
if (num > max)
max = num;
}
return max;
}
// return true if the element is an routing sink enum, e.g.:
// PCM xx Capture Enum
// Mixer Input xx Capture Enum
// Analogue Output xx Playback Enum
// S/PDIF Output xx Playback Enum
// ADAT Output xx Playback Enum
int is_elem_routing_snk(struct alsa_elem *elem) {
if (strstr(elem->name, "Capture Enum") && (
strncmp(elem->name, "PCM ", 4) == 0 ||
strncmp(elem->name, "Mixer Input ", 12) == 0 ||
strncmp(elem->name, "DSP Input ", 10) == 0
))
return 1;
if (strstr(elem->name, "Output") &&
strstr(elem->name, "Playback Enum"))
return 1;
return 0;
}
// add a callback to the list of callbacks for this element
void alsa_elem_add_callback(
struct alsa_elem *elem,
AlsaElemCallback *callback,
void *data
) {
struct alsa_elem_callback *cb = calloc(1, sizeof(struct alsa_elem_callback));
cb->callback = callback;
cb->data = data;
elem->callbacks = g_list_append(elem->callbacks, cb);
}
//
// alsa snd_ctl_elem_*() mediation functions
// for simulated elements, fake the ALSA element
// for real elements, pass through to snd_ctl_elem*()
//
// get the element type
int alsa_get_elem_type(struct alsa_elem *elem) {
snd_ctl_elem_info_t *elem_info;
snd_ctl_elem_info_alloca(&elem_info);
snd_ctl_elem_info_set_numid(elem_info, elem->numid);
snd_ctl_elem_info(elem->card->handle, elem_info);
return snd_ctl_elem_info_get_type(elem_info);
}
// get the element name
char *alsa_get_elem_name(struct alsa_elem *elem) {
snd_ctl_elem_info_t *elem_info;
snd_ctl_elem_info_alloca(&elem_info);
snd_ctl_elem_info_set_numid(elem_info, elem->numid);
snd_ctl_elem_info(elem->card->handle, elem_info);
const char *name = snd_ctl_elem_info_get_name(elem_info);
return strdup(name);
}
// get the element value
// boolean, enum, or int all returned as long ints
long alsa_get_elem_value(struct alsa_elem *elem) {
if (elem->card->num == SIMULATED_CARD_NUM)
return elem->value;
snd_ctl_elem_value_t *elem_value;
snd_ctl_elem_value_alloca(&elem_value);
snd_ctl_elem_value_set_numid(elem_value, elem->numid);
snd_ctl_elem_read(elem->card->handle, elem_value);
int type = elem->type;
if (type == SND_CTL_ELEM_TYPE_BOOLEAN) {
return snd_ctl_elem_value_get_boolean(elem_value, 0);
} else if (type == SND_CTL_ELEM_TYPE_ENUMERATED) {
return snd_ctl_elem_value_get_enumerated(elem_value, 0);
} else if (type == SND_CTL_ELEM_TYPE_INTEGER) {
return snd_ctl_elem_value_get_integer(elem_value, 0);
} else {
fprintf(
stderr,
"internal error: elem %s (%d) type %d not bool/enum/int\n",
elem->name,
elem->numid,
elem->type
);
return 0;
}
}
// for elements with multiple int values, return all the values
// the int array returned needs to be freed by the caller
int *alsa_get_elem_int_values(struct alsa_elem *elem) {
int *values = calloc(elem->count, sizeof(int));
if (elem->card->num == SIMULATED_CARD_NUM) {
for (int i = 0; i < elem->count; i++)
values[i] = 0;
return values;
}
snd_ctl_elem_value_t *elem_value;
snd_ctl_elem_value_alloca(&elem_value);
snd_ctl_elem_value_set_numid(elem_value, elem->numid);
snd_ctl_elem_read(elem->card->handle, elem_value);
for (int i = 0; i < elem->count; i++)
values[i] = snd_ctl_elem_value_get_integer(elem_value, i);
return values;
}
// set the element value
// boolean, enum, or int all set from long ints
void alsa_set_elem_value(struct alsa_elem *elem, long value) {
if (elem->card->num == SIMULATED_CARD_NUM) {
if (elem->value != value) {
elem->value = value;
alsa_elem_change(elem);
}
return;
}
snd_ctl_elem_value_t *elem_value;
snd_ctl_elem_value_alloca(&elem_value);
snd_ctl_elem_value_set_numid(elem_value, elem->numid);
int type = elem->type;
if (type == SND_CTL_ELEM_TYPE_BOOLEAN) {
snd_ctl_elem_value_set_boolean(elem_value, 0, value);
} else if (type == SND_CTL_ELEM_TYPE_ENUMERATED) {
snd_ctl_elem_value_set_enumerated(elem_value, 0, value);
} else if (type == SND_CTL_ELEM_TYPE_INTEGER) {
snd_ctl_elem_value_set_integer(elem_value, 0, value);
} else {
fprintf(
stderr,
"internal error: elem %s (%d) type %d not bool/enum/int\n",
elem->name,
elem->numid,
elem->type
);
return;
}
snd_ctl_elem_write(elem->card->handle, elem_value);
}
// return whether the element can be modified (is writable)
int alsa_get_elem_writable(struct alsa_elem *elem) {
if (elem->card->num == SIMULATED_CARD_NUM)
return elem->writable;
snd_ctl_elem_info_t *elem_info;
snd_ctl_elem_info_alloca(&elem_info);
snd_ctl_elem_info_set_numid(elem_info, elem->numid);
snd_ctl_elem_info(elem->card->handle, elem_info);
return snd_ctl_elem_info_is_writable(elem_info);
}
// get the number of values this element has
// (most are just 1; the levels element is the exception)
int alsa_get_elem_count(struct alsa_elem *elem) {
snd_ctl_elem_info_t *elem_info;
snd_ctl_elem_info_alloca(&elem_info);
snd_ctl_elem_info_set_numid(elem_info, elem->numid);
snd_ctl_elem_info(elem->card->handle, elem_info);
return snd_ctl_elem_info_get_count(elem_info);
}
// get the number of items this enum element has
int alsa_get_item_count(struct alsa_elem *elem) {
if (elem->card->num == SIMULATED_CARD_NUM)
return elem->item_count;
snd_ctl_elem_info_t *elem_info;
snd_ctl_elem_info_alloca(&elem_info);
snd_ctl_elem_info_set_numid(elem_info, elem->numid);
snd_ctl_elem_info(elem->card->handle, elem_info);
return snd_ctl_elem_info_get_items(elem_info);
}
// get the name of an item of the given enum element
char *alsa_get_item_name(struct alsa_elem *elem, int i) {
if (elem->card->num == SIMULATED_CARD_NUM)
return elem->item_names[i];
snd_ctl_elem_info_t *elem_info;
snd_ctl_elem_info_alloca(&elem_info);
snd_ctl_elem_info_set_numid(elem_info, elem->numid);
snd_ctl_elem_info_set_item(elem_info, i);
snd_ctl_elem_info(elem->card->handle, elem_info);
const char *name = snd_ctl_elem_info_get_item_name(elem_info);
return strdup(name);
}
//
// create/destroy alsa cards
//
// scan the ALSA ctl element list container and put the useful
// elements into the cards->elems array of struct alsa_elem
static void alsa_get_elem_list(struct alsa_card *card) {
snd_ctl_elem_list_t *list;
int count;
// get the list from ALSA
snd_ctl_elem_list_malloc(&list);
snd_ctl_elem_list(card->handle, list);
count = snd_ctl_elem_list_get_count(list);
snd_ctl_elem_list_alloc_space(list, count);
snd_ctl_elem_list(card->handle, list);
// for each element in the list
for (int i = 0; i < count; i++) {
// allocate a temporary struct alsa_elem (will be copied later if
// we want to keep it)
struct alsa_elem alsa_elem = {};
// keep a reference to the card in the element
alsa_elem.card = card;
// get the control's numeric identifier (different to the index
// into this array)
alsa_elem.numid = snd_ctl_elem_list_get_numid(list, i);
// get the control's info
alsa_elem.type = alsa_get_elem_type(&alsa_elem);
alsa_elem.name = alsa_get_elem_name(&alsa_elem);
alsa_elem.count = alsa_get_elem_count(&alsa_elem);
switch (alsa_elem.type) {
case SND_CTL_ELEM_TYPE_BOOLEAN:
case SND_CTL_ELEM_TYPE_ENUMERATED:
case SND_CTL_ELEM_TYPE_INTEGER:
break;
default:
continue;
}
if (strstr(alsa_elem.name, "Validity"))
continue;
if (strstr(alsa_elem.name, "Channel Map"))
continue;
// get TLV info if it's a volume control
if (alsa_elem.type == SND_CTL_ELEM_TYPE_INTEGER) {
snd_ctl_elem_info_t *elem_info;
snd_ctl_elem_info_alloca(&elem_info);
snd_ctl_elem_info_set_numid(elem_info, alsa_elem.numid);
snd_ctl_elem_info(card->handle, elem_info);
if (snd_ctl_elem_info_is_tlv_readable(elem_info)) {
snd_ctl_elem_id_t *elem_id;
unsigned int tlv[MAX_TLV_RANGE_SIZE];
unsigned int *dbrec;
int ret;
long min_dB, max_dB;
snd_ctl_elem_id_alloca(&elem_id);
snd_ctl_elem_id_set_numid(elem_id, alsa_elem.numid);
ret = snd_ctl_elem_tlv_read(
card->handle, elem_id, tlv, sizeof(tlv)
);
if (ret < 0) {
fprintf(stderr, "TLV read error %d\n", ret);
continue;
}
ret = snd_tlv_parse_dB_info(tlv, sizeof(tlv), &dbrec);
if (ret <= 0) {
fprintf(stderr, "TLV parse error %d\n", ret);
continue;
}
int min_val = snd_ctl_elem_info_get_min(elem_info);
int max_val = snd_ctl_elem_info_get_max(elem_info);
ret = snd_tlv_get_dB_range(tlv, min_val, max_val, &min_dB, &max_dB);
if (ret != 0) {
fprintf(stderr, "TLV range error %d\n", ret);
continue;
}
alsa_elem.min_val = min_val;
alsa_elem.max_val = max_val;
alsa_elem.min_dB = min_dB / 100;
alsa_elem.max_dB = max_dB / 100;
}
}
if (card->elems->len <= alsa_elem.numid)
g_array_set_size(card->elems, alsa_elem.numid + 1);
g_array_index(card->elems, struct alsa_elem, alsa_elem.numid) = alsa_elem;
}
// free the ALSA list
snd_ctl_elem_list_free_space(list);
snd_ctl_elem_list_free(list);
}
static void alsa_elem_change(struct alsa_elem *elem) {
if (!elem || !elem->callbacks)
return;
for (GList *l = elem->callbacks; l; l = l->next) {
struct alsa_elem_callback *cb = (struct alsa_elem_callback *)l->data;
if (!cb || !cb->callback)
continue;
cb->callback(elem, cb->data);
}
}
static gboolean alsa_card_callback(
GIOChannel *source,
GIOCondition condition,
void *data
) {
struct alsa_card *card = data;
snd_ctl_event_t *event;
unsigned int mask;
int err, numid;
struct alsa_elem *elem;
snd_ctl_event_alloca(&event);
if (!card->handle) {
printf("oops, no card handle??\n");
return 0;
}
err = snd_ctl_read(card->handle, event);
if (err == 0) {
printf("alsa_card_callback nothing to read??\n");
return 0;
}
if (err < 0) {
if (err == -ENODEV)
return 0;
printf("card_callback_error %d\n", err);
exit(1);
}
if (snd_ctl_event_get_type(event) != SND_CTL_EVENT_ELEM)
return 1;
numid = snd_ctl_event_elem_get_numid(event);
elem = &g_array_index(card->elems, struct alsa_elem, numid);
if (elem->numid != numid)
return 1;
mask = snd_ctl_event_elem_get_mask(event);
if (mask & (SND_CTL_EVENT_MASK_VALUE | SND_CTL_EVENT_MASK_INFO))
alsa_elem_change(elem);
return 1;
}
// go through the alsa_cards array and look for an entry with the
// matching card_num
static struct alsa_card *find_card_by_card_num(int card_num) {
for (int i = 0; i < alsa_cards->len; i++) {
struct alsa_card **card_ptr =
&g_array_index(alsa_cards, struct alsa_card *, i);
if (!*card_ptr)
continue;
if ((*card_ptr)->num == card_num)
return *card_ptr;
}
return NULL;
}
// create a new entry in the alsa_cards array (either an unused entry
// or add a new entry to the end)
struct alsa_card *card_create(int card_num) {
int i, found = 0;
struct alsa_card **card_ptr;
// look for an unused entry
for (i = 0; i < alsa_cards->len; i++) {
card_ptr = &g_array_index(alsa_cards, struct alsa_card *, i);
if (!*card_ptr) {
found = 1;
break;
}
}
// no unused entry? extend the array
if (!found) {
g_array_set_size(alsa_cards, i + 1);
card_ptr = &g_array_index(alsa_cards, struct alsa_card *, i);
}
*card_ptr = calloc(1, sizeof(struct alsa_card));
struct alsa_card *card = *card_ptr;
card->num = card_num;
card->elems = g_array_new(FALSE, TRUE, sizeof(struct alsa_elem));
return card;
}
static void card_destroy_callback(void *data) {
struct alsa_card *card = data;
// close the windows associated with this card
destroy_card_window(card);
// TODO: there is more to free
free(card->device);
free(card->name);
free(card);
// go through the alsa_cards array and clear the entry for this card
for (int i = 0; i < alsa_cards->len; i++) {
struct alsa_card **card_ptr =
&g_array_index(alsa_cards, struct alsa_card *, i);
if (*card_ptr == card)
*card_ptr = NULL;
}
}
static void alsa_add_card_callback(struct alsa_card *card) {
card->io_channel = g_io_channel_unix_new(card->pfd.fd);
card->event_source_id = g_io_add_watch_full(
card->io_channel, 0,
G_IO_IN | G_IO_ERR | G_IO_HUP,
alsa_card_callback, card, card_destroy_callback
);
}
static void alsa_subscribe(struct alsa_card *card) {
int count = snd_ctl_poll_descriptors_count(card->handle);
if (count != 1) {
printf("poll descriptors %d != 1", count);
exit(1);
}
snd_ctl_subscribe_events(card->handle, 1);
snd_ctl_poll_descriptors(card->handle, &card->pfd, 1);
}
void alsa_scan_cards(void) {
snd_ctl_card_info_t *info;
snd_ctl_t *ctl;
int card_num = -1;
char device[32];
struct alsa_card *card;
snd_ctl_card_info_alloca(&info);
while (1) {
int err = snd_card_next(&card_num);
if (err < 0)
fatal_alsa_error("snd_card_next", err);
if (card_num < 0)
break;
snprintf(device, 32, "hw:%d", card_num);
err = snd_ctl_open(&ctl, device, 0);
if (err < 0)
goto next;
err = snd_ctl_card_info(ctl, info);
if (err < 0)
goto next;
if (strncmp(snd_ctl_card_info_get_name(info), "Scarlett", 8) != 0 &&
strncmp(snd_ctl_card_info_get_name(info), "Clarett", 7) != 0)
goto next;
// is there already an entry for this card in alsa_cards?
card = find_card_by_card_num(card_num);
// yes: skip
if (card)
goto next;
// no: create
card = card_create(card_num);
card->device = strdup(device);
card->name = strdup(snd_ctl_card_info_get_name(info));
card->handle = ctl;
alsa_get_elem_list(card);
alsa_subscribe(card);
create_card_window(card);
alsa_add_card_callback(card);
continue;
next:
snd_ctl_close(ctl);
}
}
// inotify
static gboolean inotify_callback(
GIOChannel *source,
GIOCondition condition,
void *data
) {
char buf[4096] __attribute__ ((aligned(__alignof__(struct inotify_event))));
const struct inotify_event *event;
int len;
len = read(inotify_fd, &buf, sizeof(buf));
if (len < 0) {
perror("inotify read");
exit(1);
}
for (
event = (struct inotify_event *)buf;
(char *)event < buf + len;
event++
) {
if (event->mask & IN_CREATE &&
len &&
strncmp(event->name, "control", 7) == 0) {
// can't rescan for new cards too fast
sleep(1);
alsa_scan_cards();
}
}
return TRUE;
}
void alsa_inotify_init(void) {
GIOChannel *io_channel;
inotify_fd = inotify_init();
inotify_wd = inotify_add_watch(inotify_fd, "/dev/snd", IN_CREATE);
io_channel = g_io_channel_unix_new(inotify_fd);
g_io_add_watch_full(
io_channel, 0,
G_IO_IN | G_IO_ERR | G_IO_HUP,
inotify_callback, NULL, NULL
);
}