mv64x60_pic.c

来自「linux 内核源代码」· C语言 代码 · 共 298 行

C
298
字号
/* * Interrupt handling for Marvell mv64360/mv64460 host bridges (Discovery) * * Author: Dale Farnsworth <dale@farnsworth.org> * * 2007 (c) MontaVista, Software, Inc.  This file is licensed under * the terms of the GNU General Public License version 2.  This program * is licensed "as is" without any warranty of any kind, whether express * or implied. */#include <linux/stddef.h>#include <linux/kernel.h>#include <linux/init.h>#include <linux/irq.h>#include <linux/interrupt.h>#include <linux/spinlock.h>#include <asm/byteorder.h>#include <asm/io.h>#include <asm/prom.h>#include <asm/irq.h>#include "mv64x60.h"/* Interrupt Controller Interface Registers */#define MV64X60_IC_MAIN_CAUSE_LO	0x0004#define MV64X60_IC_MAIN_CAUSE_HI	0x000c#define MV64X60_IC_CPU0_INTR_MASK_LO	0x0014#define MV64X60_IC_CPU0_INTR_MASK_HI	0x001c#define MV64X60_IC_CPU0_SELECT_CAUSE	0x0024#define MV64X60_HIGH_GPP_GROUPS		0x0f000000#define MV64X60_SELECT_CAUSE_HIGH	0x40000000/* General Purpose Pins Controller Interface Registers */#define MV64x60_GPP_INTR_CAUSE		0x0008#define MV64x60_GPP_INTR_MASK		0x000c#define MV64x60_LEVEL1_LOW		0#define MV64x60_LEVEL1_HIGH		1#define MV64x60_LEVEL1_GPP		2#define MV64x60_LEVEL1_MASK		0x00000060#define MV64x60_LEVEL1_OFFSET		5#define MV64x60_LEVEL2_MASK		0x0000001f#define MV64x60_NUM_IRQS		96static DEFINE_SPINLOCK(mv64x60_lock);static void __iomem *mv64x60_irq_reg_base;static void __iomem *mv64x60_gpp_reg_base;/* * Interrupt Controller Handling * * The interrupt controller handles three groups of interrupts: *   main low:	IRQ0-IRQ31 *   main high:	IRQ32-IRQ63 *   gpp:	IRQ64-IRQ95 * * This code handles interrupts in two levels.  Level 1 selects the * interrupt group, and level 2 selects an IRQ within that group. * Each group has its own irq_chip structure. */static u32 mv64x60_cached_low_mask;static u32 mv64x60_cached_high_mask = MV64X60_HIGH_GPP_GROUPS;static u32 mv64x60_cached_gpp_mask;static struct irq_host *mv64x60_irq_host;/* * mv64x60_chip_low functions */static void mv64x60_mask_low(unsigned int virq){	int level2 = irq_map[virq].hwirq & MV64x60_LEVEL2_MASK;	unsigned long flags;	spin_lock_irqsave(&mv64x60_lock, flags);	mv64x60_cached_low_mask &= ~(1 << level2);	out_le32(mv64x60_irq_reg_base + MV64X60_IC_CPU0_INTR_MASK_LO,		 mv64x60_cached_low_mask);	spin_unlock_irqrestore(&mv64x60_lock, flags);	(void)in_le32(mv64x60_irq_reg_base + MV64X60_IC_CPU0_INTR_MASK_LO);}static void mv64x60_unmask_low(unsigned int virq){	int level2 = irq_map[virq].hwirq & MV64x60_LEVEL2_MASK;	unsigned long flags;	spin_lock_irqsave(&mv64x60_lock, flags);	mv64x60_cached_low_mask |= 1 << level2;	out_le32(mv64x60_irq_reg_base + MV64X60_IC_CPU0_INTR_MASK_LO,		 mv64x60_cached_low_mask);	spin_unlock_irqrestore(&mv64x60_lock, flags);	(void)in_le32(mv64x60_irq_reg_base + MV64X60_IC_CPU0_INTR_MASK_LO);}static struct irq_chip mv64x60_chip_low = {	.name		= "mv64x60_low",	.mask		= mv64x60_mask_low,	.mask_ack	= mv64x60_mask_low,	.unmask		= mv64x60_unmask_low,};/* * mv64x60_chip_high functions */static void mv64x60_mask_high(unsigned int virq){	int level2 = irq_map[virq].hwirq & MV64x60_LEVEL2_MASK;	unsigned long flags;	spin_lock_irqsave(&mv64x60_lock, flags);	mv64x60_cached_high_mask &= ~(1 << level2);	out_le32(mv64x60_irq_reg_base + MV64X60_IC_CPU0_INTR_MASK_HI,		 mv64x60_cached_high_mask);	spin_unlock_irqrestore(&mv64x60_lock, flags);	(void)in_le32(mv64x60_irq_reg_base + MV64X60_IC_CPU0_INTR_MASK_HI);}static void mv64x60_unmask_high(unsigned int virq){	int level2 = irq_map[virq].hwirq & MV64x60_LEVEL2_MASK;	unsigned long flags;	spin_lock_irqsave(&mv64x60_lock, flags);	mv64x60_cached_high_mask |= 1 << level2;	out_le32(mv64x60_irq_reg_base + MV64X60_IC_CPU0_INTR_MASK_HI,		 mv64x60_cached_high_mask);	spin_unlock_irqrestore(&mv64x60_lock, flags);	(void)in_le32(mv64x60_irq_reg_base + MV64X60_IC_CPU0_INTR_MASK_HI);}static struct irq_chip mv64x60_chip_high = {	.name		= "mv64x60_high",	.mask		= mv64x60_mask_high,	.mask_ack	= mv64x60_mask_high,	.unmask		= mv64x60_unmask_high,};/* * mv64x60_chip_gpp functions */static void mv64x60_mask_gpp(unsigned int virq){	int level2 = irq_map[virq].hwirq & MV64x60_LEVEL2_MASK;	unsigned long flags;	spin_lock_irqsave(&mv64x60_lock, flags);	mv64x60_cached_gpp_mask &= ~(1 << level2);	out_le32(mv64x60_gpp_reg_base + MV64x60_GPP_INTR_MASK,		 mv64x60_cached_gpp_mask);	spin_unlock_irqrestore(&mv64x60_lock, flags);	(void)in_le32(mv64x60_gpp_reg_base + MV64x60_GPP_INTR_MASK);}static void mv64x60_mask_ack_gpp(unsigned int virq){	int level2 = irq_map[virq].hwirq & MV64x60_LEVEL2_MASK;	unsigned long flags;	spin_lock_irqsave(&mv64x60_lock, flags);	mv64x60_cached_gpp_mask &= ~(1 << level2);	out_le32(mv64x60_gpp_reg_base + MV64x60_GPP_INTR_MASK,		 mv64x60_cached_gpp_mask);	out_le32(mv64x60_gpp_reg_base + MV64x60_GPP_INTR_CAUSE,		 ~(1 << level2));	spin_unlock_irqrestore(&mv64x60_lock, flags);	(void)in_le32(mv64x60_gpp_reg_base + MV64x60_GPP_INTR_CAUSE);}static void mv64x60_unmask_gpp(unsigned int virq){	int level2 = irq_map[virq].hwirq & MV64x60_LEVEL2_MASK;	unsigned long flags;	spin_lock_irqsave(&mv64x60_lock, flags);	mv64x60_cached_gpp_mask |= 1 << level2;	out_le32(mv64x60_gpp_reg_base + MV64x60_GPP_INTR_MASK,		 mv64x60_cached_gpp_mask);	spin_unlock_irqrestore(&mv64x60_lock, flags);	(void)in_le32(mv64x60_gpp_reg_base + MV64x60_GPP_INTR_MASK);}static struct irq_chip mv64x60_chip_gpp = {	.name		= "mv64x60_gpp",	.mask		= mv64x60_mask_gpp,	.mask_ack	= mv64x60_mask_ack_gpp,	.unmask		= mv64x60_unmask_gpp,};/* * mv64x60_host_ops functions */static struct irq_chip *mv64x60_chips[] = {	[MV64x60_LEVEL1_LOW]  = &mv64x60_chip_low,	[MV64x60_LEVEL1_HIGH] = &mv64x60_chip_high,	[MV64x60_LEVEL1_GPP]  = &mv64x60_chip_gpp,};static int mv64x60_host_map(struct irq_host *h, unsigned int virq,			  irq_hw_number_t hwirq){	int level1;	get_irq_desc(virq)->status |= IRQ_LEVEL;	level1 = (hwirq & MV64x60_LEVEL1_MASK) >> MV64x60_LEVEL1_OFFSET;	BUG_ON(level1 > MV64x60_LEVEL1_GPP);	set_irq_chip_and_handler(virq, mv64x60_chips[level1], handle_level_irq);	return 0;}static struct irq_host_ops mv64x60_host_ops = {	.map   = mv64x60_host_map,};/* * Global functions */void __init mv64x60_init_irq(void){	struct device_node *np;	phys_addr_t paddr;	unsigned int size;	const unsigned int *reg;	unsigned long flags;	np = of_find_compatible_node(NULL, NULL, "marvell,mv64x60-gpp");	reg = of_get_property(np, "reg", &size);	paddr = of_translate_address(np, reg);	mv64x60_gpp_reg_base = ioremap(paddr, reg[1]);	of_node_put(np);	np = of_find_compatible_node(NULL, NULL, "marvell,mv64x60-pic");	reg = of_get_property(np, "reg", &size);	paddr = of_translate_address(np, reg);	mv64x60_irq_reg_base = ioremap(paddr, reg[1]);	mv64x60_irq_host = irq_alloc_host(np, IRQ_HOST_MAP_LINEAR,					  MV64x60_NUM_IRQS,					  &mv64x60_host_ops, MV64x60_NUM_IRQS);	spin_lock_irqsave(&mv64x60_lock, flags);	out_le32(mv64x60_gpp_reg_base + MV64x60_GPP_INTR_MASK,		 mv64x60_cached_gpp_mask);	out_le32(mv64x60_irq_reg_base + MV64X60_IC_CPU0_INTR_MASK_LO,		 mv64x60_cached_low_mask);	out_le32(mv64x60_irq_reg_base + MV64X60_IC_CPU0_INTR_MASK_HI,		 mv64x60_cached_high_mask);	out_le32(mv64x60_gpp_reg_base + MV64x60_GPP_INTR_CAUSE, 0);	out_le32(mv64x60_irq_reg_base + MV64X60_IC_MAIN_CAUSE_LO, 0);	out_le32(mv64x60_irq_reg_base + MV64X60_IC_MAIN_CAUSE_HI, 0);	spin_unlock_irqrestore(&mv64x60_lock, flags);}unsigned int mv64x60_get_irq(void){	u32 cause;	int level1;	irq_hw_number_t hwirq;	int virq = NO_IRQ;	cause = in_le32(mv64x60_irq_reg_base + MV64X60_IC_CPU0_SELECT_CAUSE);	if (cause & MV64X60_SELECT_CAUSE_HIGH) {		cause &= mv64x60_cached_high_mask;		level1 = MV64x60_LEVEL1_HIGH;		if (cause & MV64X60_HIGH_GPP_GROUPS) {			cause = in_le32(mv64x60_gpp_reg_base +					MV64x60_GPP_INTR_CAUSE);			cause &= mv64x60_cached_gpp_mask;			level1 = MV64x60_LEVEL1_GPP;		}	} else {		cause &= mv64x60_cached_low_mask;		level1 = MV64x60_LEVEL1_LOW;	}	if (cause) {		hwirq = (level1 << MV64x60_LEVEL1_OFFSET) | __ilog2(cause);		virq = irq_linear_revmap(mv64x60_irq_host, hwirq);	}	return virq;}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?