forked from mirrors/linux
		
	[POWERPC] Adapt ipic driver to new host_ops interface, add set_irq_type to set IRQ sense
This converts ipic code to Benh's IRQ mods. For the IPIC, IRQ sense values in the device tree equal those in include/linux/irq.h; that's 8 for low assertion (most internal IRQs on mpc83xx), and 2 for high-to-low change. spinlocks added to [un]mask, ack operations; default handler and type now set in host_map; and redundant condition check eliminated. Signed-off-by: Kim Phillips <kim.phillips@freescale.com> Signed-off-by: Li Yang <leoli@freescale.com> Signed-off-by: Paul Mackerras <paulus@samba.org>
This commit is contained in:
		
							parent
							
								
									bf4152dd7c
								
							
						
					
					
						commit
						b9f0f1bb2b
					
				
					 2 changed files with 228 additions and 98 deletions
				
			
		| 
						 | 
					@ -19,15 +19,18 @@
 | 
				
			||||||
#include <linux/sched.h>
 | 
					#include <linux/sched.h>
 | 
				
			||||||
#include <linux/signal.h>
 | 
					#include <linux/signal.h>
 | 
				
			||||||
#include <linux/sysdev.h>
 | 
					#include <linux/sysdev.h>
 | 
				
			||||||
 | 
					#include <linux/device.h>
 | 
				
			||||||
 | 
					#include <linux/bootmem.h>
 | 
				
			||||||
 | 
					#include <linux/spinlock.h>
 | 
				
			||||||
#include <asm/irq.h>
 | 
					#include <asm/irq.h>
 | 
				
			||||||
#include <asm/io.h>
 | 
					#include <asm/io.h>
 | 
				
			||||||
 | 
					#include <asm/prom.h>
 | 
				
			||||||
#include <asm/ipic.h>
 | 
					#include <asm/ipic.h>
 | 
				
			||||||
#include <asm/mpc83xx.h>
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
#include "ipic.h"
 | 
					#include "ipic.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static struct ipic p_ipic;
 | 
					 | 
				
			||||||
