|
|
|
/* $Id: irq.c,v 1.4 2005/01/04 12:22:28 starvik Exp $
|
|
|
|
*
|
|
|
|
* linux/arch/cris/kernel/irq.c
|
|
|
|
*
|
|
|
|
* Copyright (c) 2000-2002 Axis Communications AB
|
|
|
|
*
|
|
|
|
* Authors: Bjorn Wesen (bjornw@axis.com)
|
|
|
|
*
|
|
|
|
* This file contains the interrupt vectors and some
|
|
|
|
* helper functions
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <asm/irq.h>
|
|
|
|
#include <linux/irq.h>
|
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <linux/init.h>
|
|
|
|
|
|
|
|
#define mask_irq(irq_nr) (*R_VECT_MASK_CLR = 1 << (irq_nr));
|
|
|
|
#define unmask_irq(irq_nr) (*R_VECT_MASK_SET = 1 << (irq_nr));
|
|
|
|
|
|
|
|
/* don't use set_int_vector, it bypasses the linux interrupt handlers. it is
|
|
|
|
* global just so that the kernel gdb can use it.
|
|
|
|
*/
|
|
|
|
|
|
|
|
void
|
|
|
|
set_int_vector(int n, irqvectptr addr)
|
|
|
|
{
|
|
|
|
etrax_irv->v[n + 0x20] = (irqvectptr)addr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* the breakpoint vector is obviously not made just like the normal irq handlers
|
|
|
|
* but needs to contain _code_ to jump to addr.
|
|
|
|
*
|
|
|
|
* the BREAK n instruction jumps to IBR + n * 8
|
|
|
|
*/
|
|
|
|
|
|
|
|
void
|
|
|
|
set_break_vector(int n, irqvectptr addr)
|
|
|
|
{
|
|
|
|
unsigned short *jinstr = (unsigned short *)&etrax_irv->v[n*2];
|
|
|
|
unsigned long *jaddr = (unsigned long *)(jinstr + 1);
|
|
|
|
|
|
|
|
/* if you don't know what this does, do not touch it! */
|
|
|
|
|
|
|
|
*jinstr = 0x0d3f;
|
|
|
|
*jaddr = (unsigned long)addr;
|
|
|
|
|
|
|
|
/* 00000026 <clrlop+1a> 3f0d82000000 jump 0x82 */
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This builds up the IRQ handler stubs using some ugly macros in irq.h
|
|
|
|
*
|
|
|
|
* These macros create the low-level assembly IRQ routines that do all
|
|
|
|
* the operations that are needed. They are also written to be fast - and to
|
|
|
|
* disable interrupts as little as humanly possible.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* IRQ0 and 1 are special traps */
|
|
|
|
void hwbreakpoint(void);
|
|
|
|
void IRQ1_interrupt(void);
|
|
|
|
BUILD_TIMER_IRQ(2, 0x04) /* the timer interrupt is somewhat special */
|
|
|
|
BUILD_IRQ(3, 0x08)
|
|
|
|
BUILD_IRQ(4, 0x10)
|
|
|
|
BUILD_IRQ(5, 0x20)
|
|
|
|
BUILD_IRQ(6, 0x40)
|
|
|
|
BUILD_IRQ(7, 0x80)
|
|
|
|
BUILD_IRQ(8, 0x100)
|
|
|
|
BUILD_IRQ(9, 0x200)
|
|
|
|
BUILD_IRQ(10, 0x400)
|
|
|
|
BUILD_IRQ(11, 0x800)
|
|
|
|
BUILD_IRQ(12, 0x1000)
|
|
|
|
BUILD_IRQ(13, 0x2000)
|
|
|
|
void mmu_bus_fault(void); /* IRQ 14 is the bus fault interrupt */
|
|
|
|
void multiple_interrupt(void); /* IRQ 15 is the multiple IRQ interrupt */
|
|
|
|
BUILD_IRQ(16, 0x10000)
|
|
|
|
BUILD_IRQ(17, 0x20000)
|
|
|
|
BUILD_IRQ(18, 0x40000)
|
|
|
|
BUILD_IRQ(19, 0x80000)
|
|
|
|
BUILD_IRQ(20, 0x100000)
|
|
|
|
BUILD_IRQ(21, 0x200000)
|
|
|
|
BUILD_IRQ(22, 0x400000)
|
|
|
|
BUILD_IRQ(23, 0x800000)
|
|
|
|
BUILD_IRQ(24, 0x1000000)
|
|
|
|
BUILD_IRQ(25, 0x2000000)
|
|
|
|
/* IRQ 26-30 are reserved */
|
|
|
|
BUILD_IRQ(31, 0x80000000)
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Pointers to the low-level handlers
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void (*interrupt[NR_IRQS])(void) = {
|
|
|
|
NULL, NULL, IRQ2_interrupt, IRQ3_interrupt,
|
|
|
|
IRQ4_interrupt, IRQ5_interrupt, IRQ6_interrupt, IRQ7_interrupt,
|
|
|
|
IRQ8_interrupt, IRQ9_interrupt, IRQ10_interrupt, IRQ11_interrupt,
|
|
|
|
IRQ12_interrupt, IRQ13_interrupt, NULL, NULL,
|
|
|
|
IRQ16_interrupt, IRQ17_interrupt, IRQ18_interrupt, IRQ19_interrupt,
|
|
|
|
IRQ20_interrupt, IRQ21_interrupt, IRQ22_interrupt, IRQ23_interrupt,
|
|
|
|
IRQ24_interrupt, IRQ25_interrupt, NULL, NULL, NULL, NULL, NULL,
|
|
|
|
IRQ31_interrupt
|
|
|
|
};
|
|
|
|
|
|
|
|
static void enable_crisv10_irq(unsigned int irq);
|
|
|
|
|
|
|
|
static unsigned int startup_crisv10_irq(unsigned int irq)
|
|
|
|
{
|
|
|
|
enable_crisv10_irq(irq);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define shutdown_crisv10_irq disable_crisv10_irq
|
|
|
|
|
|
|
|
static void enable_crisv10_irq(unsigned int irq)
|
|
|
|
{
|
|
|
|
unmask_irq(irq);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void disable_crisv10_irq(unsigned int irq)
|
|
|
|
{
|
|
|
|
mask_irq(irq);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ack_crisv10_irq(unsigned int irq)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static void end_crisv10_irq(unsigned int irq)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct hw_interrupt_type crisv10_irq_type = {
|
|
|
|
.typename = "CRISv10",
|
|
|
|
.startup = startup_crisv10_irq,
|
|
|
|
.shutdown = shutdown_crisv10_irq,
|
|
|
|
.enable = enable_crisv10_irq,
|
|
|
|
.disable = disable_crisv10_irq,
|
|
|
|
.ack = ack_crisv10_irq,
|
|
|
|
.end = end_crisv10_irq,
|
|
|
|
.set_affinity = NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
void weird_irq(void);
|
|
|
|
void system_call(void); /* from entry.S */
|
|
|
|
void do_sigtrap(void); /* from entry.S */
|
|
|
|
void gdb_handle_breakpoint(void); /* from entry.S */
|
|
|
|
|
|
|
|
/* init_IRQ() is called by start_kernel and is responsible for fixing IRQ masks and
|
|
|
|
setting the irq vector table.
|
|
|
|
*/
|
|
|
|
|
|
|
|
void __init
|
|
|
|
init_IRQ(void)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* clear all interrupt masks */
|
|
|
|
|
|
|
|
#ifndef CONFIG_SVINTO_SIM
|
|
|
|
*R_IRQ_MASK0_CLR = 0xffffffff;
|
|
|
|
*R_IRQ_MASK1_CLR = 0xffffffff;
|
|
|
|
*R_IRQ_MASK2_CLR = 0xffffffff;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
*R_VECT_MASK_CLR = 0xffffffff;
|
|
|
|
|
|
|
|
for (i = 0; i < 256; i++)
|
|
|
|
etrax_irv->v[i] = weird_irq;
|
|
|
|
|
|
|
|
/* Initialize IRQ handler descriptiors. */
|
|
|
|
for(i = 2; i < NR_IRQS; i++) {
|
[PATCH] genirq: rename desc->handler to desc->chip
This patch-queue improves the generic IRQ layer to be truly generic, by adding
various abstractions and features to it, without impacting existing
functionality.
While the queue can be best described as "fix and improve everything in the
generic IRQ layer that we could think of", and thus it consists of many
smaller features and lots of cleanups, the one feature that stands out most is
the new 'irq chip' abstraction.
The irq-chip abstraction is about describing and coding and IRQ controller
driver by mapping its raw hardware capabilities [and quirks, if needed] in a
straightforward way, without having to think about "IRQ flow"
(level/edge/etc.) type of details.
This stands in contrast with the current 'irq-type' model of genirq
architectures, which 'mixes' raw hardware capabilities with 'flow' details.
The patchset supports both types of irq controller designs at once, and
converts i386 and x86_64 to the new irq-chip design.
As a bonus side-effect of the irq-chip approach, chained interrupt controllers
(master/slave PIC constructs, etc.) are now supported by design as well.
The end result of this patchset intends to be simpler architecture-level code
and more consolidation between architectures.
We reused many bits of code and many concepts from Russell King's ARM IRQ
layer, the merging of which was one of the motivations for this patchset.
This patch:
rename desc->handler to desc->chip.
Originally i did not want to do this, because it's a big patch. But having
both "desc->handler", "desc->handle_irq" and "action->handler" caused a
large degree of confusion and made the code appear alot less clean than it
truly is.
I have also attempted a dual approach as well by introducing a
desc->chip alias - but that just wasnt robust enough and broke
frequently.
So lets get over with this quickly. The conversion was done automatically
via scripts and converts all the code in the kernel.
This renaming patch is the first one amongst the patches, so that the
remaining patches can stay flexible and can be merged and split up
without having some big monolithic patch act as a merge barrier.
[akpm@osdl.org: build fix]
[akpm@osdl.org: another build fix]
Signed-off-by: Ingo Molnar <mingo@elte.hu>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
19 years ago
|
|
|
irq_desc[i].chip = &crisv10_irq_type;
|
|
|
|
set_int_vector(i, interrupt[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* the entries in the break vector contain actual code to be
|
|
|
|
executed by the associated break handler, rather than just a jump
|
|
|
|
address. therefore we need to setup a default breakpoint handler
|
|
|
|
for all breakpoints */
|
|
|
|
|
|
|
|
for (i = 0; i < 16; i++)
|
|
|
|
set_break_vector(i, do_sigtrap);
|
|
|
|
|
|
|
|
/* except IRQ 15 which is the multiple-IRQ handler on Etrax100 */
|
|
|
|
|
|
|
|
set_int_vector(15, multiple_interrupt);
|
|
|
|
|
|
|
|
/* 0 and 1 which are special breakpoint/NMI traps */
|
|
|
|
|
|
|
|
set_int_vector(0, hwbreakpoint);
|
|
|
|
set_int_vector(1, IRQ1_interrupt);
|
|
|
|
|
|
|
|
/* and irq 14 which is the mmu bus fault handler */
|
|
|
|
|
|
|
|
set_int_vector(14, mmu_bus_fault);
|
|
|
|
|
|
|
|
/* setup the system-call trap, which is reached by BREAK 13 */
|
|
|
|
|
|
|
|
set_break_vector(13, system_call);
|
|
|
|
|
|
|
|
/* setup a breakpoint handler for debugging used for both user and
|
|
|
|
kernel mode debugging (which is why it is not inside an ifdef
|
|
|
|
CONFIG_ETRAX_KGDB) */
|
|
|
|
set_break_vector(8, gdb_handle_breakpoint);
|
|
|
|
|
|
|
|
#ifdef CONFIG_ETRAX_KGDB
|
|
|
|
/* setup kgdb if its enabled, and break into the debugger */
|
|
|
|
kgdb_init();
|
|
|
|
breakpoint();
|
|
|
|
#endif
|
|
|
|
}
|