1 /* Copyright (c) 2022 Intel corporation
2 * SPDX-License-Identifier: Apache-2.0
3 */
4
5 #include <zephyr/kernel.h>
6 #include <zephyr/kernel_structs.h>
7 #include <zephyr/kernel/smp.h>
8 #include <zephyr/spinlock.h>
9 #include <kswap.h>
10 #include <kernel_internal.h>
11
12 static atomic_t global_lock;
13
14 /**
15 * Flag to tell recently powered up CPU to start
16 * initialization routine.
17 *
18 * 0 to tell powered up CPU to wait.
19 * 1 to tell powered up CPU to continue initialization.
20 */
21 static atomic_t cpu_start_flag;
22
23 /**
24 * Flag to tell caller that the target CPU is now
25 * powered up and ready to be initialized.
26 *
27 * 0 if target CPU is not yet ready.
28 * 1 if target CPU has powered up and ready to be initialized.
29 */
30 static atomic_t ready_flag;
31
32 /**
33 * Struct holding the function to be called before handing off
34 * to schedule and its argument.
35 */
36 static struct cpu_start_cb {
37 /**
38 * Function to be called before handing off to scheduler.
39 * Can be NULL.
40 */
41 smp_init_fn fn;
42
43 /** Argument to @ref cpu_start_fn.fn. */
44 void *arg;
45
46 /** Invoke scheduler after CPU has started if true. */
47 bool invoke_sched;
48
49 #ifdef CONFIG_SYS_CLOCK_EXISTS
50 /** True if smp_timer_init() needs to be called. */
51 bool reinit_timer;
52 #endif /* CONFIG_SYS_CLOCK_EXISTS */
53 } cpu_start_fn;
54
55 static struct k_spinlock cpu_start_lock;
56
z_smp_global_lock(void)57 unsigned int z_smp_global_lock(void)
58 {
59 unsigned int key = arch_irq_lock();
60
61 if (!arch_current_thread()->base.global_lock_count) {
62 while (!atomic_cas(&global_lock, 0, 1)) {
63 arch_spin_relax();
64 }
65 }
66
67 arch_current_thread()->base.global_lock_count++;
68
69 return key;
70 }
71
z_smp_global_unlock(unsigned int key)72 void z_smp_global_unlock(unsigned int key)
73 {
74 if (arch_current_thread()->base.global_lock_count != 0U) {
75 arch_current_thread()->base.global_lock_count--;
76
77 if (!arch_current_thread()->base.global_lock_count) {
78 (void)atomic_clear(&global_lock);
79 }
80 }
81
82 arch_irq_unlock(key);
83 }
84
85 /* Called from within z_swap(), so assumes lock already held */
z_smp_release_global_lock(struct k_thread * thread)86 void z_smp_release_global_lock(struct k_thread *thread)
87 {
88 if (!thread->base.global_lock_count) {
89 (void)atomic_clear(&global_lock);
90 }
91 }
92
93 /* Tiny delay that relaxes bus traffic to avoid spamming a shared
94 * memory bus looking at an atomic variable
95 */
local_delay(void)96 static inline void local_delay(void)
97 {
98 for (volatile int i = 0; i < 1000; i++) {
99 }
100 }
101
wait_for_start_signal(atomic_t * start_flag)102 static void wait_for_start_signal(atomic_t *start_flag)
103 {
104 /* Wait for the signal to begin scheduling */
105 while (!atomic_get(start_flag)) {
106 local_delay();
107 }
108 }
109
smp_init_top(void * arg)110 static inline void smp_init_top(void *arg)
111 {
112 struct cpu_start_cb csc = arg ? *(struct cpu_start_cb *)arg : (struct cpu_start_cb){0};
113
114 /* Let start_cpu() know that this CPU has powered up. */
115 (void)atomic_set(&ready_flag, 1);
116
117 /* Wait for the CPU start caller to signal that
118 * we can start initialization.
119 */
120 wait_for_start_signal(&cpu_start_flag);
121
122 if ((arg == NULL) || csc.invoke_sched) {
123 /* Initialize the dummy thread struct so that
124 * the scheduler can schedule actual threads to run.
125 */
126 z_dummy_thread_init(&_thread_dummy);
127 }
128
129 #ifdef CONFIG_SYS_CLOCK_EXISTS
130 if ((arg == NULL) || csc.reinit_timer) {
131 smp_timer_init();
132 }
133 #endif /* CONFIG_SYS_CLOCK_EXISTS */
134
135 /* Do additional initialization steps if needed. */
136 if (csc.fn != NULL) {
137 csc.fn(csc.arg);
138 }
139
140 if ((arg != NULL) && !csc.invoke_sched) {
141 /* Don't invoke scheduler. */
142 return;
143 }
144
145 /* Let scheduler decide what thread to run next. */
146 z_swap_unlocked();
147
148 CODE_UNREACHABLE; /* LCOV_EXCL_LINE */
149 }
150
start_cpu(int id,struct cpu_start_cb * csc)151 static void start_cpu(int id, struct cpu_start_cb *csc)
152 {
153 /* Clear the ready flag so the newly powered up CPU can
154 * signal that it has powered up.
155 */
156 (void)atomic_clear(&ready_flag);
157
158 /* Power up the CPU */
159 arch_cpu_start(id, z_interrupt_stacks[id], CONFIG_ISR_STACK_SIZE,
160 smp_init_top, csc);
161
162 /* Wait until the newly powered up CPU to signal that
163 * it has powered up.
164 */
165 while (!atomic_get(&ready_flag)) {
166 local_delay();
167 }
168 }
169
k_smp_cpu_start(int id,smp_init_fn fn,void * arg)170 void k_smp_cpu_start(int id, smp_init_fn fn, void *arg)
171 {
172 k_spinlock_key_t key = k_spin_lock(&cpu_start_lock);
173
174 cpu_start_fn.fn = fn;
175 cpu_start_fn.arg = arg;
176 cpu_start_fn.invoke_sched = true;
177
178 #ifdef CONFIG_SYS_CLOCK_EXISTS
179 cpu_start_fn.reinit_timer = true;
180 #endif /* CONFIG_SYS_CLOCK_EXISTS */
181
182 /* We are only starting one CPU so we do not need to synchronize
183 * across all CPUs using the start_flag. So just set it to 1.
184 */
185 (void)atomic_set(&cpu_start_flag, 1); /* async, don't care */
186
187 /* Initialize various CPU structs related to this CPU. */
188 z_init_cpu(id);
189
190 /* Start the CPU! */
191 start_cpu(id, &cpu_start_fn);
192
193 k_spin_unlock(&cpu_start_lock, key);
194 }
195
k_smp_cpu_resume(int id,smp_init_fn fn,void * arg,bool reinit_timer,bool invoke_sched)196 void k_smp_cpu_resume(int id, smp_init_fn fn, void *arg,
197 bool reinit_timer, bool invoke_sched)
198 {
199 k_spinlock_key_t key = k_spin_lock(&cpu_start_lock);
200
201 cpu_start_fn.fn = fn;
202 cpu_start_fn.arg = arg;
203 cpu_start_fn.invoke_sched = invoke_sched;
204
205 #ifdef CONFIG_SYS_CLOCK_EXISTS
206 cpu_start_fn.reinit_timer = reinit_timer;
207 #else
208 ARG_UNUSED(reinit_timer);
209 #endif /* CONFIG_SYS_CLOCK_EXISTS */
210
211 /* We are only starting one CPU so we do not need to synchronize
212 * across all CPUs using the start_flag. So just set it to 1.
213 */
214 (void)atomic_set(&cpu_start_flag, 1);
215
216 /* Start the CPU! */
217 start_cpu(id, &cpu_start_fn);
218
219 k_spin_unlock(&cpu_start_lock, key);
220 }
221
z_smp_init(void)222 void z_smp_init(void)
223 {
224 /* We are powering up all CPUs and we want to synchronize their
225 * entry into scheduler. So set the start flag to 0 here.
226 */
227 (void)atomic_clear(&cpu_start_flag);
228
229 /* Just start CPUs one by one. */
230 unsigned int num_cpus = arch_num_cpus();
231
232 for (int i = 1; i < num_cpus; i++) {
233 z_init_cpu(i);
234 start_cpu(i, NULL);
235 }
236
237 /* Let loose those CPUs so they can start scheduling
238 * threads to run.
239 */
240 (void)atomic_set(&cpu_start_flag, 1);
241 }
242
z_smp_cpu_mobile(void)243 bool z_smp_cpu_mobile(void)
244 {
245 unsigned int k = arch_irq_lock();
246 bool pinned = arch_is_in_isr() || !arch_irq_unlocked(k);
247
248 arch_irq_unlock(k);
249 return !pinned;
250 }
251