Update for Renesas INTC External IRQ pin driver for v3.10
This adds support for shared interrupt lines to the Renesas INTC External IRQ pin driver which has already been queued up for v3.10 (tag renesas-intc-external-irq-for-v3.10). -----BEGIN PGP SIGNATURE----- Version: GnuPG v1.4.12 (GNU/Linux) iQIcBAABAgAGBQJRVTadAAoJENfPZGlqN0++mlQP/2tH/AimAHzPdYBpJWHneglu da5dpoqKNHtrPzrJMMJQ5VT5lupjpnX1XhNpbs5Y/GG5lA0PMJtWgQknroxEoIp+ jxYHyzMRYvJitghvwbODQzbVCbcdtLVZFHVDKaBonImEX98Kun6k/61g7L74Xdnq AyAIDKFso6sSjfnkJuKljhKQbK9n0L0Bl1L5WUQeWaKt2lF6rHsAGHqXrRvcvSN9 /V9HPCIQD+4+Efe7VYa/cz8BDsphVp1LQY4AJ9F+fjhELu6izJ7Y//MpXZSxD6/q dL/T0+fXuCrW8kwylKMyJOFAo6kMWhpOhBmHg9dvBvD5UVKvkJU1tQqpTSiT1hqm PPgmpoVCLE1p5d4rml1U4C//tj56IxDPVbr2ld3DnqykSMFqlbin64JBcl5/2tkp SRQA37Qu7oypnNYVqMrCK6KWbhJNGrrsomJqukiJda6PO+8pb6hR3az9cTk4n4+D H6Gbvb6Qf3jj0TLOjwsZ5q4yAB2xQ1sImP3QHKxi7aj0/1jvNwut4Qw2ucxKnujq EHK2rwXNzEZJ3pkcc/5hnLoPI4nCirBXMkkb0ZgrjeaPUSIISjcqOZDPyRFU1JJQ qLAYlq9fUJd6ZgMIV2G83/3cMqAaiVT+G/zIvM+BXMb0ExmW/OUYoR2D2txNBdzs CcdH0m+50ipkNLqndiin =nYUV -----END PGP SIGNATURE----- Merge tag 'renesas-intc-external-irq2-for-v3.10' of git://git.kernel.org/pub/scm/linux/kernel/git/horms/renesas into next/drivers From Simon Horman <horms+renesas@verge.net.au>: Update for Renesas INTC External IRQ pin driver for v3.10 This adds support for shared interrupt lines to the Renesas INTC External IRQ pin driver which has already been queued up for v3.10 (tag renesas-intc-external-irq-for-v3.10). * tag 'renesas-intc-external-irq2-for-v3.10' of git://git.kernel.org/pub/scm/linux/kernel/git/horms/renesas: irqchip: intc-irqpin: Add support for shared interrupt lines Signed-off-by: Arnd Bergmann <arnd@arndb.de>
This commit is contained in:
Коммит
4680ebc2c9
|
@ -74,6 +74,8 @@ struct intc_irqpin_priv {
|
||||||
struct platform_device *pdev;
|
struct platform_device *pdev;
|
||||||
struct irq_chip irq_chip;
|
struct irq_chip irq_chip;
|
||||||
struct irq_domain *irq_domain;
|
struct irq_domain *irq_domain;
|
||||||
|
bool shared_irqs;
|
||||||
|
u8 shared_irq_mask;
|
||||||
};
|
};
|
||||||
|
|
||||||
static unsigned long intc_irqpin_read32(void __iomem *iomem)
|
static unsigned long intc_irqpin_read32(void __iomem *iomem)
|
||||||
|
@ -193,6 +195,28 @@ static void intc_irqpin_irq_disable(struct irq_data *d)
|
||||||
intc_irqpin_irq_write_hwirq(p, INTC_IRQPIN_REG_MASK, hw_irq);
|
intc_irqpin_irq_write_hwirq(p, INTC_IRQPIN_REG_MASK, hw_irq);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void intc_irqpin_shared_irq_enable(struct irq_data *d)
|
||||||
|
{
|
||||||
|
struct intc_irqpin_priv *p = irq_data_get_irq_chip_data(d);
|
||||||
|
int hw_irq = irqd_to_hwirq(d);
|
||||||
|
|
||||||
|
intc_irqpin_dbg(&p->irq[hw_irq], "shared enable");
|
||||||
|
intc_irqpin_irq_write_hwirq(p, INTC_IRQPIN_REG_CLEAR, hw_irq);
|
||||||
|
|
||||||
|
p->shared_irq_mask &= ~BIT(hw_irq);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void intc_irqpin_shared_irq_disable(struct irq_data *d)
|
||||||
|
{
|
||||||
|
struct intc_irqpin_priv *p = irq_data_get_irq_chip_data(d);
|
||||||
|
int hw_irq = irqd_to_hwirq(d);
|
||||||
|
|
||||||
|
intc_irqpin_dbg(&p->irq[hw_irq], "shared disable");
|
||||||
|
intc_irqpin_irq_write_hwirq(p, INTC_IRQPIN_REG_MASK, hw_irq);
|
||||||
|
|
||||||
|
p->shared_irq_mask |= BIT(hw_irq);
|
||||||
|
}
|
||||||
|
|
||||||
static void intc_irqpin_irq_enable_force(struct irq_data *d)
|
static void intc_irqpin_irq_enable_force(struct irq_data *d)
|
||||||
{
|
{
|
||||||
struct intc_irqpin_priv *p = irq_data_get_irq_chip_data(d);
|
struct intc_irqpin_priv *p = irq_data_get_irq_chip_data(d);
|
||||||
|
@ -261,6 +285,25 @@ static irqreturn_t intc_irqpin_irq_handler(int irq, void *dev_id)
|
||||||
return IRQ_NONE;
|
return IRQ_NONE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static irqreturn_t intc_irqpin_shared_irq_handler(int irq, void *dev_id)
|
||||||
|
{
|
||||||
|
struct intc_irqpin_priv *p = dev_id;
|
||||||
|
unsigned int reg_source = intc_irqpin_read(p, INTC_IRQPIN_REG_SOURCE);
|
||||||
|
irqreturn_t status = IRQ_NONE;
|
||||||
|
int k;
|
||||||
|
|
||||||
|
for (k = 0; k < 8; k++) {
|
||||||
|
if (reg_source & BIT(7 - k)) {
|
||||||
|
if (BIT(k) & p->shared_irq_mask)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
status |= intc_irqpin_irq_handler(irq, &p->irq[k]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
static int intc_irqpin_irq_domain_map(struct irq_domain *h, unsigned int virq,
|
static int intc_irqpin_irq_domain_map(struct irq_domain *h, unsigned int virq,
|
||||||
irq_hw_number_t hw)
|
irq_hw_number_t hw)
|
||||||
{
|
{
|
||||||
|
@ -292,6 +335,7 @@ static int intc_irqpin_probe(struct platform_device *pdev)
|
||||||
void (*enable_fn)(struct irq_data *d);
|
void (*enable_fn)(struct irq_data *d);
|
||||||
void (*disable_fn)(struct irq_data *d);
|
void (*disable_fn)(struct irq_data *d);
|
||||||
const char *name = dev_name(&pdev->dev);
|
const char *name = dev_name(&pdev->dev);
|
||||||
|
int ref_irq;
|
||||||
int ret;
|
int ret;
|
||||||
int k;
|
int k;
|
||||||
|
|
||||||
|
@ -372,13 +416,29 @@ static int intc_irqpin_probe(struct platform_device *pdev)
|
||||||
for (k = 0; k < p->number_of_irqs; k++)
|
for (k = 0; k < p->number_of_irqs; k++)
|
||||||
intc_irqpin_mask_unmask_prio(p, k, 1);
|
intc_irqpin_mask_unmask_prio(p, k, 1);
|
||||||
|
|
||||||
|
/* clear all pending interrupts */
|
||||||
|
intc_irqpin_write(p, INTC_IRQPIN_REG_SOURCE, 0x0);
|
||||||
|
|
||||||
|
/* scan for shared interrupt lines */
|
||||||
|
ref_irq = p->irq[0].requested_irq;
|
||||||
|
p->shared_irqs = true;
|
||||||
|
for (k = 1; k < p->number_of_irqs; k++) {
|
||||||
|
if (ref_irq != p->irq[k].requested_irq) {
|
||||||
|
p->shared_irqs = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* use more severe masking method if requested */
|
/* use more severe masking method if requested */
|
||||||
if (p->config.control_parent) {
|
if (p->config.control_parent) {
|
||||||
enable_fn = intc_irqpin_irq_enable_force;
|
enable_fn = intc_irqpin_irq_enable_force;
|
||||||
disable_fn = intc_irqpin_irq_disable_force;
|
disable_fn = intc_irqpin_irq_disable_force;
|
||||||
} else {
|
} else if (!p->shared_irqs) {
|
||||||
enable_fn = intc_irqpin_irq_enable;
|
enable_fn = intc_irqpin_irq_enable;
|
||||||
disable_fn = intc_irqpin_irq_disable;
|
disable_fn = intc_irqpin_irq_disable;
|
||||||
|
} else {
|
||||||
|
enable_fn = intc_irqpin_shared_irq_enable;
|
||||||
|
disable_fn = intc_irqpin_shared_irq_disable;
|
||||||
}
|
}
|
||||||
|
|
||||||
irq_chip = &p->irq_chip;
|
irq_chip = &p->irq_chip;
|
||||||
|
@ -400,18 +460,34 @@ static int intc_irqpin_probe(struct platform_device *pdev)
|
||||||
goto err0;
|
goto err0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* request and set priority on interrupts one by one */
|
if (p->shared_irqs) {
|
||||||
for (k = 0; k < p->number_of_irqs; k++) {
|
/* request one shared interrupt */
|
||||||
if (devm_request_irq(&pdev->dev, p->irq[k].requested_irq,
|
if (devm_request_irq(&pdev->dev, p->irq[0].requested_irq,
|
||||||
intc_irqpin_irq_handler,
|
intc_irqpin_shared_irq_handler,
|
||||||
0, name, &p->irq[k])) {
|
IRQF_SHARED, name, p)) {
|
||||||
dev_err(&pdev->dev, "failed to request low IRQ\n");
|
dev_err(&pdev->dev, "failed to request low IRQ\n");
|
||||||
ret = -ENOENT;
|
ret = -ENOENT;
|
||||||
goto err1;
|
goto err1;
|
||||||
}
|
}
|
||||||
intc_irqpin_mask_unmask_prio(p, k, 0);
|
} else {
|
||||||
|
/* request interrupts one by one */
|
||||||
|
for (k = 0; k < p->number_of_irqs; k++) {
|
||||||
|
if (devm_request_irq(&pdev->dev,
|
||||||
|
p->irq[k].requested_irq,
|
||||||
|
intc_irqpin_irq_handler,
|
||||||
|
0, name, &p->irq[k])) {
|
||||||
|
dev_err(&pdev->dev,
|
||||||
|
"failed to request low IRQ\n");
|
||||||
|
ret = -ENOENT;
|
||||||
|
goto err1;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* unmask all interrupts on prio level */
|
||||||
|
for (k = 0; k < p->number_of_irqs; k++)
|
||||||
|
intc_irqpin_mask_unmask_prio(p, k, 0);
|
||||||
|
|
||||||
dev_info(&pdev->dev, "driving %d irqs\n", p->number_of_irqs);
|
dev_info(&pdev->dev, "driving %d irqs\n", p->number_of_irqs);
|
||||||
|
|
||||||
/* warn in case of mismatch if irq base is specified */
|
/* warn in case of mismatch if irq base is specified */
|
||||||
|
|
Загрузка…
Ссылка в новой задаче