mirror of
https://github.com/bluekitchen/btstack.git
synced 2025-01-26 03:35:20 +00:00
336 lines
12 KiB
C
336 lines
12 KiB
C
/*******************************************************************************
|
|
* Copyright (C) 2016 Maxim Integrated Products, Inc., All Rights Reserved.
|
|
*
|
|
* 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 MAXIM INTEGRATED 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.
|
|
*
|
|
* Except as contained in this notice, the name of Maxim Integrated
|
|
* Products, Inc. shall not be used except as stated in the Maxim Integrated
|
|
* Products, Inc. Branding Policy.
|
|
*
|
|
* The mere transfer of this software does not imply any licenses
|
|
* of trade secrets, proprietary technology, copyrights, patents,
|
|
* trademarks, maskwork rights, or any other form of intellectual
|
|
* property whatsoever. Maxim Integrated Products, Inc. retains all
|
|
* ownership rights.
|
|
*
|
|
* $Date: 2016-03-17 14:27:29 -0700 (Thu, 17 Mar 2016) $
|
|
* $Revision: 21966 $
|
|
*
|
|
******************************************************************************/
|
|
|
|
#include <stdio.h>
|
|
#include "mxc_config.h"
|
|
#include "mxc_assert.h"
|
|
#include "board.h"
|
|
#include "gpio.h"
|
|
#include "uart.h"
|
|
#include "spim.h"
|
|
#include "max14690n.h"
|
|
|
|
#define UART_ERRORS (MXC_F_UART_INTEN_RX_FIFO_OVERFLOW | \
|
|
MXC_F_UART_INTEN_RX_FRAMING_ERR | \
|
|
MXC_F_UART_INTEN_RX_PARITY_ERR)
|
|
|
|
/***** Global Variables *****/
|
|
|
|
// LEDs
|
|
// Note: EvKit board uses 3.3v supply so these must be open-drain.
|
|
const gpio_cfg_t led_pin[] = {
|
|
{ PORT_2, PIN_4, GPIO_FUNC_GPIO, GPIO_PAD_OPEN_DRAIN },
|
|
{ PORT_2, PIN_5, GPIO_FUNC_GPIO, GPIO_PAD_OPEN_DRAIN },
|
|
{ PORT_2, PIN_6, GPIO_FUNC_GPIO, GPIO_PAD_OPEN_DRAIN },
|
|
};
|
|
const unsigned int num_leds = (sizeof(led_pin) / sizeof(gpio_cfg_t));
|
|
|
|
// Pushbuttons
|
|
const gpio_cfg_t pb_pin[] = {
|
|
{ PORT_2, PIN_3, GPIO_FUNC_GPIO, GPIO_PAD_INPUT_PULLUP },
|
|
};
|
|
const unsigned int num_pbs = (sizeof(pb_pin) / sizeof(gpio_cfg_t));
|
|
|
|
// Console UART configuration
|
|
const uart_cfg_t console_uart_cfg = {
|
|
.parity = UART_PARITY_DISABLE,
|
|
.size = UART_DATA_SIZE_8_BITS,
|
|
.extra_stop = 0,
|
|
.cts = 0,
|
|
.rts = 0,
|
|
.baud = CONSOLE_BAUD,
|
|
};
|
|
const sys_cfg_uart_t console_sys_cfg = {
|
|
.clk_scale = CLKMAN_SCALE_AUTO,
|
|
.io_cfg = IOMAN_UART(CONSOLE_UART, IOMAN_MAP_A, IOMAN_MAP_UNUSED, IOMAN_MAP_UNUSED, 1, 0, 0)
|
|
};
|
|
|
|
// MAX14690 PMIC
|
|
const ioman_cfg_t max14690_io_cfg = IOMAN_I2CM2(IOMAN_MAP_A, 1);
|
|
const gpio_cfg_t max14690_int = { PORT_3, PIN_7, GPIO_FUNC_GPIO, GPIO_PAD_INPUT_PULLUP };
|
|
const gpio_cfg_t max14690_mpc0 = { PORT_2, PIN_7, GPIO_FUNC_GPIO, GPIO_PAD_NORMAL };
|
|
|
|
/***** File Scope Variables *****/
|
|
|
|
/******************************************************************************/
|
|
void mxc_assert(const char *expr, const char *file, int line)
|
|
{
|
|
printf("MXC_ASSERT %s #%d: (%s)\n", file, line, expr);
|
|
while (1);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int Console_Init(void)
|
|
{
|
|
int err;
|
|
|
|
if ((err = UART_Init(MXC_UART_GET_UART(CONSOLE_UART), &console_uart_cfg, &console_sys_cfg)) != E_NO_ERROR) {
|
|
MXC_ASSERT_FAIL();
|
|
return err;
|
|
}
|
|
|
|
// Setup the interrupt
|
|
NVIC_ClearPendingIRQ(MXC_UART_GET_IRQ(CONSOLE_UART));
|
|
NVIC_DisableIRQ(MXC_UART_GET_IRQ(CONSOLE_UART));
|
|
NVIC_SetPriority(MXC_UART_GET_IRQ(CONSOLE_UART), 1);
|
|
NVIC_EnableIRQ(MXC_UART_GET_IRQ(CONSOLE_UART));
|
|
MXC_UART1->inten |= (MXC_F_UART_INTEN_RX_FIFO_NOT_EMPTY | UART_ERRORS);
|
|
|
|
return E_NO_ERROR;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int Console_PrepForSleep(void)
|
|
{
|
|
fflush(stdout);
|
|
return UART_PrepForSleep(MXC_UART_GET_UART(CONSOLE_UART));
|
|
}
|
|
|
|
void UART0_IRQHandler(void)
|
|
{
|
|
UART_Handler(MXC_UART_GET_UART(0));
|
|
}
|
|
|
|
void UART1_IRQHandler(void)
|
|
{
|
|
UART_Handler(MXC_UART_GET_UART(1));
|
|
}
|
|
|
|
void UART2_IRQHandler(void)
|
|
{
|
|
UART_Handler(MXC_UART_GET_UART(2));
|
|
}
|
|
|
|
void UART3_IRQHandler(void)
|
|
{
|
|
UART_Handler(MXC_UART_GET_UART(3));
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int Board_Init(void)
|
|
{
|
|
int err;
|
|
|
|
if ((err = Console_Init()) != E_NO_ERROR) {
|
|
MXC_ASSERT_FAIL();
|
|
return err;
|
|
}
|
|
|
|
if ((err = LED_Init()) != E_NO_ERROR) {
|
|
MXC_ASSERT_FAIL();
|
|
return err;
|
|
}
|
|
|
|
if ((err = PB_Init()) != E_NO_ERROR) {
|
|
MXC_ASSERT_FAIL();
|
|
return err;
|
|
}
|
|
|
|
/* On the Pegasus board MPC1 is connected to CAP which is high when VBUS is present.
|
|
* The LDO_OUTPUT_MPC1 setting will automatically enable the output when VBUS is present.
|
|
* The LDO_OUTPUT_MPC1 setting will also disable the output when powered from the battery.
|
|
* The Pegasus board uses LDO2 for VDDB (USB), LEDs and the SD card connector.
|
|
* Use the MAX14690_LDO2setMode(mode) function to enable LDO2 when needed.
|
|
*/
|
|
if ((err = MAX14690N_Init(3.3, LDO_OUTPUT_MPC1, 3.3, LDO_OUTPUT_DISABLED)) != E_NO_ERROR) {
|
|
MXC_ASSERT_FAIL();
|
|
return err;
|
|
}
|
|
|
|
return E_NO_ERROR;
|
|
}
|
|
|
|
#ifdef CRASH_DUMP
|
|
void FaultISR_C(uint32_t *hardfault_args)
|
|
{
|
|
unsigned int stacked_r0;
|
|
unsigned int stacked_r1;
|
|
unsigned int stacked_r2;
|
|
unsigned int stacked_r3;
|
|
unsigned int stacked_r12;
|
|
unsigned int stacked_lr;
|
|
unsigned int stacked_pc;
|
|
unsigned int stacked_psr;
|
|
|
|
volatile unsigned char mmsr;
|
|
volatile unsigned char bfsr;
|
|
volatile unsigned short ufsr;
|
|
volatile unsigned int hfsr;
|
|
|
|
stacked_r0 = ((unsigned int) hardfault_args[0]);
|
|
stacked_r1 = ((unsigned int) hardfault_args[1]);
|
|
stacked_r2 = ((unsigned int) hardfault_args[2]);
|
|
stacked_r3 = ((unsigned int) hardfault_args[3]);
|
|
stacked_r12 = ((unsigned int) hardfault_args[4]);
|
|
stacked_lr = ((unsigned int) hardfault_args[5]);
|
|
stacked_pc = ((unsigned int) hardfault_args[6]);
|
|
stacked_psr = ((unsigned int) hardfault_args[7]);
|
|
|
|
printf("\n\n[Hard fault handler - all numbers in hex]\n");
|
|
printf("R0 = 0x%08x\n", stacked_r0);
|
|
printf("R1 = 0x%08x\n", stacked_r1);
|
|
printf("R2 = 0x%08x\n", stacked_r2);
|
|
printf("R3 = 0x%08x\n", stacked_r3);
|
|
printf("R12 = 0x%08x\n", stacked_r12);
|
|
printf("LR [R14] = 0x%08x subroutine call return address\n", stacked_lr);
|
|
printf("PC [R15] = 0x%08x program counter address\n", stacked_pc);
|
|
printf("PSR = 0x%08x\n", stacked_psr);
|
|
printf("MMAR = 0x%08x memory manage fault address\n", (*((volatile unsigned int *) (0xE000ED34))));
|
|
printf("BFAR = 0x%08x bus fault address\n", (*((volatile unsigned int *) (0xE000ED38))));
|
|
|
|
/***********************************************************************************************
|
|
* Memory Management Fault Status Register: (0xE000ED28)
|
|
* Bit Name Description
|
|
* 7 MMARVALID MMAR is valid (0x40)
|
|
* 4 MSTKERR Stacking error (0x10)
|
|
* 3 MUNSTKERR Unstacking error (0x8)
|
|
* 1 DACCVIOL Data access violation (0x2)
|
|
* 0 IACCVIOL Instruction access violation (0x1)
|
|
***********************************************************************************************/
|
|
mmsr = (*((volatile unsigned char *) (0xE000ED28)));
|
|
printf("MMSR = 0x%02x ", mmsr);
|
|
if (mmsr & 0x40)
|
|
printf("MMARVALID: MMAR is valid ");
|
|
if (mmsr & 0x10)
|
|
printf("MSTKERR: Stacking error\n");
|
|
else if (mmsr & 0x8)
|
|
printf("MUNSTKERR: Unstacking error\n");
|
|
else if (mmsr & 0x2)
|
|
printf("DACCVIOL: Data access violation\n");
|
|
else if (mmsr & 0x1)
|
|
printf("IACCVIOL: Instruction access violation\n");
|
|
else
|
|
printf("\n");
|
|
|
|
/***********************************************************************************************
|
|
* Bus Fault Status Register: (0xE000ED28)
|
|
* Bit Name Description
|
|
* 7 BFARVALID BFAR is valid (0x80)
|
|
* 4 STKERR Stacking error (0x10)
|
|
* 3 UNSTKERR Unstacking error (0x8)
|
|
* 2 IMPREISERR Imprecise data access violation (0x4)
|
|
* 1 PRECISERR Precise data access violation (0x2)
|
|
* 0 IBUSERR Instruction access violation (0x1)
|
|
***********************************************************************************************/
|
|
bfsr = (*((volatile unsigned char *) (0xE000ED29)));
|
|
printf("BFSR = 0x%02x ", bfsr);
|
|
if (bfsr & 0x80)
|
|
printf("BFARVALID: BFAR is valid ");
|
|
if (bfsr & 0x10)
|
|
printf("STKERR: Stacking error\n");
|
|
else if (bfsr & 0x8)
|
|
printf("UNSTKERR: Unstacking error\n");
|
|
else if (bfsr & 0x4)
|
|
printf("IMPREISERR: Imprecise data access violation\n");
|
|
else if (bfsr & 0x2)
|
|
printf("PRECISERR: Precise data access violation\n");
|
|
else if (bfsr & 0x1)
|
|
printf("IBUSERR: Instruction access violation\n");
|
|
else
|
|
printf("\n");
|
|
|
|
/***********************************************************************************************
|
|
* Usage Fault Status Register: (0xE000ED2A)
|
|
* Bit Name Description
|
|
* 9 DIVBYZERO Divide by zero will take place (0x200)
|
|
* 8 UNALIGNED Unaligned access will take place (0x100)
|
|
* 3 NOCP Attempt to execute a coprocessor instruction (0x8)
|
|
* 2 INVPC Attempt to do exception with bad value (0x4)
|
|
* 1 INVSTATE Attempt to switch to invalid state (0x2)
|
|
* 0 UNDEFINSTR Attempt to execute an undefined instruction (0x1)
|
|
***********************************************************************************************/
|
|
ufsr = (*((volatile unsigned short *) (0xE000ED2A)));
|
|
printf("UFSR = 0x%04x ", ufsr);
|
|
if (ufsr & 0x200)
|
|
printf("DIVBYZERO: Divide by zero will take place\n");
|
|
else if (ufsr & 0x100)
|
|
printf("UNALIGNED: Unaligned access will take place\n");
|
|
else if (ufsr & 0x8)
|
|
printf("NOCP: Attempt to execute a coprocessor instruction\n");
|
|
else if (ufsr & 0x4)
|
|
printf("INVPC: Attempt to do exception with bad value\n");
|
|
else if (ufsr & 0x2)
|
|
printf("INVSTATE: Attempt to switch to invalid state\n");
|
|
else if (ufsr & 0x1)
|
|
printf("UNDEFINSTR: Attempt to execute an undefined instruction\n");
|
|
else
|
|
printf("\n");
|
|
|
|
/***********************************************************************************************
|
|
* Usage Fault Status Register: (0xE000ED2A)
|
|
* Bit Name Description
|
|
* 31 DEBUGEVT Hard fault caused by debug event (0x8000_0000)
|
|
* 30 FORCED Hard fault caused by bus/memory management/usage fault (0x4000_0000)
|
|
* 1 VECTBL Hard fault caused by failed vector fetch (0x1)
|
|
***********************************************************************************************/
|
|
hfsr = (*((volatile unsigned int *) (0xE000ED2C)));
|
|
printf("HFSR = 0x%08x ", hfsr);
|
|
if (hfsr & 0x80000000)
|
|
printf("DEBUGEVT: Hard fault caused by debug event\n");
|
|
else if (hfsr & 0x40000000)
|
|
printf("FORCED: Hard fault caused by bus/memory management/usage fault\n");
|
|
else if (hfsr & 0x1)
|
|
printf("VECTBL: Hard fault caused by failed vector fetch\n");
|
|
else
|
|
printf("\n");
|
|
|
|
printf ("AFSR = 0x%08x\n", (*((volatile unsigned int *)(0xE000ED3C))));
|
|
printf ("SCB_SHCSR = %x\n", SCB->SHCSR);
|
|
|
|
while (1) ; /* Spin so we can use a debugger to anlayzer the situation */
|
|
}
|
|
#else /* ENABLE_CRASH_DUMP */
|
|
void FaultISR_C(uint32_t *hardfault_args)
|
|
{
|
|
/* spin so we can use a debugger to anlayze the situation */
|
|
while(1);
|
|
/* reset the system */
|
|
//NVIC_SystemReset();
|
|
}
|
|
#endif /* CRASH_DUMP */
|
|
|
|
|
|
void HardFault_Handler(void)
|
|
{
|
|
printf("HardFault_Handler! (main)\n");
|
|
__asm(
|
|
" TST LR, #4\n"
|
|
" ITE EQ \n"
|
|
" MRSEQ R0, MSP \n"
|
|
" MRSNE R0, PSP \n"
|
|
" B FaultISR_C \n");
|
|
}
|