ALSA: usb-audio: add support for many Roland/Yamaha devices
Add quirks to detect the various vendor-specific descriptors used by Roland and Yamaha in most of their recent USB audio and MIDI devices. Together with the previous patch, this should add audio/MIDI support for the following USB devices: - Edirol motion dive .tokyo performance package - Roland MC-808 Synthesizer - Roland BK-7m Synthesizer - Roland VIMA JM-5/8 Synthesizer - Roland SP-555 Sequencer - Roland V-Synth GT Synthesizer - Roland Music Atelier AT-75/100/300/350C/500/800/900/900C Organ - Edirol V-Mixer M-200i/300/380/400/480/R-1000 - BOSS GT-10B Effects Processor - Roland Fantom G6/G7/G8 Keyboard - Cakewalk Sonar V-Studio 20/100/700 Audio Interface - Roland GW-8 Keyboard - Roland AX-Synth Keyboard - Roland JUNO-Di/STAGE/Gi Keyboard - Roland VB-99 Effects Processor - Cakewalk UM-2G MIDI Interface - Roland A-500S Keyboard - Roland SD-50 Synthesizer - Roland OCTAPAD SPD-30 Controller - Roland Lucina AX-09 Synthesizer - BOSS BR-800 Digital Recorder - Roland DUO/TRI-CAPTURE (EX) Audio Interface - BOSS RC-300 Loop Station - Roland JUPITER-50/80 Keyboard - Roland R-26 Recorder - Roland SPD-SX Controller - BOSS JS-10 Audio Player - Roland TD-11/15/30 Drum Module - Roland A-49/88 Keyboard - Roland INTEGRA-7 Synthesizer - Roland R-88 Recorder Signed-off-by: Clemens Ladisch <clemens@ladisch.de>
This commit is contained in:
parent
ba7c2be114
commit
aafe77cc45
5 changed files with 252 additions and 3 deletions
|
@ -1947,6 +1947,44 @@ static int snd_usbmidi_detect_yamaha(struct snd_usb_midi* umidi,
|
||||||
return snd_usbmidi_detect_endpoints(umidi, endpoint, 1);
|
return snd_usbmidi_detect_endpoints(umidi, endpoint, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Detects the endpoints and ports of Roland devices.
|
||||||
|
*/
|
||||||
|
static int snd_usbmidi_detect_roland(struct snd_usb_midi* umidi,
|
||||||
|
struct snd_usb_midi_endpoint_info* endpoint)
|
||||||
|
{
|
||||||
|
struct usb_interface* intf;
|
||||||
|
struct usb_host_interface *hostif;
|
||||||
|
u8* cs_desc;
|
||||||
|
|
||||||
|
intf = umidi->iface;
|
||||||
|
if (!intf)
|
||||||
|
return -ENOENT;
|
||||||
|
hostif = intf->altsetting;
|
||||||
|
/*
|
||||||
|
* Some devices have a descriptor <06 24 F1 02 <inputs> <outputs>>,
|
||||||
|
* some have standard class descriptors, or both kinds, or neither.
|
||||||
|
*/
|
||||||
|
for (cs_desc = hostif->extra;
|
||||||
|
cs_desc < hostif->extra + hostif->extralen && cs_desc[0] >= 2;
|
||||||
|
cs_desc += cs_desc[0]) {
|
||||||
|
if (cs_desc[0] >= 6 &&
|
||||||
|
cs_desc[1] == USB_DT_CS_INTERFACE &&
|
||||||
|
cs_desc[2] == 0xf1 &&
|
||||||
|
cs_desc[3] == 0x02) {
|
||||||
|
endpoint->in_cables = (1 << cs_desc[4]) - 1;
|
||||||
|
endpoint->out_cables = (1 << cs_desc[5]) - 1;
|
||||||
|
return snd_usbmidi_detect_endpoints(umidi, endpoint, 1);
|
||||||
|
} else if (cs_desc[0] >= 7 &&
|
||||||
|
cs_desc[1] == USB_DT_CS_INTERFACE &&
|
||||||
|
cs_desc[2] == UAC_HEADER) {
|
||||||
|
return snd_usbmidi_get_ms_info(umidi, endpoint);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Creates the endpoints and their ports for Midiman devices.
|
* Creates the endpoints and their ports for Midiman devices.
|
||||||
*/
|
*/
|
||||||
|
@ -2162,6 +2200,9 @@ int snd_usbmidi_create(struct snd_card *card,
|
||||||
case QUIRK_MIDI_YAMAHA:
|
case QUIRK_MIDI_YAMAHA:
|
||||||
err = snd_usbmidi_detect_yamaha(umidi, &endpoints[0]);
|
err = snd_usbmidi_detect_yamaha(umidi, &endpoints[0]);
|
||||||
break;
|
break;
|
||||||
|
case QUIRK_MIDI_ROLAND:
|
||||||
|
err = snd_usbmidi_detect_roland(umidi, &endpoints[0]);
|
||||||
|
break;
|
||||||
case QUIRK_MIDI_MIDIMAN:
|
case QUIRK_MIDI_MIDIMAN:
|
||||||
umidi->usb_protocol_ops = &snd_usbmidi_midiman_ops;
|
umidi->usb_protocol_ops = &snd_usbmidi_midiman_ops;
|
||||||
memcpy(&endpoints[0], quirk->data,
|
memcpy(&endpoints[0], quirk->data,
|
||||||
|
|
|
@ -455,6 +455,17 @@ YAMAHA_DEVICE(0x7000, "DTX"),
|
||||||
YAMAHA_DEVICE(0x7010, "UB99"),
|
YAMAHA_DEVICE(0x7010, "UB99"),
|
||||||
#undef YAMAHA_DEVICE
|
#undef YAMAHA_DEVICE
|
||||||
#undef YAMAHA_INTERFACE
|
#undef YAMAHA_INTERFACE
|
||||||
|
/* this catches most recent vendor-specific Yamaha devices */
|
||||||
|
{
|
||||||
|
.match_flags = USB_DEVICE_ID_MATCH_VENDOR |
|
||||||
|
USB_DEVICE_ID_MATCH_INT_CLASS,
|
||||||
|
.idVendor = 0x0499,
|
||||||
|
.bInterfaceClass = USB_CLASS_VENDOR_SPEC,
|
||||||
|
.driver_info = (unsigned long) &(const struct snd_usb_audio_quirk) {
|
||||||
|
.ifnum = QUIRK_ANY_INTERFACE,
|
||||||
|
.type = QUIRK_AUTODETECT
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Roland/RolandED/Edirol/BOSS devices
|
* Roland/RolandED/Edirol/BOSS devices
|
||||||
|
@ -2031,6 +2042,17 @@ YAMAHA_DEVICE(0x7010, "UB99"),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
/* this catches most recent vendor-specific Roland devices */
|
||||||
|
{
|
||||||
|
.match_flags = USB_DEVICE_ID_MATCH_VENDOR |
|
||||||
|
USB_DEVICE_ID_MATCH_INT_CLASS,
|
||||||
|
.idVendor = 0x0582,
|
||||||
|
.bInterfaceClass = USB_CLASS_VENDOR_SPEC,
|
||||||
|
.driver_info = (unsigned long) &(const struct snd_usb_audio_quirk) {
|
||||||
|
.ifnum = QUIRK_ANY_INTERFACE,
|
||||||
|
.type = QUIRK_AUTODETECT
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
/* Guillemot devices */
|
/* Guillemot devices */
|
||||||
{
|
{
|
||||||
|
|
|
@ -18,6 +18,7 @@
|
||||||
#include <linux/slab.h>
|
#include <linux/slab.h>
|
||||||
#include <linux/usb.h>
|
#include <linux/usb.h>
|
||||||
#include <linux/usb/audio.h>
|
#include <linux/usb/audio.h>
|
||||||
|
#include <linux/usb/midi.h>
|
||||||
|
|
||||||
#include <sound/control.h>
|
#include <sound/control.h>
|
||||||
#include <sound/core.h>
|
#include <sound/core.h>
|
||||||
|
@ -175,6 +176,178 @@ static int create_fixed_stream_quirk(struct snd_usb_audio *chip,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int create_auto_pcm_quirk(struct snd_usb_audio *chip,
|
||||||
|
struct usb_interface *iface,
|
||||||
|
struct usb_driver *driver)
|
||||||
|
{
|
||||||
|
struct usb_host_interface *alts;
|
||||||
|
struct usb_interface_descriptor *altsd;
|
||||||
|
struct usb_endpoint_descriptor *epd;
|
||||||
|
struct uac1_as_header_descriptor *ashd;
|
||||||
|
struct uac_format_type_i_discrete_descriptor *fmtd;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Most Roland/Yamaha audio streaming interfaces have more or less
|
||||||
|
* standard descriptors, but older devices might lack descriptors, and
|
||||||
|
* future ones might change, so ensure that we fail silently if the
|
||||||
|
* interface doesn't look exactly right.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* must have a non-zero altsetting for streaming */
|
||||||
|
if (iface->num_altsetting < 2)
|
||||||
|
return -ENODEV;
|
||||||
|
alts = &iface->altsetting[1];
|
||||||
|
altsd = get_iface_desc(alts);
|
||||||
|
|
||||||
|
/* must have an isochronous endpoint for streaming */
|
||||||
|
if (altsd->bNumEndpoints < 1)
|
||||||
|
return -ENODEV;
|
||||||
|
epd = get_endpoint(alts, 0);
|
||||||
|
if (!usb_endpoint_xfer_isoc(epd))
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
/* must have format descriptors */
|
||||||
|
ashd = snd_usb_find_csint_desc(alts->extra, alts->extralen, NULL,
|
||||||
|
UAC_AS_GENERAL);
|
||||||
|
fmtd = snd_usb_find_csint_desc(alts->extra, alts->extralen, NULL,
|
||||||
|
UAC_FORMAT_TYPE);
|
||||||
|
if (!ashd || ashd->bLength < 7 ||
|
||||||
|
!fmtd || fmtd->bLength < 8)
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
return create_standard_audio_quirk(chip, iface, driver, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int create_yamaha_midi_quirk(struct snd_usb_audio *chip,
|
||||||
|
struct usb_interface *iface,
|
||||||
|
struct usb_driver *driver,
|
||||||
|
struct usb_host_interface *alts)
|
||||||
|
{
|
||||||
|
static const struct snd_usb_audio_quirk yamaha_midi_quirk = {
|
||||||
|
.type = QUIRK_MIDI_YAMAHA
|
||||||
|
};
|
||||||
|
struct usb_midi_in_jack_descriptor *injd;
|
||||||
|
struct usb_midi_out_jack_descriptor *outjd;
|
||||||
|
|
||||||
|
/* must have some valid jack descriptors */
|
||||||
|
injd = snd_usb_find_csint_desc(alts->extra, alts->extralen,
|
||||||
|
NULL, USB_MS_MIDI_IN_JACK);
|
||||||
|
outjd = snd_usb_find_csint_desc(alts->extra, alts->extralen,
|
||||||
|
NULL, USB_MS_MIDI_OUT_JACK);
|
||||||
|
if (!injd && !outjd)
|
||||||
|
return -ENODEV;
|
||||||
|
if (injd && (injd->bLength < 5 ||
|
||||||
|
(injd->bJackType != USB_MS_EMBEDDED &&
|
||||||
|
injd->bJackType != USB_MS_EXTERNAL)))
|
||||||
|
return -ENODEV;
|
||||||
|
if (outjd && (outjd->bLength < 6 ||
|
||||||
|
(outjd->bJackType != USB_MS_EMBEDDED &&
|
||||||
|
outjd->bJackType != USB_MS_EXTERNAL)))
|
||||||
|
return -ENODEV;
|
||||||
|
return create_any_midi_quirk(chip, iface, driver, &yamaha_midi_quirk);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int create_roland_midi_quirk(struct snd_usb_audio *chip,
|
||||||
|
struct usb_interface *iface,
|
||||||
|
struct usb_driver *driver,
|
||||||
|
struct usb_host_interface *alts)
|
||||||
|
{
|
||||||
|
static const struct snd_usb_audio_quirk roland_midi_quirk = {
|
||||||
|
.type = QUIRK_MIDI_ROLAND
|
||||||
|
};
|
||||||
|
u8 *roland_desc = NULL;
|
||||||
|
|
||||||
|
/* might have a vendor-specific descriptor <06 24 F1 02 ...> */
|
||||||
|
for (;;) {
|
||||||
|
roland_desc = snd_usb_find_csint_desc(alts->extra,
|
||||||
|
alts->extralen,
|
||||||
|
roland_desc, 0xf1);
|
||||||
|
if (!roland_desc)
|
||||||
|
return -ENODEV;
|
||||||
|
if (roland_desc[0] < 6 || roland_desc[3] != 2)
|
||||||
|
continue;
|
||||||
|
return create_any_midi_quirk(chip, iface, driver,
|
||||||
|
&roland_midi_quirk);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int create_std_midi_quirk(struct snd_usb_audio *chip,
|
||||||
|
struct usb_interface *iface,
|
||||||
|
struct usb_driver *driver,
|
||||||
|
struct usb_host_interface *alts)
|
||||||
|
{
|
||||||
|
struct usb_ms_header_descriptor *mshd;
|
||||||
|
struct usb_ms_endpoint_descriptor *msepd;
|
||||||
|
|
||||||
|
/* must have the MIDIStreaming interface header descriptor*/
|
||||||
|
mshd = (struct usb_ms_header_descriptor *)alts->extra;
|
||||||
|
if (alts->extralen < 7 ||
|
||||||
|
mshd->bLength < 7 ||
|
||||||
|
mshd->bDescriptorType != USB_DT_CS_INTERFACE ||
|
||||||
|
mshd->bDescriptorSubtype != USB_MS_HEADER)
|
||||||
|
return -ENODEV;
|
||||||
|
/* must have the MIDIStreaming endpoint descriptor*/
|
||||||
|
msepd = (struct usb_ms_endpoint_descriptor *)alts->endpoint[0].extra;
|
||||||
|
if (alts->endpoint[0].extralen < 4 ||
|
||||||
|
msepd->bLength < 4 ||
|
||||||
|
msepd->bDescriptorType != USB_DT_CS_ENDPOINT ||
|
||||||
|
msepd->bDescriptorSubtype != UAC_MS_GENERAL ||
|
||||||
|
msepd->bNumEmbMIDIJack < 1 ||
|
||||||
|
msepd->bNumEmbMIDIJack > 16)
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
return create_any_midi_quirk(chip, iface, driver, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int create_auto_midi_quirk(struct snd_usb_audio *chip,
|
||||||
|
struct usb_interface *iface,
|
||||||
|
struct usb_driver *driver)
|
||||||
|
{
|
||||||
|
struct usb_host_interface *alts;
|
||||||
|
struct usb_interface_descriptor *altsd;
|
||||||
|
struct usb_endpoint_descriptor *epd;
|
||||||
|
int err;
|
||||||
|
|
||||||
|
alts = &iface->altsetting[0];
|
||||||
|
altsd = get_iface_desc(alts);
|
||||||
|
|
||||||
|
/* must have at least one bulk/interrupt endpoint for streaming */
|
||||||
|
if (altsd->bNumEndpoints < 1)
|
||||||
|
return -ENODEV;
|
||||||
|
epd = get_endpoint(alts, 0);
|
||||||
|
if (!usb_endpoint_xfer_bulk(epd) ||
|
||||||
|
!usb_endpoint_xfer_int(epd))
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
switch (USB_ID_VENDOR(chip->usb_id)) {
|
||||||
|
case 0x0499: /* Yamaha */
|
||||||
|
err = create_yamaha_midi_quirk(chip, iface, driver, alts);
|
||||||
|
if (err < 0 && err != -ENODEV)
|
||||||
|
return err;
|
||||||
|
break;
|
||||||
|
case 0x0582: /* Roland */
|
||||||
|
err = create_roland_midi_quirk(chip, iface, driver, alts);
|
||||||
|
if (err < 0 && err != -ENODEV)
|
||||||
|
return err;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return create_std_midi_quirk(chip, iface, driver, alts);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int create_autodetect_quirk(struct snd_usb_audio *chip,
|
||||||
|
struct usb_interface *iface,
|
||||||
|
struct usb_driver *driver,
|
||||||
|
const struct snd_usb_audio_quirk *quirk)
|
||||||
|
{
|
||||||
|
int err;
|
||||||
|
|
||||||
|
err = create_auto_pcm_quirk(chip, iface, driver);
|
||||||
|
if (err == -ENODEV)
|
||||||
|
err = create_auto_midi_quirk(chip, iface, driver);
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Create a stream for an Edirol UA-700/UA-25/UA-4FX interface.
|
* Create a stream for an Edirol UA-700/UA-25/UA-4FX interface.
|
||||||
* The only way to detect the sample rate is by looking at wMaxPacketSize.
|
* The only way to detect the sample rate is by looking at wMaxPacketSize.
|
||||||
|
@ -303,9 +476,11 @@ int snd_usb_create_quirk(struct snd_usb_audio *chip,
|
||||||
static const quirk_func_t quirk_funcs[] = {
|
static const quirk_func_t quirk_funcs[] = {
|
||||||
[QUIRK_IGNORE_INTERFACE] = ignore_interface_quirk,
|
[QUIRK_IGNORE_INTERFACE] = ignore_interface_quirk,
|
||||||
[QUIRK_COMPOSITE] = create_composite_quirk,
|
[QUIRK_COMPOSITE] = create_composite_quirk,
|
||||||
|
[QUIRK_AUTODETECT] = create_autodetect_quirk,
|
||||||
[QUIRK_MIDI_STANDARD_INTERFACE] = create_any_midi_quirk,
|
[QUIRK_MIDI_STANDARD_INTERFACE] = create_any_midi_quirk,
|
||||||
[QUIRK_MIDI_FIXED_ENDPOINT] = create_any_midi_quirk,
|
[QUIRK_MIDI_FIXED_ENDPOINT] = create_any_midi_quirk,
|
||||||
[QUIRK_MIDI_YAMAHA] = create_any_midi_quirk,
|
[QUIRK_MIDI_YAMAHA] = create_any_midi_quirk,
|
||||||
|
[QUIRK_MIDI_ROLAND] = create_any_midi_quirk,
|
||||||
[QUIRK_MIDI_MIDIMAN] = create_any_midi_quirk,
|
[QUIRK_MIDI_MIDIMAN] = create_any_midi_quirk,
|
||||||
[QUIRK_MIDI_NOVATION] = create_any_midi_quirk,
|
[QUIRK_MIDI_NOVATION] = create_any_midi_quirk,
|
||||||
[QUIRK_MIDI_RAW_BYTES] = create_any_midi_quirk,
|
[QUIRK_MIDI_RAW_BYTES] = create_any_midi_quirk,
|
||||||
|
|
|
@ -493,10 +493,10 @@ int snd_usb_parse_audio_interface(struct snd_usb_audio *chip, int iface_no)
|
||||||
altsd = get_iface_desc(alts);
|
altsd = get_iface_desc(alts);
|
||||||
protocol = altsd->bInterfaceProtocol;
|
protocol = altsd->bInterfaceProtocol;
|
||||||
/* skip invalid one */
|
/* skip invalid one */
|
||||||
if ((altsd->bInterfaceClass != USB_CLASS_AUDIO &&
|
if (((altsd->bInterfaceClass != USB_CLASS_AUDIO ||
|
||||||
|
(altsd->bInterfaceSubClass != USB_SUBCLASS_AUDIOSTREAMING &&
|
||||||
|
altsd->bInterfaceSubClass != USB_SUBCLASS_VENDOR_SPEC)) &&
|
||||||
altsd->bInterfaceClass != USB_CLASS_VENDOR_SPEC) ||
|
altsd->bInterfaceClass != USB_CLASS_VENDOR_SPEC) ||
|
||||||
(altsd->bInterfaceSubClass != USB_SUBCLASS_AUDIOSTREAMING &&
|
|
||||||
altsd->bInterfaceSubClass != USB_SUBCLASS_VENDOR_SPEC) ||
|
|
||||||
altsd->bNumEndpoints < 1 ||
|
altsd->bNumEndpoints < 1 ||
|
||||||
le16_to_cpu(get_endpoint(alts, 0)->wMaxPacketSize) == 0)
|
le16_to_cpu(get_endpoint(alts, 0)->wMaxPacketSize) == 0)
|
||||||
continue;
|
continue;
|
||||||
|
@ -512,6 +512,15 @@ int snd_usb_parse_audio_interface(struct snd_usb_audio *chip, int iface_no)
|
||||||
if (snd_usb_apply_interface_quirk(chip, iface_no, altno))
|
if (snd_usb_apply_interface_quirk(chip, iface_no, altno))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Roland audio streaming interfaces are marked with protocols
|
||||||
|
* 0/1/2, but are UAC 1 compatible.
|
||||||
|
*/
|
||||||
|
if (USB_ID_VENDOR(chip->usb_id) == 0x0582 &&
|
||||||
|
altsd->bInterfaceClass == USB_CLASS_VENDOR_SPEC &&
|
||||||
|
protocol <= 2)
|
||||||
|
protocol = UAC_VERSION_1;
|
||||||
|
|
||||||
chconfig = 0;
|
chconfig = 0;
|
||||||
/* get audio formats */
|
/* get audio formats */
|
||||||
switch (protocol) {
|
switch (protocol) {
|
||||||
|
|
|
@ -72,9 +72,11 @@ struct snd_usb_audio {
|
||||||
enum quirk_type {
|
enum quirk_type {
|
||||||
QUIRK_IGNORE_INTERFACE,
|
QUIRK_IGNORE_INTERFACE,
|
||||||
QUIRK_COMPOSITE,
|
QUIRK_COMPOSITE,
|
||||||
|
QUIRK_AUTODETECT,
|
||||||
QUIRK_MIDI_STANDARD_INTERFACE,
|
QUIRK_MIDI_STANDARD_INTERFACE,
|
||||||
QUIRK_MIDI_FIXED_ENDPOINT,
|
QUIRK_MIDI_FIXED_ENDPOINT,
|
||||||
QUIRK_MIDI_YAMAHA,
|
QUIRK_MIDI_YAMAHA,
|
||||||
|
QUIRK_MIDI_ROLAND,
|
||||||
QUIRK_MIDI_MIDIMAN,
|
QUIRK_MIDI_MIDIMAN,
|
||||||
QUIRK_MIDI_NOVATION,
|
QUIRK_MIDI_NOVATION,
|
||||||
QUIRK_MIDI_RAW_BYTES,
|
QUIRK_MIDI_RAW_BYTES,
|
||||||
|
|
Loading…
Reference in a new issue