/** * \file timer/timer.c * \brief Timer Interface * \author Jaroslav Kysela * \date 1998-2001 * * Timer Interface is designed to access timers. * See \ref timer page for more details. */ /* * Timer Interface - main file * Copyright (c) 1998-2001 by Jaroslav Kysela * * * This library is free software; you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as * published by the Free Software Foundation; either version 2.1 of * the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA * */ /*! \page timer Timer interface

Timer interface is designed to use internal timers in sound hardware, but it can be driven with any timer. \section timer_general_overview General overview The timer implementation uses ring buffer to store information about timing events. In this buffer is recorded count of ticks and current tick resolution in nanoseconds. \section timer_open Opening Timer devices can be opened in two ways. When #SND_TIMER_OPEN_NONBLOCK flag is used, then the open functions return immediately with -EBUSY error code when resources are occupied with another application. When #SND_TIMER_OPEN_NONBLOCK is not used (by default) the open functions block the application requesting device until resources are not free. \section timer_events Events Events are read via snd_timer_read() function. \section timer_examples Examples The full featured examples with cross-links: \par Simple timer test program \ref example_test_timer "example code" \par This example shows opening a timer device and reading of timer events. */ /** * \example ../test/timer.c * \anchor example_test_timer */ #include "timer_local.h" #include static int snd_timer_open_conf(snd_timer_t **timer, const char *name, snd_config_t *timer_root, snd_config_t *timer_conf, int mode) { const char *str; char buf[256], errbuf[256]; int err; snd_config_t *conf, *type_conf = NULL; snd_config_iterator_t i, next; const char *id; const char *lib = NULL, *open_name = NULL; int (*open_func)(snd_timer_t **, const char *, snd_config_t *, snd_config_t *, int) = NULL; #ifndef PIC extern void *snd_timer_open_symbols(void); #endif void *h = NULL; if (snd_config_get_type(timer_conf) != SND_CONFIG_TYPE_COMPOUND) { if (name) SNDERR("Invalid type for TIMER %s definition", name); else SNDERR("Invalid type for TIMER definition"); return -EINVAL; } err = snd_config_search(timer_conf, "type", &conf); if (err < 0) { SNDERR("type is not defined"); return err; } err = snd_config_get_id(conf, &id); if (err < 0) { SNDERR("unable to get id"); return err; } err = snd_config_get_string(conf, &str); if (err < 0) { SNDERR("Invalid type for %s", id); return err; } err = snd_config_search_definition(timer_root, "timer_type", str, &type_conf); if (err >= 0) { if (snd_config_get_type(type_conf) != SND_CONFIG_TYPE_COMPOUND) { SNDERR("Invalid type for TIMER type %s definition", str); goto _err; } snd_config_for_each(i, next, type_conf) { snd_config_t *n = snd_config_iterator_entry(i); const char *id; if (snd_config_get_id(n, &id) < 0) continue; if (strcmp(id, "comment") == 0) continue; if (strcmp(id, "lib") == 0) { err = snd_config_get_string(n, &lib); if (err < 0) { SNDERR("Invalid type for %s", id); goto _err; } continue; } if (strcmp(id, "open") == 0) { err = snd_config_get_string(n, &open_name); if (err < 0) { SNDERR("Invalid type for %s", id); goto _err; } continue; } SNDERR("Unknown field %s", id); err = -EINVAL; goto _err; } } if (!open_name) { open_name = buf; snprintf(buf, sizeof(buf), "_snd_timer_%s_open", str); } #ifndef PIC snd_timer_open_symbols(); #endif h = INTERNAL(snd_dlopen)(lib, RTLD_NOW, errbuf, sizeof(errbuf)); if (h) open_func = snd_dlsym(h, open_name, SND_DLSYM_VERSION(SND_TIMER_DLSYM_VERSION)); err = 0; if (!h) { SNDERR("Cannot open shared library %s (%s)", lib, errbuf); err = -ENOENT; } else if (!open_func) { SNDERR("symbol %s is not defined inside %s", open_name, lib); snd_dlclose(h); err = -ENXIO; } _err: if (type_conf) snd_config_delete(type_conf); if (! err) { err = open_func(timer, name, timer_root, timer_conf, mode); if (err < 0) snd_dlclose(h); else (*timer)->dl_handle = h; } return err; } static int snd_timer_open_noupdate(snd_timer_t **timer, snd_config_t *root, const char *name, int mode) { int err; snd_config_t *timer_conf; err = snd_config_search_definition(root, "timer", name, &timer_conf); if (err < 0) { SNDERR("Unknown timer %s", name); return err; } err = snd_timer_open_conf(timer, name, root, timer_conf, mode); snd_config_delete(timer_conf); return err; } /** * \brief Opens a new connection to the timer interface. * \param timer Returned handle (NULL if not wanted) * \param name ASCII identifier of the timer handle * \param mode Open mode * \return 0 on success otherwise a negative error code * * Opens a new connection to the timer interface specified with * an ASCII identifier and mode. */ int snd_timer_open(snd_timer_t **timer, const char *name, int mode) { snd_config_t *top; int err; assert(timer && name); if (_snd_is_ucm_device(name)) { name = uc_mgr_alibcfg_by_device(&top, name); if (name == NULL) return -ENODEV; } else { err = snd_config_update_ref(&top); if (err < 0) return err; } err = snd_timer_open_noupdate(timer, top, name, mode); snd_config_unref(top); return err; } /** * \brief Opens a new connection to the timer interface using local configuration * \param timer Returned handle (NULL if not wanted) * \param name ASCII identifier of the timer handle * \param mode Open mode * \param lconf Local configuration * \return 0 on success otherwise a negative error code * * Opens a new connection to the timer interface specified with * an ASCII identifier and mode. */ int snd_timer_open_lconf(snd_timer_t **timer, const char *name, int mode, snd_config_t *lconf) { assert(timer && name && lconf); return snd_timer_open_noupdate(timer, lconf, name, mode); } /** * \brief close timer handle * \param timer timer handle * \return 0 on success otherwise a negative error code * * Closes the specified timer handle and frees all associated * resources. */ int snd_timer_close(snd_timer_t *timer) { int err; assert(timer); while (!list_empty(&timer->async_handlers)) { snd_async_handler_t *h = list_entry(timer->async_handlers.next, snd_async_handler_t, hlist); snd_async_del_handler(h); } err = timer->ops->close(timer); if (timer->dl_handle) snd_dlclose(timer->dl_handle); free(timer->name); free(timer); return err; } /** * \brief get identifier of timer handle * \param timer a timer handle * \return ascii identifier of timer handle * * Returns the ASCII identifier of given timer handle. It's the same * identifier specified in snd_timer_open(). */ const char *snd_timer_name(snd_timer_t *timer) { assert(timer); return timer->name; } /** * \brief get type of timer handle * \param timer a timer handle * \return type of timer handle * * Returns the type #snd_timer_type_t of given timer handle. */ snd_timer_type_t snd_timer_type(snd_timer_t *timer) { assert(timer); return timer->type; } /** * \brief Add an async handler for a timer * \param handler Returned handler handle * \param timer timer handle * \param callback Callback function * \param private_data Callback private data * \return 0 otherwise a negative error code on failure * * The asynchronous callback is called when new timer event occurs. */ int snd_async_add_timer_handler(snd_async_handler_t **handler, snd_timer_t *timer, snd_async_callback_t callback, void *private_data) { int err; int was_empty; snd_async_handler_t *h; err = snd_async_add_handler(&h, timer->poll_fd, callback, private_data); if (err < 0) return err; h->type = SND_ASYNC_HANDLER_TIMER; h->u.timer = timer; was_empty = list_empty(&timer->async_handlers); list_add_tail(&h->hlist, &timer->async_handlers); if (was_empty) { err = snd_timer_async(timer, snd_async_handler_get_signo(h), getpid()); if (err < 0) { snd_async_del_handler(h); return err; } } *handler = h; return 0; } /** * \brief Return timer handle related to an async handler * \param handler Async handler handle * \return timer handle */ snd_timer_t *snd_async_handler_get_timer(snd_async_handler_t *handler) { if (handler->type != SND_ASYNC_HANDLER_TIMER) { SNDMSG("invalid handler type %d", handler->type); return NULL; } return handler->u.timer; } /** * \brief get count of poll descriptors for timer handle * \param timer timer handle * \return count of poll descriptors */ int snd_timer_poll_descriptors_count(snd_timer_t *timer) { assert(timer); return 1; } /** * \brief get poll descriptors * \param timer timer handle * \param pfds array of poll descriptors * \param space space in the poll descriptor array * \return count of filled descriptors */ int snd_timer_poll_descriptors(snd_timer_t *timer, struct pollfd *pfds, unsigned int space) { assert(timer); if (space >= 1) { pfds->fd = timer->poll_fd; switch (timer->mode & O_ACCMODE) { case O_WRONLY: pfds->events = POLLOUT|POLLERR|POLLNVAL; break; case O_RDONLY: pfds->events = POLLIN|POLLERR|POLLNVAL; break; case O_RDWR: pfds->events = POLLOUT|POLLIN|POLLERR|POLLNVAL; break; default: return -EIO; } return 1; } return 0; } /** * \brief get returned events from poll descriptors * \param timer timer handle * \param pfds array of poll descriptors * \param nfds count of poll descriptors * \param revents returned events * \return zero if success, otherwise a negative error code */ int snd_timer_poll_descriptors_revents(snd_timer_t *timer, struct pollfd *pfds, unsigned int nfds, unsigned short *revents) { assert(timer && pfds && revents); if (nfds == 1) { *revents = pfds->revents; return 0; } return -EINVAL; } /** * \brief set nonblock mode * \param timer timer handle * \param nonblock 0 = block, 1 = nonblock mode * \return 0 on success otherwise a negative error code */ int snd_timer_nonblock(snd_timer_t *timer, int nonblock) { int err; assert(timer); if ((err = timer->ops->nonblock(timer, nonblock)) < 0) return err; if (nonblock) timer->mode |= SND_TIMER_OPEN_NONBLOCK; else timer->mode &= ~SND_TIMER_OPEN_NONBLOCK; return 0; } #ifndef DOC_HIDDEN /** * \brief set async mode * \param timer timer handle * \param sig Signal to raise: < 0 disable, 0 default (SIGIO) * \param pid Process ID to signal: 0 current * \return 0 on success otherwise a negative error code * * A signal is raised every period. */ int snd_timer_async(snd_timer_t *timer, int sig, pid_t pid) { assert(timer); if (sig == 0) sig = SIGIO; if (pid == 0) pid = getpid(); return timer->ops->async(timer, sig, pid); } #endif /** * \brief get size of the snd_timer_info_t structure in bytes * \return size of the snd_timer_info_t structure in bytes */ size_t snd_timer_info_sizeof() { return sizeof(snd_timer_info_t); } /** * \brief allocate a new snd_timer_info_t structure * \param info returned pointer * \return 0 on success otherwise a negative error code if fails * * Allocates a new snd_timer_info_t structure using the standard * malloc C library function. */ int snd_timer_info_malloc(snd_timer_info_t **info) { assert(info); *info = calloc(1, sizeof(snd_timer_info_t)); if (!*info) return -ENOMEM; return 0; } /** * \brief frees the snd_timer_info_t structure * \param info pointer to the snd_timer_info_t structure to free * * Frees the given snd_timer_info_t structure using the standard * free C library function. */ void snd_timer_info_free(snd_timer_info_t *info) { assert(info); free(info); } /** * \brief copy one snd_timer_info_t structure to another * \param dst destination snd_timer_info_t structure * \param src source snd_timer_info_t structure */ void snd_timer_info_copy(snd_timer_info_t *dst, const snd_timer_info_t *src) { assert(dst && src); *dst = *src; } /** * \brief determine, if timer is slave * \param info pointer to #snd_timer_info_t structure * \return nonzero if timer is slave */ int snd_timer_info_is_slave(snd_timer_info_t * info) { assert(info); return info->flags & SNDRV_TIMER_FLG_SLAVE ? 1 : 0; } /** * \brief get timer card * \param info pointer to #snd_timer_info_t structure * \return timer card number */ int snd_timer_info_get_card(snd_timer_info_t * info) { assert(info); return info->card; } /** * \brief get timer id * \param info pointer to #snd_timer_info_t structure * \return timer id */ const char *snd_timer_info_get_id(snd_timer_info_t * info) { assert(info); return (const char *)info->id; } /** * \brief get timer name * \param info pointer to #snd_timer_info_t structure * \return timer name */ const char *snd_timer_info_get_name(snd_timer_info_t * info) { assert(info); return (const char *)info->name; } /** * \brief get timer resolution in us * \param info pointer to #snd_timer_info_t structure * \return timer resolution */ long snd_timer_info_get_resolution(snd_timer_info_t * info) { assert(info); return info->resolution; } /** * \brief get information about timer handle * \param timer timer handle * \param info pointer to a snd_timer_info_t structure to be filled * \return 0 on success otherwise a negative error code */ int snd_timer_info(snd_timer_t *timer, snd_timer_info_t * info) { assert(timer); assert(info); return timer->ops->info(timer, info); } /** * \brief get size of the snd_timer_params_t structure in bytes * \return size of the snd_timer_params_t structure in bytes */ size_t snd_timer_params_sizeof() { return sizeof(snd_timer_params_t); } /** * \brief allocate a new snd_timer_params_t structure * \param params returned pointer * \return 0 on success otherwise a negative error code if fails * * Allocates a new snd_timer_params_t structure using the standard * malloc C library function. */ int snd_timer_params_malloc(snd_timer_params_t **params) { assert(params); *params = calloc(1, sizeof(snd_timer_params_t)); if (!*params) return -ENOMEM; return 0; } /** * \brief frees the snd_timer_params_t structure * \param params pointer to the snd_timer_params_t structure to free * * Frees the given snd_timer_params_t structure using the standard * free C library function. */ void snd_timer_params_free(snd_timer_params_t *params) { assert(params); free(params); } /** * \brief copy one snd_timer_params_t structure to another * \param dst destination snd_timer_params_t structure * \param src source snd_timer_params_t structure */ void snd_timer_params_copy(snd_timer_params_t *dst, const snd_timer_params_t *src) { assert(dst && src); *dst = *src; } /** * \brief set timer auto start * \param params pointer to #snd_timer_params_t structure * \param auto_start The boolean value to set */ int snd_timer_params_set_auto_start(snd_timer_params_t * params, int auto_start) { assert(params); if (auto_start) params->flags |= SNDRV_TIMER_PSFLG_AUTO; else params->flags &= ~SNDRV_TIMER_PSFLG_AUTO; return 0; } /** * \brief determine if timer has auto start flag * \param params pointer to #snd_timer_params_t structure * \return nonzero if timer has auto start flag */ int snd_timer_params_get_auto_start(snd_timer_params_t * params) { assert(params); return params->flags & SNDRV_TIMER_PSFLG_AUTO ? 1 : 0; } /** * \brief set timer exclusive use * \param params pointer to #snd_timer_params_t structure * \param exclusive The boolean value to set */ #ifndef DOXYGEN EXPORT_SYMBOL int INTERNAL(snd_timer_params_set_exclusive)(snd_timer_params_t * params, int exclusive) #else int snd_timer_params_set_exclusive(snd_timer_params_t * params, int exclusive) #endif { assert(params); if (exclusive) params->flags |= SNDRV_TIMER_PSFLG_EXCLUSIVE; else params->flags &= ~SNDRV_TIMER_PSFLG_EXCLUSIVE; return 0; } use_default_symbol_version(__snd_timer_params_set_exclusive, snd_timer_params_set_exclusive, ALSA_0.9.0); /** * \brief determine if timer has exclusive flag * \param params pointer to #snd_timer_params_t structure * \return nonzero if timer has exclusive flag */ #ifndef DOXYGEN EXPORT_SYMBOL int INTERNAL(snd_timer_params_get_exclusive)(snd_timer_params_t * params) #else int snd_timer_params_get_exclusive(snd_timer_params_t * params) #endif { assert(params); return params->flags & SNDRV_TIMER_PSFLG_EXCLUSIVE ? 1 : 0; } use_default_symbol_version(__snd_timer_params_get_exclusive, snd_timer_params_get_exclusive, ALSA_0.9.0); /** * \brief set timer early event * \param params pointer to #snd_timer_params_t structure * \param early_event The boolean value to set */ int snd_timer_params_set_early_event(snd_timer_params_t * params, int early_event) { assert(params); if (early_event) params->flags |= SNDRV_TIMER_PSFLG_EARLY_EVENT; else params->flags &= ~SNDRV_TIMER_PSFLG_EARLY_EVENT; return 0; } /** * \brief determine if timer has early event flag * \param params pointer to #snd_timer_params_t structure * \return nonzero if timer has early event flag set */ int snd_timer_params_get_early_event(snd_timer_params_t * params) { assert(params); return params->flags & SNDRV_TIMER_PSFLG_EARLY_EVENT ? 1 : 0; } /** * \brief set timer ticks * \param params pointer to #snd_timer_params_t structure * \param ticks Ticks to set */ void snd_timer_params_set_ticks(snd_timer_params_t * params, long ticks) { assert(params); params->ticks = ticks; } /** * \brief get timer ticks * \param params pointer to #snd_timer_params_t structure * \return timer ticks */ long snd_timer_params_get_ticks(snd_timer_params_t * params) { assert(params); return params->ticks; } /** * \brief set timer queue size (32-1024) * \param params pointer to #snd_timer_params_t structure * \param queue_size The queue size to set */ void snd_timer_params_set_queue_size(snd_timer_params_t * params, long queue_size) { assert(params); params->queue_size = queue_size; } /** * \brief get queue size * \param params pointer to #snd_timer_params_t structure * \return queue size */ long snd_timer_params_get_queue_size(snd_timer_params_t * params) { assert(params); return params->queue_size; } /** * \brief set timer event filter * \param params pointer to #snd_timer_params_t structure * \param filter The event filter bits to set */ #ifndef DOXYGEN EXPORT_SYMBOL void INTERNAL(snd_timer_params_set_filter)(snd_timer_params_t * params, unsigned int filter) #else void snd_timer_params_set_filter(snd_timer_params_t * params, unsigned int filter) #endif { assert(params); params->filter = filter; } use_default_symbol_version(__snd_timer_params_set_filter, snd_timer_params_set_filter, ALSA_0.9.0); /** * \brief get timer event filter * \param params pointer to #snd_timer_params_t structure * \return timer event filter */ #ifndef DOXYGEN EXPORT_SYMBOL unsigned int INTERNAL(snd_timer_params_get_filter)(snd_timer_params_t * params) #else unsigned int snd_timer_params_get_filter(snd_timer_params_t * params) #endif { assert(params); return params->filter; } use_default_symbol_version(__snd_timer_params_get_filter, snd_timer_params_get_filter, ALSA_0.9.0); /** * \brief set parameters for timer handle * \param timer timer handle * \param params pointer to a #snd_timer_params_t structure * \return 0 on success otherwise a negative error code */ int snd_timer_params(snd_timer_t *timer, snd_timer_params_t * params) { assert(timer); assert(params); return timer->ops->params(timer, params); } /** * \brief get size of the snd_timer_status_t structure in bytes * \return size of the snd_timer_status_t structure in bytes */ size_t snd_timer_status_sizeof() { return sizeof(snd_timer_status_t); } /** * \brief allocate a new snd_timer_status_t structure * \param status returned pointer * \return 0 on success otherwise a negative error code if fails * * Allocates a new snd_timer_status_t structure using the standard * malloc C library function. */ int snd_timer_status_malloc(snd_timer_status_t **status) { assert(status); *status = calloc(1, sizeof(snd_timer_status_t)); if (!*status) return -ENOMEM; return 0; } /** * \brief frees the snd_timer_status_t structure * \param status pointer to the snd_timer_status_t structure to free * * Frees the given snd_timer_status_t structure using the standard * free C library function. */ void snd_timer_status_free(snd_timer_status_t *status) { assert(status); free(status); } /** * \brief copy one snd_timer_status_t structure to another * \param dst destination snd_timer_status_t structure * \param src source snd_timer_status_t structure */ void snd_timer_status_copy(snd_timer_status_t *dst, const snd_timer_status_t *src) { assert(dst && src); *dst = *src; } /** * \brief get timestamp * \param status pointer to #snd_timer_status_t structure * \return timestamp */ snd_htimestamp_t snd_timer_status_get_timestamp(snd_timer_status_t * status) { assert(status); return status->tstamp; } /** * \brief get resolution in us * \param status pointer to #snd_timer_status_t structure * \return resolution */ long snd_timer_status_get_resolution(snd_timer_status_t * status) { assert(status); return status->resolution; } /** * \brief get master tick lost count * \param status pointer to #snd_timer_status_t structure * \return master tick lost count */ long snd_timer_status_get_lost(snd_timer_status_t * status) { assert(status); return status->lost; } /** * \brief get overrun count * \param status pointer to #snd_timer_status_t structure * \return overrun count */ long snd_timer_status_get_overrun(snd_timer_status_t * status) { assert(status); return status->overrun; } /** * \brief get count of used queue elements * \param status pointer to #snd_timer_status_t structure * \return count of used queue elements */ long snd_timer_status_get_queue(snd_timer_status_t * status) { assert(status); return status->queue; } /** * \brief get status from timer handle * \param timer timer handle * \param status pointer to a #snd_timer_status_t structure to be filled * \return 0 on success otherwise a negative error code */ int snd_timer_status(snd_timer_t *timer, snd_timer_status_t * status) { assert(timer); assert(status); return timer->ops->status(timer, status); } /** * \brief start the timer * \param timer timer handle * \return 0 on success otherwise a negative error code */ int snd_timer_start(snd_timer_t *timer) { assert(timer); return timer->ops->rt_start(timer); } /** * \brief stop the timer * \param timer timer handle * \return 0 on success otherwise a negative error code */ int snd_timer_stop(snd_timer_t *timer) { assert(timer); return timer->ops->rt_stop(timer); } /** * \brief continue the timer * \param timer timer handle * \return 0 on success otherwise a negative error code */ int snd_timer_continue(snd_timer_t *timer) { assert(timer); return timer->ops->rt_continue(timer); } /** * \brief read bytes using timer handle * \param timer timer handle * \param buffer buffer to store the input bytes * \param size input buffer size in bytes */ ssize_t snd_timer_read(snd_timer_t *timer, void *buffer, size_t size) { assert(timer); assert(((timer->mode & O_ACCMODE) == O_RDONLY) || ((timer->mode & O_ACCMODE) == O_RDWR)); assert(buffer || size == 0); return (timer->ops->read)(timer, buffer, size); } /** * \brief (DEPRECATED) get maximum timer ticks * \param info pointer to #snd_timer_info_t structure * \return maximum timer ticks */ long snd_timer_info_get_ticks(snd_timer_info_t * info) { assert(info); return 1; } #ifndef DOC_HIDDEN link_warning(snd_timer_info_get_ticks, "Warning: snd_timer_info_get_ticks is deprecated"); #endif