/*! \file audio_core.c \brief USB audio device class core functions \version 2020-08-01, V3.0.0, firmware for GD32F30x \version 2020-12-07, V3.0.1, firmware for GD32F30x */ /* Copyright (c) 2020, GigaDevice Semiconductor Inc. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. 3. Neither the name of the copyright holder nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "audio_out_itf.h" #include "audio_core.h" #include #define USBD_VID 0x28E9U #define USBD_PID 0x9574U #ifdef USE_USB_AUDIO_MICPHONE extern volatile uint32_t count_data; extern const char wavetestdata[]; #define LENGTH_DATA (1747 * 32) #endif /* local function prototypes ('static') */ static uint8_t audio_init (usb_dev *udev, uint8_t config_index); static uint8_t audio_deinit (usb_dev *udev, uint8_t config_index); static uint8_t audio_req_handler (usb_dev *udev, usb_req *req); static uint8_t audio_set_intf (usb_dev *udev, usb_req *req); static uint8_t audio_ctlx_out (usb_dev *udev); static uint8_t audio_data_in (usb_dev *udev, uint8_t ep_num); static uint8_t audio_data_out (usb_dev *udev, uint8_t ep_num); static uint8_t usbd_audio_sof (usb_dev *udev); usb_class_core usbd_audio_cb = { .init = audio_init, .deinit = audio_deinit, .req_proc = audio_req_handler, .set_intf = audio_set_intf, .ctlx_out = audio_ctlx_out, .data_in = audio_data_in, .data_out = audio_data_out, .SOF = usbd_audio_sof }; #define VOL_MIN 0U /* Volume Minimum Value */ #define VOL_MAX 100U /* Volume Maximum Value */ #define VOL_RES 1U /* Volume Resolution */ #define VOL_0dB 70U /* 0dB is in the middle of VOL_MIN and VOL_MAX */ /* note:it should use the c99 standard when compiling the below codes */ /* USB standard device descriptor */ const usb_desc_dev audio_dev_desc = { .header = { .bLength = USB_DEV_DESC_LEN, .bDescriptorType = USB_DESCTYPE_DEV }, .bcdUSB = 0x0200U, .bDeviceClass = 0x00U, .bDeviceSubClass = 0x00U, .bDeviceProtocol = 0x00U, .bMaxPacketSize0 = USB_FS_EP0_MAX_LEN, .idVendor = USBD_VID, .idProduct = USBD_PID, .bcdDevice = 0x0100U, .iManufacturer = STR_IDX_MFC, .iProduct = STR_IDX_PRODUCT, .iSerialNumber = STR_IDX_SERIAL, .bNumberConfigurations = USBD_CFG_MAX_NUM }; /* USB device configuration descriptor */ const usb_desc_config_set audio_config_set = { .config = { .header = { .bLength = sizeof(usb_desc_config), .bDescriptorType = USB_DESCTYPE_CONFIG }, .wTotalLength = AUDIO_CONFIG_DESC_SET_LEN, .bNumInterfaces = 0x01U + CONFIG_DESC_AS_ITF_COUNT, .bConfigurationValue = 0x01U, .iConfiguration = 0x00U, .bmAttributes = 0xC0U, .bMaxPower = 0x32U }, .std_itf = { .header = { .bLength = sizeof(usb_desc_itf), .bDescriptorType = USB_DESCTYPE_ITF }, .bInterfaceNumber = 0x00U, .bAlternateSetting = 0x00U, .bNumEndpoints = 0x00U, .bInterfaceClass = USB_CLASS_AUDIO, .bInterfaceSubClass = AUDIO_SUBCLASS_CONTROL, .bInterfaceProtocol = AUDIO_PROTOCOL_UNDEFINED, .iInterface = 0x00U }, .ac_itf = { .header = { .bLength = sizeof(usb_desc_AC_itf), .bDescriptorType = AUDIO_DESCTYPE_INTERFACE }, .bDescriptorSubtype = 0x01U, .bcdADC = 0x0100U, .wTotalLength = AC_ITF_TOTAL_LEN, .bInCollection = CONFIG_DESC_AS_ITF_COUNT, #ifdef USE_USB_AUDIO_MICPHONE .baInterfaceNr0 = 0x01U, #endif #ifdef USE_USB_AUDIO_SPEAKER .baInterfaceNr1 = 0x02U #endif }, #ifdef USE_USB_AUDIO_MICPHONE .mic_in_terminal = { .header = { .bLength = sizeof(usb_desc_input_terminal), .bDescriptorType = AUDIO_DESCTYPE_INTERFACE }, .bDescriptorSubtype = 0x02U, .bTerminalID = 0x01U, .wTerminalType = 0x0201U, .bAssocTerminal = 0x00U, .bNrChannels = 0x02U, .wChannelConfig = 0x0003U, .iChannelNames = 0x00U, .iTerminal = 0x00U }, .mic_feature_unit = { .header = { .bLength = sizeof(usb_desc_mono_feature_unit), .bDescriptorType = AUDIO_DESCTYPE_INTERFACE }, .bDescriptorSubtype = AUDIO_CONTROL_FEATURE_UNIT, .bUnitID = AUDIO_IN_STREAMING_CTRL, .bSourceID = 0x01U, .bControlSize = 0x01U, .bmaControls0 = AUDIO_CONTROL_MUTE, .bmaControls1 = AUDIO_CONTROL_VOLUME, .iFeature = 0x00U }, .mic_out_terminal = { .header = { .bLength = sizeof(usb_desc_output_terminal), .bDescriptorType = AUDIO_DESCTYPE_INTERFACE }, .bDescriptorSubtype = AUDIO_CONTROL_OUTPUT_TERMINAL, .bTerminalID = 0x03U, .wTerminalType = 0x0101U, .bAssocTerminal = 0x00U, .bSourceID = 0x02U, .iTerminal = 0x00U }, #endif #ifdef USE_USB_AUDIO_SPEAKER .speak_in_terminal = { .header = { .bLength = sizeof(usb_desc_input_terminal), .bDescriptorType = AUDIO_DESCTYPE_INTERFACE }, .bDescriptorSubtype = AUDIO_CONTROL_INPUT_TERMINAL, .bTerminalID = 0x04U, .wTerminalType = 0x0101U, .bAssocTerminal = 0x00U, .bNrChannels = 0x02U, .wChannelConfig = 0x0003U, .iChannelNames = 0x00U, .iTerminal = 0x00U }, .speak_feature_unit = { .header = { .bLength = sizeof(usb_desc_mono_feature_unit), .bDescriptorType = AUDIO_DESCTYPE_INTERFACE }, .bDescriptorSubtype = AUDIO_CONTROL_FEATURE_UNIT, .bUnitID = AUDIO_OUT_STREAMING_CTRL, .bSourceID = 0x04U, .bControlSize = 0x01U, .bmaControls0 = AUDIO_CONTROL_MUTE, .bmaControls1 = AUDIO_CONTROL_VOLUME, .iFeature = 0x00U }, .speak_out_terminal = { .header = { .bLength = sizeof(usb_desc_output_terminal), .bDescriptorType = AUDIO_DESCTYPE_INTERFACE }, .bDescriptorSubtype = AUDIO_CONTROL_OUTPUT_TERMINAL, .bTerminalID = 0x06U, .wTerminalType = 0x0301U, .bAssocTerminal = 0x00U, .bSourceID = 0x05U, .iTerminal = 0x00U }, #endif #ifdef USE_USB_AUDIO_MICPHONE .mic_std_as_itf_zeroband = { .header = { .bLength = sizeof(usb_desc_itf), .bDescriptorType = USB_DESCTYPE_ITF }, .bInterfaceNumber = 0x01U, .bAlternateSetting = 0x00U, .bNumEndpoints = 0x00U, .bInterfaceClass = USB_CLASS_AUDIO, .bInterfaceSubClass = AUDIO_SUBCLASS_AUDIOSTREAMING, .bInterfaceProtocol = AUDIO_PROTOCOL_UNDEFINED, .iInterface = 0x00U }, .mic_std_as_itf_opera = { .header = { .bLength = sizeof(usb_desc_itf), .bDescriptorType = USB_DESCTYPE_ITF }, .bInterfaceNumber = 0x01U, .bAlternateSetting = 0x01U, .bNumEndpoints = 0x01U, .bInterfaceClass = USB_CLASS_AUDIO, .bInterfaceSubClass = AUDIO_SUBCLASS_AUDIOSTREAMING, .bInterfaceProtocol = AUDIO_PROTOCOL_UNDEFINED, .iInterface = 0x00U }, .mic_as_itf = { .header = { .bLength = sizeof(usb_desc_AS_itf), .bDescriptorType = AUDIO_DESCTYPE_INTERFACE }, .bDescriptorSubtype = AUDIO_STREAMING_GENERAL, .bTerminalLink = 0x03U, .bDelay = 0x01U, .wFormatTag = 0x0001U, }, .mic_format_typeI = { .header = { .bLength = sizeof(usb_desc_format_type), .bDescriptorType = AUDIO_DESCTYPE_INTERFACE }, .bDescriptorSubtype = AUDIO_STREAMING_FORMAT_TYPE, .bFormatType = AUDIO_FORMAT_TYPE_I, .bNrChannels = MIC_IN_CHANNEL_NBR, .bSubFrameSize = 0x02U, .bBitResolution = MIC_IN_BIT_RESOLUTION, .bSamFreqType = 0x01U, .bSamFreq[0] = (uint8_t)USBD_MIC_FREQ, .bSamFreq[1] = USBD_MIC_FREQ >> 8U, .bSamFreq[2] = USBD_MIC_FREQ >> 16U }, .mic_std_endpoint = { .header = { .bLength = sizeof(usb_desc_std_ep), .bDescriptorType = USB_DESCTYPE_EP }, .bEndpointAddress = AUDIO_IN_EP, .bmAttributes = USB_ENDPOINT_TYPE_ISOCHRONOUS, .wMaxPacketSize = MIC_IN_PACKET, .bInterval = 0x01U, .bRefresh = 0x00U, .bSynchAddress = 0x00U }, .mic_as_endpoint = { .header = { .bLength = sizeof(usb_desc_AS_ep), .bDescriptorType = AUDIO_DESCTYPE_ENDPOINT }, .bDescriptorSubtype = AUDIO_ENDPOINT_GENERAL, .bmAttributes = 0x00U, .bLockDelayUnits = 0x00U, .wLockDelay = 0x0000U, }, #endif #ifdef USE_USB_AUDIO_SPEAKER .speak_std_as_itf_zeroband = { .header = { .bLength = sizeof(usb_desc_itf), .bDescriptorType = USB_DESCTYPE_ITF }, .bInterfaceNumber = 0x02U, .bAlternateSetting = 0x00U, .bNumEndpoints = 0x00U, .bInterfaceClass = USB_CLASS_AUDIO, .bInterfaceSubClass = AUDIO_SUBCLASS_AUDIOSTREAMING, .bInterfaceProtocol = AUDIO_PROTOCOL_UNDEFINED, .iInterface = 0x00U }, .speak_std_as_itf_opera = { .header = { .bLength = sizeof(usb_desc_itf), .bDescriptorType = USB_DESCTYPE_ITF }, .bInterfaceNumber = 0x02U, .bAlternateSetting = 0x01U, .bNumEndpoints = 0x01U, .bInterfaceClass = USB_CLASS_AUDIO, .bInterfaceSubClass = AUDIO_SUBCLASS_AUDIOSTREAMING, .bInterfaceProtocol = AUDIO_PROTOCOL_UNDEFINED, .iInterface = 0x00U }, .speak_as_itf = { .header = { .bLength = sizeof(usb_desc_AS_itf), .bDescriptorType = AUDIO_DESCTYPE_INTERFACE }, .bDescriptorSubtype = AUDIO_STREAMING_GENERAL, .bTerminalLink = 0x04U, .bDelay = 0x01U, .wFormatTag = 0x0001U, }, .speak_format_typeI = { .header = { .bLength = sizeof(usb_desc_format_type), .bDescriptorType = AUDIO_DESCTYPE_INTERFACE }, .bDescriptorSubtype = AUDIO_STREAMING_FORMAT_TYPE, .bFormatType = AUDIO_FORMAT_TYPE_I, .bNrChannels = SPEAKER_OUT_CHANNEL_NBR, .bSubFrameSize = 0x02U, .bBitResolution = SPEAKER_OUT_BIT_RESOLUTION, .bSamFreqType = 0x01U, .bSamFreq[0] = (uint8_t)USBD_SPEAKER_FREQ, .bSamFreq[1] = USBD_SPEAKER_FREQ >> 8U, .bSamFreq[2] = USBD_SPEAKER_FREQ >> 16U }, .speak_std_endpoint = { .header = { .bLength = sizeof(usb_desc_std_ep), .bDescriptorType = USB_DESCTYPE_EP }, .bEndpointAddress = AUDIO_OUT_EP, .bmAttributes = USB_ENDPOINT_TYPE_ISOCHRONOUS, .wMaxPacketSize = SPEAKER_OUT_PACKET, .bInterval = 0x01U, .bRefresh = 0x00U, .bSynchAddress = 0x00U }, .speak_as_endpoint = { .header = { .bLength = sizeof(usb_desc_AS_ep), .bDescriptorType = AUDIO_DESCTYPE_ENDPOINT }, .bDescriptorSubtype = AUDIO_ENDPOINT_GENERAL, .bmAttributes = 0x00U, .bLockDelayUnits = 0x00U, .wLockDelay = 0x0000U, } #endif }; /* USB language ID descriptor */ static const usb_desc_LANGID usbd_language_id_desc = { .header = { .bLength = sizeof(usb_desc_LANGID), .bDescriptorType = USB_DESCTYPE_STR }, .wLANGID = ENG_LANGID }; /* USB manufacture string */ static const usb_desc_str manufacturer_string = { .header = { .bLength = USB_STRING_LEN(10), .bDescriptorType = USB_DESCTYPE_STR, }, .unicode_string = {'G', 'i', 'g', 'a', 'D', 'e', 'v', 'i', 'c', 'e'} }; /* USB product string */ static const usb_desc_str product_string = { .header = { .bLength = USB_STRING_LEN(14), .bDescriptorType = USB_DESCTYPE_STR, }, .unicode_string = {'G', 'D', '3', '2', '-', 'U', 'S', 'B', '_', 'A', 'u', 'd', 'i', 'o'} }; /* USBD serial string */ static usb_desc_str serial_string = { .header = { .bLength = USB_STRING_LEN(12), .bDescriptorType = USB_DESCTYPE_STR, } }; /* USB string descriptor */ void *const usbd_audio_strings[] = { [STR_IDX_LANGID] = (uint8_t *)&usbd_language_id_desc, [STR_IDX_MFC] = (uint8_t *)&manufacturer_string, [STR_IDX_PRODUCT] = (uint8_t *)&product_string, [STR_IDX_SERIAL] = (uint8_t *)&serial_string }; usb_desc audio_desc = { .dev_desc = (uint8_t *)&audio_dev_desc, .config_desc = (uint8_t *)&audio_config_set, .strings = usbd_audio_strings }; /*! \brief initialize the AUDIO device \param[in] udev: pointer to USB device instance \param[in] config_index: configuration index \param[out] none \retval USB device operation status */ static uint8_t audio_init (usb_dev *udev, uint8_t config_index) { static usbd_audio_handler audio_handler; memset((void *)&audio_handler, 0, sizeof(usbd_audio_handler)); #ifdef USE_USB_AUDIO_MICPHONE { usb_desc_std_ep std_ep = audio_config_set.mic_std_endpoint; usb_desc_ep ep = { .header = std_ep.header, .bEndpointAddress = std_ep.bEndpointAddress, .bmAttributes = std_ep.bmAttributes, .wMaxPacketSize = std_ep.wMaxPacketSize, .bInterval = std_ep.bInterval }; /* initialize Tx endpoint */ usbd_ep_setup (udev, &ep); } #endif #ifdef USE_USB_AUDIO_SPEAKER { audio_handler.isoc_out_rdptr = audio_handler.isoc_out_buff; audio_handler.isoc_out_wrptr = audio_handler.isoc_out_buff; usb_desc_std_ep std_ep = audio_config_set.speak_std_endpoint; usb_desc_ep ep = { .header = std_ep.header, .bEndpointAddress = std_ep.bEndpointAddress, .bmAttributes = std_ep.bmAttributes, .wMaxPacketSize = std_ep.wMaxPacketSize, .bInterval = std_ep.bInterval }; /* initialize Rx endpoint */ usbd_ep_setup (udev, &ep); /* initialize the audio output hardware layer */ if (USBD_OK != audio_out_fops.audio_init(USBD_AUDIO_FREQ_16K, DEFAULT_VOLUME, 0U)) { return USBD_FAIL; } /* prepare OUT endpoint to receive audio data */ usbd_ep_recev (udev, AUDIO_OUT_EP, (uint8_t*)audio_handler.isoc_out_buff, SPEAKER_OUT_PACKET); } #endif udev->dev.class_data[USBD_AUDIO_INTERFACE] = (void *)&audio_handler; return USBD_OK; } /*! \brief de-initialize the AUDIO device \param[in] udev: pointer to USB device instance \param[in] config_index: configuration index \param[out] none \retval USB device operation status */ static uint8_t audio_deinit (usb_dev *udev, uint8_t config_index) { #ifdef USE_USB_AUDIO_MICPHONE /* deinitialize AUDIO endpoints */ usbd_ep_clear(udev, AUDIO_IN_EP); #endif #ifdef USE_USB_AUDIO_SPEAKER /* deinitialize AUDIO endpoints */ usbd_ep_clear(udev, AUDIO_OUT_EP); /* deinitialize the audio output hardware layer */ if (USBD_OK != audio_out_fops.audio_deinit(0U)) { return USBD_FAIL; } #endif return USBD_OK; } /*! \brief handle the AUDIO class-specific requests \param[in] udev: pointer to USB device instance \param[in] req: device class-specific request \param[out] none \retval USB device operation status */ static uint8_t audio_req_handler (usb_dev *udev, usb_req *req) { uint8_t status = REQ_NOTSUPP; usbd_audio_handler *audio = (usbd_audio_handler *)udev->dev.class_data[USBD_AUDIO_INTERFACE]; usb_transc *transc_in = &udev->dev.transc_in[0]; usb_transc *transc_out = &udev->dev.transc_out[0]; switch (req->bRequest) { case AUDIO_REQ_GET_CUR: transc_in->xfer_buf = audio->audioctl; transc_in->remain_len = req->wLength; status = REQ_SUPP; break; case AUDIO_REQ_SET_CUR: if (req->wLength) { transc_out->xfer_buf = audio->audioctl; transc_out->remain_len = req->wLength; udev->dev.class_core->command = AUDIO_REQ_SET_CUR; audio->audioctl_len = req->wLength; audio->audioctl_unit = BYTE_HIGH(req->wIndex); status = REQ_SUPP; } break; case AUDIO_REQ_GET_MIN: *((uint16_t *)audio->audioctl) = VOL_MIN; transc_in->xfer_buf = audio->audioctl; transc_in->remain_len = req->wLength; status = REQ_SUPP; break; case AUDIO_REQ_GET_MAX: *((uint16_t *)audio->audioctl) = VOL_MAX; transc_in->xfer_buf = audio->audioctl; transc_in->remain_len = req->wLength; status = REQ_SUPP; break; case AUDIO_REQ_GET_RES: *((uint16_t *)audio->audioctl) = VOL_RES; transc_in->xfer_buf = audio->audioctl; transc_in->remain_len = req->wLength; status = REQ_SUPP; break; default: break; } return status; } /*! \brief handle the AUDIO set interface requests \param[in] udev: pointer to USB device instance \param[in] req: device class-specific request \param[out] none \retval USB device operation status */ static uint8_t audio_set_intf(usb_dev *udev, usb_req *req) { udev->dev.class_core->alter_set = req->wValue; return USBD_OK; } /*! \brief handles the control transfer OUT callback \param[in] udev: pointer to USB device instance \param[out] none \retval USB device operation status */ static uint8_t audio_ctlx_out (usb_dev *udev) { #ifdef USE_USB_AUDIO_SPEAKER usbd_audio_handler *audio = (usbd_audio_handler *)udev->dev.class_data[USBD_AUDIO_INTERFACE]; /* handles audio control requests data */ /* check if an audio_control request has been issued */ if (AUDIO_REQ_SET_CUR == udev->dev.class_core->command) { /* in this driver, to simplify code, only SET_CUR request is managed */ /* check for which addressed unit the audio_control request has been issued */ if (AUDIO_OUT_STREAMING_CTRL == audio->audioctl_unit) { /* in this driver, to simplify code, only one unit is manage */ /* call the audio interface mute function */ audio_out_fops.audio_mute_ctl(audio->audioctl[0]); /* reset the audioctl_cmd variable to prevent re-entering this function */ udev->dev.class_core->command = 0U; audio->audioctl_len = 0U; } } #endif return USBD_OK; } /*! \brief handles the audio IN data stage \param[in] udev: pointer to USB device instance \param[in] ep_num: endpoint number \param[out] none \retval USB device operation status */ static uint8_t audio_data_in (usb_dev *udev, uint8_t ep_num) { #ifdef USE_USB_AUDIO_MICPHONE if(count_data < LENGTH_DATA){ /* Prepare next buffer to be sent: dummy data */ usbd_ep_send(udev, AUDIO_IN_EP,(uint8_t*)&wavetestdata[count_data],MIC_IN_PACKET); count_data += MIC_IN_PACKET; } else { usbd_ep_send(udev, AUDIO_IN_EP,(uint8_t*)wavetestdata,MIC_IN_PACKET); count_data = MIC_IN_PACKET; } #endif return USBD_OK; } /*! \brief handles the audio OUT data stage \param[in] udev: pointer to USB device instance \param[in] ep_num: endpoint number \param[out] none \retval USB device operation status */ static uint8_t audio_data_out (usb_dev *udev, uint8_t ep_num) { #ifdef USE_USB_AUDIO_SPEAKER usbd_audio_handler *audio = (usbd_audio_handler *)udev->dev.class_data[USBD_AUDIO_INTERFACE]; /* increment the Buffer pointer or roll it back when all buffers are full */ if (audio->isoc_out_wrptr >= (audio->isoc_out_buff + (SPEAKER_OUT_PACKET * OUT_PACKET_NUM))) { /* all buffers are full: roll back */ audio->isoc_out_wrptr = audio->isoc_out_buff; } else { /* increment the buffer pointer */ audio->isoc_out_wrptr += SPEAKER_OUT_PACKET; } /* Toggle the frame index */ udev->dev.transc_out[ep_num].frame_num = (udev->dev.transc_out[ep_num].frame_num)? 0U:1U; /* prepare out endpoint to receive next audio packet */ usbd_ep_recev (udev, AUDIO_OUT_EP, (uint8_t*)(audio->isoc_out_wrptr), SPEAKER_OUT_PACKET); /* trigger the start of streaming only when half buffer is full */ if ((0U == audio->play_flag) && (audio->isoc_out_wrptr >= (audio->isoc_out_buff + ((SPEAKER_OUT_PACKET * OUT_PACKET_NUM) / 2U)))) { /* enable start of streaming */ audio->play_flag = 1U; } #endif return USBD_OK; } /*! \brief handles the SOF event (data buffer update and synchronization) \param[in] udev: pointer to USB device instance \param[out] none \retval USB device operation status */ static uint8_t usbd_audio_sof (usb_dev *udev) { #ifdef USE_USB_AUDIO_SPEAKER usbd_audio_handler *audio = (usbd_audio_handler *)udev->dev.class_data[USBD_AUDIO_INTERFACE]; /* check if there are available data in stream buffer. in this function, a single variable (play_flag) is used to avoid software delays. the play operation must be executed as soon as possible after the SOF detection. */ if (audio->play_flag) { /* start playing received packet */ audio_out_fops.audio_cmd((uint8_t*)(audio->isoc_out_rdptr), /* samples buffer pointer */ SPEAKER_OUT_PACKET, /* number of samples in Bytes */ AUDIO_CMD_PLAY); /* command to be processed */ /* increment the Buffer pointer or roll it back when all buffers all full */ if (audio->isoc_out_rdptr >= (audio->isoc_out_buff + (SPEAKER_OUT_PACKET * OUT_PACKET_NUM))) { /* roll back to the start of buffer */ audio->isoc_out_rdptr = audio->isoc_out_buff; } else { /* increment to the next sub-buffer */ audio->isoc_out_rdptr += SPEAKER_OUT_PACKET; } /* if all available buffers have been consumed, stop playing */ if (audio->isoc_out_rdptr == audio->isoc_out_wrptr) { /* Pause the audio stream */ audio_out_fops.audio_cmd((uint8_t*)(audio->isoc_out_buff), /* samples buffer pointer */ SPEAKER_OUT_PACKET, /* number of samples in Bytes */ AUDIO_CMD_PAUSE); /* command to be processed */ /* stop entering play loop */ audio->play_flag = 0U; /* reset buffer pointers */ audio->isoc_out_rdptr = audio->isoc_out_buff; audio->isoc_out_wrptr = audio->isoc_out_buff; } } #endif return USBD_OK; }