| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280 |
- /*
- * SPDX-FileCopyrightText: 2021-2022 Espressif Systems (Shanghai) CO LTD
- *
- * SPDX-License-Identifier: Unlicense OR CC0-1.0
- */
- #include <stdint.h>
- #include <string.h>
- #include <stdbool.h>
- #include "freertos/FreeRTOSConfig.h"
- #include "freertos/FreeRTOS.h"
- #include "freertos/queue.h"
- #include "freertos/semphr.h"
- #include "freertos/task.h"
- #include "esp_log.h"
- #include "bt_app_core.h"
- #ifdef CONFIG_EXAMPLE_A2DP_SINK_OUTPUT_INTERNAL_DAC
- #include "driver/dac_continuous.h"
- #else
- #include "driver/i2s_std.h"
- #endif
- #include "freertos/ringbuf.h"
- #define RINGBUF_HIGHEST_WATER_LEVEL (32 * 1024)
- #define RINGBUF_PREFETCH_WATER_LEVEL (20 * 1024)
- enum {
- RINGBUFFER_MODE_PROCESSING, /* ringbuffer is buffering incoming audio data, I2S is working */
- RINGBUFFER_MODE_PREFETCHING, /* ringbuffer is buffering incoming audio data, I2S is waiting */
- RINGBUFFER_MODE_DROPPING /* ringbuffer is not buffering (dropping) incoming audio data, I2S is working */
- };
- extern uint8_t s_volume = 0;
- /*******************************
- * STATIC FUNCTION DECLARATIONS
- ******************************/
- /* handler for application task */
- static void bt_app_task_handler(void *arg);
- /* handler for I2S task */
- static void bt_i2s_task_handler(void *arg);
- /* message sender */
- static bool bt_app_send_msg(bt_app_msg_t *msg);
- /* handle dispatched messages */
- static void bt_app_work_dispatched(bt_app_msg_t *msg);
- /*******************************
- * STATIC VARIABLE DEFINITIONS
- ******************************/
- static QueueHandle_t s_bt_app_task_queue = NULL; /* handle of work queue */
- static TaskHandle_t s_bt_app_task_handle = NULL; /* handle of application task */
- static TaskHandle_t s_bt_i2s_task_handle = NULL; /* handle of I2S task */
- static RingbufHandle_t s_ringbuf_i2s = NULL; /* handle of ringbuffer for I2S */
- static SemaphoreHandle_t s_i2s_write_semaphore = NULL;
- static uint16_t ringbuffer_mode = RINGBUFFER_MODE_PROCESSING;
- /*********************************
- * EXTERNAL FUNCTION DECLARATIONS
- ********************************/
- #ifndef CONFIG_EXAMPLE_A2DP_SINK_OUTPUT_INTERNAL_DAC
- extern i2s_chan_handle_t tx_chan;
- #else
- extern dac_continuous_handle_t tx_chan;
- #endif
- /*******************************
- * STATIC FUNCTION DEFINITIONS
- ******************************/
- static bool bt_app_send_msg(bt_app_msg_t *msg)
- {
- if (msg == NULL) {
- return false;
- }
- /* send the message to work queue */
- if (xQueueSend(s_bt_app_task_queue, msg, 10 / portTICK_PERIOD_MS) != pdTRUE) {
- ESP_LOGE(BT_APP_CORE_TAG, "%s xQueue send failed", __func__);
- return false;
- }
- return true;
- }
- static void bt_app_work_dispatched(bt_app_msg_t *msg)
- {
- if (msg->cb) {
- msg->cb(msg->event, msg->param);
- }
- }
- static void bt_app_task_handler(void *arg)
- {
- bt_app_msg_t msg;
- for (;;) {
- /* receive message from work queue and handle it */
- if (pdTRUE == xQueueReceive(s_bt_app_task_queue, &msg, (TickType_t)portMAX_DELAY)) {
- ESP_LOGD(BT_APP_CORE_TAG, "%s, signal: 0x%x, event: 0x%x", __func__, msg.sig, msg.event);
- switch (msg.sig) {
- case BT_APP_SIG_WORK_DISPATCH:
- bt_app_work_dispatched(&msg);
- break;
- default:
- ESP_LOGW(BT_APP_CORE_TAG, "%s, unhandled signal: %d", __func__, msg.sig);
- break;
- } /* switch (msg.sig) */
- if (msg.param) {
- free(msg.param);
- }
- }
- }
- }
- static void bt_i2s_task_handler(void *arg)
- {
- uint8_t *data = NULL;
- size_t item_size = 0;
- /**
- * The total length of DMA buffer of I2S is:
- * `dma_frame_num * dma_desc_num * i2s_channel_num * i2s_data_bit_width / 8`.
- * Transmit `dma_frame_num * dma_desc_num` bytes to DMA is trade-off.
- */
- const size_t item_size_upto = 240 * 6;
- size_t bytes_written = 0;
- for (;;) {
- if (pdTRUE == xSemaphoreTake(s_i2s_write_semaphore, portMAX_DELAY)) {
- for (;;) {
- item_size = 0;
- /* receive data from ringbuffer and write it to I2S DMA transmit buffer */
- data = (uint8_t *)xRingbufferReceiveUpTo(s_ringbuf_i2s, &item_size, (TickType_t)pdMS_TO_TICKS(20), item_size_upto);
- if (item_size == 0) {
- ESP_LOGI(BT_APP_CORE_TAG, "ringbuffer underflowed! mode changed: RINGBUFFER_MODE_PREFETCHING");
- ringbuffer_mode = RINGBUFFER_MODE_PREFETCHING;
- break;
- }
- // ---------- 关键修改:应用音量增益 ----------
- // 假设音频格式为16位立体声(左右声道交替)
- int16_t *samples = (int16_t *)data;
- size_t num_samples = item_size / sizeof(int16_t); // 总样本数(左+右)
- for (int i = 0; i < num_samples; i++) {
- // 使用整数运算避免浮点开销
- int32_t scaled = (samples[i] * s_volume) >> 8; // 等价于 samples[i] * (gain/256)
- // 饱和处理防止溢出
- scaled = (scaled > 32767) ? 32767 : (scaled < -32768) ? -32768 : scaled;
- samples[i] = (int16_t)scaled;
- }
- // ---------- 修改结束 ----------
- #ifdef CONFIG_EXAMPLE_A2DP_SINK_OUTPUT_INTERNAL_DAC
- dac_continuous_write(tx_chan, data, item_size, &bytes_written, -1);
- #else
- i2s_channel_write(tx_chan, data, item_size, &bytes_written, portMAX_DELAY);
- #endif
- vRingbufferReturnItem(s_ringbuf_i2s, (void *)data);
- }
- }
- }
- }
- /********************************
- * EXTERNAL FUNCTION DEFINITIONS
- *******************************/
- bool bt_app_work_dispatch(bt_app_cb_t p_cback, uint16_t event, void *p_params, int param_len, bt_app_copy_cb_t p_copy_cback)
- {
- ESP_LOGD(BT_APP_CORE_TAG, "%s event: 0x%x, param len: %d", __func__, event, param_len);
- bt_app_msg_t msg;
- memset(&msg, 0, sizeof(bt_app_msg_t));
- msg.sig = BT_APP_SIG_WORK_DISPATCH;
- msg.event = event;
- msg.cb = p_cback;
- if (param_len == 0) {
- return bt_app_send_msg(&msg);
- } else if (p_params && param_len > 0) {
- if ((msg.param = malloc(param_len)) != NULL) {
- memcpy(msg.param, p_params, param_len);
- /* check if caller has provided a copy callback to do the deep copy */
- if (p_copy_cback) {
- p_copy_cback(msg.param, p_params, param_len);
- }
- return bt_app_send_msg(&msg);
- }
- }
- return false;
- }
- void bt_app_task_start_up(void)
- {
- s_bt_app_task_queue = xQueueCreate(10, sizeof(bt_app_msg_t));
- xTaskCreate(bt_app_task_handler, "BtAppTask", 3072, NULL, 10, &s_bt_app_task_handle);
- }
- void bt_app_task_shut_down(void)
- {
- if (s_bt_app_task_handle) {
- vTaskDelete(s_bt_app_task_handle);
- s_bt_app_task_handle = NULL;
- }
- if (s_bt_app_task_queue) {
- vQueueDelete(s_bt_app_task_queue);
- s_bt_app_task_queue = NULL;
- }
- }
- void bt_i2s_task_start_up(void)
- {
- ESP_LOGI(BT_APP_CORE_TAG, "ringbuffer data empty! mode changed: RINGBUFFER_MODE_PREFETCHING");
- ringbuffer_mode = RINGBUFFER_MODE_PREFETCHING;
- if ((s_i2s_write_semaphore = xSemaphoreCreateBinary()) == NULL) {
- ESP_LOGE(BT_APP_CORE_TAG, "%s, Semaphore create failed", __func__);
- return;
- }
- if ((s_ringbuf_i2s = xRingbufferCreate(RINGBUF_HIGHEST_WATER_LEVEL, RINGBUF_TYPE_BYTEBUF)) == NULL) {
- ESP_LOGE(BT_APP_CORE_TAG, "%s, ringbuffer create failed", __func__);
- return;
- }
- xTaskCreate(bt_i2s_task_handler, "BtI2STask", 2048, NULL, configMAX_PRIORITIES - 3, &s_bt_i2s_task_handle);
- }
- void bt_i2s_task_shut_down(void)
- {
- if (s_bt_i2s_task_handle) {
- vTaskDelete(s_bt_i2s_task_handle);
- s_bt_i2s_task_handle = NULL;
- }
- if (s_ringbuf_i2s) {
- vRingbufferDelete(s_ringbuf_i2s);
- s_ringbuf_i2s = NULL;
- }
- if (s_i2s_write_semaphore) {
- vSemaphoreDelete(s_i2s_write_semaphore);
- s_i2s_write_semaphore = NULL;
- }
- }
- size_t write_ringbuf(const uint8_t *data, size_t size)
- {
- size_t item_size = 0;
- BaseType_t done = pdFALSE;
- if (ringbuffer_mode == RINGBUFFER_MODE_DROPPING) {
- ESP_LOGW(BT_APP_CORE_TAG, "ringbuffer is full, drop this packet!");
- vRingbufferGetInfo(s_ringbuf_i2s, NULL, NULL, NULL, NULL, &item_size);
- if (item_size <= RINGBUF_PREFETCH_WATER_LEVEL) {
- ESP_LOGI(BT_APP_CORE_TAG, "ringbuffer data decreased! mode changed: RINGBUFFER_MODE_PROCESSING");
- ringbuffer_mode = RINGBUFFER_MODE_PROCESSING;
- }
- return 0;
- }
- done = xRingbufferSend(s_ringbuf_i2s, (void *)data, size, (TickType_t)0);
- if (!done) {
- ESP_LOGW(BT_APP_CORE_TAG, "ringbuffer overflowed, ready to decrease data! mode changed: RINGBUFFER_MODE_DROPPING");
- ringbuffer_mode = RINGBUFFER_MODE_DROPPING;
- }
- if (ringbuffer_mode == RINGBUFFER_MODE_PREFETCHING) {
- vRingbufferGetInfo(s_ringbuf_i2s, NULL, NULL, NULL, NULL, &item_size);
- if (item_size >= RINGBUF_PREFETCH_WATER_LEVEL) {
- ESP_LOGI(BT_APP_CORE_TAG, "ringbuffer data increased! mode changed: RINGBUFFER_MODE_PROCESSING");
- ringbuffer_mode = RINGBUFFER_MODE_PROCESSING;
- if (pdFALSE == xSemaphoreGive(s_i2s_write_semaphore)) {
- ESP_LOGE(BT_APP_CORE_TAG, "semphore give failed");
- }
- }
- }
- return done ? size : 0;
- }
|