| /* | 
 |  *  linux/arch/arm/mach-mmp/irq-mmp2.c | 
 |  * | 
 |  *  Generic IRQ handling, GPIO IRQ demultiplexing, etc. | 
 |  * | 
 |  *  Author:	Haojian Zhuang <haojian.zhuang@marvell.com> | 
 |  *  Copyright:	Marvell International Ltd. | 
 |  * | 
 |  *  This program is free software; you can redistribute it and/or modify | 
 |  *  it under the terms of the GNU General Public License version 2 as | 
 |  *  published by the Free Software Foundation. | 
 |  */ | 
 |  | 
 | #include <linux/init.h> | 
 | #include <linux/irq.h> | 
 | #include <linux/io.h> | 
 |  | 
 | #include <mach/regs-icu.h> | 
 | #include <mach/mmp2.h> | 
 |  | 
 | #include "common.h" | 
 |  | 
 | static void icu_mask_irq(unsigned int irq) | 
 | { | 
 | 	uint32_t r = __raw_readl(ICU_INT_CONF(irq)); | 
 |  | 
 | 	r &= ~ICU_INT_ROUTE_PJ4_IRQ; | 
 | 	__raw_writel(r, ICU_INT_CONF(irq)); | 
 | } | 
 |  | 
 | static void icu_unmask_irq(unsigned int irq) | 
 | { | 
 | 	uint32_t r = __raw_readl(ICU_INT_CONF(irq)); | 
 |  | 
 | 	r |= ICU_INT_ROUTE_PJ4_IRQ; | 
 | 	__raw_writel(r, ICU_INT_CONF(irq)); | 
 | } | 
 |  | 
 | static struct irq_chip icu_irq_chip = { | 
 | 	.name		= "icu_irq", | 
 | 	.mask		= icu_mask_irq, | 
 | 	.mask_ack	= icu_mask_irq, | 
 | 	.unmask		= icu_unmask_irq, | 
 | }; | 
 |  | 
 | static void pmic_irq_ack(unsigned int irq) | 
 | { | 
 | 	if (irq == IRQ_MMP2_PMIC) | 
 | 		mmp2_clear_pmic_int(); | 
 | } | 
 |  | 
 | #define SECOND_IRQ_MASK(_name_, irq_base, prefix)			\ | 
 | static void _name_##_mask_irq(unsigned int irq)				\ | 
 | {									\ | 
 | 	uint32_t r;							\ | 
 | 	r = __raw_readl(prefix##_MASK) | (1 << (irq - irq_base));	\ | 
 | 	__raw_writel(r, prefix##_MASK);					\ | 
 | } | 
 |  | 
 | #define SECOND_IRQ_UNMASK(_name_, irq_base, prefix)			\ | 
 | static void _name_##_unmask_irq(unsigned int irq)			\ | 
 | {									\ | 
 | 	uint32_t r;							\ | 
 | 	r = __raw_readl(prefix##_MASK) & ~(1 << (irq - irq_base));	\ | 
 | 	__raw_writel(r, prefix##_MASK);					\ | 
 | } | 
 |  | 
 | #define SECOND_IRQ_DEMUX(_name_, irq_base, prefix)			\ | 
 | static void _name_##_irq_demux(unsigned int irq, struct irq_desc *desc)	\ | 
 | {									\ | 
 | 	unsigned long status, mask, n;					\ | 
 | 	mask = __raw_readl(prefix##_MASK);				\ | 
 | 	while (1) {							\ | 
 | 		status = __raw_readl(prefix##_STATUS) & ~mask;		\ | 
 | 		if (status == 0)					\ | 
 | 			break;						\ | 
 | 		n = find_first_bit(&status, BITS_PER_LONG);		\ | 
 | 		while (n < BITS_PER_LONG) {				\ | 
 | 			generic_handle_irq(irq_base + n);		\ | 
 | 			n = find_next_bit(&status, BITS_PER_LONG, n+1);	\ | 
 | 		}							\ | 
 | 	}								\ | 
 | } | 
 |  | 
 | #define SECOND_IRQ_CHIP(_name_, irq_base, prefix)			\ | 
 | SECOND_IRQ_MASK(_name_, irq_base, prefix)				\ | 
 | SECOND_IRQ_UNMASK(_name_, irq_base, prefix)				\ | 
 | SECOND_IRQ_DEMUX(_name_, irq_base, prefix)				\ | 
 | static struct irq_chip _name_##_irq_chip = {				\ | 
 | 	.name		= #_name_,					\ | 
 | 	.mask		= _name_##_mask_irq,				\ | 
 | 	.unmask		= _name_##_unmask_irq,				\ | 
 | } | 
 |  | 
 | SECOND_IRQ_CHIP(pmic, IRQ_MMP2_PMIC_BASE, MMP2_ICU_INT4); | 
 | SECOND_IRQ_CHIP(rtc,  IRQ_MMP2_RTC_BASE,  MMP2_ICU_INT5); | 
 | SECOND_IRQ_CHIP(twsi, IRQ_MMP2_TWSI_BASE, MMP2_ICU_INT17); | 
 | SECOND_IRQ_CHIP(misc, IRQ_MMP2_MISC_BASE, MMP2_ICU_INT35); | 
 | SECOND_IRQ_CHIP(ssp,  IRQ_MMP2_SSP_BASE,  MMP2_ICU_INT51); | 
 |  | 
 | static void init_mux_irq(struct irq_chip *chip, int start, int num) | 
 | { | 
 | 	int irq; | 
 |  | 
 | 	for (irq = start; num > 0; irq++, num--) { | 
 | 		/* mask and clear the IRQ */ | 
 | 		chip->mask(irq); | 
 | 		if (chip->ack) | 
 | 			chip->ack(irq); | 
 |  | 
 | 		set_irq_chip(irq, chip); | 
 | 		set_irq_flags(irq, IRQF_VALID); | 
 | 		set_irq_handler(irq, handle_level_irq); | 
 | 	} | 
 | } | 
 |  | 
 | void __init mmp2_init_icu(void) | 
 | { | 
 | 	int irq; | 
 |  | 
 | 	for (irq = 0; irq < IRQ_MMP2_MUX_BASE; irq++) { | 
 | 		icu_mask_irq(irq); | 
 | 		set_irq_chip(irq, &icu_irq_chip); | 
 | 		set_irq_flags(irq, IRQF_VALID); | 
 |  | 
 | 		switch (irq) { | 
 | 		case IRQ_MMP2_PMIC_MUX: | 
 | 		case IRQ_MMP2_RTC_MUX: | 
 | 		case IRQ_MMP2_TWSI_MUX: | 
 | 		case IRQ_MMP2_MISC_MUX: | 
 | 		case IRQ_MMP2_SSP_MUX: | 
 | 			break; | 
 | 		default: | 
 | 			set_irq_handler(irq, handle_level_irq); | 
 | 			break; | 
 | 		} | 
 | 	} | 
 |  | 
 | 	/* NOTE: IRQ_MMP2_PMIC requires the PMIC MFPR register | 
 | 	 * to be written to clear the interrupt | 
 | 	 */ | 
 | 	pmic_irq_chip.ack = pmic_irq_ack; | 
 |  | 
 | 	init_mux_irq(&pmic_irq_chip, IRQ_MMP2_PMIC_BASE, 2); | 
 | 	init_mux_irq(&rtc_irq_chip, IRQ_MMP2_RTC_BASE, 2); | 
 | 	init_mux_irq(&twsi_irq_chip, IRQ_MMP2_TWSI_BASE, 5); | 
 | 	init_mux_irq(&misc_irq_chip, IRQ_MMP2_MISC_BASE, 15); | 
 | 	init_mux_irq(&ssp_irq_chip, IRQ_MMP2_SSP_BASE, 2); | 
 |  | 
 | 	set_irq_chained_handler(IRQ_MMP2_PMIC_MUX, pmic_irq_demux); | 
 | 	set_irq_chained_handler(IRQ_MMP2_RTC_MUX, rtc_irq_demux); | 
 | 	set_irq_chained_handler(IRQ_MMP2_TWSI_MUX, twsi_irq_demux); | 
 | 	set_irq_chained_handler(IRQ_MMP2_MISC_MUX, misc_irq_demux); | 
 | 	set_irq_chained_handler(IRQ_MMP2_SSP_MUX, ssp_irq_demux); | 
 | } |