2014-08-27 18:47:31 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2013, Freie Universitaet Berlin (FUB). All rights reserved.
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*/
|
2010-09-22 15:10:42 +02:00
|
|
|
|
|
|
|
/**
|
2020-07-16 16:25:45 +02:00
|
|
|
* @ingroup cpu_lpc23xx
|
2010-09-22 15:10:42 +02:00
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdint.h>
|
2014-12-01 17:36:00 +01:00
|
|
|
#include "cpu.h"
|
2023-01-02 18:08:35 +01:00
|
|
|
#include "kernel_init.h"
|
2014-12-01 17:36:00 +01:00
|
|
|
#include "irq.h"
|
2010-09-22 15:10:42 +02:00
|
|
|
#include "VIC.h"
|
|
|
|
|
2019-09-14 14:49:48 +02:00
|
|
|
#include "stdio_base.h"
|
|
|
|
#include "periph/init.h"
|
|
|
|
|
2020-07-16 16:24:36 +02:00
|
|
|
void lpc23xx_pclk_scale(uint32_t source, uint32_t target, uint32_t *pclksel, uint32_t *prescale)
|
2010-09-22 15:10:42 +02:00
|
|
|
{
|
2013-06-21 03:52:57 +02:00
|
|
|
uint32_t pclkdiv;
|
|
|
|
*prescale = source / target;
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if ((*prescale % 16) == 0) {
|
2013-06-21 03:52:57 +02:00
|
|
|
*pclksel = 3;
|
|
|
|
pclkdiv = 8;
|
|
|
|
}
|
2013-06-24 22:37:35 +02:00
|
|
|
else if ((*prescale % 8) == 0) {
|
2013-06-21 03:52:57 +02:00
|
|
|
*pclksel = 0;
|
|
|
|
pclkdiv = 4;
|
|
|
|
}
|
2013-06-24 22:37:35 +02:00
|
|
|
else if ((*prescale % 4) == 0) {
|
2013-06-21 03:52:57 +02:00
|
|
|
*pclksel = 2;
|
|
|
|
pclkdiv = 2;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*pclksel = 1;
|
|
|
|
pclkdiv = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
*prescale /= pclkdiv;
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (*prescale % 2) {
|
2013-06-21 03:52:57 +02:00
|
|
|
(*prescale)++;
|
|
|
|
}
|
2010-09-22 15:10:42 +02:00
|
|
|
}
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
void cpu_clock_scale(uint32_t source, uint32_t target, uint32_t *prescale)
|
|
|
|
{
|
2010-09-22 15:10:42 +02:00
|
|
|
uint32_t pclksel;
|
|
|
|
|
2020-07-16 16:24:36 +02:00
|
|
|
lpc23xx_pclk_scale(source, target, &pclksel, prescale);
|
2010-09-22 15:10:42 +02:00
|
|
|
|
2018-02-05 11:39:41 +01:00
|
|
|
PCLKSEL0 = (PCLKSEL0 & ~(BIT2 | BIT3)) | (pclksel << 2); /* timer 0 */
|
|
|
|
PCLKSEL0 = (PCLKSEL0 & ~(BIT4 | BIT5)) | (pclksel << 4); /* timer 1 */
|
|
|
|
PCLKSEL1 = (PCLKSEL1 & ~(BIT12 | BIT13)) | (pclksel << 12); /* timer 2 */
|
2010-09-22 15:10:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/******************************************************************************
|
2014-07-31 20:46:28 +02:00
|
|
|
** Function name: install_irq
|
2010-09-22 15:10:42 +02:00
|
|
|
**
|
2014-07-31 20:46:28 +02:00
|
|
|
** Descriptions: Install interrupt handler
|
|
|
|
** parameters: Interrupt number, interrupt handler address,
|
|
|
|
** interrupt priority
|
|
|
|
** Returned value: true or false, return false if IntNum is out of range
|
2010-09-22 15:10:42 +02:00
|
|
|
**
|
|
|
|
******************************************************************************/
|
|
|
|
#define VIC_BASE_ADDR 0xFFFFF000
|
|
|
|
|
2013-11-21 14:18:54 +01:00
|
|
|
bool install_irq(int IntNumber, void (*HandlerAddr)(void), int Priority)
|
2010-09-22 15:10:42 +02:00
|
|
|
{
|
2014-07-31 20:46:28 +02:00
|
|
|
VICIntEnClr = 1 << IntNumber; /* Disable Interrupt */
|
2013-06-21 03:52:57 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (IntNumber >= VIC_SIZE) {
|
2013-06-21 03:52:57 +02:00
|
|
|
return (false);
|
2010-09-22 15:10:42 +02:00
|
|
|
}
|
2013-06-21 03:52:57 +02:00
|
|
|
else {
|
|
|
|
/* find first un-assigned VIC address for the handler */
|
2014-08-11 19:51:23 +02:00
|
|
|
int *vect_addr = (int *)(VIC_BASE_ADDR + VECT_ADDR_INDEX + IntNumber * 4);
|
|
|
|
int *vect_cntl = (int *)(VIC_BASE_ADDR + VECT_CNTL_INDEX + IntNumber * 4);
|
2014-07-31 20:46:28 +02:00
|
|
|
*vect_addr = (int)HandlerAddr; /* set interrupt vector */
|
2013-06-21 03:52:57 +02:00
|
|
|
*vect_cntl = Priority;
|
2014-07-31 20:46:28 +02:00
|
|
|
VICIntEnable = 1 << IntNumber; /* Enable Interrupt */
|
2013-06-21 03:52:57 +02:00
|
|
|
return(true);
|
2010-09-22 15:10:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-14 10:40:09 +02:00
|
|
|
void arm_reset(void)
|
2014-12-01 17:36:00 +01:00
|
|
|
{
|
2018-08-14 10:40:09 +02:00
|
|
|
/*
|
|
|
|
* We abuse the watchdog timer for a reset.
|
|
|
|
*/
|
2016-03-19 09:25:47 +01:00
|
|
|
irq_disable();
|
2018-08-14 10:40:09 +02:00
|
|
|
/* Set the watchdog timeout constant to 0xFFFF */
|
|
|
|
WDTC = 0xFFFF;
|
|
|
|
/*
|
|
|
|
* Enable watchdog interrupt and enable reset on watchdog timeout.
|
|
|
|
* (The reset on watchdog timeout flag is ignored, if interrupt on watchdog
|
|
|
|
* timeout is not set. Thus, we set both. The reset takes precedence over
|
|
|
|
* the interrupt, anyway.)
|
|
|
|
*/
|
|
|
|
WDMOD = 0x03;
|
|
|
|
/*
|
|
|
|
* Feed the watchdog by writing 0xAA followed by 0x55:
|
|
|
|
* Reload the watchdog timer with the value in WDTC (0xFFFF)
|
|
|
|
*/
|
|
|
|
WDFEED = 0xAA;
|
|
|
|
WDFEED = 0x55;
|
|
|
|
/* Wait for the watchdog timer to expire, thus performing a reset */
|
2016-01-27 09:57:15 +01:00
|
|
|
while(1) {}
|
2014-12-01 17:36:00 +01:00
|
|
|
}
|
|
|
|
|
2019-09-14 14:49:48 +02:00
|
|
|
/**
|
|
|
|
* @brief Initialize the CPU, set IRQ priorities, clocks
|
|
|
|
*/
|
|
|
|
void cpu_init(void)
|
|
|
|
{
|
|
|
|
extern void board_init(void);
|
|
|
|
|
|
|
|
/* configure CPU clock */
|
|
|
|
cpu_init_clks();
|
|
|
|
|
|
|
|
/* set up GPIOs */
|
|
|
|
gpio_init_ports();
|
|
|
|
|
|
|
|
/* board specific setup of i/o pins */
|
|
|
|
board_init();
|
|
|
|
|
|
|
|
/* initialize stdio prior to periph_init() to allow use of DEBUG() there */
|
2023-01-02 18:08:35 +01:00
|
|
|
early_init();
|
2019-09-14 14:49:48 +02:00
|
|
|
|
|
|
|
/* trigger static peripheral initialization */
|
|
|
|
periph_init();
|
|
|
|
}
|
|
|
|
|
2019-11-09 23:56:29 +01:00
|
|
|
/* RSIR will only have POR bit set even when waking up from Deep Power Down
|
|
|
|
* Use signature in battery RAM to discriminate between Deep Power Down and POR
|
|
|
|
*/
|
2019-12-09 15:01:57 +01:00
|
|
|
bool cpu_backup_ram_is_initialized(void)
|
2019-11-09 23:56:29 +01:00
|
|
|
{
|
|
|
|
static char signature[] __attribute__((section(".backup.data"))) = {
|
|
|
|
'R', 'I', 'O', 'T'
|
|
|
|
};
|
|
|
|
|
2019-12-09 15:03:24 +01:00
|
|
|
/* Only in case when a reset occurs and the POR = 0, the BODR bit
|
|
|
|
* indicates if the V_DD (3V3) voltage was below 2.6 V or not.
|
|
|
|
*/
|
|
|
|
if ((RSIR & (RSIR_BODR | RSIR_POR)) == (RSIR_BODR | RSIR_POR)) {
|
|
|
|
RSIR |= RSIR_BODR;
|
|
|
|
}
|
|
|
|
|
2019-11-09 23:56:29 +01:00
|
|
|
/* external reset */
|
|
|
|
if (RSIR & RSIR_EXTR) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (signature[0] != 'R') {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (signature[1] != 'I') {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (signature[2] != 'O') {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (signature[3] != 'T') {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-12-05 22:09:39 +01:00
|
|
|
/* When we wake from Deep Sleep only POR is set, just like in the real
|
|
|
|
* POR case. Clear the bit to create a new, distinct state.
|
|
|
|
*/
|
|
|
|
RSIR |= RSIR_POR;
|
|
|
|
|
2019-11-09 23:56:29 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-09-22 15:10:42 +02:00
|
|
|
/** @} */
|