summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorEvgenii Shatokhin <e.shatokhin@yadro.com>2023-03-06 00:22:47 +0300
committerAnup Patel <anup@brainfault.org>2023-03-10 11:43:40 +0300
commitc6a092cd80112529cb2e92e180767ff5341b22a3 (patch)
tree3fe7ea041e7bc73f8f8e0df4f7648ba8f7a1cbbf
parente8e9ed3790feac05ec6fd5c279ec80b05293d67b (diff)
downloadopensbi-JH7110_VisionFive2_devel.tar.xz
Since commit 50d4fde1c5a4 ("lib: Remove redundant sbi_platform_ipi_clear() calls"), the IPI sent from the boot hart in wake_coldboot_harts() is not cleared in the secondary harts until they reach sbi_ipi_init(). However, sbi_hsm_init() and sbi_hsm_hart_wait() are called earlier, so a secondary hart might enter sbi_hsm_hart_wait() with an already pending IPI. sbi_hsm_hart_wait() makes sure the hart leaves the loop only when it is actually ready, so a pending unrelated IPI should not cause safety issues. However, it might be inefficient on certain hardware, because it prevents "wfi" from stalling the hart even if the hardware supports this, making the hart needlessly spin in a "busy-wait" loop. This behaviour can be observed, for example, in a QEMU VM (QEMU 7.2.0) with "-machine virt" running a Linux guest. Inserting delays in sbi_hsm_hart_start() allows reproducing the issue more reliably. The comment in wait_for_coldboot() suggests that the initial IPI is needed in the warm resume path, so let us clear it before init_warm_startup() only. To do this, sbi_ipi_raw_clear() was created similar to sbi_ipi_raw_send(). Signed-off-by: Evgenii Shatokhin <e.shatokhin@yadro.com> Reviewed-by: Anup Patel <anup@brainfault.org>
-rw-r--r--include/sbi/sbi_ipi.h2
-rw-r--r--lib/sbi/sbi_init.c6
-rw-r--r--lib/sbi/sbi_ipi.c6
3 files changed, 12 insertions, 2 deletions
diff --git a/include/sbi/sbi_ipi.h b/include/sbi/sbi_ipi.h
index f6ac807..f384e74 100644
--- a/include/sbi/sbi_ipi.h
+++ b/include/sbi/sbi_ipi.h
@@ -77,6 +77,8 @@ void sbi_ipi_process(void);
int sbi_ipi_raw_send(u32 target_hart);
+void sbi_ipi_raw_clear(u32 target_hart);
+
const struct sbi_ipi_device *sbi_ipi_get_device(void);
void sbi_ipi_set_device(const struct sbi_ipi_device *dev);
diff --git a/lib/sbi/sbi_init.c b/lib/sbi/sbi_init.c
index dcca2c8..ffa214c 100644
--- a/lib/sbi/sbi_init.c
+++ b/lib/sbi/sbi_init.c
@@ -442,10 +442,12 @@ static void __noreturn init_warmboot(struct sbi_scratch *scratch, u32 hartid)
if (hstate < 0)
sbi_hart_hang();
- if (hstate == SBI_HSM_STATE_SUSPENDED)
+ if (hstate == SBI_HSM_STATE_SUSPENDED) {
init_warm_resume(scratch, hartid);
- else
+ } else {
+ sbi_ipi_raw_clear(hartid);
init_warm_startup(scratch, hartid);
+ }
}
static atomic_t coldboot_lottery = ATOMIC_INITIALIZER(0);
diff --git a/lib/sbi/sbi_ipi.c b/lib/sbi/sbi_ipi.c
index 1bcc2e4..b9f6205 100644
--- a/lib/sbi/sbi_ipi.c
+++ b/lib/sbi/sbi_ipi.c
@@ -217,6 +217,12 @@ int sbi_ipi_raw_send(u32 target_hart)
return 0;
}
+void sbi_ipi_raw_clear(u32 target_hart)
+{
+ if (ipi_dev && ipi_dev->ipi_clear)
+ ipi_dev->ipi_clear(target_hart);
+}
+
const struct sbi_ipi_device *sbi_ipi_get_device(void)
{
return ipi_dev;