f360fe5b3e
1. Modified symbol names of A2DP sink APIs include: - esp_a2d_data_cb_t --> esp_a2d_sink_data_cb_t - esp_a2d_register_data_callback --> esp_a2d_sink_register_data_callback
1617 lines
56 KiB
C
1617 lines
56 KiB
C
// Copyright 2015-2016 Espressif Systems (Shanghai) PTE LTD
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
/******************************************************************************
|
|
**
|
|
** Name: btc_a2dp_source.c
|
|
**
|
|
******************************************************************************/
|
|
#include "bt_target.h"
|
|
#include "bt_trace.h"
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
#include <stdint.h>
|
|
#include <unistd.h>
|
|
#include <time.h>
|
|
#include <sys/time.h>
|
|
#include "allocator.h"
|
|
#include "alarm.h"
|
|
#include "thread.h"
|
|
#include "mutex.h"
|
|
#include "fixed_queue.h"
|
|
#include "a2d_api.h"
|
|
#include "a2d_sbc.h"
|
|
#include "bta_av_api.h"
|
|
#include "bta_av_sbc.h"
|
|
#include "bta_av_ci.h"
|
|
#include "btc_manage.h"
|
|
#include "btc_common.h"
|
|
#include "btc_av_co.h"
|
|
#include "btc_a2dp.h"
|
|
#include "btc_a2dp_control.h"
|
|
#include "btc_a2dp_source.h"
|
|
#include "btc_av.h"
|
|
#include "btc_util.h"
|
|
#include "esp_a2dp_api.h"
|
|
#include "sbc_encoder.h"
|
|
|
|
#if BTC_AV_SRC_INCLUDED
|
|
|
|
/*****************************************************************************
|
|
** Constants
|
|
*****************************************************************************/
|
|
|
|
/* BTC source command event definition */
|
|
enum {
|
|
BTC_MEDIA_TASK_INIT,
|
|
BTC_MEDIA_TASK_CLEAN_UP,
|
|
BTC_MEDIA_START_AA_TX,
|
|
BTC_MEDIA_STOP_AA_TX,
|
|
BTC_MEDIA_SBC_ENC_INIT,
|
|
BTC_MEDIA_SBC_ENC_UPDATE,
|
|
BTC_MEDIA_FLUSH_AA_TX,
|
|
BTC_MEDIA_AUDIO_FEEDING_INIT,
|
|
};
|
|
|
|
enum {
|
|
BTC_A2DP_SOURCE_STATE_OFF = 0,
|
|
BTC_A2DP_SOURCE_STATE_ON = 1,
|
|
BTC_A2DP_SOURCE_STATE_SHUTTING_DOWN = 2
|
|
};
|
|
|
|
enum {
|
|
BTC_A2DP_SOURCE_DATA_EVT = 1,
|
|
};
|
|
|
|
/* Media task tick in milliseconds, must be set to multiple of
|
|
(1000/TICKS_PER_SEC) */
|
|
#define BTC_MEDIA_TIME_TICK_MS (30)
|
|
#define A2DP_DATA_READ_POLL_MS (BTC_MEDIA_TIME_TICK_MS / 2)
|
|
|
|
#ifndef MAX_PCM_FRAME_NUM_PER_TICK
|
|
#define MAX_PCM_FRAME_NUM_PER_TICK 21 // 14 for 20ms
|
|
#endif
|
|
|
|
#define BTC_MEDIA_AA_BUF_SIZE (4096+16)
|
|
|
|
#if (BTA_AV_CO_CP_SCMS_T == TRUE)
|
|
#define BTC_MEDIA_AA_SBC_OFFSET (AVDT_MEDIA_OFFSET + BTA_AV_SBC_HDR_SIZE + 1)
|
|
#else
|
|
#define BTC_MEDIA_AA_SBC_OFFSET (AVDT_MEDIA_OFFSET + BTA_AV_SBC_HDR_SIZE)
|
|
#endif
|
|
|
|
#ifndef BTC_MEDIA_BITRATE_STEP
|
|
#define BTC_MEDIA_BITRATE_STEP 5
|
|
#endif
|
|
|
|
#ifndef BTC_A2DP_NON_EDR_MAX_RATE
|
|
#define BTC_A2DP_NON_EDR_MAX_RATE 229
|
|
#endif
|
|
|
|
/* Middle quality quality setting @ 44.1 khz */
|
|
#define DEFAULT_SBC_BITRATE 328
|
|
|
|
/*
|
|
* CONGESTION COMPENSATION CTRL ::
|
|
*
|
|
* Thus setting controls how many buffers we will hold in media task
|
|
* during temp link congestion. Together with the stack buffer queues
|
|
* it controls much temporary a2dp link congestion we can
|
|
* compensate for. It however also depends on the default run level of sinks
|
|
* jitterbuffers. Depending on type of sink this would vary.
|
|
* Ideally the (SRC) max tx buffer capacity should equal the sinks
|
|
* jitterbuffer runlevel including any intermediate buffers on the way
|
|
* towards the sinks codec.
|
|
*/
|
|
|
|
/* fixme -- define this in pcm time instead of buffer count */
|
|
|
|
/* The typical runlevel of the tx queue size is ~1 buffer
|
|
but due to link flow control or thread preemption in lower
|
|
layers we might need to temporarily buffer up data */
|
|
|
|
/* 5 frames is equivalent to 6.89*5*2.9 ~= 100 ms @ 44.1 khz, 20 ms mediatick */
|
|
#define MAX_OUTPUT_A2DP_FRAME_QUEUE_SZ (5)
|
|
#define MAX_OUTPUT_A2DP_SRC_FRAME_QUEUE_SZ (27) // 18 for 20ms tick
|
|
|
|
typedef struct {
|
|
UINT16 num_frames_to_be_processed;
|
|
UINT16 len;
|
|
UINT16 offset;
|
|
UINT16 layer_specific;
|
|
} tBT_SBC_HDR;
|
|
|
|
typedef struct {
|
|
UINT32 aa_frame_counter;
|
|
INT32 aa_feed_counter;
|
|
INT32 aa_feed_residue;
|
|
UINT32 counter;
|
|
UINT32 bytes_per_tick; /* pcm bytes read each media task tick */
|
|
} tBTC_AV_MEDIA_FEEDINGS_PCM_STATE;
|
|
|
|
typedef union {
|
|
tBTC_AV_MEDIA_FEEDINGS_PCM_STATE pcm;
|
|
} tBTC_AV_MEDIA_FEEDINGS_STATE;
|
|
|
|
typedef struct {
|
|
UINT8 TxTranscoding;
|
|
BOOLEAN tx_flush; /* discards any outgoing data when true */
|
|
BOOLEAN is_tx_timer;
|
|
UINT16 TxAaMtuSize;
|
|
UINT32 timestamp;
|
|
fixed_queue_t *TxAaQ;
|
|
tBTC_AV_FEEDING_MODE feeding_mode;
|
|
tBTC_AV_MEDIA_FEEDINGS_STATE media_feeding_state;
|
|
tBTC_AV_MEDIA_FEEDINGS media_feeding;
|
|
SBC_ENC_PARAMS encoder;
|
|
osi_alarm_t *media_alarm;
|
|
} tBTC_A2DP_SOURCE_CB;
|
|
|
|
static void btc_a2dp_source_thread_init(UNUSED_ATTR void *context);
|
|
static void btc_a2dp_source_thread_cleanup(UNUSED_ATTR void *context);
|
|
static void btc_a2dp_source_flush_q(fixed_queue_t *p_q);
|
|
|
|
static void btc_a2dp_source_feeding_state_reset(void);
|
|
static void btc_a2dp_source_send_aa_frame(void);
|
|
static void btc_a2dp_source_aa_start_tx(void);
|
|
static void btc_a2dp_source_aa_stop_tx(void);
|
|
static void btc_a2dp_source_enc_init(BT_HDR *p_msg);
|
|
static void btc_a2dp_source_enc_update(BT_HDR *p_msg);
|
|
static void btc_a2dp_source_audio_feeding_init(BT_HDR *p_msg);
|
|
static void btc_a2dp_source_aa_tx_flush(void);
|
|
static void btc_a2dp_source_prep_2_send(UINT8 nb_frame);
|
|
static void btc_a2dp_source_handle_timer(UNUSED_ATTR void *context);
|
|
static void btc_a2dp_source_encoder_init(void);
|
|
|
|
static tBTC_A2DP_SOURCE_CB btc_aa_src_cb;
|
|
static int btc_a2dp_source_state = BTC_A2DP_SOURCE_STATE_OFF;
|
|
static xTaskHandle btc_aa_src_task_hdl = NULL;
|
|
static QueueHandle_t btc_aa_src_data_queue = NULL;
|
|
static QueueHandle_t btc_aa_src_ctrl_queue = NULL;
|
|
static QueueSetHandle_t btc_aa_src_queue_set;
|
|
|
|
static esp_a2d_source_data_cb_t btc_aa_src_data_cb = NULL;
|
|
static UINT64 last_frame_us = 0;
|
|
|
|
void btc_a2dp_src_reg_data_cb(esp_a2d_source_data_cb_t callback)
|
|
{
|
|
// todo: critical section protection
|
|
btc_aa_src_data_cb = callback;
|
|
}
|
|
|
|
static inline uint32_t btc_aa_src_data_read(uint8_t *data, int32_t len)
|
|
{
|
|
// todo: critical section protection
|
|
if (btc_aa_src_data_cb) {
|
|
return btc_aa_src_data_cb(data, len);
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/*****************************************************************************
|
|
** Misc helper functions
|
|
*****************************************************************************/
|
|
static inline void btc_aa_cb_to_app(esp_a2d_cb_event_t event, esp_a2d_cb_param_t *param)
|
|
{
|
|
esp_a2d_cb_t btc_aa_cb = (esp_a2d_cb_t)btc_profile_cb_get(BTC_PID_A2DP);
|
|
if (btc_aa_cb) {
|
|
btc_aa_cb(event, param);
|
|
}
|
|
}
|
|
|
|
/*****************************************************************************
|
|
** BTC ADAPTATION
|
|
*****************************************************************************/
|
|
|
|
bool btc_a2dp_source_is_streaming(void)
|
|
{
|
|
return btc_aa_src_cb.is_tx_timer == TRUE;
|
|
}
|
|
|
|
bool btc_a2dp_source_is_task_shutting_down(void)
|
|
{
|
|
return btc_a2dp_source_state == BTC_A2DP_SOURCE_STATE_SHUTTING_DOWN;
|
|
}
|
|
|
|
static void btc_a2dp_source_ctrl_post(uint32_t sig, void *par)
|
|
{
|
|
BtTaskEvt_t *evt = (BtTaskEvt_t *)osi_malloc(sizeof(BtTaskEvt_t));
|
|
if (evt == NULL) {
|
|
return;
|
|
}
|
|
|
|
evt->sig = sig;
|
|
evt->par = par;
|
|
|
|
if (xQueueSend(btc_aa_src_ctrl_queue, &evt, portMAX_DELAY) != pdTRUE) {
|
|
APPL_TRACE_WARNING("btc_aa_src_ctrl_queue failed, sig 0x%x\n", sig);
|
|
}
|
|
}
|
|
|
|
static void btc_a2dp_source_ctrl_handler(BtTaskEvt_t *e)
|
|
{
|
|
if (e == NULL) {
|
|
return;
|
|
}
|
|
switch (e->sig) {
|
|
case BTC_MEDIA_TASK_INIT:
|
|
btc_a2dp_source_thread_init(NULL);
|
|
break;
|
|
case BTC_MEDIA_TASK_CLEAN_UP:
|
|
btc_a2dp_source_thread_cleanup(NULL);
|
|
break;
|
|
case BTC_MEDIA_START_AA_TX:
|
|
btc_a2dp_source_aa_start_tx();
|
|
break;
|
|
case BTC_MEDIA_STOP_AA_TX:
|
|
btc_a2dp_source_aa_stop_tx();
|
|
break;
|
|
case BTC_MEDIA_SBC_ENC_INIT:
|
|
btc_a2dp_source_enc_init(e->par);
|
|
break;
|
|
case BTC_MEDIA_SBC_ENC_UPDATE:
|
|
btc_a2dp_source_enc_update(e->par);
|
|
break;
|
|
case BTC_MEDIA_AUDIO_FEEDING_INIT:
|
|
btc_a2dp_source_audio_feeding_init(e->par);
|
|
break;
|
|
case BTC_MEDIA_FLUSH_AA_TX:
|
|
btc_a2dp_source_aa_tx_flush();
|
|
break;
|
|
default:
|
|
APPL_TRACE_WARNING("media task unhandled evt: 0x%x\n", e->sig);
|
|
}
|
|
if (e->par != NULL) {
|
|
osi_free(e->par);
|
|
}
|
|
}
|
|
|
|
static void btc_a2dp_source_task_handler(void *arg)
|
|
{
|
|
QueueSetMemberHandle_t xActivatedMember;
|
|
BtTaskEvt_t *e = NULL;
|
|
for (;;) {
|
|
xActivatedMember = xQueueSelectFromSet(btc_aa_src_queue_set, portMAX_DELAY);
|
|
if (xActivatedMember == btc_aa_src_data_queue) {
|
|
int32_t data_evt;
|
|
xQueueReceive(xActivatedMember, &data_evt, 0);
|
|
if (data_evt == BTC_A2DP_SOURCE_DATA_EVT) {
|
|
btc_a2dp_source_handle_timer(NULL);
|
|
}
|
|
} else if (xActivatedMember == btc_aa_src_ctrl_queue) {
|
|
xQueueReceive(xActivatedMember, &e, 0);
|
|
btc_a2dp_source_ctrl_handler(e);
|
|
osi_free(e);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool btc_a2dp_source_startup(void)
|
|
{
|
|
if (btc_a2dp_source_state != BTC_A2DP_SOURCE_STATE_OFF) {
|
|
APPL_TRACE_ERROR("warning : media task already running");
|
|
return false;
|
|
}
|
|
|
|
APPL_TRACE_EVENT("## A2DP SOURCE START MEDIA THREAD ##");
|
|
|
|
btc_aa_src_queue_set = xQueueCreateSet(BTC_MEDIA_TASK_QUEUE_SET_LEN);
|
|
configASSERT(btc_aa_src_queue_set);
|
|
btc_aa_src_data_queue = xQueueCreate(BTC_MEDIA_DATA_QUEUE_LEN, sizeof(void *));
|
|
configASSERT(btc_aa_src_data_queue);
|
|
xQueueAddToSet(btc_aa_src_data_queue, btc_aa_src_queue_set);
|
|
|
|
btc_aa_src_ctrl_queue = xQueueCreate(BTC_MEDIA_CTRL_QUEUE_LEN, sizeof(void *));
|
|
configASSERT(btc_aa_src_ctrl_queue);
|
|
xQueueAddToSet(btc_aa_src_ctrl_queue, btc_aa_src_queue_set);
|
|
|
|
if (!btc_aa_src_data_queue || !btc_aa_src_ctrl_queue || !btc_aa_src_queue_set ) {
|
|
goto error_exit;
|
|
}
|
|
|
|
xTaskCreatePinnedToCore(btc_a2dp_source_task_handler, BTC_MEDIA_TASK_NAME, BTC_MEDIA_TASK_STACK_SIZE, NULL, BTC_MEDIA_TASK_PRIO, &btc_aa_src_task_hdl, BTC_MEDIA_TASK_PINNED_TO_CORE);
|
|
if (btc_aa_src_task_hdl == NULL) {
|
|
goto error_exit;
|
|
}
|
|
|
|
btc_a2dp_source_ctrl_post(BTC_MEDIA_TASK_INIT, NULL);
|
|
|
|
APPL_TRACE_EVENT("## A2DP SOURCE MEDIA THREAD STARTED ##\n");
|
|
|
|
return true;
|
|
|
|
error_exit:;
|
|
APPL_TRACE_ERROR("%s unable to start up media thread\n", __func__);
|
|
|
|
if (btc_aa_src_task_hdl != NULL) {
|
|
vTaskDelete(btc_aa_src_task_hdl);
|
|
btc_aa_src_task_hdl = NULL;
|
|
}
|
|
|
|
if (btc_aa_src_data_queue) {
|
|
vQueueDelete(btc_aa_src_data_queue);
|
|
btc_aa_src_data_queue = NULL;
|
|
}
|
|
if (btc_aa_src_ctrl_queue) {
|
|
vQueueDelete(btc_aa_src_ctrl_queue);
|
|
btc_aa_src_ctrl_queue = NULL;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void btc_a2dp_source_shutdown(void)
|
|
{
|
|
APPL_TRACE_EVENT("## A2DP SOURCE STOP MEDIA THREAD ##\n");
|
|
|
|
// Exit thread
|
|
btc_a2dp_source_ctrl_post(BTC_MEDIA_TASK_CLEAN_UP, NULL);
|
|
|
|
vTaskDelete(btc_aa_src_task_hdl);
|
|
btc_aa_src_task_hdl = NULL;
|
|
|
|
vQueueDelete(btc_aa_src_data_queue);
|
|
btc_aa_src_data_queue = NULL;
|
|
|
|
vQueueDelete(btc_aa_src_ctrl_queue);
|
|
btc_aa_src_ctrl_queue = NULL;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_on_idle
|
|
**
|
|
*******************************************************************************/
|
|
void btc_a2dp_source_on_idle(void)
|
|
{
|
|
/* Make sure media task is stopped */
|
|
btc_a2dp_source_stop_audio_req();
|
|
}
|
|
|
|
/*****************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_on_stopped
|
|
**
|
|
*******************************************************************************/
|
|
void btc_a2dp_source_on_stopped(tBTA_AV_SUSPEND *p_av)
|
|
{
|
|
/* allow using this api for other than suspend */
|
|
if (p_av != NULL) {
|
|
if (p_av->status != BTA_AV_SUCCESS) {
|
|
APPL_TRACE_EVENT("AV STOP FAILED (%d)", p_av->status);
|
|
if (p_av->initiator) {
|
|
btc_a2dp_control_command_ack(ESP_A2D_MEDIA_CTRL_ACK_FAILURE);
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
|
|
/* ensure tx frames are immediately suspended */
|
|
btc_aa_src_cb.tx_flush = 1;
|
|
|
|
/* request to stop media task */
|
|
btc_a2dp_source_tx_flush_req();
|
|
btc_a2dp_source_stop_audio_req();
|
|
|
|
/* once stream is fully stopped we will ack back */
|
|
}
|
|
|
|
/*****************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_on_suspended
|
|
**
|
|
**
|
|
*******************************************************************************/
|
|
|
|
void btc_a2dp_source_on_suspended(tBTA_AV_SUSPEND *p_av)
|
|
{
|
|
/* check for status failures */
|
|
if (p_av->status != BTA_AV_SUCCESS) {
|
|
if (p_av->initiator == TRUE) {
|
|
btc_a2dp_control_command_ack(ESP_A2D_MEDIA_CTRL_ACK_FAILURE);
|
|
}
|
|
}
|
|
|
|
/* once stream is fully stopped we will ack back */
|
|
|
|
/* ensure tx frames are immediately flushed */
|
|
btc_aa_src_cb.tx_flush = 1;
|
|
|
|
/* stop timer tick */
|
|
btc_a2dp_source_stop_audio_req();
|
|
}
|
|
|
|
static void btc_a2dp_source_data_post(int32_t data_type)
|
|
{
|
|
if (xQueueSend(btc_aa_src_data_queue, &data_type, 0) != pdTRUE) {
|
|
APPL_TRACE_DEBUG("Media data Q filled\n");
|
|
}
|
|
}
|
|
|
|
static UINT64 time_now_us()
|
|
{
|
|
#if _POSIX_TIMERS
|
|
struct timespec ts_now;
|
|
clock_gettime(CLOCK_BOOTTIME, &ts_now);
|
|
return ((UINT64)ts_now.tv_sec * 1000000L) + ((UINT64)ts_now.tv_nsec / 1000);
|
|
#else
|
|
struct timeval ts_now;
|
|
gettimeofday(&ts_now, NULL);
|
|
return ((UINT64)ts_now.tv_sec * 1000000L) + ((UINT64)ts_now.tv_usec);
|
|
#endif
|
|
}
|
|
|
|
static void log_tstamps_us(char *comment)
|
|
{
|
|
static UINT64 prev_us = 0;
|
|
const UINT64 now_us = time_now_us();
|
|
APPL_TRACE_DEBUG("[%s] ts %08llu, diff : %08llu, queue sz %d", comment, now_us, now_us - prev_us,
|
|
fixed_queue_length(btc_aa_src_cb.TxAaQ));
|
|
prev_us = now_us;
|
|
}
|
|
|
|
/* when true media task discards any tx frames */
|
|
void btc_a2dp_source_set_tx_flush(BOOLEAN enable)
|
|
{
|
|
APPL_TRACE_EVENT("## DROP TX %d ##", enable);
|
|
btc_aa_src_cb.tx_flush = enable;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_setup_codec
|
|
**
|
|
** Description
|
|
**
|
|
** Returns
|
|
**
|
|
*******************************************************************************/
|
|
|
|
void btc_a2dp_source_setup_codec(void)
|
|
{
|
|
tBTC_AV_MEDIA_FEEDINGS media_feeding;
|
|
tBTC_AV_STATUS status;
|
|
|
|
APPL_TRACE_EVENT("## A2DP SETUP CODEC ##\n");
|
|
|
|
osi_mutex_global_lock();
|
|
|
|
/* for now hardcode 44.1 khz 16 bit stereo PCM format */
|
|
media_feeding.cfg.pcm.sampling_freq = 44100;
|
|
media_feeding.cfg.pcm.bit_per_sample = 16;
|
|
media_feeding.cfg.pcm.num_channel = 2;
|
|
media_feeding.format = BTC_AV_CODEC_PCM;
|
|
|
|
if (bta_av_co_audio_set_codec(&media_feeding, &status)) {
|
|
tBTC_MEDIA_INIT_AUDIO_FEEDING mfeed;
|
|
|
|
/* Init the encoding task */
|
|
btc_a2dp_source_encoder_init();
|
|
|
|
/* Build the media task configuration */
|
|
mfeed.feeding = media_feeding;
|
|
mfeed.feeding_mode = BTC_AV_FEEDING_ASYNCHRONOUS;
|
|
/* Send message to Media task to configure transcoding */
|
|
btc_a2dp_source_audio_feeding_init_req(&mfeed);
|
|
}
|
|
|
|
osi_mutex_global_unlock();
|
|
}
|
|
|
|
|
|
/*******************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_audio_readbuf
|
|
**
|
|
** Description This function is called by the av_co to get the next buffer to send
|
|
**
|
|
**
|
|
** Returns void
|
|
*******************************************************************************/
|
|
BT_HDR *btc_a2dp_source_audio_readbuf(void)
|
|
{
|
|
return fixed_queue_try_dequeue(btc_aa_src_cb.TxAaQ);
|
|
}
|
|
|
|
/*******************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_start_audio_req
|
|
**
|
|
** Description
|
|
**
|
|
** Returns TRUE is success
|
|
**
|
|
*******************************************************************************/
|
|
BOOLEAN btc_a2dp_source_start_audio_req(void)
|
|
{
|
|
btc_a2dp_source_ctrl_post(BTC_MEDIA_START_AA_TX, NULL);
|
|
return TRUE;
|
|
}
|
|
|
|
/*******************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_stop_audio_req
|
|
**
|
|
** Description
|
|
**
|
|
** Returns TRUE is success
|
|
**
|
|
*******************************************************************************/
|
|
BOOLEAN btc_a2dp_source_stop_audio_req(void)
|
|
{
|
|
/*
|
|
* Explicitly check whether the btc_aa_src_ctrl_queue is not NULL to
|
|
* avoid a race condition during shutdown of the Bluetooth stack.
|
|
* This race condition is triggered when A2DP audio is streaming on
|
|
* shutdown:
|
|
* "btc_a2dp_on_stopped() -> btc_a2dp_source_stop_audio_req()" is called
|
|
* to stop the particular audio stream, and this happens right after
|
|
* the "cleanup() -> btc_a2dp_stop_media_task()" processing during
|
|
* the shutdown of the Bluetooth stack.
|
|
*/
|
|
if (btc_aa_src_ctrl_queue != NULL) {
|
|
btc_a2dp_source_ctrl_post(BTC_MEDIA_STOP_AA_TX, NULL);
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
/*******************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_enc_init_req
|
|
**
|
|
** Description
|
|
**
|
|
** Returns TRUE is success
|
|
**
|
|
*******************************************************************************/
|
|
BOOLEAN btc_a2dp_source_enc_init_req(tBTC_MEDIA_INIT_AUDIO *p_msg)
|
|
{
|
|
tBTC_MEDIA_INIT_AUDIO *p_buf;
|
|
if (NULL == (p_buf = osi_malloc(sizeof(tBTC_MEDIA_INIT_AUDIO)))) {
|
|
return FALSE;
|
|
}
|
|
|
|
memcpy(p_buf, p_msg, sizeof(tBTC_MEDIA_INIT_AUDIO));
|
|
|
|
btc_a2dp_source_ctrl_post(BTC_MEDIA_SBC_ENC_INIT, p_buf);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/*******************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_enc_update_req
|
|
**
|
|
** Description
|
|
**
|
|
** Returns TRUE is success
|
|
**
|
|
*******************************************************************************/
|
|
BOOLEAN btc_a2dp_source_enc_update_req(tBTC_MEDIA_UPDATE_AUDIO *p_msg)
|
|
{
|
|
tBTC_MEDIA_UPDATE_AUDIO *p_buf;
|
|
if (NULL == (p_buf = osi_malloc(sizeof(tBTC_MEDIA_UPDATE_AUDIO)))) {
|
|
return FALSE;
|
|
}
|
|
|
|
memcpy(p_buf, p_msg, sizeof(tBTC_MEDIA_UPDATE_AUDIO));
|
|
btc_a2dp_source_ctrl_post(BTC_MEDIA_SBC_ENC_UPDATE, p_buf);
|
|
return TRUE;
|
|
}
|
|
|
|
/*******************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_audio_feeding_init_req
|
|
**
|
|
** Description
|
|
**
|
|
** Returns TRUE is success
|
|
**
|
|
*******************************************************************************/
|
|
BOOLEAN btc_a2dp_source_audio_feeding_init_req(tBTC_MEDIA_INIT_AUDIO_FEEDING *p_msg)
|
|
{
|
|
tBTC_MEDIA_INIT_AUDIO_FEEDING *p_buf;
|
|
if (NULL == (p_buf = osi_malloc(sizeof(tBTC_MEDIA_INIT_AUDIO_FEEDING)))) {
|
|
return FALSE;
|
|
}
|
|
|
|
memcpy(p_buf, p_msg, sizeof(tBTC_MEDIA_INIT_AUDIO_FEEDING));
|
|
btc_a2dp_source_ctrl_post(BTC_MEDIA_AUDIO_FEEDING_INIT, p_buf);
|
|
return TRUE;
|
|
}
|
|
|
|
/*******************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_tx_flush_req
|
|
**
|
|
** Description
|
|
**
|
|
** Returns TRUE is success
|
|
**
|
|
*******************************************************************************/
|
|
BOOLEAN btc_a2dp_source_tx_flush_req(void)
|
|
{
|
|
/*
|
|
* Explicitly check whether the btc_aa_src_ctrl_queue is not NULL to
|
|
* avoid a race condition during shutdown of the Bluetooth stack.
|
|
* This race condition is triggered when A2DP audio is streaming on
|
|
* shutdown:
|
|
* "btc_a2dp_on_stopped() -> btc_a2dp_source_tx_flush_req()" is called
|
|
* to stop the particular audio stream, and this happens right after
|
|
* the "cleanup() -> btc_a2dp_stop_media_task()" processing during
|
|
* the shutdown of the Bluetooth stack.
|
|
*/
|
|
if (btc_aa_src_ctrl_queue != NULL) {
|
|
btc_a2dp_source_ctrl_post(BTC_MEDIA_FLUSH_AA_TX, NULL);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
** BTC ADAPTATION
|
|
*****************************************************************************/
|
|
static UINT16 btc_a2dp_source_get_sbc_rate(void)
|
|
{
|
|
UINT16 rate = DEFAULT_SBC_BITRATE;
|
|
|
|
/* restrict bitrate if a2dp link is non-edr */
|
|
if (!btc_av_is_peer_edr()) {
|
|
rate = BTC_A2DP_NON_EDR_MAX_RATE;
|
|
APPL_TRACE_DEBUG("non-edr a2dp sink detected, restrict rate to %d", rate);
|
|
}
|
|
return rate;
|
|
}
|
|
|
|
static void btc_a2dp_source_encoder_init(void)
|
|
{
|
|
UINT16 minmtu;
|
|
tBTC_MEDIA_INIT_AUDIO msg;
|
|
tA2D_SBC_CIE sbc_config;
|
|
|
|
/* lookup table for converting channel mode */
|
|
UINT16 codec_mode_tbl[5] = { SBC_JOINT_STEREO, SBC_STEREO, SBC_DUAL, 0, SBC_MONO };
|
|
|
|
/* lookup table for converting number of blocks */
|
|
UINT16 codec_block_tbl[5] = { 16, 12, 8, 0, 4 };
|
|
|
|
/* lookup table to convert freq */
|
|
UINT16 freq_block_tbl[5] = { SBC_sf48000, SBC_sf44100, SBC_sf32000, 0, SBC_sf16000 };
|
|
|
|
APPL_TRACE_DEBUG("%s", __FUNCTION__);
|
|
|
|
/* Retrieve the current SBC configuration (default if currently not used) */
|
|
bta_av_co_audio_get_sbc_config(&sbc_config, &minmtu);
|
|
msg.NumOfSubBands = (sbc_config.num_subbands == A2D_SBC_IE_SUBBAND_4) ? 4 : 8;
|
|
msg.NumOfBlocks = codec_block_tbl[sbc_config.block_len >> 5];
|
|
msg.AllocationMethod = (sbc_config.alloc_mthd == A2D_SBC_IE_ALLOC_MD_L) ? SBC_LOUDNESS : SBC_SNR;
|
|
msg.ChannelMode = codec_mode_tbl[sbc_config.ch_mode >> 1];
|
|
msg.SamplingFreq = freq_block_tbl[sbc_config.samp_freq >> 5];
|
|
msg.MtuSize = minmtu;
|
|
|
|
APPL_TRACE_EVENT("msg.ChannelMode %x", msg.ChannelMode);
|
|
|
|
/* Init the media task to encode SBC properly */
|
|
btc_a2dp_source_enc_init_req(&msg);
|
|
}
|
|
|
|
void btc_a2dp_source_encoder_update(void)
|
|
{
|
|
UINT16 minmtu;
|
|
tA2D_SBC_CIE sbc_config;
|
|
tBTC_MEDIA_UPDATE_AUDIO msg;
|
|
UINT8 pref_min;
|
|
UINT8 pref_max;
|
|
|
|
APPL_TRACE_DEBUG("%s", __FUNCTION__);
|
|
|
|
/* Retrieve the current SBC configuration (default if currently not used) */
|
|
bta_av_co_audio_get_sbc_config(&sbc_config, &minmtu);
|
|
|
|
APPL_TRACE_DEBUG("%s: Common min_bitpool:%d(0x%x) max_bitpool:%d(0x%x)", __FUNCTION__,
|
|
sbc_config.min_bitpool, sbc_config.min_bitpool,
|
|
sbc_config.max_bitpool, sbc_config.max_bitpool);
|
|
|
|
if (sbc_config.min_bitpool > sbc_config.max_bitpool) {
|
|
APPL_TRACE_ERROR("%s: ERROR min_bitpool > max_bitpool", __FUNCTION__);
|
|
}
|
|
|
|
/* check if remote sink has a preferred bitpool range */
|
|
if (bta_av_co_get_remote_bitpool_pref(&pref_min, &pref_max) == TRUE) {
|
|
/* adjust our preferred bitpool with the remote preference if within
|
|
our capable range */
|
|
|
|
if (pref_min < sbc_config.min_bitpool) {
|
|
pref_min = sbc_config.min_bitpool;
|
|
}
|
|
|
|
if (pref_max > sbc_config.max_bitpool) {
|
|
pref_max = sbc_config.max_bitpool;
|
|
}
|
|
|
|
msg.MinBitPool = pref_min;
|
|
msg.MaxBitPool = pref_max;
|
|
|
|
if ((pref_min != sbc_config.min_bitpool) || (pref_max != sbc_config.max_bitpool)) {
|
|
APPL_TRACE_EVENT("## adjusted our bitpool range to peer pref [%d:%d] ##",
|
|
pref_min, pref_max);
|
|
}
|
|
} else {
|
|
msg.MinBitPool = sbc_config.min_bitpool;
|
|
msg.MaxBitPool = sbc_config.max_bitpool;
|
|
}
|
|
|
|
msg.MinMtuSize = minmtu;
|
|
|
|
/* Update the media task to encode SBC properly */
|
|
btc_a2dp_source_enc_update_req(&msg);
|
|
}
|
|
|
|
/*******************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_enc_init
|
|
**
|
|
** Description Initialize encoding task
|
|
**
|
|
** Returns void
|
|
**
|
|
*******************************************************************************/
|
|
static void btc_a2dp_source_enc_init(BT_HDR *p_msg)
|
|
{
|
|
tBTC_MEDIA_INIT_AUDIO *pInitAudio = (tBTC_MEDIA_INIT_AUDIO *) p_msg;
|
|
|
|
APPL_TRACE_DEBUG("btc_a2dp_source_enc_init");
|
|
|
|
btc_aa_src_cb.timestamp = 0;
|
|
|
|
/* SBC encoder config (enforced even if not used) */
|
|
btc_aa_src_cb.encoder.s16ChannelMode = pInitAudio->ChannelMode;
|
|
btc_aa_src_cb.encoder.s16NumOfSubBands = pInitAudio->NumOfSubBands;
|
|
btc_aa_src_cb.encoder.s16NumOfBlocks = pInitAudio->NumOfBlocks;
|
|
btc_aa_src_cb.encoder.s16AllocationMethod = pInitAudio->AllocationMethod;
|
|
btc_aa_src_cb.encoder.s16SamplingFreq = pInitAudio->SamplingFreq;
|
|
|
|
btc_aa_src_cb.encoder.u16BitRate = btc_a2dp_source_get_sbc_rate();
|
|
|
|
/* Default transcoding is PCM to SBC, modified by feeding configuration */
|
|
btc_aa_src_cb.TxTranscoding = BTC_MEDIA_TRSCD_PCM_2_SBC;
|
|
btc_aa_src_cb.TxAaMtuSize = ((BTC_MEDIA_AA_BUF_SIZE - BTC_MEDIA_AA_SBC_OFFSET - sizeof(BT_HDR))
|
|
< pInitAudio->MtuSize) ? (BTC_MEDIA_AA_BUF_SIZE - BTC_MEDIA_AA_SBC_OFFSET
|
|
- sizeof(BT_HDR)) : pInitAudio->MtuSize;
|
|
|
|
APPL_TRACE_EVENT("btc_a2dp_source_enc_init mtu %d, peer mtu %d",
|
|
btc_aa_src_cb.TxAaMtuSize, pInitAudio->MtuSize);
|
|
APPL_TRACE_EVENT(" ch mode %d, subnd %d, nb blk %d, alloc %d, rate %d, freq %d",
|
|
btc_aa_src_cb.encoder.s16ChannelMode, btc_aa_src_cb.encoder.s16NumOfSubBands,
|
|
btc_aa_src_cb.encoder.s16NumOfBlocks,
|
|
btc_aa_src_cb.encoder.s16AllocationMethod, btc_aa_src_cb.encoder.u16BitRate,
|
|
btc_aa_src_cb.encoder.s16SamplingFreq);
|
|
|
|
/* Reset entirely the SBC encoder */
|
|
SBC_Encoder_Init(&(btc_aa_src_cb.encoder));
|
|
APPL_TRACE_DEBUG("btc_a2dp_source_enc_init bit pool %d", btc_aa_src_cb.encoder.s16BitPool);
|
|
}
|
|
|
|
|
|
/*******************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_enc_update
|
|
**
|
|
** Description Update encoding task
|
|
**
|
|
** Returns void
|
|
**
|
|
*******************************************************************************/
|
|
|
|
static void btc_a2dp_source_enc_update(BT_HDR *p_msg)
|
|
{
|
|
tBTC_MEDIA_UPDATE_AUDIO *pUpdateAudio = (tBTC_MEDIA_UPDATE_AUDIO *) p_msg;
|
|
SBC_ENC_PARAMS *pstrEncParams = &btc_aa_src_cb.encoder;
|
|
UINT16 s16SamplingFreq;
|
|
SINT16 s16BitPool = 0;
|
|
SINT16 s16BitRate;
|
|
SINT16 s16FrameLen;
|
|
UINT8 protect = 0;
|
|
|
|
APPL_TRACE_DEBUG("%s : minmtu %d, maxbp %d minbp %d", __FUNCTION__,
|
|
pUpdateAudio->MinMtuSize, pUpdateAudio->MaxBitPool, pUpdateAudio->MinBitPool);
|
|
|
|
/* Only update the bitrate and MTU size while timer is running to make sure it has been initialized */
|
|
//if (btc_aa_src_cb.is_tx_timer)
|
|
{
|
|
btc_aa_src_cb.TxAaMtuSize = ((BTC_MEDIA_AA_BUF_SIZE -
|
|
BTC_MEDIA_AA_SBC_OFFSET - sizeof(BT_HDR))
|
|
< pUpdateAudio->MinMtuSize) ? (BTC_MEDIA_AA_BUF_SIZE - BTC_MEDIA_AA_SBC_OFFSET
|
|
- sizeof(BT_HDR)) : pUpdateAudio->MinMtuSize;
|
|
|
|
/* Set the initial target bit rate */
|
|
pstrEncParams->u16BitRate = btc_a2dp_source_get_sbc_rate();
|
|
|
|
if (pstrEncParams->s16SamplingFreq == SBC_sf16000) {
|
|
s16SamplingFreq = 16000;
|
|
} else if (pstrEncParams->s16SamplingFreq == SBC_sf32000) {
|
|
s16SamplingFreq = 32000;
|
|
} else if (pstrEncParams->s16SamplingFreq == SBC_sf44100) {
|
|
s16SamplingFreq = 44100;
|
|
} else {
|
|
s16SamplingFreq = 48000;
|
|
}
|
|
|
|
do {
|
|
if (pstrEncParams->s16NumOfBlocks == 0 || pstrEncParams->s16NumOfSubBands == 0
|
|
|| pstrEncParams->s16NumOfChannels == 0) {
|
|
APPL_TRACE_ERROR("%s - Avoiding division by zero...", __FUNCTION__);
|
|
APPL_TRACE_ERROR("%s - block=%d, subBands=%d, channels=%d", __FUNCTION__,
|
|
pstrEncParams->s16NumOfBlocks, pstrEncParams->s16NumOfSubBands,
|
|
pstrEncParams->s16NumOfChannels);
|
|
break;
|
|
}
|
|
|
|
if ((pstrEncParams->s16ChannelMode == SBC_JOINT_STEREO) ||
|
|
(pstrEncParams->s16ChannelMode == SBC_STEREO) ) {
|
|
s16BitPool = (SINT16)( (pstrEncParams->u16BitRate *
|
|
pstrEncParams->s16NumOfSubBands * 1000 / s16SamplingFreq)
|
|
- ( (32 + (4 * pstrEncParams->s16NumOfSubBands *
|
|
pstrEncParams->s16NumOfChannels)
|
|
+ ( (pstrEncParams->s16ChannelMode - 2) *
|
|
pstrEncParams->s16NumOfSubBands ) )
|
|
/ pstrEncParams->s16NumOfBlocks) );
|
|
|
|
s16FrameLen = 4 + (4 * pstrEncParams->s16NumOfSubBands *
|
|
pstrEncParams->s16NumOfChannels) / 8
|
|
+ ( ((pstrEncParams->s16ChannelMode - 2) *
|
|
pstrEncParams->s16NumOfSubBands)
|
|
+ (pstrEncParams->s16NumOfBlocks * s16BitPool) ) / 8;
|
|
|
|
s16BitRate = (8 * s16FrameLen * s16SamplingFreq)
|
|
/ (pstrEncParams->s16NumOfSubBands *
|
|
pstrEncParams->s16NumOfBlocks * 1000);
|
|
|
|
if (s16BitRate > pstrEncParams->u16BitRate) {
|
|
s16BitPool--;
|
|
}
|
|
|
|
if (pstrEncParams->s16NumOfSubBands == 8) {
|
|
s16BitPool = (s16BitPool > 255) ? 255 : s16BitPool;
|
|
} else {
|
|
s16BitPool = (s16BitPool > 128) ? 128 : s16BitPool;
|
|
}
|
|
} else {
|
|
s16BitPool = (SINT16)( ((pstrEncParams->s16NumOfSubBands *
|
|
pstrEncParams->u16BitRate * 1000)
|
|
/ (s16SamplingFreq * pstrEncParams->s16NumOfChannels))
|
|
- ( ( (32 / pstrEncParams->s16NumOfChannels) +
|
|
(4 * pstrEncParams->s16NumOfSubBands) )
|
|
/ pstrEncParams->s16NumOfBlocks ) );
|
|
|
|
pstrEncParams->s16BitPool = (s16BitPool >
|
|
(16 * pstrEncParams->s16NumOfSubBands))
|
|
? (16 * pstrEncParams->s16NumOfSubBands) : s16BitPool;
|
|
}
|
|
|
|
if (s16BitPool < 0) {
|
|
s16BitPool = 0;
|
|
}
|
|
|
|
APPL_TRACE_EVENT("bitpool candidate : %d (%d kbps)",
|
|
s16BitPool, pstrEncParams->u16BitRate);
|
|
|
|
if (s16BitPool > pUpdateAudio->MaxBitPool) {
|
|
APPL_TRACE_DEBUG("%s computed bitpool too large (%d)", __FUNCTION__, s16BitPool);
|
|
/* Decrease bitrate */
|
|
btc_aa_src_cb.encoder.u16BitRate -= BTC_MEDIA_BITRATE_STEP;
|
|
/* Record that we have decreased the bitrate */
|
|
protect |= 1;
|
|
} else if (s16BitPool < pUpdateAudio->MinBitPool) {
|
|
APPL_TRACE_WARNING("%s computed bitpool too small (%d)", __FUNCTION__, s16BitPool);
|
|
|
|
/* Increase bitrate */
|
|
UINT16 previous_u16BitRate = btc_aa_src_cb.encoder.u16BitRate;
|
|
btc_aa_src_cb.encoder.u16BitRate += BTC_MEDIA_BITRATE_STEP;
|
|
/* Record that we have increased the bitrate */
|
|
protect |= 2;
|
|
/* Check over-flow */
|
|
if (btc_aa_src_cb.encoder.u16BitRate < previous_u16BitRate) {
|
|
protect |= 3;
|
|
}
|
|
} else {
|
|
break;
|
|
}
|
|
/* In case we have already increased and decreased the bitrate, just stop */
|
|
if (protect == 3) {
|
|
APPL_TRACE_ERROR("%s could not find bitpool in range", __FUNCTION__);
|
|
break;
|
|
}
|
|
} while (1);
|
|
|
|
/* Finally update the bitpool in the encoder structure */
|
|
pstrEncParams->s16BitPool = s16BitPool;
|
|
|
|
APPL_TRACE_DEBUG("%s final bit rate %d, final bit pool %d", __FUNCTION__,
|
|
btc_aa_src_cb.encoder.u16BitRate, btc_aa_src_cb.encoder.s16BitPool);
|
|
|
|
/* make sure we reinitialize encoder with new settings */
|
|
SBC_Encoder_Init(&(btc_aa_src_cb.encoder));
|
|
}
|
|
}
|
|
|
|
/*******************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_pcm2sbc_init
|
|
**
|
|
** Description Init encoding task for PCM to SBC according to feeding
|
|
**
|
|
** Returns void
|
|
**
|
|
*******************************************************************************/
|
|
static void btc_a2dp_source_pcm2sbc_init(tBTC_MEDIA_INIT_AUDIO_FEEDING *p_feeding)
|
|
{
|
|
BOOLEAN reconfig_needed = FALSE;
|
|
|
|
APPL_TRACE_DEBUG("PCM feeding:");
|
|
APPL_TRACE_DEBUG("sampling_freq:%d", p_feeding->feeding.cfg.pcm.sampling_freq);
|
|
APPL_TRACE_DEBUG("num_channel:%d", p_feeding->feeding.cfg.pcm.num_channel);
|
|
APPL_TRACE_DEBUG("bit_per_sample:%d", p_feeding->feeding.cfg.pcm.bit_per_sample);
|
|
|
|
/* Check the PCM feeding sampling_freq */
|
|
switch (p_feeding->feeding.cfg.pcm.sampling_freq) {
|
|
case 8000:
|
|
case 12000:
|
|
case 16000:
|
|
case 24000:
|
|
case 32000:
|
|
case 48000:
|
|
/* For these sampling_freq the AV connection must be 48000 */
|
|
if (btc_aa_src_cb.encoder.s16SamplingFreq != SBC_sf48000) {
|
|
/* Reconfiguration needed at 48000 */
|
|
APPL_TRACE_DEBUG("SBC Reconfiguration needed at 48000");
|
|
btc_aa_src_cb.encoder.s16SamplingFreq = SBC_sf48000;
|
|
reconfig_needed = TRUE;
|
|
}
|
|
break;
|
|
|
|
case 11025:
|
|
case 22050:
|
|
case 44100:
|
|
/* For these sampling_freq the AV connection must be 44100 */
|
|
if (btc_aa_src_cb.encoder.s16SamplingFreq != SBC_sf44100) {
|
|
/* Reconfiguration needed at 44100 */
|
|
APPL_TRACE_DEBUG("SBC Reconfiguration needed at 44100");
|
|
btc_aa_src_cb.encoder.s16SamplingFreq = SBC_sf44100;
|
|
reconfig_needed = TRUE;
|
|
}
|
|
break;
|
|
default:
|
|
APPL_TRACE_DEBUG("Feeding PCM sampling_freq unsupported");
|
|
break;
|
|
}
|
|
|
|
/* Some AV Headsets do not support Mono => always ask for Stereo */
|
|
if (btc_aa_src_cb.encoder.s16ChannelMode == SBC_MONO) {
|
|
APPL_TRACE_DEBUG("SBC Reconfiguration needed in Stereo");
|
|
btc_aa_src_cb.encoder.s16ChannelMode = SBC_JOINT_STEREO;
|
|
reconfig_needed = TRUE;
|
|
}
|
|
|
|
if (reconfig_needed != FALSE) {
|
|
APPL_TRACE_DEBUG("%s :: mtu %d", __FUNCTION__, btc_aa_src_cb.TxAaMtuSize);
|
|
APPL_TRACE_DEBUG("ch mode %d, nbsubd %d, nb %d, alloc %d, rate %d, freq %d",
|
|
btc_aa_src_cb.encoder.s16ChannelMode,
|
|
btc_aa_src_cb.encoder.s16NumOfSubBands, btc_aa_src_cb.encoder.s16NumOfBlocks,
|
|
btc_aa_src_cb.encoder.s16AllocationMethod, btc_aa_src_cb.encoder.u16BitRate,
|
|
btc_aa_src_cb.encoder.s16SamplingFreq);
|
|
|
|
SBC_Encoder_Init(&(btc_aa_src_cb.encoder));
|
|
} else {
|
|
APPL_TRACE_DEBUG("%s no SBC reconfig needed", __FUNCTION__);
|
|
}
|
|
}
|
|
|
|
/*******************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_audio_feeding_init
|
|
**
|
|
** Description Initialize the audio path according to the feeding format
|
|
**
|
|
** Returns void
|
|
**
|
|
*******************************************************************************/
|
|
static void btc_a2dp_source_audio_feeding_init(BT_HDR *p_msg)
|
|
{
|
|
tBTC_MEDIA_INIT_AUDIO_FEEDING *p_feeding = (tBTC_MEDIA_INIT_AUDIO_FEEDING *) p_msg;
|
|
|
|
APPL_TRACE_DEBUG("%s format:%d", __FUNCTION__, p_feeding->feeding.format);
|
|
|
|
/* Save Media Feeding information */
|
|
btc_aa_src_cb.feeding_mode = p_feeding->feeding_mode;
|
|
btc_aa_src_cb.media_feeding = p_feeding->feeding;
|
|
|
|
/* Handle different feeding formats */
|
|
switch (p_feeding->feeding.format) {
|
|
case BTC_AV_CODEC_PCM:
|
|
btc_aa_src_cb.TxTranscoding = BTC_MEDIA_TRSCD_PCM_2_SBC;
|
|
btc_a2dp_source_pcm2sbc_init(p_feeding);
|
|
break;
|
|
|
|
default :
|
|
APPL_TRACE_ERROR("unknown feeding format %d", p_feeding->feeding.format);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/*******************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_aa_tx_flush
|
|
**
|
|
** Description
|
|
**
|
|
** Returns void
|
|
**
|
|
*******************************************************************************/
|
|
static void btc_a2dp_source_aa_tx_flush(void)
|
|
{
|
|
/* Flush all enqueued music buffers (encoded) */
|
|
APPL_TRACE_DEBUG("%s", __FUNCTION__);
|
|
|
|
btc_aa_src_cb.media_feeding_state.pcm.counter = 0;
|
|
btc_aa_src_cb.media_feeding_state.pcm.aa_feed_residue = 0;
|
|
|
|
btc_a2dp_source_flush_q(btc_aa_src_cb.TxAaQ);
|
|
|
|
btc_aa_src_data_read(NULL, -1);
|
|
}
|
|
|
|
/*******************************************************************************
|
|
**
|
|
** Function btc_get_num_aa_frame
|
|
**
|
|
** Description
|
|
**
|
|
** Returns The number of media frames in this time slice
|
|
**
|
|
*******************************************************************************/
|
|
static UINT8 btc_get_num_aa_frame(void)
|
|
{
|
|
UINT8 result = 0;
|
|
|
|
switch (btc_aa_src_cb.TxTranscoding) {
|
|
case BTC_MEDIA_TRSCD_PCM_2_SBC: {
|
|
UINT32 pcm_bytes_per_frame = btc_aa_src_cb.encoder.s16NumOfSubBands *
|
|
btc_aa_src_cb.encoder.s16NumOfBlocks *
|
|
btc_aa_src_cb.media_feeding.cfg.pcm.num_channel *
|
|
btc_aa_src_cb.media_feeding.cfg.pcm.bit_per_sample / 8;
|
|
|
|
UINT32 us_this_tick = BTC_MEDIA_TIME_TICK_MS * 1000;
|
|
UINT64 now_us = time_now_us();
|
|
if (last_frame_us != 0) {
|
|
#if _POSIX_TIMERS
|
|
us_this_tick = (now_us - last_frame_us);
|
|
#else
|
|
// consider the case that the number of day increases and timeofday wraps around
|
|
us_this_tick = (now_us > last_frame_us) ? (now_us - last_frame_us) :
|
|
(now_us + 86400000000ull - last_frame_us);
|
|
#endif
|
|
}
|
|
last_frame_us = now_us;
|
|
|
|
btc_aa_src_cb.media_feeding_state.pcm.counter +=
|
|
btc_aa_src_cb.media_feeding_state.pcm.bytes_per_tick *
|
|
us_this_tick / (BTC_MEDIA_TIME_TICK_MS * 1000);
|
|
|
|
/* calculate nbr of frames pending for this media tick */
|
|
result = btc_aa_src_cb.media_feeding_state.pcm.counter / pcm_bytes_per_frame;
|
|
if (result > MAX_PCM_FRAME_NUM_PER_TICK) {
|
|
APPL_TRACE_WARNING("%s() - Limiting frames to be sent from %d to %d"
|
|
, __FUNCTION__, result, MAX_PCM_FRAME_NUM_PER_TICK);
|
|
result = MAX_PCM_FRAME_NUM_PER_TICK;
|
|
}
|
|
btc_aa_src_cb.media_feeding_state.pcm.counter -= result * pcm_bytes_per_frame;
|
|
|
|
LOG_VERBOSE("WRITE %d FRAMES", result);
|
|
}
|
|
break;
|
|
|
|
default:
|
|
APPL_TRACE_ERROR("ERROR btc_get_num_aa_frame Unsupported transcoding format 0x%x",
|
|
btc_aa_src_cb.TxTranscoding);
|
|
result = 0;
|
|
break;
|
|
}
|
|
|
|
return (UINT8)result;
|
|
}
|
|
|
|
/*******************************************************************************
|
|
**
|
|
** Function btc_media_aa_read_feeding
|
|
**
|
|
** Description
|
|
**
|
|
** Returns void
|
|
**
|
|
*******************************************************************************/
|
|
|
|
BOOLEAN btc_media_aa_read_feeding(void)
|
|
{
|
|
UINT16 blocm_x_subband = btc_aa_src_cb.encoder.s16NumOfSubBands * \
|
|
btc_aa_src_cb.encoder.s16NumOfBlocks;
|
|
UINT32 read_size;
|
|
UINT16 sbc_sampling = 48000;
|
|
UINT32 src_samples;
|
|
UINT16 bytes_needed = blocm_x_subband * btc_aa_src_cb.encoder.s16NumOfChannels * \
|
|
btc_aa_src_cb.media_feeding.cfg.pcm.bit_per_sample / 8;
|
|
static UINT16 up_sampled_buffer[SBC_MAX_NUM_FRAME * SBC_MAX_NUM_OF_BLOCKS
|
|
* SBC_MAX_NUM_OF_CHANNELS * SBC_MAX_NUM_OF_SUBBANDS * 2];
|
|
static UINT16 read_buffer[SBC_MAX_NUM_FRAME * SBC_MAX_NUM_OF_BLOCKS
|
|
* SBC_MAX_NUM_OF_CHANNELS * SBC_MAX_NUM_OF_SUBBANDS];
|
|
UINT32 src_size_used;
|
|
UINT32 dst_size_used;
|
|
BOOLEAN fract_needed;
|
|
INT32 fract_max;
|
|
INT32 fract_threshold;
|
|
UINT32 nb_byte_read = 0;
|
|
|
|
/* Get the SBC sampling rate */
|
|
switch (btc_aa_src_cb.encoder.s16SamplingFreq) {
|
|
case SBC_sf48000:
|
|
sbc_sampling = 48000;
|
|
break;
|
|
case SBC_sf44100:
|
|
sbc_sampling = 44100;
|
|
break;
|
|
case SBC_sf32000:
|
|
sbc_sampling = 32000;
|
|
break;
|
|
case SBC_sf16000:
|
|
sbc_sampling = 16000;
|
|
break;
|
|
}
|
|
|
|
if (sbc_sampling == btc_aa_src_cb.media_feeding.cfg.pcm.sampling_freq) {
|
|
read_size = bytes_needed - btc_aa_src_cb.media_feeding_state.pcm.aa_feed_residue;
|
|
nb_byte_read = btc_aa_src_data_read(
|
|
((uint8_t *)btc_aa_src_cb.encoder.as16PcmBuffer) +
|
|
btc_aa_src_cb.media_feeding_state.pcm.aa_feed_residue,
|
|
read_size);
|
|
if (nb_byte_read == read_size) {
|
|
btc_aa_src_cb.media_feeding_state.pcm.aa_feed_residue = 0;
|
|
return TRUE;
|
|
} else {
|
|
APPL_TRACE_WARNING("### UNDERFLOW :: ONLY READ %d BYTES OUT OF %d ###",
|
|
nb_byte_read, read_size);
|
|
btc_aa_src_cb.media_feeding_state.pcm.aa_feed_residue += nb_byte_read;
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
/* Some Feeding PCM frequencies require to split the number of sample */
|
|
/* to read. */
|
|
/* E.g 128/6=21.3333 => read 22 and 21 and 21 => max = 2; threshold = 0*/
|
|
fract_needed = FALSE; /* Default */
|
|
switch (btc_aa_src_cb.media_feeding.cfg.pcm.sampling_freq) {
|
|
case 32000:
|
|
case 8000:
|
|
fract_needed = TRUE;
|
|
fract_max = 2; /* 0, 1 and 2 */
|
|
fract_threshold = 0; /* Add one for the first */
|
|
break;
|
|
case 16000:
|
|
fract_needed = TRUE;
|
|
fract_max = 2; /* 0, 1 and 2 */
|
|
fract_threshold = 1; /* Add one for the first two frames*/
|
|
break;
|
|
}
|
|
|
|
/* Compute number of sample to read from source */
|
|
src_samples = blocm_x_subband;
|
|
src_samples *= btc_aa_src_cb.media_feeding.cfg.pcm.sampling_freq;
|
|
src_samples /= sbc_sampling;
|
|
|
|
/* The previous division may have a remainder not null */
|
|
if (fract_needed) {
|
|
if (btc_aa_src_cb.media_feeding_state.pcm.aa_feed_counter <= fract_threshold) {
|
|
src_samples++; /* for every read before threshold add one sample */
|
|
}
|
|
|
|
/* do nothing if counter >= threshold */
|
|
btc_aa_src_cb.media_feeding_state.pcm.aa_feed_counter++; /* one more read */
|
|
if (btc_aa_src_cb.media_feeding_state.pcm.aa_feed_counter > fract_max) {
|
|
btc_aa_src_cb.media_feeding_state.pcm.aa_feed_counter = 0;
|
|
}
|
|
}
|
|
|
|
/* Compute number of bytes to read from source */
|
|
read_size = src_samples;
|
|
read_size *= btc_aa_src_cb.media_feeding.cfg.pcm.num_channel;
|
|
read_size *= (btc_aa_src_cb.media_feeding.cfg.pcm.bit_per_sample / 8);
|
|
|
|
/* Read Data from data channel */
|
|
nb_byte_read = btc_aa_src_data_read((uint8_t *)read_buffer, read_size);
|
|
|
|
//tput_mon(TRUE, nb_byte_read, FALSE);
|
|
|
|
if (nb_byte_read < read_size) {
|
|
APPL_TRACE_WARNING("### UNDERRUN :: ONLY READ %d BYTES OUT OF %d ###",
|
|
nb_byte_read, read_size);
|
|
|
|
if (nb_byte_read == 0) {
|
|
return FALSE;
|
|
}
|
|
|
|
if (btc_aa_src_cb.feeding_mode == BTC_AV_FEEDING_ASYNCHRONOUS) {
|
|
/* Fill the unfilled part of the read buffer with silence (0) */
|
|
memset(((UINT8 *)read_buffer) + nb_byte_read, 0, read_size - nb_byte_read);
|
|
nb_byte_read = read_size;
|
|
}
|
|
}
|
|
|
|
/* Initialize PCM up-sampling engine */
|
|
bta_av_sbc_init_up_sample(btc_aa_src_cb.media_feeding.cfg.pcm.sampling_freq,
|
|
sbc_sampling, btc_aa_src_cb.media_feeding.cfg.pcm.bit_per_sample,
|
|
btc_aa_src_cb.media_feeding.cfg.pcm.num_channel);
|
|
|
|
/* re-sample read buffer */
|
|
/* The output PCM buffer will be stereo, 16 bit per sample */
|
|
dst_size_used = bta_av_sbc_up_sample((UINT8 *)read_buffer,
|
|
(UINT8 *)up_sampled_buffer + btc_aa_src_cb.media_feeding_state.pcm.aa_feed_residue,
|
|
nb_byte_read,
|
|
sizeof(up_sampled_buffer) - btc_aa_src_cb.media_feeding_state.pcm.aa_feed_residue,
|
|
&src_size_used);
|
|
|
|
/* update the residue */
|
|
btc_aa_src_cb.media_feeding_state.pcm.aa_feed_residue += dst_size_used;
|
|
|
|
/* only copy the pcm sample when we have up-sampled enough PCM */
|
|
if (btc_aa_src_cb.media_feeding_state.pcm.aa_feed_residue >= bytes_needed) {
|
|
/* Copy the output pcm samples in SBC encoding buffer */
|
|
memcpy((UINT8 *)btc_aa_src_cb.encoder.as16PcmBuffer,
|
|
(UINT8 *)up_sampled_buffer,
|
|
bytes_needed);
|
|
/* update the residue */
|
|
btc_aa_src_cb.media_feeding_state.pcm.aa_feed_residue -= bytes_needed;
|
|
|
|
if (btc_aa_src_cb.media_feeding_state.pcm.aa_feed_residue != 0) {
|
|
memcpy((UINT8 *)up_sampled_buffer,
|
|
(UINT8 *)up_sampled_buffer + bytes_needed,
|
|
btc_aa_src_cb.media_feeding_state.pcm.aa_feed_residue);
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
/*******************************************************************************
|
|
**
|
|
** Function btc_media_aa_prep_sbc_2_send
|
|
**
|
|
** Description
|
|
**
|
|
** Returns void
|
|
**
|
|
*******************************************************************************/
|
|
static void btc_media_aa_prep_sbc_2_send(UINT8 nb_frame)
|
|
{
|
|
BT_HDR *p_buf;
|
|
UINT16 blocm_x_subband = btc_aa_src_cb.encoder.s16NumOfSubBands *
|
|
btc_aa_src_cb.encoder.s16NumOfBlocks;
|
|
|
|
while (nb_frame) {
|
|
if (NULL == (p_buf = osi_malloc(BTC_MEDIA_AA_BUF_SIZE))) {
|
|
APPL_TRACE_ERROR ("ERROR btc_media_aa_prep_sbc_2_send no buffer TxCnt %d ",
|
|
fixed_queue_length(btc_aa_src_cb.TxAaQ));
|
|
return;
|
|
}
|
|
|
|
/* Init buffer */
|
|
p_buf->offset = BTC_MEDIA_AA_SBC_OFFSET;
|
|
p_buf->len = 0;
|
|
p_buf->layer_specific = 0;
|
|
|
|
do {
|
|
/* Write @ of allocated buffer in encoder.pu8Packet */
|
|
btc_aa_src_cb.encoder.pu8Packet = (UINT8 *) (p_buf + 1) + p_buf->offset + p_buf->len;
|
|
/* Fill allocated buffer with 0 */
|
|
memset(btc_aa_src_cb.encoder.as16PcmBuffer, 0, blocm_x_subband
|
|
* btc_aa_src_cb.encoder.s16NumOfChannels);
|
|
|
|
/* Read PCM data and upsample them if needed */
|
|
if (btc_media_aa_read_feeding()) {
|
|
/* SBC encode and descramble frame */
|
|
SBC_Encoder(&(btc_aa_src_cb.encoder));
|
|
A2D_SbcChkFrInit(btc_aa_src_cb.encoder.pu8Packet);
|
|
A2D_SbcDescramble(btc_aa_src_cb.encoder.pu8Packet, btc_aa_src_cb.encoder.u16PacketLength);
|
|
/* Update SBC frame length */
|
|
p_buf->len += btc_aa_src_cb.encoder.u16PacketLength;
|
|
nb_frame--;
|
|
p_buf->layer_specific++;
|
|
} else {
|
|
APPL_TRACE_WARNING("btc_media_aa_prep_sbc_2_send underflow %d, %d",
|
|
nb_frame, btc_aa_src_cb.media_feeding_state.pcm.aa_feed_residue);
|
|
btc_aa_src_cb.media_feeding_state.pcm.counter += nb_frame *
|
|
btc_aa_src_cb.encoder.s16NumOfSubBands *
|
|
btc_aa_src_cb.encoder.s16NumOfBlocks *
|
|
btc_aa_src_cb.media_feeding.cfg.pcm.num_channel *
|
|
btc_aa_src_cb.media_feeding.cfg.pcm.bit_per_sample / 8;
|
|
/* no more pcm to read */
|
|
nb_frame = 0;
|
|
|
|
/* break read loop if timer was stopped (media task stopped) */
|
|
if ( btc_aa_src_cb.is_tx_timer == FALSE ) {
|
|
osi_free(p_buf);
|
|
return;
|
|
}
|
|
}
|
|
|
|
} while (((p_buf->len + btc_aa_src_cb.encoder.u16PacketLength) < btc_aa_src_cb.TxAaMtuSize)
|
|
&& (p_buf->layer_specific < 0x0F) && nb_frame);
|
|
|
|
if (p_buf->len) {
|
|
/* timestamp of the media packet header represent the TS of the first SBC frame
|
|
i.e the timestamp before including this frame */
|
|
*((UINT32 *) (p_buf + 1)) = btc_aa_src_cb.timestamp;
|
|
|
|
btc_aa_src_cb.timestamp += p_buf->layer_specific * blocm_x_subband;
|
|
|
|
if (btc_aa_src_cb.tx_flush) {
|
|
APPL_TRACE_DEBUG("### tx suspended, discarded frame ###");
|
|
|
|
if (fixed_queue_length(btc_aa_src_cb.TxAaQ) > 0) {
|
|
btc_a2dp_source_flush_q(btc_aa_src_cb.TxAaQ);
|
|
}
|
|
|
|
osi_free(p_buf);
|
|
return;
|
|
}
|
|
|
|
/* Enqueue the encoded SBC frame in AA Tx Queue */
|
|
fixed_queue_enqueue(btc_aa_src_cb.TxAaQ, p_buf);
|
|
} else {
|
|
osi_free(p_buf);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*******************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_prep_2_send
|
|
**
|
|
** Description
|
|
**
|
|
** Returns void
|
|
**
|
|
*******************************************************************************/
|
|
static void btc_a2dp_source_prep_2_send(UINT8 nb_frame)
|
|
{
|
|
// Check for TX queue overflow
|
|
if (nb_frame > MAX_OUTPUT_A2DP_SRC_FRAME_QUEUE_SZ) {
|
|
nb_frame = MAX_OUTPUT_A2DP_SRC_FRAME_QUEUE_SZ;
|
|
}
|
|
|
|
if (fixed_queue_length(btc_aa_src_cb.TxAaQ) > (MAX_OUTPUT_A2DP_SRC_FRAME_QUEUE_SZ - nb_frame)) {
|
|
APPL_TRACE_WARNING("TX Q overflow: %d/%d",
|
|
fixed_queue_length(btc_aa_src_cb.TxAaQ), MAX_OUTPUT_A2DP_SRC_FRAME_QUEUE_SZ - nb_frame);
|
|
}
|
|
|
|
while (fixed_queue_length(btc_aa_src_cb.TxAaQ) > (MAX_OUTPUT_A2DP_SRC_FRAME_QUEUE_SZ - nb_frame)) {
|
|
osi_free(fixed_queue_try_dequeue(btc_aa_src_cb.TxAaQ));
|
|
}
|
|
|
|
// Transcode frame
|
|
|
|
switch (btc_aa_src_cb.TxTranscoding) {
|
|
case BTC_MEDIA_TRSCD_PCM_2_SBC:
|
|
btc_media_aa_prep_sbc_2_send(nb_frame);
|
|
break;
|
|
|
|
default:
|
|
APPL_TRACE_ERROR("%s unsupported transcoding format 0x%x", __func__, btc_aa_src_cb.TxTranscoding);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/*******************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_send_aa_frame
|
|
**
|
|
** Description
|
|
**
|
|
** Returns void
|
|
**
|
|
*******************************************************************************/
|
|
static void btc_a2dp_source_send_aa_frame(void)
|
|
{
|
|
UINT8 nb_frame_2_send;
|
|
|
|
/* get the number of frame to send */
|
|
nb_frame_2_send = btc_get_num_aa_frame();
|
|
|
|
if (nb_frame_2_send != 0) {
|
|
/* format and Q buffer to send */
|
|
btc_a2dp_source_prep_2_send(nb_frame_2_send);
|
|
}
|
|
|
|
/* send it */
|
|
LOG_VERBOSE("%s: send %d frames", __FUNCTION__, nb_frame_2_send);
|
|
bta_av_ci_src_data_ready(BTA_AV_CHNL_AUDIO);
|
|
}
|
|
|
|
static void btc_a2dp_source_handle_timer(UNUSED_ATTR void *context)
|
|
{
|
|
log_tstamps_us("media task tx timer");
|
|
|
|
#if (BTA_AV_INCLUDED == TRUE)
|
|
if (btc_aa_src_cb.is_tx_timer == TRUE) {
|
|
btc_a2dp_source_send_aa_frame();
|
|
} else {
|
|
APPL_TRACE_WARNING("Media task Scheduled after Suspend");
|
|
}
|
|
#endif
|
|
}
|
|
|
|
static void btc_a2dp_source_alarm_cb(UNUSED_ATTR void *context)
|
|
{
|
|
btc_a2dp_source_data_post(BTC_A2DP_SOURCE_DATA_EVT);
|
|
}
|
|
|
|
/*******************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_feeding_state_reset
|
|
**
|
|
** Description Reset the media feeding state
|
|
**
|
|
** Returns void
|
|
**
|
|
*******************************************************************************/
|
|
static void btc_a2dp_source_feeding_state_reset(void)
|
|
{
|
|
/* By default, just clear the entire state */
|
|
memset(&btc_aa_src_cb.media_feeding_state, 0, sizeof(btc_aa_src_cb.media_feeding_state));
|
|
|
|
if (btc_aa_src_cb.TxTranscoding == BTC_MEDIA_TRSCD_PCM_2_SBC) {
|
|
btc_aa_src_cb.media_feeding_state.pcm.bytes_per_tick =
|
|
(btc_aa_src_cb.media_feeding.cfg.pcm.sampling_freq *
|
|
btc_aa_src_cb.media_feeding.cfg.pcm.bit_per_sample / 8 *
|
|
btc_aa_src_cb.media_feeding.cfg.pcm.num_channel *
|
|
BTC_MEDIA_TIME_TICK_MS) / 1000;
|
|
|
|
APPL_TRACE_WARNING("pcm bytes per tick %d",
|
|
(int)btc_aa_src_cb.media_feeding_state.pcm.bytes_per_tick);
|
|
}
|
|
}
|
|
|
|
/*******************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_aa_start_tx
|
|
**
|
|
** Description Start media task encoding
|
|
**
|
|
** Returns void
|
|
**
|
|
*******************************************************************************/
|
|
static void btc_a2dp_source_aa_start_tx(void)
|
|
{
|
|
APPL_TRACE_DEBUG("btc_a2dp_source_aa_start_tx is timer %d, feeding mode %d",
|
|
btc_aa_src_cb.is_tx_timer, btc_aa_src_cb.feeding_mode);
|
|
|
|
btc_aa_src_cb.is_tx_timer = TRUE;
|
|
last_frame_us = 0;
|
|
|
|
/* Reset the media feeding state */
|
|
btc_a2dp_source_feeding_state_reset();
|
|
|
|
APPL_TRACE_EVENT("starting timer %dms", BTC_MEDIA_TIME_TICK_MS);
|
|
|
|
assert(btc_aa_src_cb.media_alarm == NULL);
|
|
|
|
btc_aa_src_cb.media_alarm = osi_alarm_new("aaTx", btc_a2dp_source_alarm_cb, NULL, BTC_MEDIA_TIME_TICK_MS);
|
|
|
|
if (!btc_aa_src_cb.media_alarm) {
|
|
LOG_ERROR("%s unable to allocate media alarm.", __func__);
|
|
return;
|
|
}
|
|
|
|
osi_alarm_set_periodic(btc_aa_src_cb.media_alarm, BTC_MEDIA_TIME_TICK_MS);
|
|
}
|
|
|
|
/*******************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_aa_stop_tx
|
|
**
|
|
** Description Stop media task encoding
|
|
**
|
|
** Returns void
|
|
**
|
|
*******************************************************************************/
|
|
static void btc_a2dp_source_aa_stop_tx(void)
|
|
{
|
|
APPL_TRACE_DEBUG("%s is_tx_timer: %d", __func__, btc_aa_src_cb.is_tx_timer);
|
|
|
|
const bool send_ack = (btc_aa_src_cb.is_tx_timer != FALSE);
|
|
|
|
/* Stop the timer first */
|
|
if (btc_aa_src_cb.media_alarm) {
|
|
osi_alarm_cancel(btc_aa_src_cb.media_alarm);
|
|
osi_alarm_free(btc_aa_src_cb.media_alarm);
|
|
}
|
|
btc_aa_src_cb.media_alarm = NULL;
|
|
btc_aa_src_cb.is_tx_timer = FALSE;
|
|
|
|
/* Try to send acknowldegment once the media stream is
|
|
stopped. This will make sure that the A2DP HAL layer is
|
|
un-blocked on wait for acknowledgment for the sent command.
|
|
This resolves a corner cases AVDTP SUSPEND collision
|
|
when the DUT and the remote device issue SUSPEND simultaneously
|
|
and due to the processing of the SUSPEND request from the remote,
|
|
the media path is torn down. If the A2DP HAL happens to wait
|
|
for ACK for the initiated SUSPEND, it would never receive it casuing
|
|
a block/wait. Due to this acknowledgement, the A2DP HAL is guranteed
|
|
to get the ACK for any pending command in such cases. */
|
|
|
|
if (send_ack) {
|
|
btc_a2dp_control_command_ack(ESP_A2D_MEDIA_CTRL_ACK_SUCCESS);
|
|
}
|
|
|
|
/* audio engine stopped, reset tx suspended flag */
|
|
btc_aa_src_cb.tx_flush = 0;
|
|
last_frame_us = 0;
|
|
|
|
/* Reset the feeding state */
|
|
btc_a2dp_source_feeding_state_reset();
|
|
}
|
|
|
|
/*******************************************************************************
|
|
**
|
|
** Function btc_a2dp_source_flush_q
|
|
**
|
|
** Description
|
|
**
|
|
** Returns void
|
|
**
|
|
*******************************************************************************/
|
|
static void btc_a2dp_source_flush_q(fixed_queue_t *p_q)
|
|
{
|
|
while (! fixed_queue_is_empty(p_q)) {
|
|
osi_free(fixed_queue_try_dequeue(p_q));
|
|
}
|
|
}
|
|
|
|
static void btc_a2dp_source_thread_init(UNUSED_ATTR void *context)
|
|
{
|
|
APPL_TRACE_EVENT("%s\n", __func__);
|
|
memset(&btc_aa_src_cb, 0, sizeof(btc_aa_src_cb));
|
|
|
|
btc_a2dp_source_state = BTC_A2DP_SOURCE_STATE_ON;
|
|
|
|
btc_aa_src_cb.TxAaQ = fixed_queue_new(SIZE_MAX);
|
|
|
|
btc_a2dp_control_init();
|
|
}
|
|
|
|
static void btc_a2dp_source_thread_cleanup(UNUSED_ATTR void *context)
|
|
{
|
|
/* make sure no channels are restarted while shutting down */
|
|
btc_a2dp_source_state = BTC_A2DP_SOURCE_STATE_SHUTTING_DOWN;
|
|
|
|
btc_a2dp_control_set_datachnl_stat(FALSE);
|
|
/* Clear media task flag */
|
|
btc_a2dp_source_state = BTC_A2DP_SOURCE_STATE_OFF;
|
|
|
|
btc_a2dp_control_cleanup();
|
|
|
|
fixed_queue_free(btc_aa_src_cb.TxAaQ, osi_free_func);
|
|
}
|
|
|
|
#endif /* BTC_AV_INCLUDED */
|