blob: aa736f6b689788f86d5ff8f85fb0e652072343e5 [file] [log] [blame]
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001/*
Andy Ross1acd8c22018-05-03 14:51:49 -07002 * Copyright (c) 2018 Intel Corporation
Benjamin Walsh456c6da2016-09-02 18:55:39 -04003 *
David B. Kinderac74d8b2017-01-18 17:01:01 -08004 * SPDX-License-Identifier: Apache-2.0
Benjamin Walsh456c6da2016-09-02 18:55:39 -04005 */
Gerard Marull-Paretascffefc82022-05-06 11:04:23 +02006#include <zephyr/kernel.h>
Benjamin Walshb4b108d2016-10-13 10:31:48 -04007#include <ksched.h>
Gerard Marull-Paretascffefc82022-05-06 11:04:23 +02008#include <zephyr/spinlock.h>
Anas Nashiff0c7fbf2023-09-18 08:34:26 -04009#include <zephyr/kernel/internal/sched_priq.h>
Anas Nashif8634c3b2023-08-29 17:03:12 +000010#include <wait_q.h>
Andy Ross9c62cc62018-01-25 15:24:15 -080011#include <kswap.h>
Andy Ross1acd8c22018-05-03 14:51:49 -070012#include <kernel_arch_func.h>
Anas Nashif4e396172023-09-26 22:46:01 +000013#include <zephyr/internal/syscall_handler.h>
Gerard Marull-Paretascffefc82022-05-06 11:04:23 +020014#include <zephyr/drivers/timer/system_timer.h>
Flavio Ceolin80418602018-11-21 16:22:15 -080015#include <stdbool.h>
Andrew Boiefe031612019-09-21 17:54:37 -070016#include <kernel_internal.h>
Gerard Marull-Paretascffefc82022-05-06 11:04:23 +020017#include <zephyr/logging/log.h>
18#include <zephyr/sys/atomic.h>
19#include <zephyr/sys/math_extras.h>
20#include <zephyr/timing/timing.h>
Gerard Marull-Paretas4863c5f2023-04-11 15:34:39 +020021#include <zephyr/sys/util.h>
Andy Ross52351452021-09-28 09:38:43 -070022
Krzysztof Chruscinski3ed80832020-11-26 19:32:34 +010023LOG_MODULE_DECLARE(os, CONFIG_KERNEL_LOG_LEVEL);
Benjamin Walsh456c6da2016-09-02 18:55:39 -040024
Andy Ross225c74b2018-06-27 11:20:50 -070025#if defined(CONFIG_SCHED_DUMB)
Patrik Flykt4344e272019-03-08 14:19:05 -070026#define _priq_run_add z_priq_dumb_add
27#define _priq_run_remove z_priq_dumb_remove
Andy Rossab46b1b2019-01-30 15:00:42 -080028# if defined(CONFIG_SCHED_CPU_MASK)
29# define _priq_run_best _priq_dumb_mask_best
30# else
Patrik Flykt4344e272019-03-08 14:19:05 -070031# define _priq_run_best z_priq_dumb_best
Andy Rossab46b1b2019-01-30 15:00:42 -080032# endif
Andy Ross225c74b2018-06-27 11:20:50 -070033#elif defined(CONFIG_SCHED_SCALABLE)
Patrik Flykt4344e272019-03-08 14:19:05 -070034#define _priq_run_add z_priq_rb_add
35#define _priq_run_remove z_priq_rb_remove
36#define _priq_run_best z_priq_rb_best
Andy Ross9f06a352018-06-28 10:38:14 -070037#elif defined(CONFIG_SCHED_MULTIQ)
Patrik Flykt4344e272019-03-08 14:19:05 -070038#define _priq_run_add z_priq_mq_add
39#define _priq_run_remove z_priq_mq_remove
40#define _priq_run_best z_priq_mq_best
Jeremy Bettisfb1c36f2021-12-20 16:24:30 -070041static ALWAYS_INLINE void z_priq_mq_add(struct _priq_mq *pq,
42 struct k_thread *thread);
43static ALWAYS_INLINE void z_priq_mq_remove(struct _priq_mq *pq,
44 struct k_thread *thread);
Andy Rosse7ded112018-04-11 14:52:47 -070045#endif
46
Andy Ross225c74b2018-06-27 11:20:50 -070047#if defined(CONFIG_WAITQ_SCALABLE)
Patrik Flykt4344e272019-03-08 14:19:05 -070048#define z_priq_wait_add z_priq_rb_add
49#define _priq_wait_remove z_priq_rb_remove
50#define _priq_wait_best z_priq_rb_best
Andy Ross225c74b2018-06-27 11:20:50 -070051#elif defined(CONFIG_WAITQ_DUMB)
Patrik Flykt4344e272019-03-08 14:19:05 -070052#define z_priq_wait_add z_priq_dumb_add
53#define _priq_wait_remove z_priq_dumb_remove
54#define _priq_wait_best z_priq_dumb_best
Andy Ross1acd8c22018-05-03 14:51:49 -070055#endif
56
Andy Ross6b84ab32021-02-18 10:15:23 -080057struct k_spinlock sched_spinlock;
Andy Ross1acd8c22018-05-03 14:51:49 -070058
Maksim Masalski78ba2ec2021-06-01 15:44:45 +080059static void update_cache(int preempt_ok);
Andy Ross6fb6d3c2021-02-19 15:32:19 -080060static void end_thread(struct k_thread *thread);
Andrew Boie8e0f6a52020-09-05 11:50:18 -070061
Peter Mitsisf8b76f32021-11-29 09:52:11 -050062
Patrik Flykt4344e272019-03-08 14:19:05 -070063static inline int is_preempt(struct k_thread *thread)
Andy Rosse7ded112018-04-11 14:52:47 -070064{
Andy Rosse7ded112018-04-11 14:52:47 -070065 /* explanation in kernel_struct.h */
66 return thread->base.preempt <= _PREEMPT_THRESHOLD;
Andy Rosse7ded112018-04-11 14:52:47 -070067}
68
Florian Grandelcc4d1bd2023-08-28 17:31:54 +020069BUILD_ASSERT(CONFIG_NUM_COOP_PRIORITIES >= CONFIG_NUM_METAIRQ_PRIORITIES,
70 "You need to provide at least as many CONFIG_NUM_COOP_PRIORITIES as "
71 "CONFIG_NUM_METAIRQ_PRIORITIES as Meta IRQs are just a special class of cooperative "
72 "threads.");
73
Andy Ross7aa25fa2018-05-11 14:02:42 -070074static inline int is_metairq(struct k_thread *thread)
75{
76#if CONFIG_NUM_METAIRQ_PRIORITIES > 0
77 return (thread->base.prio - K_HIGHEST_THREAD_PRIO)
78 < CONFIG_NUM_METAIRQ_PRIORITIES;
79#else
Benjamin Cabéa46f1b92023-08-21 15:30:26 +020080 ARG_UNUSED(thread);
Andy Ross7aa25fa2018-05-11 14:02:42 -070081 return 0;
82#endif
83}
84
Anas Nashif80e6a972018-06-23 08:20:34 -050085#if CONFIG_ASSERT
Flavio Ceolin2df02cc2019-03-14 14:32:45 -070086static inline bool is_thread_dummy(struct k_thread *thread)
Andy Rosse7ded112018-04-11 14:52:47 -070087{
Patrik Flykt21358ba2019-03-28 14:57:54 -060088 return (thread->base.thread_state & _THREAD_DUMMY) != 0U;
Andy Ross1acd8c22018-05-03 14:51:49 -070089}
Anas Nashif80e6a972018-06-23 08:20:34 -050090#endif
Andy Ross1acd8c22018-05-03 14:51:49 -070091
James Harris2cd0f662021-03-01 09:19:57 -080092/*
93 * Return value same as e.g. memcmp
94 * > 0 -> thread 1 priority > thread 2 priority
95 * = 0 -> thread 1 priority == thread 2 priority
96 * < 0 -> thread 1 priority < thread 2 priority
97 * Do not rely on the actual value returned aside from the above.
98 * (Again, like memcmp.)
99 */
100int32_t z_sched_prio_cmp(struct k_thread *thread_1,
101 struct k_thread *thread_2)
Andy Ross4a2e50f2018-05-15 11:06:25 -0700102{
James Harris2cd0f662021-03-01 09:19:57 -0800103 /* `prio` is <32b, so the below cannot overflow. */
104 int32_t b1 = thread_1->base.prio;
105 int32_t b2 = thread_2->base.prio;
106
107 if (b1 != b2) {
108 return b2 - b1;
Andy Ross4a2e50f2018-05-15 11:06:25 -0700109 }
110
111#ifdef CONFIG_SCHED_DEADLINE
Andy Rossef626572020-07-10 09:43:36 -0700112 /* If we assume all deadlines live within the same "half" of
113 * the 32 bit modulus space (this is a documented API rule),
James Harris2cd0f662021-03-01 09:19:57 -0800114 * then the latest deadline in the queue minus the earliest is
Andy Rossef626572020-07-10 09:43:36 -0700115 * guaranteed to be (2's complement) non-negative. We can
116 * leverage that to compare the values without having to check
117 * the current time.
Andy Ross4a2e50f2018-05-15 11:06:25 -0700118 */
James Harris2cd0f662021-03-01 09:19:57 -0800119 uint32_t d1 = thread_1->base.prio_deadline;
120 uint32_t d2 = thread_2->base.prio_deadline;
Andy Ross4a2e50f2018-05-15 11:06:25 -0700121
James Harris2cd0f662021-03-01 09:19:57 -0800122 if (d1 != d2) {
123 /* Sooner deadline means higher effective priority.
124 * Doing the calculation with unsigned types and casting
125 * to signed isn't perfect, but at least reduces this
126 * from UB on overflow to impdef.
127 */
128 return (int32_t) (d2 - d1);
Andy Ross4a2e50f2018-05-15 11:06:25 -0700129 }
130#endif
James Harris2cd0f662021-03-01 09:19:57 -0800131 return 0;
Andy Ross4a2e50f2018-05-15 11:06:25 -0700132}
133
Anas Nashif9e3e7f62019-12-19 08:19:45 -0500134static ALWAYS_INLINE bool should_preempt(struct k_thread *thread,
135 int preempt_ok)
Andy Rosseace1df2018-05-30 11:23:02 -0700136{
Andy Ross43553da2018-05-31 11:13:49 -0700137 /* Preemption is OK if it's being explicitly allowed by
138 * software state (e.g. the thread called k_yield())
Andy Rosseace1df2018-05-30 11:23:02 -0700139 */
Flavio Ceolin80418602018-11-21 16:22:15 -0800140 if (preempt_ok != 0) {
141 return true;
Andy Ross43553da2018-05-31 11:13:49 -0700142 }
143
Andy Ross1763a012019-01-28 10:59:41 -0800144 __ASSERT(_current != NULL, "");
145
Andy Ross43553da2018-05-31 11:13:49 -0700146 /* Or if we're pended/suspended/dummy (duh) */
Patrik Flykt4344e272019-03-08 14:19:05 -0700147 if (z_is_thread_prevented_from_running(_current)) {
Andy Ross23c5a632019-01-04 12:52:17 -0800148 return true;
149 }
150
151 /* Edge case on ARM where a thread can be pended out of an
152 * interrupt handler before the "synchronous" swap starts
153 * context switching. Platforms with atomic swap can never
154 * hit this.
155 */
156 if (IS_ENABLED(CONFIG_SWAP_NONATOMIC)
Anas Nashif9e3e7f62019-12-19 08:19:45 -0500157 && z_is_thread_timeout_active(thread)) {
Flavio Ceolin80418602018-11-21 16:22:15 -0800158 return true;
Andy Ross43553da2018-05-31 11:13:49 -0700159 }
160
161 /* Otherwise we have to be running a preemptible thread or
162 * switching to a metairq
163 */
Anas Nashif9e3e7f62019-12-19 08:19:45 -0500164 if (is_preempt(_current) || is_metairq(thread)) {
Flavio Ceolin80418602018-11-21 16:22:15 -0800165 return true;
Andy Rosseace1df2018-05-30 11:23:02 -0700166 }
167
Flavio Ceolin80418602018-11-21 16:22:15 -0800168 return false;
Andy Rosseace1df2018-05-30 11:23:02 -0700169}
170
Andy Rossab46b1b2019-01-30 15:00:42 -0800171#ifdef CONFIG_SCHED_CPU_MASK
172static ALWAYS_INLINE struct k_thread *_priq_dumb_mask_best(sys_dlist_t *pq)
173{
174 /* With masks enabled we need to be prepared to walk the list
175 * looking for one we can run
176 */
Anas Nashif9e3e7f62019-12-19 08:19:45 -0500177 struct k_thread *thread;
Andy Rossab46b1b2019-01-30 15:00:42 -0800178
Anas Nashif9e3e7f62019-12-19 08:19:45 -0500179 SYS_DLIST_FOR_EACH_CONTAINER(pq, thread, base.qnode_dlist) {
180 if ((thread->base.cpu_mask & BIT(_current_cpu->id)) != 0) {
181 return thread;
Andy Rossab46b1b2019-01-30 15:00:42 -0800182 }
183 }
184 return NULL;
185}
186#endif
187
Flavio Ceolin2757e712023-01-06 12:51:16 -0800188#if defined(CONFIG_SCHED_DUMB) || defined(CONFIG_WAITQ_DUMB)
Peter Mitsisf8b76f32021-11-29 09:52:11 -0500189static ALWAYS_INLINE void z_priq_dumb_add(sys_dlist_t *pq,
190 struct k_thread *thread)
Andy Ross0d763e02021-09-07 15:34:04 -0700191{
192 struct k_thread *t;
193
194 __ASSERT_NO_MSG(!z_is_idle_thread_object(thread));
195
196 SYS_DLIST_FOR_EACH_CONTAINER(pq, t, base.qnode_dlist) {
197 if (z_sched_prio_cmp(thread, t) > 0) {
198 sys_dlist_insert(&t->base.qnode_dlist,
199 &thread->base.qnode_dlist);
200 return;
201 }
202 }
203
204 sys_dlist_append(pq, &thread->base.qnode_dlist);
205}
Flavio Ceolin2757e712023-01-06 12:51:16 -0800206#endif
Andy Ross0d763e02021-09-07 15:34:04 -0700207
Andy Rossb11e7962021-09-24 10:57:39 -0700208static ALWAYS_INLINE void *thread_runq(struct k_thread *thread)
Andy Ross387fdd22021-09-23 18:44:40 -0700209{
Andy Rossb11e7962021-09-24 10:57:39 -0700210#ifdef CONFIG_SCHED_CPU_MASK_PIN_ONLY
211 int cpu, m = thread->base.cpu_mask;
212
213 /* Edge case: it's legal per the API to "make runnable" a
214 * thread with all CPUs masked off (i.e. one that isn't
215 * actually runnable!). Sort of a wart in the API and maybe
216 * we should address this in docs/assertions instead to avoid
217 * the extra test.
218 */
219 cpu = m == 0 ? 0 : u32_count_trailing_zeros(m);
220
221 return &_kernel.cpus[cpu].ready_q.runq;
222#else
Benjamin Cabéa46f1b92023-08-21 15:30:26 +0200223 ARG_UNUSED(thread);
Andy Rossb11e7962021-09-24 10:57:39 -0700224 return &_kernel.ready_q.runq;
225#endif
Andy Ross387fdd22021-09-23 18:44:40 -0700226}
227
Andy Rossb11e7962021-09-24 10:57:39 -0700228static ALWAYS_INLINE void *curr_cpu_runq(void)
Andy Ross387fdd22021-09-23 18:44:40 -0700229{
Andy Rossb11e7962021-09-24 10:57:39 -0700230#ifdef CONFIG_SCHED_CPU_MASK_PIN_ONLY
231 return &arch_curr_cpu()->ready_q.runq;
232#else
233 return &_kernel.ready_q.runq;
234#endif
Andy Ross387fdd22021-09-23 18:44:40 -0700235}
236
Andy Rossb11e7962021-09-24 10:57:39 -0700237static ALWAYS_INLINE void runq_add(struct k_thread *thread)
Andy Ross387fdd22021-09-23 18:44:40 -0700238{
Andy Rossb11e7962021-09-24 10:57:39 -0700239 _priq_run_add(thread_runq(thread), thread);
240}
241
242static ALWAYS_INLINE void runq_remove(struct k_thread *thread)
243{
244 _priq_run_remove(thread_runq(thread), thread);
245}
246
247static ALWAYS_INLINE struct k_thread *runq_best(void)
248{
249 return _priq_run_best(curr_cpu_runq());
Andy Ross387fdd22021-09-23 18:44:40 -0700250}
251
Andy Ross4ff45712021-02-08 08:28:54 -0800252/* _current is never in the run queue until context switch on
253 * SMP configurations, see z_requeue_current()
254 */
255static inline bool should_queue_thread(struct k_thread *th)
256{
257 return !IS_ENABLED(CONFIG_SMP) || th != _current;
258}
259
Andy Rossc230fb32021-09-23 16:41:30 -0700260static ALWAYS_INLINE void queue_thread(struct k_thread *thread)
Andy Ross91946ef2021-02-07 13:03:09 -0800261{
262 thread->base.thread_state |= _THREAD_QUEUED;
Andy Ross4ff45712021-02-08 08:28:54 -0800263 if (should_queue_thread(thread)) {
Andy Ross387fdd22021-09-23 18:44:40 -0700264 runq_add(thread);
Andy Ross4ff45712021-02-08 08:28:54 -0800265 }
266#ifdef CONFIG_SMP
267 if (thread == _current) {
268 /* add current to end of queue means "yield" */
269 _current_cpu->swap_ok = true;
270 }
271#endif
Andy Ross91946ef2021-02-07 13:03:09 -0800272}
273
Andy Rossc230fb32021-09-23 16:41:30 -0700274static ALWAYS_INLINE void dequeue_thread(struct k_thread *thread)
Andy Ross91946ef2021-02-07 13:03:09 -0800275{
276 thread->base.thread_state &= ~_THREAD_QUEUED;
Andy Ross4ff45712021-02-08 08:28:54 -0800277 if (should_queue_thread(thread)) {
Andy Ross387fdd22021-09-23 18:44:40 -0700278 runq_remove(thread);
Andy Ross4ff45712021-02-08 08:28:54 -0800279 }
Andy Ross91946ef2021-02-07 13:03:09 -0800280}
281
Andy Rossb4e9ef02022-04-06 10:10:17 -0700282static void signal_pending_ipi(void)
283{
284 /* Synchronization note: you might think we need to lock these
285 * two steps, but an IPI is idempotent. It's OK if we do it
286 * twice. All we require is that if a CPU sees the flag true,
287 * it is guaranteed to send the IPI, and if a core sets
288 * pending_ipi, the IPI will be sent the next time through
289 * this code.
290 */
291#if defined(CONFIG_SMP) && defined(CONFIG_SCHED_IPI_SUPPORTED)
Kumar Gala4f458ba2022-10-18 11:11:46 -0500292 if (arch_num_cpus() > 1) {
Andy Rossb4e9ef02022-04-06 10:10:17 -0700293 if (_kernel.pending_ipi) {
294 _kernel.pending_ipi = false;
295 arch_sched_ipi();
296 }
297 }
298#endif
299}
300
Andy Ross4ff45712021-02-08 08:28:54 -0800301#ifdef CONFIG_SMP
302/* Called out of z_swap() when CONFIG_SMP. The current thread can
303 * never live in the run queue until we are inexorably on the context
304 * switch path on SMP, otherwise there is a deadlock condition where a
305 * set of CPUs pick a cycle of threads to run and wait for them all to
306 * context switch forever.
307 */
308void z_requeue_current(struct k_thread *curr)
309{
Andy Ross6b84ab32021-02-18 10:15:23 -0800310 if (z_is_thread_queued(curr)) {
Andy Ross387fdd22021-09-23 18:44:40 -0700311 runq_add(curr);
Andy Ross4ff45712021-02-08 08:28:54 -0800312 }
Andy Rossb4e9ef02022-04-06 10:10:17 -0700313 signal_pending_ipi();
Andy Ross4ff45712021-02-08 08:28:54 -0800314}
Andy Ross4ff45712021-02-08 08:28:54 -0800315
Andy Ross6fb6d3c2021-02-19 15:32:19 -0800316static inline bool is_aborting(struct k_thread *thread)
317{
Anas Nashifbbbc38b2021-03-29 10:03:49 -0400318 return (thread->base.thread_state & _THREAD_ABORTING) != 0U;
Andy Ross6fb6d3c2021-02-19 15:32:19 -0800319}
Jeremy Bettis1e0a36c2021-12-06 10:56:33 -0700320#endif
Andy Ross6fb6d3c2021-02-19 15:32:19 -0800321
Andy Rossb2791b02019-01-28 09:36:36 -0800322static ALWAYS_INLINE struct k_thread *next_up(void)
Andy Ross1acd8c22018-05-03 14:51:49 -0700323{
Vadim Shakirov73944c62023-07-24 15:42:52 +0300324#ifdef CONFIG_SMP
325 if (is_aborting(_current)) {
326 end_thread(_current);
327 }
328#endif
329
Andy Ross387fdd22021-09-23 18:44:40 -0700330 struct k_thread *thread = runq_best();
Andy Ross11a050b2019-11-13 09:41:52 -0800331
Florian Grandelcc4d1bd2023-08-28 17:31:54 +0200332#if (CONFIG_NUM_METAIRQ_PRIORITIES > 0) && \
333 (CONFIG_NUM_COOP_PRIORITIES > CONFIG_NUM_METAIRQ_PRIORITIES)
Andy Ross11a050b2019-11-13 09:41:52 -0800334 /* MetaIRQs must always attempt to return back to a
335 * cooperative thread they preempted and not whatever happens
336 * to be highest priority now. The cooperative thread was
337 * promised it wouldn't be preempted (by non-metairq threads)!
338 */
339 struct k_thread *mirqp = _current_cpu->metairq_preempted;
340
Anas Nashif9e3e7f62019-12-19 08:19:45 -0500341 if (mirqp != NULL && (thread == NULL || !is_metairq(thread))) {
Andy Ross11a050b2019-11-13 09:41:52 -0800342 if (!z_is_thread_prevented_from_running(mirqp)) {
Anas Nashif9e3e7f62019-12-19 08:19:45 -0500343 thread = mirqp;
Andy Ross11a050b2019-11-13 09:41:52 -0800344 } else {
345 _current_cpu->metairq_preempted = NULL;
346 }
347 }
348#endif
349
Andy Ross1acd8c22018-05-03 14:51:49 -0700350#ifndef CONFIG_SMP
351 /* In uniprocessor mode, we can leave the current thread in
352 * the queue (actually we have to, otherwise the assembly
353 * context switch code for all architectures would be
Patrik Flykt4344e272019-03-08 14:19:05 -0700354 * responsible for putting it back in z_swap and ISR return!),
Andy Ross1acd8c22018-05-03 14:51:49 -0700355 * which makes this choice simple.
356 */
Anas Nashif3f4f3f62021-03-29 17:13:47 -0400357 return (thread != NULL) ? thread : _current_cpu->idle_thread;
Andy Ross1acd8c22018-05-03 14:51:49 -0700358#else
359 /* Under SMP, the "cache" mechanism for selecting the next
360 * thread doesn't work, so we have more work to do to test
Andy Ross11a050b2019-11-13 09:41:52 -0800361 * _current against the best choice from the queue. Here, the
362 * thread selected above represents "the best thread that is
363 * not current".
Andy Rosseace1df2018-05-30 11:23:02 -0700364 *
365 * Subtle note on "queued": in SMP mode, _current does not
366 * live in the queue, so this isn't exactly the same thing as
367 * "ready", it means "is _current already added back to the
368 * queue such that we don't want to re-add it".
Andy Ross1acd8c22018-05-03 14:51:49 -0700369 */
Simon Hein02cfbfe2022-07-19 22:30:17 +0200370 bool queued = z_is_thread_queued(_current);
371 bool active = !z_is_thread_prevented_from_running(_current);
Andy Ross1acd8c22018-05-03 14:51:49 -0700372
Anas Nashif9e3e7f62019-12-19 08:19:45 -0500373 if (thread == NULL) {
374 thread = _current_cpu->idle_thread;
Andy Ross1acd8c22018-05-03 14:51:49 -0700375 }
376
Andy Rosseace1df2018-05-30 11:23:02 -0700377 if (active) {
James Harris2cd0f662021-03-01 09:19:57 -0800378 int32_t cmp = z_sched_prio_cmp(_current, thread);
Andy Ross4ff45712021-02-08 08:28:54 -0800379
380 /* Ties only switch if state says we yielded */
James Harris2cd0f662021-03-01 09:19:57 -0800381 if ((cmp > 0) || ((cmp == 0) && !_current_cpu->swap_ok)) {
Anas Nashif9e3e7f62019-12-19 08:19:45 -0500382 thread = _current;
Andy Rosseace1df2018-05-30 11:23:02 -0700383 }
384
Anas Nashif9e3e7f62019-12-19 08:19:45 -0500385 if (!should_preempt(thread, _current_cpu->swap_ok)) {
386 thread = _current;
Andy Rosseace1df2018-05-30 11:23:02 -0700387 }
Andy Ross1acd8c22018-05-03 14:51:49 -0700388 }
389
Andy Rosseace1df2018-05-30 11:23:02 -0700390 /* Put _current back into the queue */
Anas Nashif9e3e7f62019-12-19 08:19:45 -0500391 if (thread != _current && active &&
392 !z_is_idle_thread_object(_current) && !queued) {
Andy Rossc230fb32021-09-23 16:41:30 -0700393 queue_thread(_current);
Andy Ross1acd8c22018-05-03 14:51:49 -0700394 }
395
Andy Rosseace1df2018-05-30 11:23:02 -0700396 /* Take the new _current out of the queue */
Anas Nashif9e3e7f62019-12-19 08:19:45 -0500397 if (z_is_thread_queued(thread)) {
Andy Rossc230fb32021-09-23 16:41:30 -0700398 dequeue_thread(thread);
Andy Rosseace1df2018-05-30 11:23:02 -0700399 }
Andy Ross1acd8c22018-05-03 14:51:49 -0700400
Andy Ross4ff45712021-02-08 08:28:54 -0800401 _current_cpu->swap_ok = false;
Anas Nashif9e3e7f62019-12-19 08:19:45 -0500402 return thread;
Andy Ross1acd8c22018-05-03 14:51:49 -0700403#endif
404}
405
Andrew Boie8e0f6a52020-09-05 11:50:18 -0700406static void move_thread_to_end_of_prio_q(struct k_thread *thread)
407{
408 if (z_is_thread_queued(thread)) {
Andy Rossc230fb32021-09-23 16:41:30 -0700409 dequeue_thread(thread);
Andrew Boie8e0f6a52020-09-05 11:50:18 -0700410 }
Andy Rossc230fb32021-09-23 16:41:30 -0700411 queue_thread(thread);
Andrew Boie8e0f6a52020-09-05 11:50:18 -0700412 update_cache(thread == _current);
413}
414
Andy Rossc5c3ad92023-03-07 08:29:31 -0800415static void flag_ipi(void)
416{
417#if defined(CONFIG_SMP) && defined(CONFIG_SCHED_IPI_SUPPORTED)
418 if (arch_num_cpus() > 1) {
419 _kernel.pending_ipi = true;
420 }
421#endif
422}
423
Andy Ross9098a452018-09-25 10:56:09 -0700424#ifdef CONFIG_TIMESLICING
425
Gerard Marull-Paretas4863c5f2023-04-11 15:34:39 +0200426static int slice_ticks = DIV_ROUND_UP(CONFIG_TIMESLICE_SIZE * Z_HZ_ticks, Z_HZ_ms);
Nicolas Pitre524ac8a2023-03-31 12:31:28 -0400427static int slice_max_prio = CONFIG_TIMESLICE_PRIORITY;
Nicolas Pitre5879d2d2023-03-09 22:45:18 -0500428static struct _timeout slice_timeouts[CONFIG_MP_MAX_NUM_CPUS];
429static bool slice_expired[CONFIG_MP_MAX_NUM_CPUS];
Andy Ross3e696892021-11-30 18:26:26 -0800430
Andy Ross7fb8eb52019-01-04 12:54:23 -0800431#ifdef CONFIG_SWAP_NONATOMIC
Patrik Flykt4344e272019-03-08 14:19:05 -0700432/* If z_swap() isn't atomic, then it's possible for a timer interrupt
Andy Ross7fb8eb52019-01-04 12:54:23 -0800433 * to try to timeslice away _current after it has already pended
434 * itself but before the corresponding context switch. Treat that as
435 * a noop condition in z_time_slice().
436 */
437static struct k_thread *pending_current;
438#endif
439
Nicolas Pitre5879d2d2023-03-09 22:45:18 -0500440static inline int slice_time(struct k_thread *thread)
441{
442 int ret = slice_ticks;
443
444#ifdef CONFIG_TIMESLICE_PER_THREAD
445 if (thread->base.slice_ticks != 0) {
446 ret = thread->base.slice_ticks;
447 }
Benjamin Cabéa46f1b92023-08-21 15:30:26 +0200448#else
449 ARG_UNUSED(thread);
Nicolas Pitre5879d2d2023-03-09 22:45:18 -0500450#endif
451 return ret;
452}
453
454static inline bool sliceable(struct k_thread *thread)
455{
456 bool ret = is_preempt(thread)
457 && slice_time(thread) != 0
458 && !z_is_prio_higher(thread->base.prio, slice_max_prio)
459 && !z_is_thread_prevented_from_running(thread)
460 && !z_is_idle_thread_object(thread);
461
462#ifdef CONFIG_TIMESLICE_PER_THREAD
463 ret |= thread->base.slice_ticks != 0;
464#endif
465
466 return ret;
467}
468
Andy Rossf3afd5a2023-03-06 14:31:35 -0800469static void slice_timeout(struct _timeout *t)
470{
471 int cpu = ARRAY_INDEX(slice_timeouts, t);
472
473 slice_expired[cpu] = true;
Andy Rossc5c3ad92023-03-07 08:29:31 -0800474
475 /* We need an IPI if we just handled a timeslice expiration
476 * for a different CPU. Ideally this would be able to target
477 * the specific core, but that's not part of the API yet.
478 */
479 if (IS_ENABLED(CONFIG_SMP) && cpu != _current_cpu->id) {
480 flag_ipi();
481 }
Andy Rossf3afd5a2023-03-06 14:31:35 -0800482}
483
Andy Ross3e696892021-11-30 18:26:26 -0800484void z_reset_time_slice(struct k_thread *curr)
Andy Ross9098a452018-09-25 10:56:09 -0700485{
Andy Rossf3afd5a2023-03-06 14:31:35 -0800486 int cpu = _current_cpu->id;
487
488 z_abort_timeout(&slice_timeouts[cpu]);
Nicolas Pitre5879d2d2023-03-09 22:45:18 -0500489 slice_expired[cpu] = false;
490 if (sliceable(curr)) {
Andy Rossf3afd5a2023-03-06 14:31:35 -0800491 z_add_timeout(&slice_timeouts[cpu], slice_timeout,
492 K_TICKS(slice_time(curr) - 1));
Andy Rossed7d8632019-06-15 19:32:04 -0700493 }
Andy Ross9098a452018-09-25 10:56:09 -0700494}
495
Kumar Galaa1b77fd2020-05-27 11:26:57 -0500496void k_sched_time_slice_set(int32_t slice, int prio)
Andy Ross9098a452018-09-25 10:56:09 -0700497{
Florian Grandele256b7d2023-07-07 09:12:38 +0200498 K_SPINLOCK(&sched_spinlock) {
Andy Ross3e696892021-11-30 18:26:26 -0800499 slice_ticks = k_ms_to_ticks_ceil32(slice);
Andy Ross1c305142018-10-15 11:10:49 -0700500 slice_max_prio = prio;
Andy Ross3e696892021-11-30 18:26:26 -0800501 z_reset_time_slice(_current);
Andy Ross1c305142018-10-15 11:10:49 -0700502 }
Andy Ross9098a452018-09-25 10:56:09 -0700503}
504
Andy Ross3e696892021-11-30 18:26:26 -0800505#ifdef CONFIG_TIMESLICE_PER_THREAD
Daniel Leung9c0ff332023-08-03 10:28:01 -0700506void k_thread_time_slice_set(struct k_thread *th, int32_t thread_slice_ticks,
Andy Ross3e696892021-11-30 18:26:26 -0800507 k_thread_timeslice_fn_t expired, void *data)
Andy Ross9098a452018-09-25 10:56:09 -0700508{
Florian Grandele256b7d2023-07-07 09:12:38 +0200509 K_SPINLOCK(&sched_spinlock) {
Daniel Leung9c0ff332023-08-03 10:28:01 -0700510 th->base.slice_ticks = thread_slice_ticks;
Andy Ross3e696892021-11-30 18:26:26 -0800511 th->base.slice_expired = expired;
512 th->base.slice_data = data;
513 }
514}
515#endif
516
Andy Ross9098a452018-09-25 10:56:09 -0700517/* Called out of each timer interrupt */
Andy Rossf3afd5a2023-03-06 14:31:35 -0800518void z_time_slice(void)
Andy Ross9098a452018-09-25 10:56:09 -0700519{
Andrew Boie8e0f6a52020-09-05 11:50:18 -0700520 k_spinlock_key_t key = k_spin_lock(&sched_spinlock);
Nicolas Pitre5879d2d2023-03-09 22:45:18 -0500521 struct k_thread *curr = _current;
Andrew Boie8e0f6a52020-09-05 11:50:18 -0700522
Andy Ross7fb8eb52019-01-04 12:54:23 -0800523#ifdef CONFIG_SWAP_NONATOMIC
Nicolas Pitre5879d2d2023-03-09 22:45:18 -0500524 if (pending_current == curr) {
525 z_reset_time_slice(curr);
Andrew Boie8e0f6a52020-09-05 11:50:18 -0700526 k_spin_unlock(&sched_spinlock, key);
Andy Ross7fb8eb52019-01-04 12:54:23 -0800527 return;
528 }
529 pending_current = NULL;
530#endif
531
Nicolas Pitre5879d2d2023-03-09 22:45:18 -0500532 if (slice_expired[_current_cpu->id] && sliceable(curr)) {
533#ifdef CONFIG_TIMESLICE_PER_THREAD
534 if (curr->base.slice_expired) {
535 k_spin_unlock(&sched_spinlock, key);
536 curr->base.slice_expired(curr, curr->base.slice_data);
537 key = k_spin_lock(&sched_spinlock);
Andy Ross9098a452018-09-25 10:56:09 -0700538 }
Nicolas Pitre5879d2d2023-03-09 22:45:18 -0500539#endif
540 if (!z_is_thread_prevented_from_running(curr)) {
541 move_thread_to_end_of_prio_q(curr);
542 }
543 z_reset_time_slice(curr);
Andy Ross9098a452018-09-25 10:56:09 -0700544 }
Andrew Boie8e0f6a52020-09-05 11:50:18 -0700545 k_spin_unlock(&sched_spinlock, key);
Andy Ross9098a452018-09-25 10:56:09 -0700546}
Andy Ross9098a452018-09-25 10:56:09 -0700547#endif
548
Andy Ross11a050b2019-11-13 09:41:52 -0800549/* Track cooperative threads preempted by metairqs so we can return to
550 * them specifically. Called at the moment a new thread has been
551 * selected to run.
552 */
Anas Nashif9e3e7f62019-12-19 08:19:45 -0500553static void update_metairq_preempt(struct k_thread *thread)
Andy Ross11a050b2019-11-13 09:41:52 -0800554{
Florian Grandelcc4d1bd2023-08-28 17:31:54 +0200555#if (CONFIG_NUM_METAIRQ_PRIORITIES > 0) && \
556 (CONFIG_NUM_COOP_PRIORITIES > CONFIG_NUM_METAIRQ_PRIORITIES)
Anas Nashif9e3e7f62019-12-19 08:19:45 -0500557 if (is_metairq(thread) && !is_metairq(_current) &&
558 !is_preempt(_current)) {
Andy Ross11a050b2019-11-13 09:41:52 -0800559 /* Record new preemption */
560 _current_cpu->metairq_preempted = _current;
Andrew Boief5a7e1a2020-09-02 09:20:38 -0700561 } else if (!is_metairq(thread) && !z_is_idle_thread_object(thread)) {
Andy Ross11a050b2019-11-13 09:41:52 -0800562 /* Returning from existing preemption */
563 _current_cpu->metairq_preempted = NULL;
564 }
Benjamin Cabéa46f1b92023-08-21 15:30:26 +0200565#else
566 ARG_UNUSED(thread);
Andy Ross11a050b2019-11-13 09:41:52 -0800567#endif
568}
569
Andy Ross1856e222018-05-21 11:48:35 -0700570static void update_cache(int preempt_ok)
Andy Ross1acd8c22018-05-03 14:51:49 -0700571{
572#ifndef CONFIG_SMP
Anas Nashif9e3e7f62019-12-19 08:19:45 -0500573 struct k_thread *thread = next_up();
Andy Ross1856e222018-05-21 11:48:35 -0700574
Anas Nashif9e3e7f62019-12-19 08:19:45 -0500575 if (should_preempt(thread, preempt_ok)) {
Andy Rosscb3964f2019-08-16 21:29:26 -0700576#ifdef CONFIG_TIMESLICING
Anas Nashif9e3e7f62019-12-19 08:19:45 -0500577 if (thread != _current) {
Andy Ross3e696892021-11-30 18:26:26 -0800578 z_reset_time_slice(thread);
Andy Ross9098a452018-09-25 10:56:09 -0700579 }
Andy Rosscb3964f2019-08-16 21:29:26 -0700580#endif
Anas Nashif9e3e7f62019-12-19 08:19:45 -0500581 update_metairq_preempt(thread);
582 _kernel.ready_q.cache = thread;
Andy Rosseace1df2018-05-30 11:23:02 -0700583 } else {
584 _kernel.ready_q.cache = _current;
Andy Ross1856e222018-05-21 11:48:35 -0700585 }
Andy Rosseace1df2018-05-30 11:23:02 -0700586
587#else
588 /* The way this works is that the CPU record keeps its
589 * "cooperative swapping is OK" flag until the next reschedule
590 * call or context switch. It doesn't need to be tracked per
591 * thread because if the thread gets preempted for whatever
592 * reason the scheduler will make the same decision anyway.
593 */
594 _current_cpu->swap_ok = preempt_ok;
Andy Ross1acd8c22018-05-03 14:51:49 -0700595#endif
596}
597
Andy Ross05c468f2021-02-19 15:24:24 -0800598static bool thread_active_elsewhere(struct k_thread *thread)
599{
600 /* True if the thread is currently running on another CPU.
601 * There are more scalable designs to answer this question in
602 * constant time, but this is fine for now.
603 */
604#ifdef CONFIG_SMP
605 int currcpu = _current_cpu->id;
606
Kumar Galaa1195ae2022-10-18 09:45:13 -0500607 unsigned int num_cpus = arch_num_cpus();
608
609 for (int i = 0; i < num_cpus; i++) {
Andy Ross05c468f2021-02-19 15:24:24 -0800610 if ((i != currcpu) &&
611 (_kernel.cpus[i].current == thread)) {
612 return true;
613 }
614 }
615#endif
Benjamin Cabéa46f1b92023-08-21 15:30:26 +0200616 ARG_UNUSED(thread);
Andy Ross05c468f2021-02-19 15:24:24 -0800617 return false;
618}
619
Andy Ross96ccc462020-01-23 13:28:30 -0800620static void ready_thread(struct k_thread *thread)
Andy Ross1acd8c22018-05-03 14:51:49 -0700621{
Anas Nashif39f632e2020-12-07 13:15:42 -0500622#ifdef CONFIG_KERNEL_COHERENCE
Andy Rossf6d32ab2020-05-13 15:34:04 +0000623 __ASSERT_NO_MSG(arch_mem_coherent(thread));
624#endif
625
Anas Nashif081605e2020-10-16 20:00:17 -0400626 /* If thread is queued already, do not try and added it to the
627 * run queue again
628 */
629 if (!z_is_thread_queued(thread) && z_is_thread_ready(thread)) {
Torbjörn Leksellf1714432021-03-26 10:59:08 +0100630 SYS_PORT_TRACING_OBJ_FUNC(k_thread, sched_ready, thread);
631
Andy Rossc230fb32021-09-23 16:41:30 -0700632 queue_thread(thread);
Andy Ross1856e222018-05-21 11:48:35 -0700633 update_cache(0);
Andy Ross3267cd32022-04-06 09:58:20 -0700634 flag_ipi();
Andy Ross1acd8c22018-05-03 14:51:49 -0700635 }
636}
637
Andy Ross96ccc462020-01-23 13:28:30 -0800638void z_ready_thread(struct k_thread *thread)
639{
Florian Grandele256b7d2023-07-07 09:12:38 +0200640 K_SPINLOCK(&sched_spinlock) {
Andy Ross05c468f2021-02-19 15:24:24 -0800641 if (!thread_active_elsewhere(thread)) {
642 ready_thread(thread);
643 }
Andy Ross96ccc462020-01-23 13:28:30 -0800644 }
645}
646
Patrik Flykt4344e272019-03-08 14:19:05 -0700647void z_move_thread_to_end_of_prio_q(struct k_thread *thread)
Andy Ross1acd8c22018-05-03 14:51:49 -0700648{
Florian Grandele256b7d2023-07-07 09:12:38 +0200649 K_SPINLOCK(&sched_spinlock) {
Andrew Boie8e0f6a52020-09-05 11:50:18 -0700650 move_thread_to_end_of_prio_q(thread);
Andy Ross1acd8c22018-05-03 14:51:49 -0700651 }
652}
653
Andy Ross96ccc462020-01-23 13:28:30 -0800654void z_sched_start(struct k_thread *thread)
655{
656 k_spinlock_key_t key = k_spin_lock(&sched_spinlock);
657
658 if (z_has_thread_started(thread)) {
659 k_spin_unlock(&sched_spinlock, key);
660 return;
661 }
662
663 z_mark_thread_as_started(thread);
664 ready_thread(thread);
665 z_reschedule(&sched_spinlock, key);
666}
667
Andrew Boie6cf496f2020-02-14 10:52:49 -0800668void z_impl_k_thread_suspend(struct k_thread *thread)
Andy Ross8bdabcc2020-01-07 09:58:46 -0800669{
Torbjörn Leksellf1714432021-03-26 10:59:08 +0100670 SYS_PORT_TRACING_OBJ_FUNC_ENTER(k_thread, suspend, thread);
671
Andy Ross8bdabcc2020-01-07 09:58:46 -0800672 (void)z_abort_thread_timeout(thread);
673
Florian Grandele256b7d2023-07-07 09:12:38 +0200674 K_SPINLOCK(&sched_spinlock) {
Andy Ross8bdabcc2020-01-07 09:58:46 -0800675 if (z_is_thread_queued(thread)) {
Andy Rossc230fb32021-09-23 16:41:30 -0700676 dequeue_thread(thread);
Andy Ross8bdabcc2020-01-07 09:58:46 -0800677 }
678 z_mark_thread_as_suspended(thread);
679 update_cache(thread == _current);
680 }
681
682 if (thread == _current) {
683 z_reschedule_unlocked();
684 }
Torbjörn Leksellf1714432021-03-26 10:59:08 +0100685
686 SYS_PORT_TRACING_OBJ_FUNC_EXIT(k_thread, suspend, thread);
Andy Ross8bdabcc2020-01-07 09:58:46 -0800687}
688
Andrew Boie6cf496f2020-02-14 10:52:49 -0800689#ifdef CONFIG_USERSPACE
690static inline void z_vrfy_k_thread_suspend(struct k_thread *thread)
691{
692 Z_OOPS(Z_SYSCALL_OBJ(thread, K_OBJ_THREAD));
693 z_impl_k_thread_suspend(thread);
694}
695#include <syscalls/k_thread_suspend_mrsh.c>
696#endif
697
698void z_impl_k_thread_resume(struct k_thread *thread)
699{
Torbjörn Leksellf1714432021-03-26 10:59:08 +0100700 SYS_PORT_TRACING_OBJ_FUNC_ENTER(k_thread, resume, thread);
701
Andrew Boie6cf496f2020-02-14 10:52:49 -0800702 k_spinlock_key_t key = k_spin_lock(&sched_spinlock);
703
Anas Nashifbf69afc2020-10-16 19:53:56 -0400704 /* Do not try to resume a thread that was not suspended */
705 if (!z_is_thread_suspended(thread)) {
706 k_spin_unlock(&sched_spinlock, key);
707 return;
708 }
709
Andrew Boie6cf496f2020-02-14 10:52:49 -0800710 z_mark_thread_as_not_suspended(thread);
711 ready_thread(thread);
712
713 z_reschedule(&sched_spinlock, key);
Torbjörn Leksellf1714432021-03-26 10:59:08 +0100714
715 SYS_PORT_TRACING_OBJ_FUNC_EXIT(k_thread, resume, thread);
Andrew Boie6cf496f2020-02-14 10:52:49 -0800716}
717
718#ifdef CONFIG_USERSPACE
719static inline void z_vrfy_k_thread_resume(struct k_thread *thread)
720{
721 Z_OOPS(Z_SYSCALL_OBJ(thread, K_OBJ_THREAD));
722 z_impl_k_thread_resume(thread);
723}
724#include <syscalls/k_thread_resume_mrsh.c>
725#endif
726
Maksim Masalski970820e2021-05-25 14:40:14 +0800727static _wait_q_t *pended_on_thread(struct k_thread *thread)
Andy Ross8bdabcc2020-01-07 09:58:46 -0800728{
729 __ASSERT_NO_MSG(thread->base.pended_on);
730
731 return thread->base.pended_on;
732}
733
Andy Rossed6b4fb2020-01-23 13:04:15 -0800734static void unready_thread(struct k_thread *thread)
735{
736 if (z_is_thread_queued(thread)) {
Andy Rossc230fb32021-09-23 16:41:30 -0700737 dequeue_thread(thread);
Andy Rossed6b4fb2020-01-23 13:04:15 -0800738 }
739 update_cache(thread == _current);
740}
741
Andrew Boie322816e2020-02-20 16:33:06 -0800742/* sched_spinlock must be held */
743static void add_to_waitq_locked(struct k_thread *thread, _wait_q_t *wait_q)
Andy Ross1acd8c22018-05-03 14:51:49 -0700744{
Andrew Boie322816e2020-02-20 16:33:06 -0800745 unready_thread(thread);
746 z_mark_thread_as_pending(thread);
Torbjörn Leksellf1714432021-03-26 10:59:08 +0100747
748 SYS_PORT_TRACING_FUNC(k_thread, sched_pend, thread);
Andy Ross1acd8c22018-05-03 14:51:49 -0700749
Andrew Boie322816e2020-02-20 16:33:06 -0800750 if (wait_q != NULL) {
751 thread->base.pended_on = wait_q;
752 z_priq_wait_add(&wait_q->waitq, thread);
Andy Ross15d52082018-09-26 13:19:31 -0700753 }
Andrew Boie322816e2020-02-20 16:33:06 -0800754}
Andy Ross15d52082018-09-26 13:19:31 -0700755
Andy Ross78327382020-03-05 15:18:14 -0800756static void add_thread_timeout(struct k_thread *thread, k_timeout_t timeout)
Andrew Boie322816e2020-02-20 16:33:06 -0800757{
Andy Ross78327382020-03-05 15:18:14 -0800758 if (!K_TIMEOUT_EQ(timeout, K_FOREVER)) {
Andy Ross78327382020-03-05 15:18:14 -0800759 z_add_thread_timeout(thread, timeout);
Andy Ross1acd8c22018-05-03 14:51:49 -0700760 }
Andy Rosse7ded112018-04-11 14:52:47 -0700761}
762
Andy Rossc32f3762022-10-08 07:24:28 -0700763static void pend_locked(struct k_thread *thread, _wait_q_t *wait_q,
764 k_timeout_t timeout)
Andrew Boie322816e2020-02-20 16:33:06 -0800765{
Anas Nashif39f632e2020-12-07 13:15:42 -0500766#ifdef CONFIG_KERNEL_COHERENCE
Andy Ross1ba74142021-02-09 13:48:25 -0800767 __ASSERT_NO_MSG(wait_q == NULL || arch_mem_coherent(wait_q));
Andy Rossf6d32ab2020-05-13 15:34:04 +0000768#endif
Andy Rossc32f3762022-10-08 07:24:28 -0700769 add_to_waitq_locked(thread, wait_q);
Andy Ross78327382020-03-05 15:18:14 -0800770 add_thread_timeout(thread, timeout);
Andrew Boie322816e2020-02-20 16:33:06 -0800771}
772
Andy Ross78327382020-03-05 15:18:14 -0800773void z_pend_thread(struct k_thread *thread, _wait_q_t *wait_q,
774 k_timeout_t timeout)
Andy Rosse7ded112018-04-11 14:52:47 -0700775{
Patrik Flykt4344e272019-03-08 14:19:05 -0700776 __ASSERT_NO_MSG(thread == _current || is_thread_dummy(thread));
Florian Grandele256b7d2023-07-07 09:12:38 +0200777 K_SPINLOCK(&sched_spinlock) {
Andy Rossc32f3762022-10-08 07:24:28 -0700778 pend_locked(thread, wait_q, timeout);
779 }
Andy Ross1acd8c22018-05-03 14:51:49 -0700780}
781
Andrew Boieffc5bdf2020-09-05 11:44:01 -0700782static inline void unpend_thread_no_timeout(struct k_thread *thread)
783{
Maksim Masalski970820e2021-05-25 14:40:14 +0800784 _priq_wait_remove(&pended_on_thread(thread)->waitq, thread);
Andrew Boieffc5bdf2020-09-05 11:44:01 -0700785 z_mark_thread_as_not_pending(thread);
786 thread->base.pended_on = NULL;
787}
788
Patrik Flykt4344e272019-03-08 14:19:05 -0700789ALWAYS_INLINE void z_unpend_thread_no_timeout(struct k_thread *thread)
Andy Rosse7ded112018-04-11 14:52:47 -0700790{
Florian Grandele256b7d2023-07-07 09:12:38 +0200791 K_SPINLOCK(&sched_spinlock) {
Peter Mitsis31dfd84f2023-01-06 13:20:28 -0500792 if (thread->base.pended_on != NULL) {
793 unpend_thread_no_timeout(thread);
794 }
Andy Ross1acd8c22018-05-03 14:51:49 -0700795 }
Andy Rosse7ded112018-04-11 14:52:47 -0700796}
797
Aastha Grover55377762023-03-08 16:54:12 -0500798void z_sched_wake_thread(struct k_thread *thread, bool is_timeout)
799{
Florian Grandele256b7d2023-07-07 09:12:38 +0200800 K_SPINLOCK(&sched_spinlock) {
Aastha Grover55377762023-03-08 16:54:12 -0500801 bool killed = ((thread->base.thread_state & _THREAD_DEAD) ||
802 (thread->base.thread_state & _THREAD_ABORTING));
803
Aastha Grover877fc3d2023-03-08 16:56:31 -0500804#ifdef CONFIG_EVENTS
805 bool do_nothing = thread->no_wake_on_timeout && is_timeout;
806
807 thread->no_wake_on_timeout = false;
808
809 if (do_nothing) {
810 continue;
811 }
812#endif
813
Aastha Grover55377762023-03-08 16:54:12 -0500814 if (!killed) {
815 /* The thread is not being killed */
816 if (thread->base.pended_on != NULL) {
817 unpend_thread_no_timeout(thread);
818 }
819 z_mark_thread_as_started(thread);
820 if (is_timeout) {
821 z_mark_thread_as_not_suspended(thread);
822 }
823 ready_thread(thread);
824 }
825 }
826
827}
828
Andy Ross987c0e52018-09-27 16:50:00 -0700829#ifdef CONFIG_SYS_CLOCK_EXISTS
830/* Timeout handler for *_thread_timeout() APIs */
Anas Nashif9e3e7f62019-12-19 08:19:45 -0500831void z_thread_timeout(struct _timeout *timeout)
Andy Ross987c0e52018-09-27 16:50:00 -0700832{
Andy Ross37866332021-02-17 10:12:36 -0800833 struct k_thread *thread = CONTAINER_OF(timeout,
834 struct k_thread, base.timeout);
Andy Ross987c0e52018-09-27 16:50:00 -0700835
Aastha Grover55377762023-03-08 16:54:12 -0500836 z_sched_wake_thread(thread, true);
Andy Ross987c0e52018-09-27 16:50:00 -0700837}
838#endif
839
Kumar Galaa1b77fd2020-05-27 11:26:57 -0500840int z_pend_curr_irqlock(uint32_t key, _wait_q_t *wait_q, k_timeout_t timeout)
Andy Rosse7ded112018-04-11 14:52:47 -0700841{
Andy Rossc32f3762022-10-08 07:24:28 -0700842 /* This is a legacy API for pre-switch architectures and isn't
843 * correctly synchronized for multi-cpu use
844 */
845 __ASSERT_NO_MSG(!IS_ENABLED(CONFIG_SMP));
846
847 pend_locked(_current, wait_q, timeout);
Andy Ross722aeea2019-03-14 13:50:16 -0700848
Andy Ross7fb8eb52019-01-04 12:54:23 -0800849#if defined(CONFIG_TIMESLICING) && defined(CONFIG_SWAP_NONATOMIC)
850 pending_current = _current;
Andy Ross722aeea2019-03-14 13:50:16 -0700851
852 int ret = z_swap_irqlock(key);
Florian Grandele256b7d2023-07-07 09:12:38 +0200853 K_SPINLOCK(&sched_spinlock) {
Andy Ross722aeea2019-03-14 13:50:16 -0700854 if (pending_current == _current) {
855 pending_current = NULL;
856 }
857 }
858 return ret;
859#else
Patrik Flykt4344e272019-03-08 14:19:05 -0700860 return z_swap_irqlock(key);
Andy Ross722aeea2019-03-14 13:50:16 -0700861#endif
Andy Rosse7ded112018-04-11 14:52:47 -0700862}
863
Patrik Flykt4344e272019-03-08 14:19:05 -0700864int z_pend_curr(struct k_spinlock *lock, k_spinlock_key_t key,
Andy Ross78327382020-03-05 15:18:14 -0800865 _wait_q_t *wait_q, k_timeout_t timeout)
Andy Rossec554f42018-07-24 13:37:59 -0700866{
867#if defined(CONFIG_TIMESLICING) && defined(CONFIG_SWAP_NONATOMIC)
868 pending_current = _current;
869#endif
Andy Rossc32f3762022-10-08 07:24:28 -0700870 __ASSERT_NO_MSG(sizeof(sched_spinlock) == 0 || lock != &sched_spinlock);
871
872 /* We do a "lock swap" prior to calling z_swap(), such that
873 * the caller's lock gets released as desired. But we ensure
874 * that we hold the scheduler lock and leave local interrupts
875 * masked until we reach the context swich. z_swap() itself
876 * has similar code; the duplication is because it's a legacy
877 * API that doesn't expect to be called with scheduler lock
878 * held.
879 */
880 (void) k_spin_lock(&sched_spinlock);
881 pend_locked(_current, wait_q, timeout);
882 k_spin_release(lock);
883 return z_swap(&sched_spinlock, key);
Andy Rossec554f42018-07-24 13:37:59 -0700884}
885
Andy Ross604f0f42021-02-09 16:47:47 -0800886struct k_thread *z_unpend1_no_timeout(_wait_q_t *wait_q)
887{
888 struct k_thread *thread = NULL;
889
Florian Grandele256b7d2023-07-07 09:12:38 +0200890 K_SPINLOCK(&sched_spinlock) {
Andy Ross604f0f42021-02-09 16:47:47 -0800891 thread = _priq_wait_best(&wait_q->waitq);
892
893 if (thread != NULL) {
894 unpend_thread_no_timeout(thread);
895 }
896 }
897
898 return thread;
899}
900
Patrik Flykt4344e272019-03-08 14:19:05 -0700901struct k_thread *z_unpend_first_thread(_wait_q_t *wait_q)
Andy Rosse7ded112018-04-11 14:52:47 -0700902{
Andy Ross604f0f42021-02-09 16:47:47 -0800903 struct k_thread *thread = NULL;
Andy Rosse7ded112018-04-11 14:52:47 -0700904
Florian Grandele256b7d2023-07-07 09:12:38 +0200905 K_SPINLOCK(&sched_spinlock) {
Andy Ross604f0f42021-02-09 16:47:47 -0800906 thread = _priq_wait_best(&wait_q->waitq);
907
908 if (thread != NULL) {
909 unpend_thread_no_timeout(thread);
910 (void)z_abort_thread_timeout(thread);
911 }
Andy Ross1acd8c22018-05-03 14:51:49 -0700912 }
Andy Rosse7ded112018-04-11 14:52:47 -0700913
Anas Nashif9e3e7f62019-12-19 08:19:45 -0500914 return thread;
Andy Ross1acd8c22018-05-03 14:51:49 -0700915}
Andy Rosse7ded112018-04-11 14:52:47 -0700916
Patrik Flykt4344e272019-03-08 14:19:05 -0700917void z_unpend_thread(struct k_thread *thread)
Andy Ross1acd8c22018-05-03 14:51:49 -0700918{
Patrik Flykt4344e272019-03-08 14:19:05 -0700919 z_unpend_thread_no_timeout(thread);
920 (void)z_abort_thread_timeout(thread);
Andy Ross1acd8c22018-05-03 14:51:49 -0700921}
922
Andy Ross6f139802019-08-20 11:21:28 -0700923/* Priority set utility that does no rescheduling, it just changes the
924 * run queue state, returning true if a reschedule is needed later.
925 */
926bool z_set_prio(struct k_thread *thread, int prio)
Andy Ross1acd8c22018-05-03 14:51:49 -0700927{
Flavio Ceolin02ed85b2018-09-20 15:43:57 -0700928 bool need_sched = 0;
Andy Ross1acd8c22018-05-03 14:51:49 -0700929
Florian Grandele256b7d2023-07-07 09:12:38 +0200930 K_SPINLOCK(&sched_spinlock) {
Patrik Flykt4344e272019-03-08 14:19:05 -0700931 need_sched = z_is_thread_ready(thread);
Andy Ross1acd8c22018-05-03 14:51:49 -0700932
933 if (need_sched) {
Andy Ross4d8e1f22019-07-01 10:25:55 -0700934 /* Don't requeue on SMP if it's the running thread */
935 if (!IS_ENABLED(CONFIG_SMP) || z_is_thread_queued(thread)) {
Andy Rossc230fb32021-09-23 16:41:30 -0700936 dequeue_thread(thread);
Andy Ross4d8e1f22019-07-01 10:25:55 -0700937 thread->base.prio = prio;
Andy Rossc230fb32021-09-23 16:41:30 -0700938 queue_thread(thread);
Andy Ross4d8e1f22019-07-01 10:25:55 -0700939 } else {
940 thread->base.prio = prio;
941 }
Andy Ross1856e222018-05-21 11:48:35 -0700942 update_cache(1);
Andy Ross1acd8c22018-05-03 14:51:49 -0700943 } else {
944 thread->base.prio = prio;
Andy Rosse7ded112018-04-11 14:52:47 -0700945 }
946 }
Torbjörn Leksellf1714432021-03-26 10:59:08 +0100947
948 SYS_PORT_TRACING_OBJ_FUNC(k_thread, sched_priority_set, thread, prio);
Andy Rosse7ded112018-04-11 14:52:47 -0700949
Andy Ross6f139802019-08-20 11:21:28 -0700950 return need_sched;
951}
952
953void z_thread_priority_set(struct k_thread *thread, int prio)
954{
955 bool need_sched = z_set_prio(thread, prio);
956
Andy Ross3267cd32022-04-06 09:58:20 -0700957 flag_ipi();
Andy Ross5737b5c2020-02-04 13:52:09 -0800958
Anas Nashifbbbc38b2021-03-29 10:03:49 -0400959 if (need_sched && _current->base.sched_locked == 0U) {
Patrik Flykt4344e272019-03-08 14:19:05 -0700960 z_reschedule_unlocked();
Benjamin Walsh456c6da2016-09-02 18:55:39 -0400961 }
962}
963
Anas Nashif3f4f3f62021-03-29 17:13:47 -0400964static inline bool resched(uint32_t key)
Benjamin Walsh456c6da2016-09-02 18:55:39 -0400965{
Andy Rosseace1df2018-05-30 11:23:02 -0700966#ifdef CONFIG_SMP
Andy Rosseace1df2018-05-30 11:23:02 -0700967 _current_cpu->swap_ok = 0;
968#endif
969
Andrew Boie4f77c2a2019-11-07 12:43:29 -0800970 return arch_irq_unlocked(key) && !arch_is_in_isr();
Andy Rossec554f42018-07-24 13:37:59 -0700971}
972
Anas Nashif379b93f2020-08-10 15:47:02 -0400973/*
974 * Check if the next ready thread is the same as the current thread
975 * and save the trip if true.
976 */
977static inline bool need_swap(void)
978{
979 /* the SMP case will be handled in C based z_swap() */
980#ifdef CONFIG_SMP
981 return true;
982#else
983 struct k_thread *new_thread;
984
985 /* Check if the next ready thread is the same as the current thread */
Andy Ross6b84ab32021-02-18 10:15:23 -0800986 new_thread = _kernel.ready_q.cache;
Anas Nashif379b93f2020-08-10 15:47:02 -0400987 return new_thread != _current;
988#endif
989}
990
Patrik Flykt4344e272019-03-08 14:19:05 -0700991void z_reschedule(struct k_spinlock *lock, k_spinlock_key_t key)
Andy Rossec554f42018-07-24 13:37:59 -0700992{
Anas Nashif379b93f2020-08-10 15:47:02 -0400993 if (resched(key.key) && need_swap()) {
Patrik Flykt4344e272019-03-08 14:19:05 -0700994 z_swap(lock, key);
Andy Rossec554f42018-07-24 13:37:59 -0700995 } else {
996 k_spin_unlock(lock, key);
Andy Rossb4e9ef02022-04-06 10:10:17 -0700997 signal_pending_ipi();
Andy Rosseace1df2018-05-30 11:23:02 -0700998 }
Andy Rossec554f42018-07-24 13:37:59 -0700999}
Andy Rosseace1df2018-05-30 11:23:02 -07001000
Kumar Galaa1b77fd2020-05-27 11:26:57 -05001001void z_reschedule_irqlock(uint32_t key)
Andy Rossec554f42018-07-24 13:37:59 -07001002{
Andy Ross312b43f2019-05-24 10:09:13 -07001003 if (resched(key)) {
Patrik Flykt4344e272019-03-08 14:19:05 -07001004 z_swap_irqlock(key);
Andy Rossec554f42018-07-24 13:37:59 -07001005 } else {
1006 irq_unlock(key);
Andy Rossb4e9ef02022-04-06 10:10:17 -07001007 signal_pending_ipi();
Andy Rossec554f42018-07-24 13:37:59 -07001008 }
Andy Ross8606fab2018-03-26 10:54:40 -07001009}
1010
Benjamin Walshd7ad1762016-11-10 14:46:58 -05001011void k_sched_lock(void)
1012{
Florian Grandele256b7d2023-07-07 09:12:38 +02001013 K_SPINLOCK(&sched_spinlock) {
Torbjörn Leksellf1714432021-03-26 10:59:08 +01001014 SYS_PORT_TRACING_FUNC(k_thread, sched_lock);
1015
Patrik Flykt4344e272019-03-08 14:19:05 -07001016 z_sched_lock();
Andy Ross1856e222018-05-21 11:48:35 -07001017 }
Benjamin Walshd7ad1762016-11-10 14:46:58 -05001018}
1019
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001020void k_sched_unlock(void)
1021{
Florian Grandele256b7d2023-07-07 09:12:38 +02001022 K_SPINLOCK(&sched_spinlock) {
Anas Nashifbbbc38b2021-03-29 10:03:49 -04001023 __ASSERT(_current->base.sched_locked != 0U, "");
Andy Rosseefd3da2020-02-06 13:39:52 -08001024 __ASSERT(!arch_is_in_isr(), "");
1025
Andy Ross1856e222018-05-21 11:48:35 -07001026 ++_current->base.sched_locked;
Yasushi SHOJI20d07242019-07-31 11:19:08 +09001027 update_cache(0);
Andy Ross1856e222018-05-21 11:48:35 -07001028 }
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001029
Anas Nashif2c5d4042019-12-02 10:24:08 -05001030 LOG_DBG("scheduler unlocked (%p:%d)",
Benjamin Walsha4e033f2016-11-18 16:08:24 -05001031 _current, _current->base.sched_locked);
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001032
Torbjörn Leksellf1714432021-03-26 10:59:08 +01001033 SYS_PORT_TRACING_FUNC(k_thread, sched_unlock);
1034
Patrik Flykt4344e272019-03-08 14:19:05 -07001035 z_reschedule_unlocked();
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001036}
1037
Andy Ross6b84ab32021-02-18 10:15:23 -08001038struct k_thread *z_swap_next_thread(void)
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001039{
Andy Ross6b84ab32021-02-18 10:15:23 -08001040#ifdef CONFIG_SMP
Andy Rossb4e9ef02022-04-06 10:10:17 -07001041 struct k_thread *ret = next_up();
1042
1043 if (ret == _current) {
1044 /* When not swapping, have to signal IPIs here. In
1045 * the context switch case it must happen later, after
1046 * _current gets requeued.
1047 */
1048 signal_pending_ipi();
1049 }
1050 return ret;
Andy Ross6b84ab32021-02-18 10:15:23 -08001051#else
1052 return _kernel.ready_q.cache;
Benjamin Walsh62092182016-12-20 14:39:08 -05001053#endif
Andy Ross6b84ab32021-02-18 10:15:23 -08001054}
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001055
Jeremy Bettis1e0a36c2021-12-06 10:56:33 -07001056#ifdef CONFIG_USE_SWITCH
Andy Rossb18685b2019-02-19 17:24:30 -08001057/* Just a wrapper around _current = xxx with tracing */
1058static inline void set_current(struct k_thread *new_thread)
1059{
Daniel Leung11e6b432020-08-27 16:12:01 -07001060 z_thread_mark_switched_out();
Andy Rosseefd3da2020-02-06 13:39:52 -08001061 _current_cpu->current = new_thread;
Andy Rossb18685b2019-02-19 17:24:30 -08001062}
1063
Nicolas Pitrec9e3e0d2022-03-15 22:36:20 -04001064/**
1065 * @brief Determine next thread to execute upon completion of an interrupt
1066 *
1067 * Thread preemption is performed by context switching after the completion
1068 * of a non-recursed interrupt. This function determines which thread to
1069 * switch to if any. This function accepts as @p interrupted either:
1070 *
1071 * - The handle for the interrupted thread in which case the thread's context
1072 * must already be fully saved and ready to be picked up by a different CPU.
1073 *
1074 * - NULL if more work is required to fully save the thread's state after
1075 * it is known that a new thread is to be scheduled. It is up to the caller
1076 * to store the handle resulting from the thread that is being switched out
1077 * in that thread's "switch_handle" field after its
1078 * context has fully been saved, following the same requirements as with
1079 * the @ref arch_switch() function.
1080 *
1081 * If a new thread needs to be scheduled then its handle is returned.
1082 * Otherwise the same value provided as @p interrupted is returned back.
1083 * Those handles are the same opaque types used by the @ref arch_switch()
1084 * function.
1085 *
1086 * @warning
1087 * The @ref _current value may have changed after this call and not refer
1088 * to the interrupted thread anymore. It might be necessary to make a local
1089 * copy before calling this function.
1090 *
1091 * @param interrupted Handle for the thread that was interrupted or NULL.
1092 * @retval Handle for the next thread to execute, or @p interrupted when
1093 * no new thread is to be scheduled.
1094 */
Patrik Flykt4344e272019-03-08 14:19:05 -07001095void *z_get_next_switch_handle(void *interrupted)
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001096{
Andrew Boieae0d1b22019-03-29 16:25:27 -07001097 z_check_stack_sentinel();
1098
Andy Rosseace1df2018-05-30 11:23:02 -07001099#ifdef CONFIG_SMP
Andy Rossdd432212021-02-05 08:15:02 -08001100 void *ret = NULL;
1101
Florian Grandele256b7d2023-07-07 09:12:38 +02001102 K_SPINLOCK(&sched_spinlock) {
Andy Rossf6d32ab2020-05-13 15:34:04 +00001103 struct k_thread *old_thread = _current, *new_thread;
Andy Rosseace1df2018-05-30 11:23:02 -07001104
Andy Ross4ff45712021-02-08 08:28:54 -08001105 if (IS_ENABLED(CONFIG_SMP)) {
1106 old_thread->switch_handle = NULL;
1107 }
Andy Rossf6d32ab2020-05-13 15:34:04 +00001108 new_thread = next_up();
1109
Andy Ross40d12c12021-09-27 08:22:43 -07001110 z_sched_usage_switch(new_thread);
1111
Andy Rossf6d32ab2020-05-13 15:34:04 +00001112 if (old_thread != new_thread) {
1113 update_metairq_preempt(new_thread);
Andy Rossb89e4272023-05-26 09:12:51 -07001114 z_sched_switch_spin(new_thread);
Andy Rossf6d32ab2020-05-13 15:34:04 +00001115 arch_cohere_stacks(old_thread, interrupted, new_thread);
Andy Ross11a050b2019-11-13 09:41:52 -08001116
Andy Rosseace1df2018-05-30 11:23:02 -07001117 _current_cpu->swap_ok = 0;
Andy Rossf6d32ab2020-05-13 15:34:04 +00001118 set_current(new_thread);
1119
Andy Ross3e696892021-11-30 18:26:26 -08001120#ifdef CONFIG_TIMESLICING
1121 z_reset_time_slice(new_thread);
1122#endif
1123
Danny Oerndrupc9d78402019-12-13 11:24:56 +01001124#ifdef CONFIG_SPIN_VALIDATE
Andy Ross8c1bdda2019-02-20 10:07:31 -08001125 /* Changed _current! Update the spinlock
Anas Nashif6df44052021-04-30 09:58:20 -04001126 * bookkeeping so the validation doesn't get
Andy Ross8c1bdda2019-02-20 10:07:31 -08001127 * confused when the "wrong" thread tries to
1128 * release the lock.
1129 */
1130 z_spin_lock_set_owner(&sched_spinlock);
1131#endif
Andy Ross4ff45712021-02-08 08:28:54 -08001132
1133 /* A queued (runnable) old/current thread
1134 * needs to be added back to the run queue
1135 * here, and atomically with its switch handle
1136 * being set below. This is safe now, as we
1137 * will not return into it.
1138 */
1139 if (z_is_thread_queued(old_thread)) {
Andy Ross387fdd22021-09-23 18:44:40 -07001140 runq_add(old_thread);
Andy Ross4ff45712021-02-08 08:28:54 -08001141 }
Andy Rosseace1df2018-05-30 11:23:02 -07001142 }
Andy Rossf6d32ab2020-05-13 15:34:04 +00001143 old_thread->switch_handle = interrupted;
Andy Rossdd432212021-02-05 08:15:02 -08001144 ret = new_thread->switch_handle;
Andy Ross4ff45712021-02-08 08:28:54 -08001145 if (IS_ENABLED(CONFIG_SMP)) {
1146 /* Active threads MUST have a null here */
1147 new_thread->switch_handle = NULL;
1148 }
Benjamin Walshb8c21602016-12-23 19:34:41 -05001149 }
Andy Rossb4e9ef02022-04-06 10:10:17 -07001150 signal_pending_ipi();
Andy Rossdd432212021-02-05 08:15:02 -08001151 return ret;
Andy Rosseace1df2018-05-30 11:23:02 -07001152#else
Andy Ross40d12c12021-09-27 08:22:43 -07001153 z_sched_usage_switch(_kernel.ready_q.cache);
Andy Rossf6d32ab2020-05-13 15:34:04 +00001154 _current->switch_handle = interrupted;
Andy Ross6b84ab32021-02-18 10:15:23 -08001155 set_current(_kernel.ready_q.cache);
Andy Ross1acd8c22018-05-03 14:51:49 -07001156 return _current->switch_handle;
Andy Rossdd432212021-02-05 08:15:02 -08001157#endif
Andy Ross1acd8c22018-05-03 14:51:49 -07001158}
Benjamin Walshb12a8e02016-12-14 15:24:12 -05001159#endif
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001160
Patrik Flykt4344e272019-03-08 14:19:05 -07001161void z_priq_dumb_remove(sys_dlist_t *pq, struct k_thread *thread)
Andy Ross1acd8c22018-05-03 14:51:49 -07001162{
Benjamin Cabéa46f1b92023-08-21 15:30:26 +02001163 ARG_UNUSED(pq);
1164
Andrew Boie8f0bb6a2019-09-21 18:36:23 -07001165 __ASSERT_NO_MSG(!z_is_idle_thread_object(thread));
Andy Ross1acd8c22018-05-03 14:51:49 -07001166
1167 sys_dlist_remove(&thread->base.qnode_dlist);
1168}
1169
Patrik Flykt4344e272019-03-08 14:19:05 -07001170struct k_thread *z_priq_dumb_best(sys_dlist_t *pq)
Andy Ross1acd8c22018-05-03 14:51:49 -07001171{
Anas Nashif9e3e7f62019-12-19 08:19:45 -05001172 struct k_thread *thread = NULL;
Flavio Ceolin26be3352018-11-15 15:03:32 -08001173 sys_dnode_t *n = sys_dlist_peek_head(pq);
1174
Peter A. Bigot692e1032019-01-03 23:36:28 -06001175 if (n != NULL) {
Anas Nashif9e3e7f62019-12-19 08:19:45 -05001176 thread = CONTAINER_OF(n, struct k_thread, base.qnode_dlist);
Peter A. Bigot692e1032019-01-03 23:36:28 -06001177 }
Anas Nashif9e3e7f62019-12-19 08:19:45 -05001178 return thread;
Andy Ross1acd8c22018-05-03 14:51:49 -07001179}
1180
Patrik Flykt4344e272019-03-08 14:19:05 -07001181bool z_priq_rb_lessthan(struct rbnode *a, struct rbnode *b)
Andy Ross1acd8c22018-05-03 14:51:49 -07001182{
Anas Nashif9e3e7f62019-12-19 08:19:45 -05001183 struct k_thread *thread_a, *thread_b;
James Harris2cd0f662021-03-01 09:19:57 -08001184 int32_t cmp;
Andy Ross1acd8c22018-05-03 14:51:49 -07001185
Anas Nashif9e3e7f62019-12-19 08:19:45 -05001186 thread_a = CONTAINER_OF(a, struct k_thread, base.qnode_rb);
1187 thread_b = CONTAINER_OF(b, struct k_thread, base.qnode_rb);
Andy Ross1acd8c22018-05-03 14:51:49 -07001188
James Harris2cd0f662021-03-01 09:19:57 -08001189 cmp = z_sched_prio_cmp(thread_a, thread_b);
1190
1191 if (cmp > 0) {
Flavio Ceolin02ed85b2018-09-20 15:43:57 -07001192 return true;
James Harris2cd0f662021-03-01 09:19:57 -08001193 } else if (cmp < 0) {
Flavio Ceolin02ed85b2018-09-20 15:43:57 -07001194 return false;
Andy Ross1acd8c22018-05-03 14:51:49 -07001195 } else {
Anas Nashif9e3e7f62019-12-19 08:19:45 -05001196 return thread_a->base.order_key < thread_b->base.order_key
1197 ? 1 : 0;
Andy Ross1acd8c22018-05-03 14:51:49 -07001198 }
1199}
1200
Patrik Flykt4344e272019-03-08 14:19:05 -07001201void z_priq_rb_add(struct _priq_rb *pq, struct k_thread *thread)
Andy Ross1acd8c22018-05-03 14:51:49 -07001202{
1203 struct k_thread *t;
1204
Andrew Boie8f0bb6a2019-09-21 18:36:23 -07001205 __ASSERT_NO_MSG(!z_is_idle_thread_object(thread));
Andy Ross1acd8c22018-05-03 14:51:49 -07001206
1207 thread->base.order_key = pq->next_order_key++;
1208
1209 /* Renumber at wraparound. This is tiny code, and in practice
1210 * will almost never be hit on real systems. BUT on very
1211 * long-running systems where a priq never completely empties
1212 * AND that contains very large numbers of threads, it can be
1213 * a latency glitch to loop over all the threads like this.
1214 */
1215 if (!pq->next_order_key) {
1216 RB_FOR_EACH_CONTAINER(&pq->tree, t, base.qnode_rb) {
1217 t->base.order_key = pq->next_order_key++;
1218 }
1219 }
1220
1221 rb_insert(&pq->tree, &thread->base.qnode_rb);
1222}
1223
Patrik Flykt4344e272019-03-08 14:19:05 -07001224void z_priq_rb_remove(struct _priq_rb *pq, struct k_thread *thread)
Andy Ross1acd8c22018-05-03 14:51:49 -07001225{
Andrew Boie8f0bb6a2019-09-21 18:36:23 -07001226 __ASSERT_NO_MSG(!z_is_idle_thread_object(thread));
Andy Ross1acd8c22018-05-03 14:51:49 -07001227
1228 rb_remove(&pq->tree, &thread->base.qnode_rb);
1229
1230 if (!pq->tree.root) {
1231 pq->next_order_key = 0;
1232 }
1233}
1234
Patrik Flykt4344e272019-03-08 14:19:05 -07001235struct k_thread *z_priq_rb_best(struct _priq_rb *pq)
Andy Ross1acd8c22018-05-03 14:51:49 -07001236{
Anas Nashif9e3e7f62019-12-19 08:19:45 -05001237 struct k_thread *thread = NULL;
Andy Ross1acd8c22018-05-03 14:51:49 -07001238 struct rbnode *n = rb_get_min(&pq->tree);
1239
Peter A. Bigot692e1032019-01-03 23:36:28 -06001240 if (n != NULL) {
Anas Nashif9e3e7f62019-12-19 08:19:45 -05001241 thread = CONTAINER_OF(n, struct k_thread, base.qnode_rb);
Peter A. Bigot692e1032019-01-03 23:36:28 -06001242 }
Anas Nashif9e3e7f62019-12-19 08:19:45 -05001243 return thread;
Andy Ross1acd8c22018-05-03 14:51:49 -07001244}
1245
Andy Ross9f06a352018-06-28 10:38:14 -07001246#ifdef CONFIG_SCHED_MULTIQ
1247# if (K_LOWEST_THREAD_PRIO - K_HIGHEST_THREAD_PRIO) > 31
1248# error Too many priorities for multiqueue scheduler (max 32)
1249# endif
Andy Ross9f06a352018-06-28 10:38:14 -07001250
Peter Mitsisf8b76f32021-11-29 09:52:11 -05001251static ALWAYS_INLINE void z_priq_mq_add(struct _priq_mq *pq,
1252 struct k_thread *thread)
Andy Ross9f06a352018-06-28 10:38:14 -07001253{
1254 int priority_bit = thread->base.prio - K_HIGHEST_THREAD_PRIO;
1255
1256 sys_dlist_append(&pq->queues[priority_bit], &thread->base.qnode_dlist);
Flavio Ceolina9962032019-02-26 10:14:04 -08001257 pq->bitmask |= BIT(priority_bit);
Andy Ross9f06a352018-06-28 10:38:14 -07001258}
1259
Peter Mitsisf8b76f32021-11-29 09:52:11 -05001260static ALWAYS_INLINE void z_priq_mq_remove(struct _priq_mq *pq,
1261 struct k_thread *thread)
Andy Ross9f06a352018-06-28 10:38:14 -07001262{
1263 int priority_bit = thread->base.prio - K_HIGHEST_THREAD_PRIO;
1264
1265 sys_dlist_remove(&thread->base.qnode_dlist);
1266 if (sys_dlist_is_empty(&pq->queues[priority_bit])) {
Flavio Ceolina9962032019-02-26 10:14:04 -08001267 pq->bitmask &= ~BIT(priority_bit);
Andy Ross9f06a352018-06-28 10:38:14 -07001268 }
1269}
Jeremy Bettisfb1c36f2021-12-20 16:24:30 -07001270#endif
Andy Ross9f06a352018-06-28 10:38:14 -07001271
Patrik Flykt4344e272019-03-08 14:19:05 -07001272struct k_thread *z_priq_mq_best(struct _priq_mq *pq)
Andy Ross9f06a352018-06-28 10:38:14 -07001273{
1274 if (!pq->bitmask) {
1275 return NULL;
1276 }
1277
Anas Nashif9e3e7f62019-12-19 08:19:45 -05001278 struct k_thread *thread = NULL;
Andy Ross9f06a352018-06-28 10:38:14 -07001279 sys_dlist_t *l = &pq->queues[__builtin_ctz(pq->bitmask)];
Flavio Ceolin26be3352018-11-15 15:03:32 -08001280 sys_dnode_t *n = sys_dlist_peek_head(l);
Andy Ross9f06a352018-06-28 10:38:14 -07001281
Peter A. Bigot692e1032019-01-03 23:36:28 -06001282 if (n != NULL) {
Anas Nashif9e3e7f62019-12-19 08:19:45 -05001283 thread = CONTAINER_OF(n, struct k_thread, base.qnode_dlist);
Peter A. Bigot692e1032019-01-03 23:36:28 -06001284 }
Anas Nashif9e3e7f62019-12-19 08:19:45 -05001285 return thread;
Andy Ross9f06a352018-06-28 10:38:14 -07001286}
1287
Patrik Flykt4344e272019-03-08 14:19:05 -07001288int z_unpend_all(_wait_q_t *wait_q)
Andy Ross4ca0e072018-05-10 09:45:42 -07001289{
Andy Rossccf3bf72018-05-10 11:10:34 -07001290 int need_sched = 0;
Anas Nashif9e3e7f62019-12-19 08:19:45 -05001291 struct k_thread *thread;
Andy Ross4ca0e072018-05-10 09:45:42 -07001292
Anas Nashif9e3e7f62019-12-19 08:19:45 -05001293 while ((thread = z_waitq_head(wait_q)) != NULL) {
1294 z_unpend_thread(thread);
1295 z_ready_thread(thread);
Andy Ross4ca0e072018-05-10 09:45:42 -07001296 need_sched = 1;
1297 }
Andy Rossccf3bf72018-05-10 11:10:34 -07001298
1299 return need_sched;
Andy Ross4ca0e072018-05-10 09:45:42 -07001300}
1301
Andy Rossb155d062021-09-24 13:49:14 -07001302void init_ready_q(struct _ready_q *rq)
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001303{
Andy Rossb155d062021-09-24 13:49:14 -07001304#if defined(CONFIG_SCHED_SCALABLE)
1305 rq->runq = (struct _priq_rb) {
Andy Ross1acd8c22018-05-03 14:51:49 -07001306 .tree = {
Patrik Flykt4344e272019-03-08 14:19:05 -07001307 .lessthan_fn = z_priq_rb_lessthan,
Andy Ross1acd8c22018-05-03 14:51:49 -07001308 }
1309 };
Andy Rossb155d062021-09-24 13:49:14 -07001310#elif defined(CONFIG_SCHED_MULTIQ)
Andy Ross9f06a352018-06-28 10:38:14 -07001311 for (int i = 0; i < ARRAY_SIZE(_kernel.ready_q.runq.queues); i++) {
Andy Rossb155d062021-09-24 13:49:14 -07001312 sys_dlist_init(&rq->runq.queues[i]);
Andy Ross9f06a352018-06-28 10:38:14 -07001313 }
Andy Rossb155d062021-09-24 13:49:14 -07001314#else
1315 sys_dlist_init(&rq->runq);
Andy Ross9f06a352018-06-28 10:38:14 -07001316#endif
Andy Rossb155d062021-09-24 13:49:14 -07001317}
1318
1319void z_sched_init(void)
1320{
Andy Rossb11e7962021-09-24 10:57:39 -07001321#ifdef CONFIG_SCHED_CPU_MASK_PIN_ONLY
Nicolas Pitre907eea02023-03-16 17:54:25 -04001322 for (int i = 0; i < CONFIG_MP_MAX_NUM_CPUS; i++) {
Andy Rossb11e7962021-09-24 10:57:39 -07001323 init_ready_q(&_kernel.cpus[i].ready_q);
1324 }
1325#else
Andy Rossb155d062021-09-24 13:49:14 -07001326 init_ready_q(&_kernel.ready_q);
Andy Rossb11e7962021-09-24 10:57:39 -07001327#endif
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001328}
1329
Patrik Flykt4344e272019-03-08 14:19:05 -07001330int z_impl_k_thread_priority_get(k_tid_t thread)
Allan Stephens399d0ad2016-10-07 13:41:34 -05001331{
Benjamin Walshf6ca7de2016-11-08 10:36:50 -05001332 return thread->base.prio;
Allan Stephens399d0ad2016-10-07 13:41:34 -05001333}
1334
Andrew Boie76c04a22017-09-27 14:45:10 -07001335#ifdef CONFIG_USERSPACE
Andy Ross65649742019-08-06 13:34:31 -07001336static inline int z_vrfy_k_thread_priority_get(k_tid_t thread)
1337{
1338 Z_OOPS(Z_SYSCALL_OBJ(thread, K_OBJ_THREAD));
1339 return z_impl_k_thread_priority_get(thread);
1340}
1341#include <syscalls/k_thread_priority_get_mrsh.c>
Andrew Boie76c04a22017-09-27 14:45:10 -07001342#endif
1343
Anas Nashif25c87db2021-03-29 10:54:23 -04001344void z_impl_k_thread_priority_set(k_tid_t thread, int prio)
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001345{
Benjamin Walsh3cc2ba92016-11-08 15:44:05 -05001346 /*
1347 * Use NULL, since we cannot know what the entry point is (we do not
1348 * keep track of it) and idle cannot change its priority.
1349 */
Patrik Flykt4344e272019-03-08 14:19:05 -07001350 Z_ASSERT_VALID_PRIO(prio, NULL);
Andrew Boie4f77c2a2019-11-07 12:43:29 -08001351 __ASSERT(!arch_is_in_isr(), "");
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001352
Anas Nashif25c87db2021-03-29 10:54:23 -04001353 struct k_thread *th = (struct k_thread *)thread;
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001354
Anas Nashif25c87db2021-03-29 10:54:23 -04001355 z_thread_priority_set(th, prio);
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001356}
1357
Andrew Boie468190a2017-09-29 14:00:48 -07001358#ifdef CONFIG_USERSPACE
Andy Ross65649742019-08-06 13:34:31 -07001359static inline void z_vrfy_k_thread_priority_set(k_tid_t thread, int prio)
Andrew Boie468190a2017-09-29 14:00:48 -07001360{
Andrew Boie8345e5e2018-05-04 15:57:57 -07001361 Z_OOPS(Z_SYSCALL_OBJ(thread, K_OBJ_THREAD));
Anas Nashif684b8fc2023-09-27 10:41:51 +00001362 Z_OOPS(K_SYSCALL_VERIFY_MSG(_is_valid_prio(prio, NULL),
Andy Ross65649742019-08-06 13:34:31 -07001363 "invalid thread priority %d", prio));
Anas Nashif684b8fc2023-09-27 10:41:51 +00001364 Z_OOPS(K_SYSCALL_VERIFY_MSG((int8_t)prio >= thread->base.prio,
Andrew Boie8345e5e2018-05-04 15:57:57 -07001365 "thread priority may only be downgraded (%d < %d)",
1366 prio, thread->base.prio));
Andrew Boie5008fed2017-10-08 10:11:24 -07001367
Andy Ross65649742019-08-06 13:34:31 -07001368 z_impl_k_thread_priority_set(thread, prio);
Andrew Boie468190a2017-09-29 14:00:48 -07001369}
Andy Ross65649742019-08-06 13:34:31 -07001370#include <syscalls/k_thread_priority_set_mrsh.c>
Andrew Boie468190a2017-09-29 14:00:48 -07001371#endif
1372
Andy Ross4a2e50f2018-05-15 11:06:25 -07001373#ifdef CONFIG_SCHED_DEADLINE
Patrik Flykt4344e272019-03-08 14:19:05 -07001374void z_impl_k_thread_deadline_set(k_tid_t tid, int deadline)
Andy Ross4a2e50f2018-05-15 11:06:25 -07001375{
Anas Nashif9e3e7f62019-12-19 08:19:45 -05001376 struct k_thread *thread = tid;
Andy Ross4a2e50f2018-05-15 11:06:25 -07001377
Florian Grandele256b7d2023-07-07 09:12:38 +02001378 K_SPINLOCK(&sched_spinlock) {
Anas Nashif9e3e7f62019-12-19 08:19:45 -05001379 thread->base.prio_deadline = k_cycle_get_32() + deadline;
1380 if (z_is_thread_queued(thread)) {
Andy Rossc230fb32021-09-23 16:41:30 -07001381 dequeue_thread(thread);
1382 queue_thread(thread);
Andy Ross4a2e50f2018-05-15 11:06:25 -07001383 }
1384 }
1385}
1386
1387#ifdef CONFIG_USERSPACE
Andy Ross075c94f2019-08-13 11:34:34 -07001388static inline void z_vrfy_k_thread_deadline_set(k_tid_t tid, int deadline)
Andy Ross4a2e50f2018-05-15 11:06:25 -07001389{
Anas Nashif9e3e7f62019-12-19 08:19:45 -05001390 struct k_thread *thread = tid;
Andy Ross4a2e50f2018-05-15 11:06:25 -07001391
1392 Z_OOPS(Z_SYSCALL_OBJ(thread, K_OBJ_THREAD));
Anas Nashif684b8fc2023-09-27 10:41:51 +00001393 Z_OOPS(K_SYSCALL_VERIFY_MSG(deadline > 0,
Andy Ross4a2e50f2018-05-15 11:06:25 -07001394 "invalid thread deadline %d",
1395 (int)deadline));
1396
Patrik Flykt4344e272019-03-08 14:19:05 -07001397 z_impl_k_thread_deadline_set((k_tid_t)thread, deadline);
Andy Ross4a2e50f2018-05-15 11:06:25 -07001398}
Andy Ross075c94f2019-08-13 11:34:34 -07001399#include <syscalls/k_thread_deadline_set_mrsh.c>
Andy Ross4a2e50f2018-05-15 11:06:25 -07001400#endif
1401#endif
1402
Jordan Yates1ef647f2022-03-26 09:55:23 +10001403bool k_can_yield(void)
1404{
1405 return !(k_is_pre_kernel() || k_is_in_isr() ||
1406 z_is_idle_thread_object(_current));
1407}
1408
Patrik Flykt4344e272019-03-08 14:19:05 -07001409void z_impl_k_yield(void)
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001410{
Andrew Boie4f77c2a2019-11-07 12:43:29 -08001411 __ASSERT(!arch_is_in_isr(), "");
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001412
Torbjörn Leksellf1714432021-03-26 10:59:08 +01001413 SYS_PORT_TRACING_FUNC(k_thread, yield);
1414
Andy Ross851d14a2021-05-13 15:46:43 -07001415 k_spinlock_key_t key = k_spin_lock(&sched_spinlock);
James Harris6543e062021-03-01 10:14:13 -08001416
Andy Ross851d14a2021-05-13 15:46:43 -07001417 if (!IS_ENABLED(CONFIG_SMP) ||
1418 z_is_thread_queued(_current)) {
Andy Rossc230fb32021-09-23 16:41:30 -07001419 dequeue_thread(_current);
Andy Ross1acd8c22018-05-03 14:51:49 -07001420 }
Andy Rossc230fb32021-09-23 16:41:30 -07001421 queue_thread(_current);
Andy Ross851d14a2021-05-13 15:46:43 -07001422 update_cache(1);
1423 z_swap(&sched_spinlock, key);
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001424}
1425
Andrew Boie468190a2017-09-29 14:00:48 -07001426#ifdef CONFIG_USERSPACE
Andy Ross65649742019-08-06 13:34:31 -07001427static inline void z_vrfy_k_yield(void)
1428{
1429 z_impl_k_yield();
1430}
1431#include <syscalls/k_yield_mrsh.c>
Andrew Boie468190a2017-09-29 14:00:48 -07001432#endif
1433
Flavio Ceolin7a815d52020-10-19 21:37:22 -07001434static int32_t z_tick_sleep(k_ticks_t ticks)
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001435{
Benjamin Walshb12a8e02016-12-14 15:24:12 -05001436#ifdef CONFIG_MULTITHREADING
Flavio Ceolin9a160972020-11-16 10:40:46 -08001437 uint32_t expected_wakeup_ticks;
Carles Cufi9849df82016-12-02 15:31:08 +01001438
Andrew Boie4f77c2a2019-11-07 12:43:29 -08001439 __ASSERT(!arch_is_in_isr(), "");
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001440
Gerard Marull-Paretas737d7992022-11-23 13:42:04 +01001441 LOG_DBG("thread %p for %lu ticks", _current, (unsigned long)ticks);
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001442
Benjamin Walsh5596f782016-12-09 19:57:17 -05001443 /* wait of 0 ms is treated as a 'yield' */
Charles E. Youseb1863032019-05-08 13:22:46 -07001444 if (ticks == 0) {
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001445 k_yield();
Piotr Zięcik7700eb22018-10-25 17:45:08 +02001446 return 0;
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001447 }
1448
Andy Rosse9566392020-12-18 11:12:39 -08001449 k_timeout_t timeout = Z_TIMEOUT_TICKS(ticks);
Lauren Murphy4c85b462021-05-25 17:49:28 -05001450 if (Z_TICK_ABS(ticks) <= 0) {
1451 expected_wakeup_ticks = ticks + sys_clock_tick_get_32();
1452 } else {
1453 expected_wakeup_ticks = Z_TICK_ABS(ticks);
1454 }
Andy Rossd27d4e62019-02-05 15:36:01 -08001455
Andrew Boiea8775ab2020-09-05 12:53:42 -07001456 k_spinlock_key_t key = k_spin_lock(&sched_spinlock);
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001457
Andy Rossdff6b712019-02-25 21:17:29 -08001458#if defined(CONFIG_TIMESLICING) && defined(CONFIG_SWAP_NONATOMIC)
1459 pending_current = _current;
1460#endif
Andrew Boiea8775ab2020-09-05 12:53:42 -07001461 unready_thread(_current);
Andy Ross78327382020-03-05 15:18:14 -08001462 z_add_thread_timeout(_current, timeout);
Andy Ross4521e0c2019-03-22 10:30:19 -07001463 z_mark_thread_as_suspended(_current);
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001464
Andrew Boiea8775ab2020-09-05 12:53:42 -07001465 (void)z_swap(&sched_spinlock, key);
Piotr Zięcik7700eb22018-10-25 17:45:08 +02001466
Andy Ross4521e0c2019-03-22 10:30:19 -07001467 __ASSERT(!z_is_thread_state_set(_current, _THREAD_SUSPENDED), "");
1468
Anas Nashif5c90ceb2021-03-13 08:19:53 -05001469 ticks = (k_ticks_t)expected_wakeup_ticks - sys_clock_tick_get_32();
Piotr Zięcik7700eb22018-10-25 17:45:08 +02001470 if (ticks > 0) {
Charles E. Youseb1863032019-05-08 13:22:46 -07001471 return ticks;
Piotr Zięcik7700eb22018-10-25 17:45:08 +02001472 }
Benjamin Walshb12a8e02016-12-14 15:24:12 -05001473#endif
Piotr Zięcik7700eb22018-10-25 17:45:08 +02001474
1475 return 0;
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001476}
1477
Kumar Galaa1b77fd2020-05-27 11:26:57 -05001478int32_t z_impl_k_sleep(k_timeout_t timeout)
Charles E. Youseb1863032019-05-08 13:22:46 -07001479{
Andy Ross78327382020-03-05 15:18:14 -08001480 k_ticks_t ticks;
Charles E. Youseb1863032019-05-08 13:22:46 -07001481
Peter Bigot8162e582019-12-12 16:07:07 -06001482 __ASSERT(!arch_is_in_isr(), "");
Torbjörn Leksellf1714432021-03-26 10:59:08 +01001483
1484 SYS_PORT_TRACING_FUNC_ENTER(k_thread, sleep, timeout);
Peter Bigot8162e582019-12-12 16:07:07 -06001485
Anas Nashifd2c71792020-10-17 07:52:17 -04001486 /* in case of K_FOREVER, we suspend */
Andy Ross78327382020-03-05 15:18:14 -08001487 if (K_TIMEOUT_EQ(timeout, K_FOREVER)) {
Andrew Boied2b89222019-11-08 10:44:22 -08001488 k_thread_suspend(_current);
Torbjörn Leksellf1714432021-03-26 10:59:08 +01001489
1490 SYS_PORT_TRACING_FUNC_EXIT(k_thread, sleep, timeout, (int32_t) K_TICKS_FOREVER);
1491
Kumar Galaa1b77fd2020-05-27 11:26:57 -05001492 return (int32_t) K_TICKS_FOREVER;
Andrew Boied2b89222019-11-08 10:44:22 -08001493 }
1494
Andy Ross78327382020-03-05 15:18:14 -08001495 ticks = timeout.ticks;
Andy Ross78327382020-03-05 15:18:14 -08001496
Charles E. Youseb1863032019-05-08 13:22:46 -07001497 ticks = z_tick_sleep(ticks);
Torbjörn Leksellf1714432021-03-26 10:59:08 +01001498
1499 int32_t ret = k_ticks_to_ms_floor64(ticks);
1500
1501 SYS_PORT_TRACING_FUNC_EXIT(k_thread, sleep, timeout, ret);
1502
1503 return ret;
Charles E. Youseb1863032019-05-08 13:22:46 -07001504}
1505
Andrew Boie76c04a22017-09-27 14:45:10 -07001506#ifdef CONFIG_USERSPACE
Kumar Galaa1b77fd2020-05-27 11:26:57 -05001507static inline int32_t z_vrfy_k_sleep(k_timeout_t timeout)
Andrew Boie76c04a22017-09-27 14:45:10 -07001508{
Andy Ross78327382020-03-05 15:18:14 -08001509 return z_impl_k_sleep(timeout);
Charles E. Yousea5678312019-05-09 16:46:46 -07001510}
Andy Ross65649742019-08-06 13:34:31 -07001511#include <syscalls/k_sleep_mrsh.c>
Charles E. Yousea5678312019-05-09 16:46:46 -07001512#endif
1513
Kumar Galaa1b77fd2020-05-27 11:26:57 -05001514int32_t z_impl_k_usleep(int us)
Charles E. Yousea5678312019-05-09 16:46:46 -07001515{
Kumar Galaa1b77fd2020-05-27 11:26:57 -05001516 int32_t ticks;
Charles E. Yousea5678312019-05-09 16:46:46 -07001517
Torbjörn Leksellf1714432021-03-26 10:59:08 +01001518 SYS_PORT_TRACING_FUNC_ENTER(k_thread, usleep, us);
1519
Andy Ross88924062019-10-03 11:43:10 -07001520 ticks = k_us_to_ticks_ceil64(us);
Charles E. Yousea5678312019-05-09 16:46:46 -07001521 ticks = z_tick_sleep(ticks);
Torbjörn Leksellf1714432021-03-26 10:59:08 +01001522
1523 SYS_PORT_TRACING_FUNC_EXIT(k_thread, usleep, us, k_ticks_to_us_floor64(ticks));
1524
Andy Ross88924062019-10-03 11:43:10 -07001525 return k_ticks_to_us_floor64(ticks);
Charles E. Yousea5678312019-05-09 16:46:46 -07001526}
1527
1528#ifdef CONFIG_USERSPACE
Kumar Galaa1b77fd2020-05-27 11:26:57 -05001529static inline int32_t z_vrfy_k_usleep(int us)
Charles E. Yousea5678312019-05-09 16:46:46 -07001530{
1531 return z_impl_k_usleep(us);
Andrew Boie76c04a22017-09-27 14:45:10 -07001532}
Andy Ross65649742019-08-06 13:34:31 -07001533#include <syscalls/k_usleep_mrsh.c>
Andrew Boie76c04a22017-09-27 14:45:10 -07001534#endif
1535
Patrik Flykt4344e272019-03-08 14:19:05 -07001536void z_impl_k_wakeup(k_tid_t thread)
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001537{
Torbjörn Leksellf1714432021-03-26 10:59:08 +01001538 SYS_PORT_TRACING_OBJ_FUNC(k_thread, wakeup, thread);
1539
Patrik Flykt4344e272019-03-08 14:19:05 -07001540 if (z_is_thread_pending(thread)) {
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001541 return;
1542 }
1543
Patrik Flykt4344e272019-03-08 14:19:05 -07001544 if (z_abort_thread_timeout(thread) < 0) {
Andrew Boied2b89222019-11-08 10:44:22 -08001545 /* Might have just been sleeping forever */
1546 if (thread->base.thread_state != _THREAD_SUSPENDED) {
1547 return;
1548 }
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001549 }
1550
Andy Ross4521e0c2019-03-22 10:30:19 -07001551 z_mark_thread_as_not_suspended(thread);
Patrik Flykt4344e272019-03-08 14:19:05 -07001552 z_ready_thread(thread);
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001553
Andy Ross3267cd32022-04-06 09:58:20 -07001554 flag_ipi();
Andy Ross5737b5c2020-02-04 13:52:09 -08001555
Andrew Boie4f77c2a2019-11-07 12:43:29 -08001556 if (!arch_is_in_isr()) {
Patrik Flykt4344e272019-03-08 14:19:05 -07001557 z_reschedule_unlocked();
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001558 }
1559}
1560
Enjia Mai7ac40aa2020-05-28 11:29:50 +08001561#ifdef CONFIG_TRACE_SCHED_IPI
1562extern void z_trace_sched_ipi(void);
1563#endif
1564
Andy Ross42ed12a2019-02-19 16:03:39 -08001565#ifdef CONFIG_SMP
Andy Ross42ed12a2019-02-19 16:03:39 -08001566void z_sched_ipi(void)
1567{
Daniel Leungadac4cb2020-01-09 18:55:07 -08001568 /* NOTE: When adding code to this, make sure this is called
1569 * at appropriate location when !CONFIG_SCHED_IPI_SUPPORTED.
1570 */
Enjia Mai7ac40aa2020-05-28 11:29:50 +08001571#ifdef CONFIG_TRACE_SCHED_IPI
1572 z_trace_sched_ipi();
1573#endif
Andy Rossc5c3ad92023-03-07 08:29:31 -08001574
1575#ifdef CONFIG_TIMESLICING
Nicolas Pitre5879d2d2023-03-09 22:45:18 -05001576 if (sliceable(_current)) {
Andy Rossc5c3ad92023-03-07 08:29:31 -08001577 z_time_slice();
1578 }
1579#endif
Andy Ross42ed12a2019-02-19 16:03:39 -08001580}
Andy Ross42ed12a2019-02-19 16:03:39 -08001581#endif
1582
Andrew Boie468190a2017-09-29 14:00:48 -07001583#ifdef CONFIG_USERSPACE
Andy Ross65649742019-08-06 13:34:31 -07001584static inline void z_vrfy_k_wakeup(k_tid_t thread)
1585{
1586 Z_OOPS(Z_SYSCALL_OBJ(thread, K_OBJ_THREAD));
1587 z_impl_k_wakeup(thread);
1588}
1589#include <syscalls/k_wakeup_mrsh.c>
Andrew Boie468190a2017-09-29 14:00:48 -07001590#endif
1591
Daniel Leung0a50ff32023-09-25 11:56:10 -07001592k_tid_t z_impl_k_sched_current_thread_query(void)
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001593{
Andy Rosseefd3da2020-02-06 13:39:52 -08001594#ifdef CONFIG_SMP
1595 /* In SMP, _current is a field read from _current_cpu, which
1596 * can race with preemption before it is read. We must lock
1597 * local interrupts when reading it.
1598 */
1599 unsigned int k = arch_irq_lock();
1600#endif
1601
1602 k_tid_t ret = _current_cpu->current;
1603
1604#ifdef CONFIG_SMP
1605 arch_irq_unlock(k);
1606#endif
1607 return ret;
Benjamin Walsh456c6da2016-09-02 18:55:39 -04001608}
1609
Andrew Boie76c04a22017-09-27 14:45:10 -07001610#ifdef CONFIG_USERSPACE
Daniel Leung0a50ff32023-09-25 11:56:10 -07001611static inline k_tid_t z_vrfy_k_sched_current_thread_query(void)
Andy Ross65649742019-08-06 13:34:31 -07001612{
Daniel Leung0a50ff32023-09-25 11:56:10 -07001613 return z_impl_k_sched_current_thread_query();
Andy Ross65649742019-08-06 13:34:31 -07001614}
Daniel Leung0a50ff32023-09-25 11:56:10 -07001615#include <syscalls/k_sched_current_thread_query_mrsh.c>
Andrew Boie76c04a22017-09-27 14:45:10 -07001616#endif
1617
Patrik Flykt4344e272019-03-08 14:19:05 -07001618int z_impl_k_is_preempt_thread(void)
Benjamin Walsh445830d2016-11-10 15:54:27 -05001619{
Andrew Boie4f77c2a2019-11-07 12:43:29 -08001620 return !arch_is_in_isr() && is_preempt(_current);
Benjamin Walsh445830d2016-11-10 15:54:27 -05001621}
Andrew Boie468190a2017-09-29 14:00:48 -07001622
1623#ifdef CONFIG_USERSPACE
Andy Ross65649742019-08-06 13:34:31 -07001624static inline int z_vrfy_k_is_preempt_thread(void)
1625{
1626 return z_impl_k_is_preempt_thread();
1627}
1628#include <syscalls/k_is_preempt_thread_mrsh.c>
Andrew Boie468190a2017-09-29 14:00:48 -07001629#endif
Andy Rossab46b1b2019-01-30 15:00:42 -08001630
1631#ifdef CONFIG_SCHED_CPU_MASK
1632# ifdef CONFIG_SMP
Evgeniy Paltsev54e07312023-09-19 22:20:36 +01001633/* Right now we use a two byte for this mask */
1634BUILD_ASSERT(CONFIG_MP_MAX_NUM_CPUS <= 16, "Too many CPUs for mask word");
Andy Rossab46b1b2019-01-30 15:00:42 -08001635# endif
1636
1637
Kumar Galaa1b77fd2020-05-27 11:26:57 -05001638static int cpu_mask_mod(k_tid_t thread, uint32_t enable_mask, uint32_t disable_mask)
Andy Rossab46b1b2019-01-30 15:00:42 -08001639{
1640 int ret = 0;
1641
Flavio Ceolin551038e2022-05-02 14:31:04 -07001642#ifdef CONFIG_SCHED_CPU_MASK_PIN_ONLY
Kai Vehmanene81ccef2022-09-09 12:15:41 +03001643 __ASSERT(z_is_thread_prevented_from_running(thread),
1644 "Running threads cannot change CPU pin");
Flavio Ceolin551038e2022-05-02 14:31:04 -07001645#endif
1646
Florian Grandele256b7d2023-07-07 09:12:38 +02001647 K_SPINLOCK(&sched_spinlock) {
Anas Nashif9e3e7f62019-12-19 08:19:45 -05001648 if (z_is_thread_prevented_from_running(thread)) {
1649 thread->base.cpu_mask |= enable_mask;
1650 thread->base.cpu_mask &= ~disable_mask;
Andy Rossab46b1b2019-01-30 15:00:42 -08001651 } else {
1652 ret = -EINVAL;
1653 }
1654 }
Andy Rossb11e7962021-09-24 10:57:39 -07001655
1656#if defined(CONFIG_ASSERT) && defined(CONFIG_SCHED_CPU_MASK_PIN_ONLY)
1657 int m = thread->base.cpu_mask;
1658
1659 __ASSERT((m == 0) || ((m & (m - 1)) == 0),
1660 "Only one CPU allowed in mask when PIN_ONLY");
1661#endif
1662
Andy Rossab46b1b2019-01-30 15:00:42 -08001663 return ret;
1664}
1665
1666int k_thread_cpu_mask_clear(k_tid_t thread)
1667{
1668 return cpu_mask_mod(thread, 0, 0xffffffff);
1669}
1670
1671int k_thread_cpu_mask_enable_all(k_tid_t thread)
1672{
1673 return cpu_mask_mod(thread, 0xffffffff, 0);
1674}
1675
1676int k_thread_cpu_mask_enable(k_tid_t thread, int cpu)
1677{
1678 return cpu_mask_mod(thread, BIT(cpu), 0);
1679}
1680
1681int k_thread_cpu_mask_disable(k_tid_t thread, int cpu)
1682{
1683 return cpu_mask_mod(thread, 0, BIT(cpu));
1684}
1685
Anas Nashifc9d02482022-04-15 08:27:15 -04001686int k_thread_cpu_pin(k_tid_t thread, int cpu)
1687{
1688 int ret;
1689
1690 ret = k_thread_cpu_mask_clear(thread);
1691 if (ret == 0) {
1692 return k_thread_cpu_mask_enable(thread, cpu);
1693 }
1694 return ret;
1695}
1696
Andy Rossab46b1b2019-01-30 15:00:42 -08001697#endif /* CONFIG_SCHED_CPU_MASK */
Andrew Boie322816e2020-02-20 16:33:06 -08001698
Andy Ross6fb6d3c2021-02-19 15:32:19 -08001699static inline void unpend_all(_wait_q_t *wait_q)
1700{
1701 struct k_thread *thread;
1702
1703 while ((thread = z_waitq_head(wait_q)) != NULL) {
1704 unpend_thread_no_timeout(thread);
1705 (void)z_abort_thread_timeout(thread);
1706 arch_thread_return_value_set(thread, 0);
1707 ready_thread(thread);
1708 }
1709}
1710
Chen Peng10f63d112021-09-06 13:59:40 +08001711#ifdef CONFIG_CMSIS_RTOS_V1
1712extern void z_thread_cmsis_status_mask_clear(struct k_thread *thread);
1713#endif
1714
Andy Ross6fb6d3c2021-02-19 15:32:19 -08001715static void end_thread(struct k_thread *thread)
1716{
1717 /* We hold the lock, and the thread is known not to be running
1718 * anywhere.
1719 */
Anas Nashifbbbc38b2021-03-29 10:03:49 -04001720 if ((thread->base.thread_state & _THREAD_DEAD) == 0U) {
Andy Ross6fb6d3c2021-02-19 15:32:19 -08001721 thread->base.thread_state |= _THREAD_DEAD;
1722 thread->base.thread_state &= ~_THREAD_ABORTING;
1723 if (z_is_thread_queued(thread)) {
Andy Rossc230fb32021-09-23 16:41:30 -07001724 dequeue_thread(thread);
Andy Ross6fb6d3c2021-02-19 15:32:19 -08001725 }
1726 if (thread->base.pended_on != NULL) {
1727 unpend_thread_no_timeout(thread);
1728 }
1729 (void)z_abort_thread_timeout(thread);
1730 unpend_all(&thread->join_queue);
1731 update_cache(1);
1732
Grant Ramsay45701e62023-08-14 09:41:52 +12001733#if defined(CONFIG_FPU) && defined(CONFIG_FPU_SHARING)
1734 arch_float_disable(thread);
1735#endif
1736
Torbjörn Leksellf1714432021-03-26 10:59:08 +01001737 SYS_PORT_TRACING_FUNC(k_thread, sched_abort, thread);
1738
Andy Ross6fb6d3c2021-02-19 15:32:19 -08001739 z_thread_monitor_exit(thread);
1740
Chen Peng10f63d112021-09-06 13:59:40 +08001741#ifdef CONFIG_CMSIS_RTOS_V1
1742 z_thread_cmsis_status_mask_clear(thread);
1743#endif
1744
Peter Mitsis6df8efe2023-05-11 14:06:46 -04001745#ifdef CONFIG_OBJ_CORE_THREAD
Peter Mitsise6f10902023-06-01 12:16:40 -04001746#ifdef CONFIG_OBJ_CORE_STATS_THREAD
1747 k_obj_core_stats_deregister(K_OBJ_CORE(thread));
1748#endif
Peter Mitsis6df8efe2023-05-11 14:06:46 -04001749 k_obj_core_unlink(K_OBJ_CORE(thread));
1750#endif
1751
Andy Ross6fb6d3c2021-02-19 15:32:19 -08001752#ifdef CONFIG_USERSPACE
1753 z_mem_domain_exit_thread(thread);
Anas Nashif70cf96b2023-09-27 10:45:48 +00001754 k_thread_perms_all_clear(thread);
Anas Nashif7a18c2b2023-09-27 10:45:18 +00001755 k_object_uninit(thread->stack_obj);
1756 k_object_uninit(thread);
Andy Ross6fb6d3c2021-02-19 15:32:19 -08001757#endif
1758 }
1759}
1760
1761void z_thread_abort(struct k_thread *thread)
1762{
1763 k_spinlock_key_t key = k_spin_lock(&sched_spinlock);
1764
Andy Rossfb613592022-05-19 12:55:28 -07001765 if ((thread->base.user_options & K_ESSENTIAL) != 0) {
1766 k_spin_unlock(&sched_spinlock, key);
1767 __ASSERT(false, "aborting essential thread %p", thread);
1768 k_panic();
1769 return;
1770 }
1771
Anas Nashif3f4f3f62021-03-29 17:13:47 -04001772 if ((thread->base.thread_state & _THREAD_DEAD) != 0U) {
Andy Ross6fb6d3c2021-02-19 15:32:19 -08001773 k_spin_unlock(&sched_spinlock, key);
1774 return;
1775 }
1776
1777#ifdef CONFIG_SMP
1778 if (is_aborting(thread) && thread == _current && arch_is_in_isr()) {
1779 /* Another CPU is spinning for us, don't deadlock */
1780 end_thread(thread);
1781 }
1782
1783 bool active = thread_active_elsewhere(thread);
1784
1785 if (active) {
1786 /* It's running somewhere else, flag and poke */
1787 thread->base.thread_state |= _THREAD_ABORTING;
Lauren Murphyd88ce652021-03-09 16:41:43 -06001788
Andy Rossb4e9ef02022-04-06 10:10:17 -07001789 /* We're going to spin, so need a true synchronous IPI
1790 * here, not deferred!
1791 */
Lauren Murphyd88ce652021-03-09 16:41:43 -06001792#ifdef CONFIG_SCHED_IPI_SUPPORTED
Andy Ross6fb6d3c2021-02-19 15:32:19 -08001793 arch_sched_ipi();
Lauren Murphyd88ce652021-03-09 16:41:43 -06001794#endif
Andy Ross6fb6d3c2021-02-19 15:32:19 -08001795 }
1796
1797 if (is_aborting(thread) && thread != _current) {
1798 if (arch_is_in_isr()) {
1799 /* ISRs can only spin waiting another CPU */
1800 k_spin_unlock(&sched_spinlock, key);
1801 while (is_aborting(thread)) {
1802 }
Andy Rossa08e23f2023-05-26 09:39:16 -07001803
1804 /* Now we know it's dying, but not necessarily
1805 * dead. Wait for the switch to happen!
1806 */
1807 key = k_spin_lock(&sched_spinlock);
1808 z_sched_switch_spin(thread);
1809 k_spin_unlock(&sched_spinlock, key);
Andy Ross6fb6d3c2021-02-19 15:32:19 -08001810 } else if (active) {
1811 /* Threads can join */
1812 add_to_waitq_locked(_current, &thread->join_queue);
1813 z_swap(&sched_spinlock, key);
1814 }
1815 return; /* lock has been released */
1816 }
1817#endif
1818 end_thread(thread);
1819 if (thread == _current && !arch_is_in_isr()) {
1820 z_swap(&sched_spinlock, key);
1821 __ASSERT(false, "aborted _current back from dead");
1822 }
1823 k_spin_unlock(&sched_spinlock, key);
1824}
1825
1826#if !defined(CONFIG_ARCH_HAS_THREAD_ABORT)
1827void z_impl_k_thread_abort(struct k_thread *thread)
1828{
Torbjörn Leksellf1714432021-03-26 10:59:08 +01001829 SYS_PORT_TRACING_OBJ_FUNC_ENTER(k_thread, abort, thread);
1830
Andy Ross6fb6d3c2021-02-19 15:32:19 -08001831 z_thread_abort(thread);
Torbjörn Leksellf1714432021-03-26 10:59:08 +01001832
1833 SYS_PORT_TRACING_OBJ_FUNC_EXIT(k_thread, abort, thread);
Andy Ross6fb6d3c2021-02-19 15:32:19 -08001834}
1835#endif
1836
1837int z_impl_k_thread_join(struct k_thread *thread, k_timeout_t timeout)
1838{
1839 k_spinlock_key_t key = k_spin_lock(&sched_spinlock);
1840 int ret = 0;
1841
Torbjörn Leksellf1714432021-03-26 10:59:08 +01001842 SYS_PORT_TRACING_OBJ_FUNC_ENTER(k_thread, join, thread, timeout);
1843
Anas Nashif3f4f3f62021-03-29 17:13:47 -04001844 if ((thread->base.thread_state & _THREAD_DEAD) != 0U) {
Andy Rossa08e23f2023-05-26 09:39:16 -07001845 z_sched_switch_spin(thread);
Andy Ross6fb6d3c2021-02-19 15:32:19 -08001846 ret = 0;
1847 } else if (K_TIMEOUT_EQ(timeout, K_NO_WAIT)) {
1848 ret = -EBUSY;
Anas Nashif3f4f3f62021-03-29 17:13:47 -04001849 } else if ((thread == _current) ||
1850 (thread->base.pended_on == &_current->join_queue)) {
Andy Ross6fb6d3c2021-02-19 15:32:19 -08001851 ret = -EDEADLK;
1852 } else {
1853 __ASSERT(!arch_is_in_isr(), "cannot join in ISR");
1854 add_to_waitq_locked(_current, &thread->join_queue);
1855 add_thread_timeout(_current, timeout);
Torbjörn Leksellf1714432021-03-26 10:59:08 +01001856
1857 SYS_PORT_TRACING_OBJ_FUNC_BLOCKING(k_thread, join, thread, timeout);
1858 ret = z_swap(&sched_spinlock, key);
1859 SYS_PORT_TRACING_OBJ_FUNC_EXIT(k_thread, join, thread, timeout, ret);
1860
1861 return ret;
Andy Ross6fb6d3c2021-02-19 15:32:19 -08001862 }
1863
Torbjörn Leksellf1714432021-03-26 10:59:08 +01001864 SYS_PORT_TRACING_OBJ_FUNC_EXIT(k_thread, join, thread, timeout, ret);
1865
Andy Ross6fb6d3c2021-02-19 15:32:19 -08001866 k_spin_unlock(&sched_spinlock, key);
1867 return ret;
1868}
1869
Andrew Boie322816e2020-02-20 16:33:06 -08001870#ifdef CONFIG_USERSPACE
1871/* Special case: don't oops if the thread is uninitialized. This is because
1872 * the initialization bit does double-duty for thread objects; if false, means
1873 * the thread object is truly uninitialized, or the thread ran and exited for
1874 * some reason.
1875 *
1876 * Return true in this case indicating we should just do nothing and return
1877 * success to the caller.
1878 */
1879static bool thread_obj_validate(struct k_thread *thread)
1880{
Anas Nashifc25d0802023-09-27 10:49:28 +00001881 struct k_object *ko = k_object_find(thread);
Anas Nashif21254b22023-09-27 10:50:26 +00001882 int ret = k_object_validate(ko, K_OBJ_THREAD, _OBJ_INIT_TRUE);
Andrew Boie322816e2020-02-20 16:33:06 -08001883
1884 switch (ret) {
1885 case 0:
1886 return false;
1887 case -EINVAL:
1888 return true;
1889 default:
1890#ifdef CONFIG_LOG
1891 z_dump_object_error(ret, thread, ko, K_OBJ_THREAD);
1892#endif
Anas Nashif684b8fc2023-09-27 10:41:51 +00001893 Z_OOPS(K_SYSCALL_VERIFY_MSG(ret, "access denied"));
Andrew Boie322816e2020-02-20 16:33:06 -08001894 }
Enjia Mai53ca7092021-01-15 17:09:58 +08001895 CODE_UNREACHABLE; /* LCOV_EXCL_LINE */
Andrew Boie322816e2020-02-20 16:33:06 -08001896}
1897
Andy Ross78327382020-03-05 15:18:14 -08001898static inline int z_vrfy_k_thread_join(struct k_thread *thread,
1899 k_timeout_t timeout)
Andrew Boie322816e2020-02-20 16:33:06 -08001900{
1901 if (thread_obj_validate(thread)) {
1902 return 0;
1903 }
1904
1905 return z_impl_k_thread_join(thread, timeout);
1906}
1907#include <syscalls/k_thread_join_mrsh.c>
Andrew Boiea4c91902020-03-24 16:09:24 -07001908
1909static inline void z_vrfy_k_thread_abort(k_tid_t thread)
1910{
1911 if (thread_obj_validate(thread)) {
1912 return;
1913 }
1914
Anas Nashif684b8fc2023-09-27 10:41:51 +00001915 Z_OOPS(K_SYSCALL_VERIFY_MSG(!(thread->base.user_options & K_ESSENTIAL),
Andrew Boiea4c91902020-03-24 16:09:24 -07001916 "aborting essential thread %p", thread));
1917
1918 z_impl_k_thread_abort((struct k_thread *)thread);
1919}
1920#include <syscalls/k_thread_abort_mrsh.c>
Andrew Boie322816e2020-02-20 16:33:06 -08001921#endif /* CONFIG_USERSPACE */
Peter Bigot0259c862021-01-12 13:45:32 -06001922
1923/*
1924 * future scheduler.h API implementations
1925 */
1926bool z_sched_wake(_wait_q_t *wait_q, int swap_retval, void *swap_data)
1927{
1928 struct k_thread *thread;
1929 bool ret = false;
1930
Florian Grandele256b7d2023-07-07 09:12:38 +02001931 K_SPINLOCK(&sched_spinlock) {
Peter Bigot0259c862021-01-12 13:45:32 -06001932 thread = _priq_wait_best(&wait_q->waitq);
1933
1934 if (thread != NULL) {
1935 z_thread_return_value_set_with_data(thread,
1936 swap_retval,
1937 swap_data);
1938 unpend_thread_no_timeout(thread);
1939 (void)z_abort_thread_timeout(thread);
1940 ready_thread(thread);
1941 ret = true;
1942 }
1943 }
1944
1945 return ret;
1946}
1947
1948int z_sched_wait(struct k_spinlock *lock, k_spinlock_key_t key,
1949 _wait_q_t *wait_q, k_timeout_t timeout, void **data)
1950{
1951 int ret = z_pend_curr(lock, key, wait_q, timeout);
1952
1953 if (data != NULL) {
1954 *data = _current->base.swap_data;
1955 }
1956 return ret;
1957}
Peter Mitsisca583392023-01-05 11:50:21 -05001958
1959int z_sched_waitq_walk(_wait_q_t *wait_q,
1960 int (*func)(struct k_thread *, void *), void *data)
1961{
1962 struct k_thread *thread;
1963 int status = 0;
1964
Florian Grandele256b7d2023-07-07 09:12:38 +02001965 K_SPINLOCK(&sched_spinlock) {
Peter Mitsisca583392023-01-05 11:50:21 -05001966 _WAIT_Q_FOR_EACH(wait_q, thread) {
1967
1968 /*
1969 * Invoke the callback function on each waiting thread
1970 * for as long as there are both waiting threads AND
1971 * it returns 0.
1972 */
1973
1974 status = func(thread, data);
1975 if (status != 0) {
1976 break;
1977 }
1978 }
1979 }
1980
1981 return status;
1982}