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");
}