mirror of
https://github.com/edk2-porting/linux-next.git
synced 2025-01-14 00:24:15 +08:00
power: gemini-poweroff: Avoid spurious poweroff
On the D-Link DIR-685 we get spurious poweroff from
infrared. Since that block (CIR) doesn't even have a
driver this can be safely ignored, we can revisit this
code once we have a device supporting CIR.
On the D-Link DNS-313 we get spurious poweroff from
the power button. This appears to be an initialization
issue: we need to enable the block (start the state
machine) before we clear any dangling IRQ.
This patch fixes both issues.
Fixes: f7a388d6cd
("power: reset: Add a driver for the Gemini poweroff")
Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
Signed-off-by: Sebastian Reichel <sebastian.reichel@collabora.co.uk>
This commit is contained in:
parent
dde5953f05
commit
4a9be94055
@ -47,8 +47,12 @@ static irqreturn_t gemini_powerbutton_interrupt(int irq, void *data)
|
|||||||
val &= 0x70U;
|
val &= 0x70U;
|
||||||
switch (val) {
|
switch (val) {
|
||||||
case GEMINI_STAT_CIR:
|
case GEMINI_STAT_CIR:
|
||||||
dev_info(gpw->dev, "infrared poweroff\n");
|
/*
|
||||||
orderly_poweroff(true);
|
* We do not yet have a driver for the infrared
|
||||||
|
* controller so it can cause spurious poweroff
|
||||||
|
* events. Ignore those for now.
|
||||||
|
*/
|
||||||
|
dev_info(gpw->dev, "infrared poweroff - ignored\n");
|
||||||
break;
|
break;
|
||||||
case GEMINI_STAT_RTC:
|
case GEMINI_STAT_RTC:
|
||||||
dev_info(gpw->dev, "RTC poweroff\n");
|
dev_info(gpw->dev, "RTC poweroff\n");
|
||||||
@ -116,7 +120,17 @@ static int gemini_poweroff_probe(struct platform_device *pdev)
|
|||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Clear the power management IRQ */
|
/*
|
||||||
|
* Enable the power controller. This is crucial on Gemini
|
||||||
|
* systems: if this is not done, pressing the power button
|
||||||
|
* will result in unconditional poweroff without any warning.
|
||||||
|
* This makes the kernel handle the poweroff.
|
||||||
|
*/
|
||||||
|
val = readl(gpw->base + GEMINI_PWC_CTRLREG);
|
||||||
|
val |= GEMINI_CTRL_ENABLE;
|
||||||
|
writel(val, gpw->base + GEMINI_PWC_CTRLREG);
|
||||||
|
|
||||||
|
/* Now that the state machine is active, clear the IRQ */
|
||||||
val = readl(gpw->base + GEMINI_PWC_CTRLREG);
|
val = readl(gpw->base + GEMINI_PWC_CTRLREG);
|
||||||
val |= GEMINI_CTRL_IRQ_CLR;
|
val |= GEMINI_CTRL_IRQ_CLR;
|
||||||
writel(val, gpw->base + GEMINI_PWC_CTRLREG);
|
writel(val, gpw->base + GEMINI_PWC_CTRLREG);
|
||||||
@ -129,16 +143,6 @@ static int gemini_poweroff_probe(struct platform_device *pdev)
|
|||||||
pm_power_off = gemini_poweroff;
|
pm_power_off = gemini_poweroff;
|
||||||
gpw_poweroff = gpw;
|
gpw_poweroff = gpw;
|
||||||
|
|
||||||
/*
|
|
||||||
* Enable the power controller. This is crucial on Gemini
|
|
||||||
* systems: if this is not done, pressing the power button
|
|
||||||
* will result in unconditional poweroff without any warning.
|
|
||||||
* This makes the kernel handle the poweroff.
|
|
||||||
*/
|
|
||||||
val = readl(gpw->base + GEMINI_PWC_CTRLREG);
|
|
||||||
val |= GEMINI_CTRL_ENABLE;
|
|
||||||
writel(val, gpw->base + GEMINI_PWC_CTRLREG);
|
|
||||||
|
|
||||||
dev_info(dev, "Gemini poweroff driver registered\n");
|
dev_info(dev, "Gemini poweroff driver registered\n");
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
Loading…
Reference in New Issue
Block a user