2013-11-09 00:43:35 +01:00
|
|
|
/**
|
2014-02-11 18:15:43 +01:00
|
|
|
* CPU specific functions for the RIOT scheduler on NXP LPC1768
|
2013-11-09 00:43:35 +01:00
|
|
|
*
|
|
|
|
* Copyright (C) 2013 Oliver Hahm <oliver.hahm@inria.fr>
|
|
|
|
*
|
|
|
|
* This file subject to the terms and conditions of the GNU Lesser General
|
|
|
|
* Public License. See the file LICENSE in the top level directory for more
|
|
|
|
* details.
|
|
|
|
*
|
|
|
|
* @file cpu.c
|
2014-01-28 11:50:12 +01:00
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
2013-11-09 00:43:35 +01:00
|
|
|
* @author Oliver Hahm <oliver.hahm@inria.fr>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
#include "cpu.h"
|
2014-02-15 17:27:50 +01:00
|
|
|
#include "kernel.h"
|
2013-11-09 00:43:35 +01:00
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
int inISR(void)
|
|
|
|
{
|
2014-02-15 17:28:19 +01:00
|
|
|
return (__get_IPSR() & 0xFF);
|
2013-11-09 00:43:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int disableIRQ(void)
|
|
|
|
{
|
2014-02-15 17:28:19 +01:00
|
|
|
// FIXME PRIMASK is the old CPSR (FAULTMASK ??? BASEPRI ???)
|
|
|
|
//PRIMASK lesen
|
|
|
|
unsigned int uiPriMask = __get_PRIMASK();
|
|
|
|
__disable_irq();
|
|
|
|
return uiPriMask;
|
2013-11-09 00:43:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void restoreIRQ(unsigned oldPRIMASK)
|
|
|
|
{
|
2014-02-15 17:28:19 +01:00
|
|
|
//PRIMASK lesen setzen
|
|
|
|
__set_PRIMASK(oldPRIMASK);
|
2013-11-09 00:43:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
__attribute__((naked))
|
|
|
|
void HardFault_Handler(void) {
|
|
|
|
DEBUG("HARD FAULT\n");
|
|
|
|
while(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
__attribute__((naked))
|
|
|
|
void BusFault_Handler(void) {
|
|
|
|
DEBUG("BusFault_Handler\n");
|
|
|
|
while(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
__attribute__((naked))
|
|
|
|
void Usage_Handler(void) {
|
|
|
|
DEBUG("Usage FAULT\n");
|
|
|
|
while(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
__attribute__((naked))
|
|
|
|
void WWDG_Handler(void) {
|
|
|
|
DEBUG("WWDG FAULT\n");
|
|
|
|
while(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void dINT(void)
|
|
|
|
{
|
|
|
|
__disable_irq();
|
|
|
|
}
|
|
|
|
|
|
|
|
void eINT(void)
|
|
|
|
{
|
|
|
|
__enable_irq();
|
|
|
|
}
|
|
|
|
|
|
|
|
void save_context(void)
|
|
|
|
{
|
2014-02-15 17:28:19 +01:00
|
|
|
/* {r0-r3,r12,LR,PC,xPSR} are saved automatically on exception entry */
|
|
|
|
asm("push {r4-r11}");
|
|
|
|
/* save unsaved registers */
|
|
|
|
asm("push {LR}");
|
|
|
|
/* save exception return value */
|
|
|
|
|
|
|
|
asm("ldr r1, =active_thread");
|
|
|
|
/* load address of currend pdc */
|
|
|
|
asm("ldr r1, [r1]");
|
|
|
|
/* deref pdc */
|
|
|
|
asm("str sp, [r1]");
|
|
|
|
/* write sp to pdc->sp means current threads stack pointer */
|
2013-11-09 00:43:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void restore_context(void)
|
|
|
|
{
|
2014-02-15 17:28:19 +01:00
|
|
|
asm("ldr r0, =active_thread");
|
|
|
|
/* load address of currend pdc */
|
|
|
|
asm("ldr r0, [r0]");
|
|
|
|
/* deref pdc */
|
|
|
|
asm("ldr sp, [r0]");
|
|
|
|
/* load pdc->sp to sp register */
|
|
|
|
|
|
|
|
asm("pop {r0}");
|
|
|
|
/* restore exception retrun value from stack */
|
|
|
|
asm("pop {r4-r11}");
|
|
|
|
/* load unloaded register */
|
|
|
|
// asm("pop {r4}"); /*foo*/
|
|
|
|
asm("bx r0"); /* load exception return value to pc causes end of exception*/
|
|
|
|
/* {r0-r3,r12,LR,PC,xPSR} are restored automatically on exception return */
|
2013-11-09 00:43:35 +01:00
|
|
|
}
|
2014-02-15 17:27:50 +01:00
|
|
|
|
|
|
|
#define USR_RESET (0x102)
|
|
|
|
#define SWI (0xAB)
|
|
|
|
|
|
|
|
__attribute__((naked,noreturn)) void arm_reset(void)
|
|
|
|
{
|
|
|
|
int value;
|
|
|
|
|
|
|
|
asm volatile (
|
|
|
|
"mov r0, %1" "\n\t"
|
|
|
|
"mov r1, %2" "\n\t"
|
|
|
|
"bkpt" " %a3" "\n\t"
|
|
|
|
"mov %0, r0"
|
|
|
|
: "=r" (value) /* output operands */
|
|
|
|
: "r" USR_RESET, "r" NULL, "i" SWI /* input operands */
|
|
|
|
: "r0", "r1", "r2", "r3", "ip", "lr", "memory", "cc" /* list of clobbered registers */
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
NORETURN void reboot(void)
|
|
|
|
{
|
|
|
|
while (1) {
|
|
|
|
arm_reset();
|
|
|
|
}
|
|
|
|
}
|