Skip to content

Commit 02b2677

Browse files
Kevin Hilmanrjwysocki
authored andcommitted
PM / Runtime: Allow _put_sync() from interrupts-disabled context
Currently the use of pm_runtime_put_sync() is not safe from interrupts-disabled context because rpm_idle() will release the spinlock and enable interrupts for the idle callbacks. This enables interrupts during a time where interrupts were expected to be disabled, and can have strange side effects on drivers that expected interrupts to be disabled. This is not a bug since the documentation clearly states that only _put_sync_suspend() is safe in IRQ-safe mode. However, pm_runtime_put_sync() could be made safe when in IRQ-safe mode by releasing the spinlock but not re-enabling interrupts, which is what this patch aims to do. Problem was found when using some buggy drivers that set pm_runtime_irq_safe() and used _put_sync() in interrupts-disabled context. Reported-by: Colin Cross <ccross@google.com> Tested-by: Nishanth Menon <nm@ti.com> Signed-off-by: Kevin Hilman <khilman@ti.com> Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl>
1 parent fe202fd commit 02b2677

File tree

2 files changed

+13
-7
lines changed

2 files changed

+13
-7
lines changed

Documentation/power/runtime_pm.txt

Lines changed: 5 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -54,11 +54,10 @@ referred to as subsystem-level callbacks in what follows.
5454
By default, the callbacks are always invoked in process context with interrupts
5555
enabled. However, subsystems can use the pm_runtime_irq_safe() helper function
5656
to tell the PM core that a device's ->runtime_suspend() and ->runtime_resume()
57-
callbacks should be invoked in atomic context with interrupts disabled
58-
(->runtime_idle() is still invoked the default way). This implies that these
59-
callback routines must not block or sleep, but it also means that the
60-
synchronous helper functions listed at the end of Section 4 can be used within
61-
an interrupt handler or in an atomic context.
57+
callbacks should be invoked in atomic context with interrupts disabled.
58+
This implies that these callback routines must not block or sleep, but it also
59+
means that the synchronous helper functions listed at the end of Section 4 can
60+
be used within an interrupt handler or in an atomic context.
6261

6362
The subsystem-level suspend callback is _entirely_ _responsible_ for handling
6463
the suspend of the device as appropriate, which may, but need not include
@@ -483,6 +482,7 @@ pm_runtime_suspend()
483482
pm_runtime_autosuspend()
484483
pm_runtime_resume()
485484
pm_runtime_get_sync()
485+
pm_runtime_put_sync()
486486
pm_runtime_put_sync_suspend()
487487

488488
5. Runtime PM Initialization, Device Probing and Removal

drivers/base/power/runtime.c

Lines changed: 8 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -226,11 +226,17 @@ static int rpm_idle(struct device *dev, int rpmflags)
226226
callback = NULL;
227227

228228
if (callback) {
229-
spin_unlock_irq(&dev->power.lock);
229+
if (dev->power.irq_safe)
230+
spin_unlock(&dev->power.lock);
231+
else
232+
spin_unlock_irq(&dev->power.lock);
230233

231234
callback(dev);
232235

233-
spin_lock_irq(&dev->power.lock);
236+
if (dev->power.irq_safe)
237+
spin_lock(&dev->power.lock);
238+
else
239+
spin_lock_irq(&dev->power.lock);
234240
}
235241

236242
dev->power.idle_notification = false;

0 commit comments

Comments
 (0)