mirror of
https://github.com/AetherDroid/android_kernel_samsung_on5xelte.git
synced 2025-09-08 01:08:03 -04:00
Fixed MTP to work with TWRP
This commit is contained in:
commit
f6dfaef42e
50820 changed files with 20846062 additions and 0 deletions
4
sound/firewire/bebob/Makefile
Normal file
4
sound/firewire/bebob/Makefile
Normal file
|
@ -0,0 +1,4 @@
|
|||
snd-bebob-objs := bebob_command.o bebob_stream.o bebob_proc.o bebob_midi.o \
|
||||
bebob_pcm.o bebob_hwdep.o bebob_terratec.o bebob_yamaha.o \
|
||||
bebob_focusrite.o bebob_maudio.o bebob.o
|
||||
obj-m += snd-bebob.o
|
471
sound/firewire/bebob/bebob.c
Normal file
471
sound/firewire/bebob/bebob.c
Normal file
|
@ -0,0 +1,471 @@
|
|||
/*
|
||||
* bebob.c - a part of driver for BeBoB based devices
|
||||
*
|
||||
* Copyright (c) 2013-2014 Takashi Sakamoto
|
||||
*
|
||||
* Licensed under the terms of the GNU General Public License, version 2.
|
||||
*/
|
||||
|
||||
/*
|
||||
* BeBoB is 'BridgeCo enhanced Breakout Box'. This is installed to firewire
|
||||
* devices with DM1000/DM1100/DM1500 chipset. It gives common way for host
|
||||
* system to handle BeBoB based devices.
|
||||
*/
|
||||
|
||||
#include "bebob.h"
|
||||
|
||||
MODULE_DESCRIPTION("BridgeCo BeBoB driver");
|
||||
MODULE_AUTHOR("Takashi Sakamoto <o-takashi@sakamocchi.jp>");
|
||||
MODULE_LICENSE("GPL v2");
|
||||
|
||||
static int index[SNDRV_CARDS] = SNDRV_DEFAULT_IDX;
|
||||
static char *id[SNDRV_CARDS] = SNDRV_DEFAULT_STR;
|
||||
static bool enable[SNDRV_CARDS] = SNDRV_DEFAULT_ENABLE_PNP;
|
||||
|
||||
module_param_array(index, int, NULL, 0444);
|
||||
MODULE_PARM_DESC(index, "card index");
|
||||
module_param_array(id, charp, NULL, 0444);
|
||||
MODULE_PARM_DESC(id, "ID string");
|
||||
module_param_array(enable, bool, NULL, 0444);
|
||||
MODULE_PARM_DESC(enable, "enable BeBoB sound card");
|
||||
|
||||
static DEFINE_MUTEX(devices_mutex);
|
||||
static DECLARE_BITMAP(devices_used, SNDRV_CARDS);
|
||||
|
||||
/* Offsets from information register. */
|
||||
#define INFO_OFFSET_GUID 0x10
|
||||
#define INFO_OFFSET_HW_MODEL_ID 0x18
|
||||
#define INFO_OFFSET_HW_MODEL_REVISION 0x1c
|
||||
|
||||
#define VEN_EDIROL 0x000040ab
|
||||
#define VEN_PRESONUS 0x00000a92
|
||||
#define VEN_BRIDGECO 0x000007f5
|
||||
#define VEN_MACKIE 0x0000000f
|
||||
#define VEN_STANTON 0x00001260
|
||||
#define VEN_TASCAM 0x0000022e
|
||||
#define VEN_BEHRINGER 0x00001564
|
||||
#define VEN_APOGEE 0x000003db
|
||||
#define VEN_ESI 0x00000f1b
|
||||
#define VEN_ACOUSTIC 0x00000002
|
||||
#define VEN_CME 0x0000000a
|
||||
#define VEN_PHONIC 0x00001496
|
||||
#define VEN_LYNX 0x000019e5
|
||||
#define VEN_ICON 0x00001a9e
|
||||
#define VEN_PRISMSOUND 0x00001198
|
||||
#define VEN_TERRATEC 0x00000aac
|
||||
#define VEN_YAMAHA 0x0000a0de
|
||||
#define VEN_FOCUSRITE 0x0000130e
|
||||
#define VEN_MAUDIO1 0x00000d6c
|
||||
#define VEN_MAUDIO2 0x000007f5
|
||||
|
||||
#define MODEL_FOCUSRITE_SAFFIRE_BOTH 0x00000000
|
||||
#define MODEL_MAUDIO_AUDIOPHILE_BOTH 0x00010060
|
||||
#define MODEL_MAUDIO_FW1814 0x00010071
|
||||
#define MODEL_MAUDIO_PROJECTMIX 0x00010091
|
||||
|
||||
static int
|
||||
name_device(struct snd_bebob *bebob, unsigned int vendor_id)
|
||||
{
|
||||
struct fw_device *fw_dev = fw_parent_device(bebob->unit);
|
||||
char vendor[24] = {0};
|
||||
char model[32] = {0};
|
||||
u32 hw_id;
|
||||
u32 data[2] = {0};
|
||||
u32 revision;
|
||||
int err;
|
||||
|
||||
/* get vendor name from root directory */
|
||||
err = fw_csr_string(fw_dev->config_rom + 5, CSR_VENDOR,
|
||||
vendor, sizeof(vendor));
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
/* get model name from unit directory */
|
||||
err = fw_csr_string(bebob->unit->directory, CSR_MODEL,
|
||||
model, sizeof(model));
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
/* get hardware id */
|
||||
err = snd_bebob_read_quad(bebob->unit, INFO_OFFSET_HW_MODEL_ID,
|
||||
&hw_id);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
/* get hardware revision */
|
||||
err = snd_bebob_read_quad(bebob->unit, INFO_OFFSET_HW_MODEL_REVISION,
|
||||
&revision);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
/* get GUID */
|
||||
err = snd_bebob_read_block(bebob->unit, INFO_OFFSET_GUID,
|
||||
data, sizeof(data));
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
strcpy(bebob->card->driver, "BeBoB");
|
||||
strcpy(bebob->card->shortname, model);
|
||||
strcpy(bebob->card->mixername, model);
|
||||
snprintf(bebob->card->longname, sizeof(bebob->card->longname),
|
||||
"%s %s (id:%d, rev:%d), GUID %08x%08x at %s, S%d",
|
||||
vendor, model, hw_id, revision,
|
||||
data[0], data[1], dev_name(&bebob->unit->device),
|
||||
100 << fw_dev->max_speed);
|
||||
end:
|
||||
return err;
|
||||
}
|
||||
|
||||
static void
|
||||
bebob_card_free(struct snd_card *card)
|
||||
{
|
||||
struct snd_bebob *bebob = card->private_data;
|
||||
|
||||
if (bebob->card_index >= 0) {
|
||||
mutex_lock(&devices_mutex);
|
||||
clear_bit(bebob->card_index, devices_used);
|
||||
mutex_unlock(&devices_mutex);
|
||||
}
|
||||
|
||||
mutex_destroy(&bebob->mutex);
|
||||
}
|
||||
|
||||
static const struct snd_bebob_spec *
|
||||
get_saffire_spec(struct fw_unit *unit)
|
||||
{
|
||||
char name[24] = {0};
|
||||
|
||||
if (fw_csr_string(unit->directory, CSR_MODEL, name, sizeof(name)) < 0)
|
||||
return NULL;
|
||||
|
||||
if (strcmp(name, "SaffireLE") == 0)
|
||||
return &saffire_le_spec;
|
||||
else
|
||||
return &saffire_spec;
|
||||
}
|
||||
|
||||
static bool
|
||||
check_audiophile_booted(struct fw_unit *unit)
|
||||
{
|
||||
char name[24] = {0};
|
||||
|
||||
if (fw_csr_string(unit->directory, CSR_MODEL, name, sizeof(name)) < 0)
|
||||
return false;
|
||||
|
||||
return strncmp(name, "FW Audiophile Bootloader", 15) != 0;
|
||||
}
|
||||
|
||||
static int
|
||||
bebob_probe(struct fw_unit *unit,
|
||||
const struct ieee1394_device_id *entry)
|
||||
{
|
||||
struct snd_card *card;
|
||||
struct snd_bebob *bebob;
|
||||
const struct snd_bebob_spec *spec;
|
||||
unsigned int card_index;
|
||||
int err;
|
||||
|
||||
mutex_lock(&devices_mutex);
|
||||
|
||||
for (card_index = 0; card_index < SNDRV_CARDS; card_index++) {
|
||||
if (!test_bit(card_index, devices_used) && enable[card_index])
|
||||
break;
|
||||
}
|
||||
if (card_index >= SNDRV_CARDS) {
|
||||
err = -ENOENT;
|
||||
goto end;
|
||||
}
|
||||
|
||||
if ((entry->vendor_id == VEN_FOCUSRITE) &&
|
||||
(entry->model_id == MODEL_FOCUSRITE_SAFFIRE_BOTH))
|
||||
spec = get_saffire_spec(unit);
|
||||
else if ((entry->vendor_id == VEN_MAUDIO1) &&
|
||||
(entry->model_id == MODEL_MAUDIO_AUDIOPHILE_BOTH) &&
|
||||
!check_audiophile_booted(unit))
|
||||
spec = NULL;
|
||||
else
|
||||
spec = (const struct snd_bebob_spec *)entry->driver_data;
|
||||
|
||||
if (spec == NULL) {
|
||||
if ((entry->vendor_id == VEN_MAUDIO1) ||
|
||||
(entry->vendor_id == VEN_MAUDIO2))
|
||||
err = snd_bebob_maudio_load_firmware(unit);
|
||||
else
|
||||
err = -ENOSYS;
|
||||
goto end;
|
||||
}
|
||||
|
||||
err = snd_card_new(&unit->device, index[card_index], id[card_index],
|
||||
THIS_MODULE, sizeof(struct snd_bebob), &card);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
bebob = card->private_data;
|
||||
bebob->card_index = card_index;
|
||||
set_bit(card_index, devices_used);
|
||||
card->private_free = bebob_card_free;
|
||||
|
||||
bebob->card = card;
|
||||
bebob->unit = unit;
|
||||
bebob->spec = spec;
|
||||
mutex_init(&bebob->mutex);
|
||||
spin_lock_init(&bebob->lock);
|
||||
init_waitqueue_head(&bebob->hwdep_wait);
|
||||
|
||||
err = name_device(bebob, entry->vendor_id);
|
||||
if (err < 0)
|
||||
goto error;
|
||||
|
||||
if ((entry->vendor_id == VEN_MAUDIO1) &&
|
||||
(entry->model_id == MODEL_MAUDIO_FW1814))
|
||||
err = snd_bebob_maudio_special_discover(bebob, true);
|
||||
else if ((entry->vendor_id == VEN_MAUDIO1) &&
|
||||
(entry->model_id == MODEL_MAUDIO_PROJECTMIX))
|
||||
err = snd_bebob_maudio_special_discover(bebob, false);
|
||||
else
|
||||
err = snd_bebob_stream_discover(bebob);
|
||||
if (err < 0)
|
||||
goto error;
|
||||
|
||||
snd_bebob_proc_init(bebob);
|
||||
|
||||
if ((bebob->midi_input_ports > 0) ||
|
||||
(bebob->midi_output_ports > 0)) {
|
||||
err = snd_bebob_create_midi_devices(bebob);
|
||||
if (err < 0)
|
||||
goto error;
|
||||
}
|
||||
|
||||
err = snd_bebob_create_pcm_devices(bebob);
|
||||
if (err < 0)
|
||||
goto error;
|
||||
|
||||
err = snd_bebob_create_hwdep_device(bebob);
|
||||
if (err < 0)
|
||||
goto error;
|
||||
|
||||
err = snd_bebob_stream_init_duplex(bebob);
|
||||
if (err < 0)
|
||||
goto error;
|
||||
|
||||
if (!bebob->maudio_special_quirk) {
|
||||
err = snd_card_register(card);
|
||||
if (err < 0) {
|
||||
snd_bebob_stream_destroy_duplex(bebob);
|
||||
goto error;
|
||||
}
|
||||
} else {
|
||||
/*
|
||||
* This is a workaround. This bus reset seems to have an effect
|
||||
* to make devices correctly handling transactions. Without
|
||||
* this, the devices have gap_count mismatch. This causes much
|
||||
* failure of transaction.
|
||||
*
|
||||
* Just after registration, user-land application receive
|
||||
* signals from dbus and starts I/Os. To avoid I/Os till the
|
||||
* future bus reset, registration is done in next update().
|
||||
*/
|
||||
bebob->deferred_registration = true;
|
||||
fw_schedule_bus_reset(fw_parent_device(bebob->unit)->card,
|
||||
false, true);
|
||||
}
|
||||
|
||||
dev_set_drvdata(&unit->device, bebob);
|
||||
end:
|
||||
mutex_unlock(&devices_mutex);
|
||||
return err;
|
||||
error:
|
||||
mutex_unlock(&devices_mutex);
|
||||
snd_card_free(card);
|
||||
return err;
|
||||
}
|
||||
|
||||
static void
|
||||
bebob_update(struct fw_unit *unit)
|
||||
{
|
||||
struct snd_bebob *bebob = dev_get_drvdata(&unit->device);
|
||||
|
||||
if (bebob == NULL)
|
||||
return;
|
||||
|
||||
fcp_bus_reset(bebob->unit);
|
||||
snd_bebob_stream_update_duplex(bebob);
|
||||
|
||||
if (bebob->deferred_registration) {
|
||||
if (snd_card_register(bebob->card) < 0) {
|
||||
snd_bebob_stream_destroy_duplex(bebob);
|
||||
snd_card_free(bebob->card);
|
||||
}
|
||||
bebob->deferred_registration = false;
|
||||
}
|
||||
}
|
||||
|
||||
static void bebob_remove(struct fw_unit *unit)
|
||||
{
|
||||
struct snd_bebob *bebob = dev_get_drvdata(&unit->device);
|
||||
|
||||
if (bebob == NULL)
|
||||
return;
|
||||
|
||||
kfree(bebob->maudio_special_quirk);
|
||||
|
||||
snd_bebob_stream_destroy_duplex(bebob);
|
||||
snd_card_disconnect(bebob->card);
|
||||
snd_card_free_when_closed(bebob->card);
|
||||
}
|
||||
|
||||
static struct snd_bebob_rate_spec normal_rate_spec = {
|
||||
.get = &snd_bebob_stream_get_rate,
|
||||
.set = &snd_bebob_stream_set_rate
|
||||
};
|
||||
static const struct snd_bebob_spec spec_normal = {
|
||||
.clock = NULL,
|
||||
.rate = &normal_rate_spec,
|
||||
.meter = NULL
|
||||
};
|
||||
|
||||
static const struct ieee1394_device_id bebob_id_table[] = {
|
||||
/* Edirol, FA-66 */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_EDIROL, 0x00010049, &spec_normal),
|
||||
/* Edirol, FA-101 */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_EDIROL, 0x00010048, &spec_normal),
|
||||
/* Presonus, FIREBOX */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_PRESONUS, 0x00010000, &spec_normal),
|
||||
/* PreSonus, FIREPOD/FP10 */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_PRESONUS, 0x00010066, &spec_normal),
|
||||
/* PreSonus, Inspire1394 */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_PRESONUS, 0x00010001, &spec_normal),
|
||||
/* BridgeCo, RDAudio1 */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_BRIDGECO, 0x00010048, &spec_normal),
|
||||
/* BridgeCo, Audio5 */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_BRIDGECO, 0x00010049, &spec_normal),
|
||||
/* Mackie, Onyx 1220/1620/1640 (Firewire I/O Card) */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_MACKIE, 0x00010065, &spec_normal),
|
||||
/* Mackie, d.2 (Firewire Option) */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_MACKIE, 0x00010067, &spec_normal),
|
||||
/* Stanton, ScratchAmp */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_STANTON, 0x00000001, &spec_normal),
|
||||
/* Tascam, IF-FW DM */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_TASCAM, 0x00010067, &spec_normal),
|
||||
/* Behringer, XENIX UFX 1204 */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_BEHRINGER, 0x00001204, &spec_normal),
|
||||
/* Behringer, XENIX UFX 1604 */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_BEHRINGER, 0x00001604, &spec_normal),
|
||||
/* Behringer, Digital Mixer X32 series (X-UF Card) */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_BEHRINGER, 0x00000006, &spec_normal),
|
||||
/* Apogee Electronics, Rosetta 200/400 (X-FireWire card) */
|
||||
/* Apogee Electronics, DA/AD/DD-16X (X-FireWire card) */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_APOGEE, 0x00010048, &spec_normal),
|
||||
/* Apogee Electronics, Ensemble */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_APOGEE, 0x00001eee, &spec_normal),
|
||||
/* ESI, Quatafire610 */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_ESI, 0x00010064, &spec_normal),
|
||||
/* AcousticReality, eARMasterOne */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_ACOUSTIC, 0x00000002, &spec_normal),
|
||||
/* CME, MatrixKFW */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_CME, 0x00030000, &spec_normal),
|
||||
/* Phonic, Helix Board 12 MkII */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_PHONIC, 0x00050000, &spec_normal),
|
||||
/* Phonic, Helix Board 18 MkII */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_PHONIC, 0x00060000, &spec_normal),
|
||||
/* Phonic, Helix Board 24 MkII */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_PHONIC, 0x00070000, &spec_normal),
|
||||
/* Phonic, Helix Board 12 Universal/18 Universal/24 Universal */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_PHONIC, 0x00000000, &spec_normal),
|
||||
/* Lynx, Aurora 8/16 (LT-FW) */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_LYNX, 0x00000001, &spec_normal),
|
||||
/* ICON, FireXon */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_ICON, 0x00000001, &spec_normal),
|
||||
/* PrismSound, Orpheus */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_PRISMSOUND, 0x00010048, &spec_normal),
|
||||
/* PrismSound, ADA-8XR */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_PRISMSOUND, 0x0000ada8, &spec_normal),
|
||||
/* TerraTec Electronic GmbH, PHASE 88 Rack FW */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_TERRATEC, 0x00000003, &phase88_rack_spec),
|
||||
/* TerraTec Electronic GmbH, PHASE 24 FW */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_TERRATEC, 0x00000004, &phase24_series_spec),
|
||||
/* TerraTec Electronic GmbH, Phase X24 FW */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_TERRATEC, 0x00000007, &phase24_series_spec),
|
||||
/* TerraTec Electronic GmbH, EWS MIC2/MIC8 */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_TERRATEC, 0x00000005, &spec_normal),
|
||||
/* Terratec Electronic GmbH, Aureon 7.1 Firewire */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_TERRATEC, 0x00000002, &spec_normal),
|
||||
/* Yamaha, GO44 */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_YAMAHA, 0x0010000b, &yamaha_go_spec),
|
||||
/* YAMAHA, GO46 */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_YAMAHA, 0x0010000c, &yamaha_go_spec),
|
||||
/* Focusrite, SaffirePro 26 I/O */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_FOCUSRITE, 0x00000003, &saffirepro_26_spec),
|
||||
/* Focusrite, SaffirePro 10 I/O */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_FOCUSRITE, 0x00000006, &saffirepro_10_spec),
|
||||
/* Focusrite, Saffire(no label and LE) */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_FOCUSRITE, MODEL_FOCUSRITE_SAFFIRE_BOTH,
|
||||
&saffire_spec),
|
||||
/* M-Audio, Firewire 410 */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_MAUDIO2, 0x00010058, NULL), /* bootloader */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_MAUDIO2, 0x00010046, &maudio_fw410_spec),
|
||||
/* M-Audio, Firewire Audiophile */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_MAUDIO1, MODEL_MAUDIO_AUDIOPHILE_BOTH,
|
||||
&maudio_audiophile_spec),
|
||||
/* M-Audio, Firewire Solo */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_MAUDIO1, 0x00010062, &maudio_solo_spec),
|
||||
/* M-Audio, Ozonic */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_MAUDIO1, 0x0000000a, &maudio_ozonic_spec),
|
||||
/* M-Audio NRV10 */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_MAUDIO1, 0x00010081, &maudio_nrv10_spec),
|
||||
/* M-Audio, ProFireLightbridge */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_MAUDIO1, 0x000100a1, &spec_normal),
|
||||
/* Firewire 1814 */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_MAUDIO1, 0x00010070, NULL), /* bootloader */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_MAUDIO1, MODEL_MAUDIO_FW1814,
|
||||
&maudio_special_spec),
|
||||
/* M-Audio ProjectMix */
|
||||
SND_BEBOB_DEV_ENTRY(VEN_MAUDIO1, MODEL_MAUDIO_PROJECTMIX,
|
||||
&maudio_special_spec),
|
||||
/* IDs are unknown but able to be supported */
|
||||
/* Apogee, Mini-ME Firewire */
|
||||
/* Apogee, Mini-DAC Firewire */
|
||||
/* Behringer, F-Control Audio 1616 */
|
||||
/* Behringer, F-Control Audio 610 */
|
||||
/* Cakawalk, Sonar Power Studio 66 */
|
||||
/* CME, UF400e */
|
||||
/* ESI, Quotafire XL */
|
||||
/* Infrasonic, DewX */
|
||||
/* Infrasonic, Windy6 */
|
||||
/* Mackie, Digital X Bus x.200 */
|
||||
/* Mackie, Digital X Bus x.400 */
|
||||
/* Phonic, HB 12 */
|
||||
/* Phonic, HB 24 */
|
||||
/* Phonic, HB 18 */
|
||||
/* Phonic, FireFly 202 */
|
||||
/* Phonic, FireFly 302 */
|
||||
/* Rolf Spuler, Firewire Guitar */
|
||||
{}
|
||||
};
|
||||
MODULE_DEVICE_TABLE(ieee1394, bebob_id_table);
|
||||
|
||||
static struct fw_driver bebob_driver = {
|
||||
.driver = {
|
||||
.owner = THIS_MODULE,
|
||||
.name = "snd-bebob",
|
||||
.bus = &fw_bus_type,
|
||||
},
|
||||
.probe = bebob_probe,
|
||||
.update = bebob_update,
|
||||
.remove = bebob_remove,
|
||||
.id_table = bebob_id_table,
|
||||
};
|
||||
|
||||
static int __init
|
||||
snd_bebob_init(void)
|
||||
{
|
||||
return driver_register(&bebob_driver.driver);
|
||||
}
|
||||
|
||||
static void __exit
|
||||
snd_bebob_exit(void)
|
||||
{
|
||||
driver_unregister(&bebob_driver.driver);
|
||||
}
|
||||
|
||||
module_init(snd_bebob_init);
|
||||
module_exit(snd_bebob_exit);
|
255
sound/firewire/bebob/bebob.h
Normal file
255
sound/firewire/bebob/bebob.h
Normal file
|
@ -0,0 +1,255 @@
|
|||
/*
|
||||
* bebob.h - a part of driver for BeBoB based devices
|
||||
*
|
||||
* Copyright (c) 2013-2014 Takashi Sakamoto
|
||||
*
|
||||
* Licensed under the terms of the GNU General Public License, version 2.
|
||||
*/
|
||||
|
||||
#ifndef SOUND_BEBOB_H_INCLUDED
|
||||
#define SOUND_BEBOB_H_INCLUDED
|
||||
|
||||
#include <linux/compat.h>
|
||||
#include <linux/device.h>
|
||||
#include <linux/firewire.h>
|
||||
#include <linux/firewire-constants.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/mod_devicetable.h>
|
||||
#include <linux/delay.h>
|
||||
#include <linux/slab.h>
|
||||
|
||||
#include <sound/core.h>
|
||||
#include <sound/initval.h>
|
||||
#include <sound/info.h>
|
||||
#include <sound/rawmidi.h>
|
||||
#include <sound/pcm.h>
|
||||
#include <sound/pcm_params.h>
|
||||
#include <sound/firewire.h>
|
||||
#include <sound/hwdep.h>
|
||||
|
||||
#include "../lib.h"
|
||||
#include "../fcp.h"
|
||||
#include "../packets-buffer.h"
|
||||
#include "../iso-resources.h"
|
||||
#include "../amdtp.h"
|
||||
#include "../cmp.h"
|
||||
|
||||
/* basic register addresses on DM1000/DM1100/DM1500 */
|
||||
#define BEBOB_ADDR_REG_INFO 0xffffc8020000ULL
|
||||
#define BEBOB_ADDR_REG_REQ 0xffffc8021000ULL
|
||||
|
||||
struct snd_bebob;
|
||||
|
||||
#define SND_BEBOB_STRM_FMT_ENTRIES 7
|
||||
struct snd_bebob_stream_formation {
|
||||
unsigned int pcm;
|
||||
unsigned int midi;
|
||||
};
|
||||
/* this is a lookup table for index of stream formations */
|
||||
extern const unsigned int snd_bebob_rate_table[SND_BEBOB_STRM_FMT_ENTRIES];
|
||||
|
||||
/* device specific operations */
|
||||
#define SND_BEBOB_CLOCK_INTERNAL "Internal"
|
||||
struct snd_bebob_clock_spec {
|
||||
unsigned int num;
|
||||
char *const *labels;
|
||||
int (*get)(struct snd_bebob *bebob, unsigned int *id);
|
||||
};
|
||||
struct snd_bebob_rate_spec {
|
||||
int (*get)(struct snd_bebob *bebob, unsigned int *rate);
|
||||
int (*set)(struct snd_bebob *bebob, unsigned int rate);
|
||||
};
|
||||
struct snd_bebob_meter_spec {
|
||||
unsigned int num;
|
||||
char *const *labels;
|
||||
int (*get)(struct snd_bebob *bebob, u32 *target, unsigned int size);
|
||||
};
|
||||
struct snd_bebob_spec {
|
||||
struct snd_bebob_clock_spec *clock;
|
||||
struct snd_bebob_rate_spec *rate;
|
||||
struct snd_bebob_meter_spec *meter;
|
||||
};
|
||||
|
||||
struct snd_bebob {
|
||||
struct snd_card *card;
|
||||
struct fw_unit *unit;
|
||||
int card_index;
|
||||
|
||||
struct mutex mutex;
|
||||
spinlock_t lock;
|
||||
|
||||
const struct snd_bebob_spec *spec;
|
||||
|
||||
unsigned int midi_input_ports;
|
||||
unsigned int midi_output_ports;
|
||||
|
||||
/* for bus reset quirk */
|
||||
struct completion bus_reset;
|
||||
bool connected;
|
||||
|
||||
struct amdtp_stream *master;
|
||||
struct amdtp_stream tx_stream;
|
||||
struct amdtp_stream rx_stream;
|
||||
struct cmp_connection out_conn;
|
||||
struct cmp_connection in_conn;
|
||||
atomic_t capture_substreams;
|
||||
atomic_t playback_substreams;
|
||||
|
||||
struct snd_bebob_stream_formation
|
||||
tx_stream_formations[SND_BEBOB_STRM_FMT_ENTRIES];
|
||||
struct snd_bebob_stream_formation
|
||||
rx_stream_formations[SND_BEBOB_STRM_FMT_ENTRIES];
|
||||
|
||||
int sync_input_plug;
|
||||
|
||||
/* for uapi */
|
||||
int dev_lock_count;
|
||||
bool dev_lock_changed;
|
||||
wait_queue_head_t hwdep_wait;
|
||||
|
||||
/* for M-Audio special devices */
|
||||
void *maudio_special_quirk;
|
||||
bool deferred_registration;
|
||||
};
|
||||
|
||||
static inline int
|
||||
snd_bebob_read_block(struct fw_unit *unit, u64 addr, void *buf, int size)
|
||||
{
|
||||
return snd_fw_transaction(unit, TCODE_READ_BLOCK_REQUEST,
|
||||
BEBOB_ADDR_REG_INFO + addr,
|
||||
buf, size, 0);
|
||||
}
|
||||
|
||||
static inline int
|
||||
snd_bebob_read_quad(struct fw_unit *unit, u64 addr, u32 *buf)
|
||||
{
|
||||
return snd_fw_transaction(unit, TCODE_READ_QUADLET_REQUEST,
|
||||
BEBOB_ADDR_REG_INFO + addr,
|
||||
(void *)buf, sizeof(u32), 0);
|
||||
}
|
||||
|
||||
/* AV/C Audio Subunit Specification 1.0 (Oct 2000, 1394TA) */
|
||||
int avc_audio_set_selector(struct fw_unit *unit, unsigned int subunit_id,
|
||||
unsigned int fb_id, unsigned int num);
|
||||
int avc_audio_get_selector(struct fw_unit *unit, unsigned int subunit_id,
|
||||
unsigned int fb_id, unsigned int *num);
|
||||
|
||||
/*
|
||||
* AVC command extensions, AV/C Unit and Subunit, Revision 17
|
||||
* (Nov 2003, BridgeCo)
|
||||
*/
|
||||
#define AVC_BRIDGECO_ADDR_BYTES 6
|
||||
enum avc_bridgeco_plug_dir {
|
||||
AVC_BRIDGECO_PLUG_DIR_IN = 0x00,
|
||||
AVC_BRIDGECO_PLUG_DIR_OUT = 0x01
|
||||
};
|
||||
enum avc_bridgeco_plug_mode {
|
||||
AVC_BRIDGECO_PLUG_MODE_UNIT = 0x00,
|
||||
AVC_BRIDGECO_PLUG_MODE_SUBUNIT = 0x01,
|
||||
AVC_BRIDGECO_PLUG_MODE_FUNCTION_BLOCK = 0x02
|
||||
};
|
||||
enum avc_bridgeco_plug_unit {
|
||||
AVC_BRIDGECO_PLUG_UNIT_ISOC = 0x00,
|
||||
AVC_BRIDGECO_PLUG_UNIT_EXT = 0x01,
|
||||
AVC_BRIDGECO_PLUG_UNIT_ASYNC = 0x02
|
||||
};
|
||||
enum avc_bridgeco_plug_type {
|
||||
AVC_BRIDGECO_PLUG_TYPE_ISOC = 0x00,
|
||||
AVC_BRIDGECO_PLUG_TYPE_ASYNC = 0x01,
|
||||
AVC_BRIDGECO_PLUG_TYPE_MIDI = 0x02,
|
||||
AVC_BRIDGECO_PLUG_TYPE_SYNC = 0x03,
|
||||
AVC_BRIDGECO_PLUG_TYPE_ANA = 0x04,
|
||||
AVC_BRIDGECO_PLUG_TYPE_DIG = 0x05
|
||||
};
|
||||
static inline void
|
||||
avc_bridgeco_fill_unit_addr(u8 buf[AVC_BRIDGECO_ADDR_BYTES],
|
||||
enum avc_bridgeco_plug_dir dir,
|
||||
enum avc_bridgeco_plug_unit unit,
|
||||
unsigned int pid)
|
||||
{
|
||||
buf[0] = 0xff; /* Unit */
|
||||
buf[1] = dir;
|
||||
buf[2] = AVC_BRIDGECO_PLUG_MODE_UNIT;
|
||||
buf[3] = unit;
|
||||
buf[4] = 0xff & pid;
|
||||
buf[5] = 0xff; /* reserved */
|
||||
}
|
||||
static inline void
|
||||
avc_bridgeco_fill_msu_addr(u8 buf[AVC_BRIDGECO_ADDR_BYTES],
|
||||
enum avc_bridgeco_plug_dir dir,
|
||||
unsigned int pid)
|
||||
{
|
||||
buf[0] = 0x60; /* Music subunit */
|
||||
buf[1] = dir;
|
||||
buf[2] = AVC_BRIDGECO_PLUG_MODE_SUBUNIT;
|
||||
buf[3] = 0xff & pid;
|
||||
buf[4] = 0xff; /* reserved */
|
||||
buf[5] = 0xff; /* reserved */
|
||||
}
|
||||
int avc_bridgeco_get_plug_ch_pos(struct fw_unit *unit,
|
||||
u8 addr[AVC_BRIDGECO_ADDR_BYTES],
|
||||
u8 *buf, unsigned int len);
|
||||
int avc_bridgeco_get_plug_type(struct fw_unit *unit,
|
||||
u8 addr[AVC_BRIDGECO_ADDR_BYTES],
|
||||
enum avc_bridgeco_plug_type *type);
|
||||
int avc_bridgeco_get_plug_section_type(struct fw_unit *unit,
|
||||
u8 addr[AVC_BRIDGECO_ADDR_BYTES],
|
||||
unsigned int id, u8 *type);
|
||||
int avc_bridgeco_get_plug_input(struct fw_unit *unit,
|
||||
u8 addr[AVC_BRIDGECO_ADDR_BYTES],
|
||||
u8 input[7]);
|
||||
int avc_bridgeco_get_plug_strm_fmt(struct fw_unit *unit,
|
||||
u8 addr[AVC_BRIDGECO_ADDR_BYTES], u8 *buf,
|
||||
unsigned int *len, unsigned int eid);
|
||||
|
||||
/* for AMDTP streaming */
|
||||
int snd_bebob_stream_get_rate(struct snd_bebob *bebob, unsigned int *rate);
|
||||
int snd_bebob_stream_set_rate(struct snd_bebob *bebob, unsigned int rate);
|
||||
int snd_bebob_stream_check_internal_clock(struct snd_bebob *bebob,
|
||||
bool *internal);
|
||||
int snd_bebob_stream_discover(struct snd_bebob *bebob);
|
||||
int snd_bebob_stream_init_duplex(struct snd_bebob *bebob);
|
||||
int snd_bebob_stream_start_duplex(struct snd_bebob *bebob, unsigned int rate);
|
||||
void snd_bebob_stream_stop_duplex(struct snd_bebob *bebob);
|
||||
void snd_bebob_stream_update_duplex(struct snd_bebob *bebob);
|
||||
void snd_bebob_stream_destroy_duplex(struct snd_bebob *bebob);
|
||||
|
||||
void snd_bebob_stream_lock_changed(struct snd_bebob *bebob);
|
||||
int snd_bebob_stream_lock_try(struct snd_bebob *bebob);
|
||||
void snd_bebob_stream_lock_release(struct snd_bebob *bebob);
|
||||
|
||||
void snd_bebob_proc_init(struct snd_bebob *bebob);
|
||||
|
||||
int snd_bebob_create_midi_devices(struct snd_bebob *bebob);
|
||||
|
||||
int snd_bebob_create_pcm_devices(struct snd_bebob *bebob);
|
||||
|
||||
int snd_bebob_create_hwdep_device(struct snd_bebob *bebob);
|
||||
|
||||
/* model specific operations */
|
||||
extern struct snd_bebob_spec phase88_rack_spec;
|
||||
extern struct snd_bebob_spec phase24_series_spec;
|
||||
extern struct snd_bebob_spec yamaha_go_spec;
|
||||
extern struct snd_bebob_spec saffirepro_26_spec;
|
||||
extern struct snd_bebob_spec saffirepro_10_spec;
|
||||
extern struct snd_bebob_spec saffire_le_spec;
|
||||
extern struct snd_bebob_spec saffire_spec;
|
||||
extern struct snd_bebob_spec maudio_fw410_spec;
|
||||
extern struct snd_bebob_spec maudio_audiophile_spec;
|
||||
extern struct snd_bebob_spec maudio_solo_spec;
|
||||
extern struct snd_bebob_spec maudio_ozonic_spec;
|
||||
extern struct snd_bebob_spec maudio_nrv10_spec;
|
||||
extern struct snd_bebob_spec maudio_special_spec;
|
||||
int snd_bebob_maudio_special_discover(struct snd_bebob *bebob, bool is1814);
|
||||
int snd_bebob_maudio_load_firmware(struct fw_unit *unit);
|
||||
|
||||
#define SND_BEBOB_DEV_ENTRY(vendor, model, data) \
|
||||
{ \
|
||||
.match_flags = IEEE1394_MATCH_VENDOR_ID | \
|
||||
IEEE1394_MATCH_MODEL_ID, \
|
||||
.vendor_id = vendor, \
|
||||
.model_id = model, \
|
||||
.driver_data = (kernel_ulong_t)data \
|
||||
}
|
||||
|
||||
#endif
|
282
sound/firewire/bebob/bebob_command.c
Normal file
282
sound/firewire/bebob/bebob_command.c
Normal file
|
@ -0,0 +1,282 @@
|
|||
/*
|
||||
* bebob_command.c - driver for BeBoB based devices
|
||||
*
|
||||
* Copyright (c) 2013-2014 Takashi Sakamoto
|
||||
*
|
||||
* Licensed under the terms of the GNU General Public License, version 2.
|
||||
*/
|
||||
|
||||
#include "./bebob.h"
|
||||
|
||||
int avc_audio_set_selector(struct fw_unit *unit, unsigned int subunit_id,
|
||||
unsigned int fb_id, unsigned int num)
|
||||
{
|
||||
u8 *buf;
|
||||
int err;
|
||||
|
||||
buf = kzalloc(12, GFP_KERNEL);
|
||||
if (buf == NULL)
|
||||
return -ENOMEM;
|
||||
|
||||
buf[0] = 0x00; /* AV/C CONTROL */
|
||||
buf[1] = 0x08 | (0x07 & subunit_id); /* AUDIO SUBUNIT ID */
|
||||
buf[2] = 0xb8; /* FUNCTION BLOCK */
|
||||
buf[3] = 0x80; /* type is 'selector'*/
|
||||
buf[4] = 0xff & fb_id; /* function block id */
|
||||
buf[5] = 0x10; /* control attribute is CURRENT */
|
||||
buf[6] = 0x02; /* selector length is 2 */
|
||||
buf[7] = 0xff & num; /* input function block plug number */
|
||||
buf[8] = 0x01; /* control selector is SELECTOR_CONTROL */
|
||||
|
||||
err = fcp_avc_transaction(unit, buf, 12, buf, 12,
|
||||
BIT(1) | BIT(2) | BIT(3) | BIT(4) | BIT(5) |
|
||||
BIT(6) | BIT(7) | BIT(8));
|
||||
if (err > 0 && err < 9)
|
||||
err = -EIO;
|
||||
else if (buf[0] == 0x08) /* NOT IMPLEMENTED */
|
||||
err = -ENOSYS;
|
||||
else if (buf[0] == 0x0a) /* REJECTED */
|
||||
err = -EINVAL;
|
||||
else if (err > 0)
|
||||
err = 0;
|
||||
|
||||
kfree(buf);
|
||||
return err;
|
||||
}
|
||||
|
||||
int avc_audio_get_selector(struct fw_unit *unit, unsigned int subunit_id,
|
||||
unsigned int fb_id, unsigned int *num)
|
||||
{
|
||||
u8 *buf;
|
||||
int err;
|
||||
|
||||
buf = kzalloc(12, GFP_KERNEL);
|
||||
if (buf == NULL)
|
||||
return -ENOMEM;
|
||||
|
||||
buf[0] = 0x01; /* AV/C STATUS */
|
||||
buf[1] = 0x08 | (0x07 & subunit_id); /* AUDIO SUBUNIT ID */
|
||||
buf[2] = 0xb8; /* FUNCTION BLOCK */
|
||||
buf[3] = 0x80; /* type is 'selector'*/
|
||||
buf[4] = 0xff & fb_id; /* function block id */
|
||||
buf[5] = 0x10; /* control attribute is CURRENT */
|
||||
buf[6] = 0x02; /* selector length is 2 */
|
||||
buf[7] = 0xff; /* input function block plug number */
|
||||
buf[8] = 0x01; /* control selector is SELECTOR_CONTROL */
|
||||
|
||||
err = fcp_avc_transaction(unit, buf, 12, buf, 12,
|
||||
BIT(1) | BIT(2) | BIT(3) | BIT(4) | BIT(5) |
|
||||
BIT(6) | BIT(8));
|
||||
if (err > 0 && err < 9)
|
||||
err = -EIO;
|
||||
else if (buf[0] == 0x08) /* NOT IMPLEMENTED */
|
||||
err = -ENOSYS;
|
||||
else if (buf[0] == 0x0a) /* REJECTED */
|
||||
err = -EINVAL;
|
||||
else if (buf[0] == 0x0b) /* IN TRANSITION */
|
||||
err = -EAGAIN;
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
*num = buf[7];
|
||||
err = 0;
|
||||
end:
|
||||
kfree(buf);
|
||||
return err;
|
||||
}
|
||||
|
||||
static inline void
|
||||
avc_bridgeco_fill_extension_addr(u8 *buf, u8 *addr)
|
||||
{
|
||||
buf[1] = addr[0];
|
||||
memcpy(buf + 4, addr + 1, 5);
|
||||
}
|
||||
|
||||
static inline void
|
||||
avc_bridgeco_fill_plug_info_extension_command(u8 *buf, u8 *addr,
|
||||
unsigned int itype)
|
||||
{
|
||||
buf[0] = 0x01; /* AV/C STATUS */
|
||||
buf[2] = 0x02; /* AV/C GENERAL PLUG INFO */
|
||||
buf[3] = 0xc0; /* BridgeCo extension */
|
||||
avc_bridgeco_fill_extension_addr(buf, addr);
|
||||
buf[9] = itype; /* info type */
|
||||
}
|
||||
|
||||
int avc_bridgeco_get_plug_type(struct fw_unit *unit,
|
||||
u8 addr[AVC_BRIDGECO_ADDR_BYTES],
|
||||
enum avc_bridgeco_plug_type *type)
|
||||
{
|
||||
u8 *buf;
|
||||
int err;
|
||||
|
||||
buf = kzalloc(12, GFP_KERNEL);
|
||||
if (buf == NULL)
|
||||
return -ENOMEM;
|
||||
|
||||
/* Info type is 'plug type'. */
|
||||
avc_bridgeco_fill_plug_info_extension_command(buf, addr, 0x00);
|
||||
|
||||
err = fcp_avc_transaction(unit, buf, 12, buf, 12,
|
||||
BIT(1) | BIT(2) | BIT(3) | BIT(4) | BIT(5) |
|
||||
BIT(6) | BIT(7) | BIT(9));
|
||||
if ((err >= 0) && (err < 8))
|
||||
err = -EIO;
|
||||
else if (buf[0] == 0x08) /* NOT IMPLEMENTED */
|
||||
err = -ENOSYS;
|
||||
else if (buf[0] == 0x0a) /* REJECTED */
|
||||
err = -EINVAL;
|
||||
else if (buf[0] == 0x0b) /* IN TRANSITION */
|
||||
err = -EAGAIN;
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
*type = buf[10];
|
||||
err = 0;
|
||||
end:
|
||||
kfree(buf);
|
||||
return err;
|
||||
}
|
||||
|
||||
int avc_bridgeco_get_plug_ch_pos(struct fw_unit *unit,
|
||||
u8 addr[AVC_BRIDGECO_ADDR_BYTES],
|
||||
u8 *buf, unsigned int len)
|
||||
{
|
||||
int err;
|
||||
|
||||
/* Info type is 'channel position'. */
|
||||
avc_bridgeco_fill_plug_info_extension_command(buf, addr, 0x03);
|
||||
|
||||
err = fcp_avc_transaction(unit, buf, 12, buf, 256,
|
||||
BIT(1) | BIT(2) | BIT(3) | BIT(4) |
|
||||
BIT(5) | BIT(6) | BIT(7) | BIT(9));
|
||||
if ((err >= 0) && (err < 8))
|
||||
err = -EIO;
|
||||
else if (buf[0] == 0x08) /* NOT IMPLEMENTED */
|
||||
err = -ENOSYS;
|
||||
else if (buf[0] == 0x0a) /* REJECTED */
|
||||
err = -EINVAL;
|
||||
else if (buf[0] == 0x0b) /* IN TRANSITION */
|
||||
err = -EAGAIN;
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
/* Pick up specific data. */
|
||||
memmove(buf, buf + 10, err - 10);
|
||||
err = 0;
|
||||
end:
|
||||
return err;
|
||||
}
|
||||
|
||||
int avc_bridgeco_get_plug_section_type(struct fw_unit *unit,
|
||||
u8 addr[AVC_BRIDGECO_ADDR_BYTES],
|
||||
unsigned int id, u8 *type)
|
||||
{
|
||||
u8 *buf;
|
||||
int err;
|
||||
|
||||
/* section info includes charactors but this module don't need it */
|
||||
buf = kzalloc(12, GFP_KERNEL);
|
||||
if (buf == NULL)
|
||||
return -ENOMEM;
|
||||
|
||||
/* Info type is 'section info'. */
|
||||
avc_bridgeco_fill_plug_info_extension_command(buf, addr, 0x07);
|
||||
buf[10] = 0xff & ++id; /* section id */
|
||||
|
||||
err = fcp_avc_transaction(unit, buf, 12, buf, 12,
|
||||
BIT(1) | BIT(2) | BIT(3) | BIT(4) | BIT(5) |
|
||||
BIT(6) | BIT(7) | BIT(9) | BIT(10));
|
||||
if ((err >= 0) && (err < 8))
|
||||
err = -EIO;
|
||||
else if (buf[0] == 0x08) /* NOT IMPLEMENTED */
|
||||
err = -ENOSYS;
|
||||
else if (buf[0] == 0x0a) /* REJECTED */
|
||||
err = -EINVAL;
|
||||
else if (buf[0] == 0x0b) /* IN TRANSITION */
|
||||
err = -EAGAIN;
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
*type = buf[11];
|
||||
err = 0;
|
||||
end:
|
||||
kfree(buf);
|
||||
return err;
|
||||
}
|
||||
|
||||
int avc_bridgeco_get_plug_input(struct fw_unit *unit,
|
||||
u8 addr[AVC_BRIDGECO_ADDR_BYTES], u8 input[7])
|
||||
{
|
||||
int err;
|
||||
u8 *buf;
|
||||
|
||||
buf = kzalloc(18, GFP_KERNEL);
|
||||
if (buf == NULL)
|
||||
return -ENOMEM;
|
||||
|
||||
/* Info type is 'plug input'. */
|
||||
avc_bridgeco_fill_plug_info_extension_command(buf, addr, 0x05);
|
||||
|
||||
err = fcp_avc_transaction(unit, buf, 16, buf, 16,
|
||||
BIT(1) | BIT(2) | BIT(3) | BIT(4) | BIT(5) |
|
||||
BIT(6) | BIT(7));
|
||||
if ((err >= 0) && (err < 8))
|
||||
err = -EIO;
|
||||
else if (buf[0] == 0x08) /* NOT IMPLEMENTED */
|
||||
err = -ENOSYS;
|
||||
else if (buf[0] == 0x0a) /* REJECTED */
|
||||
err = -EINVAL;
|
||||
else if (buf[0] == 0x0b) /* IN TRANSITION */
|
||||
err = -EAGAIN;
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
memcpy(input, buf + 10, 5);
|
||||
err = 0;
|
||||
end:
|
||||
kfree(buf);
|
||||
return err;
|
||||
}
|
||||
|
||||
int avc_bridgeco_get_plug_strm_fmt(struct fw_unit *unit,
|
||||
u8 addr[AVC_BRIDGECO_ADDR_BYTES], u8 *buf,
|
||||
unsigned int *len, unsigned int eid)
|
||||
{
|
||||
int err;
|
||||
|
||||
/* check given buffer */
|
||||
if ((buf == NULL) || (*len < 12)) {
|
||||
err = -EINVAL;
|
||||
goto end;
|
||||
}
|
||||
|
||||
buf[0] = 0x01; /* AV/C STATUS */
|
||||
buf[2] = 0x2f; /* AV/C STREAM FORMAT SUPPORT */
|
||||
buf[3] = 0xc1; /* Bridgeco extension - List Request */
|
||||
avc_bridgeco_fill_extension_addr(buf, addr);
|
||||
buf[10] = 0xff & eid; /* Entry ID */
|
||||
|
||||
err = fcp_avc_transaction(unit, buf, 12, buf, *len,
|
||||
BIT(1) | BIT(2) | BIT(3) | BIT(4) | BIT(5) |
|
||||
BIT(6) | BIT(7) | BIT(10));
|
||||
if ((err >= 0) && (err < 12))
|
||||
err = -EIO;
|
||||
else if (buf[0] == 0x08) /* NOT IMPLEMENTED */
|
||||
err = -ENOSYS;
|
||||
else if (buf[0] == 0x0a) /* REJECTED */
|
||||
err = -EINVAL;
|
||||
else if (buf[0] == 0x0b) /* IN TRANSITION */
|
||||
err = -EAGAIN;
|
||||
else if (buf[10] != eid)
|
||||
err = -EIO;
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
/* Pick up 'stream format info'. */
|
||||
memmove(buf, buf + 11, err - 11);
|
||||
*len = err - 11;
|
||||
err = 0;
|
||||
end:
|
||||
return err;
|
||||
}
|
313
sound/firewire/bebob/bebob_focusrite.c
Normal file
313
sound/firewire/bebob/bebob_focusrite.c
Normal file
|
@ -0,0 +1,313 @@
|
|||
/*
|
||||
* bebob_focusrite.c - a part of driver for BeBoB based devices
|
||||
*
|
||||
* Copyright (c) 2013-2014 Takashi Sakamoto
|
||||
*
|
||||
* Licensed under the terms of the GNU General Public License, version 2.
|
||||
*/
|
||||
|
||||
#include "./bebob.h"
|
||||
|
||||
#define ANA_IN "Analog In"
|
||||
#define DIG_IN "Digital In"
|
||||
#define ANA_OUT "Analog Out"
|
||||
#define DIG_OUT "Digital Out"
|
||||
#define STM_IN "Stream In"
|
||||
|
||||
#define SAFFIRE_ADDRESS_BASE 0x000100000000ULL
|
||||
|
||||
#define SAFFIRE_OFFSET_CLOCK_SOURCE 0x00f8
|
||||
#define SAFFIREPRO_OFFSET_CLOCK_SOURCE 0x0174
|
||||
|
||||
/* whether sync to external device or not */
|
||||
#define SAFFIRE_OFFSET_CLOCK_SYNC_EXT 0x013c
|
||||
#define SAFFIRE_LE_OFFSET_CLOCK_SYNC_EXT 0x0432
|
||||
#define SAFFIREPRO_OFFSET_CLOCK_SYNC_EXT 0x0164
|
||||
|
||||
#define SAFFIRE_CLOCK_SOURCE_INTERNAL 0
|
||||
#define SAFFIRE_CLOCK_SOURCE_SPDIF 1
|
||||
|
||||
/* clock sources as returned from register of Saffire Pro 10 and 26 */
|
||||
#define SAFFIREPRO_CLOCK_SOURCE_INTERNAL 0
|
||||
#define SAFFIREPRO_CLOCK_SOURCE_SKIP 1 /* never used on hardware */
|
||||
#define SAFFIREPRO_CLOCK_SOURCE_SPDIF 2
|
||||
#define SAFFIREPRO_CLOCK_SOURCE_ADAT1 3 /* not used on s.pro. 10 */
|
||||
#define SAFFIREPRO_CLOCK_SOURCE_ADAT2 4 /* not used on s.pro. 10 */
|
||||
#define SAFFIREPRO_CLOCK_SOURCE_WORDCLOCK 5
|
||||
#define SAFFIREPRO_CLOCK_SOURCE_COUNT 6
|
||||
|
||||
/* S/PDIF, ADAT1, ADAT2 is enabled or not. three quadlets */
|
||||
#define SAFFIREPRO_ENABLE_DIG_IFACES 0x01a4
|
||||
|
||||
/* saffirepro has its own parameter for sampling frequency */
|
||||
#define SAFFIREPRO_RATE_NOREBOOT 0x01cc
|
||||
/* index is the value for this register */
|
||||
static const unsigned int rates[] = {
|
||||
[0] = 0,
|
||||
[1] = 44100,
|
||||
[2] = 48000,
|
||||
[3] = 88200,
|
||||
[4] = 96000,
|
||||
[5] = 176400,
|
||||
[6] = 192000
|
||||
};
|
||||
|
||||
/* saffire(no label)/saffire LE has metering */
|
||||
#define SAFFIRE_OFFSET_METER 0x0100
|
||||
#define SAFFIRE_LE_OFFSET_METER 0x0168
|
||||
|
||||
static inline int
|
||||
saffire_read_block(struct snd_bebob *bebob, u64 offset,
|
||||
u32 *buf, unsigned int size)
|
||||
{
|
||||
unsigned int i;
|
||||
int err;
|
||||
__be32 *tmp = (__be32 *)buf;
|
||||
|
||||
err = snd_fw_transaction(bebob->unit, TCODE_READ_BLOCK_REQUEST,
|
||||
SAFFIRE_ADDRESS_BASE + offset,
|
||||
tmp, size, 0);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
for (i = 0; i < size / sizeof(u32); i++)
|
||||
buf[i] = be32_to_cpu(tmp[i]);
|
||||
end:
|
||||
return err;
|
||||
}
|
||||
|
||||
static inline int
|
||||
saffire_read_quad(struct snd_bebob *bebob, u64 offset, u32 *value)
|
||||
{
|
||||
int err;
|
||||
__be32 tmp;
|
||||
|
||||
err = snd_fw_transaction(bebob->unit, TCODE_READ_QUADLET_REQUEST,
|
||||
SAFFIRE_ADDRESS_BASE + offset,
|
||||
&tmp, sizeof(__be32), 0);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
*value = be32_to_cpu(tmp);
|
||||
end:
|
||||
return err;
|
||||
}
|
||||
|
||||
static inline int
|
||||
saffire_write_quad(struct snd_bebob *bebob, u64 offset, u32 value)
|
||||
{
|
||||
__be32 data = cpu_to_be32(value);
|
||||
|
||||
return snd_fw_transaction(bebob->unit, TCODE_WRITE_QUADLET_REQUEST,
|
||||
SAFFIRE_ADDRESS_BASE + offset,
|
||||
&data, sizeof(__be32), 0);
|
||||
}
|
||||
|
||||
static char *const saffirepro_10_clk_src_labels[] = {
|
||||
SND_BEBOB_CLOCK_INTERNAL, "S/PDIF", "Word Clock"
|
||||
};
|
||||
static char *const saffirepro_26_clk_src_labels[] = {
|
||||
SND_BEBOB_CLOCK_INTERNAL, "S/PDIF", "ADAT1", "ADAT2", "Word Clock"
|
||||
};
|
||||
/* Value maps between registers and labels for SaffirePro 10/26. */
|
||||
static const signed char saffirepro_clk_maps[][SAFFIREPRO_CLOCK_SOURCE_COUNT] = {
|
||||
/* SaffirePro 10 */
|
||||
[0] = {
|
||||
[SAFFIREPRO_CLOCK_SOURCE_INTERNAL] = 0,
|
||||
[SAFFIREPRO_CLOCK_SOURCE_SKIP] = -1, /* not supported */
|
||||
[SAFFIREPRO_CLOCK_SOURCE_SPDIF] = 1,
|
||||
[SAFFIREPRO_CLOCK_SOURCE_ADAT1] = -1, /* not supported */
|
||||
[SAFFIREPRO_CLOCK_SOURCE_ADAT2] = -1, /* not supported */
|
||||
[SAFFIREPRO_CLOCK_SOURCE_WORDCLOCK] = 2,
|
||||
},
|
||||
/* SaffirePro 26 */
|
||||
[1] = {
|
||||
[SAFFIREPRO_CLOCK_SOURCE_INTERNAL] = 0,
|
||||
[SAFFIREPRO_CLOCK_SOURCE_SKIP] = -1, /* not supported */
|
||||
[SAFFIREPRO_CLOCK_SOURCE_SPDIF] = 1,
|
||||
[SAFFIREPRO_CLOCK_SOURCE_ADAT1] = 2,
|
||||
[SAFFIREPRO_CLOCK_SOURCE_ADAT2] = 3,
|
||||
[SAFFIREPRO_CLOCK_SOURCE_WORDCLOCK] = 4,
|
||||
}
|
||||
};
|
||||
|
||||
static int
|
||||
saffirepro_both_clk_freq_get(struct snd_bebob *bebob, unsigned int *rate)
|
||||
{
|
||||
u32 id;
|
||||
int err;
|
||||
|
||||
err = saffire_read_quad(bebob, SAFFIREPRO_RATE_NOREBOOT, &id);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
if (id >= ARRAY_SIZE(rates))
|
||||
err = -EIO;
|
||||
else
|
||||
*rate = rates[id];
|
||||
end:
|
||||
return err;
|
||||
}
|
||||
static int
|
||||
saffirepro_both_clk_freq_set(struct snd_bebob *bebob, unsigned int rate)
|
||||
{
|
||||
u32 id;
|
||||
|
||||
for (id = 0; id < ARRAY_SIZE(rates); id++) {
|
||||
if (rates[id] == rate)
|
||||
break;
|
||||
}
|
||||
if (id == ARRAY_SIZE(rates))
|
||||
return -EINVAL;
|
||||
|
||||
return saffire_write_quad(bebob, SAFFIREPRO_RATE_NOREBOOT, id);
|
||||
}
|
||||
|
||||
/*
|
||||
* query hardware for current clock source, return our internally
|
||||
* used clock index in *id, depending on hardware.
|
||||
*/
|
||||
static int
|
||||
saffirepro_both_clk_src_get(struct snd_bebob *bebob, unsigned int *id)
|
||||
{
|
||||
int err;
|
||||
u32 value; /* clock source read from hw register */
|
||||
const signed char *map;
|
||||
|
||||
err = saffire_read_quad(bebob, SAFFIREPRO_OFFSET_CLOCK_SOURCE, &value);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
/* depending on hardware, use a different mapping */
|
||||
if (bebob->spec->clock->labels == saffirepro_10_clk_src_labels)
|
||||
map = saffirepro_clk_maps[0];
|
||||
else
|
||||
map = saffirepro_clk_maps[1];
|
||||
|
||||
/* In a case that this driver cannot handle the value of register. */
|
||||
if (value >= SAFFIREPRO_CLOCK_SOURCE_COUNT || map[value] < 0) {
|
||||
err = -EIO;
|
||||
goto end;
|
||||
}
|
||||
|
||||
*id = (unsigned int)map[value];
|
||||
end:
|
||||
return err;
|
||||
}
|
||||
|
||||
struct snd_bebob_spec saffire_le_spec;
|
||||
static char *const saffire_both_clk_src_labels[] = {
|
||||
SND_BEBOB_CLOCK_INTERNAL, "S/PDIF"
|
||||
};
|
||||
static int
|
||||
saffire_both_clk_src_get(struct snd_bebob *bebob, unsigned int *id)
|
||||
{
|
||||
int err;
|
||||
u32 value;
|
||||
|
||||
err = saffire_read_quad(bebob, SAFFIRE_OFFSET_CLOCK_SOURCE, &value);
|
||||
if (err >= 0)
|
||||
*id = 0xff & value;
|
||||
|
||||
return err;
|
||||
};
|
||||
static char *const saffire_le_meter_labels[] = {
|
||||
ANA_IN, ANA_IN, DIG_IN,
|
||||
ANA_OUT, ANA_OUT, ANA_OUT, ANA_OUT,
|
||||
STM_IN, STM_IN
|
||||
};
|
||||
static char *const saffire_meter_labels[] = {
|
||||
ANA_IN, ANA_IN,
|
||||
STM_IN, STM_IN, STM_IN, STM_IN, STM_IN,
|
||||
};
|
||||
static int
|
||||
saffire_meter_get(struct snd_bebob *bebob, u32 *buf, unsigned int size)
|
||||
{
|
||||
struct snd_bebob_meter_spec *spec = bebob->spec->meter;
|
||||
unsigned int channels;
|
||||
u64 offset;
|
||||
int err;
|
||||
|
||||
if (spec->labels == saffire_le_meter_labels)
|
||||
offset = SAFFIRE_LE_OFFSET_METER;
|
||||
else
|
||||
offset = SAFFIRE_OFFSET_METER;
|
||||
|
||||
channels = spec->num * 2;
|
||||
if (size < channels * sizeof(u32))
|
||||
return -EIO;
|
||||
|
||||
err = saffire_read_block(bebob, offset, buf, size);
|
||||
if (err >= 0 && spec->labels == saffire_le_meter_labels) {
|
||||
swap(buf[1], buf[3]);
|
||||
swap(buf[2], buf[3]);
|
||||
swap(buf[3], buf[4]);
|
||||
|
||||
swap(buf[7], buf[10]);
|
||||
swap(buf[8], buf[10]);
|
||||
swap(buf[9], buf[11]);
|
||||
swap(buf[11], buf[12]);
|
||||
|
||||
swap(buf[15], buf[16]);
|
||||
}
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
static struct snd_bebob_rate_spec saffirepro_both_rate_spec = {
|
||||
.get = &saffirepro_both_clk_freq_get,
|
||||
.set = &saffirepro_both_clk_freq_set,
|
||||
};
|
||||
/* Saffire Pro 26 I/O */
|
||||
static struct snd_bebob_clock_spec saffirepro_26_clk_spec = {
|
||||
.num = ARRAY_SIZE(saffirepro_26_clk_src_labels),
|
||||
.labels = saffirepro_26_clk_src_labels,
|
||||
.get = &saffirepro_both_clk_src_get,
|
||||
};
|
||||
struct snd_bebob_spec saffirepro_26_spec = {
|
||||
.clock = &saffirepro_26_clk_spec,
|
||||
.rate = &saffirepro_both_rate_spec,
|
||||
.meter = NULL
|
||||
};
|
||||
/* Saffire Pro 10 I/O */
|
||||
static struct snd_bebob_clock_spec saffirepro_10_clk_spec = {
|
||||
.num = ARRAY_SIZE(saffirepro_10_clk_src_labels),
|
||||
.labels = saffirepro_10_clk_src_labels,
|
||||
.get = &saffirepro_both_clk_src_get,
|
||||
};
|
||||
struct snd_bebob_spec saffirepro_10_spec = {
|
||||
.clock = &saffirepro_10_clk_spec,
|
||||
.rate = &saffirepro_both_rate_spec,
|
||||
.meter = NULL
|
||||
};
|
||||
|
||||
static struct snd_bebob_rate_spec saffire_both_rate_spec = {
|
||||
.get = &snd_bebob_stream_get_rate,
|
||||
.set = &snd_bebob_stream_set_rate,
|
||||
};
|
||||
static struct snd_bebob_clock_spec saffire_both_clk_spec = {
|
||||
.num = ARRAY_SIZE(saffire_both_clk_src_labels),
|
||||
.labels = saffire_both_clk_src_labels,
|
||||
.get = &saffire_both_clk_src_get,
|
||||
};
|
||||
/* Saffire LE */
|
||||
static struct snd_bebob_meter_spec saffire_le_meter_spec = {
|
||||
.num = ARRAY_SIZE(saffire_le_meter_labels),
|
||||
.labels = saffire_le_meter_labels,
|
||||
.get = &saffire_meter_get,
|
||||
};
|
||||
struct snd_bebob_spec saffire_le_spec = {
|
||||
.clock = &saffire_both_clk_spec,
|
||||
.rate = &saffire_both_rate_spec,
|
||||
.meter = &saffire_le_meter_spec
|
||||
};
|
||||
/* Saffire */
|
||||
static struct snd_bebob_meter_spec saffire_meter_spec = {
|
||||
.num = ARRAY_SIZE(saffire_meter_labels),
|
||||
.labels = saffire_meter_labels,
|
||||
.get = &saffire_meter_get,
|
||||
};
|
||||
struct snd_bebob_spec saffire_spec = {
|
||||
.clock = &saffire_both_clk_spec,
|
||||
.rate = &saffire_both_rate_spec,
|
||||
.meter = &saffire_meter_spec
|
||||
};
|
199
sound/firewire/bebob/bebob_hwdep.c
Normal file
199
sound/firewire/bebob/bebob_hwdep.c
Normal file
|
@ -0,0 +1,199 @@
|
|||
/*
|
||||
* bebob_hwdep.c - a part of driver for BeBoB based devices
|
||||
*
|
||||
* Copyright (c) 2013-2014 Takashi Sakamoto
|
||||
*
|
||||
* Licensed under the terms of the GNU General Public License, version 2.
|
||||
*/
|
||||
|
||||
/*
|
||||
* This codes give three functionality.
|
||||
*
|
||||
* 1.get firewire node infomation
|
||||
* 2.get notification about starting/stopping stream
|
||||
* 3.lock/unlock stream
|
||||
*/
|
||||
|
||||
#include "bebob.h"
|
||||
|
||||
static long
|
||||
hwdep_read(struct snd_hwdep *hwdep, char __user *buf, long count,
|
||||
loff_t *offset)
|
||||
{
|
||||
struct snd_bebob *bebob = hwdep->private_data;
|
||||
DEFINE_WAIT(wait);
|
||||
union snd_firewire_event event;
|
||||
|
||||
spin_lock_irq(&bebob->lock);
|
||||
|
||||
while (!bebob->dev_lock_changed) {
|
||||
prepare_to_wait(&bebob->hwdep_wait, &wait, TASK_INTERRUPTIBLE);
|
||||
spin_unlock_irq(&bebob->lock);
|
||||
schedule();
|
||||
finish_wait(&bebob->hwdep_wait, &wait);
|
||||
if (signal_pending(current))
|
||||
return -ERESTARTSYS;
|
||||
spin_lock_irq(&bebob->lock);
|
||||
}
|
||||
|
||||
memset(&event, 0, sizeof(event));
|
||||
if (bebob->dev_lock_changed) {
|
||||
event.lock_status.type = SNDRV_FIREWIRE_EVENT_LOCK_STATUS;
|
||||
event.lock_status.status = (bebob->dev_lock_count > 0);
|
||||
bebob->dev_lock_changed = false;
|
||||
|
||||
count = min_t(long, count, sizeof(event.lock_status));
|
||||
}
|
||||
|
||||
spin_unlock_irq(&bebob->lock);
|
||||
|
||||
if (copy_to_user(buf, &event, count))
|
||||
return -EFAULT;
|
||||
|
||||
return count;
|
||||
}
|
||||
|
||||
static unsigned int
|
||||
hwdep_poll(struct snd_hwdep *hwdep, struct file *file, poll_table *wait)
|
||||
{
|
||||
struct snd_bebob *bebob = hwdep->private_data;
|
||||
unsigned int events;
|
||||
|
||||
poll_wait(file, &bebob->hwdep_wait, wait);
|
||||
|
||||
spin_lock_irq(&bebob->lock);
|
||||
if (bebob->dev_lock_changed)
|
||||
events = POLLIN | POLLRDNORM;
|
||||
else
|
||||
events = 0;
|
||||
spin_unlock_irq(&bebob->lock);
|
||||
|
||||
return events;
|
||||
}
|
||||
|
||||
static int
|
||||
hwdep_get_info(struct snd_bebob *bebob, void __user *arg)
|
||||
{
|
||||
struct fw_device *dev = fw_parent_device(bebob->unit);
|
||||
struct snd_firewire_get_info info;
|
||||
|
||||
memset(&info, 0, sizeof(info));
|
||||
info.type = SNDRV_FIREWIRE_TYPE_BEBOB;
|
||||
info.card = dev->card->index;
|
||||
*(__be32 *)&info.guid[0] = cpu_to_be32(dev->config_rom[3]);
|
||||
*(__be32 *)&info.guid[4] = cpu_to_be32(dev->config_rom[4]);
|
||||
strlcpy(info.device_name, dev_name(&dev->device),
|
||||
sizeof(info.device_name));
|
||||
|
||||
if (copy_to_user(arg, &info, sizeof(info)))
|
||||
return -EFAULT;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
hwdep_lock(struct snd_bebob *bebob)
|
||||
{
|
||||
int err;
|
||||
|
||||
spin_lock_irq(&bebob->lock);
|
||||
|
||||
if (bebob->dev_lock_count == 0) {
|
||||
bebob->dev_lock_count = -1;
|
||||
err = 0;
|
||||
} else {
|
||||
err = -EBUSY;
|
||||
}
|
||||
|
||||
spin_unlock_irq(&bebob->lock);
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
static int
|
||||
hwdep_unlock(struct snd_bebob *bebob)
|
||||
{
|
||||
int err;
|
||||
|
||||
spin_lock_irq(&bebob->lock);
|
||||
|
||||
if (bebob->dev_lock_count == -1) {
|
||||
bebob->dev_lock_count = 0;
|
||||
err = 0;
|
||||
} else {
|
||||
err = -EBADFD;
|
||||
}
|
||||
|
||||
spin_unlock_irq(&bebob->lock);
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
static int
|
||||
hwdep_release(struct snd_hwdep *hwdep, struct file *file)
|
||||
{
|
||||
struct snd_bebob *bebob = hwdep->private_data;
|
||||
|
||||
spin_lock_irq(&bebob->lock);
|
||||
if (bebob->dev_lock_count == -1)
|
||||
bebob->dev_lock_count = 0;
|
||||
spin_unlock_irq(&bebob->lock);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
hwdep_ioctl(struct snd_hwdep *hwdep, struct file *file,
|
||||
unsigned int cmd, unsigned long arg)
|
||||
{
|
||||
struct snd_bebob *bebob = hwdep->private_data;
|
||||
|
||||
switch (cmd) {
|
||||
case SNDRV_FIREWIRE_IOCTL_GET_INFO:
|
||||
return hwdep_get_info(bebob, (void __user *)arg);
|
||||
case SNDRV_FIREWIRE_IOCTL_LOCK:
|
||||
return hwdep_lock(bebob);
|
||||
case SNDRV_FIREWIRE_IOCTL_UNLOCK:
|
||||
return hwdep_unlock(bebob);
|
||||
default:
|
||||
return -ENOIOCTLCMD;
|
||||
}
|
||||
}
|
||||
|
||||
#ifdef CONFIG_COMPAT
|
||||
static int
|
||||
hwdep_compat_ioctl(struct snd_hwdep *hwdep, struct file *file,
|
||||
unsigned int cmd, unsigned long arg)
|
||||
{
|
||||
return hwdep_ioctl(hwdep, file, cmd,
|
||||
(unsigned long)compat_ptr(arg));
|
||||
}
|
||||
#else
|
||||
#define hwdep_compat_ioctl NULL
|
||||
#endif
|
||||
|
||||
static const struct snd_hwdep_ops hwdep_ops = {
|
||||
.read = hwdep_read,
|
||||
.release = hwdep_release,
|
||||
.poll = hwdep_poll,
|
||||
.ioctl = hwdep_ioctl,
|
||||
.ioctl_compat = hwdep_compat_ioctl,
|
||||
};
|
||||
|
||||
int snd_bebob_create_hwdep_device(struct snd_bebob *bebob)
|
||||
{
|
||||
struct snd_hwdep *hwdep;
|
||||
int err;
|
||||
|
||||
err = snd_hwdep_new(bebob->card, "BeBoB", 0, &hwdep);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
strcpy(hwdep->name, "BeBoB");
|
||||
hwdep->iface = SNDRV_HWDEP_IFACE_FW_BEBOB;
|
||||
hwdep->ops = hwdep_ops;
|
||||
hwdep->private_data = bebob;
|
||||
hwdep->exclusive = true;
|
||||
end:
|
||||
return err;
|
||||
}
|
||||
|
813
sound/firewire/bebob/bebob_maudio.c
Normal file
813
sound/firewire/bebob/bebob_maudio.c
Normal file
|
@ -0,0 +1,813 @@
|
|||
/*
|
||||
* bebob_maudio.c - a part of driver for BeBoB based devices
|
||||
*
|
||||
* Copyright (c) 2013-2014 Takashi Sakamoto
|
||||
*
|
||||
* Licensed under the terms of the GNU General Public License, version 2.
|
||||
*/
|
||||
|
||||
#include "./bebob.h"
|
||||
#include <sound/control.h>
|
||||
|
||||
/*
|
||||
* Just powering on, Firewire 410/Audiophile/1814 and ProjectMix I/O wait to
|
||||
* download firmware blob. To enable these devices, drivers should upload
|
||||
* firmware blob and send a command to initialize configuration to factory
|
||||
* settings when completing uploading. Then these devices generate bus reset
|
||||
* and are recognized as new devices with the firmware.
|
||||
*
|
||||
* But with firmware version 5058 or later, the firmware is stored to flash
|
||||
* memory in the device and drivers can tell bootloader to load the firmware
|
||||
* by sending a cue. This cue must be sent one time.
|
||||
*
|
||||
* For streaming, both of output and input streams are needed for Firewire 410
|
||||
* and Ozonic. The single stream is OK for the other devices even if the clock
|
||||
* source is not SYT-Match (I note no devices use SYT-Match).
|
||||
*
|
||||
* Without streaming, the devices except for Firewire Audiophile can mix any
|
||||
* input and output. For this reason, Audiophile cannot be used as standalone
|
||||
* mixer.
|
||||
*
|
||||
* Firewire 1814 and ProjectMix I/O uses special firmware. It will be freezed
|
||||
* when receiving any commands which the firmware can't understand. These
|
||||
* devices utilize completely different system to control. It is some
|
||||
* write-transaction directly into a certain address. All of addresses for mixer
|
||||
* functionality is between 0xffc700700000 to 0xffc70070009c.
|
||||
*/
|
||||
|
||||
/* Offset from information register */
|
||||
#define INFO_OFFSET_SW_DATE 0x20
|
||||
|
||||
/* Bootloader Protocol Version 1 */
|
||||
#define MAUDIO_BOOTLOADER_CUE1 0x00000001
|
||||
/*
|
||||
* Initializing configuration to factory settings (= 0x1101), (swapped in line),
|
||||
* Command code is zero (= 0x00),
|
||||
* the number of operands is zero (= 0x00)(at least significant byte)
|
||||
*/
|
||||
#define MAUDIO_BOOTLOADER_CUE2 0x01110000
|
||||
/* padding */
|
||||
#define MAUDIO_BOOTLOADER_CUE3 0x00000000
|
||||
|
||||
#define MAUDIO_SPECIFIC_ADDRESS 0xffc700000000ULL
|
||||
|
||||
#define METER_OFFSET 0x00600000
|
||||
|
||||
/* some device has sync info after metering data */
|
||||
#define METER_SIZE_SPECIAL 84 /* with sync info */
|
||||
#define METER_SIZE_FW410 76 /* with sync info */
|
||||
#define METER_SIZE_AUDIOPHILE 60 /* with sync info */
|
||||
#define METER_SIZE_SOLO 52 /* with sync info */
|
||||
#define METER_SIZE_OZONIC 48
|
||||
#define METER_SIZE_NRV10 80
|
||||
|
||||
/* labels for metering */
|
||||
#define ANA_IN "Analog In"
|
||||
#define ANA_OUT "Analog Out"
|
||||
#define DIG_IN "Digital In"
|
||||
#define SPDIF_IN "S/PDIF In"
|
||||
#define ADAT_IN "ADAT In"
|
||||
#define DIG_OUT "Digital Out"
|
||||
#define SPDIF_OUT "S/PDIF Out"
|
||||
#define ADAT_OUT "ADAT Out"
|
||||
#define STRM_IN "Stream In"
|
||||
#define AUX_OUT "Aux Out"
|
||||
#define HP_OUT "HP Out"
|
||||
/* for NRV */
|
||||
#define UNKNOWN_METER "Unknown"
|
||||
|
||||
struct special_params {
|
||||
bool is1814;
|
||||
unsigned int clk_src;
|
||||
unsigned int dig_in_fmt;
|
||||
unsigned int dig_out_fmt;
|
||||
unsigned int clk_lock;
|
||||
struct snd_ctl_elem_id *ctl_id_sync;
|
||||
};
|
||||
|
||||
/*
|
||||
* For some M-Audio devices, this module just send cue to load firmware. After
|
||||
* loading, the device generates bus reset and newly detected.
|
||||
*
|
||||
* If we make any transactions to load firmware, the operation may failed.
|
||||
*/
|
||||
int snd_bebob_maudio_load_firmware(struct fw_unit *unit)
|
||||
{
|
||||
struct fw_device *device = fw_parent_device(unit);
|
||||
int err, rcode;
|
||||
u64 date;
|
||||
__le32 cues[3] = {
|
||||
cpu_to_le32(MAUDIO_BOOTLOADER_CUE1),
|
||||
cpu_to_le32(MAUDIO_BOOTLOADER_CUE2),
|
||||
cpu_to_le32(MAUDIO_BOOTLOADER_CUE3)
|
||||
};
|
||||
|
||||
/* check date of software used to build */
|
||||
err = snd_bebob_read_block(unit, INFO_OFFSET_SW_DATE,
|
||||
&date, sizeof(u64));
|
||||
if (err < 0)
|
||||
goto end;
|
||||
/*
|
||||
* firmware version 5058 or later has date later than "20070401", but
|
||||
* 'date' is not null-terminated.
|
||||
*/
|
||||
if (date < 0x3230303730343031LL) {
|
||||
dev_err(&unit->device,
|
||||
"Use firmware version 5058 or later\n");
|
||||
err = -ENOSYS;
|
||||
goto end;
|
||||
}
|
||||
|
||||
rcode = fw_run_transaction(device->card, TCODE_WRITE_BLOCK_REQUEST,
|
||||
device->node_id, device->generation,
|
||||
device->max_speed, BEBOB_ADDR_REG_REQ,
|
||||
cues, sizeof(cues));
|
||||
if (rcode != RCODE_COMPLETE) {
|
||||
dev_err(&unit->device,
|
||||
"Failed to send a cue to load firmware\n");
|
||||
err = -EIO;
|
||||
}
|
||||
end:
|
||||
return err;
|
||||
}
|
||||
|
||||
static inline int
|
||||
get_meter(struct snd_bebob *bebob, void *buf, unsigned int size)
|
||||
{
|
||||
return snd_fw_transaction(bebob->unit, TCODE_READ_BLOCK_REQUEST,
|
||||
MAUDIO_SPECIFIC_ADDRESS + METER_OFFSET,
|
||||
buf, size, 0);
|
||||
}
|
||||
|
||||
static int
|
||||
check_clk_sync(struct snd_bebob *bebob, unsigned int size, bool *sync)
|
||||
{
|
||||
int err;
|
||||
u8 *buf;
|
||||
|
||||
buf = kmalloc(size, GFP_KERNEL);
|
||||
if (buf == NULL)
|
||||
return -ENOMEM;
|
||||
|
||||
err = get_meter(bebob, buf, size);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
/* if synced, this value is the same as SFC of FDF in CIP header */
|
||||
*sync = (buf[size - 2] != 0xff);
|
||||
end:
|
||||
kfree(buf);
|
||||
return err;
|
||||
}
|
||||
|
||||
/*
|
||||
* dig_fmt: 0x00:S/PDIF, 0x01:ADAT
|
||||
* clk_lock: 0x00:unlock, 0x01:lock
|
||||
*/
|
||||
static int
|
||||
avc_maudio_set_special_clk(struct snd_bebob *bebob, unsigned int clk_src,
|
||||
unsigned int dig_in_fmt, unsigned int dig_out_fmt,
|
||||
unsigned int clk_lock)
|
||||
{
|
||||
struct special_params *params = bebob->maudio_special_quirk;
|
||||
int err;
|
||||
u8 *buf;
|
||||
|
||||
if (amdtp_stream_running(&bebob->rx_stream) ||
|
||||
amdtp_stream_running(&bebob->tx_stream))
|
||||
return -EBUSY;
|
||||
|
||||
buf = kmalloc(12, GFP_KERNEL);
|
||||
if (buf == NULL)
|
||||
return -ENOMEM;
|
||||
|
||||
buf[0] = 0x00; /* CONTROL */
|
||||
buf[1] = 0xff; /* UNIT */
|
||||
buf[2] = 0x00; /* vendor dependent */
|
||||
buf[3] = 0x04; /* company ID high */
|
||||
buf[4] = 0x00; /* company ID middle */
|
||||
buf[5] = 0x04; /* company ID low */
|
||||
buf[6] = 0xff & clk_src; /* clock source */
|
||||
buf[7] = 0xff & dig_in_fmt; /* input digital format */
|
||||
buf[8] = 0xff & dig_out_fmt; /* output digital format */
|
||||
buf[9] = 0xff & clk_lock; /* lock these settings */
|
||||
buf[10] = 0x00; /* padding */
|
||||
buf[11] = 0x00; /* padding */
|
||||
|
||||
err = fcp_avc_transaction(bebob->unit, buf, 12, buf, 12,
|
||||
BIT(1) | BIT(2) | BIT(3) | BIT(4) |
|
||||
BIT(5) | BIT(6) | BIT(7) | BIT(8) |
|
||||
BIT(9));
|
||||
if ((err > 0) && (err < 10))
|
||||
err = -EIO;
|
||||
else if (buf[0] == 0x08) /* NOT IMPLEMENTED */
|
||||
err = -ENOSYS;
|
||||
else if (buf[0] == 0x0a) /* REJECTED */
|
||||
err = -EINVAL;
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
params->clk_src = buf[6];
|
||||
params->dig_in_fmt = buf[7];
|
||||
params->dig_out_fmt = buf[8];
|
||||
params->clk_lock = buf[9];
|
||||
|
||||
if (params->ctl_id_sync)
|
||||
snd_ctl_notify(bebob->card, SNDRV_CTL_EVENT_MASK_VALUE,
|
||||
params->ctl_id_sync);
|
||||
|
||||
err = 0;
|
||||
end:
|
||||
kfree(buf);
|
||||
return err;
|
||||
}
|
||||
static void
|
||||
special_stream_formation_set(struct snd_bebob *bebob)
|
||||
{
|
||||
static const unsigned int ch_table[2][2][3] = {
|
||||
/* AMDTP_OUT_STREAM */
|
||||
{ { 6, 6, 4 }, /* SPDIF */
|
||||
{ 12, 8, 4 } }, /* ADAT */
|
||||
/* AMDTP_IN_STREAM */
|
||||
{ { 10, 10, 2 }, /* SPDIF */
|
||||
{ 16, 12, 2 } } /* ADAT */
|
||||
};
|
||||
struct special_params *params = bebob->maudio_special_quirk;
|
||||
unsigned int i, max;
|
||||
|
||||
max = SND_BEBOB_STRM_FMT_ENTRIES - 1;
|
||||
if (!params->is1814)
|
||||
max -= 2;
|
||||
|
||||
for (i = 0; i < max; i++) {
|
||||
bebob->tx_stream_formations[i + 1].pcm =
|
||||
ch_table[AMDTP_IN_STREAM][params->dig_in_fmt][i / 2];
|
||||
bebob->tx_stream_formations[i + 1].midi = 1;
|
||||
|
||||
bebob->rx_stream_formations[i + 1].pcm =
|
||||
ch_table[AMDTP_OUT_STREAM][params->dig_out_fmt][i / 2];
|
||||
bebob->rx_stream_formations[i + 1].midi = 1;
|
||||
}
|
||||
}
|
||||
|
||||
static int add_special_controls(struct snd_bebob *bebob);
|
||||
int
|
||||
snd_bebob_maudio_special_discover(struct snd_bebob *bebob, bool is1814)
|
||||
{
|
||||
struct special_params *params;
|
||||
int err;
|
||||
|
||||
params = kzalloc(sizeof(struct special_params), GFP_KERNEL);
|
||||
if (params == NULL)
|
||||
return -ENOMEM;
|
||||
|
||||
mutex_lock(&bebob->mutex);
|
||||
|
||||
bebob->maudio_special_quirk = (void *)params;
|
||||
params->is1814 = is1814;
|
||||
|
||||
/* initialize these parameters because driver is not allowed to ask */
|
||||
bebob->rx_stream.context = ERR_PTR(-1);
|
||||
bebob->tx_stream.context = ERR_PTR(-1);
|
||||
err = avc_maudio_set_special_clk(bebob, 0x03, 0x00, 0x00, 0x00);
|
||||
if (err < 0) {
|
||||
dev_err(&bebob->unit->device,
|
||||
"fail to initialize clock params: %d\n", err);
|
||||
goto end;
|
||||
}
|
||||
|
||||
err = add_special_controls(bebob);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
special_stream_formation_set(bebob);
|
||||
|
||||
if (params->is1814) {
|
||||
bebob->midi_input_ports = 1;
|
||||
bebob->midi_output_ports = 1;
|
||||
} else {
|
||||
bebob->midi_input_ports = 2;
|
||||
bebob->midi_output_ports = 2;
|
||||
}
|
||||
end:
|
||||
if (err < 0) {
|
||||
kfree(params);
|
||||
bebob->maudio_special_quirk = NULL;
|
||||
}
|
||||
mutex_unlock(&bebob->mutex);
|
||||
return err;
|
||||
}
|
||||
|
||||
/* Input plug shows actual rate. Output plug is needless for this purpose. */
|
||||
static int special_get_rate(struct snd_bebob *bebob, unsigned int *rate)
|
||||
{
|
||||
int err, trials;
|
||||
|
||||
trials = 0;
|
||||
do {
|
||||
err = avc_general_get_sig_fmt(bebob->unit, rate,
|
||||
AVC_GENERAL_PLUG_DIR_IN, 0);
|
||||
} while (err == -EAGAIN && ++trials < 3);
|
||||
|
||||
return err;
|
||||
}
|
||||
static int special_set_rate(struct snd_bebob *bebob, unsigned int rate)
|
||||
{
|
||||
struct special_params *params = bebob->maudio_special_quirk;
|
||||
int err;
|
||||
|
||||
err = avc_general_set_sig_fmt(bebob->unit, rate,
|
||||
AVC_GENERAL_PLUG_DIR_OUT, 0);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
/*
|
||||
* Just after changing sampling rate for output, a followed command
|
||||
* for input is easy to fail. This is a workaround fot this issue.
|
||||
*/
|
||||
msleep(100);
|
||||
|
||||
err = avc_general_set_sig_fmt(bebob->unit, rate,
|
||||
AVC_GENERAL_PLUG_DIR_IN, 0);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
if (params->ctl_id_sync)
|
||||
snd_ctl_notify(bebob->card, SNDRV_CTL_EVENT_MASK_VALUE,
|
||||
params->ctl_id_sync);
|
||||
end:
|
||||
return err;
|
||||
}
|
||||
|
||||
/* Clock source control for special firmware */
|
||||
static char *const special_clk_labels[] = {
|
||||
SND_BEBOB_CLOCK_INTERNAL " with Digital Mute", "Digital",
|
||||
"Word Clock", SND_BEBOB_CLOCK_INTERNAL};
|
||||
static int special_clk_get(struct snd_bebob *bebob, unsigned int *id)
|
||||
{
|
||||
struct special_params *params = bebob->maudio_special_quirk;
|
||||
*id = params->clk_src;
|
||||
return 0;
|
||||
}
|
||||
static int special_clk_ctl_info(struct snd_kcontrol *kctl,
|
||||
struct snd_ctl_elem_info *einf)
|
||||
{
|
||||
einf->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
|
||||
einf->count = 1;
|
||||
einf->value.enumerated.items = ARRAY_SIZE(special_clk_labels);
|
||||
|
||||
if (einf->value.enumerated.item >= einf->value.enumerated.items)
|
||||
einf->value.enumerated.item = einf->value.enumerated.items - 1;
|
||||
|
||||
strcpy(einf->value.enumerated.name,
|
||||
special_clk_labels[einf->value.enumerated.item]);
|
||||
|
||||
return 0;
|
||||
}
|
||||
static int special_clk_ctl_get(struct snd_kcontrol *kctl,
|
||||
struct snd_ctl_elem_value *uval)
|
||||
{
|
||||
struct snd_bebob *bebob = snd_kcontrol_chip(kctl);
|
||||
struct special_params *params = bebob->maudio_special_quirk;
|
||||
uval->value.enumerated.item[0] = params->clk_src;
|
||||
return 0;
|
||||
}
|
||||
static int special_clk_ctl_put(struct snd_kcontrol *kctl,
|
||||
struct snd_ctl_elem_value *uval)
|
||||
{
|
||||
struct snd_bebob *bebob = snd_kcontrol_chip(kctl);
|
||||
struct special_params *params = bebob->maudio_special_quirk;
|
||||
int err, id;
|
||||
|
||||
id = uval->value.enumerated.item[0];
|
||||
if (id >= ARRAY_SIZE(special_clk_labels))
|
||||
return -EINVAL;
|
||||
|
||||
mutex_lock(&bebob->mutex);
|
||||
|
||||
err = avc_maudio_set_special_clk(bebob, id,
|
||||
params->dig_in_fmt,
|
||||
params->dig_out_fmt,
|
||||
params->clk_lock);
|
||||
mutex_unlock(&bebob->mutex);
|
||||
|
||||
if (err >= 0)
|
||||
err = 1;
|
||||
|
||||
return err;
|
||||
}
|
||||
static struct snd_kcontrol_new special_clk_ctl = {
|
||||
.name = "Clock Source",
|
||||
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
||||
.access = SNDRV_CTL_ELEM_ACCESS_READWRITE,
|
||||
.info = special_clk_ctl_info,
|
||||
.get = special_clk_ctl_get,
|
||||
.put = special_clk_ctl_put
|
||||
};
|
||||
|
||||
/* Clock synchronization control for special firmware */
|
||||
static int special_sync_ctl_info(struct snd_kcontrol *kctl,
|
||||
struct snd_ctl_elem_info *einf)
|
||||
{
|
||||
einf->type = SNDRV_CTL_ELEM_TYPE_BOOLEAN;
|
||||
einf->count = 1;
|
||||
einf->value.integer.min = 0;
|
||||
einf->value.integer.max = 1;
|
||||
|
||||
return 0;
|
||||
}
|
||||
static int special_sync_ctl_get(struct snd_kcontrol *kctl,
|
||||
struct snd_ctl_elem_value *uval)
|
||||
{
|
||||
struct snd_bebob *bebob = snd_kcontrol_chip(kctl);
|
||||
int err;
|
||||
bool synced = 0;
|
||||
|
||||
err = check_clk_sync(bebob, METER_SIZE_SPECIAL, &synced);
|
||||
if (err >= 0)
|
||||
uval->value.integer.value[0] = synced;
|
||||
|
||||
return 0;
|
||||
}
|
||||
static struct snd_kcontrol_new special_sync_ctl = {
|
||||
.name = "Sync Status",
|
||||
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
||||
.access = SNDRV_CTL_ELEM_ACCESS_READ,
|
||||
.info = special_sync_ctl_info,
|
||||
.get = special_sync_ctl_get,
|
||||
};
|
||||
|
||||
/* Digital input interface control for special firmware */
|
||||
static char *const special_dig_in_iface_labels[] = {
|
||||
"S/PDIF Optical", "S/PDIF Coaxial", "ADAT Optical"
|
||||
};
|
||||
static int special_dig_in_iface_ctl_info(struct snd_kcontrol *kctl,
|
||||
struct snd_ctl_elem_info *einf)
|
||||
{
|
||||
einf->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
|
||||
einf->count = 1;
|
||||
einf->value.enumerated.items = ARRAY_SIZE(special_dig_in_iface_labels);
|
||||
|
||||
if (einf->value.enumerated.item >= einf->value.enumerated.items)
|
||||
einf->value.enumerated.item = einf->value.enumerated.items - 1;
|
||||
|
||||
strcpy(einf->value.enumerated.name,
|
||||
special_dig_in_iface_labels[einf->value.enumerated.item]);
|
||||
|
||||
return 0;
|
||||
}
|
||||
static int special_dig_in_iface_ctl_get(struct snd_kcontrol *kctl,
|
||||
struct snd_ctl_elem_value *uval)
|
||||
{
|
||||
struct snd_bebob *bebob = snd_kcontrol_chip(kctl);
|
||||
struct special_params *params = bebob->maudio_special_quirk;
|
||||
unsigned int dig_in_iface;
|
||||
int err, val;
|
||||
|
||||
mutex_lock(&bebob->mutex);
|
||||
|
||||
err = avc_audio_get_selector(bebob->unit, 0x00, 0x04,
|
||||
&dig_in_iface);
|
||||
if (err < 0) {
|
||||
dev_err(&bebob->unit->device,
|
||||
"fail to get digital input interface: %d\n", err);
|
||||
goto end;
|
||||
}
|
||||
|
||||
/* encoded id for user value */
|
||||
val = (params->dig_in_fmt << 1) | (dig_in_iface & 0x01);
|
||||
|
||||
/* for ADAT Optical */
|
||||
if (val > 2)
|
||||
val = 2;
|
||||
|
||||
uval->value.enumerated.item[0] = val;
|
||||
end:
|
||||
mutex_unlock(&bebob->mutex);
|
||||
return err;
|
||||
}
|
||||
static int special_dig_in_iface_ctl_set(struct snd_kcontrol *kctl,
|
||||
struct snd_ctl_elem_value *uval)
|
||||
{
|
||||
struct snd_bebob *bebob = snd_kcontrol_chip(kctl);
|
||||
struct special_params *params = bebob->maudio_special_quirk;
|
||||
unsigned int id, dig_in_fmt, dig_in_iface;
|
||||
int err;
|
||||
|
||||
id = uval->value.enumerated.item[0];
|
||||
if (id >= ARRAY_SIZE(special_dig_in_iface_labels))
|
||||
return -EINVAL;
|
||||
|
||||
/* decode user value */
|
||||
dig_in_fmt = (id >> 1) & 0x01;
|
||||
dig_in_iface = id & 0x01;
|
||||
|
||||
mutex_lock(&bebob->mutex);
|
||||
|
||||
err = avc_maudio_set_special_clk(bebob,
|
||||
params->clk_src,
|
||||
dig_in_fmt,
|
||||
params->dig_out_fmt,
|
||||
params->clk_lock);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
/* For ADAT, optical interface is only available. */
|
||||
if (params->dig_in_fmt > 0) {
|
||||
err = 1;
|
||||
goto end;
|
||||
}
|
||||
|
||||
/* For S/PDIF, optical/coaxial interfaces are selectable. */
|
||||
err = avc_audio_set_selector(bebob->unit, 0x00, 0x04, dig_in_iface);
|
||||
if (err < 0)
|
||||
dev_err(&bebob->unit->device,
|
||||
"fail to set digital input interface: %d\n", err);
|
||||
err = 1;
|
||||
end:
|
||||
special_stream_formation_set(bebob);
|
||||
mutex_unlock(&bebob->mutex);
|
||||
return err;
|
||||
}
|
||||
static struct snd_kcontrol_new special_dig_in_iface_ctl = {
|
||||
.name = "Digital Input Interface",
|
||||
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
||||
.access = SNDRV_CTL_ELEM_ACCESS_READWRITE,
|
||||
.info = special_dig_in_iface_ctl_info,
|
||||
.get = special_dig_in_iface_ctl_get,
|
||||
.put = special_dig_in_iface_ctl_set
|
||||
};
|
||||
|
||||
/* Digital output interface control for special firmware */
|
||||
static char *const special_dig_out_iface_labels[] = {
|
||||
"S/PDIF Optical and Coaxial", "ADAT Optical"
|
||||
};
|
||||
static int special_dig_out_iface_ctl_info(struct snd_kcontrol *kctl,
|
||||
struct snd_ctl_elem_info *einf)
|
||||
{
|
||||
einf->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
|
||||
einf->count = 1;
|
||||
einf->value.enumerated.items = ARRAY_SIZE(special_dig_out_iface_labels);
|
||||
|
||||
if (einf->value.enumerated.item >= einf->value.enumerated.items)
|
||||
einf->value.enumerated.item = einf->value.enumerated.items - 1;
|
||||
|
||||
strcpy(einf->value.enumerated.name,
|
||||
special_dig_out_iface_labels[einf->value.enumerated.item]);
|
||||
|
||||
return 0;
|
||||
}
|
||||
static int special_dig_out_iface_ctl_get(struct snd_kcontrol *kctl,
|
||||
struct snd_ctl_elem_value *uval)
|
||||
{
|
||||
struct snd_bebob *bebob = snd_kcontrol_chip(kctl);
|
||||
struct special_params *params = bebob->maudio_special_quirk;
|
||||
mutex_lock(&bebob->mutex);
|
||||
uval->value.enumerated.item[0] = params->dig_out_fmt;
|
||||
mutex_unlock(&bebob->mutex);
|
||||
return 0;
|
||||
}
|
||||
static int special_dig_out_iface_ctl_set(struct snd_kcontrol *kctl,
|
||||
struct snd_ctl_elem_value *uval)
|
||||
{
|
||||
struct snd_bebob *bebob = snd_kcontrol_chip(kctl);
|
||||
struct special_params *params = bebob->maudio_special_quirk;
|
||||
unsigned int id;
|
||||
int err;
|
||||
|
||||
id = uval->value.enumerated.item[0];
|
||||
if (id >= ARRAY_SIZE(special_dig_out_iface_labels))
|
||||
return -EINVAL;
|
||||
|
||||
mutex_lock(&bebob->mutex);
|
||||
|
||||
err = avc_maudio_set_special_clk(bebob,
|
||||
params->clk_src,
|
||||
params->dig_in_fmt,
|
||||
id, params->clk_lock);
|
||||
if (err >= 0) {
|
||||
special_stream_formation_set(bebob);
|
||||
err = 1;
|
||||
}
|
||||
|
||||
mutex_unlock(&bebob->mutex);
|
||||
return err;
|
||||
}
|
||||
static struct snd_kcontrol_new special_dig_out_iface_ctl = {
|
||||
.name = "Digital Output Interface",
|
||||
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
||||
.access = SNDRV_CTL_ELEM_ACCESS_READWRITE,
|
||||
.info = special_dig_out_iface_ctl_info,
|
||||
.get = special_dig_out_iface_ctl_get,
|
||||
.put = special_dig_out_iface_ctl_set
|
||||
};
|
||||
|
||||
static int add_special_controls(struct snd_bebob *bebob)
|
||||
{
|
||||
struct snd_kcontrol *kctl;
|
||||
struct special_params *params = bebob->maudio_special_quirk;
|
||||
int err;
|
||||
|
||||
kctl = snd_ctl_new1(&special_clk_ctl, bebob);
|
||||
err = snd_ctl_add(bebob->card, kctl);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
kctl = snd_ctl_new1(&special_sync_ctl, bebob);
|
||||
err = snd_ctl_add(bebob->card, kctl);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
params->ctl_id_sync = &kctl->id;
|
||||
|
||||
kctl = snd_ctl_new1(&special_dig_in_iface_ctl, bebob);
|
||||
err = snd_ctl_add(bebob->card, kctl);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
kctl = snd_ctl_new1(&special_dig_out_iface_ctl, bebob);
|
||||
err = snd_ctl_add(bebob->card, kctl);
|
||||
end:
|
||||
return err;
|
||||
}
|
||||
|
||||
/* Hardware metering for special firmware */
|
||||
static char *const special_meter_labels[] = {
|
||||
ANA_IN, ANA_IN, ANA_IN, ANA_IN,
|
||||
SPDIF_IN,
|
||||
ADAT_IN, ADAT_IN, ADAT_IN, ADAT_IN,
|
||||
ANA_OUT, ANA_OUT,
|
||||
SPDIF_OUT,
|
||||
ADAT_OUT, ADAT_OUT, ADAT_OUT, ADAT_OUT,
|
||||
HP_OUT, HP_OUT,
|
||||
AUX_OUT
|
||||
};
|
||||
static int
|
||||
special_meter_get(struct snd_bebob *bebob, u32 *target, unsigned int size)
|
||||
{
|
||||
u16 *buf;
|
||||
unsigned int i, c, channels;
|
||||
int err;
|
||||
|
||||
channels = ARRAY_SIZE(special_meter_labels) * 2;
|
||||
if (size < channels * sizeof(u32))
|
||||
return -EINVAL;
|
||||
|
||||
/* omit last 4 bytes because it's clock info. */
|
||||
buf = kmalloc(METER_SIZE_SPECIAL - 4, GFP_KERNEL);
|
||||
if (buf == NULL)
|
||||
return -ENOMEM;
|
||||
|
||||
err = get_meter(bebob, (void *)buf, METER_SIZE_SPECIAL - 4);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
/* Its format is u16 and some channels are unknown. */
|
||||
i = 0;
|
||||
for (c = 2; c < channels + 2; c++)
|
||||
target[i++] = be16_to_cpu(buf[c]) << 16;
|
||||
end:
|
||||
kfree(buf);
|
||||
return err;
|
||||
}
|
||||
|
||||
/* last 4 bytes are omitted because it's clock info. */
|
||||
static char *const fw410_meter_labels[] = {
|
||||
ANA_IN, DIG_IN,
|
||||
ANA_OUT, ANA_OUT, ANA_OUT, ANA_OUT, DIG_OUT,
|
||||
HP_OUT
|
||||
};
|
||||
static char *const audiophile_meter_labels[] = {
|
||||
ANA_IN, DIG_IN,
|
||||
ANA_OUT, ANA_OUT, DIG_OUT,
|
||||
HP_OUT, AUX_OUT,
|
||||
};
|
||||
static char *const solo_meter_labels[] = {
|
||||
ANA_IN, DIG_IN,
|
||||
STRM_IN, STRM_IN,
|
||||
ANA_OUT, DIG_OUT
|
||||
};
|
||||
|
||||
/* no clock info */
|
||||
static char *const ozonic_meter_labels[] = {
|
||||
ANA_IN, ANA_IN,
|
||||
STRM_IN, STRM_IN,
|
||||
ANA_OUT, ANA_OUT
|
||||
};
|
||||
/* TODO: need testers. these positions are based on authour's assumption */
|
||||
static char *const nrv10_meter_labels[] = {
|
||||
ANA_IN, ANA_IN, ANA_IN, ANA_IN,
|
||||
DIG_IN,
|
||||
ANA_OUT, ANA_OUT, ANA_OUT, ANA_OUT,
|
||||
DIG_IN
|
||||
};
|
||||
static int
|
||||
normal_meter_get(struct snd_bebob *bebob, u32 *buf, unsigned int size)
|
||||
{
|
||||
struct snd_bebob_meter_spec *spec = bebob->spec->meter;
|
||||
unsigned int c, channels;
|
||||
int err;
|
||||
|
||||
channels = spec->num * 2;
|
||||
if (size < channels * sizeof(u32))
|
||||
return -EINVAL;
|
||||
|
||||
err = get_meter(bebob, (void *)buf, size);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
for (c = 0; c < channels; c++)
|
||||
be32_to_cpus(&buf[c]);
|
||||
|
||||
/* swap stream channels because inverted */
|
||||
if (spec->labels == solo_meter_labels) {
|
||||
swap(buf[4], buf[6]);
|
||||
swap(buf[5], buf[7]);
|
||||
}
|
||||
end:
|
||||
return err;
|
||||
}
|
||||
|
||||
/* for special customized devices */
|
||||
static struct snd_bebob_rate_spec special_rate_spec = {
|
||||
.get = &special_get_rate,
|
||||
.set = &special_set_rate,
|
||||
};
|
||||
static struct snd_bebob_clock_spec special_clk_spec = {
|
||||
.num = ARRAY_SIZE(special_clk_labels),
|
||||
.labels = special_clk_labels,
|
||||
.get = &special_clk_get,
|
||||
};
|
||||
static struct snd_bebob_meter_spec special_meter_spec = {
|
||||
.num = ARRAY_SIZE(special_meter_labels),
|
||||
.labels = special_meter_labels,
|
||||
.get = &special_meter_get
|
||||
};
|
||||
struct snd_bebob_spec maudio_special_spec = {
|
||||
.clock = &special_clk_spec,
|
||||
.rate = &special_rate_spec,
|
||||
.meter = &special_meter_spec
|
||||
};
|
||||
|
||||
/* Firewire 410 specification */
|
||||
static struct snd_bebob_rate_spec usual_rate_spec = {
|
||||
.get = &snd_bebob_stream_get_rate,
|
||||
.set = &snd_bebob_stream_set_rate,
|
||||
};
|
||||
static struct snd_bebob_meter_spec fw410_meter_spec = {
|
||||
.num = ARRAY_SIZE(fw410_meter_labels),
|
||||
.labels = fw410_meter_labels,
|
||||
.get = &normal_meter_get
|
||||
};
|
||||
struct snd_bebob_spec maudio_fw410_spec = {
|
||||
.clock = NULL,
|
||||
.rate = &usual_rate_spec,
|
||||
.meter = &fw410_meter_spec
|
||||
};
|
||||
|
||||
/* Firewire Audiophile specification */
|
||||
static struct snd_bebob_meter_spec audiophile_meter_spec = {
|
||||
.num = ARRAY_SIZE(audiophile_meter_labels),
|
||||
.labels = audiophile_meter_labels,
|
||||
.get = &normal_meter_get
|
||||
};
|
||||
struct snd_bebob_spec maudio_audiophile_spec = {
|
||||
.clock = NULL,
|
||||
.rate = &usual_rate_spec,
|
||||
.meter = &audiophile_meter_spec
|
||||
};
|
||||
|
||||
/* Firewire Solo specification */
|
||||
static struct snd_bebob_meter_spec solo_meter_spec = {
|
||||
.num = ARRAY_SIZE(solo_meter_labels),
|
||||
.labels = solo_meter_labels,
|
||||
.get = &normal_meter_get
|
||||
};
|
||||
struct snd_bebob_spec maudio_solo_spec = {
|
||||
.clock = NULL,
|
||||
.rate = &usual_rate_spec,
|
||||
.meter = &solo_meter_spec
|
||||
};
|
||||
|
||||
/* Ozonic specification */
|
||||
static struct snd_bebob_meter_spec ozonic_meter_spec = {
|
||||
.num = ARRAY_SIZE(ozonic_meter_labels),
|
||||
.labels = ozonic_meter_labels,
|
||||
.get = &normal_meter_get
|
||||
};
|
||||
struct snd_bebob_spec maudio_ozonic_spec = {
|
||||
.clock = NULL,
|
||||
.rate = &usual_rate_spec,
|
||||
.meter = &ozonic_meter_spec
|
||||
};
|
||||
|
||||
/* NRV10 specification */
|
||||
static struct snd_bebob_meter_spec nrv10_meter_spec = {
|
||||
.num = ARRAY_SIZE(nrv10_meter_labels),
|
||||
.labels = nrv10_meter_labels,
|
||||
.get = &normal_meter_get
|
||||
};
|
||||
struct snd_bebob_spec maudio_nrv10_spec = {
|
||||
.clock = NULL,
|
||||
.rate = &usual_rate_spec,
|
||||
.meter = &nrv10_meter_spec
|
||||
};
|
168
sound/firewire/bebob/bebob_midi.c
Normal file
168
sound/firewire/bebob/bebob_midi.c
Normal file
|
@ -0,0 +1,168 @@
|
|||
/*
|
||||
* bebob_midi.c - a part of driver for BeBoB based devices
|
||||
*
|
||||
* Copyright (c) 2013-2014 Takashi Sakamoto
|
||||
*
|
||||
* Licensed under the terms of the GNU General Public License, version 2.
|
||||
*/
|
||||
|
||||
#include "bebob.h"
|
||||
|
||||
static int midi_capture_open(struct snd_rawmidi_substream *substream)
|
||||
{
|
||||
struct snd_bebob *bebob = substream->rmidi->private_data;
|
||||
int err;
|
||||
|
||||
err = snd_bebob_stream_lock_try(bebob);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
atomic_inc(&bebob->capture_substreams);
|
||||
err = snd_bebob_stream_start_duplex(bebob, 0);
|
||||
if (err < 0)
|
||||
snd_bebob_stream_lock_release(bebob);
|
||||
end:
|
||||
return err;
|
||||
}
|
||||
|
||||
static int midi_playback_open(struct snd_rawmidi_substream *substream)
|
||||
{
|
||||
struct snd_bebob *bebob = substream->rmidi->private_data;
|
||||
int err;
|
||||
|
||||
err = snd_bebob_stream_lock_try(bebob);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
atomic_inc(&bebob->playback_substreams);
|
||||
err = snd_bebob_stream_start_duplex(bebob, 0);
|
||||
if (err < 0)
|
||||
snd_bebob_stream_lock_release(bebob);
|
||||
end:
|
||||
return err;
|
||||
}
|
||||
|
||||
static int midi_capture_close(struct snd_rawmidi_substream *substream)
|
||||
{
|
||||
struct snd_bebob *bebob = substream->rmidi->private_data;
|
||||
|
||||
atomic_dec(&bebob->capture_substreams);
|
||||
snd_bebob_stream_stop_duplex(bebob);
|
||||
|
||||
snd_bebob_stream_lock_release(bebob);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int midi_playback_close(struct snd_rawmidi_substream *substream)
|
||||
{
|
||||
struct snd_bebob *bebob = substream->rmidi->private_data;
|
||||
|
||||
atomic_dec(&bebob->playback_substreams);
|
||||
snd_bebob_stream_stop_duplex(bebob);
|
||||
|
||||
snd_bebob_stream_lock_release(bebob);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void midi_capture_trigger(struct snd_rawmidi_substream *substrm, int up)
|
||||
{
|
||||
struct snd_bebob *bebob = substrm->rmidi->private_data;
|
||||
unsigned long flags;
|
||||
|
||||
spin_lock_irqsave(&bebob->lock, flags);
|
||||
|
||||
if (up)
|
||||
amdtp_stream_midi_trigger(&bebob->tx_stream,
|
||||
substrm->number, substrm);
|
||||
else
|
||||
amdtp_stream_midi_trigger(&bebob->tx_stream,
|
||||
substrm->number, NULL);
|
||||
|
||||
spin_unlock_irqrestore(&bebob->lock, flags);
|
||||
}
|
||||
|
||||
static void midi_playback_trigger(struct snd_rawmidi_substream *substrm, int up)
|
||||
{
|
||||
struct snd_bebob *bebob = substrm->rmidi->private_data;
|
||||
unsigned long flags;
|
||||
|
||||
spin_lock_irqsave(&bebob->lock, flags);
|
||||
|
||||
if (up)
|
||||
amdtp_stream_midi_trigger(&bebob->rx_stream,
|
||||
substrm->number, substrm);
|
||||
else
|
||||
amdtp_stream_midi_trigger(&bebob->rx_stream,
|
||||
substrm->number, NULL);
|
||||
|
||||
spin_unlock_irqrestore(&bebob->lock, flags);
|
||||
}
|
||||
|
||||
static struct snd_rawmidi_ops midi_capture_ops = {
|
||||
.open = midi_capture_open,
|
||||
.close = midi_capture_close,
|
||||
.trigger = midi_capture_trigger,
|
||||
};
|
||||
|
||||
static struct snd_rawmidi_ops midi_playback_ops = {
|
||||
.open = midi_playback_open,
|
||||
.close = midi_playback_close,
|
||||
.trigger = midi_playback_trigger,
|
||||
};
|
||||
|
||||
static void set_midi_substream_names(struct snd_bebob *bebob,
|
||||
struct snd_rawmidi_str *str)
|
||||
{
|
||||
struct snd_rawmidi_substream *subs;
|
||||
|
||||
list_for_each_entry(subs, &str->substreams, list) {
|
||||
snprintf(subs->name, sizeof(subs->name),
|
||||
"%s MIDI %d",
|
||||
bebob->card->shortname, subs->number + 1);
|
||||
}
|
||||
}
|
||||
|
||||
int snd_bebob_create_midi_devices(struct snd_bebob *bebob)
|
||||
{
|
||||
struct snd_rawmidi *rmidi;
|
||||
struct snd_rawmidi_str *str;
|
||||
int err;
|
||||
|
||||
/* create midi ports */
|
||||
err = snd_rawmidi_new(bebob->card, bebob->card->driver, 0,
|
||||
bebob->midi_output_ports, bebob->midi_input_ports,
|
||||
&rmidi);
|
||||
if (err < 0)
|
||||
return err;
|
||||
|
||||
snprintf(rmidi->name, sizeof(rmidi->name),
|
||||
"%s MIDI", bebob->card->shortname);
|
||||
rmidi->private_data = bebob;
|
||||
|
||||
if (bebob->midi_input_ports > 0) {
|
||||
rmidi->info_flags |= SNDRV_RAWMIDI_INFO_INPUT;
|
||||
|
||||
snd_rawmidi_set_ops(rmidi, SNDRV_RAWMIDI_STREAM_INPUT,
|
||||
&midi_capture_ops);
|
||||
|
||||
str = &rmidi->streams[SNDRV_RAWMIDI_STREAM_INPUT];
|
||||
|
||||
set_midi_substream_names(bebob, str);
|
||||
}
|
||||
|
||||
if (bebob->midi_output_ports > 0) {
|
||||
rmidi->info_flags |= SNDRV_RAWMIDI_INFO_OUTPUT;
|
||||
|
||||
snd_rawmidi_set_ops(rmidi, SNDRV_RAWMIDI_STREAM_OUTPUT,
|
||||
&midi_playback_ops);
|
||||
|
||||
str = &rmidi->streams[SNDRV_RAWMIDI_STREAM_OUTPUT];
|
||||
|
||||
set_midi_substream_names(bebob, str);
|
||||
}
|
||||
|
||||
if ((bebob->midi_output_ports > 0) && (bebob->midi_input_ports > 0))
|
||||
rmidi->info_flags |= SNDRV_RAWMIDI_INFO_DUPLEX;
|
||||
|
||||
return 0;
|
||||
}
|
378
sound/firewire/bebob/bebob_pcm.c
Normal file
378
sound/firewire/bebob/bebob_pcm.c
Normal file
|
@ -0,0 +1,378 @@
|
|||
/*
|
||||
* bebob_pcm.c - a part of driver for BeBoB based devices
|
||||
*
|
||||
* Copyright (c) 2013-2014 Takashi Sakamoto
|
||||
*
|
||||
* Licensed under the terms of the GNU General Public License, version 2.
|
||||
*/
|
||||
|
||||
#include "./bebob.h"
|
||||
|
||||
static int
|
||||
hw_rule_rate(struct snd_pcm_hw_params *params, struct snd_pcm_hw_rule *rule)
|
||||
{
|
||||
struct snd_bebob_stream_formation *formations = rule->private;
|
||||
struct snd_interval *r =
|
||||
hw_param_interval(params, SNDRV_PCM_HW_PARAM_RATE);
|
||||
const struct snd_interval *c =
|
||||
hw_param_interval_c(params, SNDRV_PCM_HW_PARAM_CHANNELS);
|
||||
struct snd_interval t = {
|
||||
.min = UINT_MAX, .max = 0, .integer = 1
|
||||
};
|
||||
unsigned int i;
|
||||
|
||||
for (i = 0; i < SND_BEBOB_STRM_FMT_ENTRIES; i++) {
|
||||
/* entry is invalid */
|
||||
if (formations[i].pcm == 0)
|
||||
continue;
|
||||
|
||||
if (!snd_interval_test(c, formations[i].pcm))
|
||||
continue;
|
||||
|
||||
t.min = min(t.min, snd_bebob_rate_table[i]);
|
||||
t.max = max(t.max, snd_bebob_rate_table[i]);
|
||||
|
||||
}
|
||||
return snd_interval_refine(r, &t);
|
||||
}
|
||||
|
||||
static int
|
||||
hw_rule_channels(struct snd_pcm_hw_params *params, struct snd_pcm_hw_rule *rule)
|
||||
{
|
||||
struct snd_bebob_stream_formation *formations = rule->private;
|
||||
struct snd_interval *c =
|
||||
hw_param_interval(params, SNDRV_PCM_HW_PARAM_CHANNELS);
|
||||
const struct snd_interval *r =
|
||||
hw_param_interval_c(params, SNDRV_PCM_HW_PARAM_RATE);
|
||||
struct snd_interval t = {
|
||||
.min = UINT_MAX, .max = 0, .integer = 1
|
||||
};
|
||||
|
||||
unsigned int i;
|
||||
|
||||
for (i = 0; i < SND_BEBOB_STRM_FMT_ENTRIES; i++) {
|
||||
/* entry is invalid */
|
||||
if (formations[i].pcm == 0)
|
||||
continue;
|
||||
|
||||
if (!snd_interval_test(r, snd_bebob_rate_table[i]))
|
||||
continue;
|
||||
|
||||
t.min = min(t.min, formations[i].pcm);
|
||||
t.max = max(t.max, formations[i].pcm);
|
||||
}
|
||||
|
||||
return snd_interval_refine(c, &t);
|
||||
}
|
||||
|
||||
static void
|
||||
limit_channels_and_rates(struct snd_pcm_hardware *hw,
|
||||
struct snd_bebob_stream_formation *formations)
|
||||
{
|
||||
unsigned int i;
|
||||
|
||||
hw->channels_min = UINT_MAX;
|
||||
hw->channels_max = 0;
|
||||
|
||||
hw->rate_min = UINT_MAX;
|
||||
hw->rate_max = 0;
|
||||
hw->rates = 0;
|
||||
|
||||
for (i = 0; i < SND_BEBOB_STRM_FMT_ENTRIES; i++) {
|
||||
/* entry has no PCM channels */
|
||||
if (formations[i].pcm == 0)
|
||||
continue;
|
||||
|
||||
hw->channels_min = min(hw->channels_min, formations[i].pcm);
|
||||
hw->channels_max = max(hw->channels_max, formations[i].pcm);
|
||||
|
||||
hw->rate_min = min(hw->rate_min, snd_bebob_rate_table[i]);
|
||||
hw->rate_max = max(hw->rate_max, snd_bebob_rate_table[i]);
|
||||
hw->rates |= snd_pcm_rate_to_rate_bit(snd_bebob_rate_table[i]);
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
limit_period_and_buffer(struct snd_pcm_hardware *hw)
|
||||
{
|
||||
hw->periods_min = 2; /* SNDRV_PCM_INFO_BATCH */
|
||||
hw->periods_max = UINT_MAX;
|
||||
|
||||
hw->period_bytes_min = 4 * hw->channels_max; /* bytes for a frame */
|
||||
|
||||
/* Just to prevent from allocating much pages. */
|
||||
hw->period_bytes_max = hw->period_bytes_min * 2048;
|
||||
hw->buffer_bytes_max = hw->period_bytes_max * hw->periods_min;
|
||||
}
|
||||
|
||||
static int
|
||||
pcm_init_hw_params(struct snd_bebob *bebob,
|
||||
struct snd_pcm_substream *substream)
|
||||
{
|
||||
struct snd_pcm_runtime *runtime = substream->runtime;
|
||||
struct amdtp_stream *s;
|
||||
struct snd_bebob_stream_formation *formations;
|
||||
int err;
|
||||
|
||||
runtime->hw.info = SNDRV_PCM_INFO_BATCH |
|
||||
SNDRV_PCM_INFO_BLOCK_TRANSFER |
|
||||
SNDRV_PCM_INFO_INTERLEAVED |
|
||||
SNDRV_PCM_INFO_JOINT_DUPLEX |
|
||||
SNDRV_PCM_INFO_MMAP |
|
||||
SNDRV_PCM_INFO_MMAP_VALID;
|
||||
|
||||
if (substream->stream == SNDRV_PCM_STREAM_CAPTURE) {
|
||||
runtime->hw.formats = AMDTP_IN_PCM_FORMAT_BITS;
|
||||
s = &bebob->tx_stream;
|
||||
formations = bebob->tx_stream_formations;
|
||||
} else {
|
||||
runtime->hw.formats = AMDTP_OUT_PCM_FORMAT_BITS;
|
||||
s = &bebob->rx_stream;
|
||||
formations = bebob->rx_stream_formations;
|
||||
}
|
||||
|
||||
limit_channels_and_rates(&runtime->hw, formations);
|
||||
limit_period_and_buffer(&runtime->hw);
|
||||
|
||||
err = snd_pcm_hw_rule_add(runtime, 0, SNDRV_PCM_HW_PARAM_CHANNELS,
|
||||
hw_rule_channels, formations,
|
||||
SNDRV_PCM_HW_PARAM_RATE, -1);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
err = snd_pcm_hw_rule_add(runtime, 0, SNDRV_PCM_HW_PARAM_RATE,
|
||||
hw_rule_rate, formations,
|
||||
SNDRV_PCM_HW_PARAM_CHANNELS, -1);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
err = amdtp_stream_add_pcm_hw_constraints(s, runtime);
|
||||
end:
|
||||
return err;
|
||||
}
|
||||
|
||||
static int
|
||||
pcm_open(struct snd_pcm_substream *substream)
|
||||
{
|
||||
struct snd_bebob *bebob = substream->private_data;
|
||||
struct snd_bebob_rate_spec *spec = bebob->spec->rate;
|
||||
unsigned int sampling_rate;
|
||||
bool internal;
|
||||
int err;
|
||||
|
||||
err = snd_bebob_stream_lock_try(bebob);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
err = pcm_init_hw_params(bebob, substream);
|
||||
if (err < 0)
|
||||
goto err_locked;
|
||||
|
||||
err = snd_bebob_stream_check_internal_clock(bebob, &internal);
|
||||
if (err < 0)
|
||||
goto err_locked;
|
||||
|
||||
/*
|
||||
* When source of clock is internal or any PCM stream are running,
|
||||
* the available sampling rate is limited at current sampling rate.
|
||||
*/
|
||||
if (!internal ||
|
||||
amdtp_stream_pcm_running(&bebob->tx_stream) ||
|
||||
amdtp_stream_pcm_running(&bebob->rx_stream)) {
|
||||
err = spec->get(bebob, &sampling_rate);
|
||||
if (err < 0) {
|
||||
dev_err(&bebob->unit->device,
|
||||
"fail to get sampling rate: %d\n", err);
|
||||
goto err_locked;
|
||||
}
|
||||
|
||||
substream->runtime->hw.rate_min = sampling_rate;
|
||||
substream->runtime->hw.rate_max = sampling_rate;
|
||||
}
|
||||
|
||||
snd_pcm_set_sync(substream);
|
||||
end:
|
||||
return err;
|
||||
err_locked:
|
||||
snd_bebob_stream_lock_release(bebob);
|
||||
return err;
|
||||
}
|
||||
|
||||
static int
|
||||
pcm_close(struct snd_pcm_substream *substream)
|
||||
{
|
||||
struct snd_bebob *bebob = substream->private_data;
|
||||
snd_bebob_stream_lock_release(bebob);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
pcm_capture_hw_params(struct snd_pcm_substream *substream,
|
||||
struct snd_pcm_hw_params *hw_params)
|
||||
{
|
||||
struct snd_bebob *bebob = substream->private_data;
|
||||
|
||||
if (substream->runtime->status->state == SNDRV_PCM_STATE_OPEN)
|
||||
atomic_inc(&bebob->capture_substreams);
|
||||
amdtp_stream_set_pcm_format(&bebob->tx_stream,
|
||||
params_format(hw_params));
|
||||
return snd_pcm_lib_alloc_vmalloc_buffer(substream,
|
||||
params_buffer_bytes(hw_params));
|
||||
}
|
||||
static int
|
||||
pcm_playback_hw_params(struct snd_pcm_substream *substream,
|
||||
struct snd_pcm_hw_params *hw_params)
|
||||
{
|
||||
struct snd_bebob *bebob = substream->private_data;
|
||||
|
||||
if (substream->runtime->status->state == SNDRV_PCM_STATE_OPEN)
|
||||
atomic_inc(&bebob->playback_substreams);
|
||||
amdtp_stream_set_pcm_format(&bebob->rx_stream,
|
||||
params_format(hw_params));
|
||||
return snd_pcm_lib_alloc_vmalloc_buffer(substream,
|
||||
params_buffer_bytes(hw_params));
|
||||
}
|
||||
|
||||
static int
|
||||
pcm_capture_hw_free(struct snd_pcm_substream *substream)
|
||||
{
|
||||
struct snd_bebob *bebob = substream->private_data;
|
||||
|
||||
if (substream->runtime->status->state != SNDRV_PCM_STATE_OPEN)
|
||||
atomic_dec(&bebob->capture_substreams);
|
||||
|
||||
snd_bebob_stream_stop_duplex(bebob);
|
||||
|
||||
return snd_pcm_lib_free_vmalloc_buffer(substream);
|
||||
}
|
||||
static int
|
||||
pcm_playback_hw_free(struct snd_pcm_substream *substream)
|
||||
{
|
||||
struct snd_bebob *bebob = substream->private_data;
|
||||
|
||||
if (substream->runtime->status->state != SNDRV_PCM_STATE_OPEN)
|
||||
atomic_dec(&bebob->playback_substreams);
|
||||
|
||||
snd_bebob_stream_stop_duplex(bebob);
|
||||
|
||||
return snd_pcm_lib_free_vmalloc_buffer(substream);
|
||||
}
|
||||
|
||||
static int
|
||||
pcm_capture_prepare(struct snd_pcm_substream *substream)
|
||||
{
|
||||
struct snd_bebob *bebob = substream->private_data;
|
||||
struct snd_pcm_runtime *runtime = substream->runtime;
|
||||
int err;
|
||||
|
||||
err = snd_bebob_stream_start_duplex(bebob, runtime->rate);
|
||||
if (err >= 0)
|
||||
amdtp_stream_pcm_prepare(&bebob->tx_stream);
|
||||
|
||||
return err;
|
||||
}
|
||||
static int
|
||||
pcm_playback_prepare(struct snd_pcm_substream *substream)
|
||||
{
|
||||
struct snd_bebob *bebob = substream->private_data;
|
||||
struct snd_pcm_runtime *runtime = substream->runtime;
|
||||
int err;
|
||||
|
||||
err = snd_bebob_stream_start_duplex(bebob, runtime->rate);
|
||||
if (err >= 0)
|
||||
amdtp_stream_pcm_prepare(&bebob->rx_stream);
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
static int
|
||||
pcm_capture_trigger(struct snd_pcm_substream *substream, int cmd)
|
||||
{
|
||||
struct snd_bebob *bebob = substream->private_data;
|
||||
|
||||
switch (cmd) {
|
||||
case SNDRV_PCM_TRIGGER_START:
|
||||
amdtp_stream_pcm_trigger(&bebob->tx_stream, substream);
|
||||
break;
|
||||
case SNDRV_PCM_TRIGGER_STOP:
|
||||
amdtp_stream_pcm_trigger(&bebob->tx_stream, NULL);
|
||||
break;
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
static int
|
||||
pcm_playback_trigger(struct snd_pcm_substream *substream, int cmd)
|
||||
{
|
||||
struct snd_bebob *bebob = substream->private_data;
|
||||
|
||||
switch (cmd) {
|
||||
case SNDRV_PCM_TRIGGER_START:
|
||||
amdtp_stream_pcm_trigger(&bebob->rx_stream, substream);
|
||||
break;
|
||||
case SNDRV_PCM_TRIGGER_STOP:
|
||||
amdtp_stream_pcm_trigger(&bebob->rx_stream, NULL);
|
||||
break;
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static snd_pcm_uframes_t
|
||||
pcm_capture_pointer(struct snd_pcm_substream *sbstrm)
|
||||
{
|
||||
struct snd_bebob *bebob = sbstrm->private_data;
|
||||
return amdtp_stream_pcm_pointer(&bebob->tx_stream);
|
||||
}
|
||||
static snd_pcm_uframes_t
|
||||
pcm_playback_pointer(struct snd_pcm_substream *sbstrm)
|
||||
{
|
||||
struct snd_bebob *bebob = sbstrm->private_data;
|
||||
return amdtp_stream_pcm_pointer(&bebob->rx_stream);
|
||||
}
|
||||
|
||||
static const struct snd_pcm_ops pcm_capture_ops = {
|
||||
.open = pcm_open,
|
||||
.close = pcm_close,
|
||||
.ioctl = snd_pcm_lib_ioctl,
|
||||
.hw_params = pcm_capture_hw_params,
|
||||
.hw_free = pcm_capture_hw_free,
|
||||
.prepare = pcm_capture_prepare,
|
||||
.trigger = pcm_capture_trigger,
|
||||
.pointer = pcm_capture_pointer,
|
||||
.page = snd_pcm_lib_get_vmalloc_page,
|
||||
};
|
||||
static const struct snd_pcm_ops pcm_playback_ops = {
|
||||
.open = pcm_open,
|
||||
.close = pcm_close,
|
||||
.ioctl = snd_pcm_lib_ioctl,
|
||||
.hw_params = pcm_playback_hw_params,
|
||||
.hw_free = pcm_playback_hw_free,
|
||||
.prepare = pcm_playback_prepare,
|
||||
.trigger = pcm_playback_trigger,
|
||||
.pointer = pcm_playback_pointer,
|
||||
.page = snd_pcm_lib_get_vmalloc_page,
|
||||
.mmap = snd_pcm_lib_mmap_vmalloc,
|
||||
};
|
||||
|
||||
int snd_bebob_create_pcm_devices(struct snd_bebob *bebob)
|
||||
{
|
||||
struct snd_pcm *pcm;
|
||||
int err;
|
||||
|
||||
err = snd_pcm_new(bebob->card, bebob->card->driver, 0, 1, 1, &pcm);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
pcm->private_data = bebob;
|
||||
snprintf(pcm->name, sizeof(pcm->name),
|
||||
"%s PCM", bebob->card->shortname);
|
||||
snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_PLAYBACK, &pcm_playback_ops);
|
||||
snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_CAPTURE, &pcm_capture_ops);
|
||||
end:
|
||||
return err;
|
||||
}
|
196
sound/firewire/bebob/bebob_proc.c
Normal file
196
sound/firewire/bebob/bebob_proc.c
Normal file
|
@ -0,0 +1,196 @@
|
|||
/*
|
||||
* bebob_proc.c - a part of driver for BeBoB based devices
|
||||
*
|
||||
* Copyright (c) 2013-2014 Takashi Sakamoto
|
||||
*
|
||||
* Licensed under the terms of the GNU General Public License, version 2.
|
||||
*/
|
||||
|
||||
#include "./bebob.h"
|
||||
|
||||
/* contents of information register */
|
||||
struct hw_info {
|
||||
u64 manufacturer;
|
||||
u32 protocol_ver;
|
||||
u32 bld_ver;
|
||||
u32 guid[2];
|
||||
u32 model_id;
|
||||
u32 model_rev;
|
||||
u64 fw_date;
|
||||
u64 fw_time;
|
||||
u32 fw_id;
|
||||
u32 fw_ver;
|
||||
u32 base_addr;
|
||||
u32 max_size;
|
||||
u64 bld_date;
|
||||
u64 bld_time;
|
||||
/* may not used in product
|
||||
u64 dbg_date;
|
||||
u64 dbg_time;
|
||||
u32 dbg_id;
|
||||
u32 dbg_version;
|
||||
*/
|
||||
} __packed;
|
||||
|
||||
static void
|
||||
proc_read_hw_info(struct snd_info_entry *entry,
|
||||
struct snd_info_buffer *buffer)
|
||||
{
|
||||
struct snd_bebob *bebob = entry->private_data;
|
||||
struct hw_info *info;
|
||||
|
||||
info = kzalloc(sizeof(struct hw_info), GFP_KERNEL);
|
||||
if (info == NULL)
|
||||
return;
|
||||
|
||||
if (snd_bebob_read_block(bebob->unit, 0,
|
||||
info, sizeof(struct hw_info)) < 0)
|
||||
goto end;
|
||||
|
||||
snd_iprintf(buffer, "Manufacturer:\t%.8s\n",
|
||||
(char *)&info->manufacturer);
|
||||
snd_iprintf(buffer, "Protocol Ver:\t%d\n", info->protocol_ver);
|
||||
snd_iprintf(buffer, "Build Ver:\t%d\n", info->bld_ver);
|
||||
snd_iprintf(buffer, "GUID:\t\t0x%.8X%.8X\n",
|
||||
info->guid[0], info->guid[1]);
|
||||
snd_iprintf(buffer, "Model ID:\t0x%02X\n", info->model_id);
|
||||
snd_iprintf(buffer, "Model Rev:\t%d\n", info->model_rev);
|
||||
snd_iprintf(buffer, "Firmware Date:\t%.8s\n", (char *)&info->fw_date);
|
||||
snd_iprintf(buffer, "Firmware Time:\t%.8s\n", (char *)&info->fw_time);
|
||||
snd_iprintf(buffer, "Firmware ID:\t0x%X\n", info->fw_id);
|
||||
snd_iprintf(buffer, "Firmware Ver:\t%d\n", info->fw_ver);
|
||||
snd_iprintf(buffer, "Base Addr:\t0x%X\n", info->base_addr);
|
||||
snd_iprintf(buffer, "Max Size:\t%d\n", info->max_size);
|
||||
snd_iprintf(buffer, "Loader Date:\t%.8s\n", (char *)&info->bld_date);
|
||||
snd_iprintf(buffer, "Loader Time:\t%.8s\n", (char *)&info->bld_time);
|
||||
|
||||
end:
|
||||
kfree(info);
|
||||
}
|
||||
|
||||
static void
|
||||
proc_read_meters(struct snd_info_entry *entry,
|
||||
struct snd_info_buffer *buffer)
|
||||
{
|
||||
struct snd_bebob *bebob = entry->private_data;
|
||||
struct snd_bebob_meter_spec *spec = bebob->spec->meter;
|
||||
u32 *buf;
|
||||
unsigned int i, c, channels, size;
|
||||
|
||||
if (spec == NULL)
|
||||
return;
|
||||
|
||||
channels = spec->num * 2;
|
||||
size = channels * sizeof(u32);
|
||||
buf = kmalloc(size, GFP_KERNEL);
|
||||
if (buf == NULL)
|
||||
return;
|
||||
|
||||
if (spec->get(bebob, buf, size) < 0)
|
||||
goto end;
|
||||
|
||||
for (i = 0, c = 1; i < channels; i++) {
|
||||
snd_iprintf(buffer, "%s %d:\t%d\n",
|
||||
spec->labels[i / 2], c++, buf[i]);
|
||||
if ((i + 1 < channels - 1) &&
|
||||
(strcmp(spec->labels[i / 2],
|
||||
spec->labels[(i + 1) / 2]) != 0))
|
||||
c = 1;
|
||||
}
|
||||
end:
|
||||
kfree(buf);
|
||||
}
|
||||
|
||||
static void
|
||||
proc_read_formation(struct snd_info_entry *entry,
|
||||
struct snd_info_buffer *buffer)
|
||||
{
|
||||
struct snd_bebob *bebob = entry->private_data;
|
||||
struct snd_bebob_stream_formation *formation;
|
||||
unsigned int i;
|
||||
|
||||
snd_iprintf(buffer, "Output Stream from device:\n");
|
||||
snd_iprintf(buffer, "\tRate\tPCM\tMIDI\n");
|
||||
formation = bebob->tx_stream_formations;
|
||||
for (i = 0; i < SND_BEBOB_STRM_FMT_ENTRIES; i++) {
|
||||
snd_iprintf(buffer,
|
||||
"\t%d\t%d\t%d\n", snd_bebob_rate_table[i],
|
||||
formation[i].pcm, formation[i].midi);
|
||||
}
|
||||
|
||||
snd_iprintf(buffer, "Input Stream to device:\n");
|
||||
snd_iprintf(buffer, "\tRate\tPCM\tMIDI\n");
|
||||
formation = bebob->rx_stream_formations;
|
||||
for (i = 0; i < SND_BEBOB_STRM_FMT_ENTRIES; i++) {
|
||||
snd_iprintf(buffer,
|
||||
"\t%d\t%d\t%d\n", snd_bebob_rate_table[i],
|
||||
formation[i].pcm, formation[i].midi);
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
proc_read_clock(struct snd_info_entry *entry,
|
||||
struct snd_info_buffer *buffer)
|
||||
{
|
||||
struct snd_bebob *bebob = entry->private_data;
|
||||
struct snd_bebob_rate_spec *rate_spec = bebob->spec->rate;
|
||||
struct snd_bebob_clock_spec *clk_spec = bebob->spec->clock;
|
||||
unsigned int rate, id;
|
||||
bool internal;
|
||||
|
||||
if (rate_spec->get(bebob, &rate) >= 0)
|
||||
snd_iprintf(buffer, "Sampling rate: %d\n", rate);
|
||||
|
||||
if (clk_spec) {
|
||||
if (clk_spec->get(bebob, &id) >= 0)
|
||||
snd_iprintf(buffer, "Clock Source: %s\n",
|
||||
clk_spec->labels[id]);
|
||||
} else {
|
||||
if (snd_bebob_stream_check_internal_clock(bebob,
|
||||
&internal) >= 0)
|
||||
snd_iprintf(buffer, "Clock Source: %s (MSU-dest: %d)\n",
|
||||
(internal) ? "Internal" : "External",
|
||||
bebob->sync_input_plug);
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
add_node(struct snd_bebob *bebob, struct snd_info_entry *root, const char *name,
|
||||
void (*op)(struct snd_info_entry *e, struct snd_info_buffer *b))
|
||||
{
|
||||
struct snd_info_entry *entry;
|
||||
|
||||
entry = snd_info_create_card_entry(bebob->card, name, root);
|
||||
if (entry == NULL)
|
||||
return;
|
||||
|
||||
snd_info_set_text_ops(entry, bebob, op);
|
||||
if (snd_info_register(entry) < 0)
|
||||
snd_info_free_entry(entry);
|
||||
}
|
||||
|
||||
void snd_bebob_proc_init(struct snd_bebob *bebob)
|
||||
{
|
||||
struct snd_info_entry *root;
|
||||
|
||||
/*
|
||||
* All nodes are automatically removed at snd_card_disconnect(),
|
||||
* by following to link list.
|
||||
*/
|
||||
root = snd_info_create_card_entry(bebob->card, "firewire",
|
||||
bebob->card->proc_root);
|
||||
if (root == NULL)
|
||||
return;
|
||||
root->mode = S_IFDIR | S_IRUGO | S_IXUGO;
|
||||
if (snd_info_register(root) < 0) {
|
||||
snd_info_free_entry(root);
|
||||
return;
|
||||
}
|
||||
|
||||
add_node(bebob, root, "clock", proc_read_clock);
|
||||
add_node(bebob, root, "firmware", proc_read_hw_info);
|
||||
add_node(bebob, root, "formation", proc_read_formation);
|
||||
|
||||
if (bebob->spec->meter != NULL)
|
||||
add_node(bebob, root, "meter", proc_read_meters);
|
||||
}
|
1033
sound/firewire/bebob/bebob_stream.c
Normal file
1033
sound/firewire/bebob/bebob_stream.c
Normal file
File diff suppressed because it is too large
Load diff
73
sound/firewire/bebob/bebob_terratec.c
Normal file
73
sound/firewire/bebob/bebob_terratec.c
Normal file
|
@ -0,0 +1,73 @@
|
|||
/*
|
||||
* bebob_terratec.c - a part of driver for BeBoB based devices
|
||||
*
|
||||
* Copyright (c) 2013-2014 Takashi Sakamoto
|
||||
*
|
||||
* Licensed under the terms of the GNU General Public License, version 2.
|
||||
*/
|
||||
|
||||
#include "./bebob.h"
|
||||
|
||||
static char *const phase88_rack_clk_src_labels[] = {
|
||||
SND_BEBOB_CLOCK_INTERNAL, "Digital In", "Word Clock"
|
||||
};
|
||||
static int
|
||||
phase88_rack_clk_src_get(struct snd_bebob *bebob, unsigned int *id)
|
||||
{
|
||||
unsigned int enable_ext, enable_word;
|
||||
int err;
|
||||
|
||||
err = avc_audio_get_selector(bebob->unit, 0, 9, &enable_ext);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
err = avc_audio_get_selector(bebob->unit, 0, 8, &enable_word);
|
||||
if (err < 0)
|
||||
goto end;
|
||||
|
||||
if (enable_ext == 0)
|
||||
*id = 0;
|
||||
else if (enable_word == 0)
|
||||
*id = 1;
|
||||
else
|
||||
*id = 2;
|
||||
end:
|
||||
return err;
|
||||
}
|
||||
|
||||
static char *const phase24_series_clk_src_labels[] = {
|
||||
SND_BEBOB_CLOCK_INTERNAL, "Digital In"
|
||||
};
|
||||
static int
|
||||
phase24_series_clk_src_get(struct snd_bebob *bebob, unsigned int *id)
|
||||
{
|
||||
return avc_audio_get_selector(bebob->unit, 0, 4, id);
|
||||
}
|
||||
|
||||
static struct snd_bebob_rate_spec phase_series_rate_spec = {
|
||||
.get = &snd_bebob_stream_get_rate,
|
||||
.set = &snd_bebob_stream_set_rate,
|
||||
};
|
||||
|
||||
/* PHASE 88 Rack FW */
|
||||
static struct snd_bebob_clock_spec phase88_rack_clk = {
|
||||
.num = ARRAY_SIZE(phase88_rack_clk_src_labels),
|
||||
.labels = phase88_rack_clk_src_labels,
|
||||
.get = &phase88_rack_clk_src_get,
|
||||
};
|
||||
struct snd_bebob_spec phase88_rack_spec = {
|
||||
.clock = &phase88_rack_clk,
|
||||
.rate = &phase_series_rate_spec,
|
||||
.meter = NULL
|
||||
};
|
||||
|
||||
/* 'PHASE 24 FW' and 'PHASE X24 FW' */
|
||||
static struct snd_bebob_clock_spec phase24_series_clk = {
|
||||
.num = ARRAY_SIZE(phase24_series_clk_src_labels),
|
||||
.labels = phase24_series_clk_src_labels,
|
||||
.get = &phase24_series_clk_src_get,
|
||||
};
|
||||
struct snd_bebob_spec phase24_series_spec = {
|
||||
.clock = &phase24_series_clk,
|
||||
.rate = &phase_series_rate_spec,
|
||||
.meter = NULL
|
||||
};
|
50
sound/firewire/bebob/bebob_yamaha.c
Normal file
50
sound/firewire/bebob/bebob_yamaha.c
Normal file
|
@ -0,0 +1,50 @@
|
|||
/*
|
||||
* bebob_yamaha.c - a part of driver for BeBoB based devices
|
||||
*
|
||||
* Copyright (c) 2013-2014 Takashi Sakamoto
|
||||
*
|
||||
* Licensed under the terms of the GNU General Public License, version 2.
|
||||
*/
|
||||
|
||||
#include "./bebob.h"
|
||||
|
||||
/*
|
||||
* NOTE:
|
||||
* Yamaha GO44 is not designed to be used as stand-alone mixer. So any streams
|
||||
* must be accompanied. If changing the state, a LED on the device starts to
|
||||
* blink and its sync status is false. In this state, the device sounds nothing
|
||||
* even if streaming. To start streaming at the current sampling rate is only
|
||||
* way to revocer this state. GO46 is better for stand-alone mixer.
|
||||
*
|
||||
* Both of them have a capability to change its sampling rate up to 192.0kHz.
|
||||
* At 192.0kHz, the device reports 4 PCM-in, 1 MIDI-in, 6 PCM-out, 1 MIDI-out.
|
||||
* But Yamaha's driver reduce 2 PCM-in, 1 MIDI-in, 2 PCM-out, 1 MIDI-out to use
|
||||
* 'Extended Stream Format Information Command - Single Request' in 'Additional
|
||||
* AVC commands' defined by BridgeCo.
|
||||
* This ALSA driver don't do this because a bit tiresome. Then isochronous
|
||||
* streaming with many asynchronous transactions brings sounds with noises.
|
||||
* Unfortunately current 'ffado-mixer' generated many asynchronous transaction
|
||||
* to observe device's state, mainly check cmp connection and signal format. I
|
||||
* reccomend users to close ffado-mixer at 192.0kHz if mixer is needless.
|
||||
*/
|
||||
|
||||
static char *const clk_src_labels[] = {SND_BEBOB_CLOCK_INTERNAL, "SPDIF"};
|
||||
static int
|
||||
clk_src_get(struct snd_bebob *bebob, unsigned int *id)
|
||||
{
|
||||
return avc_audio_get_selector(bebob->unit, 0, 4, id);
|
||||
}
|
||||
static struct snd_bebob_clock_spec clock_spec = {
|
||||
.num = ARRAY_SIZE(clk_src_labels),
|
||||
.labels = clk_src_labels,
|
||||
.get = &clk_src_get,
|
||||
};
|
||||
static struct snd_bebob_rate_spec rate_spec = {
|
||||
.get = &snd_bebob_stream_get_rate,
|
||||
.set = &snd_bebob_stream_set_rate,
|
||||
};
|
||||
struct snd_bebob_spec yamaha_go_spec = {
|
||||
.clock = &clock_spec,
|
||||
.rate = &rate_spec,
|
||||
.meter = NULL
|
||||
};
|
Loading…
Add table
Add a link
Reference in a new issue