static struct ipic * primary_ipic;
 | 
					static struct ipic * primary_ipic;
 | 
				
			||||||
 | 
					static DEFINE_SPINLOCK(ipic_lock);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static struct ipic_info ipic_info[] = {
 | 
					static struct ipic_info ipic_info[] = {
 | 
				
			||||||
	[9] = {
 | 
						[9] = {
 | 
				
			||||||
| 
						 | 
					@ -373,74 +376,220 @@ static inline void ipic_write(volatile u32 __iomem *base, unsigned int reg, u32
 | 
				
			||||||
	out_be32(base + (reg >> 2), value);
 | 
						out_be32(base + (reg >> 2), value);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static inline struct ipic * ipic_from_irq(unsigned int irq)
 | 
					static inline struct ipic * ipic_from_irq(unsigned int virq)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	return primary_ipic;
 | 
						return primary_ipic;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void ipic_enable_irq(unsigned int irq)
 | 
					#define ipic_irq_to_hw(virq)	((unsigned int)irq_map[virq].hwirq)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void ipic_unmask_irq(unsigned int virq)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct ipic *ipic = ipic_from_irq(irq);
 | 
						struct ipic *ipic = ipic_from_irq(virq);
 | 
				
			||||||
	unsigned int src = irq - ipic->irq_offset;
 | 
						unsigned int src = ipic_irq_to_hw(virq);
 | 
				
			||||||
 | 
						unsigned long flags;
 | 
				
			||||||
	u32 temp;
 | 
						u32 temp;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						spin_lock_irqsave(&ipic_lock, flags);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	temp = ipic_read(ipic->regs, ipic_info[src].mask);
 | 
						temp = ipic_read(ipic->regs, ipic_info[src].mask);
 | 
				
			||||||
	temp |= (1 << (31 - ipic_info[src].bit));
 | 
						temp |= (1 << (31 - ipic_info[src].bit));
 | 
				
			||||||
	ipic_write(ipic->regs, ipic_info[src].mask, temp);
 | 
						ipic_write(ipic->regs, ipic_info[src].mask, temp);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						spin_unlock_irqrestore(&ipic_lock, flags);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void ipic_disable_irq(unsigned int irq)
 | 
					static void ipic_mask_irq(unsigned int virq)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct ipic *ipic = ipic_from_irq(irq);
 | 
						struct ipic *ipic = ipic_from_irq(virq);
 | 
				
			||||||
	unsigned int src = irq - ipic->irq_offset;
 | 
						unsigned int src = ipic_irq_to_hw(virq);
 | 
				
			||||||
 | 
						unsigned long flags;
 | 
				
			||||||
	u32 temp;
 | 
						u32 temp;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						spin_lock_irqsave(&ipic_lock, flags);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	temp = ipic_read(ipic->regs, ipic_info[src].mask);
 | 
						temp = ipic_read(ipic->regs, ipic_info[src].mask);
 | 
				
			||||||
	temp &= ~(1 << (31 - ipic_info[src].bit));
 | 
						temp &= ~(1 << (31 - ipic_info[src].bit));
 | 
				
			||||||
	ipic_write(ipic->regs, ipic_info[src].mask, temp);
 | 
						ipic_write(ipic->regs, ipic_info[src].mask, temp);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						spin_unlock_irqrestore(&ipic_lock, flags);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void ipic_disable_irq_and_ack(unsigned int irq)
 | 
					static void ipic_ack_irq(unsigned int virq)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct ipic *ipic = ipic_from_irq(irq);
 | 
						struct ipic *ipic = ipic_from_irq(virq);
 | 
				
			||||||
	unsigned int src = irq - ipic->irq_offset;
 | 
						unsigned int src = ipic_irq_to_hw(virq);
 | 
				
			||||||
 | 
						unsigned long flags;
 | 
				
			||||||
	u32 temp;
 | 
						u32 temp;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	ipic_disable_irq(irq);
 | 
						spin_lock_irqsave(&ipic_lock, flags);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	temp = ipic_read(ipic->regs, ipic_info[src].pend);
 | 
						temp = ipic_read(ipic->regs, ipic_info[src].pend);
 | 
				
			||||||
	temp |= (1 << (31 - ipic_info[src].bit));
 | 
						temp |= (1 << (31 - ipic_info[src].bit));
 | 
				
			||||||
	ipic_write(ipic->regs, ipic_info[src].pend, temp);
 | 
						ipic_write(ipic->regs, ipic_info[src].pend, temp);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						spin_unlock_irqrestore(&ipic_lock, flags);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void ipic_end_irq(unsigned int irq)
 | 
					static void ipic_mask_irq_and_ack(unsigned int virq)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	if (!(irq_desc[irq].status & (IRQ_DISABLED|IRQ_INPROGRESS)))
 | 
						struct ipic *ipic = ipic_from_irq(virq);
 | 
				
			||||||
		ipic_enable_irq(irq);
 | 
						unsigned int src = ipic_irq_to_hw(virq);
 | 
				
			||||||
 | 
						unsigned long flags;
 | 
				
			||||||
 | 
						u32 temp;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						spin_lock_irqsave(&ipic_lock, flags);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						temp = ipic_read(ipic->regs, ipic_info[src].mask);
 | 
				
			||||||
 | 
						temp &= ~(1 << (31 - ipic_info[src].bit));
 | 
				
			||||||
 | 
						ipic_write(ipic->regs, ipic_info[src].mask, temp);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						temp = ipic_read(ipic->regs, ipic_info[src].pend);
 | 
				
			||||||
 | 
						temp |= (1 << (31 - ipic_info[src].bit));
 | 
				
			||||||
 | 
						ipic_write(ipic->regs, ipic_info[src].pend, temp);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						spin_unlock_irqrestore(&ipic_lock, flags);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
struct hw_interrupt_type ipic = {
 | 
					static int ipic_set_irq_type(unsigned int virq, unsigned int flow_type)
 | 
				
			||||||
	.typename = " IPIC  ",
 | 
					{
 | 
				
			||||||
	.enable = ipic_enable_irq,
 | 
						struct ipic *ipic = ipic_from_irq(virq);
 | 
				
			||||||
	.disable = ipic_disable_irq,
 | 
						unsigned int src = ipic_irq_to_hw(virq);
 | 
				
			||||||
	.ack = ipic_disable_irq_and_ack,
 | 
						struct irq_desc *desc = get_irq_desc(virq);
 | 
				
			||||||
	.end = ipic_end_irq,
 | 
						unsigned int vold, vnew, edibit;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (flow_type == IRQ_TYPE_NONE)
 | 
				
			||||||
 | 
							flow_type = IRQ_TYPE_LEVEL_LOW;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* ipic supports only low assertion and high-to-low change senses
 | 
				
			||||||
 | 
						 */
 | 
				
			||||||
 | 
						if (!(flow_type & (IRQ_TYPE_LEVEL_LOW | IRQ_TYPE_EDGE_FALLING))) {
 | 
				
			||||||
 | 
							printk(KERN_ERR "ipic: sense type 0x%x not supported\n",
 | 
				
			||||||
 | 
								flow_type);
 | 
				
			||||||
 | 
							return -EINVAL;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						desc->status &= ~(IRQ_TYPE_SENSE_MASK | IRQ_LEVEL);
 | 
				
			||||||
 | 
						desc->status |= flow_type & IRQ_TYPE_SENSE_MASK;
 | 
				
			||||||
 | 
						if (flow_type & IRQ_TYPE_LEVEL_LOW)  {
 | 
				
			||||||
 | 
							desc->status |= IRQ_LEVEL;
 | 
				
			||||||
 | 
							set_irq_handler(virq, handle_level_irq);
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							set_irq_handler(virq, handle_edge_irq);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* only EXT IRQ senses are programmable on ipic
 | 
				
			||||||
 | 
						 * internal IRQ senses are LEVEL_LOW
 | 
				
			||||||
 | 
						 */
 | 
				
			||||||
 | 
						if (src == IPIC_IRQ_EXT0)
 | 
				
			||||||
 | 
							edibit = 15;
 | 
				
			||||||
 | 
						else
 | 
				
			||||||
 | 
							if (src >= IPIC_IRQ_EXT1 && src <= IPIC_IRQ_EXT7)
 | 
				
			||||||
 | 
								edibit = (14 - (src - IPIC_IRQ_EXT1));
 | 
				
			||||||
 | 
							else
 | 
				
			||||||
 | 
								return (flow_type & IRQ_TYPE_LEVEL_LOW) ? 0 : -EINVAL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						vold = ipic_read(ipic->regs, IPIC_SECNR);
 | 
				
			||||||
 | 
						if ((flow_type & IRQ_TYPE_SENSE_MASK) == IRQ_TYPE_EDGE_FALLING) {
 | 
				
			||||||
 | 
							vnew = vold | (1 << edibit);
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							vnew = vold & ~(1 << edibit);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if (vold != vnew)
 | 
				
			||||||
 | 
							ipic_write(ipic->regs, IPIC_SECNR, vnew);
 | 
				
			||||||
 | 
						return 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static struct irq_chip ipic_irq_chip = {
 | 
				
			||||||
 | 
						.typename	= " IPIC  ",
 | 
				
			||||||
 | 
						.unmask		= ipic_unmask_irq,
 | 
				
			||||||
 | 
						.mask		= ipic_mask_irq,
 | 
				
			||||||
 | 
						.mask_ack	= ipic_mask_irq_and_ack,
 | 
				
			||||||
 | 
						.ack		= ipic_ack_irq,
 | 
				
			||||||
 | 
						.set_type	= ipic_set_irq_type,
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void __init ipic_init(phys_addr_t phys_addr,
 | 
					static int ipic_host_match(struct irq_host *h, struct device_node *node)
 | 
				
			||||||
		unsigned int flags,
 | 
					 | 
				
			||||||
		unsigned int irq_offset,
 | 
					 | 
				
			||||||
		unsigned char *senses,
 | 
					 | 
				
			||||||
		unsigned int senses_count)
 | 
					 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	u32 i, temp = 0;
 | 
						struct ipic *ipic = h->host_data;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	primary_ipic = &p_ipic;
 | 
						/* Exact match, unless ipic node is NULL */
 | 
				
			||||||
	primary_ipic->regs = ioremap(phys_addr, MPC83xx_IPIC_SIZE);
 | 
						return ipic->of_node == NULL || ipic->of_node == node;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	primary_ipic->irq_offset = irq_offset;
 | 
					static int ipic_host_map(struct irq_host *h, unsigned int virq,
 | 
				
			||||||
 | 
								 irq_hw_number_t hw)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						struct ipic *ipic = h->host_data;
 | 
				
			||||||
 | 
						struct irq_chip *chip;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	ipic_write(primary_ipic->regs, IPIC_SICNR, 0x0);
 | 
						/* Default chip */
 | 
				
			||||||
 | 
						chip = &ipic->hc_irq;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						set_irq_chip_data(virq, ipic);
 | 
				
			||||||
 | 
						set_irq_chip_and_handler(virq, chip, handle_level_irq);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Set default irq type */
 | 
				
			||||||
 | 
						set_irq_type(virq, IRQ_TYPE_NONE);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static int ipic_host_xlate(struct irq_host *h, struct device_node *ct,
 | 
				
			||||||
 | 
								   u32 *intspec, unsigned int intsize,
 | 
				
			||||||
 | 
								   irq_hw_number_t *out_hwirq, unsigned int *out_flags)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						/* interrupt sense values coming from the device tree equal either
 | 
				
			||||||
 | 
						 * LEVEL_LOW (low assertion) or EDGE_FALLING (high-to-low change)
 | 
				
			||||||
 | 
						 */
 | 
				
			||||||
 | 
						*out_hwirq = intspec[0];
 | 
				
			||||||
 | 
						if (intsize > 1)
 | 
				
			||||||
 | 
							*out_flags = intspec[1];
 | 
				
			||||||
 | 
						else
 | 
				
			||||||
 | 
							*out_flags = IRQ_TYPE_NONE;
 | 
				
			||||||
 | 
						return 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static struct irq_host_ops ipic_host_ops = {
 | 
				
			||||||
 | 
						.match	= ipic_host_match,
 | 
				
			||||||
 | 
						.map	= ipic_host_map,
 | 
				
			||||||
 | 
						.xlate	= ipic_host_xlate,
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void __init ipic_init(struct device_node *node,
 | 
				
			||||||
 | 
							unsigned int flags)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						struct ipic	*ipic;
 | 
				
			||||||
 | 
						struct resource res;
 | 
				
			||||||
 | 
						u32 temp = 0, ret;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ipic = alloc_bootmem(sizeof(struct ipic));
 | 
				
			||||||
 | 
						if (ipic == NULL)
 | 
				
			||||||
 | 
							return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						memset(ipic, 0, sizeof(struct ipic));
 | 
				
			||||||
 | 
						ipic->of_node = node ? of_node_get(node) : NULL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ipic->irqhost = irq_alloc_host(IRQ_HOST_MAP_LINEAR,
 | 
				
			||||||
 | 
									       NR_IPIC_INTS,
 | 
				
			||||||
 | 
									       &ipic_host_ops, 0);
 | 
				
			||||||
 | 
						if (ipic->irqhost == NULL) {
 | 
				
			||||||
 | 
							of_node_put(node);
 | 
				
			||||||
 | 
							return;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ret = of_address_to_resource(node, 0, &res);
 | 
				
			||||||
 | 
						if (ret)
 | 
				
			||||||
 | 
							return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ipic->regs = ioremap(res.start, res.end - res.start + 1);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ipic->irqhost->host_data = ipic;
 | 
				
			||||||
 | 
						ipic->hc_irq = ipic_irq_chip;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* init hw */
 | 
				
			||||||
 | 
						ipic_write(ipic->regs, IPIC_SICNR, 0x0);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* default priority scheme is grouped. If spread mode is required
 | 
						/* default priority scheme is grouped. If spread mode is required
 | 
				
			||||||
	 * configure SICFR accordingly */
 | 
						 * configure SICFR accordingly */
 | 
				
			||||||
| 
						 | 
					@ -453,49 +602,35 @@ void __init ipic_init(phys_addr_t phys_addr,
 | 
				
			||||||
	if (flags & IPIC_SPREADMODE_MIX_B)
 | 
						if (flags & IPIC_SPREADMODE_MIX_B)
 | 
				
			||||||
		temp |= SICFR_MPSB;
 | 
							temp |= SICFR_MPSB;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	ipic_write(primary_ipic->regs, IPIC_SICNR, temp);
 | 
						ipic_write(ipic->regs, IPIC_SICNR, temp);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* handle MCP route */
 | 
						/* handle MCP route */
 | 
				
			||||||
	temp = 0;
 | 
						temp = 0;
 | 
				
			||||||
	if (flags & IPIC_DISABLE_MCP_OUT)
 | 
						if (flags & IPIC_DISABLE_MCP_OUT)
 | 
				
			||||||
		temp = SERCR_MCPR;
 | 
							temp = SERCR_MCPR;
 | 
				
			||||||
	ipic_write(primary_ipic->regs, IPIC_SERCR, temp);
 | 
						ipic_write(ipic->regs, IPIC_SERCR, temp);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* handle routing of IRQ0 to MCP */
 | 
						/* handle routing of IRQ0 to MCP */
 | 
				
			||||||
	temp = ipic_read(primary_ipic->regs, IPIC_SEMSR);
 | 
						temp = ipic_read(ipic->regs, IPIC_SEMSR);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (flags & IPIC_IRQ0_MCP)
 | 
						if (flags & IPIC_IRQ0_MCP)
 | 
				
			||||||
		temp |= SEMSR_SIRQ0;
 | 
							temp |= SEMSR_SIRQ0;
 | 
				
			||||||
	else
 | 
						else
 | 
				
			||||||
		temp &= ~SEMSR_SIRQ0;
 | 
							temp &= ~SEMSR_SIRQ0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	ipic_write(primary_ipic->regs, IPIC_SEMSR, temp);
 | 
						ipic_write(ipic->regs, IPIC_SEMSR, temp);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	for (i = 0 ; i < NR_IPIC_INTS ; i++) {
 | 
						primary_ipic = ipic;
 | 
				
			||||||
		irq_desc[i+irq_offset].chip = &ipic;
 | 
						irq_set_default_host(primary_ipic->irqhost);
 | 
				
			||||||
		irq_desc[i+irq_offset].status = IRQ_LEVEL;
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	temp = 0;
 | 
						printk ("IPIC (%d IRQ sources) at %p\n", NR_IPIC_INTS,
 | 
				
			||||||
	for (i = 0 ; i < senses_count ; i++) {
 | 
								primary_ipic->regs);
 | 
				
			||||||
		if ((senses[i] & IRQ_SENSE_MASK) == IRQ_SENSE_EDGE) {
 | 
					 | 
				
			||||||
			temp |= 1 << (15 - i);
 | 
					 | 
				
			||||||
			if (i != 0)
 | 
					 | 
				
			||||||
				irq_desc[i + irq_offset + MPC83xx_IRQ_EXT1 - 1].status = 0;
 | 
					 | 
				
			||||||
			else
 | 
					 | 
				
			||||||
				irq_desc[irq_offset + MPC83xx_IRQ_EXT0].status = 0;
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
	ipic_write(primary_ipic->regs, IPIC_SECNR, temp);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	printk ("IPIC (%d IRQ sources, %d External IRQs) at %p\n", NR_IPIC_INTS,
 | 
					 | 
				
			||||||
			senses_count, primary_ipic->regs);
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
int ipic_set_priority(unsigned int irq, unsigned int priority)
 | 
					int ipic_set_priority(unsigned int virq, unsigned int priority)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct ipic *ipic = ipic_from_irq(irq);
 | 
						struct ipic *ipic = ipic_from_irq(virq);
 | 
				
			||||||
	unsigned int src = irq - ipic->irq_offset;
 | 
						unsigned int src = ipic_irq_to_hw(virq);
 | 
				
			||||||
	u32 temp;
 | 
						u32 temp;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (priority > 7)
 | 
						if (priority > 7)
 | 
				
			||||||
| 
						 | 
					@ -520,10 +655,10 @@ int ipic_set_priority(unsigned int irq, unsigned int priority)
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void ipic_set_highest_priority(unsigned int irq)
 | 
					void ipic_set_highest_priority(unsigned int virq)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct ipic *ipic = ipic_from_irq(irq);
 | 
						struct ipic *ipic = ipic_from_irq(virq);
 | 
				
			||||||
	unsigned int src = irq - ipic->irq_offset;
 | 
						unsigned int src = ipic_irq_to_hw(virq);
 | 
				
			||||||
	u32 temp;
 | 
						u32 temp;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	temp = ipic_read(ipic->regs, IPIC_SICFR);
 | 
						temp = ipic_read(ipic->regs, IPIC_SICFR);
 | 
				
			||||||
| 
						 | 
					@ -537,37 +672,10 @@ void ipic_set_highest_priority(unsigned int irq)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void ipic_set_default_priority(void)
 | 
					void ipic_set_default_priority(void)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_TSEC1_TX, 0);
 | 
						ipic_write(primary_ipic->regs, IPIC_SIPRR_A, IPIC_SIPRR_A_DEFAULT);
 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_TSEC1_RX, 1);
 | 
						ipic_write(primary_ipic->regs, IPIC_SIPRR_D, IPIC_SIPRR_D_DEFAULT);
 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_TSEC1_ERROR, 2);
 | 
						ipic_write(primary_ipic->regs, IPIC_SMPRR_A, IPIC_SMPRR_A_DEFAULT);
 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_TSEC2_TX, 3);
 | 
						ipic_write(primary_ipic->regs, IPIC_SMPRR_B, IPIC_SMPRR_B_DEFAULT);
 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_TSEC2_RX, 4);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_TSEC2_ERROR, 5);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_USB2_DR, 6);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_USB2_MPH, 7);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_UART1, 0);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_UART2, 1);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_SEC2, 2);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_IIC1, 5);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_IIC2, 6);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_SPI, 7);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_RTC_SEC, 0);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_PIT, 1);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_PCI1, 2);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_PCI2, 3);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_EXT0, 4);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_EXT1, 5);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_EXT2, 6);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_EXT3, 7);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_RTC_ALR, 0);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_MU, 1);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_SBA, 2);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_DMA, 3);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_EXT4, 4);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_EXT5, 5);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_EXT6, 6);
 | 
					 | 
				
			||||||
	ipic_set_priority(MPC83xx_IRQ_EXT7, 7);
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void ipic_enable_mcp(enum ipic_mcp_irq mcp_irq)
 | 
					void ipic_enable_mcp(enum ipic_mcp_irq mcp_irq)
 | 
				
			||||||
