| From 24b569214a16c94e5d9e6e469109b73ce5daedb9 Mon Sep 17 00:00:00 2001 |
| From: Sasha Levin <sashal@kernel.org> |
| Date: Wed, 17 May 2023 11:30:29 +0800 |
| Subject: i3c: master: svc: fix cpu schedule in spin lock |
| |
| From: Clark Wang <xiaoning.wang@nxp.com> |
| |
| [ Upstream commit 33beadb3b1ab74e69db2c49d9663f3a93a273943 ] |
| |
| pm_runtime_resume_and_get() may call sleep(). It cannot be used in |
| svc_i3c_master_start_xfer_locked(), because it is in a spin lock. |
| |
| Move the pm runtime operations to svc_i3c_master_enqueue_xfer(). |
| |
| Signed-off-by: Clark Wang <xiaoning.wang@nxp.com> |
| Fixes: 05be23ef78f7 ("i3c: master: svc: add runtime pm support") |
| Reviewed-by: Miquel Raynal <miquel.raynal@bootlin.com> |
| Link: https://lore.kernel.org/r/20230517033030.3068085-2-xiaoning.wang@nxp.com |
| Signed-off-by: Alexandre Belloni <alexandre.belloni@bootlin.com> |
| Signed-off-by: Sasha Levin <sashal@kernel.org> |
| --- |
| drivers/i3c/master/svc-i3c-master.c | 19 ++++++++++--------- |
| 1 file changed, 10 insertions(+), 9 deletions(-) |
| |
| diff --git a/drivers/i3c/master/svc-i3c-master.c b/drivers/i3c/master/svc-i3c-master.c |
| index e3f454123805e..79b08942a925d 100644 |
| --- a/drivers/i3c/master/svc-i3c-master.c |
| +++ b/drivers/i3c/master/svc-i3c-master.c |
| @@ -1090,12 +1090,6 @@ static void svc_i3c_master_start_xfer_locked(struct svc_i3c_master *master) |
| if (!xfer) |
| return; |
| |
| - ret = pm_runtime_resume_and_get(master->dev); |
| - if (ret < 0) { |
| - dev_err(master->dev, "<%s> Cannot get runtime PM.\n", __func__); |
| - return; |
| - } |
| - |
| svc_i3c_master_clear_merrwarn(master); |
| svc_i3c_master_flush_fifo(master); |
| |
| @@ -1110,9 +1104,6 @@ static void svc_i3c_master_start_xfer_locked(struct svc_i3c_master *master) |
| break; |
| } |
| |
| - pm_runtime_mark_last_busy(master->dev); |
| - pm_runtime_put_autosuspend(master->dev); |
| - |
| xfer->ret = ret; |
| complete(&xfer->comp); |
| |
| @@ -1133,6 +1124,13 @@ static void svc_i3c_master_enqueue_xfer(struct svc_i3c_master *master, |
| struct svc_i3c_xfer *xfer) |
| { |
| unsigned long flags; |
| + int ret; |
| + |
| + ret = pm_runtime_resume_and_get(master->dev); |
| + if (ret < 0) { |
| + dev_err(master->dev, "<%s> Cannot get runtime PM.\n", __func__); |
| + return; |
| + } |
| |
| init_completion(&xfer->comp); |
| spin_lock_irqsave(&master->xferqueue.lock, flags); |
| @@ -1143,6 +1141,9 @@ static void svc_i3c_master_enqueue_xfer(struct svc_i3c_master *master, |
| svc_i3c_master_start_xfer_locked(master); |
| } |
| spin_unlock_irqrestore(&master->xferqueue.lock, flags); |
| + |
| + pm_runtime_mark_last_busy(master->dev); |
| + pm_runtime_put_autosuspend(master->dev); |
| } |
| |
| static bool |
| -- |
| 2.39.2 |
| |