2021-03-11 15:53:46 +01:00

430 lines
12 KiB
C

#define __BTSTACK_FILE__ "port.c"
// include STM32 first to avoid warning about redefinition of UNUSED
#include "stm32f4xx_hal.h"
#include "main.h"
#include "port.h"
#include "btstack.h"
#include "btstack_debug.h"
#include "btstack_audio.h"
#include "btstack_chipset_cc256x.h"
#include "btstack_run_loop_embedded.h"
#include "btstack_tlv.h"
#include "btstack_tlv_flash_bank.h"
#include "ble/le_device_db_tlv.h"
#include "classic/btstack_link_key_db_static.h"
#include "classic/btstack_link_key_db_tlv.h"
#include "hal_flash_bank_stm32.h"
#ifdef ENABLE_SEGGER_RTT
#include "SEGGER_RTT.h"
#include "hci_dump_segger_rtt_stdout.h"
#else
#include "hci_dump_embedded_stdout.h"
#endif
//
extern UART_HandleTypeDef huart2;
extern UART_HandleTypeDef huart3;
//
static btstack_packet_callback_registration_t hci_event_callback_registration;
static const hci_transport_config_uart_t config = {
HCI_TRANSPORT_CONFIG_UART,
115200,
4000000,
1,
NULL
};
// hal_time_ms.h
#include "hal_time_ms.h"
uint32_t hal_time_ms(void){
return HAL_GetTick();
}
// hal_cpu.h implementation
#include "hal_cpu.h"
void hal_cpu_disable_irqs(void){
__disable_irq();
}
void hal_cpu_enable_irqs(void){
__enable_irq();
}
void hal_cpu_enable_irqs_and_sleep(void){
__enable_irq();
__asm__("wfe"); // go to sleep if event flag isn't set. if set, just clear it. IRQs set event flag
}
// hal_stdin.h
#include "hal_stdin.h"
static uint8_t stdin_buffer[1];
static void (*stdin_handler)(char c);
void hal_stdin_setup(void (*handler)(char c)){
stdin_handler = handler;
// start receiving
HAL_UART_Receive_IT(&huart2, &stdin_buffer[0], 1);
}
static void stdin_rx_complete(void){
if (stdin_handler){
(*stdin_handler)(stdin_buffer[0]);
}
HAL_UART_Receive_IT(&huart2, &stdin_buffer[0], 1);
}
// hal_uart_dma.h
// hal_uart_dma.c implementation
#include "hal_uart_dma.h"
static void dummy_handler(void);
// handlers
static void (*rx_done_handler)(void) = &dummy_handler;
static void (*tx_done_handler)(void) = &dummy_handler;
static void (*cts_irq_handler)(void) = &dummy_handler;
static void dummy_handler(void){};
static int hal_uart_needed_during_sleep;
void hal_uart_dma_set_sleep(uint8_t sleep){
// RTS is on PD12 - manually set it during sleep
GPIO_InitTypeDef RTS_InitStruct;
RTS_InitStruct.Pin = GPIO_PIN_12;
RTS_InitStruct.Pull = GPIO_NOPULL;
RTS_InitStruct.Alternate = GPIO_AF7_USART3;
if (sleep){
HAL_GPIO_WritePin(GPIOD, GPIO_PIN_12, GPIO_PIN_SET);
RTS_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
RTS_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
} else {
RTS_InitStruct.Mode = GPIO_MODE_AF_PP;
RTS_InitStruct.Speed = GPIO_SPEED_FREQ_VERY_HIGH;
}
HAL_GPIO_Init(GPIOD, &RTS_InitStruct);
// if (sleep){
// HAL_GPIO_WritePin(GPIOD, GPIO_PIN_12, GPIO_PIN_SET);
// }
hal_uart_needed_during_sleep = !sleep;
}
// reset Bluetooth using n_shutdown
static void bluetooth_power_cycle(void){
printf("Bluetooth power cycle\n");
HAL_GPIO_WritePin( CC_nSHUTD_GPIO_Port, CC_nSHUTD_Pin, GPIO_PIN_RESET );
HAL_Delay( 250 );
HAL_GPIO_WritePin( CC_nSHUTD_GPIO_Port, CC_nSHUTD_Pin, GPIO_PIN_SET );
HAL_Delay( 500 );
}
void HAL_UART_TxCpltCallback(UART_HandleTypeDef *huart){
if (huart == &huart3){
(*tx_done_handler)();
}
}
void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart){
if (huart == &huart3){
(*rx_done_handler)();
}
if (huart == &huart2){
stdin_rx_complete();
}
}
void hal_uart_dma_init(void){
bluetooth_power_cycle();
}
void hal_uart_dma_set_block_received( void (*the_block_handler)(void)){
rx_done_handler = the_block_handler;
}
void hal_uart_dma_set_block_sent( void (*the_block_handler)(void)){
tx_done_handler = the_block_handler;
}
void EXTI15_10_IRQHandler(void){
__HAL_GPIO_EXTI_CLEAR_IT(GPIO_PIN_11);
if (cts_irq_handler){
(*cts_irq_handler)();
}
}
void hal_uart_dma_set_csr_irq_handler( void (*the_irq_handler)(void)){
GPIO_InitTypeDef CTS_InitStruct = {
.Pin = GPIO_PIN_11,
.Mode = GPIO_MODE_AF_PP,
.Pull = GPIO_PULLUP,
.Speed = GPIO_SPEED_FREQ_VERY_HIGH,
.Alternate = GPIO_AF7_USART3,
};
if( the_irq_handler ) {
/* Configure the EXTI11 interrupt (USART3_CTS is on PD11) */
HAL_NVIC_EnableIRQ( EXTI15_10_IRQn );
CTS_InitStruct.Mode = GPIO_MODE_IT_RISING;
CTS_InitStruct.Pull = GPIO_NOPULL;
HAL_GPIO_Init( GPIOD, &CTS_InitStruct );
log_info("enabled CTS irq");
}
else {
CTS_InitStruct.Mode = GPIO_MODE_AF_PP;
CTS_InitStruct.Pull = GPIO_PULLUP;
HAL_GPIO_Init( GPIOD, &CTS_InitStruct );
HAL_NVIC_DisableIRQ( EXTI15_10_IRQn );
log_info("disabled CTS irq");
}
cts_irq_handler = the_irq_handler;
}
int hal_uart_dma_set_baud(uint32_t baud){
huart3.Init.BaudRate = baud;
HAL_UART_Init(&huart3);
return 0;
}
void hal_uart_dma_send_block(const uint8_t *data, uint16_t size){
HAL_UART_Transmit_DMA( &huart3, (uint8_t *) data, size);
}
void hal_uart_dma_receive_block(uint8_t *data, uint16_t size){
HAL_UART_Receive_DMA( &huart3, data, size );
}
#ifndef ENABLE_SEGGER_RTT
/**
* Use USART_CONSOLE as a console.
* This is a syscall for newlib
* @param file
* @param ptr
* @param len
* @return
*/
#include <stdio.h>
#include <unistd.h>
#include <errno.h>
int _write(int file, char *ptr, int len);
int _write(int file, char *ptr, int len){
#if 1
uint8_t cr = '\r';
int i;
if (file == STDOUT_FILENO || file == STDERR_FILENO) {
for (i = 0; i < len; i++) {
if (ptr[i] == '\n') {
HAL_UART_Transmit( &huart2, &cr, 1, HAL_MAX_DELAY );
}
HAL_UART_Transmit( &huart2, (uint8_t *) &ptr[i], 1, HAL_MAX_DELAY );
}
return i;
}
errno = EIO;
return -1;
#else
return len;
#endif
}
int _read(int file, char * ptr, int len){
UNUSED(file);
UNUSED(ptr);
UNUSED(len);
return -1;
}
#endif
int _close(int file){
UNUSED(file);
return -1;
}
int _isatty(int file){
UNUSED(file);
return -1;
}
int _lseek(int file){
UNUSED(file);
return -1;
}
int _fstat(int file){
UNUSED(file);
return -1;
}
// main.c
static void packet_handler (uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){
UNUSED(size);
UNUSED(channel);
bd_addr_t local_addr;
if (packet_type != HCI_EVENT_PACKET) return;
switch(hci_event_packet_get_type(packet)){
case BTSTACK_EVENT_STATE:
if (btstack_event_state_get_state(packet) != HCI_STATE_WORKING) return;
gap_local_bd_addr(local_addr);
printf("BTstack up and running on %s.\n", bd_addr_to_str(local_addr));
break;
case HCI_EVENT_COMMAND_COMPLETE:
if (HCI_EVENT_IS_COMMAND_COMPLETE(packet, hci_read_local_version_information)){
uint16_t manufacturer = little_endian_read_16(packet, 10);
uint16_t lmp_subversion = little_endian_read_16(packet, 12);
// assert manufacturer is TI
if (manufacturer != BLUETOOTH_COMPANY_ID_TEXAS_INSTRUMENTS_INC){
printf("ERROR: Expected Bluetooth Chipset from TI but got manufacturer 0x%04x\n", manufacturer);
break;
}
// assert correct init script is used based on expected lmp_subversion
if (lmp_subversion != btstack_chipset_cc256x_lmp_subversion()){
printf("Error: LMP Subversion does not match initscript! ");
printf("Your initscripts is for %s chipset\n", btstack_chipset_cc256x_lmp_subversion() < lmp_subversion ? "an older" : "a newer");
printf("Please update Makefile to include the appropriate bluetooth_init_cc256???.c file\n");
break;
}
}
break;
default:
break;
}
}
static btstack_tlv_flash_bank_t btstack_tlv_flash_bank_context;
static hal_flash_bank_stm32_t hal_flash_bank_context;
#define HAL_FLASH_BANK_SIZE (128 * 1024)
#define HAL_FLASH_BANK_0_ADDR 0x080C0000
#define HAL_FLASH_BANK_1_ADDR 0x080E0000
#define HAL_FLASH_BANK_0_SECTOR FLASH_SECTOR_10
#define HAL_FLASH_BANK_1_SECTOR FLASH_SECTOR_11
//
int btstack_main(int argc, char ** argv);
void port_main(void){
// start with BTstack init - especially configure HCI Transport
btstack_memory_init();
btstack_run_loop_init(btstack_run_loop_embedded_get_instance());
// uncomment to enable packet logger
#ifdef ENABLE_SEGGER_RTT
// hci_dump_init(hci_dump_segger_rtt_stdout_get_instance());
#else
// hci_dump_init(hci_dump_embedded_stdout_get_instance());
#endif
// init HCI
hci_init(hci_transport_h4_instance(btstack_uart_block_embedded_instance()), (void*) &config);
hci_set_chipset(btstack_chipset_cc256x_instance());
// setup TLV Flash Sector implementation
const hal_flash_bank_t * hal_flash_bank_impl = hal_flash_bank_stm32_init_instance(
&hal_flash_bank_context,
HAL_FLASH_BANK_SIZE,
HAL_FLASH_BANK_0_SECTOR,
HAL_FLASH_BANK_1_SECTOR,
HAL_FLASH_BANK_0_ADDR,
HAL_FLASH_BANK_1_ADDR);
const btstack_tlv_t * btstack_tlv_impl = btstack_tlv_flash_bank_init_instance(
&btstack_tlv_flash_bank_context,
hal_flash_bank_impl,
&hal_flash_bank_context);
// setup global tlv
btstack_tlv_set_instance(btstack_tlv_impl, &btstack_tlv_flash_bank_context);
// setup Link Key DB using TLV
const btstack_link_key_db_t * btstack_link_key_db = btstack_link_key_db_tlv_get_instance(btstack_tlv_impl, &btstack_tlv_flash_bank_context);
hci_set_link_key_db(btstack_link_key_db);
// setup LE Device DB using TLV
le_device_db_tlv_configure(btstack_tlv_impl, &btstack_tlv_flash_bank_context);
#ifdef HAVE_HAL_AUDIO
// setup audio
btstack_audio_sink_set_instance(btstack_audio_embedded_sink_get_instance());
btstack_audio_source_set_instance(btstack_audio_embedded_source_get_instance());
#endif
// inform about BTstack state
hci_event_callback_registration.callback = &packet_handler;
hci_add_event_handler(&hci_event_callback_registration);
// hand over to btstack embedded code
btstack_main(0, NULL);
// go
btstack_run_loop_execute();
}
#if 0
// Help with debugging hard faults - from FreeRTOS docu
// https://www.freertos.org/Debugging-Hard-Faults-On-Cortex-M-Microcontrollers.html
void prvGetRegistersFromStack( uint32_t *pulFaultStackAddress );
void prvGetRegistersFromStack( uint32_t *pulFaultStackAddress ) {
/* These are volatile to try and prevent the compiler/linker optimising them
away as the variables never actually get used. If the debugger won't show the
values of the variables, make them global my moving their declaration outside
of this function. */
volatile uint32_t r0;
volatile uint32_t r1;
volatile uint32_t r2;
volatile uint32_t r3;
volatile uint32_t r12;
volatile uint32_t lr; /* Link register. */
volatile uint32_t pc; /* Program counter. */
volatile uint32_t psr;/* Program status register. */
r0 = pulFaultStackAddress[ 0 ];
r1 = pulFaultStackAddress[ 1 ];
r2 = pulFaultStackAddress[ 2 ];
r3 = pulFaultStackAddress[ 3 ];
r12 = pulFaultStackAddress[ 4 ];
lr = pulFaultStackAddress[ 5 ];
pc = pulFaultStackAddress[ 6 ];
psr = pulFaultStackAddress[ 7 ];
/* When the following line is hit, the variables contain the register values. */
for( ;; );
}
/* The prototype shows it is a naked function - in effect this is just an
assembly function. */
void HardFault_Handler( void ) __attribute__( ( naked ) );
/* The fault handler implementation calls a function called
prvGetRegistersFromStack(). */
void HardFault_Handler(void)
{
__asm volatile
(
" tst lr, #4 \n"
" ite eq \n"
" mrseq r0, msp \n"
" mrsne r0, psp \n"
" ldr r1, [r0, #24] \n"
" ldr r2, handler2_address_const \n"
" bx r2 \n"
" handler2_address_const: .word prvGetRegistersFromStack \n"
);
}
#endif