| 
						 | 
					@ -600,17 +708,20 @@ void ipic_clear_mcp_status(u32 mask)
 | 
				
			||||||
	ipic_write(primary_ipic->regs, IPIC_SERMR, mask);
 | 
						ipic_write(primary_ipic->regs, IPIC_SERMR, mask);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* Return an interrupt vector or -1 if no interrupt is pending. */
 | 
					/* Return an interrupt vector or NO_IRQ if no interrupt is pending. */
 | 
				
			||||||
int ipic_get_irq(struct pt_regs *regs)
 | 
					unsigned int ipic_get_irq(struct pt_regs *regs)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	int irq;
 | 
						int irq;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	irq = ipic_read(primary_ipic->regs, IPIC_SIVCR) & 0x7f;
 | 
						BUG_ON(primary_ipic == NULL);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#define IPIC_SIVCR_VECTOR_MASK	0x7f
 | 
				
			||||||
 | 
						irq = ipic_read(primary_ipic->regs, IPIC_SIVCR) & IPIC_SIVCR_VECTOR_MASK;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (irq == 0)    /* 0 --> no irq is pending */
 | 
						if (irq == 0)    /* 0 --> no irq is pending */
 | 
				
			||||||
		irq = -1;
 | 
							return NO_IRQ;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return irq;
 | 
						return irq_linear_revmap(primary_ipic->irqhost, irq);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static struct sysdev_class ipic_sysclass = {
 | 
					static struct sysdev_class ipic_sysclass = {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -15,7 +15,18 @@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#include <asm/ipic.h>
 | 
					#include <asm/ipic.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define MPC83xx_IPIC_SIZE	(0x00100)
 | 
					#define NR_IPIC_INTS 128
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* External IRQS */
 | 
				
			||||||
 | 
					#define IPIC_IRQ_EXT0 48
 | 
				
			||||||
 | 
					#define IPIC_IRQ_EXT1 17
 | 
				
			||||||
 | 
					#define IPIC_IRQ_EXT7 23
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Default Priority Registers */
 | 
				
			||||||
 | 
					#define IPIC_SIPRR_A_DEFAULT 0x05309770
 | 
				
			||||||
 | 
					#define IPIC_SIPRR_D_DEFAULT 0x05309770
 | 
				
			||||||
 | 
					#define IPIC_SMPRR_A_DEFAULT 0x05309770
 | 
				
			||||||
 | 
					#define IPIC_SMPRR_B_DEFAULT 0x05309770
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* System Global Interrupt Configuration Register */
 | 
					/* System Global Interrupt Configuration Register */
 | 
				
			||||||
#define	SICFR_IPSA	0x00010000
 | 
					#define	SICFR_IPSA	0x00010000
 | 
				
			||||||
| 
						 | 
					@ -31,7 +42,15 @@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
struct ipic {
 | 
					struct ipic {
 | 
				
			||||||
	volatile u32 __iomem	*regs;
 | 
						volatile u32 __iomem	*regs;
 | 
				
			||||||
	unsigned int		irq_offset;
 | 
					
 | 
				
			||||||
 | 
						/* The remapper for this IPIC */
 | 
				
			||||||
 | 
						struct irq_host		*irqhost;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* The "linux" controller struct */
 | 
				
			||||||
 | 
						struct irq_chip		hc_irq;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* The device node of the interrupt controller */
 | 
				
			||||||
 | 
						struct device_node	*of_node;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
struct ipic_info {
 | 
					struct ipic_info {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue