diff options
Diffstat (limited to 'arch/mips/kernel/irq_cpu.c')
| -rw-r--r-- | arch/mips/kernel/irq_cpu.c | 113 | 
1 files changed, 74 insertions, 39 deletions
diff --git a/arch/mips/kernel/irq_cpu.c b/arch/mips/kernel/irq_cpu.c index 0262abe0912..e498f2b3646 100644 --- a/arch/mips/kernel/irq_cpu.c +++ b/arch/mips/kernel/irq_cpu.c @@ -3,13 +3,13 @@   * Author: Jun Sun, jsun@mvista.com or jsun@junsun.net   *   * Copyright (C) 2001 Ralf Baechle - * Copyright (C) 2005  MIPS Technologies, Inc.  All rights reserved. - *      Author: Maciej W. Rozycki <macro@mips.com> + * Copyright (C) 2005  MIPS Technologies, Inc.	All rights reserved. + *	Author: Maciej W. Rozycki <macro@mips.com>   *   * This file define the irq handler for MIPS CPU interrupts.   * - * This program is free software; you can redistribute  it and/or modify it - * under  the terms of  the GNU General  Public License as published by the + * This program is free software; you can redistribute	it and/or modify it + * under  the terms of	the GNU General	 Public License as published by the   * Free Software Foundation;  either version 2 of the  License, or (at your   * option) any later version.   */ @@ -31,48 +31,44 @@  #include <linux/interrupt.h>  #include <linux/kernel.h>  #include <linux/irq.h> +#include <linux/irqdomain.h>  #include <asm/irq_cpu.h>  #include <asm/mipsregs.h>  #include <asm/mipsmtregs.h> -#include <asm/system.h> -static inline void unmask_mips_irq(unsigned int irq) +static inline void unmask_mips_irq(struct irq_data *d)  { -	set_c0_status(0x100 << (irq - MIPS_CPU_IRQ_BASE)); +	set_c0_status(0x100 << (d->irq - MIPS_CPU_IRQ_BASE));  	irq_enable_hazard();  } -static inline void mask_mips_irq(unsigned int irq) +static inline void mask_mips_irq(struct irq_data *d)  { -	clear_c0_status(0x100 << (irq - MIPS_CPU_IRQ_BASE)); +	clear_c0_status(0x100 << (d->irq - MIPS_CPU_IRQ_BASE));  	irq_disable_hazard();  }  static struct irq_chip mips_cpu_irq_controller = {  	.name		= "MIPS", -	.ack		= mask_mips_irq, -	.mask		= mask_mips_irq, -	.mask_ack	= mask_mips_irq, -	.unmask		= unmask_mips_irq, -	.eoi		= unmask_mips_irq, +	.irq_ack	= mask_mips_irq, +	.irq_mask	= mask_mips_irq, +	.irq_mask_ack	= mask_mips_irq, +	.irq_unmask	= unmask_mips_irq, +	.irq_eoi	= unmask_mips_irq,  };  /*   * Basically the same as above but taking care of all the MT stuff   */ -#define unmask_mips_mt_irq	unmask_mips_irq -#define mask_mips_mt_irq	mask_mips_irq - -static unsigned int mips_mt_cpu_irq_startup(unsigned int irq) +static unsigned int mips_mt_cpu_irq_startup(struct irq_data *d)  {  	unsigned int vpflags = dvpe(); -	clear_c0_cause(0x100 << (irq - MIPS_CPU_IRQ_BASE)); +	clear_c0_cause(0x100 << (d->irq - MIPS_CPU_IRQ_BASE));  	evpe(vpflags); -	unmask_mips_mt_irq(irq); - +	unmask_mips_irq(d);  	return 0;  } @@ -80,22 +76,22 @@ static unsigned int mips_mt_cpu_irq_startup(unsigned int irq)   * While we ack the interrupt interrupts are disabled and thus we don't need   * to deal with concurrency issues.  Same for mips_cpu_irq_end.   */ -static void mips_mt_cpu_irq_ack(unsigned int irq) +static void mips_mt_cpu_irq_ack(struct irq_data *d)  {  	unsigned int vpflags = dvpe(); -	clear_c0_cause(0x100 << (irq - MIPS_CPU_IRQ_BASE)); +	clear_c0_cause(0x100 << (d->irq - MIPS_CPU_IRQ_BASE));  	evpe(vpflags); -	mask_mips_mt_irq(irq); +	mask_mips_irq(d);  }  static struct irq_chip mips_mt_cpu_irq_controller = {  	.name		= "MIPS", -	.startup	= mips_mt_cpu_irq_startup, -	.ack		= mips_mt_cpu_irq_ack, -	.mask		= mask_mips_mt_irq, -	.mask_ack	= mips_mt_cpu_irq_ack, -	.unmask		= unmask_mips_mt_irq, -	.eoi		= unmask_mips_mt_irq, +	.irq_startup	= mips_mt_cpu_irq_startup, +	.irq_ack	= mips_mt_cpu_irq_ack, +	.irq_mask	= mask_mips_irq, +	.irq_mask_ack	= mips_mt_cpu_irq_ack, +	.irq_unmask	= unmask_mips_irq, +	.irq_eoi	= unmask_mips_irq,  };  void __init mips_cpu_irq_init(void) @@ -107,16 +103,55 @@ void __init mips_cpu_irq_init(void)  	clear_c0_status(ST0_IM);  	clear_c0_cause(CAUSEF_IP); -	/* -	 * Only MT is using the software interrupts currently, so we just -	 * leave them uninitialized for other processors. -	 */ -	if (cpu_has_mipsmt) -		for (i = irq_base; i < irq_base + 2; i++) -			set_irq_chip_and_handler(i, &mips_mt_cpu_irq_controller, -						 handle_percpu_irq); +	/* Software interrupts are used for MT/CMT IPI */ +	for (i = irq_base; i < irq_base + 2; i++) +		irq_set_chip_and_handler(i, cpu_has_mipsmt ? +					 &mips_mt_cpu_irq_controller : +					 &mips_cpu_irq_controller, +					 handle_percpu_irq);  	for (i = irq_base + 2; i < irq_base + 8; i++) -		set_irq_chip_and_handler(i, &mips_cpu_irq_controller, +		irq_set_chip_and_handler(i, &mips_cpu_irq_controller,  					 handle_percpu_irq);  } + +#ifdef CONFIG_IRQ_DOMAIN +static int mips_cpu_intc_map(struct irq_domain *d, unsigned int irq, +			     irq_hw_number_t hw) +{ +	static struct irq_chip *chip; + +	if (hw < 2 && cpu_has_mipsmt) { +		/* Software interrupts are used for MT/CMT IPI */ +		chip = &mips_mt_cpu_irq_controller; +	} else { +		chip = &mips_cpu_irq_controller; +	} + +	irq_set_chip_and_handler(irq, chip, handle_percpu_irq); + +	return 0; +} + +static const struct irq_domain_ops mips_cpu_intc_irq_domain_ops = { +	.map = mips_cpu_intc_map, +	.xlate = irq_domain_xlate_onecell, +}; + +int __init mips_cpu_intc_init(struct device_node *of_node, +			      struct device_node *parent) +{ +	struct irq_domain *domain; + +	/* Mask interrupts. */ +	clear_c0_status(ST0_IM); +	clear_c0_cause(CAUSEF_IP); + +	domain = irq_domain_add_legacy(of_node, 8, MIPS_CPU_IRQ_BASE, 0, +				       &mips_cpu_intc_irq_domain_ops, NULL); +	if (!domain) +		panic("Failed to add irqdomain for MIPS CPU"); + +	return 0; +} +#endif /* CONFIG_IRQ_DOMAIN */  | 
