2010-09-22 15:10:42 +02:00
|
|
|
/******************************************************************************
|
2013-06-18 17:21:38 +02:00
|
|
|
Copyright (C) 2013, Freie Universitaet Berlin (FUB). All rights reserved.
|
2010-09-22 15:10:42 +02:00
|
|
|
|
|
|
|
These sources were developed at the Freie Universitaet Berlin, Computer Systems
|
|
|
|
and Telematics group (http://cst.mi.fu-berlin.de).
|
|
|
|
-------------------------------------------------------------------------------
|
2013-03-07 20:51:26 +01:00
|
|
|
This file is part of RIOT.
|
2010-09-22 15:10:42 +02:00
|
|
|
|
2013-11-23 13:43:47 +01:00
|
|
|
This file is subject to the terms and conditions of the LGPLv2.
|
|
|
|
See the file LICENSE in the top level directory for more details.
|
2010-09-22 15:10:42 +02:00
|
|
|
*******************************************************************************/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup lpc2387
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "cpu.h"
|
|
|
|
#include <stdint.h>
|
|
|
|
#include "VIC.h"
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
void lpc2387_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;
|
|
|
|
|
|
|
|
lpc2387_pclk_scale(source, target, &pclksel, prescale);
|
|
|
|
|
2013-06-21 03:52:57 +02: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
|
|
|
}
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
** Function name: install_irq
|
|
|
|
**
|
|
|
|
** 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
|
|
|
|
**
|
|
|
|
******************************************************************************/
|
|
|
|
#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
|
|
|
{
|
|
|
|
int *vect_addr;
|
|
|
|
int *vect_cntl;
|
|
|
|
|
|
|
|
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 */
|
|
|
|
vect_addr = (int *)(VIC_BASE_ADDR + VECT_ADDR_INDEX + IntNumber * 4);
|
|
|
|
vect_cntl = (int *)(VIC_BASE_ADDR + VECT_CNTL_INDEX + IntNumber * 4);
|
|
|
|
*vect_addr = (int)HandlerAddr; /* set interrupt vector */
|
|
|
|
*vect_cntl = Priority;
|
|
|
|
VICIntEnable = 1 << IntNumber; /* Enable Interrupt */
|
|
|
|
return(true);
|
2010-09-22 15:10:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @} */
|