|
@@ -219,3 +219,172 @@ static inline void s3c_irq_eint_ack(struct irq_data *data)
|
|
|
__raw_writel((u32)data->chip_data, S3C64XX_EINT0PEND);
|
|
|
}
|
|
|
|
|
|
+static void s3c_irq_eint_maskack(struct irq_data *data)
|
|
|
+{
|
|
|
+ /* compiler should in-line these */
|
|
|
+ s3c_irq_eint_mask(data);
|
|
|
+ s3c_irq_eint_ack(data);
|
|
|
+}
|
|
|
+
|
|
|
+static int s3c_irq_eint_set_type(struct irq_data *data, unsigned int type)
|
|
|
+{
|
|
|
+ int offs = eint_offset(data->irq);
|
|
|
+ int pin, pin_val;
|
|
|
+ int shift;
|
|
|
+ u32 ctrl, mask;
|
|
|
+ u32 newvalue = 0;
|
|
|
+ void __iomem *reg;
|
|
|
+
|
|
|
+ if (offs > 27)
|
|
|
+ return -EINVAL;
|
|
|
+
|
|
|
+ if (offs <= 15)
|
|
|
+ reg = S3C64XX_EINT0CON0;
|
|
|
+ else
|
|
|
+ reg = S3C64XX_EINT0CON1;
|
|
|
+
|
|
|
+ switch (type) {
|
|
|
+ case IRQ_TYPE_NONE:
|
|
|
+ printk(KERN_WARNING "No edge setting!\n");
|
|
|
+ break;
|
|
|
+
|
|
|
+ case IRQ_TYPE_EDGE_RISING:
|
|
|
+ newvalue = S3C2410_EXTINT_RISEEDGE;
|
|
|
+ break;
|
|
|
+
|
|
|
+ case IRQ_TYPE_EDGE_FALLING:
|
|
|
+ newvalue = S3C2410_EXTINT_FALLEDGE;
|
|
|
+ break;
|
|
|
+
|
|
|
+ case IRQ_TYPE_EDGE_BOTH:
|
|
|
+ newvalue = S3C2410_EXTINT_BOTHEDGE;
|
|
|
+ break;
|
|
|
+
|
|
|
+ case IRQ_TYPE_LEVEL_LOW:
|
|
|
+ newvalue = S3C2410_EXTINT_LOWLEV;
|
|
|
+ break;
|
|
|
+
|
|
|
+ case IRQ_TYPE_LEVEL_HIGH:
|
|
|
+ newvalue = S3C2410_EXTINT_HILEV;
|
|
|
+ break;
|
|
|
+
|
|
|
+ default:
|
|
|
+ printk(KERN_ERR "No such irq type %d", type);
|
|
|
+ return -1;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (offs <= 15)
|
|
|
+ shift = (offs / 2) * 4;
|
|
|
+ else
|
|
|
+ shift = ((offs - 16) / 2) * 4;
|
|
|
+ mask = 0x7 << shift;
|
|
|
+
|
|
|
+ ctrl = __raw_readl(reg);
|
|
|
+ ctrl &= ~mask;
|
|
|
+ ctrl |= newvalue << shift;
|
|
|
+ __raw_writel(ctrl, reg);
|
|
|
+
|
|
|
+ /* set the GPIO pin appropriately */
|
|
|
+
|
|
|
+ if (offs < 16) {
|
|
|
+ pin = S3C64XX_GPN(offs);
|
|
|
+ pin_val = S3C_GPIO_SFN(2);
|
|
|
+ } else if (offs < 23) {
|
|
|
+ pin = S3C64XX_GPL(offs + 8 - 16);
|
|
|
+ pin_val = S3C_GPIO_SFN(3);
|
|
|
+ } else {
|
|
|
+ pin = S3C64XX_GPM(offs - 23);
|
|
|
+ pin_val = S3C_GPIO_SFN(3);
|
|
|
+ }
|
|
|
+
|
|
|
+ s3c_gpio_cfgpin(pin, pin_val);
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+static struct irq_chip s3c_irq_eint = {
|
|
|
+ .name = "s3c-eint",
|
|
|
+ .irq_mask = s3c_irq_eint_mask,
|
|
|
+ .irq_unmask = s3c_irq_eint_unmask,
|
|
|
+ .irq_mask_ack = s3c_irq_eint_maskack,
|
|
|
+ .irq_ack = s3c_irq_eint_ack,
|
|
|
+ .irq_set_type = s3c_irq_eint_set_type,
|
|
|
+ .irq_set_wake = s3c_irqext_wake,
|
|
|
+};
|
|
|
+
|
|
|
+/* s3c_irq_demux_eint
|
|
|
+ *
|
|
|
+ * This function demuxes the IRQ from the group0 external interrupts,
|
|
|
+ * from IRQ_EINT(0) to IRQ_EINT(27). It is designed to be inlined into
|
|
|
+ * the specific handlers s3c_irq_demux_eintX_Y.
|
|
|
+ */
|
|
|
+static inline void s3c_irq_demux_eint(unsigned int start, unsigned int end)
|
|
|
+{
|
|
|
+ u32 status = __raw_readl(S3C64XX_EINT0PEND);
|
|
|
+ u32 mask = __raw_readl(S3C64XX_EINT0MASK);
|
|
|
+ unsigned int irq;
|
|
|
+
|
|
|
+ status &= ~mask;
|
|
|
+ status >>= start;
|
|
|
+ status &= (1 << (end - start + 1)) - 1;
|
|
|
+
|
|
|
+ for (irq = IRQ_EINT(start); irq <= IRQ_EINT(end); irq++) {
|
|
|
+ if (status & 1)
|
|
|
+ generic_handle_irq(irq);
|
|
|
+
|
|
|
+ status >>= 1;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+static void s3c_irq_demux_eint0_3(unsigned int irq, struct irq_desc *desc)
|
|
|
+{
|
|
|
+ s3c_irq_demux_eint(0, 3);
|
|
|
+}
|
|
|
+
|
|
|
+static void s3c_irq_demux_eint4_11(unsigned int irq, struct irq_desc *desc)
|
|
|
+{
|
|
|
+ s3c_irq_demux_eint(4, 11);
|
|
|
+}
|
|
|
+
|
|
|
+static void s3c_irq_demux_eint12_19(unsigned int irq, struct irq_desc *desc)
|
|
|
+{
|
|
|
+ s3c_irq_demux_eint(12, 19);
|
|
|
+}
|
|
|
+
|
|
|
+static void s3c_irq_demux_eint20_27(unsigned int irq, struct irq_desc *desc)
|
|
|
+{
|
|
|
+ s3c_irq_demux_eint(20, 27);
|
|
|
+}
|
|
|
+
|
|
|
+static int __init s3c64xx_init_irq_eint(void)
|
|
|
+{
|
|
|
+ int irq;
|
|
|
+
|
|
|
+ for (irq = IRQ_EINT(0); irq <= IRQ_EINT(27); irq++) {
|
|
|
+ irq_set_chip_and_handler(irq, &s3c_irq_eint, handle_level_irq);
|
|
|
+ irq_set_chip_data(irq, (void *)eint_irq_to_bit(irq));
|
|
|
+ set_irq_flags(irq, IRQF_VALID);
|
|
|
+ }
|
|
|
+
|
|
|
+ irq_set_chained_handler(IRQ_EINT0_3, s3c_irq_demux_eint0_3);
|
|
|
+ irq_set_chained_handler(IRQ_EINT4_11, s3c_irq_demux_eint4_11);
|
|
|
+ irq_set_chained_handler(IRQ_EINT12_19, s3c_irq_demux_eint12_19);
|
|
|
+ irq_set_chained_handler(IRQ_EINT20_27, s3c_irq_demux_eint20_27);
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+arch_initcall(s3c64xx_init_irq_eint);
|
|
|
+
|
|
|
+void s3c64xx_restart(char mode, const char *cmd)
|
|
|
+{
|
|
|
+ if (mode != 's')
|
|
|
+ arch_wdt_reset();
|
|
|
+
|
|
|
+ /* if all else fails, or mode was for soft, jump to 0 */
|
|
|
+ soft_restart(0);
|
|
|
+}
|
|
|
+
|
|
|
+void __init s3c64xx_init_late(void)
|
|
|
+{
|
|
|
+ s3c64xx_pm_late_initcall();
|
|
|
+}
|