root/drivers/irqchip/irq-vt8500.c
// SPDX-License-Identifier: GPL-2.0-or-later
/*
 *  arch/arm/mach-vt8500/irq.c
 *
 *  Copyright (C) 2012 Tony Prisk <linux@prisktech.co.nz>
 *  Copyright (C) 2010 Alexey Charkov <alchark@gmail.com>
 */

/*
 * This file is copied and modified from the original irq.c provided by
 * Alexey Charkov. Minor changes have been made for Device Tree Support.
 */

#include <linux/slab.h>
#include <linux/io.h>
#include <linux/irq.h>
#include <linux/irqchip.h>
#include <linux/irqchip/chained_irq.h>
#include <linux/irqdomain.h>
#include <linux/interrupt.h>
#include <linux/bitops.h>

#include <linux/of.h>
#include <linux/of_irq.h>
#include <linux/of_address.h>

#include <asm/irq.h>
#include <asm/exception.h>
#include <asm/mach/irq.h>

#define VT8500_ICPC_IRQ         0x20
#define VT8500_ICPC_FIQ         0x24
#define VT8500_ICDC             0x40            /* Destination Control 64*u32 */
#define VT8500_ICIS             0x80            /* Interrupt status, 16*u32 */

/* ICPC */
#define ICPC_MASK               0x3F
#define ICPC_ROTATE             BIT(6)

/* IC_DCTR */
#define ICDC_IRQ                0x00
#define ICDC_FIQ                0x01
#define ICDC_DSS0               0x02
#define ICDC_DSS1               0x03
#define ICDC_DSS2               0x04
#define ICDC_DSS3               0x05
#define ICDC_DSS4               0x06
#define ICDC_DSS5               0x07

#define VT8500_INT_DISABLE      0
#define VT8500_INT_ENABLE       BIT(3)

#define VT8500_TRIGGER_HIGH     0
#define VT8500_TRIGGER_RISING   BIT(5)
#define VT8500_TRIGGER_FALLING  BIT(6)
#define VT8500_EDGE             ( VT8500_TRIGGER_RISING \
                                | VT8500_TRIGGER_FALLING)

/* vt8500 has 1 intc, wm8505 and wm8650 have 2 */
#define VT8500_INTC_MAX         2

struct vt8500_irq_data {
        void __iomem            *base;          /* IO Memory base address */
        struct irq_domain       *domain;        /* Domain for this controller */
};

/* Primary interrupt controller data */
static struct vt8500_irq_data *primary_intc;

static void vt8500_irq_ack(struct irq_data *d)
{
        struct vt8500_irq_data *priv = d->domain->host_data;
        void __iomem *base = priv->base;
        void __iomem *stat_reg = base + VT8500_ICIS + (d->hwirq < 32 ? 0 : 4);
        u32 status = (1 << (d->hwirq & 0x1f));

        writel(status, stat_reg);
}

static void vt8500_irq_mask(struct irq_data *d)
{
        struct vt8500_irq_data *priv = d->domain->host_data;
        void __iomem *base = priv->base;
        u8 dctr;

        dctr = readb(base + VT8500_ICDC + d->hwirq);
        dctr &= ~VT8500_INT_ENABLE;
        writeb(dctr, base + VT8500_ICDC + d->hwirq);
}

static void vt8500_irq_unmask(struct irq_data *d)
{
        struct vt8500_irq_data *priv = d->domain->host_data;
        void __iomem *base = priv->base;
        u8 dctr;

        dctr = readb(base + VT8500_ICDC + d->hwirq);
        dctr |= VT8500_INT_ENABLE;
        writeb(dctr, base + VT8500_ICDC + d->hwirq);
}

static int vt8500_irq_set_type(struct irq_data *d, unsigned int flow_type)
{
        struct vt8500_irq_data *priv = d->domain->host_data;
        void __iomem *base = priv->base;
        u8 dctr;

        dctr = readb(base + VT8500_ICDC + d->hwirq);
        dctr &= ~VT8500_EDGE;

        switch (flow_type) {
        case IRQF_TRIGGER_LOW:
                return -EINVAL;
        case IRQF_TRIGGER_HIGH:
                dctr |= VT8500_TRIGGER_HIGH;
                irq_set_handler_locked(d, handle_level_irq);
                break;
        case IRQF_TRIGGER_FALLING:
                dctr |= VT8500_TRIGGER_FALLING;
                irq_set_handler_locked(d, handle_edge_irq);
                break;
        case IRQF_TRIGGER_RISING:
                dctr |= VT8500_TRIGGER_RISING;
                irq_set_handler_locked(d, handle_edge_irq);
                break;
        }
        writeb(dctr, base + VT8500_ICDC + d->hwirq);

        return 0;
}

