2f2c267989
Reduce the function pointer mess of the m68knommu timer code by calling directly to the local hardware's timer setup, and expose the local common timer interrupt handler to the lower level hardware timer. Ultimately this will save definitions of all these functions across all the platform code to setup the function pointers (which for any given m68knommu CPU family member can be only one set of hardware timer functions). Signed-off-by: Greg Ungerer <gerg@uclinux.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
107 lines
2.7 KiB
C
107 lines
2.7 KiB
C
/***************************************************************************/
|
|
|
|
/*
|
|
* linux/arch/m68knommu/platform/5407/config.c
|
|
*
|
|
* Copyright (C) 1999-2002, Greg Ungerer (gerg@snapgear.com)
|
|
* Copyright (C) 2000, Lineo (www.lineo.com)
|
|
*/
|
|
|
|
/***************************************************************************/
|
|
|
|
#include <linux/kernel.h>
|
|
#include <linux/param.h>
|
|
#include <linux/init.h>
|
|
#include <linux/interrupt.h>
|
|
#include <asm/dma.h>
|
|
#include <asm/machdep.h>
|
|
#include <asm/coldfire.h>
|
|
#include <asm/mcfsim.h>
|
|
#include <asm/mcfdma.h>
|
|
|
|
/***************************************************************************/
|
|
|
|
void coldfire_reset(void);
|
|
|
|
extern unsigned int mcf_timervector;
|
|
extern unsigned int mcf_profilevector;
|
|
extern unsigned int mcf_timerlevel;
|
|
|
|
/***************************************************************************/
|
|
|
|
/*
|
|
* DMA channel base address table.
|
|
*/
|
|
unsigned int dma_base_addr[MAX_M68K_DMA_CHANNELS] = {
|
|
MCF_MBAR + MCFDMA_BASE0,
|
|
MCF_MBAR + MCFDMA_BASE1,
|
|
MCF_MBAR + MCFDMA_BASE2,
|
|
MCF_MBAR + MCFDMA_BASE3,
|
|
};
|
|
|
|
unsigned int dma_device_address[MAX_M68K_DMA_CHANNELS];
|
|
|
|
/***************************************************************************/
|
|
|
|
void mcf_autovector(unsigned int vec)
|
|
{
|
|
volatile unsigned char *mbar;
|
|
|
|
if ((vec >= 25) && (vec <= 31)) {
|
|
mbar = (volatile unsigned char *) MCF_MBAR;
|
|
vec = 0x1 << (vec - 24);
|
|
*(mbar + MCFSIM_AVR) |= vec;
|
|
mcf_setimr(mcf_getimr() & ~vec);
|
|
}
|
|
}
|
|
|
|
/***************************************************************************/
|
|
|
|
void mcf_settimericr(unsigned int timer, unsigned int level)
|
|
{
|
|
volatile unsigned char *icrp;
|
|
unsigned int icr, imr;
|
|
|
|
if (timer <= 2) {
|
|
switch (timer) {
|
|
case 2: icr = MCFSIM_TIMER2ICR; imr = MCFSIM_IMR_TIMER2; break;
|
|
default: icr = MCFSIM_TIMER1ICR; imr = MCFSIM_IMR_TIMER1; break;
|
|
}
|
|
|
|
icrp = (volatile unsigned char *) (MCF_MBAR + icr);
|
|
*icrp = MCFSIM_ICR_AUTOVEC | (level << 2) | MCFSIM_ICR_PRI3;
|
|
mcf_setimr(mcf_getimr() & ~imr);
|
|
}
|
|
}
|
|
|
|
/***************************************************************************/
|
|
|
|
int mcf_timerirqpending(int timer)
|
|
{
|
|
unsigned int imr = 0;
|
|
|
|
switch (timer) {
|
|
case 1: imr = MCFSIM_IMR_TIMER1; break;
|
|
case 2: imr = MCFSIM_IMR_TIMER2; break;
|
|
default: break;
|
|
}
|
|
return (mcf_getipr() & imr);
|
|
}
|
|
|
|
/***************************************************************************/
|
|
|
|
void config_BSP(char *commandp, int size)
|
|
{
|
|
mcf_setimr(MCFSIM_IMR_MASKALL);
|
|
|
|
#if defined(CONFIG_CLEOPATRA)
|
|
/* Different timer setup - to prevent device clash */
|
|
mcf_timervector = 30;
|
|
mcf_profilevector = 31;
|
|
mcf_timerlevel = 6;
|
|
#endif
|
|
|
|
mach_reset = coldfire_reset;
|
|
}
|
|
|
|
/***************************************************************************/
|