2019-05-29 16:57:59 -07:00
// SPDX-License-Identifier: GPL-2.0-only
2015-10-01 22:02:11 +09:00
/*
* tascam . c - a part of driver for TASCAM FireWire series
*
* Copyright ( c ) 2015 Takashi Sakamoto
*/
# include "tascam.h"
MODULE_DESCRIPTION ( " TASCAM FireWire series Driver " ) ;
MODULE_AUTHOR ( " Takashi Sakamoto <o-takashi@sakamocchi.jp> " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;
2017-08-22 22:58:15 +09:00
static const struct snd_tscm_spec model_specs [ ] = {
2015-10-01 22:02:12 +09:00
{
. name = " FW-1884 " ,
. has_adat = true ,
. has_spdif = true ,
. pcm_capture_analog_channels = 8 ,
. pcm_playback_analog_channels = 8 ,
. midi_capture_ports = 4 ,
. midi_playback_ports = 4 ,
} ,
{
. name = " FW-1082 " ,
. has_adat = false ,
. has_spdif = true ,
. pcm_capture_analog_channels = 8 ,
. pcm_playback_analog_channels = 2 ,
. midi_capture_ports = 2 ,
. midi_playback_ports = 2 ,
} ,
2016-02-05 09:56:06 +09:00
{
. name = " FW-1804 " ,
. has_adat = true ,
. has_spdif = true ,
. pcm_capture_analog_channels = 8 ,
. pcm_playback_analog_channels = 2 ,
. midi_capture_ports = 2 ,
. midi_playback_ports = 4 ,
} ,
2015-10-01 22:02:12 +09:00
} ;
2015-10-12 19:10:25 +09:00
static int identify_model ( struct snd_tscm * tscm )
2015-10-01 22:02:11 +09:00
{
struct fw_device * fw_dev = fw_parent_device ( tscm - > unit ) ;
2015-10-12 19:10:25 +09:00
const u32 * config_rom = fw_dev - > config_rom ;
2015-10-19 14:29:27 +03:00
char model [ 9 ] ;
2015-10-12 19:10:25 +09:00
unsigned int i ;
u8 c ;
if ( fw_dev - > config_rom_length < 30 ) {
dev_err ( & tscm - > unit - > device ,
" Configuration ROM is too short. \n " ) ;
return - ENODEV ;
}
/* Pick up model name from certain addresses. */
for ( i = 0 ; i < 8 ; i + + ) {
c = config_rom [ 28 + i / 4 ] > > ( 24 - 8 * ( i % 4 ) ) ;
if ( c = = ' \0 ' )
break ;
model [ i ] = c ;
}
model [ i ] = ' \0 ' ;
for ( i = 0 ; i < ARRAY_SIZE ( model_specs ) ; i + + ) {
if ( strcmp ( model , model_specs [ i ] . name ) = = 0 ) {
tscm - > spec = & model_specs [ i ] ;
break ;
}
}
if ( tscm - > spec = = NULL )
return - ENODEV ;
2015-10-01 22:02:11 +09:00
strcpy ( tscm - > card - > driver , " FW-TASCAM " ) ;
strcpy ( tscm - > card - > shortname , model ) ;
strcpy ( tscm - > card - > mixername , model ) ;
snprintf ( tscm - > card - > longname , sizeof ( tscm - > card - > longname ) ,
2015-10-12 19:10:25 +09:00
" TASCAM %s, GUID %08x%08x at %s, S%d " , model ,
2015-10-18 22:39:53 +09:00
fw_dev - > config_rom [ 3 ] , fw_dev - > config_rom [ 4 ] ,
2015-10-01 22:02:11 +09:00
dev_name ( & tscm - > unit - > device ) , 100 < < fw_dev - > max_speed ) ;
return 0 ;
}
2018-10-10 15:35:02 +09:00
static void tscm_card_free ( struct snd_card * card )
2015-10-01 22:02:11 +09:00
{
2018-10-10 15:35:02 +09:00
struct snd_tscm * tscm = card - > private_data ;
2015-10-12 19:10:21 +09:00
snd_tscm_transaction_unregister ( tscm ) ;
2015-10-01 22:02:15 +09:00
snd_tscm_stream_destroy_duplex ( tscm ) ;
2021-06-07 17:12:47 +09:00
mutex_destroy ( & tscm - > mutex ) ;
fw_unit_put ( tscm - > unit ) ;
2015-10-01 22:02:11 +09:00
}
2021-06-07 17:12:47 +09:00
static int snd_tscm_probe ( struct fw_unit * unit ,
const struct ieee1394_device_id * entry )
2016-03-31 08:47:09 +09:00
{
2021-06-07 17:12:47 +09:00
struct snd_card * card ;
struct snd_tscm * tscm ;
2015-10-01 22:02:11 +09:00
int err ;
2021-06-07 17:12:47 +09:00
err = snd_card_new ( & unit - > device , - 1 , NULL , THIS_MODULE , sizeof ( * tscm ) , & card ) ;
2015-10-01 22:02:11 +09:00
if ( err < 0 )
2021-06-07 17:12:47 +09:00
return err ;
card - > private_free = tscm_card_free ;
tscm = card - > private_data ;
tscm - > unit = fw_unit_get ( unit ) ;
dev_set_drvdata ( & unit - > device , tscm ) ;
tscm - > card = card ;
mutex_init ( & tscm - > mutex ) ;
spin_lock_init ( & tscm - > lock ) ;
init_waitqueue_head ( & tscm - > hwdep_wait ) ;
2015-10-01 22:02:11 +09:00
2015-10-12 19:10:25 +09:00
err = identify_model ( tscm ) ;
2015-10-01 22:02:11 +09:00
if ( err < 0 )
goto error ;
2016-03-31 08:47:09 +09:00
err = snd_tscm_transaction_register ( tscm ) ;
2015-10-01 22:02:15 +09:00
if ( err < 0 )
goto error ;
2016-03-31 08:47:09 +09:00
err = snd_tscm_stream_init_duplex ( tscm ) ;
2015-10-01 22:02:16 +09:00
if ( err < 0 )
goto error ;
2016-03-31 08:47:09 +09:00
snd_tscm_proc_init ( tscm ) ;
err = snd_tscm_create_pcm_devices ( tscm ) ;
2015-10-12 19:10:21 +09:00
if ( err < 0 )
goto error ;
2015-10-12 19:10:23 +09:00
err = snd_tscm_create_midi_devices ( tscm ) ;
if ( err < 0 )
goto error ;
2015-10-01 22:02:17 +09:00
err = snd_tscm_create_hwdep_device ( tscm ) ;
if ( err < 0 )
goto error ;
2021-06-07 17:12:47 +09:00
err = snd_card_register ( card ) ;
2015-10-01 22:02:11 +09:00
if ( err < 0 )
goto error ;
2016-03-31 08:47:09 +09:00
return 0 ;
2021-06-07 17:12:47 +09:00
error :
snd_card_free ( card ) ;
return err ;
2015-10-01 22:02:11 +09:00
}
static void snd_tscm_update ( struct fw_unit * unit )
{
2015-10-01 22:02:15 +09:00
struct snd_tscm * tscm = dev_get_drvdata ( & unit - > device ) ;
2015-10-12 19:10:21 +09:00
snd_tscm_transaction_reregister ( tscm ) ;
2021-06-07 17:12:47 +09:00
mutex_lock ( & tscm - > mutex ) ;
snd_tscm_stream_update_duplex ( tscm ) ;
mutex_unlock ( & tscm - > mutex ) ;
2015-10-01 22:02:11 +09:00
}
static void snd_tscm_remove ( struct fw_unit * unit )
{
struct snd_tscm * tscm = dev_get_drvdata ( & unit - > device ) ;
2021-06-07 17:12:47 +09:00
// Block till all of ALSA character devices are released.
snd_card_free ( tscm - > card ) ;
2015-10-01 22:02:11 +09:00
}
static const struct ieee1394_device_id snd_tscm_id_table [ ] = {
2020-08-23 16:55:37 +09:00
// Tascam, FW-1884.
{
. match_flags = IEEE1394_MATCH_VENDOR_ID |
IEEE1394_MATCH_SPECIFIER_ID |
IEEE1394_MATCH_VERSION ,
. vendor_id = 0x00022e ,
. specifier_id = 0x00022e ,
. version = 0x800000 ,
} ,
// Tascam, FE-8 (.version = 0x800001)
// This kernel module doesn't support FE-8 because the most of features
// can be implemented in userspace without any specific support of this
// module.
//
// .version = 0x800002 is unknown.
//
// Tascam, FW-1082.
{
. match_flags = IEEE1394_MATCH_VENDOR_ID |
IEEE1394_MATCH_SPECIFIER_ID |
IEEE1394_MATCH_VERSION ,
. vendor_id = 0x00022e ,
. specifier_id = 0x00022e ,
. version = 0x800003 ,
} ,
// Tascam, FW-1804.
2015-10-01 22:02:11 +09:00
{
. match_flags = IEEE1394_MATCH_VENDOR_ID |
2020-08-23 16:55:37 +09:00
IEEE1394_MATCH_SPECIFIER_ID |
IEEE1394_MATCH_VERSION ,
2015-10-01 22:02:11 +09:00
. vendor_id = 0x00022e ,
. specifier_id = 0x00022e ,
2020-08-23 16:55:37 +09:00
. version = 0x800004 ,
2015-10-01 22:02:11 +09:00
} ,
{ }
} ;
MODULE_DEVICE_TABLE ( ieee1394 , snd_tscm_id_table ) ;
static struct fw_driver tscm_driver = {
. driver = {
. owner = THIS_MODULE ,
2020-03-06 22:52:29 +09:00
. name = KBUILD_MODNAME ,
2015-10-01 22:02:11 +09:00
. bus = & fw_bus_type ,
} ,
. probe = snd_tscm_probe ,
. update = snd_tscm_update ,
. remove = snd_tscm_remove ,
. id_table = snd_tscm_id_table ,
} ;
static int __init snd_tscm_init ( void )
{
return driver_register ( & tscm_driver . driver ) ;
}
static void __exit snd_tscm_exit ( void )
{
driver_unregister ( & tscm_driver . driver ) ;
}
module_init ( snd_tscm_init ) ;
module_exit ( snd_tscm_exit ) ;