pinebuds/platform/drivers/bt/bt_drv_common.c

566 lines
16 KiB
C

/***************************************************************************
*
* Copyright 2015-2019 BES.
* All rights reserved. All unpublished rights reserved.
*
* No part of this work may be used or reproduced in any form or by any
* means, or stored in a database or retrieval system, without prior written
* permission of BES.
*
* Use of this work is governed by a license granted by BES.
* This work contains confidential and proprietary information of
* BES. which is protected by copyright, trade secret,
* trademark and other intellectual property rights.
*
****************************************************************************/
#include "app_utils.h"
#include "besbt_string.h"
#include "bt_drv.h"
#include "bt_drv_interface.h"
#include "bt_drv_internal.h"
#include "bt_drv_reg_op.h"
#include "hal_chipid.h"
#include "hal_i2c.h"
#include "hal_uart.h"
#include "plat_types.h"
#include "string.h"
#define BT_DUMP_LEN_LINE 100
#define SLOT_SIZE 625
#define CLK_SUB(clock_a, clock_b) \
((uint32_t)(((clock_a) - (clock_b)) & MAX_SLOT_CLOCK))
#define CLK_ADD_2(clock_a, clock_b) \
((uint32_t)(((clock_a) + (clock_b)) & MAX_SLOT_CLOCK))
struct bt_cb_tag bt_drv_func_cb = {NULL};
struct bt_cb_tag *bt_drv_get_func_cb_ptr(void) { return &bt_drv_func_cb; }
/// only used for bt chip write patch data for speed up
void btdrv_memory_copy(uint32_t *dest, const uint32_t *src, uint16_t length) {
// memcpy(dest,src,length);
uint16_t i;
for (i = 0; i < length / 4; i++) {
*dest++ = *src++;
}
}
const uint8_t lmp_sniffer_filter_tab[51] = {
50, // array length
LMP_NAME_REQ_OPCODE,
LMP_NAME_RES_OPCODE,
LMP_ACCEPTED_OPCODE,
LMP_NOT_ACCEPTED_OPCODE,
LMP_CLK_OFF_REQ_OPCODE,
LMP_CLK_OFF_RES_OPCODE,
// LMP_DETACH_OPCODE,
LMP_INRAND_OPCODE,
LMP_COMBKEY_OPCODE,
LMP_UNITKEY_OPCODE,
LMP_AURAND_OPCODE,
LMP_SRES_OPCODE,
LMP_TEMPRAND_OPCODE,
LMP_TEMPKEY_OPCODE,
LMP_ENC_MODE_REQ_OPCODE,
LMP_ENC_KEY_SIZE_REQ_OPCODE,
LMP_START_ENC_REQ_OPCODE,
LMP_STOP_ENC_REQ_OPCODE,
LMP_SWITCH_REQ_OPCODE,
LMP_HOLD_OPCODE,
LMP_HOLD_REQ_OPCODE,
LMP_SNIFF_REQ_OPCODE,
// LMP_UNSNIFF_REQ_OPCODE,
LMP_PARK_REQ_OPCODE,
LMP_SET_BSWIN_OPCODE,
LMP_MODIF_BEACON_OPCODE,
LMP_UNPARK_BD_REQ_OPCODE,
LMP_UNPARK_PM_REQ_OPCODE,
// LMP_INCR_PWR_REQ_OPCODE,
// LMP_DECR_PWR_REQ_OPCODE,
// LMP_MAX_PWR_OPCODE,
// LMP_MIN_PWR_OPCODE,
// LMP_AUTO_RATE_OPCODE,
// LMP_PREF_RATE_OPCODE,
LMP_VER_REQ_OPCODE,
LMP_VER_RES_OPCODE,
LMP_FEATS_REQ_OPCODE,
LMP_FEATS_RES_OPCODE,
// LMP_QOS_OPCODE,
// LMP_QOS_REQ_OPCODE,
LMP_SCO_LINK_REQ_OPCODE,
LMP_RMV_SCO_LINK_REQ_OPCODE,
// LMP_MAX_SLOT_OPCODE,
// LMP_MAX_SLOT_REQ_OPCODE,
LMP_TIMING_ACCU_REQ_OPCODE,
LMP_TIMING_ACCU_RES_OPCODE,
LMP_SETUP_CMP_OPCODE,
LMP_USE_SEMI_PERM_KEY_OPCODE,
LMP_HOST_CON_REQ_OPCODE,
LMP_SLOT_OFF_OPCODE,
LMP_PAGE_MODE_REQ_OPCODE,
LMP_PAGE_SCAN_MODE_REQ_OPCODE,
// LMP_SUPV_TO_OPCODE,
LMP_TEST_ACTIVATE_OPCODE,
LMP_TEST_CTRL_OPCODE,
LMP_ENC_KEY_SIZE_MASK_REQ_OPCODE,
LMP_ENC_KEY_SIZE_MASK_RES_OPCODE,
// LMP_SET_AFH_OPCODE,
LMP_ENCAPS_HDR_OPCODE,
LMP_ENCAPS_PAYL_OPCODE,
LMP_SP_CFM_OPCODE,
LMP_SP_NB_OPCODE,
LMP_DHKEY_CHK_OPCODE,
LMP_PAUSE_ENC_AES_REQ_OPCODE,
};
const uint8_t lmp_ext_sniffer_filter_tab[18] = {
17, // array length
LMP_ACCEPTED_EXT_EXTOPCODE,
LMP_NOT_ACCEPTED_EXT_EXTOPCODE,
LMP_FEATS_REQ_EXT_EXTOPCODE,
LMP_FEATS_RES_EXT_EXTOPCODE,
// LMP_CLK_ADJ_EXTOPCODE,
// LMP_CLK_ADJ_ACK_EXTOPCODE,
// LMP_CLK_ADJ_REQ_EXTOPCODE,
// LMP_PKT_TYPE_TBL_REQ_EXTOPCODE,
// LMP_ESCO_LINK_REQ_EXTOPCODE,
// LMP_RMV_ESCO_LINK_REQ_EXTOPCODE,
// LMP_CH_CLASS_REQ_EXTOPCODE,
// LMP_CH_CLASS_EXTOPCODE,
LMP_SSR_REQ_EXTOPCODE,
LMP_SSR_RES_EXTOPCODE,
LMP_PAUSE_ENC_REQ_EXTOPCODE,
LMP_RESUME_ENC_REQ_EXTOPCODE,
LMP_IO_CAP_REQ_EXTOPCODE,
LMP_IO_CAP_RES_EXTOPCODE,
LMP_NUM_COMPARISON_FAIL_EXTOPCODE,
LMP_PASSKEY_FAIL_EXTOPCODE,
LMP_OOB_FAIL_EXTOPCODE,
LMP_KEYPRESS_NOTIF_EXTOPCODE,
// LMP_PWR_CTRL_REQ_EXTOPCODE,
LMP_PWR_CTRL_RES_EXTOPCODE,
LMP_PING_REQ_EXTOPCODE,
LMP_PING_RES_EXTOPCODE,
};
const uint8_t lmp_ext_sniffer_fast_cfm_tab[4] = {
3, // array length
LMP_ACCEPTED_EXT_EXTOPCODE,
LMP_NOT_ACCEPTED_EXT_EXTOPCODE,
LMP_CH_CLASS_EXTOPCODE,
};
extern uint8_t sleep_param[];
void btdrv_set_lpo_times(void) {
*(uint32_t *)0xd0330044 &= ~0xff;
*(uint32_t *)0xd0330044 |= (uint32_t)(sleep_param[4] - 1);
}
int btdrv_slave2master_clkcnt_convert(uint32_t local_clk, uint16_t local_cnt,
int32_t clk_offset, uint16_t bit_offset,
uint32_t *master_clk,
uint16_t *master_cnt) {
// Adjust bit offset and clock offset if needed
uint32_t new_clk;
int16_t new_cnt;
new_clk = CLK_ADD_2(local_clk, clk_offset);
new_cnt = (int16_t)local_cnt + bit_offset;
if (new_cnt > SLOT_SIZE) {
new_cnt -= SLOT_SIZE;
new_clk = CLK_SUB(new_clk, 1);
}
*master_clk = new_clk;
*master_cnt = new_cnt;
return 0;
}
void bt_drv_set_fa_invert_enable(uint8_t en) {
BTDIGITAL_REG_SET_FIELD(0xd0220468, 1, 17, en);
BT_DRV_TRACE(1, "BT_DRV:set fa invert en=%d", en);
}
void btdrv_trigger_coredump(void) {
uint8_t *null_pointer = 0;
*null_pointer = 0; // MPU trigger coredump
}
int btdrv_clkcnt_diff(int32_t clk1, int16_t cnt1, int32_t clk2, int16_t cnt2,
int32_t *diff_clk, uint16_t *diff_bit) {
int32_t new_clk;
int16_t new_cnt;
int diff_us;
new_clk = (int32_t)clk1 - (int32_t)clk2;
new_cnt = cnt1 - cnt2;
if (new_cnt < 0) {
new_cnt += SLOT_SIZE;
} else {
new_clk -= 1;
}
*diff_clk = new_clk;
*diff_bit = new_cnt;
diff_us = new_clk * SLOT_SIZE + (SLOT_SIZE - new_cnt);
return diff_us;
}
void WEAK btdrv_set_controller_trace_enable(uint8_t trace_level) {}
void WEAK btdrv_set_lmp_trace_enable(void) {}
static bool g_controller_trace_dump_enable = false;
bool btdrv_get_controller_trace_dump_enable(void) {
return g_controller_trace_dump_enable;
}
void btdrv_set_controller_trace_dump_enable(void) {
g_controller_trace_dump_enable = true;
}
void btdrv_btc_fault_dump(void) {
app_wdt_close();
bt_drv_reg_op_crash_dump();
}
void btdrv_dump_mem(uint8_t *dump_mem_start, uint32_t dump_length,
uint8_t dump_type) {
#if CONTROLLER_DUMP_ENABLE
uint32_t dump_counter = 0;
uint32_t dump_line = 0;
hal_sys_timer_delay(MS_TO_TICKS(100));
switch (dump_type) {
case BT_SUB_SYS_TYPE:
BT_DRV_TRACE(1, "controller dump start, dump len=0x%x", dump_length);
break;
case MCU_SYS_TYPE:
BT_DRV_TRACE(1, "mcu dump start, dump len=0x%x", dump_length);
break;
case BT_EM_AREA_1_TYPE:
BT_DRV_TRACE(1, "em_area1 dump start, dump len=0x%x", dump_length);
break;
case BT_EM_AREA_2_TYPE:
BT_DRV_TRACE(1, "em_area2 dump start, dump len=0x%x", dump_length);
break;
}
while (dump_counter < dump_length) {
uint32_t data_left = dump_length - dump_counter;
dump_line += 1;
if (data_left >= BT_DUMP_LEN_LINE) {
BT_DRV_TRACE(1, "dump_line=%d", dump_line);
DUMP8("%02x", dump_mem_start, BT_DUMP_LEN_LINE);
hal_trace_flush_buffer();
dump_mem_start += BT_DUMP_LEN_LINE;
dump_counter += BT_DUMP_LEN_LINE;
hal_sys_timer_delay(MS_TO_TICKS(50));
} else {
BT_DRV_TRACE(1, "dump_line=%d", dump_line);
DUMP8("%02x", dump_mem_start, data_left);
hal_trace_flush_buffer();
hal_sys_timer_delay(MS_TO_TICKS(50));
break;
}
}
switch (dump_type) {
case BT_SUB_SYS_TYPE:
BT_DRV_TRACE(0, "controller dump end");
break;
case MCU_SYS_TYPE:
BT_DRV_TRACE(0, "mcu dump end");
break;
case BT_EM_AREA_1_TYPE:
BT_DRV_TRACE(0, "em_area1 dump end");
break;
case BT_EM_AREA_2_TYPE:
BT_DRV_TRACE(0, "em_area2 dump end");
break;
}
#endif
}
void btdrv_trace_config(BT_CONTROLER_TRACE_TYPE trace_config) {
if (trace_config & BT_CONTROLER_TRACE_TYPE_INTERSYS) {
btdrv_set_intersys_trace_enable();
}
if (trace_config & BT_CONTROLER_TRACE_TYPE_CONTROLLER) {
btdrv_set_controller_trace_enable(0x00);
}
if (trace_config & BT_CONTROLER_TRACE_TYPE_LMP_TRACE) {
btdrv_set_lmp_trace_enable();
}
if (trace_config & BT_CONTROLER_TRACE_TYPE_SPUV_HCI_BUFF) {
btdrv_set_spuv_hci_buff_trace_enable();
}
if (trace_config & BT_CONTROLER_FILTER_TRACE_TYPE_A2DP_STREAM) {
btdrv_set_a2dp_stream_trace_disable();
}
#if CONTROLLER_MEM_LOG_ENABLE
btdrv_set_controller_trace_dump_enable();
#endif
}
bool bt_drv_is_enhanced_ibrt_rom(void) {
bool ret = false;
#ifdef CHIP_BEST2300P
if (hal_get_chip_metal_id() >= HAL_CHIP_METAL_ID_1)
ret = true;
#elif CHIP_BEST1400
if (hal_get_chip_metal_id() >= HAL_CHIP_METAL_ID_2)
ret = true;
#elif CHIP_BEST1402
ret = true;
#elif CHIP_BEST2300A
ret = true;
#elif CHIP_BEST2300
ret = true;
#endif
return ret;
}
bool bt_drv_is_esco_auto_accept_support(void) {
bool ret = false;
#ifdef CHIP_BEST2300P
if (hal_get_chip_metal_id() >= HAL_CHIP_METAL_ID_2)
ret = true;
#elif CHIP_BEST1400
if (hal_get_chip_metal_id() >= HAL_CHIP_METAL_ID_4)
ret = true;
#elif CHIP_BEST1402
if (hal_get_chip_metal_id() >= HAL_CHIP_METAL_ID_1)
ret = true;
#elif CHIP_BEST2300A
ret = true;
#endif
return ret;
}
bool btdrv_is_acl_ecc_softbit_support(void) {
bool ret = false;
#ifdef CHIP_BEST2300A
ret = true;
#endif
return ret;
}
void btdrv_softbit_enable(uint16_t connhdl, uint8_t type1, uint8_t type2,
uint8_t type3, uint8_t num) {
if (btdrv_is_acl_ecc_softbit_support()) {
#if (defined(SOFTBIT_EN))
btdrv_softbit_config(connhdl, type1, type2, type3, num);
#endif
}
}
void bt_drv_bt_tport_type_config(void) {
uint32_t tport_type = 0xb1b1;
#ifdef __BT_DEBUG_TPORTS__
tport_type = TPORT_TYPE;
#endif
BTDIGITAL_REG(0xd0220050) = tport_type;
BTDIGITAL_REG(0xd0340000) = 0x23620200;
BT_DRV_TRACE(1, "BT_DRV: tport type=0x%x", tport_type);
}
bool bt_drv_is_bes_fa_mode_en(void) {
bool ret = false;
#ifdef __BES_FA_MODE__
ret = true;
#endif
return ret;
}
bool bt_drv_is_support_multipoint_ibrt(void) {
bool ret = false;
return ret;
}
void bt_drv_enable_hw_spi(uint8_t elt_idx, bool hwspien) {
#ifdef CHIP_BEST2300A
bt_drv_reg_op_hw_spi_en_setf(elt_idx, hwspien);
#endif
}
struct btstack_chip_config_t {
uint16_t hci_dbg_set_sync_config_cmd_opcode;
uint16_t hci_dbg_set_sco_switch_cmd_opcode;
} g_bt_drv_btstack_chip_config = {
HCI_DBG_SET_SYNC_CONFIG_CMD_OPCODE,
HCI_DBG_SET_SCO_SWITCH_CMD_OPCODE,
};
void *bt_drv_get_btstack_chip_config(void) {
return &g_bt_drv_btstack_chip_config;
}
void bt_drv_hwspi_select(uint8_t link_id, uint8_t spi_idx) {
#ifdef __FA_RX_GAIN_CTRL__
btdrv_spi_trig_select(link_id, spi_idx);
#endif
}
/// BCH size
#define LD_BCH_SIZE 8
static const uint8_t ld_util_bch_ppolynom[LD_BCH_SIZE] = {
0x83, 0x84, 0x8D, 0x96, 0xBB, 0xCC, 0x54, 0xFC};
static const uint8_t ld_util_bch_gpolynom[LD_BCH_SIZE] = {
0x00, 0x00, 0x00, 0x05, 0x85, 0x71, 0x3D, 0xA9};
static void ld_util_bch_modulo(uint8_t *dividend, const uint8_t *divisor,
uint8_t *rest) {
uint8_t clock;
uint8_t carry;
uint8_t index;
uint8_t dividend_tmp[LD_BCH_SIZE];
// Copy dividend vector in a temporary vector
// And reset rest vector
for (index = 0; index < LD_BCH_SIZE; index++) {
dividend_tmp[index] = dividend[index];
rest[index] = 0;
}
// Execute 64 times the LFSR process
for (clock = 0; clock < 64; clock++) {
// Store bit Rest(degree-1) in carry (assume degree of G is 32)
carry = rest[3] & 0x02;
// the rest is shifted of 1 bit Left
// The MSB of rest if lost and the MSB of dividend is shifted in LSB
rest[0] = (rest[0] << 1) | ((rest[1] & 0x80) >> 7);
rest[1] = (rest[1] << 1) | ((rest[2] & 0x80) >> 7);
rest[2] = (rest[2] << 1) | ((rest[3] & 0x80) >> 7);
rest[3] = (rest[3] << 1) | ((rest[4] & 0x80) >> 7);
rest[4] = (rest[4] << 1) | ((rest[5] & 0x80) >> 7);
rest[5] = (rest[5] << 1) | ((rest[6] & 0x80) >> 7);
rest[6] = (rest[6] << 1) | ((rest[7] & 0x80) >> 7);
rest[7] = (rest[7] << 1) | ((dividend_tmp[0] & 0x80) >> 7);
// the dividend_tmp is shifted of 1 bit Left (a 0 is shifted in LSB)
dividend_tmp[0] = (dividend_tmp[0] << 1) | ((dividend_tmp[1] & 0x80) >> 7);
dividend_tmp[1] = (dividend_tmp[1] << 1) | ((dividend_tmp[2] & 0x80) >> 7);
dividend_tmp[2] = (dividend_tmp[2] << 1) | ((dividend_tmp[3] & 0x80) >> 7);
dividend_tmp[3] = (dividend_tmp[3] << 1) | ((dividend_tmp[4] & 0x80) >> 7);
dividend_tmp[4] = (dividend_tmp[4] << 1) | ((dividend_tmp[5] & 0x80) >> 7);
dividend_tmp[5] = (dividend_tmp[5] << 1) | ((dividend_tmp[6] & 0x80) >> 7);
dividend_tmp[6] = (dividend_tmp[6] << 1) | ((dividend_tmp[7] & 0x80) >> 7);
dividend_tmp[7] = (dividend_tmp[7] << 1);
// If bit carry value was 1
if (carry != 0) {
// rest = rest XOR Divisor
for (index = 0; index < LD_BCH_SIZE; index++) {
rest[index] ^= divisor[index];
}
}
}
}
void ld_util_bch_create(uint8_t *lap, uint8_t *bch) {
uint8_t vector[LD_BCH_SIZE];
uint8_t vector2[LD_BCH_SIZE];
uint8_t index;
// Copy lap and Appended in bch
if (lap[2] & 0x80) {
vector[0] = 0x4C; // If a23 = 1
} else {
vector[0] = 0xB0; // If a23 = 0
}
vector[0] |= lap[2] >> 6;
vector[1] = lap[2] << 2 | lap[1] >> 6;
vector[2] = lap[1] << 2 | lap[0] >> 6;
vector[3] = lap[0] << 2;
// Xor Vector and PN (Vector contains only 30 significant bits)
for (index = 0; index < 4; index++) {
vector[index] ^= ld_util_bch_ppolynom[index];
}
// Reset 34 last bits
vector[3] &= 0xFC;
vector[4] = 0;
vector[5] = 0;
vector[6] = 0;
vector[7] = 0;
// Generate Parity bits Vector Modulo G
ld_util_bch_modulo(vector, ld_util_bch_gpolynom, vector2);
// Create CodeWord (concatenate Modulo result and Xored Vector)
vector[3] |= vector2[3];
vector[4] = vector2[4];
vector[5] = vector2[5];
vector[6] = vector2[6];
vector[7] = vector2[7];
// Xor codeWord and PN
for (index = 0; index < 8; index++) {
bch[7 - index] = vector[index] ^ ld_util_bch_ppolynom[index];
}
}
void bt_drv_rssi_dump_handler(void) {
rx_agc_t bt_agc = {0, 0xf};
for (int i = 0; i < MAX_NB_ACTIVE_ACL; i++) {
bt_drv_reg_op_read_rssi_in_dbm(BT_ACL_CONHDL_BIT + i, &bt_agc);
BT_DRV_TRACE(3, "BT=0x%x,rssi=%d,gain=%d", (BT_ACL_CONHDL_BIT + i),
bt_agc.rssi, bt_agc.rxgain);
bt_agc.rssi = 0;
bt_agc.rxgain = 0xf;
}
rx_agc_t ble_agc = {0, 0xf};
bt_drv_reg_op_read_ble_rssi_in_dbm(0, &ble_agc);
BT_DRV_TRACE(2, "BLE rssi=%d,gain=%d\n", ble_agc.rssi, ble_agc.rxgain);
}
bool bt_drv_error_check_handler(void) {
bool ret = false;
#if defined(CHIP_BEST2300) || defined(CHIP_BEST1400) || \
defined(CHIP_BEST1402) || defined(CHIP_BEST2300P) || \
defined(CHIP_BEST2300A)
if (BTDIGITAL_REG(BT_ERRORTYPESTAT_ADDR) != 0 ||
BTDIGITAL_REG(BT_CONTROLLER_CRASH_DUMP_ADDR_BASE) != 0) {
BT_DRV_TRACE(1, "BT_DRV:digital assert,error code=0x%x",
BTDIGITAL_REG(BT_ERRORTYPESTAT_ADDR));
ret = true;
}
#endif
return ret;
}
size_t memcpy_s(void *dst, size_t dstMax, const void *src, size_t srcMax) {
memcpy(dst, src, srcMax);
return 0;
}
size_t memset_s(dst0, dstMax, c0, srcMax) void *dst0;
size_t dstMax;
int c0;
size_t srcMax;
{
memset(dst0, c0, srcMax);
return 0;
}