Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

Merge branch 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/tiwai/sound-2.6

* 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/tiwai/sound-2.6:
ALSA: hda-intel - fix wallclk variable update and condition
ALSA: asihpi - Fix uninitialized variable
ALSA: hda: Use LPIB for ASUS M2V
usb/gadget: Replace the old USB audio FU definitions in f_audio.c
ASoC: MX31ads sound support should depend on MACH_MX31ADS_WM1133_EV1
ASoC: Add missing Kconfig entry for Phytec boards
ALSA: usb-audio: export UAC2 clock selectors as mixer controls
ALSA: usb-audio: clean up find_audio_control_unit()
ALSA: usb-audio: add UAC2 sepecific Feature Unit controls
ALSA: usb-audio: unify constants from specification
ALSA: usb-audio: parse clock topology of UAC2 devices
ALSA: usb-audio: fix selector unit string index accessor
include/linux/usb/audio-v2.h: add more UAC2 details
ALSA: usb-audio: support partially write-protected UAC2 controls
ALSA: usb-audio: UAC2: clean up parsing of bmaControls
ALSA: hda: Use LPIB for another mainboard
ALSA: hda: Use mb31 quirk for an iMac model
ALSA: hda: Use LPIB for an ASUS device

+631 -256
+2 -2
drivers/usb/gadget/f_audio.c
··· 101 101 static struct usb_audio_control mute_control = { 102 102 .list = LIST_HEAD_INIT(mute_control.list), 103 103 .name = "Mute Control", 104 - .type = UAC_MUTE_CONTROL, 104 + .type = UAC_FU_MUTE, 105 105 /* Todo: add real Mute control code */ 106 106 .set = generic_set_cmd, 107 107 .get = generic_get_cmd, ··· 110 110 static struct usb_audio_control volume_control = { 111 111 .list = LIST_HEAD_INIT(volume_control.list), 112 112 .name = "Volume Control", 113 - .type = UAC_VOLUME_CONTROL, 113 + .type = UAC_FU_VOLUME, 114 114 /* Todo: add real Volume control code */ 115 115 .set = generic_set_cmd, 116 116 .get = generic_get_cmd,
+30 -1
include/linux/usb/audio-v2.h
··· 18 18 /* v1.0 and v2.0 of this standard have many things in common. For the rest 19 19 * of the definitions, please refer to audio.h */ 20 20 21 + static inline bool uac2_control_is_readable(u32 bmControls, u8 control) 22 + { 23 + return (bmControls >> (control * 2)) & 0x1; 24 + } 25 + 26 + static inline bool uac2_control_is_writeable(u32 bmControls, u8 control) 27 + { 28 + return (bmControls >> (control * 2)) & 0x2; 29 + } 30 + 21 31 /* 4.7.2.1 Clock Source Descriptor */ 22 32 23 33 struct uac_clock_source_descriptor { ··· 41 31 __u8 iClockSource; 42 32 } __attribute__((packed)); 43 33 34 + /* bmAttribute fields */ 35 + #define UAC_CLOCK_SOURCE_TYPE_EXT 0x0 36 + #define UAC_CLOCK_SOURCE_TYPE_INT_FIXED 0x1 37 + #define UAC_CLOCK_SOURCE_TYPE_INT_VAR 0x2 38 + #define UAC_CLOCK_SOURCE_TYPE_INT_PROG 0x3 39 + #define UAC_CLOCK_SOURCE_SYNCED_TO_SOF (1 << 2) 40 + 44 41 /* 4.7.2.2 Clock Source Descriptor */ 45 42 46 43 struct uac_clock_selector_descriptor { ··· 56 39 __u8 bDescriptorSubtype; 57 40 __u8 bClockID; 58 41 __u8 bNrInPins; 59 - __u8 bmControls; 60 42 __u8 baCSourceID[]; 43 + /* bmControls, bAssocTerminal and iClockSource omitted */ 44 + } __attribute__((packed)); 45 + 46 + /* 4.7.2.3 Clock Multiplier Descriptor */ 47 + 48 + struct uac_clock_multiplier_descriptor { 49 + __u8 bLength; 50 + __u8 bDescriptorType; 51 + __u8 bDescriptorSubtype; 52 + __u8 bClockID; 53 + __u8 bCSourceID; 54 + __u8 bmControls; 55 + __u8 iClockMultiplier; 61 56 } __attribute__((packed)); 62 57 63 58 /* 4.7.2.4 Input terminal descriptor */
+64 -26
include/linux/usb/audio.h
··· 47 47 #define UAC_FORMAT_TYPE 0x02 48 48 #define UAC_FORMAT_SPECIFIC 0x03 49 49 50 + /* A.7 Processing Unit Process Types */ 51 + #define UAC_PROCESS_UNDEFINED 0x00 52 + #define UAC_PROCESS_UP_DOWNMIX 0x01 53 + #define UAC_PROCESS_DOLBY_PROLOGIC 0x02 54 + #define UAC_PROCESS_STEREO_EXTENDER 0x03 55 + #define UAC_PROCESS_REVERB 0x04 56 + #define UAC_PROCESS_CHORUS 0x05 57 + #define UAC_PROCESS_DYN_RANGE_COMP 0x06 58 + 50 59 /* A.8 Audio Class-Specific Endpoint Descriptor Subtypes */ 51 60 #define UAC_EP_GENERAL 0x01 52 61 ··· 81 72 #define UAC_GET_MEM (UAC_GET_ | UAC__MEM) 82 73 83 74 #define UAC_GET_STAT 0xff 75 + 76 + /* A.10 Control Selector Codes */ 77 + 78 + /* A.10.1 Terminal Control Selectors */ 79 + #define UAC_TERM_COPY_PROTECT 0x01 80 + 81 + /* A.10.2 Feature Unit Control Selectors */ 82 + #define UAC_FU_MUTE 0x01 83 + #define UAC_FU_VOLUME 0x02 84 + #define UAC_FU_BASS 0x03 85 + #define UAC_FU_MID 0x04 86 + #define UAC_FU_TREBLE 0x05 87 + #define UAC_FU_GRAPHIC_EQUALIZER 0x06 88 + #define UAC_FU_AUTOMATIC_GAIN 0x07 89 + #define UAC_FU_DELAY 0x08 90 + #define UAC_FU_BASS_BOOST 0x09 91 + #define UAC_FU_LOUDNESS 0x0a 92 + 93 + #define UAC_CONTROL_BIT(CS) (1 << ((CS) - 1)) 94 + 95 + /* A.10.3.1 Up/Down-mix Processing Unit Controls Selectors */ 96 + #define UAC_UD_ENABLE 0x01 97 + #define UAC_UD_MODE_SELECT 0x02 98 + 99 + /* A.10.3.2 Dolby Prologic (tm) Processing Unit Controls Selectors */ 100 + #define UAC_DP_ENABLE 0x01 101 + #define UAC_DP_MODE_SELECT 0x02 102 + 103 + /* A.10.3.3 3D Stereo Extender Processing Unit Control Selectors */ 104 + #define UAC_3D_ENABLE 0x01 105 + #define UAC_3D_SPACE 0x02 106 + 107 + /* A.10.3.4 Reverberation Processing Unit Control Selectors */ 108 + #define UAC_REVERB_ENABLE 0x01 109 + #define UAC_REVERB_LEVEL 0x02 110 + #define UAC_REVERB_TIME 0x03 111 + #define UAC_REVERB_FEEDBACK 0x04 112 + 113 + /* A.10.3.5 Chorus Processing Unit Control Selectors */ 114 + #define UAC_CHORUS_ENABLE 0x01 115 + #define UAC_CHORUS_LEVEL 0x02 116 + #define UAC_CHORUS_RATE 0x03 117 + #define UAC_CHORUS_DEPTH 0x04 118 + 119 + /* A.10.3.6 Dynamic Range Compressor Unit Control Selectors */ 120 + #define UAC_DCR_ENABLE 0x01 121 + #define UAC_DCR_RATE 0x02 122 + #define UAC_DCR_MAXAMPL 0x03 123 + #define UAC_DCR_THRESHOLD 0x04 124 + #define UAC_DCR_ATTACK_TIME 0x05 125 + #define UAC_DCR_RELEASE_TIME 0x06 126 + 127 + /* A.10.4 Extension Unit Control Selectors */ 128 + #define UAC_XU_ENABLE 0x01 84 129 85 130 /* MIDI - A.1 MS Class-Specific Interface Descriptor Subtypes */ 86 131 #define UAC_MS_HEADER 0x01 ··· 307 244 static inline __u8 uac_selector_unit_iSelector(struct uac_selector_unit_descriptor *desc) 308 245 { 309 246 __u8 *raw = (__u8 *) desc; 310 - return raw[9 + desc->bLength - 1]; 247 + return raw[desc->bLength - 1]; 311 248 } 312 249 313 250 /* 4.3.2.5 Feature Unit Descriptor */ ··· 525 462 #define UAC_EP_CS_ATTR_SAMPLE_RATE 0x01 526 463 #define UAC_EP_CS_ATTR_PITCH_CONTROL 0x02 527 464 #define UAC_EP_CS_ATTR_FILL_MAX 0x80 528 - 529 - /* A.10.2 Feature Unit Control Selectors */ 530 - 531 - #define UAC_FU_CONTROL_UNDEFINED 0x00 532 - #define UAC_MUTE_CONTROL 0x01 533 - #define UAC_VOLUME_CONTROL 0x02 534 - #define UAC_BASS_CONTROL 0x03 535 - #define UAC_MID_CONTROL 0x04 536 - #define UAC_TREBLE_CONTROL 0x05 537 - #define UAC_GRAPHIC_EQUALIZER_CONTROL 0x06 538 - #define UAC_AUTOMATIC_GAIN_CONTROL 0x07 539 - #define UAC_DELAY_CONTROL 0x08 540 - #define UAC_BASS_BOOST_CONTROL 0x09 541 - #define UAC_LOUDNESS_CONTROL 0x0a 542 - 543 - #define UAC_FU_MUTE (1 << (UAC_MUTE_CONTROL - 1)) 544 - #define UAC_FU_VOLUME (1 << (UAC_VOLUME_CONTROL - 1)) 545 - #define UAC_FU_BASS (1 << (UAC_BASS_CONTROL - 1)) 546 - #define UAC_FU_MID (1 << (UAC_MID_CONTROL - 1)) 547 - #define UAC_FU_TREBLE (1 << (UAC_TREBLE_CONTROL - 1)) 548 - #define UAC_FU_GRAPHIC_EQ (1 << (UAC_GRAPHIC_EQUALIZER_CONTROL - 1)) 549 - #define UAC_FU_AUTO_GAIN (1 << (UAC_AUTOMATIC_GAIN_CONTROL - 1)) 550 - #define UAC_FU_DELAY (1 << (UAC_DELAY_CONTROL - 1)) 551 - #define UAC_FU_BASS_BOOST (1 << (UAC_BASS_BOOST_CONTROL - 1)) 552 - #define UAC_FU_LOUDNESS (1 << (UAC_LOUDNESS_CONTROL - 1)) 553 465 554 466 /* status word format (3.7.1.1) */ 555 467
+3
sound/pci/asihpi/asihpi.c
··· 2578 2578 if (err) 2579 2579 return -err; 2580 2580 2581 + memset(&prev_ctl, 0, sizeof(prev_ctl)); 2582 + prev_ctl.control_type = -1; 2583 + 2581 2584 for (idx = 0; idx < 2000; idx++) { 2582 2585 err = hpi_mixer_get_control_by_index( 2583 2586 ss, asihpi->h_mixer,
+5 -2
sound/pci/hda/hda_intel.c
··· 1913 1913 if (WARN_ONCE(!azx_dev->period_bytes, 1914 1914 "hda-intel: zero azx_dev->period_bytes")) 1915 1915 return -1; /* this shouldn't happen! */ 1916 - if (wallclk <= azx_dev->period_wallclk && 1916 + if (wallclk < (azx_dev->period_wallclk * 5) / 4 && 1917 1917 pos % azx_dev->period_bytes > azx_dev->period_bytes / 2) 1918 1918 /* NG - it's below the first next period boundary */ 1919 1919 return bdl_pos_adj[chip->dev_index] ? 0 : -1; 1920 - azx_dev->start_wallclk = wallclk; 1920 + azx_dev->start_wallclk += wallclk; 1921 1921 return 1; /* OK, it's fine */ 1922 1922 } 1923 1923 ··· 2288 2288 SND_PCI_QUIRK(0x1028, 0x01f6, "Dell Latitude 131L", POS_FIX_LPIB), 2289 2289 SND_PCI_QUIRK(0x103c, 0x306d, "HP dv3", POS_FIX_LPIB), 2290 2290 SND_PCI_QUIRK(0x1043, 0x813d, "ASUS P5AD2", POS_FIX_LPIB), 2291 + SND_PCI_QUIRK(0x1043, 0x81b3, "ASUS", POS_FIX_LPIB), 2292 + SND_PCI_QUIRK(0x1043, 0x81e7, "ASUS M2V", POS_FIX_LPIB), 2291 2293 SND_PCI_QUIRK(0x104d, 0x9069, "Sony VPCS11V9E", POS_FIX_LPIB), 2292 2294 SND_PCI_QUIRK(0x1106, 0x3288, "ASUS M2V-MX SE", POS_FIX_LPIB), 2293 2295 SND_PCI_QUIRK(0x1179, 0xff10, "Toshiba A100-259", POS_FIX_LPIB), ··· 2298 2296 SND_PCI_QUIRK(0x1462, 0x1002, "MSI Wind U115", POS_FIX_LPIB), 2299 2297 SND_PCI_QUIRK(0x1565, 0x820f, "Biostar Microtech", POS_FIX_LPIB), 2300 2298 SND_PCI_QUIRK(0x1565, 0x8218, "Biostar Microtech", POS_FIX_LPIB), 2299 + SND_PCI_QUIRK(0x1849, 0x0888, "775Dual-VSTA", POS_FIX_LPIB), 2301 2300 SND_PCI_QUIRK(0x8086, 0x2503, "DG965OT AAD63733-203", POS_FIX_LPIB), 2302 2301 SND_PCI_QUIRK(0x8086, 0xd601, "eMachines T5212", POS_FIX_LPIB), 2303 2302 {}
+1
sound/pci/hda/patch_realtek.c
··· 9476 9476 SND_PCI_QUIRK(0x106b, 0x1000, "iMac 24", ALC885_IMAC24), 9477 9477 SND_PCI_QUIRK(0x106b, 0x2800, "AppleTV", ALC885_IMAC24), 9478 9478 SND_PCI_QUIRK(0x106b, 0x2c00, "MacbookPro rev3", ALC885_MBP3), 9479 + SND_PCI_QUIRK(0x106b, 0x3000, "iMac", ALC889A_MB31), 9479 9480 SND_PCI_QUIRK(0x106b, 0x3600, "Macbook 3,1", ALC889A_MB31), 9480 9481 SND_PCI_QUIRK(0x106b, 0x3800, "MacbookPro 4,1", ALC885_MBP3), 9481 9482 SND_PCI_QUIRK(0x106b, 0x3e00, "iMac 24 Aluminum", ALC885_IMAC24),
+10 -1
sound/soc/imx/Kconfig
··· 13 13 14 14 config SND_MXC_SOC_WM1133_EV1 15 15 tristate "Audio on the the i.MX31ADS with WM1133-EV1 fitted" 16 - depends on SND_IMX_SOC && EXPERIMENTAL 16 + depends on SND_IMX_SOC && MACH_MX31ADS_WM1133_EV1 && EXPERIMENTAL 17 17 select SND_SOC_WM8350 18 18 select SND_MXC_SOC_SSI 19 19 help 20 20 Enable support for audio on the i.MX31ADS with the WM1133-EV1 21 21 PMIC board with WM8835x fitted. 22 + 23 + config SND_SOC_PHYCORE_AC97 24 + tristate "SoC Audio support for Phytec phyCORE (and phyCARD) boards" 25 + depends on MACH_PCM043 || MACH_PCA100 26 + select SND_MXC_SOC_SSI 27 + select SND_SOC_WM9712 28 + help 29 + Say Y if you want to add support for SoC audio on Phytec phyCORE 30 + and phyCARD boards in AC97 mode
+2 -1
sound/usb/Makefile
··· 11 11 endpoint.o \ 12 12 urb.o \ 13 13 pcm.o \ 14 - helper.o 14 + helper.o \ 15 + clock.o 15 16 16 17 snd-usbmidi-lib-objs := midi.o 17 18
+2 -16
sound/usb/card.c
··· 236 236 } 237 237 238 238 case UAC_VERSION_2: { 239 - struct uac_clock_source_descriptor *cs; 240 239 struct usb_interface_assoc_descriptor *assoc = 241 240 usb_ifnum_to_if(dev, ctrlif)->intf_assoc; 242 241 ··· 243 244 snd_printk(KERN_ERR "Audio class v2 interfaces need an interface association\n"); 244 245 return -EINVAL; 245 246 } 246 - 247 - /* FIXME: for now, we expect there is at least one clock source 248 - * descriptor and we always take the first one. 249 - * We should properly support devices with multiple clock sources, 250 - * clock selectors and sample rate conversion units. */ 251 - 252 - cs = snd_usb_find_csint_desc(host_iface->extra, host_iface->extralen, 253 - NULL, UAC2_CLOCK_SOURCE); 254 - 255 - if (!cs) { 256 - snd_printk(KERN_ERR "CLOCK_SOURCE descriptor not found\n"); 257 - return -EINVAL; 258 - } 259 - 260 - chip->clock_id = cs->bClockID; 261 247 262 248 for (i = 0; i < assoc->bInterfaceCount; i++) { 263 249 int intf = assoc->bFirstInterface + i; ··· 464 480 if ((err = snd_usb_create_quirk(chip, intf, &usb_audio_driver, quirk)) < 0) 465 481 goto __error; 466 482 } 483 + 484 + chip->ctrl_intf = alts; 467 485 468 486 if (err > 0) { 469 487 /* create normal USB audio interfaces */
+1
sound/usb/card.h
··· 25 25 unsigned int rate_min, rate_max; /* min/max rates */ 26 26 unsigned int nr_rates; /* number of rate table entries */ 27 27 unsigned int *rate_table; /* rate table */ 28 + unsigned char clock; /* associated clock */ 28 29 }; 29 30 30 31 struct snd_usb_substream;
+311
sound/usb/clock.c
··· 1 + /* 2 + * Clock domain and sample rate management functions 3 + * 4 + * This program is free software; you can redistribute it and/or modify 5 + * it under the terms of the GNU General Public License as published by 6 + * the Free Software Foundation; either version 2 of the License, or 7 + * (at your option) any later version. 8 + * 9 + * This program is distributed in the hope that it will be useful, 10 + * but WITHOUT ANY WARRANTY; without even the implied warranty of 11 + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 12 + * GNU General Public License for more details. 13 + * 14 + * You should have received a copy of the GNU General Public License 15 + * along with this program; if not, write to the Free Software 16 + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA 17 + * 18 + */ 19 + 20 + #include <linux/bitops.h> 21 + #include <linux/init.h> 22 + #include <linux/list.h> 23 + #include <linux/slab.h> 24 + #include <linux/string.h> 25 + #include <linux/usb.h> 26 + #include <linux/moduleparam.h> 27 + #include <linux/mutex.h> 28 + #include <linux/usb/audio.h> 29 + #include <linux/usb/audio-v2.h> 30 + 31 + #include <sound/core.h> 32 + #include <sound/info.h> 33 + #include <sound/pcm.h> 34 + #include <sound/pcm_params.h> 35 + #include <sound/initval.h> 36 + 37 + #include "usbaudio.h" 38 + #include "card.h" 39 + #include "midi.h" 40 + #include "mixer.h" 41 + #include "proc.h" 42 + #include "quirks.h" 43 + #include "endpoint.h" 44 + #include "helper.h" 45 + #include "debug.h" 46 + #include "pcm.h" 47 + #include "urb.h" 48 + #include "format.h" 49 + 50 + static struct uac_clock_source_descriptor * 51 + snd_usb_find_clock_source(struct usb_host_interface *ctrl_iface, 52 + int clock_id) 53 + { 54 + struct uac_clock_source_descriptor *cs = NULL; 55 + 56 + while ((cs = snd_usb_find_csint_desc(ctrl_iface->extra, 57 + ctrl_iface->extralen, 58 + cs, UAC2_CLOCK_SOURCE))) { 59 + if (cs->bClockID == clock_id) 60 + return cs; 61 + } 62 + 63 + return NULL; 64 + } 65 + 66 + static struct uac_clock_selector_descriptor * 67 + snd_usb_find_clock_selector(struct usb_host_interface *ctrl_iface, 68 + int clock_id) 69 + { 70 + struct uac_clock_selector_descriptor *cs = NULL; 71 + 72 + while ((cs = snd_usb_find_csint_desc(ctrl_iface->extra, 73 + ctrl_iface->extralen, 74 + cs, UAC2_CLOCK_SELECTOR))) { 75 + if (cs->bClockID == clock_id) 76 + return cs; 77 + } 78 + 79 + return NULL; 80 + } 81 + 82 + static struct uac_clock_multiplier_descriptor * 83 + snd_usb_find_clock_multiplier(struct usb_host_interface *ctrl_iface, 84 + int clock_id) 85 + { 86 + struct uac_clock_multiplier_descriptor *cs = NULL; 87 + 88 + while ((cs = snd_usb_find_csint_desc(ctrl_iface->extra, 89 + ctrl_iface->extralen, 90 + cs, UAC2_CLOCK_MULTIPLIER))) { 91 + if (cs->bClockID == clock_id) 92 + return cs; 93 + } 94 + 95 + return NULL; 96 + } 97 + 98 + static int uac_clock_selector_get_val(struct snd_usb_audio *chip, int selector_id) 99 + { 100 + unsigned char buf; 101 + int ret; 102 + 103 + ret = snd_usb_ctl_msg(chip->dev, usb_rcvctrlpipe(chip->dev, 0), 104 + UAC2_CS_CUR, 105 + USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_IN, 106 + UAC2_CX_CLOCK_SELECTOR << 8, selector_id << 8, 107 + &buf, sizeof(buf), 1000); 108 + 109 + if (ret < 0) 110 + return ret; 111 + 112 + return buf; 113 + } 114 + 115 + static bool uac_clock_source_is_valid(struct snd_usb_audio *chip, int source_id) 116 + { 117 + int err; 118 + unsigned char data; 119 + struct usb_device *dev = chip->dev; 120 + 121 + err = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0), UAC2_CS_CUR, 122 + USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN, 123 + UAC2_CS_CONTROL_CLOCK_VALID << 8, source_id << 8, 124 + &data, sizeof(data), 1000); 125 + 126 + if (err < 0) { 127 + snd_printk(KERN_WARNING "%s(): cannot get clock validity for id %d\n", 128 + __func__, source_id); 129 + return err; 130 + } 131 + 132 + return !!data; 133 + } 134 + 135 + /* Try to find the clock source ID of a given clock entity */ 136 + 137 + static int __uac_clock_find_source(struct snd_usb_audio *chip, 138 + struct usb_host_interface *host_iface, 139 + int entity_id, unsigned long *visited) 140 + { 141 + struct uac_clock_source_descriptor *source; 142 + struct uac_clock_selector_descriptor *selector; 143 + struct uac_clock_multiplier_descriptor *multiplier; 144 + 145 + entity_id &= 0xff; 146 + 147 + if (test_and_set_bit(entity_id, visited)) { 148 + snd_printk(KERN_WARNING 149 + "%s(): recursive clock topology detected, id %d.\n", 150 + __func__, entity_id); 151 + return -EINVAL; 152 + } 153 + 154 + /* first, see if the ID we're looking for is a clock source already */ 155 + source = snd_usb_find_clock_source(host_iface, entity_id); 156 + if (source) 157 + return source->bClockID; 158 + 159 + selector = snd_usb_find_clock_selector(host_iface, entity_id); 160 + if (selector) { 161 + int ret; 162 + 163 + /* the entity ID we are looking for is a selector. 164 + * find out what it currently selects */ 165 + ret = uac_clock_selector_get_val(chip, selector->bClockID); 166 + if (ret < 0) 167 + return ret; 168 + 169 + if (ret > selector->bNrInPins || ret < 1) { 170 + printk(KERN_ERR 171 + "%s(): selector reported illegal value, id %d, ret %d\n", 172 + __func__, selector->bClockID, ret); 173 + 174 + return -EINVAL; 175 + } 176 + 177 + return __uac_clock_find_source(chip, host_iface, 178 + selector->baCSourceID[ret-1], 179 + visited); 180 + } 181 + 182 + /* FIXME: multipliers only act as pass-thru element for now */ 183 + multiplier = snd_usb_find_clock_multiplier(host_iface, entity_id); 184 + if (multiplier) 185 + return __uac_clock_find_source(chip, host_iface, 186 + multiplier->bCSourceID, visited); 187 + 188 + return -EINVAL; 189 + } 190 + 191 + int snd_usb_clock_find_source(struct snd_usb_audio *chip, 192 + struct usb_host_interface *host_iface, 193 + int entity_id) 194 + { 195 + DECLARE_BITMAP(visited, 256); 196 + memset(visited, 0, sizeof(visited)); 197 + return __uac_clock_find_source(chip, host_iface, entity_id, visited); 198 + } 199 + 200 + static int set_sample_rate_v1(struct snd_usb_audio *chip, int iface, 201 + struct usb_host_interface *alts, 202 + struct audioformat *fmt, int rate) 203 + { 204 + struct usb_device *dev = chip->dev; 205 + unsigned int ep; 206 + unsigned char data[3]; 207 + int err, crate; 208 + 209 + ep = get_endpoint(alts, 0)->bEndpointAddress; 210 + 211 + /* if endpoint doesn't have sampling rate control, bail out */ 212 + if (!(fmt->attributes & UAC_EP_CS_ATTR_SAMPLE_RATE)) { 213 + snd_printk(KERN_WARNING "%d:%d:%d: endpoint lacks sample rate attribute bit, cannot set.\n", 214 + dev->devnum, iface, fmt->altsetting); 215 + return 0; 216 + } 217 + 218 + data[0] = rate; 219 + data[1] = rate >> 8; 220 + data[2] = rate >> 16; 221 + if ((err = snd_usb_ctl_msg(dev, usb_sndctrlpipe(dev, 0), UAC_SET_CUR, 222 + USB_TYPE_CLASS | USB_RECIP_ENDPOINT | USB_DIR_OUT, 223 + UAC_EP_CS_ATTR_SAMPLE_RATE << 8, ep, 224 + data, sizeof(data), 1000)) < 0) { 225 + snd_printk(KERN_ERR "%d:%d:%d: cannot set freq %d to ep %#x\n", 226 + dev->devnum, iface, fmt->altsetting, rate, ep); 227 + return err; 228 + } 229 + 230 + if ((err = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0), UAC_GET_CUR, 231 + USB_TYPE_CLASS | USB_RECIP_ENDPOINT | USB_DIR_IN, 232 + UAC_EP_CS_ATTR_SAMPLE_RATE << 8, ep, 233 + data, sizeof(data), 1000)) < 0) { 234 + snd_printk(KERN_WARNING "%d:%d:%d: cannot get freq at ep %#x\n", 235 + dev->devnum, iface, fmt->altsetting, ep); 236 + return 0; /* some devices don't support reading */ 237 + } 238 + 239 + crate = data[0] | (data[1] << 8) | (data[2] << 16); 240 + if (crate != rate) { 241 + snd_printd(KERN_WARNING "current rate %d is different from the runtime rate %d\n", crate, rate); 242 + // runtime->rate = crate; 243 + } 244 + 245 + return 0; 246 + } 247 + 248 + static int set_sample_rate_v2(struct snd_usb_audio *chip, int iface, 249 + struct usb_host_interface *alts, 250 + struct audioformat *fmt, int rate) 251 + { 252 + struct usb_device *dev = chip->dev; 253 + unsigned char data[4]; 254 + int err, crate; 255 + int clock = snd_usb_clock_find_source(chip, chip->ctrl_intf, fmt->clock); 256 + 257 + if (clock < 0) 258 + return clock; 259 + 260 + if (!uac_clock_source_is_valid(chip, clock)) { 261 + snd_printk(KERN_ERR "%d:%d:%d: clock source %d is not valid, cannot use\n", 262 + dev->devnum, iface, fmt->altsetting, clock); 263 + return -ENXIO; 264 + } 265 + 266 + data[0] = rate; 267 + data[1] = rate >> 8; 268 + data[2] = rate >> 16; 269 + data[3] = rate >> 24; 270 + if ((err = snd_usb_ctl_msg(dev, usb_sndctrlpipe(dev, 0), UAC2_CS_CUR, 271 + USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_OUT, 272 + UAC2_CS_CONTROL_SAM_FREQ << 8, clock << 8, 273 + data, sizeof(data), 1000)) < 0) { 274 + snd_printk(KERN_ERR "%d:%d:%d: cannot set freq %d (v2)\n", 275 + dev->devnum, iface, fmt->altsetting, rate); 276 + return err; 277 + } 278 + 279 + if ((err = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0), UAC2_CS_CUR, 280 + USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN, 281 + UAC2_CS_CONTROL_SAM_FREQ << 8, clock << 8, 282 + data, sizeof(data), 1000)) < 0) { 283 + snd_printk(KERN_WARNING "%d:%d:%d: cannot get freq (v2)\n", 284 + dev->devnum, iface, fmt->altsetting); 285 + return err; 286 + } 287 + 288 + crate = data[0] | (data[1] << 8) | (data[2] << 16) | (data[3] << 24); 289 + if (crate != rate) 290 + snd_printd(KERN_WARNING "current rate %d is different from the runtime rate %d\n", crate, rate); 291 + 292 + return 0; 293 + } 294 + 295 + int snd_usb_init_sample_rate(struct snd_usb_audio *chip, int iface, 296 + struct usb_host_interface *alts, 297 + struct audioformat *fmt, int rate) 298 + { 299 + struct usb_interface_descriptor *altsd = get_iface_desc(alts); 300 + 301 + switch (altsd->bInterfaceProtocol) { 302 + case UAC_VERSION_1: 303 + return set_sample_rate_v1(chip, iface, alts, fmt, rate); 304 + 305 + case UAC_VERSION_2: 306 + return set_sample_rate_v2(chip, iface, alts, fmt, rate); 307 + } 308 + 309 + return -EINVAL; 310 + } 311 +
+12
sound/usb/clock.h
··· 1 + #ifndef __USBAUDIO_CLOCK_H 2 + #define __USBAUDIO_CLOCK_H 3 + 4 + int snd_usb_init_sample_rate(struct snd_usb_audio *chip, int iface, 5 + struct usb_host_interface *alts, 6 + struct audioformat *fmt, int rate); 7 + 8 + int snd_usb_clock_find_source(struct snd_usb_audio *chip, 9 + struct usb_host_interface *host_iface, 10 + int entity_id); 11 + 12 + #endif /* __USBAUDIO_CLOCK_H */
+55 -2
sound/usb/endpoint.c
··· 190 190 return attributes; 191 191 } 192 192 193 + static struct uac2_input_terminal_descriptor * 194 + snd_usb_find_input_terminal_descriptor(struct usb_host_interface *ctrl_iface, 195 + int terminal_id) 196 + { 197 + struct uac2_input_terminal_descriptor *term = NULL; 198 + 199 + while ((term = snd_usb_find_csint_desc(ctrl_iface->extra, 200 + ctrl_iface->extralen, 201 + term, UAC_INPUT_TERMINAL))) { 202 + if (term->bTerminalID == terminal_id) 203 + return term; 204 + } 205 + 206 + return NULL; 207 + } 208 + 209 + static struct uac2_output_terminal_descriptor * 210 + snd_usb_find_output_terminal_descriptor(struct usb_host_interface *ctrl_iface, 211 + int terminal_id) 212 + { 213 + struct uac2_output_terminal_descriptor *term = NULL; 214 + 215 + while ((term = snd_usb_find_csint_desc(ctrl_iface->extra, 216 + ctrl_iface->extralen, 217 + term, UAC_OUTPUT_TERMINAL))) { 218 + if (term->bTerminalID == terminal_id) 219 + return term; 220 + } 221 + 222 + return NULL; 223 + } 224 + 193 225 int snd_usb_parse_audio_endpoints(struct snd_usb_audio *chip, int iface_no) 194 226 { 195 227 struct usb_device *dev; ··· 231 199 int i, altno, err, stream; 232 200 int format = 0, num_channels = 0; 233 201 struct audioformat *fp = NULL; 234 - int num, protocol; 202 + int num, protocol, clock = 0; 235 203 struct uac_format_type_i_continuous_descriptor *fmt; 236 204 237 205 dev = chip->dev; ··· 295 263 } 296 264 297 265 case UAC_VERSION_2: { 266 + struct uac2_input_terminal_descriptor *input_term; 267 + struct uac2_output_terminal_descriptor *output_term; 298 268 struct uac_as_header_descriptor_v2 *as = 299 269 snd_usb_find_csint_desc(alts->extra, alts->extralen, NULL, UAC_AS_GENERAL); 300 270 ··· 315 281 num_channels = as->bNrChannels; 316 282 format = le32_to_cpu(as->bmFormats); 317 283 318 - break; 284 + /* lookup the terminal associated to this interface 285 + * to extract the clock */ 286 + input_term = snd_usb_find_input_terminal_descriptor(chip->ctrl_intf, 287 + as->bTerminalLink); 288 + if (input_term) { 289 + clock = input_term->bCSourceID; 290 + break; 291 + } 292 + 293 + output_term = snd_usb_find_output_terminal_descriptor(chip->ctrl_intf, 294 + as->bTerminalLink); 295 + if (output_term) { 296 + clock = output_term->bCSourceID; 297 + break; 298 + } 299 + 300 + snd_printk(KERN_ERR "%d:%u:%d : bogus bTerminalLink %d\n", 301 + dev->devnum, iface_no, altno, as->bTerminalLink); 302 + continue; 319 303 } 320 304 321 305 default: ··· 390 338 fp->maxpacksize = (((fp->maxpacksize >> 11) & 3) + 1) 391 339 * (fp->maxpacksize & 0x7ff); 392 340 fp->attributes = parse_uac_endpoint_attributes(chip, alts, protocol, iface_no); 341 + fp->clock = clock; 393 342 394 343 /* some quirks for attributes here */ 395 344
+10 -6
sound/usb/format.c
··· 29 29 #include "quirks.h" 30 30 #include "helper.h" 31 31 #include "debug.h" 32 + #include "clock.h" 32 33 33 34 /* 34 35 * parse the audio format type I descriptor ··· 216 215 struct usb_device *dev = chip->dev; 217 216 unsigned char tmp[2], *data; 218 217 int i, nr_rates, data_size, ret = 0; 218 + int clock = snd_usb_clock_find_source(chip, chip->ctrl_intf, fp->clock); 219 219 220 220 /* get the number of sample rates first by only fetching 2 bytes */ 221 221 ret = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0), UAC2_CS_RANGE, 222 222 USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN, 223 - UAC2_CS_CONTROL_SAM_FREQ << 8, chip->clock_id << 8, 223 + UAC2_CS_CONTROL_SAM_FREQ << 8, clock << 8, 224 224 tmp, sizeof(tmp), 1000); 225 225 226 226 if (ret < 0) { 227 - snd_printk(KERN_ERR "unable to retrieve number of sample rates\n"); 227 + snd_printk(KERN_ERR "%s(): unable to retrieve number of sample rates (clock %d)\n", 228 + __func__, clock); 228 229 goto err; 229 230 } 230 231 ··· 240 237 241 238 /* now get the full information */ 242 239 ret = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0), UAC2_CS_RANGE, 243 - USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN, 244 - UAC2_CS_CONTROL_SAM_FREQ << 8, chip->clock_id << 8, 245 - data, data_size, 1000); 240 + USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN, 241 + UAC2_CS_CONTROL_SAM_FREQ << 8, clock << 8, 242 + data, data_size, 1000); 246 243 247 244 if (ret < 0) { 248 - snd_printk(KERN_ERR "unable to retrieve sample rate range\n"); 245 + snd_printk(KERN_ERR "%s(): unable to retrieve sample rate range (clock %d)\n", 246 + __func__, clock); 249 247 ret = -EINVAL; 250 248 goto err_free; 251 249 }
+116 -97
sound/usb/mixer.c
··· 78 78 USB_MIXER_U16, 79 79 }; 80 80 81 - enum { 82 - USB_PROC_UPDOWN = 1, 83 - USB_PROC_UPDOWN_SWITCH = 1, 84 - USB_PROC_UPDOWN_MODE_SEL = 2, 85 - 86 - USB_PROC_PROLOGIC = 2, 87 - USB_PROC_PROLOGIC_SWITCH = 1, 88 - USB_PROC_PROLOGIC_MODE_SEL = 2, 89 - 90 - USB_PROC_3DENH = 3, 91 - USB_PROC_3DENH_SWITCH = 1, 92 - USB_PROC_3DENH_SPACE = 2, 93 - 94 - USB_PROC_REVERB = 4, 95 - USB_PROC_REVERB_SWITCH = 1, 96 - USB_PROC_REVERB_LEVEL = 2, 97 - USB_PROC_REVERB_TIME = 3, 98 - USB_PROC_REVERB_DELAY = 4, 99 - 100 - USB_PROC_CHORUS = 5, 101 - USB_PROC_CHORUS_SWITCH = 1, 102 - USB_PROC_CHORUS_LEVEL = 2, 103 - USB_PROC_CHORUS_RATE = 3, 104 - USB_PROC_CHORUS_DEPTH = 4, 105 - 106 - USB_PROC_DCR = 6, 107 - USB_PROC_DCR_SWITCH = 1, 108 - USB_PROC_DCR_RATIO = 2, 109 - USB_PROC_DCR_MAX_AMP = 3, 110 - USB_PROC_DCR_THRESHOLD = 4, 111 - USB_PROC_DCR_ATTACK = 5, 112 - USB_PROC_DCR_RELEASE = 6, 113 - }; 114 81 115 82 /*E-mu 0202(0404) eXtension Unit(XU) control*/ 116 83 enum { ··· 165 198 166 199 /* 167 200 * find an audio control unit with the given unit id 168 - * this doesn't return any clock related units, so they need to be handled elsewhere 169 201 */ 170 202 static void *find_audio_control_unit(struct mixer_build *state, unsigned char unit) 171 203 { 172 - unsigned char *p; 204 + /* we just parse the header */ 205 + struct uac_feature_unit_descriptor *hdr = NULL; 173 206 174 - p = NULL; 175 - while ((p = snd_usb_find_desc(state->buffer, state->buflen, p, 176 - USB_DT_CS_INTERFACE)) != NULL) { 177 - if (p[0] >= 4 && p[2] >= UAC_INPUT_TERMINAL && p[2] <= UAC2_EXTENSION_UNIT_V2 && p[3] == unit) 178 - return p; 207 + while ((hdr = snd_usb_find_desc(state->buffer, state->buflen, hdr, 208 + USB_DT_CS_INTERFACE)) != NULL) { 209 + if (hdr->bLength >= 4 && 210 + hdr->bDescriptorSubtype >= UAC_INPUT_TERMINAL && 211 + hdr->bDescriptorSubtype <= UAC2_SAMPLE_RATE_CONVERTER && 212 + hdr->bUnitID == unit) 213 + return hdr; 179 214 } 215 + 180 216 return NULL; 181 217 } 182 - 183 218 184 219 /* 185 220 * copy a string with the given id ··· 313 344 buf, sizeof(buf), 1000); 314 345 315 346 if (ret < 0) { 316 - snd_printdd(KERN_ERR "cannot get ctl value: req = %#x, wValue = %#x, wIndex = %#x, type = %d\n", 317 - request, validx, cval->mixer->ctrlif | (cval->id << 8), cval->val_type); 347 + snd_printk(KERN_ERR "cannot get ctl value: req = %#x, wValue = %#x, wIndex = %#x, type = %d\n", 348 + request, validx, cval->mixer->ctrlif | (cval->id << 8), cval->val_type); 318 349 return ret; 319 350 } 320 351 ··· 431 462 int index, int value) 432 463 { 433 464 int err; 465 + unsigned int read_only = (channel == 0) ? 466 + cval->master_readonly : 467 + cval->ch_readonly & (1 << (channel - 1)); 468 + 469 + if (read_only) { 470 + snd_printdd(KERN_INFO "%s(): channel %d of control %d is read_only\n", 471 + __func__, channel, cval->control); 472 + return 0; 473 + } 474 + 434 475 err = snd_usb_mixer_set_ctl_value(cval, UAC_SET_CUR, (cval->control << 8) | channel, 435 476 value); 436 477 if (err < 0) ··· 610 631 */ 611 632 static int check_input_term(struct mixer_build *state, int id, struct usb_audio_term *term) 612 633 { 634 + int err; 613 635 void *p1; 614 636 615 637 memset(term, 0, sizeof(*term)); ··· 631 651 term->channels = d->bNrChannels; 632 652 term->chconfig = le32_to_cpu(d->bmChannelConfig); 633 653 term->name = d->iTerminal; 654 + 655 + /* call recursively to get the clock selectors */ 656 + err = check_input_term(state, d->bCSourceID, term); 657 + if (err < 0) 658 + return err; 634 659 } 635 660 return 0; 636 661 case UAC_FEATURE_UNIT: { ··· 652 667 term->name = uac_mixer_unit_iMixer(d); 653 668 return 0; 654 669 } 655 - case UAC_SELECTOR_UNIT: { 670 + case UAC_SELECTOR_UNIT: 671 + case UAC2_CLOCK_SELECTOR: { 656 672 struct uac_selector_unit_descriptor *d = p1; 657 673 /* call recursively to retrieve the channel info */ 658 674 if (check_input_term(state, d->baSourceID[0], term) < 0) ··· 676 690 term->name = uac_processing_unit_iProcessing(d, state->mixer->protocol); 677 691 return 0; 678 692 } 693 + case UAC2_CLOCK_SOURCE: { 694 + struct uac_clock_source_descriptor *d = p1; 695 + term->type = d->bDescriptorSubtype << 16; /* virtual type */ 696 + term->id = id; 697 + term->name = d->iClockSource; 698 + return 0; 699 + } 679 700 default: 680 701 return -ENODEV; 681 702 } ··· 702 709 }; 703 710 704 711 static struct usb_feature_control_info audio_feature_info[] = { 705 - { "Mute", USB_MIXER_INV_BOOLEAN }, 706 - { "Volume", USB_MIXER_S16 }, 712 + { "Mute", USB_MIXER_INV_BOOLEAN }, 713 + { "Volume", USB_MIXER_S16 }, 707 714 { "Tone Control - Bass", USB_MIXER_S8 }, 708 715 { "Tone Control - Mid", USB_MIXER_S8 }, 709 716 { "Tone Control - Treble", USB_MIXER_S8 }, 710 717 { "Graphic Equalizer", USB_MIXER_S8 }, /* FIXME: not implemeted yet */ 711 - { "Auto Gain Control", USB_MIXER_BOOLEAN }, 712 - { "Delay Control", USB_MIXER_U16 }, 713 - { "Bass Boost", USB_MIXER_BOOLEAN }, 714 - { "Loudness", USB_MIXER_BOOLEAN }, 718 + { "Auto Gain Control", USB_MIXER_BOOLEAN }, 719 + { "Delay Control", USB_MIXER_U16 }, 720 + { "Bass Boost", USB_MIXER_BOOLEAN }, 721 + { "Loudness", USB_MIXER_BOOLEAN }, 722 + /* UAC2 specific */ 723 + { "Input Gain Control", USB_MIXER_U16 }, 724 + { "Input Gain Pad Control", USB_MIXER_BOOLEAN }, 725 + { "Phase Inverter Control", USB_MIXER_BOOLEAN }, 715 726 }; 716 727 717 728 ··· 955 958 static void build_feature_ctl(struct mixer_build *state, void *raw_desc, 956 959 unsigned int ctl_mask, int control, 957 960 struct usb_audio_term *iterm, int unitid, 958 - int read_only) 961 + int readonly_mask) 959 962 { 960 963 struct uac_feature_unit_descriptor *desc = raw_desc; 961 964 unsigned int len = 0; ··· 967 970 968 971 control++; /* change from zero-based to 1-based value */ 969 972 970 - if (control == UAC_GRAPHIC_EQUALIZER_CONTROL) { 973 + if (control == UAC_FU_GRAPHIC_EQUALIZER) { 971 974 /* FIXME: not supported yet */ 972 975 return; 973 976 } ··· 986 989 cval->control = control; 987 990 cval->cmask = ctl_mask; 988 991 cval->val_type = audio_feature_info[control-1].type; 989 - if (ctl_mask == 0) 992 + if (ctl_mask == 0) { 990 993 cval->channels = 1; /* master channel */ 991 - else { 994 + cval->master_readonly = readonly_mask; 995 + } else { 992 996 int i, c = 0; 993 997 for (i = 0; i < 16; i++) 994 998 if (ctl_mask & (1 << i)) 995 999 c++; 996 1000 cval->channels = c; 1001 + cval->ch_readonly = readonly_mask; 997 1002 } 998 1003 999 1004 /* get min/max values */ 1000 1005 get_min_max(cval, 0); 1001 1006 1002 - if (read_only) 1007 + /* if all channels in the mask are marked read-only, make the control 1008 + * read-only. set_cur_mix_value() will check the mask again and won't 1009 + * issue write commands to read-only channels. */ 1010 + if (cval->channels == readonly_mask) 1003 1011 kctl = snd_ctl_new1(&usb_feature_unit_ctl_ro, cval); 1004 1012 else 1005 1013 kctl = snd_ctl_new1(&usb_feature_unit_ctl, cval); ··· 1023 1021 kctl->id.name, sizeof(kctl->id.name)); 1024 1022 1025 1023 switch (control) { 1026 - case UAC_MUTE_CONTROL: 1027 - case UAC_VOLUME_CONTROL: 1024 + case UAC_FU_MUTE: 1025 + case UAC_FU_VOLUME: 1028 1026 /* determine the control name. the rule is: 1029 1027 * - if a name id is given in descriptor, use it. 1030 1028 * - if the connected input can be determined, then use the name ··· 1051 1049 len = append_ctl_name(kctl, " Playback"); 1052 1050 } 1053 1051 } 1054 - append_ctl_name(kctl, control == UAC_MUTE_CONTROL ? 1052 + append_ctl_name(kctl, control == UAC_FU_MUTE ? 1055 1053 " Switch" : " Volume"); 1056 - if (control == UAC_VOLUME_CONTROL) { 1054 + if (control == UAC_FU_VOLUME) { 1057 1055 kctl->tlv.c = mixer_vol_tlv; 1058 1056 kctl->vd[0].access |= 1059 1057 SNDRV_CTL_ELEM_ACCESS_TLV_READ | ··· 1152 1150 snd_printk(KERN_INFO 1153 1151 "usbmixer: master volume quirk for PCM2702 chip\n"); 1154 1152 /* disable non-functional volume control */ 1155 - master_bits &= ~UAC_FU_VOLUME; 1153 + master_bits &= ~UAC_CONTROL_BIT(UAC_FU_VOLUME); 1156 1154 break; 1157 1155 } 1158 1156 if (channels > 0) ··· 1190 1188 1191 1189 for (j = 0; j < channels; j++) { 1192 1190 unsigned int mask = snd_usb_combine_bytes(bmaControls + csize * (j+1), csize); 1193 - if (mask & (1 << (i * 2))) { 1191 + if (uac2_control_is_readable(mask, i)) { 1194 1192 ch_bits |= (1 << j); 1195 - if (~mask & (1 << ((i * 2) + 1))) 1193 + if (!uac2_control_is_writeable(mask, i)) 1196 1194 ch_read_only |= (1 << j); 1197 1195 } 1198 1196 } 1199 1197 1200 - /* FIXME: the whole unit is read-only if any of the channels is marked read-only */ 1198 + /* NOTE: build_feature_ctl() will mark the control read-only if all channels 1199 + * are marked read-only in the descriptors. Otherwise, the control will be 1200 + * reported as writeable, but the driver will not actually issue a write 1201 + * command for read-only channels */ 1201 1202 if (ch_bits & 1) /* the first channel must be set (for ease of programming) */ 1202 - build_feature_ctl(state, _ftr, ch_bits, i, &iterm, unitid, !!ch_read_only); 1203 - if (master_bits & (1 << i * 2)) 1203 + build_feature_ctl(state, _ftr, ch_bits, i, &iterm, unitid, ch_read_only); 1204 + if (uac2_control_is_readable(master_bits, i)) 1204 1205 build_feature_ctl(state, _ftr, 0, i, &iterm, unitid, 1205 - ~master_bits & (1 << ((i * 2) + 1))); 1206 + !uac2_control_is_writeable(master_bits, i)); 1206 1207 } 1207 1208 } 1208 1209 ··· 1397 1392 }; 1398 1393 1399 1394 static struct procunit_value_info updown_proc_info[] = { 1400 - { USB_PROC_UPDOWN_SWITCH, "Switch", USB_MIXER_BOOLEAN }, 1401 - { USB_PROC_UPDOWN_MODE_SEL, "Mode Select", USB_MIXER_U8, 1 }, 1395 + { UAC_UD_ENABLE, "Switch", USB_MIXER_BOOLEAN }, 1396 + { UAC_UD_MODE_SELECT, "Mode Select", USB_MIXER_U8, 1 }, 1402 1397 { 0 } 1403 1398 }; 1404 1399 static struct procunit_value_info prologic_proc_info[] = { 1405 - { USB_PROC_PROLOGIC_SWITCH, "Switch", USB_MIXER_BOOLEAN }, 1406 - { USB_PROC_PROLOGIC_MODE_SEL, "Mode Select", USB_MIXER_U8, 1 }, 1400 + { UAC_DP_ENABLE, "Switch", USB_MIXER_BOOLEAN }, 1401 + { UAC_DP_MODE_SELECT, "Mode Select", USB_MIXER_U8, 1 }, 1407 1402 { 0 } 1408 1403 }; 1409 1404 static struct procunit_value_info threed_enh_proc_info[] = { 1410 - { USB_PROC_3DENH_SWITCH, "Switch", USB_MIXER_BOOLEAN }, 1411 - { USB_PROC_3DENH_SPACE, "Spaciousness", USB_MIXER_U8 }, 1405 + { UAC_3D_ENABLE, "Switch", USB_MIXER_BOOLEAN }, 1406 + { UAC_3D_SPACE, "Spaciousness", USB_MIXER_U8 }, 1412 1407 { 0 } 1413 1408 }; 1414 1409 static struct procunit_value_info reverb_proc_info[] = { 1415 - { USB_PROC_REVERB_SWITCH, "Switch", USB_MIXER_BOOLEAN }, 1416 - { USB_PROC_REVERB_LEVEL, "Level", USB_MIXER_U8 }, 1417 - { USB_PROC_REVERB_TIME, "Time", USB_MIXER_U16 }, 1418 - { USB_PROC_REVERB_DELAY, "Delay", USB_MIXER_U8 }, 1410 + { UAC_REVERB_ENABLE, "Switch", USB_MIXER_BOOLEAN }, 1411 + { UAC_REVERB_LEVEL, "Level", USB_MIXER_U8 }, 1412 + { UAC_REVERB_TIME, "Time", USB_MIXER_U16 }, 1413 + { UAC_REVERB_FEEDBACK, "Feedback", USB_MIXER_U8 }, 1419 1414 { 0 } 1420 1415 }; 1421 1416 static struct procunit_value_info chorus_proc_info[] = { 1422 - { USB_PROC_CHORUS_SWITCH, "Switch", USB_MIXER_BOOLEAN }, 1423 - { USB_PROC_CHORUS_LEVEL, "Level", USB_MIXER_U8 }, 1424 - { USB_PROC_CHORUS_RATE, "Rate", USB_MIXER_U16 }, 1425 - { USB_PROC_CHORUS_DEPTH, "Depth", USB_MIXER_U16 }, 1417 + { UAC_CHORUS_ENABLE, "Switch", USB_MIXER_BOOLEAN }, 1418 + { UAC_CHORUS_LEVEL, "Level", USB_MIXER_U8 }, 1419 + { UAC_CHORUS_RATE, "Rate", USB_MIXER_U16 }, 1420 + { UAC_CHORUS_DEPTH, "Depth", USB_MIXER_U16 }, 1426 1421 { 0 } 1427 1422 }; 1428 1423 static struct procunit_value_info dcr_proc_info[] = { 1429 - { USB_PROC_DCR_SWITCH, "Switch", USB_MIXER_BOOLEAN }, 1430 - { USB_PROC_DCR_RATIO, "Ratio", USB_MIXER_U16 }, 1431 - { USB_PROC_DCR_MAX_AMP, "Max Amp", USB_MIXER_S16 }, 1432 - { USB_PROC_DCR_THRESHOLD, "Threshold", USB_MIXER_S16 }, 1433 - { USB_PROC_DCR_ATTACK, "Attack Time", USB_MIXER_U16 }, 1434 - { USB_PROC_DCR_RELEASE, "Release Time", USB_MIXER_U16 }, 1424 + { UAC_DCR_ENABLE, "Switch", USB_MIXER_BOOLEAN }, 1425 + { UAC_DCR_RATE, "Ratio", USB_MIXER_U16 }, 1426 + { UAC_DCR_MAXAMPL, "Max Amp", USB_MIXER_S16 }, 1427 + { UAC_DCR_THRESHOLD, "Threshold", USB_MIXER_S16 }, 1428 + { UAC_DCR_ATTACK_TIME, "Attack Time", USB_MIXER_U16 }, 1429 + { UAC_DCR_RELEASE_TIME, "Release Time", USB_MIXER_U16 }, 1435 1430 { 0 } 1436 1431 }; 1437 1432 1438 1433 static struct procunit_info procunits[] = { 1439 - { USB_PROC_UPDOWN, "Up Down", updown_proc_info }, 1440 - { USB_PROC_PROLOGIC, "Dolby Prologic", prologic_proc_info }, 1441 - { USB_PROC_3DENH, "3D Stereo Extender", threed_enh_proc_info }, 1442 - { USB_PROC_REVERB, "Reverb", reverb_proc_info }, 1443 - { USB_PROC_CHORUS, "Chorus", chorus_proc_info }, 1444 - { USB_PROC_DCR, "DCR", dcr_proc_info }, 1434 + { UAC_PROCESS_UP_DOWNMIX, "Up Down", updown_proc_info }, 1435 + { UAC_PROCESS_DOLBY_PROLOGIC, "Dolby Prologic", prologic_proc_info }, 1436 + { UAC_PROCESS_STEREO_EXTENDER, "3D Stereo Extender", threed_enh_proc_info }, 1437 + { UAC_PROCESS_REVERB, "Reverb", reverb_proc_info }, 1438 + { UAC_PROCESS_CHORUS, "Chorus", chorus_proc_info }, 1439 + { UAC_PROCESS_DYN_RANGE_COMP, "DCR", dcr_proc_info }, 1445 1440 { 0 }, 1446 1441 }; 1447 1442 /* ··· 1529 1524 cval->channels = 1; 1530 1525 1531 1526 /* get min/max values */ 1532 - if (type == USB_PROC_UPDOWN && cval->control == USB_PROC_UPDOWN_MODE_SEL) { 1527 + if (type == UAC_PROCESS_UP_DOWNMIX && cval->control == UAC_UD_MODE_SELECT) { 1533 1528 __u8 *control_spec = uac_processing_unit_specific(desc, state->mixer->protocol); 1534 1529 /* FIXME: hard-coded */ 1535 1530 cval->min = 1; ··· 1624 1619 struct usb_mixer_elem_info *cval = kcontrol->private_data; 1625 1620 int val, err; 1626 1621 1627 - err = get_cur_ctl_value(cval, 0, &val); 1622 + err = get_cur_ctl_value(cval, cval->control << 8, &val); 1628 1623 if (err < 0) { 1629 1624 if (cval->mixer->ignore_ctl_error) { 1630 1625 ucontrol->value.enumerated.item[0] = 0; ··· 1643 1638 struct usb_mixer_elem_info *cval = kcontrol->private_data; 1644 1639 int val, oval, err; 1645 1640 1646 - err = get_cur_ctl_value(cval, 0, &oval); 1641 + err = get_cur_ctl_value(cval, cval->control << 8, &oval); 1647 1642 if (err < 0) { 1648 1643 if (cval->mixer->ignore_ctl_error) 1649 1644 return 0; ··· 1652 1647 val = ucontrol->value.enumerated.item[0]; 1653 1648 val = get_abs_value(cval, val); 1654 1649 if (val != oval) { 1655 - set_cur_ctl_value(cval, 0, val); 1650 + set_cur_ctl_value(cval, cval->control << 8, val); 1656 1651 return 1; 1657 1652 } 1658 1653 return 0; ··· 1734 1729 cval->res = 1; 1735 1730 cval->initialized = 1; 1736 1731 1732 + if (desc->bDescriptorSubtype == UAC2_CLOCK_SELECTOR) 1733 + cval->control = UAC2_CX_CLOCK_SELECTOR; 1734 + else 1735 + cval->control = 0; 1736 + 1737 1737 namelist = kmalloc(sizeof(char *) * desc->bNrInPins, GFP_KERNEL); 1738 1738 if (! namelist) { 1739 1739 snd_printk(KERN_ERR "cannot malloc\n"); ··· 1788 1778 if (! len) 1789 1779 strlcpy(kctl->id.name, "USB", sizeof(kctl->id.name)); 1790 1780 1791 - if ((state->oterm.type & 0xff00) == 0x0100) 1781 + if (desc->bDescriptorSubtype == UAC2_CLOCK_SELECTOR) 1782 + append_ctl_name(kctl, " Clock Source"); 1783 + else if ((state->oterm.type & 0xff00) == 0x0100) 1792 1784 append_ctl_name(kctl, " Capture Source"); 1793 1785 else 1794 1786 append_ctl_name(kctl, " Playback Source"); ··· 1824 1812 1825 1813 switch (p1[2]) { 1826 1814 case UAC_INPUT_TERMINAL: 1815 + case UAC2_CLOCK_SOURCE: 1827 1816 return 0; /* NOP */ 1828 1817 case UAC_MIXER_UNIT: 1829 1818 return parse_audio_mixer_unit(state, unitid, p1); 1830 1819 case UAC_SELECTOR_UNIT: 1820 + case UAC2_CLOCK_SELECTOR: 1831 1821 return parse_audio_selector_unit(state, unitid, p1); 1832 1822 case UAC_FEATURE_UNIT: 1833 1823 return parse_audio_feature_unit(state, unitid, p1); ··· 1924 1910 state.oterm.type = le16_to_cpu(desc->wTerminalType); 1925 1911 state.oterm.name = desc->iTerminal; 1926 1912 err = parse_audio_unit(&state, desc->bSourceID); 1913 + if (err < 0) 1914 + return err; 1915 + 1916 + /* for UAC2, use the same approach to also add the clock selectors */ 1917 + err = parse_audio_unit(&state, desc->bCSourceID); 1927 1918 if (err < 0) 1928 1919 return err; 1929 1920 }
+2
sound/usb/mixer.h
··· 34 34 unsigned int id; 35 35 unsigned int control; /* CS or ICN (high byte) */ 36 36 unsigned int cmask; /* channel mask bitmap: 0 = master */ 37 + unsigned int ch_readonly; 38 + unsigned int master_readonly; 37 39 int channels; 38 40 int val_type; 39 41 int min, max, res;
+2 -2
sound/usb/mixer_maps.c
··· 85 85 /* 16: MU (w/o controls) */ 86 86 { 17, NULL, 1 }, /* DISABLED: PU-switch (any effect?) */ 87 87 { 17, "Channel Routing", 2 }, /* PU: mode select */ 88 - { 18, "Tone Control - Bass", UAC_BASS_CONTROL }, /* FU */ 89 - { 18, "Tone Control - Treble", UAC_TREBLE_CONTROL }, /* FU */ 88 + { 18, "Tone Control - Bass", UAC_FU_BASS }, /* FU */ 89 + { 18, "Tone Control - Treble", UAC_FU_TREBLE }, /* FU */ 90 90 { 18, "Master Playback" }, /* FU; others */ 91 91 /* 19: OT speaker */ 92 92 /* 20: OT headphone */
+1 -97
sound/usb/pcm.c
··· 31 31 #include "urb.h" 32 32 #include "helper.h" 33 33 #include "pcm.h" 34 + #include "clock.h" 34 35 35 36 /* 36 37 * return the current pcm pointer. just based on the hwptr_done value. ··· 177 176 178 177 case UAC_VERSION_2: 179 178 return init_pitch_v2(chip, iface, alts, fmt); 180 - } 181 - 182 - return -EINVAL; 183 - } 184 - 185 - static int set_sample_rate_v1(struct snd_usb_audio *chip, int iface, 186 - struct usb_host_interface *alts, 187 - struct audioformat *fmt, int rate) 188 - { 189 - struct usb_device *dev = chip->dev; 190 - unsigned int ep; 191 - unsigned char data[3]; 192 - int err, crate; 193 - 194 - ep = get_endpoint(alts, 0)->bEndpointAddress; 195 - /* if endpoint doesn't have sampling rate control, bail out */ 196 - if (!(fmt->attributes & UAC_EP_CS_ATTR_SAMPLE_RATE)) { 197 - snd_printk(KERN_WARNING "%d:%d:%d: endpoint lacks sample rate attribute bit, cannot set.\n", 198 - dev->devnum, iface, fmt->altsetting); 199 - return 0; 200 - } 201 - 202 - data[0] = rate; 203 - data[1] = rate >> 8; 204 - data[2] = rate >> 16; 205 - if ((err = snd_usb_ctl_msg(dev, usb_sndctrlpipe(dev, 0), UAC_SET_CUR, 206 - USB_TYPE_CLASS|USB_RECIP_ENDPOINT|USB_DIR_OUT, 207 - UAC_EP_CS_ATTR_SAMPLE_RATE << 8, ep, 208 - data, sizeof(data), 1000)) < 0) { 209 - snd_printk(KERN_ERR "%d:%d:%d: cannot set freq %d to ep %#x\n", 210 - dev->devnum, iface, fmt->altsetting, rate, ep); 211 - return err; 212 - } 213 - if ((err = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0), UAC_GET_CUR, 214 - USB_TYPE_CLASS|USB_RECIP_ENDPOINT|USB_DIR_IN, 215 - UAC_EP_CS_ATTR_SAMPLE_RATE << 8, ep, 216 - data, sizeof(data), 1000)) < 0) { 217 - snd_printk(KERN_WARNING "%d:%d:%d: cannot get freq at ep %#x\n", 218 - dev->devnum, iface, fmt->altsetting, ep); 219 - return 0; /* some devices don't support reading */ 220 - } 221 - crate = data[0] | (data[1] << 8) | (data[2] << 16); 222 - if (crate != rate) { 223 - snd_printd(KERN_WARNING "current rate %d is different from the runtime rate %d\n", crate, rate); 224 - // runtime->rate = crate; 225 - } 226 - 227 - return 0; 228 - } 229 - 230 - static int set_sample_rate_v2(struct snd_usb_audio *chip, int iface, 231 - struct usb_host_interface *alts, 232 - struct audioformat *fmt, int rate) 233 - { 234 - struct usb_device *dev = chip->dev; 235 - unsigned char data[4]; 236 - int err, crate; 237 - 238 - data[0] = rate; 239 - data[1] = rate >> 8; 240 - data[2] = rate >> 16; 241 - data[3] = rate >> 24; 242 - if ((err = snd_usb_ctl_msg(dev, usb_sndctrlpipe(dev, 0), UAC2_CS_CUR, 243 - USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_OUT, 244 - UAC2_CS_CONTROL_SAM_FREQ << 8, chip->clock_id << 8, 245 - data, sizeof(data), 1000)) < 0) { 246 - snd_printk(KERN_ERR "%d:%d:%d: cannot set freq %d (v2)\n", 247 - dev->devnum, iface, fmt->altsetting, rate); 248 - return err; 249 - } 250 - if ((err = snd_usb_ctl_msg(dev, usb_rcvctrlpipe(dev, 0), UAC2_CS_CUR, 251 - USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN, 252 - UAC2_CS_CONTROL_SAM_FREQ << 8, chip->clock_id << 8, 253 - data, sizeof(data), 1000)) < 0) { 254 - snd_printk(KERN_WARNING "%d:%d:%d: cannot get freq (v2)\n", 255 - dev->devnum, iface, fmt->altsetting); 256 - return err; 257 - } 258 - crate = data[0] | (data[1] << 8) | (data[2] << 16) | (data[3] << 24); 259 - if (crate != rate) 260 - snd_printd(KERN_WARNING "current rate %d is different from the runtime rate %d\n", crate, rate); 261 - 262 - return 0; 263 - } 264 - 265 - int snd_usb_init_sample_rate(struct snd_usb_audio *chip, int iface, 266 - struct usb_host_interface *alts, 267 - struct audioformat *fmt, int rate) 268 - { 269 - struct usb_interface_descriptor *altsd = get_iface_desc(alts); 270 - 271 - switch (altsd->bInterfaceProtocol) { 272 - case UAC_VERSION_1: 273 - return set_sample_rate_v1(chip, iface, alts, fmt, rate); 274 - 275 - case UAC_VERSION_2: 276 - return set_sample_rate_v2(chip, iface, alts, fmt, rate); 277 179 } 278 180 279 181 return -EINVAL;
+2 -3
sound/usb/usbaudio.h
··· 40 40 int num_interfaces; 41 41 int num_suspended_intf; 42 42 43 - /* for audio class v2 */ 44 - int clock_id; 45 - 46 43 struct list_head pcm_list; /* list of pcm streams */ 47 44 int pcm_devs; 48 45 ··· 50 53 int setup; /* from the 'device_setup' module param */ 51 54 int nrpacks; /* from the 'nrpacks' module param */ 52 55 int async_unlink; /* from the 'async_unlink' module param */ 56 + 57 + struct usb_host_interface *ctrl_intf; /* the audio control interface */ 53 58 }; 54 59 55 60 /*