2005-04-17 02:20:36 +04:00
/*
* Hardware dependent layer
* Copyright ( c ) by Jaroslav Kysela < perex @ suse . cz >
*
*
* 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 <sound/driver.h>
# include <linux/major.h>
# include <linux/init.h>
# include <linux/smp_lock.h>
# include <linux/slab.h>
# include <linux/time.h>
# include <sound/core.h>
# include <sound/control.h>
# include <sound/minors.h>
# include <sound/hwdep.h>
# include <sound/info.h>
MODULE_AUTHOR ( " Jaroslav Kysela <perex@suse.cz> " ) ;
MODULE_DESCRIPTION ( " Hardware dependent layer " ) ;
MODULE_LICENSE ( " GPL " ) ;
2005-11-20 16:06:59 +03:00
static LIST_HEAD ( snd_hwdep_devices ) ;
2005-04-17 02:20:36 +04:00
static DECLARE_MUTEX ( register_mutex ) ;
2005-11-17 15:58:21 +03:00
static int snd_hwdep_free ( struct snd_hwdep * hwdep ) ;
static int snd_hwdep_dev_free ( struct snd_device * device ) ;
static int snd_hwdep_dev_register ( struct snd_device * device ) ;
static int snd_hwdep_dev_unregister ( struct snd_device * device ) ;
2005-04-17 02:20:36 +04:00
2005-11-20 16:06:59 +03:00
static struct snd_hwdep * snd_hwdep_search ( struct snd_card * card , int device )
{
struct list_head * p ;
struct snd_hwdep * hwdep ;
list_for_each ( p , & snd_hwdep_devices ) {
hwdep = list_entry ( p , struct snd_hwdep , list ) ;
if ( hwdep - > card = = card & & hwdep - > device = = device )
return hwdep ;
}
return NULL ;
}
2005-04-17 02:20:36 +04:00
static loff_t snd_hwdep_llseek ( struct file * file , loff_t offset , int orig )
{
2005-11-17 15:58:21 +03:00
struct snd_hwdep * hw = file - > private_data ;
2005-04-17 02:20:36 +04:00
if ( hw - > ops . llseek )
return hw - > ops . llseek ( hw , file , offset , orig ) ;
return - ENXIO ;
}
2005-11-17 15:58:21 +03:00
static ssize_t snd_hwdep_read ( struct file * file , char __user * buf ,
size_t count , loff_t * offset )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:58:21 +03:00
struct snd_hwdep * hw = file - > private_data ;
2005-04-17 02:20:36 +04:00
if ( hw - > ops . read )
return hw - > ops . read ( hw , buf , count , offset ) ;
return - ENXIO ;
}
2005-11-17 15:58:21 +03:00
static ssize_t snd_hwdep_write ( struct file * file , const char __user * buf ,
size_t count , loff_t * offset )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:58:21 +03:00
struct snd_hwdep * hw = file - > private_data ;
2005-04-17 02:20:36 +04:00
if ( hw - > ops . write )
return hw - > ops . write ( hw , buf , count , offset ) ;
return - ENXIO ;
}
static int snd_hwdep_open ( struct inode * inode , struct file * file )
{
int major = imajor ( inode ) ;
2005-11-17 15:58:21 +03:00
struct snd_hwdep * hw ;
2005-04-17 02:20:36 +04:00
int err ;
wait_queue_t wait ;
2005-10-24 19:05:03 +04:00
if ( major = = snd_major ) {
2005-11-20 16:06:59 +03:00
hw = snd_lookup_minor_data ( iminor ( inode ) ,
SNDRV_DEVICE_TYPE_HWDEP ) ;
2005-04-17 02:20:36 +04:00
# ifdef CONFIG_SND_OSSEMUL
2005-10-24 19:05:03 +04:00
} else if ( major = = SOUND_MAJOR ) {
2005-11-20 16:06:59 +03:00
hw = snd_lookup_oss_minor_data ( iminor ( inode ) ,
SNDRV_OSS_DEVICE_TYPE_DMFM ) ;
2005-04-17 02:20:36 +04:00
# endif
2005-10-24 19:05:03 +04:00
} else
2005-04-17 02:20:36 +04:00
return - ENXIO ;
if ( hw = = NULL )
return - ENODEV ;
if ( ! hw - > ops . open )
return - ENXIO ;
if ( ! try_module_get ( hw - > card - > module ) )
return - EFAULT ;
init_waitqueue_entry ( & wait , current ) ;
add_wait_queue ( & hw - > open_wait , & wait ) ;
down ( & hw - > open_mutex ) ;
while ( 1 ) {
if ( hw - > exclusive & & hw - > used > 0 ) {
err = - EBUSY ;
break ;
}
err = hw - > ops . open ( hw , file ) ;
if ( err > = 0 )
break ;
if ( err = = - EAGAIN ) {
if ( file - > f_flags & O_NONBLOCK ) {
err = - EBUSY ;
break ;
}
} else
break ;
set_current_state ( TASK_INTERRUPTIBLE ) ;
up ( & hw - > open_mutex ) ;
schedule ( ) ;
down ( & hw - > open_mutex ) ;
if ( signal_pending ( current ) ) {
err = - ERESTARTSYS ;
break ;
}
}
remove_wait_queue ( & hw - > open_wait , & wait ) ;
if ( err > = 0 ) {
err = snd_card_file_add ( hw - > card , file ) ;
if ( err > = 0 ) {
file - > private_data = hw ;
hw - > used + + ;
} else {
if ( hw - > ops . release )
hw - > ops . release ( hw , file ) ;
}
}
up ( & hw - > open_mutex ) ;
if ( err < 0 )
module_put ( hw - > card - > module ) ;
return err ;
}
static int snd_hwdep_release ( struct inode * inode , struct file * file )
{
int err = - ENXIO ;
2005-11-17 15:58:21 +03:00
struct snd_hwdep * hw = file - > private_data ;
2005-04-17 02:20:36 +04:00
down ( & hw - > open_mutex ) ;
if ( hw - > ops . release ) {
err = hw - > ops . release ( hw , file ) ;
wake_up ( & hw - > open_wait ) ;
}
if ( hw - > used > 0 )
hw - > used - - ;
snd_card_file_remove ( hw - > card , file ) ;
up ( & hw - > open_mutex ) ;
module_put ( hw - > card - > module ) ;
return err ;
}
static unsigned int snd_hwdep_poll ( struct file * file , poll_table * wait )
{
2005-11-17 15:58:21 +03:00
struct snd_hwdep * hw = file - > private_data ;
2005-04-17 02:20:36 +04:00
if ( hw - > ops . poll )
return hw - > ops . poll ( hw , file , wait ) ;
return 0 ;
}
2005-11-17 15:58:21 +03:00
static int snd_hwdep_info ( struct snd_hwdep * hw ,
struct snd_hwdep_info __user * _info )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:58:21 +03:00
struct snd_hwdep_info info ;
2005-04-17 02:20:36 +04:00
memset ( & info , 0 , sizeof ( info ) ) ;
info . card = hw - > card - > number ;
strlcpy ( info . id , hw - > id , sizeof ( info . id ) ) ;
strlcpy ( info . name , hw - > name , sizeof ( info . name ) ) ;
info . iface = hw - > iface ;
if ( copy_to_user ( _info , & info , sizeof ( info ) ) )
return - EFAULT ;
return 0 ;
}
2005-11-17 15:58:21 +03:00
static int snd_hwdep_dsp_status ( struct snd_hwdep * hw ,
struct snd_hwdep_dsp_status __user * _info )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:58:21 +03:00
struct snd_hwdep_dsp_status info ;
2005-04-17 02:20:36 +04:00
int err ;
if ( ! hw - > ops . dsp_status )
return - ENXIO ;
memset ( & info , 0 , sizeof ( info ) ) ;
info . dsp_loaded = hw - > dsp_loaded ;
if ( ( err = hw - > ops . dsp_status ( hw , & info ) ) < 0 )
return err ;
if ( copy_to_user ( _info , & info , sizeof ( info ) ) )
return - EFAULT ;
return 0 ;
}
2005-11-17 15:58:21 +03:00
static int snd_hwdep_dsp_load ( struct snd_hwdep * hw ,
struct snd_hwdep_dsp_image __user * _info )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:58:21 +03:00
struct snd_hwdep_dsp_image info ;
2005-04-17 02:20:36 +04:00
int err ;
if ( ! hw - > ops . dsp_load )
return - ENXIO ;
memset ( & info , 0 , sizeof ( info ) ) ;
if ( copy_from_user ( & info , _info , sizeof ( info ) ) )
return - EFAULT ;
/* check whether the dsp was already loaded */
if ( hw - > dsp_loaded & ( 1 < < info . index ) )
return - EBUSY ;
if ( ! access_ok ( VERIFY_READ , info . image , info . length ) )
return - EFAULT ;
err = hw - > ops . dsp_load ( hw , & info ) ;
if ( err < 0 )
return err ;
hw - > dsp_loaded | = ( 1 < < info . index ) ;
return 0 ;
}
2005-11-17 15:58:21 +03:00
static long snd_hwdep_ioctl ( struct file * file , unsigned int cmd ,
unsigned long arg )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:58:21 +03:00
struct snd_hwdep * hw = file - > private_data ;
2005-04-17 02:20:36 +04:00
void __user * argp = ( void __user * ) arg ;
switch ( cmd ) {
case SNDRV_HWDEP_IOCTL_PVERSION :
return put_user ( SNDRV_HWDEP_VERSION , ( int __user * ) argp ) ;
case SNDRV_HWDEP_IOCTL_INFO :
return snd_hwdep_info ( hw , argp ) ;
case SNDRV_HWDEP_IOCTL_DSP_STATUS :
return snd_hwdep_dsp_status ( hw , argp ) ;
case SNDRV_HWDEP_IOCTL_DSP_LOAD :
return snd_hwdep_dsp_load ( hw , argp ) ;
}
if ( hw - > ops . ioctl )
return hw - > ops . ioctl ( hw , file , cmd , arg ) ;
return - ENOTTY ;
}
static int snd_hwdep_mmap ( struct file * file , struct vm_area_struct * vma )
{
2005-11-17 15:58:21 +03:00
struct snd_hwdep * hw = file - > private_data ;
2005-04-17 02:20:36 +04:00
if ( hw - > ops . mmap )
return hw - > ops . mmap ( hw , file , vma ) ;
return - ENXIO ;
}
2005-11-17 15:58:21 +03:00
static int snd_hwdep_control_ioctl ( struct snd_card * card ,
struct snd_ctl_file * control ,
2005-04-17 02:20:36 +04:00
unsigned int cmd , unsigned long arg )
{
switch ( cmd ) {
case SNDRV_CTL_IOCTL_HWDEP_NEXT_DEVICE :
{
int device ;
if ( get_user ( device , ( int __user * ) arg ) )
return - EFAULT ;
2005-11-20 16:06:59 +03:00
down ( & register_mutex ) ;
2005-04-17 02:20:36 +04:00
device = device < 0 ? 0 : device + 1 ;
while ( device < SNDRV_MINOR_HWDEPS ) {
2005-11-20 16:06:59 +03:00
if ( snd_hwdep_search ( card , device ) )
2005-04-17 02:20:36 +04:00
break ;
device + + ;
}
if ( device > = SNDRV_MINOR_HWDEPS )
device = - 1 ;
2005-11-20 16:06:59 +03:00
up ( & register_mutex ) ;
2005-04-17 02:20:36 +04:00
if ( put_user ( device , ( int __user * ) arg ) )
return - EFAULT ;
return 0 ;
}
case SNDRV_CTL_IOCTL_HWDEP_INFO :
{
2005-11-17 15:58:21 +03:00
struct snd_hwdep_info __user * info = ( struct snd_hwdep_info __user * ) arg ;
2005-11-20 16:06:59 +03:00
int device , err ;
2005-11-17 15:58:21 +03:00
struct snd_hwdep * hwdep ;
2005-04-17 02:20:36 +04:00
if ( get_user ( device , & info - > device ) )
return - EFAULT ;
2005-11-20 16:06:59 +03:00
down ( & register_mutex ) ;
hwdep = snd_hwdep_search ( card , device ) ;
if ( hwdep )
err = snd_hwdep_info ( hwdep , info ) ;
else
err = - ENXIO ;
up ( & register_mutex ) ;
return err ;
2005-04-17 02:20:36 +04:00
}
}
return - ENOIOCTLCMD ;
}
# ifdef CONFIG_COMPAT
# include "hwdep_compat.c"
# else
# define snd_hwdep_ioctl_compat NULL
# endif
/*
*/
static struct file_operations snd_hwdep_f_ops =
{
. owner = THIS_MODULE ,
. llseek = snd_hwdep_llseek ,
. read = snd_hwdep_read ,
. write = snd_hwdep_write ,
. open = snd_hwdep_open ,
. release = snd_hwdep_release ,
. poll = snd_hwdep_poll ,
. unlocked_ioctl = snd_hwdep_ioctl ,
. compat_ioctl = snd_hwdep_ioctl_compat ,
. mmap = snd_hwdep_mmap ,
} ;
/**
* snd_hwdep_new - create a new hwdep instance
* @ card : the card instance
* @ id : the id string
* @ device : the device index ( zero - based )
* @ rhwdep : the pointer to store the new hwdep instance
*
* Creates a new hwdep instance with the given index on the card .
* The callbacks ( hwdep - > ops ) must be set on the returned instance
* after this call manually by the caller .
*
* Returns zero if successful , or a negative error code on failure .
*/
2005-11-17 15:58:21 +03:00
int snd_hwdep_new ( struct snd_card * card , char * id , int device ,
struct snd_hwdep * * rhwdep )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:58:21 +03:00
struct snd_hwdep * hwdep ;
2005-04-17 02:20:36 +04:00
int err ;
2005-11-17 15:58:21 +03:00
static struct snd_device_ops ops = {
2005-04-17 02:20:36 +04:00
. dev_free = snd_hwdep_dev_free ,
. dev_register = snd_hwdep_dev_register ,
. dev_unregister = snd_hwdep_dev_unregister
} ;
snd_assert ( rhwdep ! = NULL , return - EINVAL ) ;
* rhwdep = NULL ;
snd_assert ( card ! = NULL , return - ENXIO ) ;
2005-09-09 16:20:23 +04:00
hwdep = kzalloc ( sizeof ( * hwdep ) , GFP_KERNEL ) ;
2005-11-17 19:44:01 +03:00
if ( hwdep = = NULL ) {
snd_printk ( KERN_ERR " hwdep: cannot allocate \n " ) ;
2005-04-17 02:20:36 +04:00
return - ENOMEM ;
2005-11-17 19:44:01 +03:00
}
2005-04-17 02:20:36 +04:00
hwdep - > card = card ;
hwdep - > device = device ;
2005-11-17 19:44:01 +03:00
if ( id )
2005-04-17 02:20:36 +04:00
strlcpy ( hwdep - > id , id , sizeof ( hwdep - > id ) ) ;
# ifdef CONFIG_SND_OSSEMUL
hwdep - > oss_type = - 1 ;
# endif
if ( ( err = snd_device_new ( card , SNDRV_DEV_HWDEP , hwdep , & ops ) ) < 0 ) {
snd_hwdep_free ( hwdep ) ;
return err ;
}
init_waitqueue_head ( & hwdep - > open_wait ) ;
init_MUTEX ( & hwdep - > open_mutex ) ;
* rhwdep = hwdep ;
return 0 ;
}
2005-11-17 15:58:21 +03:00
static int snd_hwdep_free ( struct snd_hwdep * hwdep )
2005-04-17 02:20:36 +04:00
{
snd_assert ( hwdep ! = NULL , return - ENXIO ) ;
if ( hwdep - > private_free )
hwdep - > private_free ( hwdep ) ;
kfree ( hwdep ) ;
return 0 ;
}
2005-11-17 15:58:21 +03:00
static int snd_hwdep_dev_free ( struct snd_device * device )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:58:21 +03:00
struct snd_hwdep * hwdep = device - > device_data ;
2005-04-17 02:20:36 +04:00
return snd_hwdep_free ( hwdep ) ;
}
2005-11-17 15:58:21 +03:00
static int snd_hwdep_dev_register ( struct snd_device * device )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:58:21 +03:00
struct snd_hwdep * hwdep = device - > device_data ;
2005-11-20 16:06:59 +03:00
int err ;
2005-04-17 02:20:36 +04:00
char name [ 32 ] ;
down ( & register_mutex ) ;
2005-11-20 16:06:59 +03:00
if ( snd_hwdep_search ( hwdep - > card , hwdep - > device ) ) {
2005-04-17 02:20:36 +04:00
up ( & register_mutex ) ;
return - EBUSY ;
}
2005-11-20 16:06:59 +03:00
list_add_tail ( & hwdep - > list , & snd_hwdep_devices ) ;
2005-04-17 02:20:36 +04:00
sprintf ( name , " hwC%iD%i " , hwdep - > card - > number , hwdep - > device ) ;
if ( ( err = snd_register_device ( SNDRV_DEVICE_TYPE_HWDEP ,
hwdep - > card , hwdep - > device ,
2005-11-20 16:06:59 +03:00
& snd_hwdep_f_ops , hwdep , name ) ) < 0 ) {
2005-04-17 02:20:36 +04:00
snd_printk ( KERN_ERR " unable to register hardware dependent device %i:%i \n " ,
hwdep - > card - > number , hwdep - > device ) ;
2005-11-20 16:06:59 +03:00
list_del ( & hwdep - > list ) ;
2005-04-17 02:20:36 +04:00
up ( & register_mutex ) ;
return err ;
}
# ifdef CONFIG_SND_OSSEMUL
hwdep - > ossreg = 0 ;
if ( hwdep - > oss_type > = 0 ) {
if ( ( hwdep - > oss_type = = SNDRV_OSS_DEVICE_TYPE_DMFM ) & & ( hwdep - > device ! = 0 ) ) {
snd_printk ( KERN_WARNING " only hwdep device 0 can be registered as OSS direct FM device! \n " ) ;
} else {
if ( snd_register_oss_device ( hwdep - > oss_type ,
hwdep - > card , hwdep - > device ,
2005-11-20 16:06:59 +03:00
& snd_hwdep_f_ops , hwdep ,
2005-11-20 16:03:48 +03:00
hwdep - > oss_dev ) < 0 ) {
2005-04-17 02:20:36 +04:00
snd_printk ( KERN_ERR " unable to register OSS compatibility device %i:%i \n " ,
hwdep - > card - > number , hwdep - > device ) ;
} else
hwdep - > ossreg = 1 ;
}
}
# endif
up ( & register_mutex ) ;
return 0 ;
}
2005-11-17 15:58:21 +03:00
static int snd_hwdep_dev_unregister ( struct snd_device * device )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:58:21 +03:00
struct snd_hwdep * hwdep = device - > device_data ;
2005-04-17 02:20:36 +04:00
snd_assert ( hwdep ! = NULL , return - ENXIO ) ;
down ( & register_mutex ) ;
2005-11-20 16:06:59 +03:00
if ( snd_hwdep_search ( hwdep - > card , hwdep - > device ) ! = hwdep ) {
2005-04-17 02:20:36 +04:00
up ( & register_mutex ) ;
return - EINVAL ;
}
# ifdef CONFIG_SND_OSSEMUL
if ( hwdep - > ossreg )
snd_unregister_oss_device ( hwdep - > oss_type , hwdep - > card , hwdep - > device ) ;
# endif
snd_unregister_device ( SNDRV_DEVICE_TYPE_HWDEP , hwdep - > card , hwdep - > device ) ;
2005-11-20 16:06:59 +03:00
list_del ( & hwdep - > list ) ;
2005-04-17 02:20:36 +04:00
up ( & register_mutex ) ;
return snd_hwdep_free ( hwdep ) ;
}
/*
* Info interface
*/
2005-11-17 15:58:21 +03:00
static void snd_hwdep_proc_read ( struct snd_info_entry * entry ,
struct snd_info_buffer * buffer )
2005-04-17 02:20:36 +04:00
{
2005-11-20 16:06:59 +03:00
struct list_head * p ;
2005-11-17 15:58:21 +03:00
struct snd_hwdep * hwdep ;
2005-04-17 02:20:36 +04:00
down ( & register_mutex ) ;
2005-11-20 16:06:59 +03:00
list_for_each ( p , & snd_hwdep_devices ) {
hwdep = list_entry ( p , struct snd_hwdep , list ) ;
2005-04-17 02:20:36 +04:00
snd_iprintf ( buffer , " %02i-%02i: %s \n " ,
2005-11-20 16:06:59 +03:00
hwdep - > card - > number , hwdep - > device , hwdep - > name ) ;
2005-04-17 02:20:36 +04:00
}
up ( & register_mutex ) ;
}
/*
* ENTRY functions
*/
2005-11-17 15:58:21 +03:00
static struct snd_info_entry * snd_hwdep_proc_entry = NULL ;
2005-04-17 02:20:36 +04:00
static int __init alsa_hwdep_init ( void )
{
2005-11-17 15:58:21 +03:00
struct snd_info_entry * entry ;
2005-04-17 02:20:36 +04:00
if ( ( entry = snd_info_create_module_entry ( THIS_MODULE , " hwdep " , NULL ) ) ! = NULL ) {
2005-11-20 16:06:59 +03:00
entry - > c . text . read_size = PAGE_SIZE ;
2005-04-17 02:20:36 +04:00
entry - > c . text . read = snd_hwdep_proc_read ;
if ( snd_info_register ( entry ) < 0 ) {
snd_info_free_entry ( entry ) ;
entry = NULL ;
}
}
snd_hwdep_proc_entry = entry ;
snd_ctl_register_ioctl ( snd_hwdep_control_ioctl ) ;
snd_ctl_register_ioctl_compat ( snd_hwdep_control_ioctl ) ;
return 0 ;
}
static void __exit alsa_hwdep_exit ( void )
{
snd_ctl_unregister_ioctl ( snd_hwdep_control_ioctl ) ;
snd_ctl_unregister_ioctl_compat ( snd_hwdep_control_ioctl ) ;
if ( snd_hwdep_proc_entry ) {
snd_info_unregister ( snd_hwdep_proc_entry ) ;
snd_hwdep_proc_entry = NULL ;
}
}
module_init ( alsa_hwdep_init )
module_exit ( alsa_hwdep_exit )
EXPORT_SYMBOL ( snd_hwdep_new ) ;