Commit a2c22510 authored by Sudeep Holla's avatar Sudeep Holla Committed by Jason Cooper

irqchip: gic-v3: Refactor gic_enable_redist to support both enabling and disabling

Currently gic_enable_redist configures the redistributors to never
assert WakeRequest signal. However when powering down the processors
with wake-up enabled(i.e suspend), we need to configure it to assert
that signal.

This patch extends gic_enable_redist so that the redistributor can be
configure to assert WakeRequest and hold interrupts as pending. This is
useful in suspending the processors.

This patch also adds check to make sure GICR_WAKER is accessible when
configuring it.

Cc: Lorenzo Pieralisi <lorenzo.pieralisi@arm.com>
Signed-off-by: default avatarSudeep Holla <sudeep.holla@arm.com>
[maz: removed reference to GICD_CTLR.DS and added read-back of
      GICR_WAKER to check that it is not RAZ/WI]
Signed-off-by: default avatarMarc Zyngier <marc.zyngier@arm.com>
Acked-by: default avatarMarc Zyngier <marc.zyngier@arm.com>
Link: https://lkml.kernel.org/r/1409065415-20176-2-git-send-email-sudeep.holla@arm.comSigned-off-by: default avatarJason Cooper <jason@lakedaemon.net>
parent 32289506
...@@ -155,7 +155,7 @@ static void gic_enable_sre(void) ...@@ -155,7 +155,7 @@ static void gic_enable_sre(void)
pr_err("GIC: unable to set SRE (disabled at EL2), panic ahead\n"); pr_err("GIC: unable to set SRE (disabled at EL2), panic ahead\n");
} }
static void gic_enable_redist(void) static void gic_enable_redist(bool enable)
{ {
void __iomem *rbase; void __iomem *rbase;
u32 count = 1000000; /* 1s! */ u32 count = 1000000; /* 1s! */
...@@ -163,20 +163,30 @@ static void gic_enable_redist(void) ...@@ -163,20 +163,30 @@ static void gic_enable_redist(void)
rbase = gic_data_rdist_rd_base(); rbase = gic_data_rdist_rd_base();
/* Wake up this CPU redistributor */
val = readl_relaxed(rbase + GICR_WAKER); val = readl_relaxed(rbase + GICR_WAKER);
if (enable)
/* Wake up this CPU redistributor */
val &= ~GICR_WAKER_ProcessorSleep; val &= ~GICR_WAKER_ProcessorSleep;
else
val |= GICR_WAKER_ProcessorSleep;
writel_relaxed(val, rbase + GICR_WAKER); writel_relaxed(val, rbase + GICR_WAKER);
while (readl_relaxed(rbase + GICR_WAKER) & GICR_WAKER_ChildrenAsleep) { if (!enable) { /* Check that GICR_WAKER is writeable */
count--; val = readl_relaxed(rbase + GICR_WAKER);
if (!count) { if (!(val & GICR_WAKER_ProcessorSleep))
pr_err_ratelimited("redist didn't wake up...\n"); return; /* No PM support in this redistributor */
return;
} }
while (count--) {
val = readl_relaxed(rbase + GICR_WAKER);
if (enable ^ (val & GICR_WAKER_ChildrenAsleep))
break;
cpu_relax(); cpu_relax();
udelay(1); udelay(1);
}; };
if (!count)
pr_err_ratelimited("redistributor failed to %s...\n",
enable ? "wakeup" : "sleep");
} }
/* /*
...@@ -381,7 +391,7 @@ static void gic_cpu_init(void) ...@@ -381,7 +391,7 @@ static void gic_cpu_init(void)
if (gic_populate_rdist()) if (gic_populate_rdist())
return; return;
gic_enable_redist(); gic_enable_redist(true);
rbase = gic_data_rdist_sgi_base(); rbase = gic_data_rdist_sgi_base();
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment