2019-05-27 14:55:05 +08:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2010-03-05 02:46:13 +08:00
|
|
|
/*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/init.h>
|
2010-03-29 13:02:50 +08:00
|
|
|
#include <linux/slab.h>
|
2010-03-05 02:46:13 +08:00
|
|
|
#include <linux/usb.h>
|
|
|
|
|
|
|
|
#include "usbaudio.h"
|
|
|
|
#include "helper.h"
|
2012-09-04 16:23:07 +08:00
|
|
|
#include "quirks.h"
|
2010-03-05 02:46:13 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* combine bytes and get an integer value
|
|
|
|
*/
|
|
|
|
unsigned int snd_usb_combine_bytes(unsigned char *bytes, int size)
|
|
|
|
{
|
|
|
|
switch (size) {
|
|
|
|
case 1: return *bytes;
|
|
|
|
case 2: return combine_word(bytes);
|
|
|
|
case 3: return combine_triple(bytes);
|
|
|
|
case 4: return combine_quad(bytes);
|
|
|
|
default: return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* parse descriptor buffer and return the pointer starting the given
|
|
|
|
* descriptor type.
|
|
|
|
*/
|
|
|
|
void *snd_usb_find_desc(void *descstart, int desclen, void *after, u8 dtype)
|
|
|
|
{
|
|
|
|
u8 *p, *end, *next;
|
|
|
|
|
|
|
|
p = descstart;
|
|
|
|
end = p + desclen;
|
|
|
|
for (; p < end;) {
|
|
|
|
if (p[0] < 2)
|
|
|
|
return NULL;
|
|
|
|
next = p + p[0];
|
|
|
|
if (next > end)
|
|
|
|
return NULL;
|
|
|
|
if (p[1] == dtype && (!after || (void *)p > after)) {
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
p = next;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* find a class-specified interface descriptor with the given subtype.
|
|
|
|
*/
|
|
|
|
void *snd_usb_find_csint_desc(void *buffer, int buflen, void *after, u8 dsubtype)
|
|
|
|
{
|
|
|
|
unsigned char *p = after;
|
|
|
|
|
|
|
|
while ((p = snd_usb_find_desc(buffer, buflen, p,
|
|
|
|
USB_DT_CS_INTERFACE)) != NULL) {
|
|
|
|
if (p[0] >= 3 && p[2] == dsubtype)
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Wrapper for usb_control_msg().
|
|
|
|
* Allocates a temp buffer to prevent dmaing from/to the stack.
|
|
|
|
*/
|
|
|
|
int snd_usb_ctl_msg(struct usb_device *dev, unsigned int pipe, __u8 request,
|
|
|
|
__u8 requesttype, __u16 value, __u16 index, void *data,
|
2011-09-27 03:15:27 +08:00
|
|
|
__u16 size)
|
2010-03-05 02:46:13 +08:00
|
|
|
{
|
|
|
|
int err;
|
|
|
|
void *buf = NULL;
|
2013-04-21 05:06:17 +08:00
|
|
|
int timeout;
|
2010-03-05 02:46:13 +08:00
|
|
|
|
2020-09-14 23:37:46 +08:00
|
|
|
if (usb_pipe_type_check(dev, pipe))
|
2019-06-24 21:08:28 +08:00
|
|
|
return -EINVAL;
|
|
|
|
|
2010-03-05 02:46:13 +08:00
|
|
|
if (size > 0) {
|
|
|
|
buf = kmemdup(data, size, GFP_KERNEL);
|
|
|
|
if (!buf)
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
2013-04-21 05:06:17 +08:00
|
|
|
|
|
|
|
if (requesttype & USB_DIR_IN)
|
|
|
|
timeout = USB_CTRL_GET_TIMEOUT;
|
|
|
|
else
|
|
|
|
timeout = USB_CTRL_SET_TIMEOUT;
|
|
|
|
|
2010-03-05 02:46:13 +08:00
|
|
|
err = usb_control_msg(dev, pipe, request, requesttype,
|
2013-04-21 05:06:17 +08:00
|
|
|
value, index, buf, size, timeout);
|
|
|
|
|
2010-03-05 02:46:13 +08:00
|
|
|
if (size > 0) {
|
|
|
|
memcpy(data, buf, size);
|
|
|
|
kfree(buf);
|
|
|
|
}
|
2012-09-04 16:23:07 +08:00
|
|
|
|
|
|
|
snd_usb_ctl_msg_quirk(dev, pipe, request, requesttype,
|
|
|
|
value, index, data, size);
|
|
|
|
|
2010-03-05 02:46:13 +08:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned char snd_usb_parse_datainterval(struct snd_usb_audio *chip,
|
|
|
|
struct usb_host_interface *alts)
|
|
|
|
{
|
2010-08-14 03:42:07 +08:00
|
|
|
switch (snd_usb_get_speed(chip->dev)) {
|
|
|
|
case USB_SPEED_HIGH:
|
|
|
|
case USB_SPEED_SUPER:
|
2016-05-04 20:18:39 +08:00
|
|
|
case USB_SPEED_SUPER_PLUS:
|
2010-08-14 03:42:07 +08:00
|
|
|
if (get_endpoint(alts, 0)->bInterval >= 1 &&
|
|
|
|
get_endpoint(alts, 0)->bInterval <= 4)
|
|
|
|
return get_endpoint(alts, 0)->bInterval - 1;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
2010-03-05 02:46:13 +08:00
|
|
|
}
|
|
|
|
|
2020-11-23 16:53:22 +08:00
|
|
|
struct usb_host_interface *
|
|
|
|
snd_usb_get_host_interface(struct snd_usb_audio *chip, int ifnum, int altsetting)
|
|
|
|
{
|
|
|
|
struct usb_interface *iface;
|
|
|
|
|
|
|
|
iface = usb_ifnum_to_if(chip->dev, ifnum);
|
|
|
|
if (!iface)
|
|
|
|
return NULL;
|
|
|
|
return usb_altnum_to_altsetting(iface, altsetting);
|
|
|
|
}
|