tinyusb/src/class/audio/audio_device.h

330 lines
13 KiB
C
Raw Normal View History

2020-05-22 12:09:34 +02:00
/*
* The MIT License (MIT)
*
* Copyright (c) 2020 Ha Thach (tinyusb.org)
* Copyright (c) 2020 Reinhard Panhuber
2020-05-22 12:09:34 +02:00
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*
* This file is part of the TinyUSB stack.
*/
#ifndef _TUSB_AUDIO_DEVICE_H_
#define _TUSB_AUDIO_DEVICE_H_
#include "assert.h"
#include "common/tusb_common.h"
#include "device/usbd.h"
#include "audio.h"
#include "tusb_config.h"
//--------------------------------------------------------------------+
// Class Driver Configuration
//--------------------------------------------------------------------+
// Number of Standard AS Interface Descriptors (4.9.1) defined per audio function - this is required to be able to remember the current alternate settings of these interfaces - We restrict us here to have a constant number for all audio functions (which means this has to be the maximum number of AS interfaces an audio function has and a second audio function with less AS interfaces just waste a few bytes)
#ifndef CFG_TUD_AUDIO_N_AS_INT
2020-08-20 20:09:44 +02:00
#define CFG_TUD_AUDIO_N_AS_INT 0
2020-05-22 12:09:34 +02:00
#endif
// Size of control buffer used to receive and send control messages via EP0 - has to be big enough to hold your biggest request structure e.g. range requests with multiple intervals defined or cluster descriptors
#ifndef CFG_TUD_AUDIO_CTRL_BUF_SIZE
#error You must define an audio class control request buffer size!
#endif
// Use of TX/RX FIFOs - If sizes are not zero, audio.c implements FIFOs for RX and TX (whatever defined).
2020-05-22 12:09:34 +02:00
// For RX: the input stream gets decoded into its corresponding channels, where for each channel a FIFO is setup to hold its data -> see: audio_rx_done_cb().
// For TX: the output stream is composed from CFG_TUD_AUDIO_N_CHANNELS_TX channels, where for each channel a FIFO is defined.
// Further, it implements encoding and decoding of the individual channels (parameterized by the defines below).
// If you don't use the FIFOs you need to handle encoding and decoding on your own in audio_rx_done_cb() and Y. This, however, allows for optimizations.
#ifndef CFG_TUD_AUDIO_TX_FIFO_SIZE
2020-08-20 20:09:44 +02:00
#define CFG_TUD_AUDIO_TX_FIFO_SIZE 0 // Buffer size per channel
2020-05-22 12:09:34 +02:00
#endif
#ifndef CFG_TUD_AUDIO_RX_FIFO_SIZE
2020-08-20 20:09:44 +02:00
#define CFG_TUD_AUDIO_RX_FIFO_SIZE 0 // Buffer size per channel
2020-05-22 12:09:34 +02:00
#endif
// End point sizes - Limits: Full Speed <= 1023, High Speed <= 1024
#ifndef CFG_TUD_AUDIO_EPSIZE_IN
2020-08-20 20:09:44 +02:00
#define CFG_TUD_AUDIO_EPSIZE_IN 0 // TX
2020-05-22 12:09:34 +02:00
#endif
#ifndef CFG_TUD_AUDIO_EPSIZE_OUT
2020-08-20 20:09:44 +02:00
#define CFG_TUD_AUDIO_EPSIZE_OUT 0 // RX
2020-05-22 12:09:34 +02:00
#endif
#ifndef CFG_TUD_AUDIO_ENABLE_FEEDBACK_EP
2020-08-20 20:09:44 +02:00
#define CFG_TUD_AUDIO_ENABLE_FEEDBACK_EP 0 // Feedback
2020-05-22 12:09:34 +02:00
#endif
#ifndef CFG_TUD_AUDIO_INT_CTR_EPSIZE_IN
2020-08-20 20:09:44 +02:00
#define CFG_TUD_AUDIO_INT_CTR_EPSIZE_IN 0 // Audio interrupt control
2020-05-22 12:09:34 +02:00
#endif
#if CFG_TUD_AUDIO_INT_CTR_EPSIZE_IN
2020-05-22 12:09:34 +02:00
#ifndef CFG_TUD_AUDIO_INT_CTR_BUFSIZE
2020-08-20 20:09:44 +02:00
#define CFG_TUD_AUDIO_INT_CTR_BUFSIZE 6 // Buffer size of audio control interrupt EP - 6 Bytes according to UAC 2 specification (p. 74)
2020-05-22 12:09:34 +02:00
#endif
#endif
#ifndef CFG_TUD_AUDIO_N_CHANNELS_TX
2020-08-20 20:09:44 +02:00
#define CFG_TUD_AUDIO_N_CHANNELS_TX 1
2020-05-22 12:09:34 +02:00
#endif
#ifndef CFG_TUD_AUDIO_N_CHANNELS_RX
2020-08-20 20:09:44 +02:00
#define CFG_TUD_AUDIO_N_CHANNELS_RX 1
2020-05-22 12:09:34 +02:00
#endif
// Audio data format types
#ifndef CFG_TUD_AUDIO_FORMAT_TYPE_TX
2020-08-20 20:09:44 +02:00
#define CFG_TUD_AUDIO_FORMAT_TYPE_TX AUDIO_FORMAT_TYPE_UNDEFINED // If this option is used, an encoding function has to be implemented in audio_device.c
2020-05-22 12:09:34 +02:00
#endif
#ifndef CFG_TUD_AUDIO_FORMAT_TYPE_RX
2020-08-20 20:09:44 +02:00
#define CFG_TUD_AUDIO_FORMAT_TYPE_RX AUDIO_FORMAT_TYPE_UNDEFINED // If this option is used, a decoding function has to be implemented in audio_device.c
2020-05-22 12:09:34 +02:00
#endif
// Audio data format type I specifications
#if CFG_TUD_AUDIO_FORMAT_TYPE_TX == AUDIO_FORMAT_TYPE_I
// Type definitions - for possible formats see: audio_data_format_type_I_t and further in UAC2 specifications.
#ifndef CFG_TUD_AUDIO_FORMAT_TYPE_I_TX
2020-08-20 20:09:44 +02:00
#define CFG_TUD_AUDIO_FORMAT_TYPE_I_TX AUDIO_DATA_FORMAT_TYPE_I_PCM
2020-05-22 12:09:34 +02:00
#endif
2020-08-20 20:09:44 +02:00
#ifndef CFG_TUD_AUDIO_N_BYTES_PER_SAMPLE_TX // bSubslotSize
#define CFG_TUD_AUDIO_N_BYTES_PER_SAMPLE_TX 1
2020-05-22 12:09:34 +02:00
#endif
#if CFG_TUD_AUDIO_N_BYTES_PER_SAMPLE_TX == 1
#define CFG_TUD_AUDIO_TX_ITEMSIZE 1
#elif CFG_TUD_AUDIO_N_BYTES_PER_SAMPLE_TX == 2
#define CFG_TUD_AUDIO_TX_ITEMSIZE 2
#else
#define CFG_TUD_AUDIO_TX_ITEMSIZE 4
#endif
#endif
#if CFG_TUD_AUDIO_FORMAT_TYPE_RX == AUDIO_FORMAT_TYPE_I
#ifndef CFG_TUD_AUDIO_FORMAT_TYPE_I_RX
2020-08-20 20:09:44 +02:00
#define CFG_TUD_AUDIO_FORMAT_TYPE_I_RX AUDIO_DATA_FORMAT_TYPE_I_PCM
2020-05-22 12:09:34 +02:00
#endif
2020-08-20 20:09:44 +02:00
#ifndef CFG_TUD_AUDIO_N_BYTES_PER_SAMPLE_RX // bSubslotSize
#define CFG_TUD_AUDIO_N_BYTES_PER_SAMPLE_RX 1
2020-05-22 12:09:34 +02:00
#endif
#if CFG_TUD_AUDIO_N_BYTES_PER_SAMPLE_RX == 1
#define CFG_TUD_AUDIO_RX_ITEMSIZE 1
#elif CFG_TUD_AUDIO_N_BYTES_PER_SAMPLE_RX == 2
#define CFG_TUD_AUDIO_RX_ITEMSIZE 2
#else
#define CFG_TUD_AUDIO_RX_ITEMSIZE 4
#endif
#endif
//static_assert(sizeof(tud_audio_desc_lengths) != CFG_TUD_AUDIO, "Supply audio function descriptor pack length!");
// Supported types of this driver:
2020-08-20 20:09:44 +02:00
// AUDIO_DATA_FORMAT_TYPE_I_PCM - Required definitions: CFG_TUD_AUDIO_N_CHANNELS and CFG_TUD_AUDIO_BYTES_PER_CHANNEL
2020-05-22 12:09:34 +02:00
#ifdef __cplusplus
extern "C" {
#endif
/** \addtogroup AUDIO_Serial Serial
* @{
* \defgroup AUDIO_Serial_Device Device
* @{ */
//--------------------------------------------------------------------+
// Application API (Multiple Interfaces)
// CFG_TUD_AUDIO > 1
//--------------------------------------------------------------------+
bool tud_audio_n_mounted (uint8_t itf);
#if CFG_TUD_AUDIO_EPSIZE_OUT && CFG_TUD_AUDIO_RX_BUFSIZE
2020-05-22 12:09:34 +02:00
uint16_t tud_audio_n_available (uint8_t itf, uint8_t channelId);
uint16_t tud_audio_n_read (uint8_t itf, uint8_t channelId, void* buffer, uint16_t bufsize);
void tud_audio_n_read_flush (uint8_t itf, uint8_t channelId);
#endif
#if CFG_TUD_AUDIO_EPSIZE_IN && CFG_TUD_AUDIO_TX_FIFO_SIZE
2020-08-20 20:09:44 +02:00
uint16_t tud_audio_n_write (uint8_t itf, uint8_t channelId, uint8_t const* buffer, uint16_t bufsize);
2020-05-22 12:09:34 +02:00
#endif
#if CFG_TUD_AUDIO_INT_CTR_EPSIZE_IN > 0
2020-08-20 20:09:44 +02:00
uint16_t tud_audio_int_ctr_n_available (uint8_t itf);
uint16_t tud_audio_int_ctr_n_read (uint8_t itf, void* buffer, uint16_t bufsize);
void tud_audio_int_ctr_n_read_flush (uint8_t itf);
uint16_t tud_audio_int_ctr_n_write (uint8_t itf, uint8_t const* buffer, uint16_t bufsize);
2020-05-22 12:09:34 +02:00
#endif
//--------------------------------------------------------------------+
// Application API (Interface0)
//--------------------------------------------------------------------+
2020-08-20 20:09:44 +02:00
inline bool tud_audio_mounted (void);
2020-05-22 12:09:34 +02:00
#if CFG_TUD_AUDIO_EPSIZE_OUT && CFG_TUD_AUDIO_RX_BUFSIZE
2020-08-20 20:09:44 +02:00
inline uint16_t tud_audio_available (void);
inline uint16_t tud_audio_read (void* buffer, uint16_t bufsize);
inline void tud_audio_read_flush (void);
2020-05-22 12:09:34 +02:00
#endif
#if CFG_TUD_AUDIO_EPSIZE_IN && CFG_TUD_AUDIO_TX_FIFO_SIZE
2020-05-22 12:09:34 +02:00
inline uint16_t tud_audio_write (uint8_t channelId, uint8_t const* buffer, uint16_t bufsize);
#endif
#if CFG_TUD_AUDIO_INT_CTR_EPSIZE_IN > 0
2020-08-20 20:09:44 +02:00
inline uint32_t tud_audio_int_ctr_available (void);
inline uint32_t tud_audio_int_ctr_read (void* buffer, uint32_t bufsize);
inline void tud_audio_int_ctr_read_flush (void);
inline uint32_t tud_audio_int_ctr_write (uint8_t const* buffer, uint32_t bufsize);
2020-05-22 12:09:34 +02:00
#endif
2020-07-25 11:18:50 +02:00
// Buffer control EP data and schedule a transmit
// This function is intended to be used if you do not have a persistent buffer or memory location available (e.g. non-local variables) and need to answer onto a
// get request. This function buffers your answer request frame into the control buffer of the corresponding audio driver and schedules a transmit for sending it.
// Since transmission is triggered via interrupts, a persistent memory location is required onto which the buffer pointer in pointing. If you already have such
// available you may directly use 'tud_control_xfer(...)'. In this case data does not need to be copied into an additional buffer and you save some time.
// If the request's wLength is zero, a status packet is sent instead.
bool tud_audio_buffer_and_schedule_control_xfer(uint8_t rhport, tusb_control_request_t const * p_request, void* data, uint16_t len);
2020-05-22 12:09:34 +02:00
//--------------------------------------------------------------------+
// Application Callback API (weak is optional)
//--------------------------------------------------------------------+
#if CFG_TUD_AUDIO_EPSIZE_IN
TU_ATTR_WEAK bool tud_audio_tx_done_cb(uint8_t rhport, uint16_t * n_bytes_copied);
2020-05-22 12:09:34 +02:00
#endif
#if CFG_TUD_AUDIO_EPSIZE_OUT
TU_ATTR_WEAK bool tud_audio_rx_done_cb(uint8_t rhport, uint8_t * buffer, uint16_t bufsize);
2020-05-22 12:09:34 +02:00
#endif
#if CFG_TUD_AUDIO_EPSIZE_OUT > 0 && CFG_TUD_AUDIO_ENABLE_FEEDBACK_EP
TU_ATTR_WEAK bool tud_audio_fb_done_cb(uint8_t rhport);
2020-05-22 12:09:34 +02:00
#endif
#if CFG_TUD_AUDIO_INT_CTR_EPSIZE_IN
TU_ATTR_WEAK bool tud_audio_int_ctr_done_cb(uint8_t rhport, uint16_t * n_bytes_copied);
#endif
2020-07-17 08:40:10 +02:00
// Invoked when audio set interface request received
TU_ATTR_WEAK bool tud_audio_set_itf_cb(uint8_t rhport, tusb_control_request_t const * p_request);
// Invoked when audio class specific set request received for an EP
2020-07-25 11:18:50 +02:00
TU_ATTR_WEAK bool tud_audio_set_req_ep_cb(uint8_t rhport, tusb_control_request_t const * p_request, uint8_t *pBuff);
2020-07-17 08:40:10 +02:00
// Invoked when audio class specific set request received for an interface
2020-07-25 11:18:50 +02:00
TU_ATTR_WEAK bool tud_audio_set_req_itf_cb(uint8_t rhport, tusb_control_request_t const * p_request, uint8_t *pBuff);
2020-07-17 08:40:10 +02:00
// Invoked when audio class specific set request received for an entity
2020-07-25 11:18:50 +02:00
TU_ATTR_WEAK bool tud_audio_set_req_entity_cb(uint8_t rhport, tusb_control_request_t const * p_request, uint8_t *pBuff);
2020-07-17 08:40:10 +02:00
// Invoked when audio class specific get request received for an EP
TU_ATTR_WEAK bool tud_audio_get_req_ep_cb(uint8_t rhport, tusb_control_request_t const * p_request);
// Invoked when audio class specific get request received for an interface
TU_ATTR_WEAK bool tud_audio_get_req_itf_cb(uint8_t rhport, tusb_control_request_t const * p_request);
// Invoked when audio class specific get request received for an entity
TU_ATTR_WEAK bool tud_audio_get_req_entity_cb(uint8_t rhport, tusb_control_request_t const * p_request);
2020-05-22 12:09:34 +02:00
//--------------------------------------------------------------------+
// Inline Functions
//--------------------------------------------------------------------+
inline bool tud_audio_mounted(void)
{
2020-08-20 20:09:44 +02:00
return tud_audio_n_mounted(0);
2020-05-22 12:09:34 +02:00
}
#if CFG_TUD_AUDIO_EPSIZE_IN && CFG_TUD_AUDIO_TX_FIFO_SIZE
2020-08-20 20:09:44 +02:00
inline uint16_t tud_audio_write (uint8_t channelId, uint8_t const* buffer, uint16_t bufsize) // Short version if only one audio function is used
2020-05-22 12:09:34 +02:00
{
2020-08-20 20:09:44 +02:00
return tud_audio_n_write(0, channelId, buffer, bufsize);
2020-05-22 12:09:34 +02:00
}
#endif // CFG_TUD_AUDIO_EPSIZE_IN && CFG_TUD_AUDIO_TX_FIFO_SIZE
2020-05-22 12:09:34 +02:00
#if CFG_TUD_AUDIO_EPSIZE_OUT && CFG_TUD_AUDIO_RX_BUFSIZE
2020-05-22 12:09:34 +02:00
inline uint16_t tud_audio_available(uint8_t channelId)
{
2020-08-20 20:09:44 +02:00
return tud_audio_n_available(0, channelId);
2020-05-22 12:09:34 +02:00
}
inline uint16_t tud_audio_read(uint8_t channelId, void* buffer, uint16_t bufsize)
{
2020-08-20 20:09:44 +02:00
return tud_audio_n_read(0, channelId, buffer, bufsize);
2020-05-22 12:09:34 +02:00
}
inline void tud_audio_read_flush(uint8_t channelId)
{
2020-08-20 20:09:44 +02:00
tud_audio_n_read_flush(0, channelId);
2020-05-22 12:09:34 +02:00
}
#endif
#if CFG_TUD_AUDIO_INT_CTR_EPSIZE_IN > 0
inline uint16_t tud_audio_int_ctr_available(void)
{
2020-08-20 20:09:44 +02:00
return tud_audio_int_ctr_n_available(0);
2020-05-22 12:09:34 +02:00
}
inline uint16_t tud_audio_int_ctr_read(void* buffer, uint16_t bufsize)
{
2020-08-20 20:09:44 +02:00
return tud_audio_int_ctr_n_read(0, buffer, bufsize);
2020-05-22 12:09:34 +02:00
}
inline void tud_audio_int_ctr_read_flush(void)
{
2020-08-20 20:09:44 +02:00
return tud_audio_int_ctr_n_read_flush(0);
2020-05-22 12:09:34 +02:00
}
inline uint16_t tud_audio_int_ctr_write(uint8_t const* buffer, uint16_t bufsize)
{
2020-08-20 20:09:44 +02:00
return tud_audio_int_ctr_n_write(0, buffer, bufsize);
2020-05-22 12:09:34 +02:00
}
#endif
//--------------------------------------------------------------------+
// Internal Class Driver API
//--------------------------------------------------------------------+
void audiod_init (void);
void audiod_reset (uint8_t rhport);
2020-08-20 20:09:44 +02:00
uint16_t audiod_open (uint8_t rhport, tusb_desc_interface_t const * itf_desc, uint16_t max_len);
2020-05-22 12:09:34 +02:00
bool audiod_control_request (uint8_t rhport, tusb_control_request_t const * request);
bool audiod_control_complete (uint8_t rhport, tusb_control_request_t const * request);
bool audiod_xfer_cb (uint8_t rhport, uint8_t edpt_addr, xfer_result_t result, uint32_t xferred_bytes);
#ifdef __cplusplus
}
#endif
#endif /* _TUSB_AUDIO_DEVICE_H_ */
/** @} */
/** @} */