Blame src/async.c

Packit Service db8eaa
/**
Packit Service db8eaa
 * \file async.c
Packit Service db8eaa
 * \brief Async notification helpers
Packit Service db8eaa
 * \author Abramo Bagnara <abramo@alsa-project.org>
Packit Service db8eaa
 * \date 2001
Packit Service db8eaa
 */
Packit Service db8eaa
/*
Packit Service db8eaa
 *  Async notification helpers
Packit Service db8eaa
 *  Copyright (c) 2001 by Abramo Bagnara <abramo@alsa-project.org>
Packit Service db8eaa
 *
Packit Service db8eaa
 *   This library is free software; you can redistribute it and/or modify
Packit Service db8eaa
 *   it under the terms of the GNU Lesser General Public License as
Packit Service db8eaa
 *   published by the Free Software Foundation; either version 2.1 of
Packit Service db8eaa
 *   the License, or (at your option) any later version.
Packit Service db8eaa
 *
Packit Service db8eaa
 *   This program is distributed in the hope that it will be useful,
Packit Service db8eaa
 *   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service db8eaa
 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit Service db8eaa
 *   GNU Lesser General Public License for more details.
Packit Service db8eaa
 *
Packit Service db8eaa
 *   You should have received a copy of the GNU Lesser General Public
Packit Service db8eaa
 *   License along with this library; if not, write to the Free Software
Packit Service db8eaa
 *   Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
Packit Service db8eaa
 *
Packit Service db8eaa
 */
Packit Service db8eaa
Packit Service db8eaa
#include "pcm/pcm_local.h"
Packit Service db8eaa
#include "control/control_local.h"
Packit Service db8eaa
#include <signal.h>
Packit Service db8eaa
Packit Service db8eaa
static struct sigaction previous_action;
Packit Service db8eaa
#define MAX_SIG_FUNCTION_CODE 10 /* i.e. SIG_DFL SIG_IGN SIG_HOLD et al */
Packit Service db8eaa
Packit Service db8eaa
#ifdef SND_ASYNC_RT_SIGNAL
Packit Service db8eaa
/** async signal number */
Packit Service db8eaa
static int snd_async_signo;
Packit Service db8eaa
Packit Service db8eaa
void snd_async_init(void) __attribute__ ((constructor));
Packit Service db8eaa
Packit Service db8eaa
void snd_async_init(void)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_async_signo = __libc_allocate_rtsig(0);
Packit Service db8eaa
	if (snd_async_signo < 0) {
Packit Service db8eaa
		SNDERR("Unable to find a RT signal to use for snd_async");
Packit Service db8eaa
		exit(1);
Packit Service db8eaa
	}
