[PATCH v3 08/14] lib: time: hook uthread_schedule() into udelay()
Jerome Forissier
jerome.forissier at linaro.org
Tue Mar 4 16:24:56 CET 2025
Introduce a uthread scheduling loop into udelay() when CONFIG_UTHREAD
is enabled. This means that any uthread calling into udelay() may yield
to uthread and be scheduled again later. The granunlarity of the
scheduling loop is defined by CONFIG_UTHREAD_GRANULARITY. The lower the
value, the tighter the loop. The default value of 10 ms was chosen by
testing the concurrent initialization of the two USB busses of an
i.MX93 board (introduced in a later commit). The total time is as
follows:
UTHREAD_GRANULARITY (us) | "usb start" time (s)
-------------------------+----------------------
0 | 2.426
1000 | 2.448
10000 | 2.512
100000 | 2.992
300000 | 3.407
For reference, if uthread_schedule() is not called at all in udelay()
except once via schedule(), the "usb start" time is 4.377s and without
UTHREAD it is 4.591 s.
Signed-off-by: Jerome Forissier <jerome.forissier at linaro.org>
---
lib/Kconfig | 8 ++++++++
lib/time.c | 21 ++++++++++++++++++++-
2 files changed, 28 insertions(+), 1 deletion(-)
diff --git a/lib/Kconfig b/lib/Kconfig
index 8321f20e154..40964f5bb4a 100644
--- a/lib/Kconfig
+++ b/lib/Kconfig
@@ -1276,6 +1276,14 @@ config UTHREAD_STACK_SIZE
When the stack_sz argument to uthread_create() is zero then this
value is used.
+config UTHREAD_GRANULARITY
+ int "Scheduling period for uthreads inside udelay()"
+ depends on UTHREAD
+ default 10000
+ help
+ The delay in microseconds between calls to uthread_schedule() inside
+ the udelay() function.
+
endmenu
source "lib/fwu_updates/Kconfig"
diff --git a/lib/time.c b/lib/time.c
index d88edafb196..7c34424bf64 100644
--- a/lib/time.c
+++ b/lib/time.c
@@ -17,6 +17,7 @@
#include <asm/global_data.h>
#include <asm/io.h>
#include <linux/delay.h>
+#include <uthread.h>
#ifndef CFG_WD_PERIOD
# define CFG_WD_PERIOD (10 * 1000 * 1000) /* 10 seconds default */
@@ -190,6 +191,23 @@ void __weak __udelay(unsigned long usec)
/* ------------------------------------------------------------------------- */
+static void do_delay(unsigned long kv)
+{
+#if CONFIG_IS_ENABLED(UTHREAD)
+ ulong t0 = timer_get_us();
+ ulong gr = CONFIG_UTHREAD_GRANULARITY;
+
+ if (kv < CONFIG_UTHREAD_GRANULARITY)
+ gr = kv;
+ while (timer_get_us() - t0 < kv) {
+ uthread_schedule();
+ __udelay(gr);
+ }
+#else
+ __udelay(kv);
+#endif
+}
+
void udelay(unsigned long usec)
{
ulong kv;
@@ -197,7 +215,8 @@ void udelay(unsigned long usec)
do {
schedule();
kv = usec > CFG_WD_PERIOD ? CFG_WD_PERIOD : usec;
- __udelay(kv);
+ do_delay(kv);
usec -= kv;
} while(usec);
+
}
--
2.43.0
More information about the U-Boot
mailing list