mirror of
https://github.com/sakumisu/CherryUSB.git
synced 2025-05-09 00:21:44 +08:00
472 lines
17 KiB
C
472 lines
17 KiB
C
/*
|
|
* Copyright (c) 2022, sakumisu
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
#include "usbd_core.h"
|
|
#include "usbd_audio.h"
|
|
|
|
#if CONFIG_USBDEV_AUDIO_VERSION < 0x0200
|
|
struct usbd_audio_volume_info {
|
|
uint16_t vol_min;
|
|
uint16_t vol_max;
|
|
uint16_t vol_res;
|
|
uint16_t vol_current;
|
|
};
|
|
|
|
struct usbd_audio_attribute_control {
|
|
struct usbd_audio_volume_info volume[CONFIG_USBDEV_AUDIO_MAX_CHANNEL];
|
|
uint8_t mute[CONFIG_USBDEV_AUDIO_MAX_CHANNEL];
|
|
uint8_t automatic_gain[CONFIG_USBDEV_AUDIO_MAX_CHANNEL];
|
|
};
|
|
#else
|
|
struct audio_v2_control_range2_param_block_default {
|
|
uint16_t wNumSubRanges;
|
|
struct
|
|
{
|
|
uint16_t wMin;
|
|
uint16_t wMax;
|
|
uint16_t wRes;
|
|
} subrange[CONFIG_USBDEV_AUDIO_MAX_CHANNEL];
|
|
} __PACKED;
|
|
|
|
struct usbd_audio_attribute_control {
|
|
uint32_t volume_bCUR;
|
|
uint32_t mute_bCUR;
|
|
struct audio_v2_control_range2_param_block_default volume;
|
|
uint8_t mute[CONFIG_USBDEV_AUDIO_MAX_CHANNEL];
|
|
};
|
|
#endif
|
|
struct audio_entity_info {
|
|
usb_slist_t list;
|
|
uint8_t bDescriptorSubtype;
|
|
uint8_t bEntityId;
|
|
void *priv[2];
|
|
};
|
|
|
|
static usb_slist_t usbd_audio_entity_info_head = USB_SLIST_OBJECT_INIT(usbd_audio_entity_info_head);
|
|
|
|
#if CONFIG_USBDEV_AUDIO_VERSION >= 0x0200
|
|
const uint8_t default_sampling_freq_table[] = {
|
|
AUDIO_SAMPLE_FREQ_NUM(1),
|
|
// AUDIO_SAMPLE_FREQ_4B(8000),
|
|
// AUDIO_SAMPLE_FREQ_4B(8000),
|
|
// AUDIO_SAMPLE_FREQ_4B(0x00),
|
|
AUDIO_SAMPLE_FREQ_4B(16000),
|
|
AUDIO_SAMPLE_FREQ_4B(16000),
|
|
AUDIO_SAMPLE_FREQ_4B(0x00),
|
|
AUDIO_SAMPLE_FREQ_4B(32000),
|
|
AUDIO_SAMPLE_FREQ_4B(32000),
|
|
AUDIO_SAMPLE_FREQ_4B(0x00),
|
|
AUDIO_SAMPLE_FREQ_4B(44100),
|
|
AUDIO_SAMPLE_FREQ_4B(44100),
|
|
AUDIO_SAMPLE_FREQ_4B(0x00),
|
|
AUDIO_SAMPLE_FREQ_4B(48000),
|
|
AUDIO_SAMPLE_FREQ_4B(48000),
|
|
AUDIO_SAMPLE_FREQ_4B(0x00),
|
|
AUDIO_SAMPLE_FREQ_4B(88200),
|
|
AUDIO_SAMPLE_FREQ_4B(88200),
|
|
AUDIO_SAMPLE_FREQ_4B(0x00),
|
|
AUDIO_SAMPLE_FREQ_4B(96000),
|
|
AUDIO_SAMPLE_FREQ_4B(96000),
|
|
AUDIO_SAMPLE_FREQ_4B(0x00),
|
|
AUDIO_SAMPLE_FREQ_4B(192000),
|
|
AUDIO_SAMPLE_FREQ_4B(192000),
|
|
AUDIO_SAMPLE_FREQ_4B(0x00),
|
|
};
|
|
#endif
|
|
|
|
#if CONFIG_USBDEV_AUDIO_VERSION < 0x0200
|
|
static int audio_custom_request_handler(struct usb_setup_packet *setup, uint8_t **data, uint32_t *len)
|
|
{
|
|
uint8_t control_selector;
|
|
uint32_t sampling_freq = 0;
|
|
uint8_t pitch_enable;
|
|
uint8_t ep;
|
|
|
|
if ((setup->bmRequestType & USB_REQUEST_RECIPIENT_MASK) == USB_REQUEST_RECIPIENT_ENDPOINT) {
|
|
control_selector = HI_BYTE(setup->wValue);
|
|
ep = LO_BYTE(setup->wIndex);
|
|
|
|
switch (setup->bRequest) {
|
|
case AUDIO_REQUEST_SET_CUR:
|
|
switch (control_selector) {
|
|
case AUDIO_EP_CONTROL_SAMPLING_FEQ:
|
|
memcpy((uint8_t *)&sampling_freq, *data, *len);
|
|
USB_LOG_DBG("Set ep:%02x %d Hz\r\n", ep, (int)sampling_freq);
|
|
usbd_audio_set_sampling_freq(0, ep, sampling_freq);
|
|
break;
|
|
case AUDIO_EP_CONTROL_PITCH:
|
|
pitch_enable = (*data)[0];
|
|
usbd_audio_set_pitch(ep, pitch_enable);
|
|
break;
|
|
default:
|
|
USB_LOG_WRN("Unhandled Audio Class control selector 0x%02x\r\n", control_selector);
|
|
return -1;
|
|
}
|
|
break;
|
|
case AUDIO_REQUEST_GET_CUR:
|
|
sampling_freq = 16000;
|
|
memcpy(*data, &sampling_freq, 4);
|
|
*len = 4;
|
|
break;
|
|
default:
|
|
USB_LOG_WRN("Unhandled Audio Class bRequest 0x%02x\r\n", setup->bRequest);
|
|
return -1;
|
|
}
|
|
} else {
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
static int audio_class_request_handler(struct usb_setup_packet *setup, uint8_t **data, uint32_t *len)
|
|
{
|
|
USB_LOG_DBG("AUDIO Class request: "
|
|
"bRequest 0x%02x\r\n",
|
|
setup->bRequest);
|
|
|
|
struct audio_entity_info *current_entity_info = NULL;
|
|
struct usbd_audio_attribute_control *current_feature_control = NULL;
|
|
#if CONFIG_USBDEV_AUDIO_VERSION >= 0x0200
|
|
uint32_t *sampling_freq;
|
|
#endif
|
|
usb_slist_t *i;
|
|
uint8_t entity_id;
|
|
uint8_t control_selector;
|
|
uint8_t ch;
|
|
uint8_t mute;
|
|
uint16_t volume;
|
|
const char *mute_string[2] = { "off", "on" };
|
|
|
|
entity_id = HI_BYTE(setup->wIndex);
|
|
control_selector = HI_BYTE(setup->wValue);
|
|
ch = LO_BYTE(setup->wValue);
|
|
|
|
ARG_UNUSED(mute_string);
|
|
if (ch > (CONFIG_USBDEV_AUDIO_MAX_CHANNEL - 1)) {
|
|
return -2;
|
|
}
|
|
|
|
usb_slist_for_each(i, &usbd_audio_entity_info_head)
|
|
{
|
|
struct audio_entity_info *tmp_entity_info = usb_slist_entry(i, struct audio_entity_info, list);
|
|
if (tmp_entity_info->bEntityId == entity_id) {
|
|
current_entity_info = tmp_entity_info;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (current_entity_info == NULL) {
|
|
return -2;
|
|
}
|
|
|
|
current_feature_control = (struct usbd_audio_attribute_control *)current_entity_info->priv[0];
|
|
#if CONFIG_USBDEV_AUDIO_VERSION >= 0x0200
|
|
sampling_freq = (uint32_t *)current_entity_info->priv[1];
|
|
#endif
|
|
if (current_entity_info->bDescriptorSubtype == AUDIO_CONTROL_FEATURE_UNIT) {
|
|
#if CONFIG_USBDEV_AUDIO_VERSION < 0x0200
|
|
float volume2db = 0.0;
|
|
|
|
switch (control_selector) {
|
|
case AUDIO_FU_CONTROL_MUTE:
|
|
switch (setup->bRequest) {
|
|
case AUDIO_REQUEST_SET_CUR:
|
|
mute = (*data)[0];
|
|
current_feature_control->mute[ch] = mute;
|
|
USB_LOG_DBG("Set UnitId:%d ch[%d] mute %s\r\n", entity_id, ch, mute_string[mute]);
|
|
usbd_audio_set_mute(entity_id, ch, mute);
|
|
break;
|
|
case AUDIO_REQUEST_GET_CUR:
|
|
(*data)[0] = current_feature_control->mute[ch];
|
|
break;
|
|
default:
|
|
USB_LOG_WRN("Unhandled Audio Class bRequest 0x%02x\r\n", setup->bRequest);
|
|
return -1;
|
|
}
|
|
|
|
break;
|
|
case AUDIO_FU_CONTROL_VOLUME:
|
|
switch (setup->bRequest) {
|
|
case AUDIO_REQUEST_SET_CUR:
|
|
volume = (((uint16_t)(*data)[1] << 8) | ((uint16_t)(*data)[0]));
|
|
current_feature_control->volume[ch].vol_current = volume;
|
|
|
|
if (volume < 0x8000) {
|
|
volume2db = 0.00390625 * volume;
|
|
} else if (volume > 0x8000) {
|
|
volume2db = -0.00390625 * (0xffff - volume + 1);
|
|
}
|
|
|
|
USB_LOG_DBG("Set UnitId:%d ch[%d] %0.4f dB\r\n", entity_id, ch, volume2db);
|
|
usbd_audio_set_volume(entity_id, ch, volume2db);
|
|
break;
|
|
case AUDIO_REQUEST_GET_CUR:
|
|
memcpy(*data, ¤t_feature_control->volume[ch].vol_current, 2);
|
|
*len = 2;
|
|
break;
|
|
|
|
case AUDIO_REQUEST_GET_MIN:
|
|
memcpy(*data, ¤t_feature_control->volume[ch].vol_min, 2);
|
|
*len = 2;
|
|
break;
|
|
|
|
case AUDIO_REQUEST_GET_MAX:
|
|
memcpy(*data, ¤t_feature_control->volume[ch].vol_max, 2);
|
|
*len = 2;
|
|
break;
|
|
|
|
case AUDIO_REQUEST_GET_RES:
|
|
memcpy(*data, ¤t_feature_control->volume[ch].vol_res, 2);
|
|
*len = 2;
|
|
break;
|
|
|
|
case AUDIO_REQUEST_SET_RES:
|
|
memcpy(¤t_feature_control->volume[ch].vol_res, *data, 2);
|
|
*len = 2;
|
|
break;
|
|
default:
|
|
USB_LOG_WRN("Unhandled Audio Class bRequest 0x%02x\r\n", setup->bRequest);
|
|
return -1;
|
|
}
|
|
break;
|
|
default:
|
|
USB_LOG_WRN("Unhandled Audio Class control selector 0x%02x\r\n", control_selector);
|
|
return -1;
|
|
}
|
|
#else
|
|
switch (setup->bRequest) {
|
|
case AUDIO_REQUEST_CUR:
|
|
switch (control_selector) {
|
|
case AUDIO_FU_CONTROL_MUTE:
|
|
if (setup->bmRequestType & USB_REQUEST_DIR_MASK) {
|
|
(*data)[0] = current_feature_control->mute_bCUR;
|
|
*len = 1;
|
|
} else {
|
|
mute = (*data)[0];
|
|
USB_LOG_DBG("Set UnitId:%d ch[%d] mute %s\r\n", entity_id, ch, mute_string[mute]);
|
|
usbd_audio_set_mute(entity_id, ch, mute);
|
|
}
|
|
break;
|
|
case AUDIO_FU_CONTROL_VOLUME:
|
|
if (setup->bmRequestType & USB_REQUEST_DIR_MASK) {
|
|
(*data)[0] = current_feature_control->volume_bCUR & 0XFF;
|
|
(*data)[1] = (current_feature_control->volume_bCUR >> 8) & 0xff;
|
|
*len = 2;
|
|
} else {
|
|
volume = (((uint16_t)(*data)[1] << 8) | ((uint16_t)(*data)[0]));
|
|
current_feature_control->volume_bCUR = volume;
|
|
USB_LOG_DBG("Set UnitId:%d ch[%d] %d dB\r\n", entity_id, ch, volume);
|
|
usbd_audio_set_volume(entity_id, ch, volume);
|
|
}
|
|
break;
|
|
default:
|
|
USB_LOG_WRN("Unhandled Audio Class control selector 0x%02x\r\n", control_selector);
|
|
return -1;
|
|
}
|
|
break;
|
|
case AUDIO_REQUEST_RANGE:
|
|
switch (control_selector) {
|
|
case AUDIO_FU_CONTROL_VOLUME:
|
|
if (setup->bmRequestType & USB_REQUEST_DIR_MASK) {
|
|
*((uint16_t *)(*data + 0)) = current_feature_control->volume.wNumSubRanges;
|
|
*((uint16_t *)(*data + 2)) = current_feature_control->volume.subrange[ch].wMin;
|
|
*((uint16_t *)(*data + 4)) = current_feature_control->volume.subrange[ch].wMax;
|
|
*((uint16_t *)(*data + 6)) = current_feature_control->volume.subrange[ch].wRes;
|
|
*len = 8;
|
|
} else {
|
|
}
|
|
break;
|
|
default:
|
|
USB_LOG_WRN("Unhandled Audio Class control selector 0x%02x\r\n", control_selector);
|
|
return -1;
|
|
}
|
|
|
|
break;
|
|
default:
|
|
USB_LOG_WRN("Unhandled Audio Class bRequest 0x%02x\r\n", setup->bRequest);
|
|
return -1;
|
|
}
|
|
#endif
|
|
}
|
|
#if CONFIG_USBDEV_AUDIO_VERSION >= 0x0200
|
|
else if (current_entity_info->bDescriptorSubtype == AUDIO_CONTROL_CLOCK_SOURCE) {
|
|
switch (setup->bRequest) {
|
|
case AUDIO_REQUEST_CUR:
|
|
switch (control_selector) {
|
|
case AUDIO_CS_CONTROL_SAM_FREQ:
|
|
if (setup->bmRequestType & USB_REQUEST_DIR_MASK) {
|
|
memcpy(*data, &sampling_freq[ch], sizeof(uint32_t));
|
|
*len = 4;
|
|
} else {
|
|
memcpy(&sampling_freq[ch], *data, setup->wLength);
|
|
USB_LOG_DBG("Set ClockId:%d ch[%d] %d Hz\r\n", entity_id, ch, (int)sampling_freq[ch]);
|
|
usbd_audio_set_sampling_freq(entity_id, ch, sampling_freq[ch]);
|
|
}
|
|
break;
|
|
case AUDIO_CS_CONTROL_CLOCK_VALID:
|
|
if (setup->bmRequestType & USB_REQUEST_DIR_MASK) {
|
|
(*data)[0] = 1;
|
|
*len = 1;
|
|
} else {
|
|
}
|
|
break;
|
|
default:
|
|
USB_LOG_WRN("Unhandled Audio Class control selector 0x%02x\r\n", control_selector);
|
|
return -1;
|
|
}
|
|
break;
|
|
case AUDIO_REQUEST_RANGE:
|
|
switch (control_selector) {
|
|
case AUDIO_CS_CONTROL_SAM_FREQ:
|
|
if (setup->bmRequestType & USB_REQUEST_DIR_MASK) {
|
|
uint8_t *sampling_freq_table = NULL;
|
|
uint16_t num;
|
|
|
|
usbd_audio_get_sampling_freq_table(entity_id, &sampling_freq_table);
|
|
num = (uint16_t)((uint16_t)(sampling_freq_table[1] << 8) | ((uint16_t)sampling_freq_table[0]));
|
|
*data = sampling_freq_table;
|
|
*len = (12 * num + 2);
|
|
} else {
|
|
}
|
|
break;
|
|
default:
|
|
USB_LOG_WRN("Unhandled Audio Class control selector 0x%02x\r\n", control_selector);
|
|
return -1;
|
|
}
|
|
|
|
break;
|
|
default:
|
|
USB_LOG_WRN("Unhandled Audio Class bRequest 0x%02x\r\n", setup->bRequest);
|
|
return -1;
|
|
}
|
|
}
|
|
#endif
|
|
else {
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static void audio_notify_handler(uint8_t event, void *arg)
|
|
{
|
|
switch (event) {
|
|
case USBD_EVENT_RESET:
|
|
|
|
break;
|
|
|
|
case USBD_EVENT_SOF:
|
|
usbd_audio_sof_callback();
|
|
break;
|
|
|
|
case USBD_EVENT_SET_INTERFACE: {
|
|
struct usb_interface_descriptor *intf = (struct usb_interface_descriptor *)arg;
|
|
if (intf->bAlternateSetting == 1) {
|
|
usbd_audio_open(intf->bInterfaceNumber);
|
|
} else {
|
|
usbd_audio_close(intf->bInterfaceNumber);
|
|
}
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
void usbd_audio_add_interface(usbd_class_t *devclass, usbd_interface_t *intf)
|
|
{
|
|
static usbd_class_t *last_class = NULL;
|
|
|
|
if (last_class != devclass) {
|
|
last_class = devclass;
|
|
usbd_class_register(devclass);
|
|
}
|
|
|
|
intf->class_handler = audio_class_request_handler;
|
|
#if CONFIG_USBDEV_AUDIO_VERSION < 0x0200
|
|
intf->custom_handler = audio_custom_request_handler;
|
|
#else
|
|
intf->custom_handler = NULL;
|
|
#endif
|
|
intf->vendor_handler = NULL;
|
|
intf->notify_handler = audio_notify_handler;
|
|
usbd_class_add_interface(devclass, intf);
|
|
}
|
|
|
|
void usbd_audio_add_entity(uint8_t entity_id, uint16_t bDescriptorSubtype)
|
|
{
|
|
struct audio_entity_info *entity_info = usb_malloc(sizeof(struct audio_entity_info));
|
|
memset(entity_info, 0, sizeof(struct audio_entity_info));
|
|
entity_info->bEntityId = entity_id;
|
|
entity_info->bDescriptorSubtype = bDescriptorSubtype;
|
|
|
|
if (bDescriptorSubtype == AUDIO_CONTROL_FEATURE_UNIT) {
|
|
#if CONFIG_USBDEV_AUDIO_VERSION < 0x0200
|
|
struct usbd_audio_attribute_control *control = usb_malloc(sizeof(struct usbd_audio_attribute_control));
|
|
memset(control, 0, sizeof(struct usbd_audio_attribute_control));
|
|
for (uint8_t ch = 0; ch < CONFIG_USBDEV_AUDIO_MAX_CHANNEL; ch++) {
|
|
control->volume[ch].vol_min = 0xdb00;
|
|
control->volume[ch].vol_max = 0x0000;
|
|
control->volume[ch].vol_res = 0x0100;
|
|
control->volume[ch].vol_current = 0xf600;
|
|
control->mute[ch] = 0;
|
|
control->automatic_gain[ch] = 0;
|
|
}
|
|
#else
|
|
struct usbd_audio_attribute_control *control = usb_malloc(sizeof(struct usbd_audio_attribute_control));
|
|
memset(control, 0, sizeof(struct usbd_audio_attribute_control));
|
|
for (uint8_t ch = 0; ch < CONFIG_USBDEV_AUDIO_MAX_CHANNEL; ch++) {
|
|
control->volume.wNumSubRanges = 1;
|
|
control->volume.subrange[ch].wMin = 0;
|
|
control->volume.subrange[ch].wMax = 100;
|
|
control->volume.subrange[ch].wRes = 1;
|
|
control->mute[ch] = 0;
|
|
control->volume_bCUR = 50;
|
|
control->mute_bCUR = 0;
|
|
}
|
|
#endif
|
|
entity_info->priv[0] = control;
|
|
} else if (bDescriptorSubtype == AUDIO_CONTROL_CLOCK_SOURCE) {
|
|
#if CONFIG_USBDEV_AUDIO_VERSION >= 0x0200
|
|
uint32_t *sampling_freq = usb_malloc(sizeof(uint32_t) * CONFIG_USBDEV_AUDIO_MAX_CHANNEL);
|
|
for (size_t ch = 0; ch < CONFIG_USBDEV_AUDIO_MAX_CHANNEL; ch++) {
|
|
sampling_freq[ch] = 16000;
|
|
}
|
|
entity_info->priv[1] = sampling_freq;
|
|
#else
|
|
entity_info->priv[1] = NULL;
|
|
#endif
|
|
}
|
|
|
|
usb_slist_add_tail(&usbd_audio_entity_info_head, &entity_info->list);
|
|
}
|
|
|
|
__WEAK void usbd_audio_set_volume(uint8_t entity_id, uint8_t ch, float dB)
|
|
{
|
|
}
|
|
|
|
__WEAK void usbd_audio_set_mute(uint8_t entity_id, uint8_t ch, uint8_t enable)
|
|
{
|
|
}
|
|
|
|
__WEAK void usbd_audio_set_sampling_freq(uint8_t entity_id, uint8_t ep_ch, uint32_t sampling_freq)
|
|
{
|
|
}
|
|
#if CONFIG_USBDEV_AUDIO_VERSION >= 0x0200
|
|
__WEAK void usbd_audio_get_sampling_freq_table(uint8_t entity_id, uint8_t **sampling_freq_table)
|
|
{
|
|
*sampling_freq_table = (uint8_t *)default_sampling_freq_table;
|
|
}
|
|
#endif
|
|
__WEAK void usbd_audio_set_pitch(uint8_t ep, bool enable)
|
|
{
|
|
}
|
|
|
|
__WEAK void usbd_audio_sof_callback(void)
|
|
{
|
|
}
|