2019-05-27 08:55:06 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2014-04-04 20:40:50 +02:00
/*
* Behringer BCD2000 driver
*
* Copyright ( C ) 2014 Mario Kicherer ( dev @ kicherer . org )
*/
# include <linux/kernel.h>
# include <linux/errno.h>
# include <linux/init.h>
# include <linux/slab.h>
# include <linux/module.h>
# include <linux/bitmap.h>
# include <linux/usb.h>
# include <linux/usb/audio.h>
# include <sound/core.h>
# include <sound/initval.h>
# include <sound/rawmidi.h>
# define PREFIX "snd-bcd2000: "
# define BUFSIZE 64
2017-08-06 12:18:54 +05:30
static const struct usb_device_id id_table [ ] = {
2014-04-04 20:40:50 +02:00
{ USB_DEVICE ( 0x1397 , 0x00bd ) } ,
{ } ,
} ;
2020-01-05 15:47:41 +01:00
static const unsigned char device_cmd_prefix [ ] = { 0x03 , 0x00 } ;
2014-04-04 20:40:50 +02:00
2020-01-05 15:47:41 +01:00
static const unsigned char bcd2000_init_sequence [ ] = {
2014-04-04 20:40:50 +02:00
0x07 , 0x00 , 0x00 , 0x00 , 0x78 , 0x48 , 0x1c , 0x81 ,
0xc4 , 0x00 , 0x00 , 0x00 , 0x5e , 0x53 , 0x4a , 0xf7 ,
0x18 , 0xfa , 0x11 , 0xff , 0x6c , 0xf3 , 0x90 , 0xff ,
0x00 , 0x00 , 0x00 , 0x00 , 0x01 , 0x00 , 0x00 , 0x00 ,
0x18 , 0xfa , 0x11 , 0xff , 0x14 , 0x00 , 0x00 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0xf2 , 0x34 , 0x4a , 0xf7 ,
0x18 , 0xfa , 0x11 , 0xff
} ;
struct bcd2000 {
struct usb_device * dev ;
struct snd_card * card ;
struct usb_interface * intf ;
int card_index ;
int midi_out_active ;
struct snd_rawmidi * rmidi ;
struct snd_rawmidi_substream * midi_receive_substream ;
struct snd_rawmidi_substream * midi_out_substream ;
unsigned char midi_in_buf [ BUFSIZE ] ;
unsigned char midi_out_buf [ BUFSIZE ] ;
struct urb * midi_out_urb ;
struct urb * midi_in_urb ;
struct usb_anchor anchor ;
} ;
static int index [ SNDRV_CARDS ] = SNDRV_DEFAULT_IDX ;
static char * id [ SNDRV_CARDS ] = SNDRV_DEFAULT_STR ;
static DEFINE_MUTEX ( devices_mutex ) ;
2015-05-26 12:57:09 +02:00
static DECLARE_BITMAP ( devices_used , SNDRV_CARDS ) ;
2014-04-04 20:40:50 +02:00
static struct usb_driver bcd2000_driver ;
# ifdef CONFIG_SND_DEBUG
static void bcd2000_dump_buffer ( const char * prefix , const char * buf , int len )
{
print_hex_dump ( KERN_DEBUG , prefix ,
DUMP_PREFIX_NONE , 16 , 1 ,
buf , len , false ) ;
}
# else
static void bcd2000_dump_buffer ( const char * prefix , const char * buf , int len ) { }
# endif
static int bcd2000_midi_input_open ( struct snd_rawmidi_substream * substream )
{
return 0 ;
}
static int bcd2000_midi_input_close ( struct snd_rawmidi_substream * substream )
{
return 0 ;
}
/* (de)register midi substream from client */
static void bcd2000_midi_input_trigger ( struct snd_rawmidi_substream * substream ,
int up )
{
struct bcd2000 * bcd2k = substream - > rmidi - > private_data ;
bcd2k - > midi_receive_substream = up ? substream : NULL ;
}
static void bcd2000_midi_handle_input ( struct bcd2000 * bcd2k ,
const unsigned char * buf , unsigned int buf_len )
{
unsigned int payload_length , tocopy ;
struct snd_rawmidi_substream * midi_receive_substream ;
locking/atomics: COCCINELLE/treewide: Convert trivial ACCESS_ONCE() patterns to READ_ONCE()/WRITE_ONCE()
Please do not apply this to mainline directly, instead please re-run the
coccinelle script shown below and apply its output.
For several reasons, it is desirable to use {READ,WRITE}_ONCE() in
preference to ACCESS_ONCE(), and new code is expected to use one of the
former. So far, there's been no reason to change most existing uses of
ACCESS_ONCE(), as these aren't harmful, and changing them results in
churn.
However, for some features, the read/write distinction is critical to
correct operation. To distinguish these cases, separate read/write
accessors must be used. This patch migrates (most) remaining
ACCESS_ONCE() instances to {READ,WRITE}_ONCE(), using the following
coccinelle script:
----
// Convert trivial ACCESS_ONCE() uses to equivalent READ_ONCE() and
// WRITE_ONCE()
// $ make coccicheck COCCI=/home/mark/once.cocci SPFLAGS="--include-headers" MODE=patch
virtual patch
@ depends on patch @
expression E1, E2;
@@
- ACCESS_ONCE(E1) = E2
+ WRITE_ONCE(E1, E2)
@ depends on patch @
expression E;
@@
- ACCESS_ONCE(E)
+ READ_ONCE(E)
----
Signed-off-by: Mark Rutland <mark.rutland@arm.com>
Signed-off-by: Paul E. McKenney <paulmck@linux.vnet.ibm.com>
Cc: Linus Torvalds <torvalds@linux-foundation.org>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: davem@davemloft.net
Cc: linux-arch@vger.kernel.org
Cc: mpe@ellerman.id.au
Cc: shuah@kernel.org
Cc: snitzer@redhat.com
Cc: thor.thayer@linux.intel.com
Cc: tj@kernel.org
Cc: viro@zeniv.linux.org.uk
Cc: will.deacon@arm.com
Link: http://lkml.kernel.org/r/1508792849-3115-19-git-send-email-paulmck@linux.vnet.ibm.com
Signed-off-by: Ingo Molnar <mingo@kernel.org>
2017-10-23 14:07:29 -07:00
midi_receive_substream = READ_ONCE ( bcd2k - > midi_receive_substream ) ;
2014-04-04 20:40:50 +02:00
if ( ! midi_receive_substream )
return ;
bcd2000_dump_buffer ( PREFIX " received from device: " , buf , buf_len ) ;
if ( buf_len < 2 )
return ;
payload_length = buf [ 0 ] ;
/* ignore packets without payload */
if ( payload_length = = 0 )
return ;
tocopy = min ( payload_length , buf_len - 1 ) ;
bcd2000_dump_buffer ( PREFIX " sending to userspace: " ,
& buf [ 1 ] , tocopy ) ;
snd_rawmidi_receive ( midi_receive_substream ,
& buf [ 1 ] , tocopy ) ;
}
static void bcd2000_midi_send ( struct bcd2000 * bcd2k )
{
int len , ret ;
struct snd_rawmidi_substream * midi_out_substream ;
BUILD_BUG_ON ( sizeof ( device_cmd_prefix ) > = BUFSIZE ) ;
locking/atomics: COCCINELLE/treewide: Convert trivial ACCESS_ONCE() patterns to READ_ONCE()/WRITE_ONCE()
Please do not apply this to mainline directly, instead please re-run the
coccinelle script shown below and apply its output.
For several reasons, it is desirable to use {READ,WRITE}_ONCE() in
preference to ACCESS_ONCE(), and new code is expected to use one of the
former. So far, there's been no reason to change most existing uses of
ACCESS_ONCE(), as these aren't harmful, and changing them results in
churn.
However, for some features, the read/write distinction is critical to
correct operation. To distinguish these cases, separate read/write
accessors must be used. This patch migrates (most) remaining
ACCESS_ONCE() instances to {READ,WRITE}_ONCE(), using the following
coccinelle script:
----
// Convert trivial ACCESS_ONCE() uses to equivalent READ_ONCE() and
// WRITE_ONCE()
// $ make coccicheck COCCI=/home/mark/once.cocci SPFLAGS="--include-headers" MODE=patch
virtual patch
@ depends on patch @
expression E1, E2;
@@
- ACCESS_ONCE(E1) = E2
+ WRITE_ONCE(E1, E2)
@ depends on patch @
expression E;
@@
- ACCESS_ONCE(E)
+ READ_ONCE(E)
----
Signed-off-by: Mark Rutland <mark.rutland@arm.com>
Signed-off-by: Paul E. McKenney <paulmck@linux.vnet.ibm.com>
Cc: Linus Torvalds <torvalds@linux-foundation.org>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: davem@davemloft.net
Cc: linux-arch@vger.kernel.org
Cc: mpe@ellerman.id.au
Cc: shuah@kernel.org
Cc: snitzer@redhat.com
Cc: thor.thayer@linux.intel.com
Cc: tj@kernel.org
Cc: viro@zeniv.linux.org.uk
Cc: will.deacon@arm.com
Link: http://lkml.kernel.org/r/1508792849-3115-19-git-send-email-paulmck@linux.vnet.ibm.com
Signed-off-by: Ingo Molnar <mingo@kernel.org>
2017-10-23 14:07:29 -07:00
midi_out_substream = READ_ONCE ( bcd2k - > midi_out_substream ) ;
2014-04-04 20:40:50 +02:00
if ( ! midi_out_substream )
return ;
/* copy command prefix bytes */
memcpy ( bcd2k - > midi_out_buf , device_cmd_prefix ,
sizeof ( device_cmd_prefix ) ) ;
/*
* get MIDI packet and leave space for command prefix
* and payload length
*/
len = snd_rawmidi_transmit ( midi_out_substream ,
bcd2k - > midi_out_buf + 3 , BUFSIZE - 3 ) ;
if ( len < 0 )
dev_err ( & bcd2k - > dev - > dev , " %s: snd_rawmidi_transmit error %d \n " ,
__func__ , len ) ;
if ( len < = 0 )
return ;
/* set payload length */
bcd2k - > midi_out_buf [ 2 ] = len ;
bcd2k - > midi_out_urb - > transfer_buffer_length = BUFSIZE ;
bcd2000_dump_buffer ( PREFIX " sending to device: " ,
bcd2k - > midi_out_buf , len + 3 ) ;
/* send packet to the BCD2000 */
ret = usb_submit_urb ( bcd2k - > midi_out_urb , GFP_ATOMIC ) ;
if ( ret < 0 )
dev_err ( & bcd2k - > dev - > dev , PREFIX
" %s (%p): usb_submit_urb() failed, ret=%d, len=%d \n " ,
__func__ , midi_out_substream , ret , len ) ;
else
bcd2k - > midi_out_active = 1 ;
}
static int bcd2000_midi_output_open ( struct snd_rawmidi_substream * substream )
{
return 0 ;
}
static int bcd2000_midi_output_close ( struct snd_rawmidi_substream * substream )
{
struct bcd2000 * bcd2k = substream - > rmidi - > private_data ;
if ( bcd2k - > midi_out_active ) {
usb_kill_urb ( bcd2k - > midi_out_urb ) ;
bcd2k - > midi_out_active = 0 ;
}
return 0 ;
}
/* (de)register midi substream from client */
static void bcd2000_midi_output_trigger ( struct snd_rawmidi_substream * substream ,
int up )
{
struct bcd2000 * bcd2k = substream - > rmidi - > private_data ;
if ( up ) {
bcd2k - > midi_out_substream = substream ;
/* check if there is data userspace wants to send */
if ( ! bcd2k - > midi_out_active )
bcd2000_midi_send ( bcd2k ) ;
} else {
bcd2k - > midi_out_substream = NULL ;
}
}
static void bcd2000_output_complete ( struct urb * urb )
{
struct bcd2000 * bcd2k = urb - > context ;
bcd2k - > midi_out_active = 0 ;
if ( urb - > status )
dev_warn ( & urb - > dev - > dev ,
PREFIX " output urb->status: %d \n " , urb - > status ) ;
if ( urb - > status = = - ESHUTDOWN )
return ;
/* check if there is more data userspace wants to send */
bcd2000_midi_send ( bcd2k ) ;
}
static void bcd2000_input_complete ( struct urb * urb )
{
int ret ;
struct bcd2000 * bcd2k = urb - > context ;
if ( urb - > status )
dev_warn ( & urb - > dev - > dev ,
PREFIX " input urb->status: %i \n " , urb - > status ) ;
if ( ! bcd2k | | urb - > status = = - ESHUTDOWN )
return ;
if ( urb - > actual_length > 0 )
bcd2000_midi_handle_input ( bcd2k , urb - > transfer_buffer ,
urb - > actual_length ) ;
/* return URB to device */
ret = usb_submit_urb ( bcd2k - > midi_in_urb , GFP_ATOMIC ) ;
if ( ret < 0 )
dev_err ( & bcd2k - > dev - > dev , PREFIX
" %s: usb_submit_urb() failed, ret=%d \n " ,
__func__ , ret ) ;
}
2017-01-05 17:30:12 +01:00
static const struct snd_rawmidi_ops bcd2000_midi_output = {
2014-04-04 20:40:50 +02:00
. open = bcd2000_midi_output_open ,
. close = bcd2000_midi_output_close ,
. trigger = bcd2000_midi_output_trigger ,
} ;
2017-01-05 17:30:12 +01:00
static const struct snd_rawmidi_ops bcd2000_midi_input = {
2014-04-04 20:40:50 +02:00
. open = bcd2000_midi_input_open ,
. close = bcd2000_midi_input_close ,
. trigger = bcd2000_midi_input_trigger ,
} ;
static void bcd2000_init_device ( struct bcd2000 * bcd2k )
{
int ret ;
init_usb_anchor ( & bcd2k - > anchor ) ;
usb_anchor_urb ( bcd2k - > midi_out_urb , & bcd2k - > anchor ) ;
usb_anchor_urb ( bcd2k - > midi_in_urb , & bcd2k - > anchor ) ;
/* copy init sequence into buffer */
memcpy ( bcd2k - > midi_out_buf , bcd2000_init_sequence , 52 ) ;
bcd2k - > midi_out_urb - > transfer_buffer_length = 52 ;
/* submit sequence */
ret = usb_submit_urb ( bcd2k - > midi_out_urb , GFP_KERNEL ) ;
if ( ret < 0 )
dev_err ( & bcd2k - > dev - > dev , PREFIX
" %s: usb_submit_urb() out failed, ret=%d: " ,
__func__ , ret ) ;
else
bcd2k - > midi_out_active = 1 ;
/* pass URB to device to enable button and controller events */
ret = usb_submit_urb ( bcd2k - > midi_in_urb , GFP_KERNEL ) ;
if ( ret < 0 )
dev_err ( & bcd2k - > dev - > dev , PREFIX
" %s: usb_submit_urb() in failed, ret=%d: " ,
__func__ , ret ) ;
/* ensure initialization is finished */
usb_wait_anchor_empty_timeout ( & bcd2k - > anchor , 1000 ) ;
}
static int bcd2000_init_midi ( struct bcd2000 * bcd2k )
{
int ret ;
struct snd_rawmidi * rmidi ;
ret = snd_rawmidi_new ( bcd2k - > card , bcd2k - > card - > shortname , 0 ,
1 , /* output */
1 , /* input */
& rmidi ) ;
if ( ret < 0 )
return ret ;
ALSA: Convert strlcpy to strscpy when return value is unused
strlcpy is deprecated. see: Documentation/process/deprecated.rst
Change the calls that do not use the strlcpy return value to the
preferred strscpy.
Done with cocci script:
@@
expression e1, e2, e3;
@@
- strlcpy(
+ strscpy(
e1, e2, e3);
This cocci script leaves the instances where the return value is
used unchanged.
After this patch, sound/ has 3 uses of strlcpy() that need to be
manually inspected for conversion and changed one day.
$ git grep -w strlcpy sound/
sound/usb/card.c: len = strlcpy(card->longname, s, sizeof(card->longname));
sound/usb/mixer.c: return strlcpy(buf, p->name, buflen);
sound/usb/mixer.c: return strlcpy(buf, p->names[index], buflen);
Miscellenea:
o Remove trailing whitespace in conversion of sound/core/hwdep.c
Link: https://lore.kernel.org/lkml/CAHk-=wgfRnXz0W3D37d01q3JFkr_i_uTL=V6A6G1oUZcprmknw@mail.gmail.com/
Signed-off-by: Joe Perches <joe@perches.com>
Acked-by: Mark Brown <broonie@kernel.org>
Link: https://lore.kernel.org/r/22b393d1790bb268769d0bab7bacf0866dcb0c14.camel@perches.com
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2021-01-04 09:17:34 -08:00
strscpy ( rmidi - > name , bcd2k - > card - > shortname , sizeof ( rmidi - > name ) ) ;
2014-04-04 20:40:50 +02:00
rmidi - > info_flags = SNDRV_RAWMIDI_INFO_DUPLEX ;
rmidi - > private_data = bcd2k ;
rmidi - > info_flags | = SNDRV_RAWMIDI_INFO_OUTPUT ;
snd_rawmidi_set_ops ( rmidi , SNDRV_RAWMIDI_STREAM_OUTPUT ,
& bcd2000_midi_output ) ;
rmidi - > info_flags | = SNDRV_RAWMIDI_INFO_INPUT ;
snd_rawmidi_set_ops ( rmidi , SNDRV_RAWMIDI_STREAM_INPUT ,
& bcd2000_midi_input ) ;
bcd2k - > rmidi = rmidi ;
bcd2k - > midi_in_urb = usb_alloc_urb ( 0 , GFP_KERNEL ) ;
bcd2k - > midi_out_urb = usb_alloc_urb ( 0 , GFP_KERNEL ) ;
if ( ! bcd2k - > midi_in_urb | | ! bcd2k - > midi_out_urb ) {
dev_err ( & bcd2k - > dev - > dev , PREFIX " usb_alloc_urb failed \n " ) ;
return - ENOMEM ;
}
usb_fill_int_urb ( bcd2k - > midi_in_urb , bcd2k - > dev ,
usb_rcvintpipe ( bcd2k - > dev , 0x81 ) ,
bcd2k - > midi_in_buf , BUFSIZE ,
bcd2000_input_complete , bcd2k , 1 ) ;
usb_fill_int_urb ( bcd2k - > midi_out_urb , bcd2k - > dev ,
usb_sndintpipe ( bcd2k - > dev , 0x1 ) ,
bcd2k - > midi_out_buf , BUFSIZE ,
bcd2000_output_complete , bcd2k , 1 ) ;
2017-10-04 15:03:40 +02:00
/* sanity checks of EPs before actually submitting */
if ( usb_urb_ep_type_check ( bcd2k - > midi_in_urb ) | |
usb_urb_ep_type_check ( bcd2k - > midi_out_urb ) ) {
dev_err ( & bcd2k - > dev - > dev , " invalid MIDI EP \n " ) ;
return - EINVAL ;
}
2014-04-04 20:40:50 +02:00
bcd2000_init_device ( bcd2k ) ;
return 0 ;
}
static void bcd2000_free_usb_related_resources ( struct bcd2000 * bcd2k ,
struct usb_interface * interface )
{
2022-07-15 09:05:15 +08:00
usb_kill_urb ( bcd2k - > midi_out_urb ) ;
usb_kill_urb ( bcd2k - > midi_in_urb ) ;
2014-04-04 20:40:50 +02:00
usb_free_urb ( bcd2k - > midi_out_urb ) ;
usb_free_urb ( bcd2k - > midi_in_urb ) ;
if ( bcd2k - > intf ) {
usb_set_intfdata ( bcd2k - > intf , NULL ) ;
bcd2k - > intf = NULL ;
}
}
static int bcd2000_probe ( struct usb_interface * interface ,
const struct usb_device_id * usb_id )
{
struct snd_card * card ;
struct bcd2000 * bcd2k ;
unsigned int card_index ;
char usb_path [ 32 ] ;
int err ;
mutex_lock ( & devices_mutex ) ;
for ( card_index = 0 ; card_index < SNDRV_CARDS ; + + card_index )
if ( ! test_bit ( card_index , devices_used ) )
break ;
if ( card_index > = SNDRV_CARDS ) {
mutex_unlock ( & devices_mutex ) ;
return - ENOENT ;
}
err = snd_card_new ( & interface - > dev , index [ card_index ] , id [ card_index ] ,
THIS_MODULE , sizeof ( * bcd2k ) , & card ) ;
if ( err < 0 ) {
mutex_unlock ( & devices_mutex ) ;
return err ;
}
bcd2k = card - > private_data ;
bcd2k - > dev = interface_to_usbdev ( interface ) ;
bcd2k - > card = card ;
bcd2k - > card_index = card_index ;
bcd2k - > intf = interface ;
snd_card_set_dev ( card , & interface - > dev ) ;
ALSA: bcd2000: refactor deprecated strncpy
`strncpy` is deprecated for use on NUL-terminated destination strings [1].
A suitable replacement is `strscpy` [2] due to the fact that it
guarantees NUL-termination on its destination buffer argument which is
_not_ always the case for `strncpy`!
It should be noted that, in this case, the destination buffer has a
length strictly greater than the source string. Moreover, the source
string is NUL-terminated (and so is the destination) which means there
was no real bug happening here. Nonetheless, this patch would get us one
step closer to eliminating the `strncpy` API in the kernel, as its use
is too ambiguous. We need to favor less ambiguous replacements such as:
strscpy, strscpy_pad, strtomem and strtomem_pad (amongst others).
Technically, my patch yields subtly different behavior. The original
implementation with `strncpy` would fill the entire destination buffer
with null bytes [3] while `strscpy` will leave the junk, uninitialized
bytes trailing after the _mandatory_ NUL-termination. So, if somehow
`card->driver` or `card->shortname` require this NUL-padding behavior
then `strscpy_pad` should be used. My interpretation, though, is that
the aforementioned fields are just fine as NUL-terminated strings.
Please correct my assumptions if needed and I'll send in a v2.
[1]: www.kernel.org/doc/html/latest/process/deprecated.html#strncpy-on-nul-terminated-strings
[2]: manpages.debian.org/testing/linux-manual-4.8/strscpy.9.en.html
[3]: https://linux.die.net/man/3/strncpy
Link: https://github.com/KSPP/linux/issues/90
Link: https://lore.kernel.org/r/20230727-sound-xen-v1-1-89dd161351f1@google.com (related ALSA patch)
Signed-off-by: Justin Stitt <justinstitt@google.com>
Link: https://lore.kernel.org/r/20230727-sound-usb-bcd2000-v1-1-0dc73684b2f0@google.com
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2023-07-27 22:09:29 +00:00
strscpy ( card - > driver , " snd-bcd2000 " , sizeof ( card - > driver ) ) ;
strscpy ( card - > shortname , " BCD2000 " , sizeof ( card - > shortname ) ) ;
2014-04-04 20:40:50 +02:00
usb_make_path ( bcd2k - > dev , usb_path , sizeof ( usb_path ) ) ;
snprintf ( bcd2k - > card - > longname , sizeof ( bcd2k - > card - > longname ) ,
" Behringer BCD2000 at %s " ,
usb_path ) ;
err = bcd2000_init_midi ( bcd2k ) ;
if ( err < 0 )
goto probe_error ;
err = snd_card_register ( card ) ;
if ( err < 0 )
goto probe_error ;
usb_set_intfdata ( interface , bcd2k ) ;
set_bit ( card_index , devices_used ) ;
mutex_unlock ( & devices_mutex ) ;
return 0 ;
probe_error :
dev_info ( & bcd2k - > dev - > dev , PREFIX " error during probing " ) ;
bcd2000_free_usb_related_resources ( bcd2k , interface ) ;
snd_card_free ( card ) ;
mutex_unlock ( & devices_mutex ) ;
return err ;
}
static void bcd2000_disconnect ( struct usb_interface * interface )
{
struct bcd2000 * bcd2k = usb_get_intfdata ( interface ) ;
if ( ! bcd2k )
return ;
mutex_lock ( & devices_mutex ) ;
/* make sure that userspace cannot create new requests */
snd_card_disconnect ( bcd2k - > card ) ;
bcd2000_free_usb_related_resources ( bcd2k , interface ) ;
clear_bit ( bcd2k - > card_index , devices_used ) ;
snd_card_free_when_closed ( bcd2k - > card ) ;
mutex_unlock ( & devices_mutex ) ;
}
static struct usb_driver bcd2000_driver = {
. name = " snd-bcd2000 " ,
. probe = bcd2000_probe ,
. disconnect = bcd2000_disconnect ,
. id_table = id_table ,
} ;
module_usb_driver ( bcd2000_driver ) ;
MODULE_DEVICE_TABLE ( usb , id_table ) ;
MODULE_AUTHOR ( " Mario Kicherer, dev@kicherer.org " ) ;
MODULE_DESCRIPTION ( " Behringer BCD2000 driver " ) ;
MODULE_LICENSE ( " GPL " ) ;