2005-04-16 15:20:36 -07:00
/*
* Advanced Linux Sound Architecture
2007-10-15 09:50:19 +02:00
* Copyright ( c ) by Jaroslav Kysela < perex @ perex . cz >
2005-04-16 15:20:36 -07: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/init.h>
# include <linux/slab.h>
2008-05-15 16:39:03 -06:00
# include <linux/smp_lock.h>
2005-04-16 15:20:36 -07:00
# include <linux/time.h>
2005-11-22 15:46:41 +01:00
# include <linux/device.h>
2005-04-16 15:20:36 -07:00
# include <linux/moduleparam.h>
# include <sound/core.h>
# include <sound/minors.h>
# include <sound/info.h>
# include <sound/version.h>
# include <sound/control.h>
# include <sound/initval.h>
# include <linux/kmod.h>
2006-01-16 16:29:08 +01:00
# include <linux/mutex.h>
2005-04-16 15:20:36 -07:00
static int major = CONFIG_SND_MAJOR ;
int snd_major ;
2006-04-28 15:13:39 +02:00
EXPORT_SYMBOL ( snd_major ) ;
2005-04-16 15:20:36 -07:00
static int cards_limit = 1 ;
2007-10-15 09:50:19 +02:00
MODULE_AUTHOR ( " Jaroslav Kysela <perex@perex.cz> " ) ;
2005-04-16 15:20:36 -07:00
MODULE_DESCRIPTION ( " Advanced Linux Sound Architecture driver for soundcards. " ) ;
MODULE_LICENSE ( " GPL " ) ;
module_param ( major , int , 0444 ) ;
MODULE_PARM_DESC ( major , " Major # for sound driver. " ) ;
module_param ( cards_limit , int , 0444 ) ;
MODULE_PARM_DESC ( cards_limit , " Count of auto-loadable soundcards. " ) ;
MODULE_ALIAS_CHARDEV_MAJOR ( CONFIG_SND_MAJOR ) ;
/* this one holds the actual max. card number currently available.
* as default , it ' s identical with cards_limit option . when more
* modules are loaded manually , this limit number increases , too .
*/
int snd_ecards_limit ;
2006-04-28 15:13:39 +02:00
EXPORT_SYMBOL ( snd_ecards_limit ) ;
2005-04-16 15:20:36 -07:00
2005-11-20 14:05:49 +01:00
static struct snd_minor * snd_minors [ SNDRV_OS_MINORS ] ;
2006-01-16 16:29:08 +01:00
static DEFINE_MUTEX ( sound_mutex ) ;
2005-04-16 15:20:36 -07:00
2008-07-09 10:28:41 +02:00
# ifdef CONFIG_MODULES
2005-04-16 15:20:36 -07:00
/**
* snd_request_card - try to load the card module
* @ card : the card number
*
* Tries to load the module " snd-card-X " for the given card number
2008-07-09 10:28:41 +02:00
* via request_module . Returns immediately if already loaded .
2005-04-16 15:20:36 -07:00
*/
void snd_request_card ( int card )
{
2006-05-15 19:49:05 +02:00
if ( snd_card_locked ( card ) )
2005-04-16 15:20:36 -07:00
return ;
if ( card < 0 | | card > = cards_limit )
return ;
request_module ( " snd-card-%i " , card ) ;
}
2006-04-28 15:13:39 +02:00
EXPORT_SYMBOL ( snd_request_card ) ;
2005-04-16 15:20:36 -07:00
static void snd_request_other ( int minor )
{
char * str ;
switch ( minor ) {
case SNDRV_MINOR_SEQUENCER : str = " snd-seq " ; break ;
case SNDRV_MINOR_TIMER : str = " snd-timer " ; break ;
default : return ;
}
request_module ( str ) ;
}
2008-07-09 10:28:41 +02:00
# endif /* modular kernel */
2005-04-16 15:20:36 -07:00
2005-11-20 14:06:59 +01:00
/**
* snd_lookup_minor_data - get user data of a registered device
* @ minor : the minor number
* @ type : device type ( SNDRV_DEVICE_TYPE_XXX )
*
* Checks that a minor device with the specified type is registered , and returns
* its user data pointer .
*/
void * snd_lookup_minor_data ( unsigned int minor , int type )
{
struct snd_minor * mreg ;
void * private_data ;
2006-03-13 14:14:10 +01:00
if ( minor > = ARRAY_SIZE ( snd_minors ) )
2005-11-20 14:06:59 +01:00
return NULL ;
2006-01-16 16:29:08 +01:00
mutex_lock ( & sound_mutex ) ;
2005-11-20 14:06:59 +01:00
mreg = snd_minors [ minor ] ;
if ( mreg & & mreg - > type = = type )
private_data = mreg - > private_data ;
else
private_data = NULL ;
2006-01-16 16:29:08 +01:00
mutex_unlock ( & sound_mutex ) ;
2005-11-20 14:06:59 +01:00
return private_data ;
}
2006-04-28 15:13:39 +02:00
EXPORT_SYMBOL ( snd_lookup_minor_data ) ;
2008-05-15 16:39:03 -06:00
static int __snd_open ( struct inode * inode , struct file * file )
2005-04-16 15:20:36 -07:00
{
2005-11-20 14:07:47 +01:00
unsigned int minor = iminor ( inode ) ;
2005-11-17 13:51:18 +01:00
struct snd_minor * mptr = NULL ;
2006-03-28 01:56:41 -08:00
const struct file_operations * old_fops ;
2005-04-16 15:20:36 -07:00
int err = 0 ;
2006-03-13 14:14:10 +01:00
if ( minor > = ARRAY_SIZE ( snd_minors ) )
2005-11-20 14:07:47 +01:00
return - ENODEV ;
mptr = snd_minors [ minor ] ;
if ( mptr = = NULL ) {
2008-07-09 10:28:41 +02:00
# ifdef CONFIG_MODULES
2005-11-20 14:07:47 +01:00
int dev = SNDRV_MINOR_DEVICE ( minor ) ;
if ( dev = = SNDRV_MINOR_CONTROL ) {
/* /dev/aloadC? */
int card = SNDRV_MINOR_CARD ( minor ) ;
2005-04-16 15:20:36 -07:00
if ( snd_cards [ card ] = = NULL )
2005-11-20 14:07:47 +01:00
snd_request_card ( card ) ;
} else if ( dev = = SNDRV_MINOR_GLOBAL ) {
/* /dev/aloadSEQ */
2005-04-16 15:20:36 -07:00
snd_request_other ( minor ) ;
2005-11-20 14:07:47 +01:00
}
# ifndef CONFIG_SND_DYNAMIC_MINORS
/* /dev/snd/{controlC?,seq} */
mptr = snd_minors [ minor ] ;
if ( mptr = = NULL )
# endif
2005-04-16 15:20:36 -07:00
# endif
2005-11-20 14:07:47 +01:00
return - ENODEV ;
2005-04-16 15:20:36 -07:00
}
old_fops = file - > f_op ;
file - > f_op = fops_get ( mptr - > f_ops ) ;
2009-01-06 14:40:40 -08:00
if ( file - > f_op = = NULL ) {
file - > f_op = old_fops ;
return - ENODEV ;
}
2005-04-16 15:20:36 -07:00
if ( file - > f_op - > open )
err = file - > f_op - > open ( inode , file ) ;
if ( err ) {
fops_put ( file - > f_op ) ;
file - > f_op = fops_get ( old_fops ) ;
}
fops_put ( old_fops ) ;
return err ;
}
2008-05-15 16:39:03 -06:00
/* BKL pushdown: nasty #ifdef avoidance wrapper */
static int snd_open ( struct inode * inode , struct file * file )
{
int ret ;
lock_kernel ( ) ;
ret = __snd_open ( inode , file ) ;
unlock_kernel ( ) ;
return ret ;
}
2007-02-12 00:55:37 -08:00
static const struct file_operations snd_fops =
2005-04-16 15:20:36 -07:00
{
. owner = THIS_MODULE ,
. open = snd_open
} ;
2005-11-20 14:07:47 +01:00
# ifdef CONFIG_SND_DYNAMIC_MINORS
static int snd_find_free_minor ( void )
{
int minor ;
for ( minor = 0 ; minor < ARRAY_SIZE ( snd_minors ) ; + + minor ) {
/* skip minors still used statically for autoloading devices */
if ( SNDRV_MINOR_DEVICE ( minor ) = = SNDRV_MINOR_CONTROL | |
minor = = SNDRV_MINOR_SEQUENCER )
continue ;
if ( ! snd_minors [ minor ] )
return minor ;
}
return - EBUSY ;
}
# else
2005-11-17 13:51:18 +01:00
static int snd_kernel_minor ( int type , struct snd_card * card , int dev )
2005-04-16 15:20:36 -07:00
{
int minor ;
switch ( type ) {
case SNDRV_DEVICE_TYPE_SEQUENCER :
case SNDRV_DEVICE_TYPE_TIMER :
minor = type ;
break ;
case SNDRV_DEVICE_TYPE_CONTROL :
2008-08-08 17:09:09 +02:00
if ( snd_BUG_ON ( ! card ) )
return - EINVAL ;
2005-04-16 15:20:36 -07:00
minor = SNDRV_MINOR ( card - > number , type ) ;
break ;
case SNDRV_DEVICE_TYPE_HWDEP :
case SNDRV_DEVICE_TYPE_RAWMIDI :
case SNDRV_DEVICE_TYPE_PCM_PLAYBACK :
case SNDRV_DEVICE_TYPE_PCM_CAPTURE :
2008-08-08 17:09:09 +02:00
if ( snd_BUG_ON ( ! card ) )
return - EINVAL ;
2005-04-16 15:20:36 -07:00
minor = SNDRV_MINOR ( card - > number , type + dev ) ;
break ;
default :
return - EINVAL ;
}
2008-08-08 17:09:09 +02:00
if ( snd_BUG_ON ( minor < 0 | | minor > = SNDRV_OS_MINORS ) )
return - EINVAL ;
2005-04-16 15:20:36 -07:00
return minor ;
}
2005-11-20 14:07:47 +01:00
# endif
2005-04-16 15:20:36 -07:00
/**
2006-10-05 15:05:34 +02:00
* snd_register_device_for_dev - Register the ALSA device file for the card
2005-04-16 15:20:36 -07:00
* @ type : the device type , SNDRV_DEVICE_TYPE_XXX
* @ card : the card instance
* @ dev : the device index
2005-11-20 14:03:48 +01:00
* @ f_ops : the file operations
2005-11-20 14:06:59 +01:00
* @ private_data : user pointer for f_ops - > open ( )
2005-04-16 15:20:36 -07:00
* @ name : the device file name
2006-10-05 15:05:34 +02:00
* @ device : the & struct device to link this new device to
2005-04-16 15:20:36 -07:00
*
* Registers an ALSA device file for the given card .
* The operators have to be set in reg parameter .
*
2006-10-05 15:05:34 +02:00
* Returns zero if successful , or a negative error code on failure .
2005-04-16 15:20:36 -07:00
*/
2006-10-05 15:05:34 +02:00
int snd_register_device_for_dev ( int type , struct snd_card * card , int dev ,
const struct file_operations * f_ops ,
void * private_data ,
const char * name , struct device * device )
2005-04-16 15:20:36 -07:00
{
2005-11-20 14:07:47 +01:00
int minor ;
2005-11-17 13:51:18 +01:00
struct snd_minor * preg ;
2005-04-16 15:20:36 -07:00
2008-08-08 17:09:09 +02:00
if ( snd_BUG_ON ( ! name ) )
return - EINVAL ;
2006-07-05 11:24:22 +02:00
preg = kmalloc ( sizeof * preg , GFP_KERNEL ) ;
2005-04-16 15:20:36 -07:00
if ( preg = = NULL )
return - ENOMEM ;
2005-11-20 14:03:48 +01:00
preg - > type = type ;
2005-11-20 14:05:49 +01:00
preg - > card = card ? card - > number : - 1 ;
2005-04-16 15:20:36 -07:00
preg - > device = dev ;
2005-11-20 14:03:48 +01:00
preg - > f_ops = f_ops ;
2005-11-20 14:06:59 +01:00
preg - > private_data = private_data ;
2006-01-16 16:29:08 +01:00
mutex_lock ( & sound_mutex ) ;
2005-11-20 14:07:47 +01:00
# ifdef CONFIG_SND_DYNAMIC_MINORS
minor = snd_find_free_minor ( ) ;
# else
minor = snd_kernel_minor ( type , card , dev ) ;
if ( minor > = 0 & & snd_minors [ minor ] )
minor = - EBUSY ;
# endif
if ( minor < 0 ) {
2006-01-16 16:29:08 +01:00
mutex_unlock ( & sound_mutex ) ;
2005-04-16 15:20:36 -07:00
kfree ( preg ) ;
2005-11-20 14:07:47 +01:00
return minor ;
2005-04-16 15:20:36 -07:00
}
2005-11-20 14:05:49 +01:00
snd_minors [ minor ] = preg ;
2008-07-21 20:03:34 -07:00
preg - > dev = device_create ( sound_class , device , MKDEV ( major , minor ) ,
private_data , " %s " , name ) ;
2007-10-09 10:34:06 +02:00
if ( IS_ERR ( preg - > dev ) ) {
snd_minors [ minor ] = NULL ;
mutex_unlock ( & sound_mutex ) ;
minor = PTR_ERR ( preg - > dev ) ;
kfree ( preg ) ;
return minor ;
}
2006-01-16 16:29:08 +01:00
mutex_unlock ( & sound_mutex ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
2006-10-05 15:05:34 +02:00
EXPORT_SYMBOL ( snd_register_device_for_dev ) ;
2006-04-28 15:13:39 +02:00
2006-09-06 14:27:46 +02:00
/* find the matching minor record
* return the index of snd_minor , or - 1 if not found
*/
static int find_snd_minor ( int type , struct snd_card * card , int dev )
{
int cardnum , minor ;
struct snd_minor * mptr ;
cardnum = card ? card - > number : - 1 ;
for ( minor = 0 ; minor < ARRAY_SIZE ( snd_minors ) ; + + minor )
if ( ( mptr = snd_minors [ minor ] ) ! = NULL & &
mptr - > type = = type & &
mptr - > card = = cardnum & &
mptr - > device = = dev )
return minor ;
return - 1 ;
}
2005-04-16 15:20:36 -07:00
/**
* snd_unregister_device - unregister the device on the given card
* @ type : the device type , SNDRV_DEVICE_TYPE_XXX
* @ card : the card instance
* @ dev : the device index
*
* Unregisters the device file already registered via
* snd_register_device ( ) .
*
* Returns zero if sucecessful , or a negative error code on failure
*/
2005-11-17 13:51:18 +01:00
int snd_unregister_device ( int type , struct snd_card * card , int dev )
2005-04-16 15:20:36 -07:00
{
2006-09-06 14:27:46 +02:00
int minor ;
2005-04-16 15:20:36 -07:00
2006-01-16 16:29:08 +01:00
mutex_lock ( & sound_mutex ) ;
2006-09-06 14:27:46 +02:00
minor = find_snd_minor ( type , card , dev ) ;
if ( minor < 0 ) {
2006-01-16 16:29:08 +01:00
mutex_unlock ( & sound_mutex ) ;
2005-04-16 15:20:36 -07:00
return - EINVAL ;
}
2006-08-07 22:19:37 -07:00
device_destroy ( sound_class , MKDEV ( major , minor ) ) ;
2005-04-16 15:20:36 -07:00
2006-09-06 14:27:46 +02:00
kfree ( snd_minors [ minor ] ) ;
2005-11-20 14:05:49 +01:00
snd_minors [ minor ] = NULL ;
2006-01-16 16:29:08 +01:00
mutex_unlock ( & sound_mutex ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
2006-04-28 15:13:39 +02:00
EXPORT_SYMBOL ( snd_unregister_device ) ;
2006-09-06 14:27:46 +02:00
int snd_add_device_sysfs_file ( int type , struct snd_card * card , int dev ,
2006-08-07 22:19:37 -07:00
struct device_attribute * attr )
2006-09-06 14:27:46 +02:00
{
int minor , ret = - EINVAL ;
2006-08-07 22:19:37 -07:00
struct device * d ;
2006-09-06 14:27:46 +02:00
mutex_lock ( & sound_mutex ) ;
minor = find_snd_minor ( type , card , dev ) ;
2006-08-07 22:19:37 -07:00
if ( minor > = 0 & & ( d = snd_minors [ minor ] - > dev ) ! = NULL )
ret = device_create_file ( d , attr ) ;
2006-09-06 14:27:46 +02:00
mutex_unlock ( & sound_mutex ) ;
return ret ;
}
EXPORT_SYMBOL ( snd_add_device_sysfs_file ) ;
2005-12-01 10:42:42 +01:00
# ifdef CONFIG_PROC_FS
2005-04-16 15:20:36 -07:00
/*
* INFO PART
*/
2006-05-17 17:14:51 +02:00
static struct snd_info_entry * snd_minor_info_entry ;
2005-04-16 15:20:36 -07:00
2005-11-20 14:03:48 +01:00
static const char * snd_device_type_name ( int type )
{
switch ( type ) {
case SNDRV_DEVICE_TYPE_CONTROL :
return " control " ;
case SNDRV_DEVICE_TYPE_HWDEP :
return " hardware dependent " ;
case SNDRV_DEVICE_TYPE_RAWMIDI :
return " raw midi " ;
case SNDRV_DEVICE_TYPE_PCM_PLAYBACK :
return " digital audio playback " ;
case SNDRV_DEVICE_TYPE_PCM_CAPTURE :
return " digital audio capture " ;
case SNDRV_DEVICE_TYPE_SEQUENCER :
return " sequencer " ;
case SNDRV_DEVICE_TYPE_TIMER :
return " timer " ;
default :
return " ? " ;
}
}
2005-11-17 13:51:18 +01:00
static void snd_minor_info_read ( struct snd_info_entry * entry , struct snd_info_buffer * buffer )
2005-04-16 15:20:36 -07:00
{
2005-11-20 14:05:49 +01:00
int minor ;
2005-11-17 13:51:18 +01:00
struct snd_minor * mptr ;
2005-04-16 15:20:36 -07:00
2006-01-16 16:29:08 +01:00
mutex_lock ( & sound_mutex ) ;
2005-11-20 14:05:49 +01:00
for ( minor = 0 ; minor < SNDRV_OS_MINORS ; + + minor ) {
if ( ! ( mptr = snd_minors [ minor ] ) )
continue ;
if ( mptr - > card > = 0 ) {
if ( mptr - > device > = 0 )
2005-11-20 14:09:05 +01:00
snd_iprintf ( buffer , " %3i: [%2i-%2i]: %s \n " ,
2005-11-20 14:05:49 +01:00
minor , mptr - > card , mptr - > device ,
snd_device_type_name ( mptr - > type ) ) ;
else
2005-11-20 14:09:05 +01:00
snd_iprintf ( buffer , " %3i: [%2i] : %s \n " ,
2005-11-20 14:05:49 +01:00
minor , mptr - > card ,
snd_device_type_name ( mptr - > type ) ) ;
} else
2005-11-20 14:09:05 +01:00
snd_iprintf ( buffer , " %3i: : %s \n " , minor ,
2005-11-20 14:05:49 +01:00
snd_device_type_name ( mptr - > type ) ) ;
2005-04-16 15:20:36 -07:00
}
2006-01-16 16:29:08 +01:00
mutex_unlock ( & sound_mutex ) ;
2005-04-16 15:20:36 -07:00
}
int __init snd_minor_info_init ( void )
{
2005-11-17 13:51:18 +01:00
struct snd_info_entry * entry ;
2005-04-16 15:20:36 -07:00
entry = snd_info_create_module_entry ( THIS_MODULE , " devices " , NULL ) ;
if ( entry ) {
entry - > c . text . read = snd_minor_info_read ;
if ( snd_info_register ( entry ) < 0 ) {
snd_info_free_entry ( entry ) ;
entry = NULL ;
}
}
snd_minor_info_entry = entry ;
return 0 ;
}
int __exit snd_minor_info_done ( void )
{
2006-06-23 14:37:59 +02:00
snd_info_free_entry ( snd_minor_info_entry ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
2005-12-01 10:42:42 +01:00
# endif /* CONFIG_PROC_FS */
2005-04-16 15:20:36 -07:00
/*
* INIT PART
*/
static int __init alsa_sound_init ( void )
{
snd_major = major ;
snd_ecards_limit = cards_limit ;
if ( register_chrdev ( major , " alsa " , & snd_fops ) ) {
snd_printk ( KERN_ERR " unable to register native major device number %d \n " , major ) ;
return - EIO ;
}
if ( snd_info_init ( ) < 0 ) {
unregister_chrdev ( major , " alsa " ) ;
return - ENOMEM ;
}
snd_info_minor_register ( ) ;
# ifndef MODULE
printk ( KERN_INFO " Advanced Linux Sound Architecture Driver Version " CONFIG_SND_VERSION CONFIG_SND_DATE " . \n " ) ;
# endif
return 0 ;
}
static void __exit alsa_sound_exit ( void )
{
snd_info_minor_unregister ( ) ;
snd_info_done ( ) ;
2007-07-19 01:47:50 -07:00
unregister_chrdev ( major , " alsa " ) ;
2005-04-16 15:20:36 -07:00
}
module_init ( alsa_sound_init )
module_exit ( alsa_sound_exit )