Packit Service db8eaa
}
Packit Service db8eaa
#else
Packit Service db8eaa
/** async signal number */
Packit Service db8eaa
static const int snd_async_signo = SIGIO;
Packit Service db8eaa
#endif
Packit Service db8eaa
Packit Service db8eaa
static LIST_HEAD(snd_async_handlers);
Packit Service db8eaa
Packit Service db8eaa
static void snd_async_handler(int signo ATTRIBUTE_UNUSED, siginfo_t *siginfo, void *context ATTRIBUTE_UNUSED)
Packit Service db8eaa
{
Packit Service db8eaa
	int fd;
Packit Service db8eaa
	struct list_head *i;
Packit Service db8eaa
	//assert(siginfo->si_code == SI_SIGIO);
Packit Service db8eaa
	if (signo == SIGIO
Packit Service db8eaa
	 && (unsigned long)(previous_action.sa_sigaction) > MAX_SIG_FUNCTION_CODE)
Packit Service db8eaa
		previous_action.sa_sigaction(signo, siginfo, context);
Packit Service db8eaa
	fd = siginfo->si_fd;
Packit Service db8eaa
	list_for_each(i, &snd_async_handlers) {
Packit Service db8eaa
		snd_async_handler_t *h = list_entry(i, snd_async_handler_t, glist);
Packit Service db8eaa
		if (h->fd == fd && h->callback)
Packit Service db8eaa
			h->callback(h);
Packit Service db8eaa
	}
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
/**
Packit Service db8eaa
 * \brief Registers an async handler.
Packit Service db8eaa
 * \param handler The function puts the pointer to the new async handler
Packit Service db8eaa
 *                object at the address specified by \p handler.
Packit Service db8eaa
 * \param fd The file descriptor to be associated with the callback.
Packit Service db8eaa
 * \param callback The async callback function.
Packit Service db8eaa
 * \param private_data Private data for the async callback function.
Packit Service db8eaa
 * \result Zero if successful, otherwise a negative error code.
Packit Service db8eaa
 *
Packit Service db8eaa
 * This function associates the callback function with the given file,
Packit Service db8eaa
 * and saves this association in a \c snd_async_handler_t object.
Packit Service db8eaa
 *
Packit Service db8eaa
 * Whenever the \c SIGIO signal is raised for the file \p fd, the callback
Packit Service db8eaa
 * function will be called with its parameter pointing to the async handler
Packit Service db8eaa
 * object returned by this function.
Packit Service db8eaa
 *
Packit Service db8eaa
 * The ALSA \c sigaction handler for the \c SIGIO signal automatically
Packit Service db8eaa
 * multiplexes the notifications to the registered async callbacks.
Packit Service db8eaa
 * However, the application is responsible for instructing the device driver
Packit Service db8eaa
 * to generate the \c SIGIO signal.
Packit Service db8eaa
 *
Packit Service db8eaa
 * The \c SIGIO signal may have been replaced with another signal,
Packit Service db8eaa
 * see #snd_async_handler_get_signo.
Packit Service db8eaa
 *
Packit Service db8eaa
 * When the async handler isn't needed anymore, you must delete it with
Packit Service db8eaa
 * #snd_async_del_handler.
Packit Service db8eaa
 *
Packit Service db8eaa
 * \see snd_async_add_pcm_handler, snd_async_add_ctl_handler
Packit Service db8eaa
 */
Packit Service db8eaa
int snd_async_add_handler(snd_async_handler_t **handler, int fd, 
Packit Service db8eaa
			  snd_async_callback_t callback, void *private_data)
Packit Service db8eaa
{
Packit Service db8eaa
	snd_async_handler_t *h;
Packit Service db8eaa
	int was_empty;
Packit Service db8eaa
	assert(handler);
Packit Service db8eaa
	h = malloc(sizeof(*h));
Packit Service db8eaa
	if (!h)
Packit Service db8eaa
		return -ENOMEM;
Packit Service db8eaa
	h->fd = fd;
Packit Service db8eaa
	h->callback = callback;
Packit Service db8eaa
	h->private_data = private_data;
Packit Service db8eaa
	was_empty = list_empty(&snd_async_handlers);
Packit Service db8eaa
	list_add_tail(&h->glist, &snd_async_handlers);
Packit Service db8eaa
	INIT_LIST_HEAD(&h->hlist);
Packit Service db8eaa
	*handler = h;
Packit Service db8eaa
	if (was_empty) {
Packit Service db8eaa
		int err;
Packit Service db8eaa
		struct sigaction act;
Packit Service db8eaa
		memset(&act, 0, sizeof(act));
Packit Service db8eaa
		act.sa_flags = SA_RESTART | SA_SIGINFO;
Packit Service db8eaa
		act.sa_sigaction = snd_async_handler;
Packit Service db8eaa
		sigemptyset(&act.sa_mask);
Packit Service db8eaa
		assert(!previous_action.sa_sigaction);
Packit Service db8eaa
		err = sigaction(snd_async_signo, &act, &previous_action);
Packit Service db8eaa
		if (err < 0) {
Packit Service db8eaa
			SYSERR("sigaction");
Packit Service db8eaa
			return -errno;
Packit Service db8eaa
		}
Packit Service db8eaa
	}
Packit Service db8eaa
	return 0;
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
/**
Packit Service db8eaa
 * \brief Deletes an async handler.
Packit Service db8eaa
 * \param handler Handle of the async handler to delete.
Packit Service db8eaa
 * \result Zero if successful, otherwise a negative error code.
Packit Service db8eaa
 */
Packit Service db8eaa
int snd_async_del_handler(snd_async_handler_t *handler)
Packit Service db8eaa
{
Packit Service db8eaa
	int err = 0;
Packit Service db8eaa
	int was_empty = list_empty(&snd_async_handlers);
Packit Service db8eaa
	assert(handler);
Packit Service db8eaa
	list_del(&handler->glist);
Packit Service db8eaa
	if (!was_empty
Packit Service db8eaa
	 && list_empty(&snd_async_handlers)) {
Packit Service db8eaa
		err = sigaction(snd_async_signo, &previous_action, NULL);
Packit Service db8eaa
		if (err < 0) {
Packit Service db8eaa
			SYSERR("sigaction");
Packit Service db8eaa
			return -errno;
Packit Service db8eaa
		}
Packit Service db8eaa
		memset(&previous_action, 0, sizeof(previous_action));
Packit Service db8eaa
	}
Packit Service db8eaa
	if (handler->type == SND_ASYNC_HANDLER_GENERIC)
Packit Service db8eaa
		goto _end;
Packit Service db8eaa
	if (!list_empty(&handler->hlist))
Packit Service db8eaa
		list_del(&handler->hlist);
Packit Service db8eaa
	if (!list_empty(&handler->hlist))
Packit Service db8eaa
		goto _end;
Packit Service db8eaa
	switch (handler->type) {
Packit Service db8eaa
#ifdef BUILD_PCM
Packit Service db8eaa
	case SND_ASYNC_HANDLER_PCM:
Packit Service db8eaa
		err = snd_pcm_async(handler->u.pcm, -1, 1);
Packit Service db8eaa
		break;
Packit Service db8eaa
#endif
Packit Service db8eaa
	case SND_ASYNC_HANDLER_CTL:
Packit Service db8eaa
		err = snd_ctl_async(handler->u.ctl, -1, 1);
Packit Service db8eaa
		break;
Packit Service db8eaa
	default:
Packit Service db8eaa
		assert(0);
Packit Service db8eaa
	}
Packit Service db8eaa
 _end:
Packit Service db8eaa
	free(handler);
Packit Service db8eaa
	return err;
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
/**
Packit Service db8eaa
 * \brief Returns the signal number assigned to an async handler.
Packit Service db8eaa
 * \param handler Handle to an async handler.
Packit Service db8eaa
 * \result The signal number if successful, otherwise a negative error code.
Packit Service db8eaa
 *
Packit Service db8eaa
 * The signal number for async handlers usually is \c SIGIO,
Packit Service db8eaa
 * but wizards can redefine it to a realtime signal
Packit Service db8eaa
 * when compiling the ALSA library.
Packit Service db8eaa
 */
Packit Service db8eaa
int snd_async_handler_get_signo(snd_async_handler_t *handler)
Packit Service db8eaa
{
Packit Service db8eaa
	assert(handler);
Packit Service db8eaa
	return snd_async_signo;
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
/**
Packit Service db8eaa
 * \brief Returns the file descriptor assigned to an async handler.
Packit Service db8eaa
 * \param handler Handle to an async handler.
Packit Service db8eaa
 * \result The file descriptor if successful, otherwise a negative error code.
Packit Service db8eaa
 */
Packit Service db8eaa
int snd_async_handler_get_fd(snd_async_handler_t *handler)
Packit Service db8eaa
{
Packit Service db8eaa
	assert(handler);
Packit Service db8eaa
	return handler->fd;
Packit Service db8eaa
}
Packit Service db8eaa
Packit Service db8eaa
/**
Packit Service db8eaa
 * \brief Returns the private data assigned to an async handler.
Packit Service db8eaa
 * \param handler Handle to an async handler.
Packit Service db8eaa
 * \result The \c private_data value registered with the async handler.
Packit Service db8eaa
 */
Packit Service db8eaa
void *snd_async_handler_get_callback_private(snd_async_handler_t *handler)
Packit Service db8eaa
{
Packit Service db8eaa
	assert(handler);
Packit Service db8eaa
	return handler->private_data;
Packit Service db8eaa
}
Packit Service db8eaa