mirror of
https://github.com/alsa-project/alsa-utils
synced 2025-01-05 16:16:42 +01:00
alsactl: use a list of source for event dispatcher instead of an array of source
In a previous commit, handlers of control nodes are maintained by link list. This commit uses the list to register/unregister event sources to dispatcher. Signed-off-by: Takashi Sakamoto <o-takashi@sakamocchi.jp> Signed-off-by: Jaroslav Kysela <perex@perex.cz>
This commit is contained in:
parent
5e010366d3
commit
89ab7146ba
1 changed files with 32 additions and 46 deletions
|
@ -68,6 +68,8 @@ static const char *snd_card_iterator_next(struct snd_card_iterator *iter)
|
||||||
static void remove_source_entry(struct src_entry *entry)
|
static void remove_source_entry(struct src_entry *entry)
|
||||||
{
|
{
|
||||||
list_del(&entry->list);
|
list_del(&entry->list);
|
||||||
|
if (entry->handle)
|
||||||
|
snd_ctl_close(entry->handle);
|
||||||
free(entry->name);
|
free(entry->name);
|
||||||
free(entry);
|
free(entry);
|
||||||
}
|
}
|
||||||
|
@ -138,7 +140,7 @@ static int open_ctl(const char *name, snd_ctl_t **ctlp)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int print_event(int card, snd_ctl_t *ctl)
|
static int print_event(snd_ctl_t *ctl, const char *name)
|
||||||
{
|
{
|
||||||
snd_ctl_event_t *event;
|
snd_ctl_event_t *event;
|
||||||
unsigned int mask;
|
unsigned int mask;
|
||||||
|
@ -152,9 +154,8 @@ static int print_event(int card, snd_ctl_t *ctl)
|
||||||
if (snd_ctl_event_get_type(event) != SND_CTL_EVENT_ELEM)
|
if (snd_ctl_event_get_type(event) != SND_CTL_EVENT_ELEM)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
if (card >= 0)
|
printf("node %s, #%d (%i,%i,%i,%s,%i)",
|
||||||
printf("card %d, ", card);
|
name,
|
||||||
printf("#%d (%i,%i,%i,%s,%i)",
|
|
||||||
snd_ctl_event_elem_get_numid(event),
|
snd_ctl_event_elem_get_numid(event),
|
||||||
snd_ctl_event_elem_get_interface(event),
|
snd_ctl_event_elem_get_interface(event),
|
||||||
snd_ctl_event_elem_get_device(event),
|
snd_ctl_event_elem_get_device(event),
|
||||||
|
@ -181,36 +182,28 @@ static int print_event(int card, snd_ctl_t *ctl)
|
||||||
}
|
}
|
||||||
|
|
||||||
static int operate_dispatcher(int epfd, uint32_t op, struct epoll_event *epev,
|
static int operate_dispatcher(int epfd, uint32_t op, struct epoll_event *epev,
|
||||||
snd_ctl_t *ctl)
|
struct src_entry *entry)
|
||||||
{
|
{
|
||||||
struct pollfd *pfds;
|
struct pollfd *pfds;
|
||||||
int count;
|
int count;
|
||||||
unsigned int pfd_count;
|
|
||||||
int i;
|
int i;
|
||||||
int err;
|
int err = 0;
|
||||||
|
|
||||||
count = snd_ctl_poll_descriptors_count(ctl);
|
pfds = calloc(entry->pfd_count, sizeof(*pfds));
|
||||||
if (count < 0)
|
|
||||||
return count;
|
|
||||||
if (count == 0)
|
|
||||||
return -ENXIO;
|
|
||||||
pfd_count = count;
|
|
||||||
|
|
||||||
pfds = calloc(pfd_count, sizeof(*pfds));
|
|
||||||
if (!pfds)
|
if (!pfds)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
count = snd_ctl_poll_descriptors(ctl, pfds, pfd_count);
|
count = snd_ctl_poll_descriptors(entry->handle, pfds, entry->pfd_count);
|
||||||
if (count < 0) {
|
if (count < 0) {
|
||||||
err = count;
|
err = count;
|
||||||
goto end;
|
goto end;
|
||||||
}
|
}
|
||||||
if (count != pfd_count) {
|
if (count != entry->pfd_count) {
|
||||||
err = -EIO;
|
err = -EIO;
|
||||||
goto end;
|
goto end;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (i = 0; i < pfd_count; ++i) {
|
for (i = 0; i < entry->pfd_count; ++i) {
|
||||||
err = epoll_ctl(epfd, op, pfds[i].fd, epev);
|
err = epoll_ctl(epfd, op, pfds[i].fd, epev);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
break;
|
break;
|
||||||
|
@ -220,17 +213,16 @@ end:
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int prepare_dispatcher(int epfd, snd_ctl_t **ctls, int ncards)
|
static int prepare_dispatcher(int epfd, struct list_head *srcs)
|
||||||
{
|
{
|
||||||
int i;
|
struct src_entry *entry;
|
||||||
int err = 0;
|
int err = 0;
|
||||||
|
|
||||||
for (i = 0; i < ncards; ++i) {
|
list_for_each_entry(entry, srcs, list) {
|
||||||
snd_ctl_t *ctl = ctls[i];
|
|
||||||
struct epoll_event ev;
|
struct epoll_event ev;
|
||||||
ev.events = EPOLLIN;
|
ev.events = EPOLLIN;
|
||||||
ev.data.ptr = (void *)ctl;
|
ev.data.ptr = (void *)entry;
|
||||||
err = operate_dispatcher(epfd, EPOLL_CTL_ADD, &ev, ctl);
|
err = operate_dispatcher(epfd, EPOLL_CTL_ADD, &ev, entry);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -238,11 +230,17 @@ static int prepare_dispatcher(int epfd, snd_ctl_t **ctls, int ncards)
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int run_dispatcher(int epfd, unsigned int max_ev_count, int show_cards)
|
static int run_dispatcher(int epfd, struct list_head *srcs)
|
||||||
{
|
{
|
||||||
|
struct src_entry *entry;
|
||||||
|
unsigned int max_ev_count;
|
||||||
struct epoll_event *epev;
|
struct epoll_event *epev;
|
||||||
int err = 0;
|
int err = 0;
|
||||||
|
|
||||||
|
max_ev_count = 0;
|
||||||
|
list_for_each_entry(entry, srcs, list)
|
||||||
|
max_ev_count += entry->pfd_count;
|
||||||
|
|
||||||
epev = calloc(max_ev_count, sizeof(*epev));
|
epev = calloc(max_ev_count, sizeof(*epev));
|
||||||
if (!epev)
|
if (!epev)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
@ -261,10 +259,9 @@ static int run_dispatcher(int epfd, unsigned int max_ev_count, int show_cards)
|
||||||
|
|
||||||
for (i = 0; i < count; ++i) {
|
for (i = 0; i < count; ++i) {
|
||||||
struct epoll_event *ev = epev + i;
|
struct epoll_event *ev = epev + i;
|
||||||
snd_ctl_t *handle = (snd_ctl_t *)ev->data.ptr;
|
struct src_entry *entry = (struct src_entry *)ev->data.ptr;
|
||||||
|
|
||||||
if (ev->events & EPOLLIN)
|
if (ev->events & EPOLLIN)
|
||||||
print_event(show_cards ? i : -1, handle);
|
print_event(entry->handle, entry->name);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -273,14 +270,12 @@ static int run_dispatcher(int epfd, unsigned int max_ev_count, int show_cards)
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void clear_dispatcher(int epfd, snd_ctl_t **ctls, int ncards)
|
static void clear_dispatcher(int epfd, struct list_head *srcs)
|
||||||
{
|
{
|
||||||
int i;
|
struct src_entry *entry;
|
||||||
|
|
||||||
for (i = 0; i < ncards; ++i) {
|
list_for_each_entry(entry, srcs, list)
|
||||||
snd_ctl_t *ctl = ctls[i];
|
operate_dispatcher(epfd, EPOLL_CTL_DEL, NULL, entry);
|
||||||
operate_dispatcher(epfd, EPOLL_CTL_DEL, NULL, ctl);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int monitor(const char *name)
|
int monitor(const char *name)
|
||||||
|
@ -288,9 +283,7 @@ int monitor(const char *name)
|
||||||
LIST_HEAD(srcs);
|
LIST_HEAD(srcs);
|
||||||
snd_ctl_t *ctls[MAX_CARDS] = {0};
|
snd_ctl_t *ctls[MAX_CARDS] = {0};
|
||||||
int ncards = 0;
|
int ncards = 0;
|
||||||
int show_cards;
|
|
||||||
int epfd;
|
int epfd;
|
||||||
int i;
|
|
||||||
int err = 0;
|
int err = 0;
|
||||||
|
|
||||||
epfd = epoll_create(1);
|
epfd = epoll_create(1);
|
||||||
|
@ -311,7 +304,6 @@ int monitor(const char *name)
|
||||||
goto error;
|
goto error;
|
||||||
ncards++;
|
ncards++;
|
||||||
}
|
}
|
||||||
show_cards = 1;
|
|
||||||
} else {
|
} else {
|
||||||
err = open_ctl(name, &ctls[0]);
|
err = open_ctl(name, &ctls[0]);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
|
@ -320,20 +312,14 @@ int monitor(const char *name)
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
goto error;
|
goto error;
|
||||||
ncards++;
|
ncards++;
|
||||||
show_cards = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
err = prepare_dispatcher(epfd, ctls, ncards);
|
err = prepare_dispatcher(epfd, &srcs);
|
||||||
if (err >= 0)
|
if (err >= 0)
|
||||||
err = run_dispatcher(epfd, ncards, show_cards);
|
err = run_dispatcher(epfd, &srcs);
|
||||||
clear_dispatcher(epfd, ctls, ncards);
|
clear_dispatcher(epfd, &srcs);
|
||||||
|
|
||||||
error:
|
error:
|
||||||
clear_source_list(&srcs);
|
clear_source_list(&srcs);
|
||||||
for (i = 0; i < ncards; i++) {
|
|
||||||
if (ctls[i])
|
|
||||||
snd_ctl_close(ctls[i]);
|
|
||||||
}
|
|
||||||
|
|
||||||
close(epfd);
|
close(epfd);
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue