Change descriptor and code to handle features for USB Audiomain
@@ -44,6 +44,8 @@ audio_block_t * AudioInputUSB::ready_right; | |||
uint16_t AudioInputUSB::incoming_count; | |||
uint8_t AudioInputUSB::receive_flag; | |||
struct usb_audio_features_struct AudioInputUSB::features = {0,0,FEATURE_MAX_VOLUME}; | |||
#define DMABUFATTR __attribute__ ((section(".dmabuffers"), aligned (4))) | |||
uint16_t usb_audio_receive_buffer[AUDIO_RX_SIZE/2] DMABUFATTR; | |||
uint32_t usb_audio_sync_feedback DMABUFATTR; | |||
@@ -342,6 +344,70 @@ unsigned int usb_audio_transmit_callback(void) | |||
return target * 4; | |||
} | |||
int usb_audio_get_feature(void *stp, uint8_t *data, uint32_t *datalen) | |||
{ | |||
struct setup_struct setup = *((struct setup_struct *)stp); | |||
if (setup.bmRequestType==0xA1) { // should check bRequest too, and UnitID | |||
if (setup.bChannel==0) { // only support main, but in theory could do left/right as well | |||
if (setup.bCS==0x01) { // mute | |||
data[0] = AudioInputUSB::features.mute; // 1=mute, 0=unmute | |||
*datalen = 1; | |||
return 1; | |||
} | |||
else if (setup.bCS==0x02) { // volume | |||
if (setup.bRequest==0x81) { // GET_CURR | |||
data[0] = AudioInputUSB::features.volume & 0xFF; | |||
data[1] = (AudioInputUSB::features.volume>>8) & 0xFF; | |||
} | |||
else if (setup.bRequest==0x82) { // GET_MIN | |||
//serial_print("vol get_min\n"); | |||
data[0] = 0; // min level is 0 | |||
data[1] = 0; | |||
} | |||
else if (setup.bRequest==0x83) { // GET_MAX | |||
data[0] = FEATURE_MAX_VOLUME & 0xFF; // max level, for range of 0 to MAX | |||
data[1] = (FEATURE_MAX_VOLUME>>8) & 0x0F; | |||
} | |||
else if (setup.bRequest==0x84) { // GET_RES | |||
data[0] = 1; // increment vol by by 1 | |||
data[1] = 0; | |||
} | |||
else { // pass over SET_MEM, etc. | |||
return 0; | |||
} | |||
*datalen = 2; | |||
return 1; | |||
} | |||
} | |||
return 0; | |||
} | |||
return 0; | |||
} | |||
int usb_audio_set_feature(void *stp, uint8_t *buf) | |||
{ | |||
struct setup_struct setup = *((struct setup_struct *)stp); | |||
if (setup.bmRequestType==0x21) { // should check bRequest too, and UnitID | |||
if (setup.bChannel==0) { | |||
if (setup.bCS==0x01) { // mute | |||
if (setup.bRequest==0x01) { // SET_CUR | |||
AudioInputUSB::features.mute = buf[0]; // 1=mute,0=unmute | |||
AudioInputUSB::features.change = 1; | |||
return 1; | |||
} | |||
} | |||
else if (setup.bCS==0x02) { // volume | |||
if (setup.bRequest==0x01) { // SET_CUR | |||
AudioInputUSB::features.volume = buf[0] + (buf[1]<<8); | |||
AudioInputUSB::features.change = 1; | |||
return 1; | |||
} | |||
} | |||
} | |||
} | |||
return 0; | |||
} | |||
#endif // F_CPU | |||
#endif // AUDIO_INTERFACE |
@@ -4,6 +4,8 @@ | |||
#include "usb_desc.h" | |||
#ifdef AUDIO_INTERFACE | |||
#define FEATURE_MAX_VOLUME 0xFFF // volume accepted from 0 to 0xFFF | |||
#ifdef __cplusplus | |||
extern "C" { | |||
#endif | |||
@@ -11,12 +13,48 @@ extern uint16_t usb_audio_receive_buffer[]; | |||
extern uint16_t usb_audio_transmit_buffer[]; | |||
extern void usb_audio_receive_callback(unsigned int len); | |||
extern unsigned int usb_audio_transmit_callback(void); | |||
int usb_audio_set_feature(void *stp, uint8_t *buf); | |||
int usb_audio_get_feature(void *stp, uint8_t *data, uint32_t *datalen); | |||
extern uint32_t usb_audio_sync_feedback; | |||
extern uint8_t usb_audio_receive_setting; | |||
extern uint8_t usb_audio_transmit_setting; | |||
#ifdef __cplusplus | |||
} | |||
// setup struct definition could be moved from usb_dev.c to usb_dev.h so we can reuse | |||
// it instead of redefining it here | |||
struct setup_struct { | |||
union { | |||
struct { | |||
uint8_t bmRequestType; | |||
uint8_t bRequest; | |||
union { | |||
struct { | |||
uint8_t bChannel; // 0=main, 1=left, 2=right | |||
uint8_t bCS; // Control Selector | |||
}; | |||
uint16_t wValue; | |||
}; | |||
union { | |||
struct { | |||
uint8_t bIfEp; // type of entity | |||
uint8_t bEntityId; // UnitID, TerminalID, etc. | |||
}; | |||
uint16_t wIndex; | |||
}; | |||
uint16_t wLength; | |||
}; | |||
}; | |||
}; | |||
// audio features supported | |||
struct usb_audio_features_struct { | |||
int change; // set to 1 when any value is changed | |||
int mute; // 1=mute, 0=unmute | |||
int volume; // volume from 0 to FEATURE_MAX_VOLUME, maybe should be float from 0.0 to 1.0 | |||
}; | |||
#include "AudioStream.h" | |||
class AudioInputUSB : public AudioStream | |||
@@ -26,6 +64,11 @@ public: | |||
virtual void update(void); | |||
void begin(void); | |||
friend void usb_audio_receive_callback(unsigned int len); | |||
friend int usb_audio_set_feature(void *stp, uint8_t *buf); | |||
friend int usb_audio_get_feature(void *stp, uint8_t *data, uint32_t *datalen); | |||
static struct usb_audio_features_struct features; | |||
private: | |||
static bool update_responsibility; | |||
static audio_block_t *incoming_left; |
@@ -416,7 +416,7 @@ static uint8_t flightsim_report_desc[] = { | |||
#define AUDIO_INTERFACE_DESC_POS KEYMEDIA_INTERFACE_DESC_POS+KEYMEDIA_INTERFACE_DESC_SIZE | |||
#ifdef AUDIO_INTERFACE | |||
#define AUDIO_INTERFACE_DESC_SIZE 9+10+12+9+12+9 + 9+9+7+11+9+7 + 9+9+7+11+9+7+9 | |||
#define AUDIO_INTERFACE_DESC_SIZE 9+10+12+9+12+10+9 + 9+9+7+11+9+7 + 9+9+7+11+9+7+9 | |||
#else | |||
#define AUDIO_INTERFACE_DESC_SIZE 0 | |||
#endif | |||
@@ -920,6 +920,17 @@ static uint8_t config_descriptor[CONFIG_DESC_SIZE] = { | |||
0x03, 0x00, // wChannelConfig, 0x0003 = Left & Right Front | |||
0, // iChannelNames | |||
0, // iTerminal | |||
// Volume feature descriptor | |||
10, // bLength | |||
0x24, // bDescriptorType = CS_INTERFACE | |||
0x06, // bDescriptorSubType = FEATURE_UNIT | |||
0x31, // bUnitID | |||
0x03, // bSourceID (Input Terminal) | |||
0x01, // bControlSize (each channel is 1 byte, 3 channels) | |||
0x03, // bmaControls(0) Master: Volume & Mute | |||
0x00, // bmaControls(1) Left: None | |||
0x00, // bmaControls(2) Right: None | |||
0x00, // iFeature | |||
// Output Terminal Descriptor | |||
// USB DCD for Audio Devices 1.0, Table 4-4, page 40 | |||
9, // bLength | |||
@@ -929,7 +940,7 @@ static uint8_t config_descriptor[CONFIG_DESC_SIZE] = { | |||
//0x02, 0x03, // wTerminalType, 0x0302 = Headphones | |||
0x02, 0x06, // wTerminalType, 0x0602 = Digital Audio | |||
0, // bAssocTerminal, 0 = unidirectional | |||
3, // bCSourceID, connected to input terminal, ID=3 | |||
0x31, // bCSourceID, connected to feature, ID=31 | |||
0, // iTerminal | |||
// Standard AS Interface Descriptor | |||
// USB DCD for Audio Devices 1.0, Section 4.5.1, Table 4-18, page 59 |
@@ -415,8 +415,25 @@ static void usb_setup(void) | |||
return; | |||
} | |||
break; | |||
case 0x0122: // SET_CUR (wValue=0, wIndex=interface, wLength=len) | |||
case 0x0121: // SET FEATURE | |||
case 0x0221: | |||
case 0x0321: | |||
case 0x0421: | |||
// handle these on the next packet. See usb_audio_set_feature() | |||
return; | |||
case 0x81A1: // GET FEATURE | |||
case 0x82A1: | |||
case 0x83A1: | |||
case 0x84A1: | |||
if (usb_audio_get_feature(&setup, reply_buffer, &datalen)) { | |||
data = reply_buffer; | |||
} | |||
else { | |||
endpoint0_stall(); | |||
return; | |||
} | |||
break; | |||
case 0x81A2: // GET_CUR (wValue=0, wIndex=interface, wLength=len) | |||
if (setup.wLength >= 3) { | |||
reply_buffer[0] = 44100 & 255; | |||
@@ -574,8 +591,7 @@ static void usb_control(uint32_t stat) | |||
} | |||
#endif | |||
#ifdef AUDIO_INTERFACE | |||
if (setup.wRequestAndType == 0x0122 /* SET_CUR */) { | |||
// TODO: actually check data, do something with it? | |||
if (usb_audio_set_feature(&setup, buf)) { | |||
endpoint0_transmit(NULL, 0); | |||
} | |||
#endif |