-
Notifications
You must be signed in to change notification settings - Fork 7.2k
/
Copy paththread_monitor.c
110 lines (94 loc) · 2.88 KB
/
thread_monitor.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
/*
* Copyright (c) 2010-2014 Wind River Systems, Inc.
* Copyright (c) 2024 Intel Corporation
*
* SPDX-License-Identifier: Apache-2.0
*/
#include <zephyr/kernel.h>
#include <kthread.h>
struct k_spinlock z_thread_monitor_lock;
/*
* Remove a thread from the kernel's list of active threads.
*/
void z_thread_monitor_exit(struct k_thread *thread)
{
k_spinlock_key_t key = k_spin_lock(&z_thread_monitor_lock);
if (thread == _kernel.threads) {
_kernel.threads = _kernel.threads->next_thread;
} else {
struct k_thread *prev_thread;
prev_thread = _kernel.threads;
while ((prev_thread != NULL) &&
(thread != prev_thread->next_thread)) {
prev_thread = prev_thread->next_thread;
}
if (prev_thread != NULL) {
prev_thread->next_thread = thread->next_thread;
}
}
k_spin_unlock(&z_thread_monitor_lock, key);
}
/*
* Helper function to iterate over threads with optional filtering and locking behavior.
*/
static void thread_foreach_helper(k_thread_user_cb_t user_cb, void *user_data,
bool unlocked, bool filter_by_cpu, unsigned int cpu)
{
struct k_thread *thread;
k_spinlock_key_t key;
__ASSERT(user_cb != NULL, "user_cb can not be NULL");
if (filter_by_cpu) {
__ASSERT(cpu < CONFIG_MP_MAX_NUM_CPUS, "cpu filter out of bounds");
}
key = k_spin_lock(&z_thread_monitor_lock);
for (thread = _kernel.threads; thread; thread = thread->next_thread) {
/* cpu is only defined when SMP=y*/
#ifdef CONFIG_SMP
bool on_cpu = (thread->base.cpu == cpu);
#else
bool on_cpu = false;
#endif
if (filter_by_cpu && !on_cpu) {
continue;
}
if (unlocked) {
k_spin_unlock(&z_thread_monitor_lock, key);
user_cb(thread, user_data);
key = k_spin_lock(&z_thread_monitor_lock);
} else {
user_cb(thread, user_data);
}
}
k_spin_unlock(&z_thread_monitor_lock, key);
}
/*
* Public API functions using the helper.
*/
void k_thread_foreach(k_thread_user_cb_t user_cb, void *user_data)
{
SYS_PORT_TRACING_FUNC_ENTER(k_thread, foreach);
thread_foreach_helper(user_cb, user_data, false, false, 0);
SYS_PORT_TRACING_FUNC_EXIT(k_thread, foreach);
}
void k_thread_foreach_unlocked(k_thread_user_cb_t user_cb, void *user_data)
{
SYS_PORT_TRACING_FUNC_ENTER(k_thread, foreach_unlocked);
thread_foreach_helper(user_cb, user_data, true, false, 0);
SYS_PORT_TRACING_FUNC_EXIT(k_thread, foreach_unlocked);
}
#ifdef CONFIG_SMP
void k_thread_foreach_filter_by_cpu(unsigned int cpu, k_thread_user_cb_t user_cb,
void *user_data)
{
SYS_PORT_TRACING_FUNC_ENTER(k_thread, foreach);
thread_foreach_helper(user_cb, user_data, false, true, cpu);
SYS_PORT_TRACING_FUNC_EXIT(k_thread, foreach);
}
void k_thread_foreach_unlocked_filter_by_cpu(unsigned int cpu, k_thread_user_cb_t user_cb,
void *user_data)
{
SYS_PORT_TRACING_FUNC_ENTER(k_thread, foreach_unlocked);
thread_foreach_helper(user_cb, user_data, true, true, cpu);
SYS_PORT_TRACING_FUNC_EXIT(k_thread, foreach_unlocked);
}
#endif /* CONFIG_SMP */