static struct irq_chip vt8500_irq_chip = {
        .name           = "vt8500",
        .irq_ack        = vt8500_irq_ack,
        .irq_mask       = vt8500_irq_mask,
        .irq_unmask     = vt8500_irq_unmask,
        .irq_set_type   = vt8500_irq_set_type,
};

static void __init vt8500_init_irq_hw(void __iomem *base)
{
        u32 i;

        /* Enable rotating priority for IRQ */
        writel(ICPC_ROTATE, base + VT8500_ICPC_IRQ);
        writel(0x00, base + VT8500_ICPC_FIQ);

        /* Disable all interrupts and route them to IRQ */
        for (i = 0; i < 64; i++)
                writeb(VT8500_INT_DISABLE | ICDC_IRQ, base + VT8500_ICDC + i);
}

static int vt8500_irq_map(struct irq_domain *h, unsigned int virq,
                                                        irq_hw_number_t hw)
{
        irq_set_chip_and_handler(virq, &vt8500_irq_chip, handle_level_irq);

        return 0;
}

static const struct irq_domain_ops vt8500_irq_domain_ops = {
        .map = vt8500_irq_map,
        .xlate = irq_domain_xlate_onecell,
};

static inline void vt8500_handle_irq_common(struct vt8500_irq_data *intc)
{
        unsigned long irqnr = readl_relaxed(intc->base) & 0x3F;
        unsigned long stat;

        /*
         * Highest Priority register default = 63, so check that this
         * is a real interrupt by checking the status register
         */
        if (irqnr == 63) {
                stat = readl_relaxed(intc->base + VT8500_ICIS + 4);
                if (!(stat & BIT(31)))
                        return;
        }

        generic_handle_domain_irq(intc->domain, irqnr);
}

static void __exception_irq_entry vt8500_handle_irq(struct pt_regs *regs)
{
        vt8500_handle_irq_common(primary_intc);
}

static void vt8500_handle_irq_chained(struct irq_desc *desc)
{
        struct irq_domain *d = irq_desc_get_handler_data(desc);
        struct irq_chip *chip = irq_desc_get_chip(desc);
        struct vt8500_irq_data *intc = d->host_data;

        chained_irq_enter(chip, desc);
        vt8500_handle_irq_common(intc);
        chained_irq_exit(chip, desc);
}

static int __init vt8500_irq_init(struct device_node *node,
                                  struct device_node *parent)
{
        struct vt8500_irq_data *intc;
        int irq, i, ret = 0;

        intc = kzalloc_obj(*intc);
        if (!intc)
                return -ENOMEM;

        intc->base = of_iomap(node, 0);
        if (!intc->base) {
                pr_err("%s: Unable to map IO memory\n", __func__);
                ret = -ENOMEM;
                goto err_free;
        }

        intc->domain = irq_domain_create_linear(of_fwnode_handle(node), 64,
                                                &vt8500_irq_domain_ops, intc);
        if (!intc->domain) {
                pr_err("%s: Unable to add irq domain!\n", __func__);
                ret = -ENOMEM;
                goto err_unmap;
        }

        vt8500_init_irq_hw(intc->base);

        pr_info("vt8500-irq: Added interrupt controller\n");

        /* check if this is a chained controller */
        if (of_irq_count(node) != 0) {
                for (i = 0; i < of_irq_count(node); i++) {
                        irq = irq_of_parse_and_map(node, i);
                        irq_set_chained_handler_and_data(irq, vt8500_handle_irq_chained,
                                                         intc);
                }

                pr_info("vt8500-irq: Enabled slave->parent interrupts\n");
        } else {
                primary_intc = intc;
                set_handle_irq(vt8500_handle_irq);
        }
        return 0;

err_unmap:
        iounmap(intc->base);
err_free:
        kfree(intc);
        return ret;
}

IRQCHIP_DECLARE(vt8500_irq, "via,vt8500-intc", vt8500_irq_init);