2005-04-17 02:20:36 +04:00
/*
* Timers abstract layer
2007-10-15 11:50:19 +04:00
* Copyright ( c ) by Jaroslav Kysela < perex @ perex . cz >
2005-04-17 02:20:36 +04:00
*
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 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 General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*
*/
# include <linux/delay.h>
# include <linux/init.h>
# include <linux/slab.h>
# include <linux/time.h>
2006-01-16 18:29:08 +03:00
# include <linux/mutex.h>
2012-01-22 20:23:42 +04:00
# include <linux/device.h>
2011-07-15 21:13:37 +04:00
# include <linux/module.h>
2005-06-23 11:09:02 +04:00
# include <linux/string.h>
2017-02-02 21:15:33 +03:00
# include <linux/sched/signal.h>
2005-04-17 02:20:36 +04:00
# include <sound/core.h>
# include <sound/timer.h>
# include <sound/control.h>
# include <sound/info.h>
# include <sound/minors.h>
# include <sound/initval.h>
# include <linux/kmod.h>
2016-09-07 16:45:31 +03:00
/* internal flags */
# define SNDRV_TIMER_IFLG_PAUSED 0x00010000
2014-02-10 12:48:47 +04:00
# if IS_ENABLED(CONFIG_SND_HRTIMER)
2010-11-18 11:53:54 +03:00
# define DEFAULT_TIMER_LIMIT 4
2005-04-17 02:20:36 +04:00
# else
# define DEFAULT_TIMER_LIMIT 1
# endif
static int timer_limit = DEFAULT_TIMER_LIMIT ;
2007-12-13 12:19:42 +03:00
static int timer_tstamp_monotonic = 1 ;
2007-10-15 11:50:19 +04:00
MODULE_AUTHOR ( " Jaroslav Kysela <perex@perex.cz>, Takashi Iwai <tiwai@suse.de> " ) ;
2005-04-17 02:20:36 +04:00
MODULE_DESCRIPTION ( " ALSA timer interface " ) ;
MODULE_LICENSE ( " GPL " ) ;
module_param ( timer_limit , int , 0444 ) ;
MODULE_PARM_DESC ( timer_limit , " Maximum global timers in system. " ) ;
2007-12-13 12:19:42 +03:00
module_param ( timer_tstamp_monotonic , int , 0444 ) ;
MODULE_PARM_DESC ( timer_tstamp_monotonic , " Use posix monotonic clock source for timestamps (default). " ) ;
2005-04-17 02:20:36 +04:00
2010-11-23 19:43:19 +03:00
MODULE_ALIAS_CHARDEV ( CONFIG_SND_MAJOR , SNDRV_MINOR_TIMER ) ;
MODULE_ALIAS ( " devname:snd/timer " ) ;
2005-11-17 15:56:05 +03:00
struct snd_timer_user {
struct snd_timer_instance * timeri ;
2005-10-12 19:20:21 +04:00
int tread ; /* enhanced read with timestamps and events */
2005-04-17 02:20:36 +04:00
unsigned long ticks ;
unsigned long overrun ;
int qhead ;
int qtail ;
int qused ;
int queue_size ;
2016-01-21 19:19:31 +03:00
bool disconnected ;
2005-11-17 15:56:05 +03:00
struct snd_timer_read * queue ;
struct snd_timer_tread * tqueue ;
2005-04-17 02:20:36 +04:00
spinlock_t qlock ;
unsigned long last_resolution ;
unsigned int filter ;
struct timespec tstamp ; /* trigger tstamp */
wait_queue_head_t qchange_sleep ;
struct fasync_struct * fasync ;
2016-01-13 19:48:01 +03:00
struct mutex ioctl_lock ;
2005-11-17 15:56:05 +03:00
} ;
2005-04-17 02:20:36 +04:00
/* list of timers */
static LIST_HEAD ( snd_timer_list ) ;
/* list of slave instances */
static LIST_HEAD ( snd_timer_slave_list ) ;
/* lock for slave active lists */
static DEFINE_SPINLOCK ( slave_active_lock ) ;
2006-01-16 18:29:08 +03:00
static DEFINE_MUTEX ( register_mutex ) ;
2005-04-17 02:20:36 +04:00
2005-11-17 15:56:05 +03:00
static int snd_timer_free ( struct snd_timer * timer ) ;
static int snd_timer_dev_free ( struct snd_device * device ) ;
static int snd_timer_dev_register ( struct snd_device * device ) ;
2006-06-23 16:38:23 +04:00
static int snd_timer_dev_disconnect ( struct snd_device * device ) ;
2005-04-17 02:20:36 +04:00
2005-11-17 15:56:05 +03:00
static void snd_timer_reschedule ( struct snd_timer * timer , unsigned long ticks_left ) ;
2005-04-17 02:20:36 +04:00
/*
* create a timer instance with the given owner string .
* when timer is not NULL , increments the module counter
*/
2005-11-17 15:56:05 +03:00
static struct snd_timer_instance * snd_timer_instance_new ( char * owner ,
struct snd_timer * timer )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer_instance * timeri ;
2005-09-09 16:20:23 +04:00
timeri = kzalloc ( sizeof ( * timeri ) , GFP_KERNEL ) ;
2005-04-17 02:20:36 +04:00
if ( timeri = = NULL )
return NULL ;
2005-06-23 11:09:02 +04:00
timeri - > owner = kstrdup ( owner , GFP_KERNEL ) ;
2005-04-17 02:20:36 +04:00
if ( ! timeri - > owner ) {
kfree ( timeri ) ;
return NULL ;
}
INIT_LIST_HEAD ( & timeri - > open_list ) ;
INIT_LIST_HEAD ( & timeri - > active_list ) ;
INIT_LIST_HEAD ( & timeri - > ack_list ) ;
INIT_LIST_HEAD ( & timeri - > slave_list_head ) ;
INIT_LIST_HEAD ( & timeri - > slave_active_head ) ;
timeri - > timer = timer ;
2005-10-12 19:12:31 +04:00
if ( timer & & ! try_module_get ( timer - > module ) ) {
2005-04-17 02:20:36 +04:00
kfree ( timeri - > owner ) ;
kfree ( timeri ) ;
return NULL ;
}
return timeri ;
}
/*
* find a timer instance from the given timer id
*/
2005-11-17 15:56:05 +03:00
static struct snd_timer * snd_timer_find ( struct snd_timer_id * tid )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer * timer = NULL ;
2005-04-17 02:20:36 +04:00
2006-10-05 18:02:22 +04:00
list_for_each_entry ( timer , & snd_timer_list , device_list ) {
2005-04-17 02:20:36 +04:00
if ( timer - > tmr_class ! = tid - > dev_class )
continue ;
if ( ( timer - > tmr_class = = SNDRV_TIMER_CLASS_CARD | |
timer - > tmr_class = = SNDRV_TIMER_CLASS_PCM ) & &
( timer - > card = = NULL | |
timer - > card - > number ! = tid - > card ) )
continue ;
if ( timer - > tmr_device ! = tid - > device )
continue ;
if ( timer - > tmr_subdevice ! = tid - > subdevice )
continue ;
return timer ;
}
return NULL ;
}
2008-07-09 12:28:41 +04:00
# ifdef CONFIG_MODULES
2005-04-17 02:20:36 +04:00
2005-11-17 15:56:05 +03:00
static void snd_timer_request ( struct snd_timer_id * tid )
2005-04-17 02:20:36 +04:00
{
switch ( tid - > dev_class ) {
case SNDRV_TIMER_CLASS_GLOBAL :
if ( tid - > device < timer_limit )
request_module ( " snd-timer-%i " , tid - > device ) ;
break ;
case SNDRV_TIMER_CLASS_CARD :
case SNDRV_TIMER_CLASS_PCM :
if ( tid - > card < snd_ecards_limit )
request_module ( " snd-card-%i " , tid - > card ) ;
break ;
default :
break ;
}
}
# endif
/*
* look for a master instance matching with the slave id of the given slave .
* when found , relink the open_link of the slave .
*
* call this with register_mutex down .
*/
2017-11-05 12:07:43 +03:00
static int snd_timer_check_slave ( struct snd_timer_instance * slave )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer * timer ;
struct snd_timer_instance * master ;
2005-04-17 02:20:36 +04:00
/* FIXME: it's really dumb to look up all entries.. */
2006-10-05 18:02:22 +04:00
list_for_each_entry ( timer , & snd_timer_list , device_list ) {
list_for_each_entry ( master , & timer - > open_list_head , open_list ) {
2005-04-17 02:20:36 +04:00
if ( slave - > slave_class = = master - > slave_class & &
slave - > slave_id = = master - > slave_id ) {
2017-11-05 12:07:43 +03:00
if ( master - > timer - > num_instances > =
master - > timer - > max_instances )
return - EBUSY ;
2011-03-16 19:10:11 +03:00
list_move_tail ( & slave - > open_list ,
& master - > slave_list_head ) ;
2017-11-05 12:07:43 +03:00
master - > timer - > num_instances + + ;
2005-04-17 02:20:36 +04:00
spin_lock_irq ( & slave_active_lock ) ;
slave - > master = master ;
slave - > timer = master - > timer ;
spin_unlock_irq ( & slave_active_lock ) ;
2017-11-05 12:07:43 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
}
}
2017-11-05 12:07:43 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
/*
* look for slave instances matching with the slave id of the given master .
* when found , relink the open_link of slaves .
*
* call this with register_mutex down .
*/
2017-11-05 12:07:43 +03:00
static int snd_timer_check_master ( struct snd_timer_instance * master )
2005-04-17 02:20:36 +04:00
{
2006-10-05 18:02:22 +04:00
struct snd_timer_instance * slave , * tmp ;
2005-04-17 02:20:36 +04:00
/* check all pending slaves */
2006-10-05 18:02:22 +04:00
list_for_each_entry_safe ( slave , tmp , & snd_timer_slave_list , open_list ) {
2005-04-17 02:20:36 +04:00
if ( slave - > slave_class = = master - > slave_class & &
slave - > slave_id = = master - > slave_id ) {
2017-11-05 12:07:43 +03:00
if ( master - > timer - > num_instances > =
master - > timer - > max_instances )
return - EBUSY ;
2006-10-05 18:02:22 +04:00
list_move_tail ( & slave - > open_list , & master - > slave_list_head ) ;
2017-11-05 12:07:43 +03:00
master - > timer - > num_instances + + ;
2005-04-17 02:20:36 +04:00
spin_lock_irq ( & slave_active_lock ) ;
ALSA: timer: Harden slave timer list handling
A slave timer instance might be still accessible in a racy way while
operating the master instance as it lacks of locking. Since the
master operation is mostly protected with timer->lock, we should cope
with it while changing the slave instance, too. Also, some linked
lists (active_list and ack_list) of slave instances aren't unlinked
immediately at stopping or closing, and this may lead to unexpected
accesses.
This patch tries to address these issues. It adds spin lock of
timer->lock (either from master or slave, which is equivalent) in a
few places. For avoiding a deadlock, we ensure that the global
slave_active_lock is always locked at first before each timer lock.
Also, ack and active_list of slave instances are properly unlinked at
snd_timer_stop() and snd_timer_close().
Last but not least, remove the superfluous call of _snd_timer_stop()
at removing slave links. This is a noop, and calling it may confuse
readers wrt locking. Further cleanup will follow in a later patch.
Actually we've got reports of use-after-free by syzkaller fuzzer, and
this hopefully fixes these issues.
Reported-by: Dmitry Vyukov <dvyukov@google.com>
Cc: <stable@vger.kernel.org>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2016-01-14 18:30:58 +03:00
spin_lock ( & master - > timer - > lock ) ;
2005-04-17 02:20:36 +04:00
slave - > master = master ;
slave - > timer = master - > timer ;
if ( slave - > flags & SNDRV_TIMER_IFLG_RUNNING )
2005-10-12 19:20:21 +04:00
list_add_tail ( & slave - > active_list ,
& master - > slave_active_head ) ;
ALSA: timer: Harden slave timer list handling
A slave timer instance might be still accessible in a racy way while
operating the master instance as it lacks of locking. Since the
master operation is mostly protected with timer->lock, we should cope
with it while changing the slave instance, too. Also, some linked
lists (active_list and ack_list) of slave instances aren't unlinked
immediately at stopping or closing, and this may lead to unexpected
accesses.
This patch tries to address these issues. It adds spin lock of
timer->lock (either from master or slave, which is equivalent) in a
few places. For avoiding a deadlock, we ensure that the global
slave_active_lock is always locked at first before each timer lock.
Also, ack and active_list of slave instances are properly unlinked at
snd_timer_stop() and snd_timer_close().
Last but not least, remove the superfluous call of _snd_timer_stop()
at removing slave links. This is a noop, and calling it may confuse
readers wrt locking. Further cleanup will follow in a later patch.
Actually we've got reports of use-after-free by syzkaller fuzzer, and
this hopefully fixes these issues.
Reported-by: Dmitry Vyukov <dvyukov@google.com>
Cc: <stable@vger.kernel.org>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2016-01-14 18:30:58 +03:00
spin_unlock ( & master - > timer - > lock ) ;
2005-04-17 02:20:36 +04:00
spin_unlock_irq ( & slave_active_lock ) ;
}
}
2017-11-05 12:07:43 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
2017-11-05 12:07:43 +03:00
static int snd_timer_close_locked ( struct snd_timer_instance * timeri ) ;
2005-04-17 02:20:36 +04:00
/*
* open a timer instance
* when opening a master , the slave id must be here given .
*/
2005-11-17 15:56:05 +03:00
int snd_timer_open ( struct snd_timer_instance * * ti ,
char * owner , struct snd_timer_id * tid ,
2005-04-17 02:20:36 +04:00
unsigned int slave_id )
{
2005-11-17 15:56:05 +03:00
struct snd_timer * timer ;
struct snd_timer_instance * timeri = NULL ;
2017-11-05 12:07:43 +03:00
int err ;
2005-10-12 19:20:21 +04:00
2005-04-17 02:20:36 +04:00
if ( tid - > dev_class = = SNDRV_TIMER_CLASS_SLAVE ) {
/* open a slave instance */
if ( tid - > dev_sclass < = SNDRV_TIMER_SCLASS_NONE | |
tid - > dev_sclass > SNDRV_TIMER_SCLASS_OSS_SEQUENCER ) {
2014-02-04 21:22:39 +04:00
pr_debug ( " ALSA: timer: invalid slave class %i \n " ,
tid - > dev_sclass ) ;
2005-04-17 02:20:36 +04:00
return - EINVAL ;
}
2006-01-16 18:29:08 +03:00
mutex_lock ( & register_mutex ) ;
2005-04-17 02:20:36 +04:00
timeri = snd_timer_instance_new ( owner , NULL ) ;
2005-10-12 19:10:35 +04:00
if ( ! timeri ) {
2006-01-16 18:29:08 +03:00
mutex_unlock ( & register_mutex ) ;
2005-10-12 19:10:35 +04:00
return - ENOMEM ;
}
2005-04-17 02:20:36 +04:00
timeri - > slave_class = tid - > dev_sclass ;
timeri - > slave_id = tid - > device ;
timeri - > flags | = SNDRV_TIMER_IFLG_SLAVE ;
list_add_tail ( & timeri - > open_list , & snd_timer_slave_list ) ;
2017-11-05 12:07:43 +03:00
err = snd_timer_check_slave ( timeri ) ;
if ( err < 0 ) {
snd_timer_close_locked ( timeri ) ;
timeri = NULL ;
}
2006-01-16 18:29:08 +03:00
mutex_unlock ( & register_mutex ) ;
2005-04-17 02:20:36 +04:00
* ti = timeri ;
2017-11-05 12:07:43 +03:00
return err ;
2005-04-17 02:20:36 +04:00
}
/* open a master instance */
2006-01-16 18:29:08 +03:00
mutex_lock ( & register_mutex ) ;
2005-04-17 02:20:36 +04:00
timer = snd_timer_find ( tid ) ;
2008-07-09 12:28:41 +04:00
# ifdef CONFIG_MODULES
if ( ! timer ) {
2006-01-16 18:29:08 +03:00
mutex_unlock ( & register_mutex ) ;
2005-04-17 02:20:36 +04:00
snd_timer_request ( tid ) ;
2006-01-16 18:29:08 +03:00
mutex_lock ( & register_mutex ) ;
2005-04-17 02:20:36 +04:00
timer = snd_timer_find ( tid ) ;
}
# endif
2005-10-12 19:10:35 +04:00
if ( ! timer ) {
2006-01-16 18:29:08 +03:00
mutex_unlock ( & register_mutex ) ;
2005-04-17 02:20:36 +04:00
return - ENODEV ;
}
2005-10-12 19:10:35 +04:00
if ( ! list_empty ( & timer - > open_list_head ) ) {
timeri = list_entry ( timer - > open_list_head . next ,
2005-11-17 15:56:05 +03:00
struct snd_timer_instance , open_list ) ;
2005-10-12 19:10:35 +04:00
if ( timeri - > flags & SNDRV_TIMER_IFLG_EXCLUSIVE ) {
2006-01-16 18:29:08 +03:00
mutex_unlock ( & register_mutex ) ;
2005-10-12 19:10:35 +04:00
return - EBUSY ;
}
}
2017-11-05 12:07:43 +03:00
if ( timer - > num_instances > = timer - > max_instances ) {
mutex_unlock ( & register_mutex ) ;
return - EBUSY ;
}
2005-10-12 19:10:35 +04:00
timeri = snd_timer_instance_new ( owner , timer ) ;
if ( ! timeri ) {
2006-01-16 18:29:08 +03:00
mutex_unlock ( & register_mutex ) ;
2005-10-12 19:10:35 +04:00
return - ENOMEM ;
}
2016-01-21 19:19:31 +03:00
/* take a card refcount for safe disconnection */
if ( timer - > card )
get_device ( & timer - > card - > card_dev ) ;
2005-10-12 19:10:35 +04:00
timeri - > slave_class = tid - > dev_sclass ;
timeri - > slave_id = slave_id ;
2016-08-29 01:33:51 +03:00
if ( list_empty ( & timer - > open_list_head ) & & timer - > hw . open ) {
int err = timer - > hw . open ( timer ) ;
if ( err ) {
kfree ( timeri - > owner ) ;
kfree ( timeri ) ;
if ( timer - > card )
put_device ( & timer - > card - > card_dev ) ;
module_put ( timer - > module ) ;
mutex_unlock ( & register_mutex ) ;
return err ;
}
}
2005-10-12 19:10:35 +04:00
list_add_tail ( & timeri - > open_list , & timer - > open_list_head ) ;
2017-11-05 12:07:43 +03:00
timer - > num_instances + + ;
err = snd_timer_check_master ( timeri ) ;
if ( err < 0 ) {
snd_timer_close_locked ( timeri ) ;
timeri = NULL ;
}
2006-01-16 18:29:08 +03:00
mutex_unlock ( & register_mutex ) ;
2005-04-17 02:20:36 +04:00
* ti = timeri ;
2017-11-05 12:07:43 +03:00
return err ;
2005-04-17 02:20:36 +04:00
}
2017-06-16 17:16:05 +03:00
EXPORT_SYMBOL ( snd_timer_open ) ;
2005-04-17 02:20:36 +04:00
/*
* close a timer instance
2017-11-05 12:07:43 +03:00
* call this with register_mutex down .
2005-04-17 02:20:36 +04:00
*/
2017-11-05 12:07:43 +03:00
static int snd_timer_close_locked ( struct snd_timer_instance * timeri )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer * timer = NULL ;
2006-10-05 18:02:22 +04:00
struct snd_timer_instance * slave , * tmp ;
2005-04-17 02:20:36 +04:00
2016-02-10 13:53:30 +03:00
list_del ( & timeri - > open_list ) ;
2005-04-17 02:20:36 +04:00
/* force to stop the timer */
snd_timer_stop ( timeri ) ;
2016-02-10 13:53:30 +03:00
timer = timeri - > timer ;
if ( timer ) {
2017-11-05 12:07:43 +03:00
timer - > num_instances - - ;
2005-04-17 02:20:36 +04:00
/* wait, until the active callback is finished */
spin_lock_irq ( & timer - > lock ) ;
while ( timeri - > flags & SNDRV_TIMER_IFLG_CALLBACK ) {
spin_unlock_irq ( & timer - > lock ) ;
udelay ( 10 ) ;
spin_lock_irq ( & timer - > lock ) ;
}
spin_unlock_irq ( & timer - > lock ) ;
2016-02-10 13:53:30 +03:00
2005-04-17 02:20:36 +04:00
/* remove slave links */
ALSA: timer: Harden slave timer list handling
A slave timer instance might be still accessible in a racy way while
operating the master instance as it lacks of locking. Since the
master operation is mostly protected with timer->lock, we should cope
with it while changing the slave instance, too. Also, some linked
lists (active_list and ack_list) of slave instances aren't unlinked
immediately at stopping or closing, and this may lead to unexpected
accesses.
This patch tries to address these issues. It adds spin lock of
timer->lock (either from master or slave, which is equivalent) in a
few places. For avoiding a deadlock, we ensure that the global
slave_active_lock is always locked at first before each timer lock.
Also, ack and active_list of slave instances are properly unlinked at
snd_timer_stop() and snd_timer_close().
Last but not least, remove the superfluous call of _snd_timer_stop()
at removing slave links. This is a noop, and calling it may confuse
readers wrt locking. Further cleanup will follow in a later patch.
Actually we've got reports of use-after-free by syzkaller fuzzer, and
this hopefully fixes these issues.
Reported-by: Dmitry Vyukov <dvyukov@google.com>
Cc: <stable@vger.kernel.org>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2016-01-14 18:30:58 +03:00
spin_lock_irq ( & slave_active_lock ) ;
spin_lock ( & timer - > lock ) ;
2006-10-05 18:02:22 +04:00
list_for_each_entry_safe ( slave , tmp , & timeri - > slave_list_head ,
open_list ) {
list_move_tail ( & slave - > open_list , & snd_timer_slave_list ) ;
2017-11-05 12:07:43 +03:00
timer - > num_instances - - ;
2005-04-17 02:20:36 +04:00
slave - > master = NULL ;
slave - > timer = NULL ;
ALSA: timer: Harden slave timer list handling
A slave timer instance might be still accessible in a racy way while
operating the master instance as it lacks of locking. Since the
master operation is mostly protected with timer->lock, we should cope
with it while changing the slave instance, too. Also, some linked
lists (active_list and ack_list) of slave instances aren't unlinked
immediately at stopping or closing, and this may lead to unexpected
accesses.
This patch tries to address these issues. It adds spin lock of
timer->lock (either from master or slave, which is equivalent) in a
few places. For avoiding a deadlock, we ensure that the global
slave_active_lock is always locked at first before each timer lock.
Also, ack and active_list of slave instances are properly unlinked at
snd_timer_stop() and snd_timer_close().
Last but not least, remove the superfluous call of _snd_timer_stop()
at removing slave links. This is a noop, and calling it may confuse
readers wrt locking. Further cleanup will follow in a later patch.
Actually we've got reports of use-after-free by syzkaller fuzzer, and
this hopefully fixes these issues.
Reported-by: Dmitry Vyukov <dvyukov@google.com>
Cc: <stable@vger.kernel.org>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2016-01-14 18:30:58 +03:00
list_del_init ( & slave - > ack_list ) ;
list_del_init ( & slave - > active_list ) ;
2005-04-17 02:20:36 +04:00
}
ALSA: timer: Harden slave timer list handling
A slave timer instance might be still accessible in a racy way while
operating the master instance as it lacks of locking. Since the
master operation is mostly protected with timer->lock, we should cope
with it while changing the slave instance, too. Also, some linked
lists (active_list and ack_list) of slave instances aren't unlinked
immediately at stopping or closing, and this may lead to unexpected
accesses.
This patch tries to address these issues. It adds spin lock of
timer->lock (either from master or slave, which is equivalent) in a
few places. For avoiding a deadlock, we ensure that the global
slave_active_lock is always locked at first before each timer lock.
Also, ack and active_list of slave instances are properly unlinked at
snd_timer_stop() and snd_timer_close().
Last but not least, remove the superfluous call of _snd_timer_stop()
at removing slave links. This is a noop, and calling it may confuse
readers wrt locking. Further cleanup will follow in a later patch.
Actually we've got reports of use-after-free by syzkaller fuzzer, and
this hopefully fixes these issues.
Reported-by: Dmitry Vyukov <dvyukov@google.com>
Cc: <stable@vger.kernel.org>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2016-01-14 18:30:58 +03:00
spin_unlock ( & timer - > lock ) ;
spin_unlock_irq ( & slave_active_lock ) ;
2016-02-10 13:53:30 +03:00
/* slave doesn't need to release timer resources below */
if ( timeri - > flags & SNDRV_TIMER_IFLG_SLAVE )
timer = NULL ;
2005-04-17 02:20:36 +04:00
}
2016-02-10 13:53:30 +03:00
2005-04-17 02:20:36 +04:00
if ( timeri - > private_free )
timeri - > private_free ( timeri ) ;
kfree ( timeri - > owner ) ;
kfree ( timeri ) ;
2016-02-10 13:53:30 +03:00
if ( timer ) {
if ( list_empty ( & timer - > open_list_head ) & & timer - > hw . close )
timer - > hw . close ( timer ) ;
/* release a card refcount for safe disconnection */
if ( timer - > card )
put_device ( & timer - > card - > card_dev ) ;
2005-10-12 19:12:31 +04:00
module_put ( timer - > module ) ;
2016-02-10 13:53:30 +03:00
}
2005-04-17 02:20:36 +04:00
return 0 ;
}
2017-11-05 12:07:43 +03:00
/*
* close a timer instance
*/
int snd_timer_close ( struct snd_timer_instance * timeri )
{
int err ;
if ( snd_BUG_ON ( ! timeri ) )
return - ENXIO ;
mutex_lock ( & register_mutex ) ;
err = snd_timer_close_locked ( timeri ) ;
mutex_unlock ( & register_mutex ) ;
return err ;
}
2017-06-16 17:16:05 +03:00
EXPORT_SYMBOL ( snd_timer_close ) ;
2005-04-17 02:20:36 +04:00
2005-11-17 15:56:05 +03:00
unsigned long snd_timer_resolution ( struct snd_timer_instance * timeri )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer * timer ;
2005-04-17 02:20:36 +04:00
if ( timeri = = NULL )
return 0 ;
2017-08-23 10:45:06 +03:00
timer = timeri - > timer ;
if ( timer ) {
2005-04-17 02:20:36 +04:00
if ( timer - > hw . c_resolution )
return timer - > hw . c_resolution ( timer ) ;
return timer - > hw . resolution ;
}
return 0 ;
}
2017-06-16 17:16:05 +03:00
EXPORT_SYMBOL ( snd_timer_resolution ) ;
2005-04-17 02:20:36 +04:00
2005-11-17 15:56:05 +03:00
static void snd_timer_notify1 ( struct snd_timer_instance * ti , int event )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer * timer ;
2005-04-17 02:20:36 +04:00
unsigned long resolution = 0 ;
2005-11-17 15:56:05 +03:00
struct snd_timer_instance * ts ;
2005-04-17 02:20:36 +04:00
struct timespec tstamp ;
2007-12-13 12:19:42 +03:00
if ( timer_tstamp_monotonic )
2014-06-12 03:59:14 +04:00
ktime_get_ts ( & tstamp ) ;
2007-12-13 12:19:42 +03:00
else
getnstimeofday ( & tstamp ) ;
2008-08-08 19:09:09 +04:00
if ( snd_BUG_ON ( event < SNDRV_TIMER_EVENT_START | |
event > SNDRV_TIMER_EVENT_PAUSE ) )
return ;
2005-10-12 19:20:21 +04:00
if ( event = = SNDRV_TIMER_EVENT_START | |
event = = SNDRV_TIMER_EVENT_CONTINUE )
2005-04-17 02:20:36 +04:00
resolution = snd_timer_resolution ( ti ) ;
if ( ti - > ccallback )
2010-03-03 13:05:55 +03:00
ti - > ccallback ( ti , event , & tstamp , resolution ) ;
2005-04-17 02:20:36 +04:00
if ( ti - > flags & SNDRV_TIMER_IFLG_SLAVE )
return ;
timer = ti - > timer ;
if ( timer = = NULL )
return ;
if ( timer - > hw . flags & SNDRV_TIMER_HW_SLAVE )
return ;
2006-10-05 18:02:22 +04:00
list_for_each_entry ( ts , & ti - > slave_active_head , active_list )
2005-04-17 02:20:36 +04:00
if ( ts - > ccallback )
2016-02-08 19:36:25 +03:00
ts - > ccallback ( ts , event + 100 , & tstamp , resolution ) ;
2005-04-17 02:20:36 +04:00
}
2016-02-10 14:47:03 +03:00
/* start/continue a master timer */
static int snd_timer_start1 ( struct snd_timer_instance * timeri ,
bool start , unsigned long ticks )
2005-04-17 02:20:36 +04:00
{
2016-02-10 14:47:03 +03:00
struct snd_timer * timer ;
int result ;
unsigned long flags ;
timer = timeri - > timer ;
if ( ! timer )
return - EINVAL ;
spin_lock_irqsave ( & timer - > lock , flags ) ;
if ( timer - > card & & timer - > card - > shutdown ) {
result = - ENODEV ;
goto unlock ;
}
if ( timeri - > flags & ( SNDRV_TIMER_IFLG_RUNNING |
SNDRV_TIMER_IFLG_START ) ) {
result = - EBUSY ;
goto unlock ;
}
if ( start )
timeri - > ticks = timeri - > cticks = ticks ;
else if ( ! timeri - > cticks )
timeri - > cticks = 1 ;
timeri - > pticks = 0 ;
2011-03-16 19:10:11 +03:00
list_move_tail ( & timeri - > active_list , & timer - > active_list_head ) ;
2005-04-17 02:20:36 +04:00
if ( timer - > running ) {
if ( timer - > hw . flags & SNDRV_TIMER_HW_SLAVE )
goto __start_now ;
timer - > flags | = SNDRV_TIMER_FLG_RESCHED ;
timeri - > flags | = SNDRV_TIMER_IFLG_START ;
2016-02-10 14:47:03 +03:00
result = 1 ; /* delayed start */
2005-04-17 02:20:36 +04:00
} else {
2016-02-10 14:47:03 +03:00
if ( start )
timer - > sticks = ticks ;
2005-04-17 02:20:36 +04:00
timer - > hw . start ( timer ) ;
__start_now :
timer - > running + + ;
timeri - > flags | = SNDRV_TIMER_IFLG_RUNNING ;
2016-02-10 14:47:03 +03:00
result = 0 ;
2005-04-17 02:20:36 +04:00
}
2016-02-10 14:47:03 +03:00
snd_timer_notify1 ( timeri , start ? SNDRV_TIMER_EVENT_START :
SNDRV_TIMER_EVENT_CONTINUE ) ;
unlock :
spin_unlock_irqrestore ( & timer - > lock , flags ) ;
return result ;
2005-04-17 02:20:36 +04:00
}
2016-02-10 14:47:03 +03:00
/* start/continue a slave timer */
static int snd_timer_start_slave ( struct snd_timer_instance * timeri ,
bool start )
2005-04-17 02:20:36 +04:00
{
unsigned long flags ;
spin_lock_irqsave ( & slave_active_lock , flags ) ;
2016-01-31 01:09:08 +03:00
if ( timeri - > flags & SNDRV_TIMER_IFLG_RUNNING ) {
spin_unlock_irqrestore ( & slave_active_lock , flags ) ;
return - EBUSY ;
}
2005-04-17 02:20:36 +04:00
timeri - > flags | = SNDRV_TIMER_IFLG_RUNNING ;
ALSA: timer: Harden slave timer list handling
A slave timer instance might be still accessible in a racy way while
operating the master instance as it lacks of locking. Since the
master operation is mostly protected with timer->lock, we should cope
with it while changing the slave instance, too. Also, some linked
lists (active_list and ack_list) of slave instances aren't unlinked
immediately at stopping or closing, and this may lead to unexpected
accesses.
This patch tries to address these issues. It adds spin lock of
timer->lock (either from master or slave, which is equivalent) in a
few places. For avoiding a deadlock, we ensure that the global
slave_active_lock is always locked at first before each timer lock.
Also, ack and active_list of slave instances are properly unlinked at
snd_timer_stop() and snd_timer_close().
Last but not least, remove the superfluous call of _snd_timer_stop()
at removing slave links. This is a noop, and calling it may confuse
readers wrt locking. Further cleanup will follow in a later patch.
Actually we've got reports of use-after-free by syzkaller fuzzer, and
this hopefully fixes these issues.
Reported-by: Dmitry Vyukov <dvyukov@google.com>
Cc: <stable@vger.kernel.org>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2016-01-14 18:30:58 +03:00
if ( timeri - > master & & timeri - > timer ) {
spin_lock ( & timeri - > timer - > lock ) ;
2005-10-12 19:20:21 +04:00
list_add_tail ( & timeri - > active_list ,
& timeri - > master - > slave_active_head ) ;
2016-02-10 14:47:03 +03:00
snd_timer_notify1 ( timeri , start ? SNDRV_TIMER_EVENT_START :
SNDRV_TIMER_EVENT_CONTINUE ) ;
ALSA: timer: Harden slave timer list handling
A slave timer instance might be still accessible in a racy way while
operating the master instance as it lacks of locking. Since the
master operation is mostly protected with timer->lock, we should cope
with it while changing the slave instance, too. Also, some linked
lists (active_list and ack_list) of slave instances aren't unlinked
immediately at stopping or closing, and this may lead to unexpected
accesses.
This patch tries to address these issues. It adds spin lock of
timer->lock (either from master or slave, which is equivalent) in a
few places. For avoiding a deadlock, we ensure that the global
slave_active_lock is always locked at first before each timer lock.
Also, ack and active_list of slave instances are properly unlinked at
snd_timer_stop() and snd_timer_close().
Last but not least, remove the superfluous call of _snd_timer_stop()
at removing slave links. This is a noop, and calling it may confuse
readers wrt locking. Further cleanup will follow in a later patch.
Actually we've got reports of use-after-free by syzkaller fuzzer, and
this hopefully fixes these issues.
Reported-by: Dmitry Vyukov <dvyukov@google.com>
Cc: <stable@vger.kernel.org>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2016-01-14 18:30:58 +03:00
spin_unlock ( & timeri - > timer - > lock ) ;
}
2005-04-17 02:20:36 +04:00
spin_unlock_irqrestore ( & slave_active_lock , flags ) ;
return 1 ; /* delayed start */
}
2016-02-10 14:47:03 +03:00
/* stop/pause a master timer */
static int snd_timer_stop1 ( struct snd_timer_instance * timeri , bool stop )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer * timer ;
2016-02-10 14:47:03 +03:00
int result = 0 ;
2005-04-17 02:20:36 +04:00
unsigned long flags ;
timer = timeri - > timer ;
if ( ! timer )
return - EINVAL ;
spin_lock_irqsave ( & timer - > lock , flags ) ;
2016-01-31 01:09:08 +03:00
if ( ! ( timeri - > flags & ( SNDRV_TIMER_IFLG_RUNNING |
SNDRV_TIMER_IFLG_START ) ) ) {
2016-02-10 14:47:03 +03:00
result = - EBUSY ;
goto unlock ;
2016-01-31 01:09:08 +03:00
}
2005-04-17 02:20:36 +04:00
list_del_init ( & timeri - > ack_list ) ;
list_del_init ( & timeri - > active_list ) ;
2016-02-10 14:47:03 +03:00
if ( timer - > card & & timer - > card - > shutdown )
goto unlock ;
if ( stop ) {
timeri - > cticks = timeri - > ticks ;
timeri - > pticks = 0 ;
2016-01-21 19:19:31 +03:00
}
2005-04-17 02:20:36 +04:00
if ( ( timeri - > flags & SNDRV_TIMER_IFLG_RUNNING ) & &
! ( - - timer - > running ) ) {
timer - > hw . stop ( timer ) ;
if ( timer - > flags & SNDRV_TIMER_FLG_RESCHED ) {
timer - > flags & = ~ SNDRV_TIMER_FLG_RESCHED ;
snd_timer_reschedule ( timer , 0 ) ;
if ( timer - > flags & SNDRV_TIMER_FLG_CHANGE ) {
timer - > flags & = ~ SNDRV_TIMER_FLG_CHANGE ;
timer - > hw . start ( timer ) ;
}
}
}
2016-01-14 19:01:46 +03:00
timeri - > flags & = ~ ( SNDRV_TIMER_IFLG_RUNNING | SNDRV_TIMER_IFLG_START ) ;
2016-09-07 16:45:31 +03:00
if ( stop )
timeri - > flags & = ~ SNDRV_TIMER_IFLG_PAUSED ;
else
timeri - > flags | = SNDRV_TIMER_IFLG_PAUSED ;
2016-02-10 14:47:03 +03:00
snd_timer_notify1 ( timeri , stop ? SNDRV_TIMER_EVENT_STOP :
SNDRV_TIMER_EVENT_CONTINUE ) ;
unlock :
2005-04-17 02:20:36 +04:00
spin_unlock_irqrestore ( & timer - > lock , flags ) ;
2016-02-10 14:47:03 +03:00
return result ;
}
/* stop/pause a slave timer */
static int snd_timer_stop_slave ( struct snd_timer_instance * timeri , bool stop )
{
unsigned long flags ;
spin_lock_irqsave ( & slave_active_lock , flags ) ;
if ( ! ( timeri - > flags & SNDRV_TIMER_IFLG_RUNNING ) ) {
spin_unlock_irqrestore ( & slave_active_lock , flags ) ;
return - EBUSY ;
}
timeri - > flags & = ~ SNDRV_TIMER_IFLG_RUNNING ;
if ( timeri - > timer ) {
spin_lock ( & timeri - > timer - > lock ) ;
list_del_init ( & timeri - > ack_list ) ;
list_del_init ( & timeri - > active_list ) ;
snd_timer_notify1 ( timeri , stop ? SNDRV_TIMER_EVENT_STOP :
SNDRV_TIMER_EVENT_CONTINUE ) ;
spin_unlock ( & timeri - > timer - > lock ) ;
}
spin_unlock_irqrestore ( & slave_active_lock , flags ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2016-02-10 14:47:03 +03:00
/*
* start the timer instance
*/
int snd_timer_start ( struct snd_timer_instance * timeri , unsigned int ticks )
{
if ( timeri = = NULL | | ticks < 1 )
return - EINVAL ;
if ( timeri - > flags & SNDRV_TIMER_IFLG_SLAVE )
return snd_timer_start_slave ( timeri , true ) ;
else
return snd_timer_start1 ( timeri , true , ticks ) ;
}
2017-06-16 17:16:05 +03:00
EXPORT_SYMBOL ( snd_timer_start ) ;
2016-02-10 14:47:03 +03:00
2005-04-17 02:20:36 +04:00
/*
* stop the timer instance .
*
* do not call this from the timer callback !
*/
2005-11-17 15:56:05 +03:00
int snd_timer_stop ( struct snd_timer_instance * timeri )
2005-04-17 02:20:36 +04:00
{
2016-02-10 14:47:03 +03:00
if ( timeri - > flags & SNDRV_TIMER_IFLG_SLAVE )
return snd_timer_stop_slave ( timeri , true ) ;
else
return snd_timer_stop1 ( timeri , true ) ;
2005-04-17 02:20:36 +04:00
}
2017-06-16 17:16:05 +03:00
EXPORT_SYMBOL ( snd_timer_stop ) ;
2005-04-17 02:20:36 +04:00
/*
* start again . . the tick is kept .
*/
2005-11-17 15:56:05 +03:00
int snd_timer_continue ( struct snd_timer_instance * timeri )
2005-04-17 02:20:36 +04:00
{
2016-09-07 16:45:31 +03:00
/* timer can continue only after pause */
if ( ! ( timeri - > flags & SNDRV_TIMER_IFLG_PAUSED ) )
return - EINVAL ;
2005-04-17 02:20:36 +04:00
if ( timeri - > flags & SNDRV_TIMER_IFLG_SLAVE )
2016-02-10 14:47:03 +03:00
return snd_timer_start_slave ( timeri , false ) ;
else
return snd_timer_start1 ( timeri , false , 0 ) ;
2005-04-17 02:20:36 +04:00
}
2017-06-16 17:16:05 +03:00
EXPORT_SYMBOL ( snd_timer_continue ) ;
2005-04-17 02:20:36 +04:00
/*
* pause . . remember the ticks left
*/
2005-11-17 15:56:05 +03:00
int snd_timer_pause ( struct snd_timer_instance * timeri )
2005-04-17 02:20:36 +04:00
{
2016-02-10 14:47:03 +03:00
if ( timeri - > flags & SNDRV_TIMER_IFLG_SLAVE )
return snd_timer_stop_slave ( timeri , false ) ;
else
return snd_timer_stop1 ( timeri , false ) ;
2005-04-17 02:20:36 +04:00
}
2017-06-16 17:16:05 +03:00
EXPORT_SYMBOL ( snd_timer_pause ) ;
2005-04-17 02:20:36 +04:00
/*
* reschedule the timer
*
* start pending instances and check the scheduling ticks .
* when the scheduling ticks is changed set CHANGE flag to reprogram the timer .
*/
2005-11-17 15:56:05 +03:00
static void snd_timer_reschedule ( struct snd_timer * timer , unsigned long ticks_left )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer_instance * ti ;
2005-04-17 02:20:36 +04:00
unsigned long ticks = ~ 0UL ;
2006-10-05 18:02:22 +04:00
list_for_each_entry ( ti , & timer - > active_list_head , active_list ) {
2005-04-17 02:20:36 +04:00
if ( ti - > flags & SNDRV_TIMER_IFLG_START ) {
ti - > flags & = ~ SNDRV_TIMER_IFLG_START ;
ti - > flags | = SNDRV_TIMER_IFLG_RUNNING ;
timer - > running + + ;
}
if ( ti - > flags & SNDRV_TIMER_IFLG_RUNNING ) {
if ( ticks > ti - > cticks )
ticks = ti - > cticks ;
}
}
if ( ticks = = ~ 0UL ) {
timer - > flags & = ~ SNDRV_TIMER_FLG_RESCHED ;
return ;
}
if ( ticks > timer - > hw . ticks )
ticks = timer - > hw . ticks ;
if ( ticks_left ! = ticks )
timer - > flags | = SNDRV_TIMER_FLG_CHANGE ;
timer - > sticks = ticks ;
}
/*
* timer tasklet
*
*/
static void snd_timer_tasklet ( unsigned long arg )
{
2005-11-17 15:56:05 +03:00
struct snd_timer * timer = ( struct snd_timer * ) arg ;
struct snd_timer_instance * ti ;
2005-04-17 02:20:36 +04:00
struct list_head * p ;
unsigned long resolution , ticks ;
2006-07-05 19:16:58 +04:00
unsigned long flags ;
2005-04-17 02:20:36 +04:00
2016-01-21 19:19:31 +03:00
if ( timer - > card & & timer - > card - > shutdown )
return ;
2006-07-05 19:16:58 +04:00
spin_lock_irqsave ( & timer - > lock , flags ) ;
2005-04-17 02:20:36 +04:00
/* now process all callbacks */
while ( ! list_empty ( & timer - > sack_list_head ) ) {
p = timer - > sack_list_head . next ; /* get first item */
2005-11-17 15:56:05 +03:00
ti = list_entry ( p , struct snd_timer_instance , ack_list ) ;
2005-04-17 02:20:36 +04:00
/* remove from ack_list and make empty */
list_del_init ( p ) ;
2005-10-12 19:20:21 +04:00
2005-04-17 02:20:36 +04:00
ticks = ti - > pticks ;
ti - > pticks = 0 ;
resolution = ti - > resolution ;
ti - > flags | = SNDRV_TIMER_IFLG_CALLBACK ;
spin_unlock ( & timer - > lock ) ;
if ( ti - > callback )
ti - > callback ( ti , resolution , ticks ) ;
spin_lock ( & timer - > lock ) ;
ti - > flags & = ~ SNDRV_TIMER_IFLG_CALLBACK ;
}
2006-07-05 19:16:58 +04:00
spin_unlock_irqrestore ( & timer - > lock , flags ) ;
2005-04-17 02:20:36 +04:00
}
/*
* timer interrupt
*
* ticks_left is usually equal to timer - > sticks .
*
*/
2005-11-17 15:56:05 +03:00
void snd_timer_interrupt ( struct snd_timer * timer , unsigned long ticks_left )
2005-04-17 02:20:36 +04:00
{
2006-10-05 18:02:22 +04:00
struct snd_timer_instance * ti , * ts , * tmp ;
2005-04-17 02:20:36 +04:00
unsigned long resolution , ticks ;
2006-10-05 18:02:22 +04:00
struct list_head * p , * ack_list_head ;
2005-11-18 20:52:14 +03:00
unsigned long flags ;
2005-04-17 02:20:36 +04:00
int use_tasklet = 0 ;
if ( timer = = NULL )
return ;
2016-01-21 19:19:31 +03:00
if ( timer - > card & & timer - > card - > shutdown )
return ;
2005-11-18 20:52:14 +03:00
spin_lock_irqsave ( & timer - > lock , flags ) ;
2005-04-17 02:20:36 +04:00
/* remember the current resolution */
if ( timer - > hw . c_resolution )
resolution = timer - > hw . c_resolution ( timer ) ;
else
resolution = timer - > hw . resolution ;
/* loop for all active instances
2006-10-05 18:02:22 +04:00
* Here we cannot use list_for_each_entry because the active_list of a
2005-10-12 19:20:21 +04:00
* processed instance is relinked to done_list_head before the callback
* is called .
2005-04-17 02:20:36 +04:00
*/
2006-10-05 18:02:22 +04:00
list_for_each_entry_safe ( ti , tmp , & timer - > active_list_head ,
active_list ) {
2005-04-17 02:20:36 +04:00
if ( ! ( ti - > flags & SNDRV_TIMER_IFLG_RUNNING ) )
continue ;
ti - > pticks + = ticks_left ;
ti - > resolution = resolution ;
if ( ti - > cticks < ticks_left )
ti - > cticks = 0 ;
else
ti - > cticks - = ticks_left ;
if ( ti - > cticks ) /* not expired */
continue ;
if ( ti - > flags & SNDRV_TIMER_IFLG_AUTO ) {
ti - > cticks = ti - > ticks ;
} else {
ti - > flags & = ~ SNDRV_TIMER_IFLG_RUNNING ;
2016-02-04 19:06:13 +03:00
- - timer - > running ;
list_del_init ( & ti - > active_list ) ;
2005-04-17 02:20:36 +04:00
}
2005-10-12 19:20:21 +04:00
if ( ( timer - > hw . flags & SNDRV_TIMER_HW_TASKLET ) | |
( ti - > flags & SNDRV_TIMER_IFLG_FAST ) )
ack_list_head = & timer - > ack_list_head ;
else
ack_list_head = & timer - > sack_list_head ;
if ( list_empty ( & ti - > ack_list ) )
list_add_tail ( & ti - > ack_list , ack_list_head ) ;
2006-10-05 18:02:22 +04:00
list_for_each_entry ( ts , & ti - > slave_active_head , active_list ) {
2005-04-17 02:20:36 +04:00
ts - > pticks = ti - > pticks ;
ts - > resolution = resolution ;
2005-10-12 19:20:21 +04:00
if ( list_empty ( & ts - > ack_list ) )
list_add_tail ( & ts - > ack_list , ack_list_head ) ;
2005-04-17 02:20:36 +04:00
}
}
if ( timer - > flags & SNDRV_TIMER_FLG_RESCHED )
2006-07-17 18:53:57 +04:00
snd_timer_reschedule ( timer , timer - > sticks ) ;
2005-04-17 02:20:36 +04:00
if ( timer - > running ) {
if ( timer - > hw . flags & SNDRV_TIMER_HW_STOP ) {
timer - > hw . stop ( timer ) ;
timer - > flags | = SNDRV_TIMER_FLG_CHANGE ;
}
if ( ! ( timer - > hw . flags & SNDRV_TIMER_HW_AUTO ) | |
( timer - > flags & SNDRV_TIMER_FLG_CHANGE ) ) {
/* restart timer */
timer - > flags & = ~ SNDRV_TIMER_FLG_CHANGE ;
timer - > hw . start ( timer ) ;
}
} else {
timer - > hw . stop ( timer ) ;
}
/* now process all fast callbacks */
while ( ! list_empty ( & timer - > ack_list_head ) ) {
p = timer - > ack_list_head . next ; /* get first item */
2005-11-17 15:56:05 +03:00
ti = list_entry ( p , struct snd_timer_instance , ack_list ) ;
2005-10-12 19:20:21 +04:00
2005-04-17 02:20:36 +04:00
/* remove from ack_list and make empty */
list_del_init ( p ) ;
2005-10-12 19:20:21 +04:00
2005-04-17 02:20:36 +04:00
ticks = ti - > pticks ;
ti - > pticks = 0 ;
ti - > flags | = SNDRV_TIMER_IFLG_CALLBACK ;
spin_unlock ( & timer - > lock ) ;
if ( ti - > callback )
ti - > callback ( ti , resolution , ticks ) ;
spin_lock ( & timer - > lock ) ;
ti - > flags & = ~ SNDRV_TIMER_IFLG_CALLBACK ;
}
/* do we have any slow callbacks? */
use_tasklet = ! list_empty ( & timer - > sack_list_head ) ;
2005-11-18 20:52:14 +03:00
spin_unlock_irqrestore ( & timer - > lock , flags ) ;
2005-04-17 02:20:36 +04:00
if ( use_tasklet )
2008-12-18 14:17:55 +03:00
tasklet_schedule ( & timer - > task_queue ) ;
2005-04-17 02:20:36 +04:00
}
2017-06-16 17:16:05 +03:00
EXPORT_SYMBOL ( snd_timer_interrupt ) ;
2005-04-17 02:20:36 +04:00
/*
*/
2005-11-17 15:56:05 +03:00
int snd_timer_new ( struct snd_card * card , char * id , struct snd_timer_id * tid ,
struct snd_timer * * rtimer )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer * timer ;
2005-04-17 02:20:36 +04:00
int err ;
2005-11-17 15:56:05 +03:00
static struct snd_device_ops ops = {
2005-04-17 02:20:36 +04:00
. dev_free = snd_timer_dev_free ,
. dev_register = snd_timer_dev_register ,
2006-06-23 16:38:23 +04:00
. dev_disconnect = snd_timer_dev_disconnect ,
2005-04-17 02:20:36 +04:00
} ;
2008-08-08 19:09:09 +04:00
if ( snd_BUG_ON ( ! tid ) )
return - EINVAL ;
if ( rtimer )
* rtimer = NULL ;
2005-09-09 16:20:23 +04:00
timer = kzalloc ( sizeof ( * timer ) , GFP_KERNEL ) ;
2015-03-10 17:42:14 +03:00
if ( ! timer )
2005-04-17 02:20:36 +04:00
return - ENOMEM ;
timer - > tmr_class = tid - > dev_class ;
timer - > card = card ;
timer - > tmr_device = tid - > device ;
timer - > tmr_subdevice = tid - > subdevice ;
if ( id )
strlcpy ( timer - > id , id , sizeof ( timer - > id ) ) ;
2016-08-29 01:33:50 +03:00
timer - > sticks = 1 ;
2005-04-17 02:20:36 +04:00
INIT_LIST_HEAD ( & timer - > device_list ) ;
INIT_LIST_HEAD ( & timer - > open_list_head ) ;
INIT_LIST_HEAD ( & timer - > active_list_head ) ;
INIT_LIST_HEAD ( & timer - > ack_list_head ) ;
INIT_LIST_HEAD ( & timer - > sack_list_head ) ;
spin_lock_init ( & timer - > lock ) ;
2005-10-12 19:20:21 +04:00
tasklet_init ( & timer - > task_queue , snd_timer_tasklet ,
( unsigned long ) timer ) ;
2017-11-05 12:07:43 +03:00
timer - > max_instances = 1000 ; /* default limit per timer */
2005-04-17 02:20:36 +04:00
if ( card ! = NULL ) {
2005-10-12 19:12:31 +04:00
timer - > module = card - > module ;
2005-10-12 19:20:21 +04:00
err = snd_device_new ( card , SNDRV_DEV_TIMER , timer , & ops ) ;
if ( err < 0 ) {
2005-04-17 02:20:36 +04:00
snd_timer_free ( timer ) ;
return err ;
}
}
2008-08-08 19:09:09 +04:00
if ( rtimer )
* rtimer = timer ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2017-06-16 17:16:05 +03:00
EXPORT_SYMBOL ( snd_timer_new ) ;
2005-04-17 02:20:36 +04:00
2005-11-17 15:56:05 +03:00
static int snd_timer_free ( struct snd_timer * timer )
2005-04-17 02:20:36 +04:00
{
2008-08-08 19:09:09 +04:00
if ( ! timer )
return 0 ;
2006-06-23 16:38:23 +04:00
mutex_lock ( & register_mutex ) ;
if ( ! list_empty ( & timer - > open_list_head ) ) {
struct list_head * p , * n ;
struct snd_timer_instance * ti ;
2014-02-04 21:22:39 +04:00
pr_warn ( " ALSA: timer %p is busy? \n " , timer ) ;
2006-06-23 16:38:23 +04:00
list_for_each_safe ( p , n , & timer - > open_list_head ) {
list_del_init ( p ) ;
ti = list_entry ( p , struct snd_timer_instance , open_list ) ;
ti - > timer = NULL ;
}
}
list_del ( & timer - > device_list ) ;
mutex_unlock ( & register_mutex ) ;
2005-04-17 02:20:36 +04:00
if ( timer - > private_free )
timer - > private_free ( timer ) ;
kfree ( timer ) ;
return 0 ;
}
2005-11-17 15:56:05 +03:00
static int snd_timer_dev_free ( struct snd_device * device )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer * timer = device - > device_data ;
2005-04-17 02:20:36 +04:00
return snd_timer_free ( timer ) ;
}
2005-11-17 15:56:05 +03:00
static int snd_timer_dev_register ( struct snd_device * dev )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer * timer = dev - > device_data ;
struct snd_timer * timer1 ;
2005-04-17 02:20:36 +04:00
2008-08-08 19:09:09 +04:00
if ( snd_BUG_ON ( ! timer | | ! timer - > hw . start | | ! timer - > hw . stop ) )
return - ENXIO ;
2005-04-17 02:20:36 +04:00
if ( ! ( timer - > hw . flags & SNDRV_TIMER_HW_SLAVE ) & &
! timer - > hw . resolution & & timer - > hw . c_resolution = = NULL )
return - EINVAL ;
2006-01-16 18:29:08 +03:00
mutex_lock ( & register_mutex ) ;
2006-10-05 18:02:22 +04:00
list_for_each_entry ( timer1 , & snd_timer_list , device_list ) {
2005-04-17 02:20:36 +04:00
if ( timer1 - > tmr_class > timer - > tmr_class )
break ;
if ( timer1 - > tmr_class < timer - > tmr_class )
continue ;
if ( timer1 - > card & & timer - > card ) {
if ( timer1 - > card - > number > timer - > card - > number )
break ;
if ( timer1 - > card - > number < timer - > card - > number )
continue ;
}
if ( timer1 - > tmr_device > timer - > tmr_device )
break ;
if ( timer1 - > tmr_device < timer - > tmr_device )
continue ;
if ( timer1 - > tmr_subdevice > timer - > tmr_subdevice )
break ;
if ( timer1 - > tmr_subdevice < timer - > tmr_subdevice )
continue ;
/* conflicts.. */
2006-01-16 18:29:08 +03:00
mutex_unlock ( & register_mutex ) ;
2005-04-17 02:20:36 +04:00
return - EBUSY ;
}
2006-10-05 18:02:22 +04:00
list_add_tail ( & timer - > device_list , & timer1 - > device_list ) ;
2006-01-16 18:29:08 +03:00
mutex_unlock ( & register_mutex ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2006-06-23 16:38:23 +04:00
static int snd_timer_dev_disconnect ( struct snd_device * device )
2005-04-17 02:20:36 +04:00
{
2006-06-23 16:38:23 +04:00
struct snd_timer * timer = device - > device_data ;
2016-01-21 19:19:31 +03:00
struct snd_timer_instance * ti ;
2006-01-16 18:29:08 +03:00
mutex_lock ( & register_mutex ) ;
2006-06-23 16:38:23 +04:00
list_del_init ( & timer - > device_list ) ;
2016-01-21 19:19:31 +03:00
/* wake up pending sleepers */
list_for_each_entry ( ti , & timer - > open_list_head , open_list ) {
2016-01-21 19:43:08 +03:00
if ( ti - > disconnect )
ti - > disconnect ( ti ) ;
2016-01-21 19:19:31 +03:00
}
2006-01-16 18:29:08 +03:00
mutex_unlock ( & register_mutex ) ;
2006-06-23 16:38:23 +04:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
2005-11-17 15:56:05 +03:00
void snd_timer_notify ( struct snd_timer * timer , int event , struct timespec * tstamp )
2005-04-17 02:20:36 +04:00
{
unsigned long flags ;
unsigned long resolution = 0 ;
2005-11-17 15:56:05 +03:00
struct snd_timer_instance * ti , * ts ;
2005-04-17 02:20:36 +04:00
2016-01-21 19:19:31 +03:00
if ( timer - > card & & timer - > card - > shutdown )
return ;
2005-10-10 13:46:31 +04:00
if ( ! ( timer - > hw . flags & SNDRV_TIMER_HW_SLAVE ) )
return ;
2008-08-08 19:09:09 +04:00
if ( snd_BUG_ON ( event < SNDRV_TIMER_EVENT_MSTART | |
event > SNDRV_TIMER_EVENT_MRESUME ) )
return ;
2005-04-17 02:20:36 +04:00
spin_lock_irqsave ( & timer - > lock , flags ) ;
2005-08-16 13:09:05 +04:00
if ( event = = SNDRV_TIMER_EVENT_MSTART | |
event = = SNDRV_TIMER_EVENT_MCONTINUE | |
event = = SNDRV_TIMER_EVENT_MRESUME ) {
2005-04-17 02:20:36 +04:00
if ( timer - > hw . c_resolution )
resolution = timer - > hw . c_resolution ( timer ) ;
else
resolution = timer - > hw . resolution ;
}
2006-10-05 18:02:22 +04:00
list_for_each_entry ( ti , & timer - > active_list_head , active_list ) {
2005-04-17 02:20:36 +04:00
if ( ti - > ccallback )
ti - > ccallback ( ti , event , tstamp , resolution ) ;
2006-10-05 18:02:22 +04:00
list_for_each_entry ( ts , & ti - > slave_active_head , active_list )
2005-04-17 02:20:36 +04:00
if ( ts - > ccallback )
ts - > ccallback ( ts , event , tstamp , resolution ) ;
}
spin_unlock_irqrestore ( & timer - > lock , flags ) ;
}
2017-06-16 17:16:05 +03:00
EXPORT_SYMBOL ( snd_timer_notify ) ;
2005-04-17 02:20:36 +04:00
/*
* exported functions for global timers
*/
2005-11-17 15:56:05 +03:00
int snd_timer_global_new ( char * id , int device , struct snd_timer * * rtimer )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer_id tid ;
2005-10-12 19:20:21 +04:00
2005-04-17 02:20:36 +04:00
tid . dev_class = SNDRV_TIMER_CLASS_GLOBAL ;
tid . dev_sclass = SNDRV_TIMER_SCLASS_NONE ;
tid . card = - 1 ;
tid . device = device ;
tid . subdevice = 0 ;
return snd_timer_new ( NULL , id , & tid , rtimer ) ;
}
2017-06-16 17:16:05 +03:00
EXPORT_SYMBOL ( snd_timer_global_new ) ;
2005-04-17 02:20:36 +04:00
2005-11-17 15:56:05 +03:00
int snd_timer_global_free ( struct snd_timer * timer )
2005-04-17 02:20:36 +04:00
{
return snd_timer_free ( timer ) ;
}
2017-06-16 17:16:05 +03:00
EXPORT_SYMBOL ( snd_timer_global_free ) ;
2005-04-17 02:20:36 +04:00
2005-11-17 15:56:05 +03:00
int snd_timer_global_register ( struct snd_timer * timer )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_device dev ;
2005-04-17 02:20:36 +04:00
memset ( & dev , 0 , sizeof ( dev ) ) ;
dev . device_data = timer ;
return snd_timer_dev_register ( & dev ) ;
}
2017-06-16 17:16:05 +03:00
EXPORT_SYMBOL ( snd_timer_global_register ) ;
2005-04-17 02:20:36 +04:00
2005-10-12 19:20:21 +04:00
/*
2005-04-17 02:20:36 +04:00
* System timer
*/
struct snd_timer_system_private {
struct timer_list tlist ;
2017-10-05 03:53:33 +03:00
struct snd_timer * snd_timer ;
2005-04-17 02:20:36 +04:00
unsigned long last_expires ;
unsigned long last_jiffies ;
unsigned long correction ;
} ;
2017-10-05 03:53:33 +03:00
static void snd_timer_s_function ( struct timer_list * t )
2005-04-17 02:20:36 +04:00
{
2017-10-05 03:53:33 +03:00
struct snd_timer_system_private * priv = from_timer ( priv , t ,
tlist ) ;
struct snd_timer * timer = priv - > snd_timer ;
2005-04-17 02:20:36 +04:00
unsigned long jiff = jiffies ;
if ( time_after ( jiff , priv - > last_expires ) )
2006-07-17 18:51:37 +04:00
priv - > correction + = ( long ) jiff - ( long ) priv - > last_expires ;
2005-04-17 02:20:36 +04:00
snd_timer_interrupt ( timer , ( long ) jiff - ( long ) priv - > last_jiffies ) ;
}
2005-11-17 15:56:05 +03:00
static int snd_timer_s_start ( struct snd_timer * timer )
2005-04-17 02:20:36 +04:00
{
struct snd_timer_system_private * priv ;
unsigned long njiff ;
priv = ( struct snd_timer_system_private * ) timer - > private_data ;
njiff = ( priv - > last_jiffies = jiffies ) ;
if ( priv - > correction > timer - > sticks - 1 ) {
priv - > correction - = timer - > sticks - 1 ;
njiff + + ;
} else {
njiff + = timer - > sticks - priv - > correction ;
2006-07-17 18:50:56 +04:00
priv - > correction = 0 ;
2005-04-17 02:20:36 +04:00
}
2016-04-01 13:28:16 +03:00
priv - > last_expires = njiff ;
mod_timer ( & priv - > tlist , njiff ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2005-11-17 15:56:05 +03:00
static int snd_timer_s_stop ( struct snd_timer * timer )
2005-04-17 02:20:36 +04:00
{
struct snd_timer_system_private * priv ;
unsigned long jiff ;
priv = ( struct snd_timer_system_private * ) timer - > private_data ;
del_timer ( & priv - > tlist ) ;
jiff = jiffies ;
if ( time_before ( jiff , priv - > last_expires ) )
timer - > sticks = priv - > last_expires - jiff ;
else
timer - > sticks = 1 ;
2006-07-17 18:52:09 +04:00
priv - > correction = 0 ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2016-02-02 16:14:10 +03:00
static int snd_timer_s_close ( struct snd_timer * timer )
{
struct snd_timer_system_private * priv ;
priv = ( struct snd_timer_system_private * ) timer - > private_data ;
del_timer_sync ( & priv - > tlist ) ;
return 0 ;
}
2005-11-17 15:56:05 +03:00
static struct snd_timer_hardware snd_timer_system =
2005-04-17 02:20:36 +04:00
{
. flags = SNDRV_TIMER_HW_FIRST | SNDRV_TIMER_HW_TASKLET ,
. resolution = 1000000000L / HZ ,
. ticks = 10000000L ,
2016-02-02 16:14:10 +03:00
. close = snd_timer_s_close ,
2005-04-17 02:20:36 +04:00
. start = snd_timer_s_start ,
. stop = snd_timer_s_stop
} ;
2005-11-17 15:56:05 +03:00
static void snd_timer_free_system ( struct snd_timer * timer )
2005-04-17 02:20:36 +04:00
{
kfree ( timer - > private_data ) ;
}
static int snd_timer_register_system ( void )
{
2005-11-17 15:56:05 +03:00
struct snd_timer * timer ;
2005-04-17 02:20:36 +04:00
struct snd_timer_system_private * priv ;
int err ;
2005-10-12 19:20:21 +04:00
err = snd_timer_global_new ( " system " , SNDRV_TIMER_GLOBAL_SYSTEM , & timer ) ;
if ( err < 0 )
2005-04-17 02:20:36 +04:00
return err ;
strcpy ( timer - > name , " system timer " ) ;
timer - > hw = snd_timer_system ;
2005-09-09 16:20:23 +04:00
priv = kzalloc ( sizeof ( * priv ) , GFP_KERNEL ) ;
2005-04-17 02:20:36 +04:00
if ( priv = = NULL ) {
snd_timer_free ( timer ) ;
return - ENOMEM ;
}
2017-10-05 03:53:33 +03:00
priv - > snd_timer = timer ;
timer_setup ( & priv - > tlist , snd_timer_s_function , 0 ) ;
2005-04-17 02:20:36 +04:00
timer - > private_data = priv ;
timer - > private_free = snd_timer_free_system ;
return snd_timer_global_register ( timer ) ;
}
2015-05-27 14:45:45 +03:00
# ifdef CONFIG_SND_PROC_FS
2005-04-17 02:20:36 +04:00
/*
* Info interface
*/
2005-11-17 15:56:05 +03:00
static void snd_timer_proc_read ( struct snd_info_entry * entry ,
struct snd_info_buffer * buffer )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer * timer ;
struct snd_timer_instance * ti ;
2005-04-17 02:20:36 +04:00
2006-01-16 18:29:08 +03:00
mutex_lock ( & register_mutex ) ;
2006-10-05 18:02:22 +04:00
list_for_each_entry ( timer , & snd_timer_list , device_list ) {
2016-01-21 19:19:31 +03:00
if ( timer - > card & & timer - > card - > shutdown )
continue ;
2005-04-17 02:20:36 +04:00
switch ( timer - > tmr_class ) {
case SNDRV_TIMER_CLASS_GLOBAL :
snd_iprintf ( buffer , " G%i: " , timer - > tmr_device ) ;
break ;
case SNDRV_TIMER_CLASS_CARD :
2005-10-12 19:20:21 +04:00
snd_iprintf ( buffer , " C%i-%i: " ,
timer - > card - > number , timer - > tmr_device ) ;
2005-04-17 02:20:36 +04:00
break ;
case SNDRV_TIMER_CLASS_PCM :
2005-10-12 19:20:21 +04:00
snd_iprintf ( buffer , " P%i-%i-%i: " , timer - > card - > number ,
timer - > tmr_device , timer - > tmr_subdevice ) ;
2005-04-17 02:20:36 +04:00
break ;
default :
2005-10-12 19:20:21 +04:00
snd_iprintf ( buffer , " ?%i-%i-%i-%i: " , timer - > tmr_class ,
timer - > card ? timer - > card - > number : - 1 ,
timer - > tmr_device , timer - > tmr_subdevice ) ;
2005-04-17 02:20:36 +04:00
}
snd_iprintf ( buffer , " %s : " , timer - > name ) ;
if ( timer - > hw . resolution )
2005-10-12 19:20:21 +04:00
snd_iprintf ( buffer , " %lu.%03luus (%lu ticks) " ,
timer - > hw . resolution / 1000 ,
timer - > hw . resolution % 1000 ,
timer - > hw . ticks ) ;
2005-04-17 02:20:36 +04:00
if ( timer - > hw . flags & SNDRV_TIMER_HW_SLAVE )
snd_iprintf ( buffer , " SLAVE " ) ;
snd_iprintf ( buffer , " \n " ) ;
2006-10-05 18:02:22 +04:00
list_for_each_entry ( ti , & timer - > open_list_head , open_list )
2005-10-12 19:20:21 +04:00
snd_iprintf ( buffer , " Client %s : %s \n " ,
ti - > owner ? ti - > owner : " unknown " ,
ti - > flags & ( SNDRV_TIMER_IFLG_START |
SNDRV_TIMER_IFLG_RUNNING )
? " running " : " stopped " ) ;
2005-04-17 02:20:36 +04:00
}
2006-01-16 18:29:08 +03:00
mutex_unlock ( & register_mutex ) ;
2005-04-17 02:20:36 +04:00
}
2006-05-17 19:14:51 +04:00
static struct snd_info_entry * snd_timer_proc_entry ;
2005-12-01 12:42:42 +03:00
static void __init snd_timer_proc_init ( void )
{
struct snd_info_entry * entry ;
entry = snd_info_create_module_entry ( THIS_MODULE , " timers " , NULL ) ;
if ( entry ! = NULL ) {
entry - > c . text . read = snd_timer_proc_read ;
if ( snd_info_register ( entry ) < 0 ) {
snd_info_free_entry ( entry ) ;
entry = NULL ;
}
}
snd_timer_proc_entry = entry ;
}
static void __exit snd_timer_proc_done ( void )
{
2006-06-23 16:37:59 +04:00
snd_info_free_entry ( snd_timer_proc_entry ) ;
2005-12-01 12:42:42 +03:00
}
2015-05-27 14:45:45 +03:00
# else /* !CONFIG_SND_PROC_FS */
2005-12-01 12:42:42 +03:00
# define snd_timer_proc_init()
# define snd_timer_proc_done()
# endif
2005-04-17 02:20:36 +04:00
/*
* USER SPACE interface
*/
2005-11-17 15:56:05 +03:00
static void snd_timer_user_interrupt ( struct snd_timer_instance * timeri ,
2005-04-17 02:20:36 +04:00
unsigned long resolution ,
unsigned long ticks )
{
2005-11-17 15:56:05 +03:00
struct snd_timer_user * tu = timeri - > callback_data ;
struct snd_timer_read * r ;
2005-04-17 02:20:36 +04:00
int prev ;
2005-10-12 19:20:21 +04:00
2005-04-17 02:20:36 +04:00
spin_lock ( & tu - > qlock ) ;
if ( tu - > qused > 0 ) {
prev = tu - > qtail = = 0 ? tu - > queue_size - 1 : tu - > qtail - 1 ;
r = & tu - > queue [ prev ] ;
if ( r - > resolution = = resolution ) {
r - > ticks + = ticks ;
goto __wake ;
}
}
if ( tu - > qused > = tu - > queue_size ) {
tu - > overrun + + ;
} else {
r = & tu - > queue [ tu - > qtail + + ] ;
tu - > qtail % = tu - > queue_size ;
r - > resolution = resolution ;
r - > ticks = ticks ;
tu - > qused + + ;
}
__wake :
spin_unlock ( & tu - > qlock ) ;
kill_fasync ( & tu - > fasync , SIGIO , POLL_IN ) ;
wake_up ( & tu - > qchange_sleep ) ;
}
2005-11-17 15:56:05 +03:00
static void snd_timer_user_append_to_tqueue ( struct snd_timer_user * tu ,
struct snd_timer_tread * tread )
2005-04-17 02:20:36 +04:00
{
if ( tu - > qused > = tu - > queue_size ) {
tu - > overrun + + ;
} else {
memcpy ( & tu - > tqueue [ tu - > qtail + + ] , tread , sizeof ( * tread ) ) ;
tu - > qtail % = tu - > queue_size ;
tu - > qused + + ;
}
}
2005-11-17 15:56:05 +03:00
static void snd_timer_user_ccallback ( struct snd_timer_instance * timeri ,
int event ,
2005-04-17 02:20:36 +04:00
struct timespec * tstamp ,
unsigned long resolution )
{
2005-11-17 15:56:05 +03:00
struct snd_timer_user * tu = timeri - > callback_data ;
struct snd_timer_tread r1 ;
2010-04-28 12:29:14 +04:00
unsigned long flags ;
2005-04-17 02:20:36 +04:00
2005-10-12 19:20:21 +04:00
if ( event > = SNDRV_TIMER_EVENT_START & &
event < = SNDRV_TIMER_EVENT_PAUSE )
2005-04-17 02:20:36 +04:00
tu - > tstamp = * tstamp ;
if ( ( tu - > filter & ( 1 < < event ) ) = = 0 | | ! tu - > tread )
return ;
2016-05-03 23:44:20 +03:00
memset ( & r1 , 0 , sizeof ( r1 ) ) ;
2005-04-17 02:20:36 +04:00
r1 . event = event ;
r1 . tstamp = * tstamp ;
r1 . val = resolution ;
2010-04-28 12:29:14 +04:00
spin_lock_irqsave ( & tu - > qlock , flags ) ;
2005-04-17 02:20:36 +04:00
snd_timer_user_append_to_tqueue ( tu , & r1 ) ;
2010-04-28 12:29:14 +04:00
spin_unlock_irqrestore ( & tu - > qlock , flags ) ;
2005-04-17 02:20:36 +04:00
kill_fasync ( & tu - > fasync , SIGIO , POLL_IN ) ;
wake_up ( & tu - > qchange_sleep ) ;
}
2016-01-21 19:43:08 +03:00
static void snd_timer_user_disconnect ( struct snd_timer_instance * timeri )
{
struct snd_timer_user * tu = timeri - > callback_data ;
tu - > disconnected = true ;
wake_up ( & tu - > qchange_sleep ) ;
}
2005-11-17 15:56:05 +03:00
static void snd_timer_user_tinterrupt ( struct snd_timer_instance * timeri ,
2005-04-17 02:20:36 +04:00
unsigned long resolution ,
unsigned long ticks )
{
2005-11-17 15:56:05 +03:00
struct snd_timer_user * tu = timeri - > callback_data ;
struct snd_timer_tread * r , r1 ;
2005-04-17 02:20:36 +04:00
struct timespec tstamp ;
int prev , append = 0 ;
2017-03-31 18:22:23 +03:00
memset ( & r1 , 0 , sizeof ( r1 ) ) ;
2005-10-10 13:49:49 +04:00
memset ( & tstamp , 0 , sizeof ( tstamp ) ) ;
2005-04-17 02:20:36 +04:00
spin_lock ( & tu - > qlock ) ;
2005-10-12 19:20:21 +04:00
if ( ( tu - > filter & ( ( 1 < < SNDRV_TIMER_EVENT_RESOLUTION ) |
( 1 < < SNDRV_TIMER_EVENT_TICK ) ) ) = = 0 ) {
2005-04-17 02:20:36 +04:00
spin_unlock ( & tu - > qlock ) ;
return ;
}
2007-12-13 12:19:42 +03:00
if ( tu - > last_resolution ! = resolution | | ticks > 0 ) {
if ( timer_tstamp_monotonic )
2014-06-12 03:59:14 +04:00
ktime_get_ts ( & tstamp ) ;
2007-12-13 12:19:42 +03:00
else
getnstimeofday ( & tstamp ) ;
}
2005-10-12 19:20:21 +04:00
if ( ( tu - > filter & ( 1 < < SNDRV_TIMER_EVENT_RESOLUTION ) ) & &
tu - > last_resolution ! = resolution ) {
2005-04-17 02:20:36 +04:00
r1 . event = SNDRV_TIMER_EVENT_RESOLUTION ;
r1 . tstamp = tstamp ;
r1 . val = resolution ;
snd_timer_user_append_to_tqueue ( tu , & r1 ) ;
tu - > last_resolution = resolution ;
append + + ;
}
if ( ( tu - > filter & ( 1 < < SNDRV_TIMER_EVENT_TICK ) ) = = 0 )
goto __wake ;
if ( ticks = = 0 )
goto __wake ;
if ( tu - > qused > 0 ) {
prev = tu - > qtail = = 0 ? tu - > queue_size - 1 : tu - > qtail - 1 ;
r = & tu - > tqueue [ prev ] ;
if ( r - > event = = SNDRV_TIMER_EVENT_TICK ) {
r - > tstamp = tstamp ;
r - > val + = ticks ;
append + + ;
goto __wake ;
}
}
r1 . event = SNDRV_TIMER_EVENT_TICK ;
r1 . tstamp = tstamp ;
r1 . val = ticks ;
snd_timer_user_append_to_tqueue ( tu , & r1 ) ;
append + + ;
__wake :
spin_unlock ( & tu - > qlock ) ;
if ( append = = 0 )
return ;
kill_fasync ( & tu - > fasync , SIGIO , POLL_IN ) ;
wake_up ( & tu - > qchange_sleep ) ;
}
ALSA: timer: Improve user queue reallocation
ALSA timer may reallocate the user queue upon request, and it happens
at three places for now: at opening, at SNDRV_TIMER_IOCTL_PARAMS, and
at SNDRV_TIMER_IOCTL_SELECT. However, the last one,
snd_timer_user_tselect(), doesn't need to reallocate the buffer since
it doesn't change the queue size. It does just because tu->tread
might have been changed before starting the timer.
Instead of *_SELECT ioctl, we should reallocate the queue at
SNDRV_TIMER_IOCTL_TREAD; then the timer is guaranteed to be stopped,
thus we can reassign the buffer more safely.
This patch implements that with a slight code refactoring.
Essentially, the patch achieves:
- Introduce realloc_user_queue() for (re-)allocating the ring buffer,
and call it from all places. Also, realloc_user_queue() uses
kcalloc() for avoiding possible leaks.
- Add the buffer reallocation at SNDRV_TIMER_IOCTL_TREAD. When it
fails, tu->tread is restored to the old value, too.
- Drop the buffer reallocation at snd_timer_user_tselect().
Tested-by: Alexander Potapenko <glider@google.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2017-06-02 18:16:59 +03:00
static int realloc_user_queue ( struct snd_timer_user * tu , int size )
{
struct snd_timer_read * queue = NULL ;
struct snd_timer_tread * tqueue = NULL ;
if ( tu - > tread ) {
tqueue = kcalloc ( size , sizeof ( * tqueue ) , GFP_KERNEL ) ;
if ( ! tqueue )
return - ENOMEM ;
} else {
queue = kcalloc ( size , sizeof ( * queue ) , GFP_KERNEL ) ;
if ( ! queue )
return - ENOMEM ;
}
spin_lock_irq ( & tu - > qlock ) ;
kfree ( tu - > queue ) ;
kfree ( tu - > tqueue ) ;
tu - > queue_size = size ;
tu - > queue = queue ;
tu - > tqueue = tqueue ;
tu - > qhead = tu - > qtail = tu - > qused = 0 ;
spin_unlock_irq ( & tu - > qlock ) ;
return 0 ;
}
2005-04-17 02:20:36 +04:00
static int snd_timer_user_open ( struct inode * inode , struct file * file )
{
2005-11-17 15:56:05 +03:00
struct snd_timer_user * tu ;
2010-04-13 13:49:04 +04:00
int err ;
err = nonseekable_open ( inode , file ) ;
if ( err < 0 )
return err ;
2005-10-12 19:20:21 +04:00
2005-09-09 16:20:23 +04:00
tu = kzalloc ( sizeof ( * tu ) , GFP_KERNEL ) ;
2005-04-17 02:20:36 +04:00
if ( tu = = NULL )
return - ENOMEM ;
spin_lock_init ( & tu - > qlock ) ;
init_waitqueue_head ( & tu - > qchange_sleep ) ;
2016-01-13 19:48:01 +03:00
mutex_init ( & tu - > ioctl_lock ) ;
2005-04-17 02:20:36 +04:00
tu - > ticks = 1 ;
ALSA: timer: Improve user queue reallocation
ALSA timer may reallocate the user queue upon request, and it happens
at three places for now: at opening, at SNDRV_TIMER_IOCTL_PARAMS, and
at SNDRV_TIMER_IOCTL_SELECT. However, the last one,
snd_timer_user_tselect(), doesn't need to reallocate the buffer since
it doesn't change the queue size. It does just because tu->tread
might have been changed before starting the timer.
Instead of *_SELECT ioctl, we should reallocate the queue at
SNDRV_TIMER_IOCTL_TREAD; then the timer is guaranteed to be stopped,
thus we can reassign the buffer more safely.
This patch implements that with a slight code refactoring.
Essentially, the patch achieves:
- Introduce realloc_user_queue() for (re-)allocating the ring buffer,
and call it from all places. Also, realloc_user_queue() uses
kcalloc() for avoiding possible leaks.
- Add the buffer reallocation at SNDRV_TIMER_IOCTL_TREAD. When it
fails, tu->tread is restored to the old value, too.
- Drop the buffer reallocation at snd_timer_user_tselect().
Tested-by: Alexander Potapenko <glider@google.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2017-06-02 18:16:59 +03:00
if ( realloc_user_queue ( tu , 128 ) < 0 ) {
2005-04-17 02:20:36 +04:00
kfree ( tu ) ;
return - ENOMEM ;
}
file - > private_data = tu ;
return 0 ;
}
static int snd_timer_user_release ( struct inode * inode , struct file * file )
{
2005-11-17 15:56:05 +03:00
struct snd_timer_user * tu ;
2005-04-17 02:20:36 +04:00
if ( file - > private_data ) {
tu = file - > private_data ;
file - > private_data = NULL ;
2016-01-13 19:48:01 +03:00
mutex_lock ( & tu - > ioctl_lock ) ;
2005-04-17 02:20:36 +04:00
if ( tu - > timeri )
snd_timer_close ( tu - > timeri ) ;
2016-01-13 19:48:01 +03:00
mutex_unlock ( & tu - > ioctl_lock ) ;
2005-04-17 02:20:36 +04:00
kfree ( tu - > queue ) ;
kfree ( tu - > tqueue ) ;
kfree ( tu ) ;
}
return 0 ;
}
2005-11-17 15:56:05 +03:00
static void snd_timer_user_zero_id ( struct snd_timer_id * id )
2005-04-17 02:20:36 +04:00
{
id - > dev_class = SNDRV_TIMER_CLASS_NONE ;
id - > dev_sclass = SNDRV_TIMER_SCLASS_NONE ;
id - > card = - 1 ;
id - > device = - 1 ;
id - > subdevice = - 1 ;
}
2005-11-17 15:56:05 +03:00
static void snd_timer_user_copy_id ( struct snd_timer_id * id , struct snd_timer * timer )
2005-04-17 02:20:36 +04:00
{
id - > dev_class = timer - > tmr_class ;
id - > dev_sclass = SNDRV_TIMER_SCLASS_NONE ;
id - > card = timer - > card ? timer - > card - > number : - 1 ;
id - > device = timer - > tmr_device ;
id - > subdevice = timer - > tmr_subdevice ;
}
2005-11-17 15:56:05 +03:00
static int snd_timer_user_next_device ( struct snd_timer_id __user * _tid )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer_id id ;
struct snd_timer * timer ;
2005-04-17 02:20:36 +04:00
struct list_head * p ;
2005-10-12 19:20:21 +04:00
2005-04-17 02:20:36 +04:00
if ( copy_from_user ( & id , _tid , sizeof ( id ) ) )
return - EFAULT ;
2006-01-16 18:29:08 +03:00
mutex_lock ( & register_mutex ) ;
2005-04-17 02:20:36 +04:00
if ( id . dev_class < 0 ) { /* first item */
if ( list_empty ( & snd_timer_list ) )
snd_timer_user_zero_id ( & id ) ;
else {
2005-10-12 19:14:55 +04:00
timer = list_entry ( snd_timer_list . next ,
2005-11-17 15:56:05 +03:00
struct snd_timer , device_list ) ;
2005-04-17 02:20:36 +04:00
snd_timer_user_copy_id ( & id , timer ) ;
}
} else {
switch ( id . dev_class ) {
case SNDRV_TIMER_CLASS_GLOBAL :
id . device = id . device < 0 ? 0 : id . device + 1 ;
list_for_each ( p , & snd_timer_list ) {
2005-11-17 15:56:05 +03:00
timer = list_entry ( p , struct snd_timer , device_list ) ;
2005-04-17 02:20:36 +04:00
if ( timer - > tmr_class > SNDRV_TIMER_CLASS_GLOBAL ) {
snd_timer_user_copy_id ( & id , timer ) ;
break ;
}
if ( timer - > tmr_device > = id . device ) {
snd_timer_user_copy_id ( & id , timer ) ;
break ;
}
}
if ( p = = & snd_timer_list )
snd_timer_user_zero_id ( & id ) ;
break ;
case SNDRV_TIMER_CLASS_CARD :
case SNDRV_TIMER_CLASS_PCM :
if ( id . card < 0 ) {
id . card = 0 ;
} else {
2017-03-31 18:21:41 +03:00
if ( id . device < 0 ) {
id . device = 0 ;
2005-04-17 02:20:36 +04:00
} else {
2017-03-31 18:21:41 +03:00
if ( id . subdevice < 0 )
id . subdevice = 0 ;
else
id . subdevice + + ;
2005-04-17 02:20:36 +04:00
}
}
list_for_each ( p , & snd_timer_list ) {
2005-11-17 15:56:05 +03:00
timer = list_entry ( p , struct snd_timer , device_list ) ;
2005-04-17 02:20:36 +04:00
if ( timer - > tmr_class > id . dev_class ) {
snd_timer_user_copy_id ( & id , timer ) ;
break ;
}
if ( timer - > tmr_class < id . dev_class )
continue ;
if ( timer - > card - > number > id . card ) {
snd_timer_user_copy_id ( & id , timer ) ;
break ;
}
if ( timer - > card - > number < id . card )
continue ;
if ( timer - > tmr_device > id . device ) {
snd_timer_user_copy_id ( & id , timer ) ;
break ;
}
if ( timer - > tmr_device < id . device )
continue ;
if ( timer - > tmr_subdevice > id . subdevice ) {
snd_timer_user_copy_id ( & id , timer ) ;
break ;
}
if ( timer - > tmr_subdevice < id . subdevice )
continue ;
snd_timer_user_copy_id ( & id , timer ) ;
break ;
}
if ( p = = & snd_timer_list )
snd_timer_user_zero_id ( & id ) ;
break ;
default :
snd_timer_user_zero_id ( & id ) ;
}
}
2006-01-16 18:29:08 +03:00
mutex_unlock ( & register_mutex ) ;
2005-04-17 02:20:36 +04:00
if ( copy_to_user ( _tid , & id , sizeof ( * _tid ) ) )
return - EFAULT ;
return 0 ;
2005-10-12 19:20:21 +04:00
}
2005-04-17 02:20:36 +04:00
2005-10-12 19:20:21 +04:00
static int snd_timer_user_ginfo ( struct file * file ,
2005-11-17 15:56:05 +03:00
struct snd_timer_ginfo __user * _ginfo )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer_ginfo * ginfo ;
struct snd_timer_id tid ;
struct snd_timer * t ;
2005-04-17 02:20:36 +04:00
struct list_head * p ;
int err = 0 ;
2009-04-10 05:43:08 +04:00
ginfo = memdup_user ( _ginfo , sizeof ( * ginfo ) ) ;
if ( IS_ERR ( ginfo ) )
return PTR_ERR ( ginfo ) ;
2005-04-17 02:20:36 +04:00
tid = ginfo - > tid ;
memset ( ginfo , 0 , sizeof ( * ginfo ) ) ;
ginfo - > tid = tid ;
2006-01-16 18:29:08 +03:00
mutex_lock ( & register_mutex ) ;
2005-04-17 02:20:36 +04:00
t = snd_timer_find ( & tid ) ;
if ( t ! = NULL ) {
ginfo - > card = t - > card ? t - > card - > number : - 1 ;
if ( t - > hw . flags & SNDRV_TIMER_HW_SLAVE )
ginfo - > flags | = SNDRV_TIMER_FLG_SLAVE ;
strlcpy ( ginfo - > id , t - > id , sizeof ( ginfo - > id ) ) ;
strlcpy ( ginfo - > name , t - > name , sizeof ( ginfo - > name ) ) ;
ginfo - > resolution = t - > hw . resolution ;
if ( t - > hw . resolution_min > 0 ) {
ginfo - > resolution_min = t - > hw . resolution_min ;
ginfo - > resolution_max = t - > hw . resolution_max ;
}
list_for_each ( p , & t - > open_list_head ) {
ginfo - > clients + + ;
}
} else {
err = - ENODEV ;
}
2006-01-16 18:29:08 +03:00
mutex_unlock ( & register_mutex ) ;
2005-04-17 02:20:36 +04:00
if ( err > = 0 & & copy_to_user ( _ginfo , ginfo , sizeof ( * ginfo ) ) )
err = - EFAULT ;
kfree ( ginfo ) ;
return err ;
}
2016-03-23 02:03:59 +03:00
static int timer_set_gparams ( struct snd_timer_gparams * gparams )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer * t ;
2005-04-17 02:20:36 +04:00
int err ;
2006-01-16 18:29:08 +03:00
mutex_lock ( & register_mutex ) ;
2016-03-23 02:03:59 +03:00
t = snd_timer_find ( & gparams - > tid ) ;
2005-10-12 19:20:21 +04:00
if ( ! t ) {
2005-04-17 02:20:36 +04:00
err = - ENODEV ;
2005-10-12 19:20:21 +04:00
goto _error ;
}
if ( ! list_empty ( & t - > open_list_head ) ) {
err = - EBUSY ;
goto _error ;
}
if ( ! t - > hw . set_period ) {
err = - ENOSYS ;
goto _error ;
2005-04-17 02:20:36 +04:00
}
2016-03-23 02:03:59 +03:00
err = t - > hw . set_period ( t , gparams - > period_num , gparams - > period_den ) ;
2005-10-12 19:20:21 +04:00
_error :
2006-01-16 18:29:08 +03:00
mutex_unlock ( & register_mutex ) ;
2005-04-17 02:20:36 +04:00
return err ;
}
2016-03-23 02:03:59 +03:00
static int snd_timer_user_gparams ( struct file * file ,
struct snd_timer_gparams __user * _gparams )
{
struct snd_timer_gparams gparams ;
if ( copy_from_user ( & gparams , _gparams , sizeof ( gparams ) ) )
return - EFAULT ;
return timer_set_gparams ( & gparams ) ;
}
2005-10-12 19:20:21 +04:00
static int snd_timer_user_gstatus ( struct file * file ,
2005-11-17 15:56:05 +03:00
struct snd_timer_gstatus __user * _gstatus )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer_gstatus gstatus ;
struct snd_timer_id tid ;
struct snd_timer * t ;
2005-04-17 02:20:36 +04:00
int err = 0 ;
if ( copy_from_user ( & gstatus , _gstatus , sizeof ( gstatus ) ) )
return - EFAULT ;
tid = gstatus . tid ;
memset ( & gstatus , 0 , sizeof ( gstatus ) ) ;
gstatus . tid = tid ;
2006-01-16 18:29:08 +03:00
mutex_lock ( & register_mutex ) ;
2005-04-17 02:20:36 +04:00
t = snd_timer_find ( & tid ) ;
if ( t ! = NULL ) {
if ( t - > hw . c_resolution )
gstatus . resolution = t - > hw . c_resolution ( t ) ;
else
gstatus . resolution = t - > hw . resolution ;
if ( t - > hw . precise_resolution ) {
2005-10-12 19:20:21 +04:00
t - > hw . precise_resolution ( t , & gstatus . resolution_num ,
& gstatus . resolution_den ) ;
2005-04-17 02:20:36 +04:00
} else {
gstatus . resolution_num = gstatus . resolution ;
gstatus . resolution_den = 1000000000uL ;
}
} else {
err = - ENODEV ;
}
2006-01-16 18:29:08 +03:00
mutex_unlock ( & register_mutex ) ;
2005-04-17 02:20:36 +04:00
if ( err > = 0 & & copy_to_user ( _gstatus , & gstatus , sizeof ( gstatus ) ) )
err = - EFAULT ;
return err ;
}
2005-10-12 19:20:21 +04:00
static int snd_timer_user_tselect ( struct file * file ,
2005-11-17 15:56:05 +03:00
struct snd_timer_select __user * _tselect )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer_user * tu ;
struct snd_timer_select tselect ;
2005-04-17 02:20:36 +04:00
char str [ 32 ] ;
2005-04-04 18:44:58 +04:00
int err = 0 ;
2005-10-12 19:20:21 +04:00
2005-04-17 02:20:36 +04:00
tu = file - > private_data ;
2005-04-04 18:44:58 +04:00
if ( tu - > timeri ) {
2005-04-17 02:20:36 +04:00
snd_timer_close ( tu - > timeri ) ;
2005-04-04 18:44:58 +04:00
tu - > timeri = NULL ;
}
if ( copy_from_user ( & tselect , _tselect , sizeof ( tselect ) ) ) {
err = - EFAULT ;
goto __err ;
}
2005-04-17 02:20:36 +04:00
sprintf ( str , " application %i " , current - > pid ) ;
if ( tselect . id . dev_class ! = SNDRV_TIMER_CLASS_SLAVE )
tselect . id . dev_sclass = SNDRV_TIMER_SCLASS_APPLICATION ;
2005-10-12 19:20:21 +04:00
err = snd_timer_open ( & tu - > timeri , str , & tselect . id , current - > pid ) ;
if ( err < 0 )
2005-04-04 18:44:58 +04:00
goto __err ;
2005-04-17 02:20:36 +04:00
ALSA: timer: Improve user queue reallocation
ALSA timer may reallocate the user queue upon request, and it happens
at three places for now: at opening, at SNDRV_TIMER_IOCTL_PARAMS, and
at SNDRV_TIMER_IOCTL_SELECT. However, the last one,
snd_timer_user_tselect(), doesn't need to reallocate the buffer since
it doesn't change the queue size. It does just because tu->tread
might have been changed before starting the timer.
Instead of *_SELECT ioctl, we should reallocate the queue at
SNDRV_TIMER_IOCTL_TREAD; then the timer is guaranteed to be stopped,
thus we can reassign the buffer more safely.
This patch implements that with a slight code refactoring.
Essentially, the patch achieves:
- Introduce realloc_user_queue() for (re-)allocating the ring buffer,
and call it from all places. Also, realloc_user_queue() uses
kcalloc() for avoiding possible leaks.
- Add the buffer reallocation at SNDRV_TIMER_IOCTL_TREAD. When it
fails, tu->tread is restored to the old value, too.
- Drop the buffer reallocation at snd_timer_user_tselect().
Tested-by: Alexander Potapenko <glider@google.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2017-06-02 18:16:59 +03:00
tu - > timeri - > flags | = SNDRV_TIMER_IFLG_FAST ;
tu - > timeri - > callback = tu - > tread
2005-10-12 19:20:21 +04:00
? snd_timer_user_tinterrupt : snd_timer_user_interrupt ;
ALSA: timer: Improve user queue reallocation
ALSA timer may reallocate the user queue upon request, and it happens
at three places for now: at opening, at SNDRV_TIMER_IOCTL_PARAMS, and
at SNDRV_TIMER_IOCTL_SELECT. However, the last one,
snd_timer_user_tselect(), doesn't need to reallocate the buffer since
it doesn't change the queue size. It does just because tu->tread
might have been changed before starting the timer.
Instead of *_SELECT ioctl, we should reallocate the queue at
SNDRV_TIMER_IOCTL_TREAD; then the timer is guaranteed to be stopped,
thus we can reassign the buffer more safely.
This patch implements that with a slight code refactoring.
Essentially, the patch achieves:
- Introduce realloc_user_queue() for (re-)allocating the ring buffer,
and call it from all places. Also, realloc_user_queue() uses
kcalloc() for avoiding possible leaks.
- Add the buffer reallocation at SNDRV_TIMER_IOCTL_TREAD. When it
fails, tu->tread is restored to the old value, too.
- Drop the buffer reallocation at snd_timer_user_tselect().
Tested-by: Alexander Potapenko <glider@google.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2017-06-02 18:16:59 +03:00
tu - > timeri - > ccallback = snd_timer_user_ccallback ;
tu - > timeri - > callback_data = ( void * ) tu ;
tu - > timeri - > disconnect = snd_timer_user_disconnect ;
2005-04-04 18:44:58 +04:00
__err :
return err ;
2005-04-17 02:20:36 +04:00
}
2005-10-12 19:20:21 +04:00
static int snd_timer_user_info ( struct file * file ,
2005-11-17 15:56:05 +03:00
struct snd_timer_info __user * _info )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer_user * tu ;
struct snd_timer_info * info ;
struct snd_timer * t ;
2005-04-17 02:20:36 +04:00
int err = 0 ;
tu = file - > private_data ;
2007-07-16 13:01:30 +04:00
if ( ! tu - > timeri )
return - EBADFD ;
2005-04-17 02:20:36 +04:00
t = tu - > timeri - > timer ;
2007-07-16 13:01:30 +04:00
if ( ! t )
return - EBADFD ;
2005-04-17 02:20:36 +04:00
2005-09-09 16:20:23 +04:00
info = kzalloc ( sizeof ( * info ) , GFP_KERNEL ) ;
2005-04-17 02:20:36 +04:00
if ( ! info )
return - ENOMEM ;
info - > card = t - > card ? t - > card - > number : - 1 ;
if ( t - > hw . flags & SNDRV_TIMER_HW_SLAVE )
info - > flags | = SNDRV_TIMER_FLG_SLAVE ;
strlcpy ( info - > id , t - > id , sizeof ( info - > id ) ) ;
strlcpy ( info - > name , t - > name , sizeof ( info - > name ) ) ;
info - > resolution = t - > hw . resolution ;
if ( copy_to_user ( _info , info , sizeof ( * _info ) ) )
err = - EFAULT ;
kfree ( info ) ;
return err ;
}
2005-10-12 19:20:21 +04:00
static int snd_timer_user_params ( struct file * file ,
2005-11-17 15:56:05 +03:00
struct snd_timer_params __user * _params )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer_user * tu ;
struct snd_timer_params params ;
struct snd_timer * t ;
2005-04-17 02:20:36 +04:00
int err ;
2005-10-12 19:20:21 +04:00
2005-04-17 02:20:36 +04:00
tu = file - > private_data ;
2007-07-16 13:01:30 +04:00
if ( ! tu - > timeri )
return - EBADFD ;
2005-04-17 02:20:36 +04:00
t = tu - > timeri - > timer ;
2007-07-16 13:01:30 +04:00
if ( ! t )
return - EBADFD ;
2005-04-17 02:20:36 +04:00
if ( copy_from_user ( & params , _params , sizeof ( params ) ) )
return - EFAULT ;
2017-02-28 16:49:07 +03:00
if ( ! ( t - > hw . flags & SNDRV_TIMER_HW_SLAVE ) ) {
u64 resolution ;
if ( params . ticks < 1 ) {
err = - EINVAL ;
goto _end ;
}
/* Don't allow resolution less than 1ms */
resolution = snd_timer_resolution ( tu - > timeri ) ;
resolution * = params . ticks ;
if ( resolution < 1000000 ) {
err = - EINVAL ;
goto _end ;
}
2005-04-17 02:20:36 +04:00
}
2005-10-12 19:20:21 +04:00
if ( params . queue_size > 0 & &
( params . queue_size < 32 | | params . queue_size > 1024 ) ) {
2005-04-17 02:20:36 +04:00
err = - EINVAL ;
goto _end ;
}
if ( params . filter & ~ ( ( 1 < < SNDRV_TIMER_EVENT_RESOLUTION ) |
( 1 < < SNDRV_TIMER_EVENT_TICK ) |
( 1 < < SNDRV_TIMER_EVENT_START ) |
( 1 < < SNDRV_TIMER_EVENT_STOP ) |
( 1 < < SNDRV_TIMER_EVENT_CONTINUE ) |
( 1 < < SNDRV_TIMER_EVENT_PAUSE ) |
2005-08-16 13:09:05 +04:00
( 1 < < SNDRV_TIMER_EVENT_SUSPEND ) |
( 1 < < SNDRV_TIMER_EVENT_RESUME ) |
2005-04-17 02:20:36 +04:00
( 1 < < SNDRV_TIMER_EVENT_MSTART ) |
( 1 < < SNDRV_TIMER_EVENT_MSTOP ) |
( 1 < < SNDRV_TIMER_EVENT_MCONTINUE ) |
2005-08-16 15:05:43 +04:00
( 1 < < SNDRV_TIMER_EVENT_MPAUSE ) |
( 1 < < SNDRV_TIMER_EVENT_MSUSPEND ) |
2005-08-16 13:09:05 +04:00
( 1 < < SNDRV_TIMER_EVENT_MRESUME ) ) ) {
2005-04-17 02:20:36 +04:00
err = - EINVAL ;
goto _end ;
}
snd_timer_stop ( tu - > timeri ) ;
spin_lock_irq ( & t - > lock ) ;
tu - > timeri - > flags & = ~ ( SNDRV_TIMER_IFLG_AUTO |
SNDRV_TIMER_IFLG_EXCLUSIVE |
SNDRV_TIMER_IFLG_EARLY_EVENT ) ;
if ( params . flags & SNDRV_TIMER_PSFLG_AUTO )
tu - > timeri - > flags | = SNDRV_TIMER_IFLG_AUTO ;
if ( params . flags & SNDRV_TIMER_PSFLG_EXCLUSIVE )
tu - > timeri - > flags | = SNDRV_TIMER_IFLG_EXCLUSIVE ;
if ( params . flags & SNDRV_TIMER_PSFLG_EARLY_EVENT )
tu - > timeri - > flags | = SNDRV_TIMER_IFLG_EARLY_EVENT ;
spin_unlock_irq ( & t - > lock ) ;
2005-10-12 19:20:21 +04:00
if ( params . queue_size > 0 & &
( unsigned int ) tu - > queue_size ! = params . queue_size ) {
ALSA: timer: Improve user queue reallocation
ALSA timer may reallocate the user queue upon request, and it happens
at three places for now: at opening, at SNDRV_TIMER_IOCTL_PARAMS, and
at SNDRV_TIMER_IOCTL_SELECT. However, the last one,
snd_timer_user_tselect(), doesn't need to reallocate the buffer since
it doesn't change the queue size. It does just because tu->tread
might have been changed before starting the timer.
Instead of *_SELECT ioctl, we should reallocate the queue at
SNDRV_TIMER_IOCTL_TREAD; then the timer is guaranteed to be stopped,
thus we can reassign the buffer more safely.
This patch implements that with a slight code refactoring.
Essentially, the patch achieves:
- Introduce realloc_user_queue() for (re-)allocating the ring buffer,
and call it from all places. Also, realloc_user_queue() uses
kcalloc() for avoiding possible leaks.
- Add the buffer reallocation at SNDRV_TIMER_IOCTL_TREAD. When it
fails, tu->tread is restored to the old value, too.
- Drop the buffer reallocation at snd_timer_user_tselect().
Tested-by: Alexander Potapenko <glider@google.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2017-06-02 18:16:59 +03:00
err = realloc_user_queue ( tu , params . queue_size ) ;
if ( err < 0 )
goto _end ;
2005-04-17 02:20:36 +04:00
}
2017-06-02 18:35:30 +03:00
spin_lock_irq ( & tu - > qlock ) ;
2005-04-17 02:20:36 +04:00
tu - > qhead = tu - > qtail = tu - > qused = 0 ;
if ( tu - > timeri - > flags & SNDRV_TIMER_IFLG_EARLY_EVENT ) {
if ( tu - > tread ) {
2005-11-17 15:56:05 +03:00
struct snd_timer_tread tread ;
2016-05-03 23:44:07 +03:00
memset ( & tread , 0 , sizeof ( tread ) ) ;
2005-04-17 02:20:36 +04:00
tread . event = SNDRV_TIMER_EVENT_EARLY ;
tread . tstamp . tv_sec = 0 ;
tread . tstamp . tv_nsec = 0 ;
tread . val = 0 ;
snd_timer_user_append_to_tqueue ( tu , & tread ) ;
} else {
2005-11-17 15:56:05 +03:00
struct snd_timer_read * r = & tu - > queue [ 0 ] ;
2005-04-17 02:20:36 +04:00
r - > resolution = 0 ;
r - > ticks = 0 ;
tu - > qused + + ;
tu - > qtail + + ;
}
}
tu - > filter = params . filter ;
tu - > ticks = params . ticks ;
2017-06-02 18:35:30 +03:00
spin_unlock_irq ( & tu - > qlock ) ;
2005-04-17 02:20:36 +04:00
err = 0 ;
_end :
if ( copy_to_user ( _params , & params , sizeof ( params ) ) )
return - EFAULT ;
return err ;
}
2005-10-12 19:20:21 +04:00
static int snd_timer_user_status ( struct file * file ,
2005-11-17 15:56:05 +03:00
struct snd_timer_status __user * _status )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer_user * tu ;
struct snd_timer_status status ;
2005-10-12 19:20:21 +04:00
2005-04-17 02:20:36 +04:00
tu = file - > private_data ;
2007-07-16 13:01:30 +04:00
if ( ! tu - > timeri )
return - EBADFD ;
2005-04-17 02:20:36 +04:00
memset ( & status , 0 , sizeof ( status ) ) ;
status . tstamp = tu - > tstamp ;
status . resolution = snd_timer_resolution ( tu - > timeri ) ;
status . lost = tu - > timeri - > lost ;
status . overrun = tu - > overrun ;
spin_lock_irq ( & tu - > qlock ) ;
status . queue = tu - > qused ;
spin_unlock_irq ( & tu - > qlock ) ;
if ( copy_to_user ( _status , & status , sizeof ( status ) ) )
return - EFAULT ;
return 0 ;
}
static int snd_timer_user_start ( struct file * file )
{
int err ;
2005-11-17 15:56:05 +03:00
struct snd_timer_user * tu ;
2005-10-12 19:20:21 +04:00
2005-04-17 02:20:36 +04:00
tu = file - > private_data ;
2007-07-16 13:01:30 +04:00
if ( ! tu - > timeri )
return - EBADFD ;
2005-04-17 02:20:36 +04:00
snd_timer_stop ( tu - > timeri ) ;
tu - > timeri - > lost = 0 ;
tu - > last_resolution = 0 ;
return ( err = snd_timer_start ( tu - > timeri , tu - > ticks ) ) < 0 ? err : 0 ;
}
static int snd_timer_user_stop ( struct file * file )
{
int err ;
2005-11-17 15:56:05 +03:00
struct snd_timer_user * tu ;
2005-10-12 19:20:21 +04:00
2005-04-17 02:20:36 +04:00
tu = file - > private_data ;
2007-07-16 13:01:30 +04:00
if ( ! tu - > timeri )
return - EBADFD ;
2005-04-17 02:20:36 +04:00
return ( err = snd_timer_stop ( tu - > timeri ) ) < 0 ? err : 0 ;
}
static int snd_timer_user_continue ( struct file * file )
{
int err ;
2005-11-17 15:56:05 +03:00
struct snd_timer_user * tu ;
2005-10-12 19:20:21 +04:00
2005-04-17 02:20:36 +04:00
tu = file - > private_data ;
2007-07-16 13:01:30 +04:00
if ( ! tu - > timeri )
return - EBADFD ;
2016-09-07 16:45:31 +03:00
/* start timer instead of continue if it's not used before */
if ( ! ( tu - > timeri - > flags & SNDRV_TIMER_IFLG_PAUSED ) )
return snd_timer_user_start ( file ) ;
2005-04-17 02:20:36 +04:00
tu - > timeri - > lost = 0 ;
return ( err = snd_timer_continue ( tu - > timeri ) ) < 0 ? err : 0 ;
}
2005-05-15 17:04:14 +04:00
static int snd_timer_user_pause ( struct file * file )
{
int err ;
2005-11-17 15:56:05 +03:00
struct snd_timer_user * tu ;
2005-10-12 19:20:21 +04:00
2005-05-15 17:04:14 +04:00
tu = file - > private_data ;
2007-07-16 13:01:30 +04:00
if ( ! tu - > timeri )
return - EBADFD ;
2005-05-16 15:51:39 +04:00
return ( err = snd_timer_pause ( tu - > timeri ) ) < 0 ? err : 0 ;
2005-05-15 17:04:14 +04:00
}
2005-05-15 17:43:54 +04:00
enum {
SNDRV_TIMER_IOCTL_START_OLD = _IO ( ' T ' , 0x20 ) ,
SNDRV_TIMER_IOCTL_STOP_OLD = _IO ( ' T ' , 0x21 ) ,
SNDRV_TIMER_IOCTL_CONTINUE_OLD = _IO ( ' T ' , 0x22 ) ,
SNDRV_TIMER_IOCTL_PAUSE_OLD = _IO ( ' T ' , 0x23 ) ,
} ;
2016-01-13 19:48:01 +03:00
static long __snd_timer_user_ioctl ( struct file * file , unsigned int cmd ,
2005-10-12 19:20:21 +04:00
unsigned long arg )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer_user * tu ;
2005-04-17 02:20:36 +04:00
void __user * argp = ( void __user * ) arg ;
int __user * p = argp ;
2005-10-12 19:20:21 +04:00
2005-04-17 02:20:36 +04:00
tu = file - > private_data ;
switch ( cmd ) {
case SNDRV_TIMER_IOCTL_PVERSION :
return put_user ( SNDRV_TIMER_VERSION , p ) ? - EFAULT : 0 ;
case SNDRV_TIMER_IOCTL_NEXT_DEVICE :
return snd_timer_user_next_device ( argp ) ;
case SNDRV_TIMER_IOCTL_TREAD :
{
ALSA: timer: Improve user queue reallocation
ALSA timer may reallocate the user queue upon request, and it happens
at three places for now: at opening, at SNDRV_TIMER_IOCTL_PARAMS, and
at SNDRV_TIMER_IOCTL_SELECT. However, the last one,
snd_timer_user_tselect(), doesn't need to reallocate the buffer since
it doesn't change the queue size. It does just because tu->tread
might have been changed before starting the timer.
Instead of *_SELECT ioctl, we should reallocate the queue at
SNDRV_TIMER_IOCTL_TREAD; then the timer is guaranteed to be stopped,
thus we can reassign the buffer more safely.
This patch implements that with a slight code refactoring.
Essentially, the patch achieves:
- Introduce realloc_user_queue() for (re-)allocating the ring buffer,
and call it from all places. Also, realloc_user_queue() uses
kcalloc() for avoiding possible leaks.
- Add the buffer reallocation at SNDRV_TIMER_IOCTL_TREAD. When it
fails, tu->tread is restored to the old value, too.
- Drop the buffer reallocation at snd_timer_user_tselect().
Tested-by: Alexander Potapenko <glider@google.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2017-06-02 18:16:59 +03:00
int xarg , old_tread ;
2005-10-12 19:20:21 +04:00
2016-01-13 19:48:01 +03:00
if ( tu - > timeri ) /* too late */
2005-04-17 02:20:36 +04:00
return - EBUSY ;
2016-01-13 19:48:01 +03:00
if ( get_user ( xarg , p ) )
2005-04-17 02:20:36 +04:00
return - EFAULT ;
ALSA: timer: Improve user queue reallocation
ALSA timer may reallocate the user queue upon request, and it happens
at three places for now: at opening, at SNDRV_TIMER_IOCTL_PARAMS, and
at SNDRV_TIMER_IOCTL_SELECT. However, the last one,
snd_timer_user_tselect(), doesn't need to reallocate the buffer since
it doesn't change the queue size. It does just because tu->tread
might have been changed before starting the timer.
Instead of *_SELECT ioctl, we should reallocate the queue at
SNDRV_TIMER_IOCTL_TREAD; then the timer is guaranteed to be stopped,
thus we can reassign the buffer more safely.
This patch implements that with a slight code refactoring.
Essentially, the patch achieves:
- Introduce realloc_user_queue() for (re-)allocating the ring buffer,
and call it from all places. Also, realloc_user_queue() uses
kcalloc() for avoiding possible leaks.
- Add the buffer reallocation at SNDRV_TIMER_IOCTL_TREAD. When it
fails, tu->tread is restored to the old value, too.
- Drop the buffer reallocation at snd_timer_user_tselect().
Tested-by: Alexander Potapenko <glider@google.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2017-06-02 18:16:59 +03:00
old_tread = tu - > tread ;
2005-04-17 02:20:36 +04:00
tu - > tread = xarg ? 1 : 0 ;
ALSA: timer: Improve user queue reallocation
ALSA timer may reallocate the user queue upon request, and it happens
at three places for now: at opening, at SNDRV_TIMER_IOCTL_PARAMS, and
at SNDRV_TIMER_IOCTL_SELECT. However, the last one,
snd_timer_user_tselect(), doesn't need to reallocate the buffer since
it doesn't change the queue size. It does just because tu->tread
might have been changed before starting the timer.
Instead of *_SELECT ioctl, we should reallocate the queue at
SNDRV_TIMER_IOCTL_TREAD; then the timer is guaranteed to be stopped,
thus we can reassign the buffer more safely.
This patch implements that with a slight code refactoring.
Essentially, the patch achieves:
- Introduce realloc_user_queue() for (re-)allocating the ring buffer,
and call it from all places. Also, realloc_user_queue() uses
kcalloc() for avoiding possible leaks.
- Add the buffer reallocation at SNDRV_TIMER_IOCTL_TREAD. When it
fails, tu->tread is restored to the old value, too.
- Drop the buffer reallocation at snd_timer_user_tselect().
Tested-by: Alexander Potapenko <glider@google.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2017-06-02 18:16:59 +03:00
if ( tu - > tread ! = old_tread & &
realloc_user_queue ( tu , tu - > queue_size ) < 0 ) {
tu - > tread = old_tread ;
return - ENOMEM ;
}
2005-04-17 02:20:36 +04:00
return 0 ;
}
case SNDRV_TIMER_IOCTL_GINFO :
return snd_timer_user_ginfo ( file , argp ) ;
case SNDRV_TIMER_IOCTL_GPARAMS :
return snd_timer_user_gparams ( file , argp ) ;
case SNDRV_TIMER_IOCTL_GSTATUS :
return snd_timer_user_gstatus ( file , argp ) ;
case SNDRV_TIMER_IOCTL_SELECT :
return snd_timer_user_tselect ( file , argp ) ;
case SNDRV_TIMER_IOCTL_INFO :
return snd_timer_user_info ( file , argp ) ;
case SNDRV_TIMER_IOCTL_PARAMS :
return snd_timer_user_params ( file , argp ) ;
case SNDRV_TIMER_IOCTL_STATUS :
return snd_timer_user_status ( file , argp ) ;
case SNDRV_TIMER_IOCTL_START :
2005-05-15 17:43:54 +04:00
case SNDRV_TIMER_IOCTL_START_OLD :
2005-04-17 02:20:36 +04:00
return snd_timer_user_start ( file ) ;
case SNDRV_TIMER_IOCTL_STOP :
2005-05-15 17:43:54 +04:00
case SNDRV_TIMER_IOCTL_STOP_OLD :
2005-04-17 02:20:36 +04:00
return snd_timer_user_stop ( file ) ;
case SNDRV_TIMER_IOCTL_CONTINUE :
2005-05-15 17:43:54 +04:00
case SNDRV_TIMER_IOCTL_CONTINUE_OLD :
2005-04-17 02:20:36 +04:00
return snd_timer_user_continue ( file ) ;
2005-05-15 17:04:14 +04:00
case SNDRV_TIMER_IOCTL_PAUSE :
2005-05-15 17:43:54 +04:00
case SNDRV_TIMER_IOCTL_PAUSE_OLD :
2005-05-15 17:04:14 +04:00
return snd_timer_user_pause ( file ) ;
2005-04-17 02:20:36 +04:00
}
return - ENOTTY ;
}
2016-01-13 19:48:01 +03:00
static long snd_timer_user_ioctl ( struct file * file , unsigned int cmd ,
unsigned long arg )
{
struct snd_timer_user * tu = file - > private_data ;
long ret ;
mutex_lock ( & tu - > ioctl_lock ) ;
ret = __snd_timer_user_ioctl ( file , cmd , arg ) ;
mutex_unlock ( & tu - > ioctl_lock ) ;
return ret ;
}
2005-04-17 02:20:36 +04:00
static int snd_timer_user_fasync ( int fd , struct file * file , int on )
{
2005-11-17 15:56:05 +03:00
struct snd_timer_user * tu ;
2005-10-12 19:20:21 +04:00
2005-04-17 02:20:36 +04:00
tu = file - > private_data ;
2009-02-02 00:52:56 +03:00
return fasync_helper ( fd , file , on , & tu - > fasync ) ;
2005-04-17 02:20:36 +04:00
}
2005-10-12 19:20:21 +04:00
static ssize_t snd_timer_user_read ( struct file * file , char __user * buffer ,
size_t count , loff_t * offset )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:05 +03:00
struct snd_timer_user * tu ;
2005-04-17 02:20:36 +04:00
long result = 0 , unit ;
2016-02-08 19:26:58 +03:00
int qhead ;
2005-04-17 02:20:36 +04:00
int err = 0 ;
2005-10-12 19:20:21 +04:00
2005-04-17 02:20:36 +04:00
tu = file - > private_data ;
2005-11-17 15:56:05 +03:00
unit = tu - > tread ? sizeof ( struct snd_timer_tread ) : sizeof ( struct snd_timer_read ) ;
2017-06-02 16:03:38 +03:00
mutex_lock ( & tu - > ioctl_lock ) ;
2005-04-17 02:20:36 +04:00
spin_lock_irq ( & tu - > qlock ) ;
while ( ( long ) count - result > = unit ) {
while ( ! tu - > qused ) {
2017-06-20 13:06:13 +03:00
wait_queue_entry_t wait ;
2005-04-17 02:20:36 +04:00
if ( ( file - > f_flags & O_NONBLOCK ) ! = 0 | | result > 0 ) {
err = - EAGAIN ;
2016-02-08 19:26:58 +03:00
goto _error ;
2005-04-17 02:20:36 +04:00
}
set_current_state ( TASK_INTERRUPTIBLE ) ;
init_waitqueue_entry ( & wait , current ) ;
add_wait_queue ( & tu - > qchange_sleep , & wait ) ;
spin_unlock_irq ( & tu - > qlock ) ;
2017-06-02 16:03:38 +03:00
mutex_unlock ( & tu - > ioctl_lock ) ;
2005-04-17 02:20:36 +04:00
schedule ( ) ;
2017-06-02 16:03:38 +03:00
mutex_lock ( & tu - > ioctl_lock ) ;
2005-04-17 02:20:36 +04:00
spin_lock_irq ( & tu - > qlock ) ;
remove_wait_queue ( & tu - > qchange_sleep , & wait ) ;
2016-01-21 19:19:31 +03:00
if ( tu - > disconnected ) {
err = - ENODEV ;
2016-02-08 19:26:58 +03:00
goto _error ;
2016-01-21 19:19:31 +03:00
}
2005-04-17 02:20:36 +04:00
if ( signal_pending ( current ) ) {
err = - ERESTARTSYS ;
2016-02-08 19:26:58 +03:00
goto _error ;
2005-04-17 02:20:36 +04:00
}
}
2016-02-08 19:26:58 +03:00
qhead = tu - > qhead + + ;
tu - > qhead % = tu - > queue_size ;
2016-07-04 15:02:15 +03:00
tu - > qused - - ;
2005-04-17 02:20:36 +04:00
spin_unlock_irq ( & tu - > qlock ) ;
if ( tu - > tread ) {
2016-02-08 19:26:58 +03:00
if ( copy_to_user ( buffer , & tu - > tqueue [ qhead ] ,
sizeof ( struct snd_timer_tread ) ) )
2005-04-17 02:20:36 +04:00
err = - EFAULT ;
} else {
2016-02-08 19:26:58 +03:00
if ( copy_to_user ( buffer , & tu - > queue [ qhead ] ,
sizeof ( struct snd_timer_read ) ) )
2005-04-17 02:20:36 +04:00
err = - EFAULT ;
}
spin_lock_irq ( & tu - > qlock ) ;
2016-02-08 19:26:58 +03:00
if ( err < 0 )
goto _error ;
result + = unit ;
buffer + = unit ;
2005-04-17 02:20:36 +04:00
}
_error :
2016-02-08 19:26:58 +03:00
spin_unlock_irq ( & tu - > qlock ) ;
2017-06-02 16:03:38 +03:00
mutex_unlock ( & tu - > ioctl_lock ) ;
2005-04-17 02:20:36 +04:00
return result > 0 ? result : err ;
}
static unsigned int snd_timer_user_poll ( struct file * file , poll_table * wait )
{
unsigned int mask ;
2005-11-17 15:56:05 +03:00
struct snd_timer_user * tu ;
2005-04-17 02:20:36 +04:00
tu = file - > private_data ;
poll_wait ( file , & tu - > qchange_sleep , wait ) ;
2005-10-12 19:20:21 +04:00
2005-04-17 02:20:36 +04:00
mask = 0 ;
2017-06-02 18:35:30 +03:00
spin_lock_irq ( & tu - > qlock ) ;
2005-04-17 02:20:36 +04:00
if ( tu - > qused )
mask | = POLLIN | POLLRDNORM ;
2016-01-21 19:19:31 +03:00
if ( tu - > disconnected )
mask | = POLLERR ;
2017-06-02 18:35:30 +03:00
spin_unlock_irq ( & tu - > qlock ) ;
2005-04-17 02:20:36 +04:00
return mask ;
}
# ifdef CONFIG_COMPAT
# include "timer_compat.c"
# else
# define snd_timer_user_ioctl_compat NULL
# endif
2007-02-12 11:55:37 +03:00
static const struct file_operations snd_timer_f_ops =
2005-04-17 02:20:36 +04:00
{
. owner = THIS_MODULE ,
. read = snd_timer_user_read ,
. open = snd_timer_user_open ,
. release = snd_timer_user_release ,
2010-04-13 13:49:04 +04:00
. llseek = no_llseek ,
2005-04-17 02:20:36 +04:00
. poll = snd_timer_user_poll ,
. unlocked_ioctl = snd_timer_user_ioctl ,
. compat_ioctl = snd_timer_user_ioctl_compat ,
. fasync = snd_timer_user_fasync ,
} ;
2015-01-29 20:09:05 +03:00
/* unregister the system timer */
static void snd_timer_free_all ( void )
{
struct snd_timer * timer , * n ;
list_for_each_entry_safe ( timer , n , & snd_timer_list , device_list )
snd_timer_free ( timer ) ;
}
2015-01-29 20:12:26 +03:00
static struct device timer_dev ;
2005-04-17 02:20:36 +04:00
/*
* ENTRY functions
*/
static int __init alsa_timer_init ( void )
{
int err ;
2015-01-29 20:12:26 +03:00
snd_device_initialize ( & timer_dev , NULL ) ;
dev_set_name ( & timer_dev , " timer " ) ;
2005-04-17 02:20:36 +04:00
# ifdef SNDRV_OSS_INFO_DEV_TIMERS
2005-10-12 19:20:21 +04:00
snd_oss_info_register ( SNDRV_OSS_INFO_DEV_TIMERS , SNDRV_CARDS - 1 ,
" system timer " ) ;
2005-04-17 02:20:36 +04:00
# endif
2005-12-01 12:42:42 +03:00
2015-01-29 20:09:05 +03:00
err = snd_timer_register_system ( ) ;
if ( err < 0 ) {
2014-02-04 21:22:39 +04:00
pr_err ( " ALSA: unable to register system timer (%i) \n " , err ) ;
2017-08-23 10:30:41 +03:00
goto put_timer ;
2015-01-29 20:09:05 +03:00
}
2015-01-30 10:34:58 +03:00
err = snd_register_device ( SNDRV_DEVICE_TYPE_TIMER , NULL , 0 ,
& snd_timer_f_ops , NULL , & timer_dev ) ;
2015-01-29 20:09:05 +03:00
if ( err < 0 ) {
2014-02-04 21:22:39 +04:00
pr_err ( " ALSA: unable to register timer device (%i) \n " , err ) ;
2015-01-29 20:09:05 +03:00
snd_timer_free_all ( ) ;
2017-08-23 10:30:41 +03:00
goto put_timer ;
2015-01-29 20:09:05 +03:00
}
2005-12-01 12:42:42 +03:00
snd_timer_proc_init ( ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
2017-08-23 10:30:41 +03:00
put_timer :
put_device ( & timer_dev ) ;
return err ;
2005-04-17 02:20:36 +04:00
}
static void __exit alsa_timer_exit ( void )
{
2015-01-30 10:34:58 +03:00
snd_unregister_device ( & timer_dev ) ;
2015-01-29 20:09:05 +03:00
snd_timer_free_all ( ) ;
2015-01-29 20:12:26 +03:00
put_device ( & timer_dev ) ;
2005-12-01 12:42:42 +03:00
snd_timer_proc_done ( ) ;
2005-04-17 02:20:36 +04:00
# ifdef SNDRV_OSS_INFO_DEV_TIMERS
snd_oss_info_unregister ( SNDRV_OSS_INFO_DEV_TIMERS , SNDRV_CARDS - 1 ) ;
# endif
}
module_init ( alsa_timer_init )
module_exit ( alsa_timer_exit )