]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
pinctrl: amd: Don't save/restore interrupt status and wake status bits
authorBasavaraj Natikar <Basavaraj.Natikar@amd.com>
Mon, 13 Jun 2022 06:41:26 +0000 (12:11 +0530)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Wed, 31 Aug 2022 15:15:14 +0000 (17:15 +0200)
commit b8c824a869f220c6b46df724f85794349bafbf23 upstream.

Saving/restoring interrupt and wake status bits across suspend can
cause the suspend to fail if an IRQ is serviced across the
suspend cycle.

Signed-off-by: Mario Limonciello <mario.limonciello@amd.com>
Signed-off-by: Basavaraj Natikar <Basavaraj.Natikar@amd.com>
Fixes: 79d2c8bede2c ("pinctrl/amd: save pin registers over suspend/resume")
Link: https://lore.kernel.org/r/20220613064127.220416-3-Basavaraj.Natikar@amd.com
Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/pinctrl/pinctrl-amd.c

index e20bcc835d6a84f627084f3e61410910f0509b8c..82b658a3c220a84bc0d1b363d90bf122c8475281 100644 (file)
@@ -815,6 +815,7 @@ static int amd_gpio_suspend(struct device *dev)
 {
        struct amd_gpio *gpio_dev = dev_get_drvdata(dev);
        struct pinctrl_desc *desc = gpio_dev->pctrl->desc;
+       unsigned long flags;
        int i;
 
        for (i = 0; i < desc->npins; i++) {
@@ -823,7 +824,9 @@ static int amd_gpio_suspend(struct device *dev)
                if (!amd_gpio_should_save(gpio_dev, pin))
                        continue;
 
-               gpio_dev->saved_regs[i] = readl(gpio_dev->base + pin*4);
+               raw_spin_lock_irqsave(&gpio_dev->lock, flags);
+               gpio_dev->saved_regs[i] = readl(gpio_dev->base + pin * 4) & ~PIN_IRQ_PENDING;
+               raw_spin_unlock_irqrestore(&gpio_dev->lock, flags);
        }
 
        return 0;
@@ -833,6 +836,7 @@ static int amd_gpio_resume(struct device *dev)
 {
        struct amd_gpio *gpio_dev = dev_get_drvdata(dev);
        struct pinctrl_desc *desc = gpio_dev->pctrl->desc;
+       unsigned long flags;
        int i;
 
        for (i = 0; i < desc->npins; i++) {
@@ -841,7 +845,10 @@ static int amd_gpio_resume(struct device *dev)
                if (!amd_gpio_should_save(gpio_dev, pin))
                        continue;
 
-               writel(gpio_dev->saved_regs[i], gpio_dev->base + pin*4);
+               raw_spin_lock_irqsave(&gpio_dev->lock, flags);
+               gpio_dev->saved_regs[i] |= readl(gpio_dev->base + pin * 4) & PIN_IRQ_PENDING;
+               writel(gpio_dev->saved_regs[i], gpio_dev->base + pin * 4);
+               raw_spin_unlock_irqrestore(&gpio_dev->lock, flags);
        }
 
        return 0;