2019-05-27 09:55:05 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
2005-04-17 02:20:36 +04:00
/*
* Dummy soundcard for virtual rawmidi devices
*
* Copyright ( c ) 2000 by Takashi Iwai < tiwai @ suse . de >
*/
/*
* VIRTUAL RAW MIDI DEVICE CARDS
*
* This dummy card contains up to 4 virtual rawmidi devices .
* They are not real rawmidi devices but just associated with sequencer
* clients , so that any input / output sources can be connected as a raw
* MIDI device arbitrary .
* Also , multiple access is allowed to a single rawmidi device .
*
* Typical usage is like following :
* - Load snd - virmidi module .
* # modprobe snd - virmidi index = 2
* Then , sequencer clients 72 : 0 to 75 : 0 will be created , which are
* mapped from / dev / snd / midiC1D0 to / dev / snd / midiC1D3 , respectively .
*
* - Connect input / output via aconnect .
* % aconnect 64 : 0 72 : 0 # keyboard input redirection 64 : 0 - > 72 : 0
* % aconnect 72 : 0 65 : 0 # output device redirection 72 : 0 - > 65 : 0
*
* - Run application using a midi device ( eg . / dev / snd / midiC1D0 )
*/
# include <linux/init.h>
# include <linux/wait.h>
2005-11-17 18:03:26 +03:00
# include <linux/err.h>
# include <linux/platform_device.h>
2011-07-15 21:13:37 +04:00
# include <linux/module.h>
2005-04-17 02:20:36 +04:00
# include <sound/core.h>
# include <sound/seq_kernel.h>
# include <sound/seq_virmidi.h>
# include <sound/initval.h>
/* hack: OSS defines midi_devs, so undefine it (versioned symbols) */
# undef midi_devs
MODULE_AUTHOR ( " Takashi Iwai <tiwai@suse.de> " ) ;
MODULE_DESCRIPTION ( " Dummy soundcard for virtual rawmidi devices " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_SUPPORTED_DEVICE ( " {{ALSA,Virtual rawmidi device}} " ) ;
2005-11-20 16:08:28 +03:00
# define MAX_MIDI_DEVICES 4
2005-04-17 02:20:36 +04:00
static int index [ SNDRV_CARDS ] = SNDRV_DEFAULT_IDX ; /* Index 0-MAX */
static char * id [ SNDRV_CARDS ] = SNDRV_DEFAULT_STR ; /* ID for this card */
2011-12-15 07:19:36 +04:00
static bool enable [ SNDRV_CARDS ] = { 1 , [ 1 . . . ( SNDRV_CARDS - 1 ) ] = 0 } ;
2005-04-17 02:20:36 +04:00
static int midi_devs [ SNDRV_CARDS ] = { [ 0 . . . ( SNDRV_CARDS - 1 ) ] = 4 } ;
module_param_array ( index , int , NULL , 0444 ) ;
MODULE_PARM_DESC ( index , " Index value for virmidi soundcard. " ) ;
module_param_array ( id , charp , NULL , 0444 ) ;
MODULE_PARM_DESC ( id , " ID string for virmidi soundcard. " ) ;
module_param_array ( enable , bool , NULL , 0444 ) ;
MODULE_PARM_DESC ( enable , " Enable this soundcard. " ) ;
module_param_array ( midi_devs , int , NULL , 0444 ) ;
2005-11-20 16:08:28 +03:00
MODULE_PARM_DESC ( midi_devs , " MIDI devices # (1-4) " ) ;
2005-04-17 02:20:36 +04:00
2005-11-17 16:27:28 +03:00
struct snd_card_virmidi {
struct snd_card * card ;
struct snd_rawmidi * midi [ MAX_MIDI_DEVICES ] ;
} ;
2005-04-17 02:20:36 +04:00
2005-12-07 11:13:42 +03:00
static struct platform_device * devices [ SNDRV_CARDS ] ;
2005-04-17 02:20:36 +04:00
2012-12-06 21:35:27 +04:00
static int snd_virmidi_probe ( struct platform_device * devptr )
2005-04-17 02:20:36 +04:00
{
2005-11-17 16:27:28 +03:00
struct snd_card * card ;
2005-04-17 02:20:36 +04:00
struct snd_card_virmidi * vmidi ;
int idx , err ;
2005-11-17 18:03:26 +03:00
int dev = devptr - > id ;
2005-04-17 02:20:36 +04:00
2014-01-29 15:59:08 +04:00
err = snd_card_new ( & devptr - > dev , index [ dev ] , id [ dev ] , THIS_MODULE ,
sizeof ( struct snd_card_virmidi ) , & card ) ;
2008-12-28 18:45:02 +03:00
if ( err < 0 )
return err ;
2010-09-05 05:52:54 +04:00
vmidi = card - > private_data ;
2005-04-17 02:20:36 +04:00
vmidi - > card = card ;
if ( midi_devs [ dev ] > MAX_MIDI_DEVICES ) {
2009-02-05 17:51:50 +03:00
snd_printk ( KERN_WARNING
2014-11-28 21:59:56 +03:00
" too much midi devices for virmidi %d: force to use %d \n " ,
dev , MAX_MIDI_DEVICES ) ;
2005-04-17 02:20:36 +04:00
midi_devs [ dev ] = MAX_MIDI_DEVICES ;
}
for ( idx = 0 ; idx < midi_devs [ dev ] ; idx + + ) {
2005-11-17 16:27:28 +03:00
struct snd_rawmidi * rmidi ;
struct snd_virmidi_dev * rdev ;
2014-11-28 21:59:56 +03:00
err = snd_virmidi_new ( card , idx , & rmidi ) ;
if ( err < 0 )
2005-04-17 02:20:36 +04:00
goto __nodev ;
rdev = rmidi - > private_data ;
vmidi - > midi [ idx ] = rmidi ;
strcpy ( rmidi - > name , " Virtual Raw MIDI " ) ;
rdev - > seq_mode = SNDRV_VIRMIDI_SEQ_DISPATCH ;
}
2014-11-28 21:59:56 +03:00
2005-04-17 02:20:36 +04:00
strcpy ( card - > driver , " VirMIDI " ) ;
strcpy ( card - > shortname , " VirMIDI " ) ;
sprintf ( card - > longname , " Virtual MIDI Card %i " , dev + 1 ) ;
2005-09-05 19:17:58 +04:00
2014-11-28 21:59:56 +03:00
err = snd_card_register ( card ) ;
2014-11-30 22:13:16 +03:00
if ( ! err ) {
2005-11-17 18:03:26 +03:00
platform_set_drvdata ( devptr , card ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2014-11-28 21:59:56 +03:00
__nodev :
2005-04-17 02:20:36 +04:00
snd_card_free ( card ) ;
return err ;
}
2012-12-06 21:35:27 +04:00
static int snd_virmidi_remove ( struct platform_device * devptr )
2005-04-17 02:20:36 +04:00
{
2005-11-17 18:03:26 +03:00
snd_card_free ( platform_get_drvdata ( devptr ) ) ;
return 0 ;
}
2005-04-17 02:20:36 +04:00
2005-11-17 18:03:26 +03:00
# define SND_VIRMIDI_DRIVER "snd_virmidi"
static struct platform_driver snd_virmidi_driver = {
. probe = snd_virmidi_probe ,
2012-12-06 21:35:27 +04:00
. remove = snd_virmidi_remove ,
2005-11-17 18:03:26 +03:00
. driver = {
2012-07-02 12:50:24 +04:00
. name = SND_VIRMIDI_DRIVER ,
2005-11-17 18:03:26 +03:00
} ,
} ;
2007-06-25 14:07:38 +04:00
static void snd_virmidi_unregister_all ( void )
2005-12-07 11:13:42 +03:00
{
int i ;
for ( i = 0 ; i < ARRAY_SIZE ( devices ) ; + + i )
platform_device_unregister ( devices [ i ] ) ;
platform_driver_unregister ( & snd_virmidi_driver ) ;
}
2005-11-17 18:03:26 +03:00
static int __init alsa_card_virmidi_init ( void )
{
int i , cards , err ;
2014-11-28 21:59:56 +03:00
err = platform_driver_register ( & snd_virmidi_driver ) ;
if ( err < 0 )
2005-11-17 18:03:26 +03:00
return err ;
cards = 0 ;
2006-02-20 13:57:34 +03:00
for ( i = 0 ; i < SNDRV_CARDS ; i + + ) {
2005-11-17 18:03:26 +03:00
struct platform_device * device ;
2014-11-28 21:59:56 +03:00
if ( ! enable [ i ] )
2006-02-20 13:57:34 +03:00
continue ;
2005-11-17 18:03:26 +03:00
device = platform_device_register_simple ( SND_VIRMIDI_DRIVER ,
i , NULL , 0 ) ;
2006-04-13 14:57:11 +04:00
if ( IS_ERR ( device ) )
continue ;
2006-04-13 14:58:06 +04:00
if ( ! platform_get_drvdata ( device ) ) {
platform_device_unregister ( device ) ;
continue ;
}
2005-12-07 11:13:42 +03:00
devices [ i ] = device ;
2005-04-17 02:20:36 +04:00
cards + + ;
}
if ( ! cards ) {
# ifdef MODULE
printk ( KERN_ERR " Card-VirMIDI soundcard not found or device busy \n " ) ;
# endif
2006-04-13 14:57:11 +04:00
snd_virmidi_unregister_all ( ) ;
return - ENODEV ;
2005-04-17 02:20:36 +04:00
}
return 0 ;
}
static void __exit alsa_card_virmidi_exit ( void )
{
2005-12-07 11:13:42 +03:00
snd_virmidi_unregister_all ( ) ;
2005-04-17 02:20:36 +04:00
}
module_init ( alsa_card_virmidi_init )
module_exit ( alsa_card_virmidi_exit )