1 /*
2 * Copyright (c) 2010-2014 Wind River Systems, Inc.
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 /**
8 * @file
9 * @brief Floating point register sharing routines
10 *
11 * This module allows multiple preemptible threads to safely share the system's
12 * floating point registers, by allowing the system to save FPU state info
13 * in a thread's stack region when a preemptive context switch occurs.
14 *
15 * Note: If the kernel has been built without floating point register sharing
16 * support (CONFIG_FPU_SHARING), the floating point registers can still be used
17 * safely by one or more cooperative threads OR by a single preemptive thread,
18 * but not by both.
19 *
20 * This code is not necessary for systems with CONFIG_EAGER_FPU_SHARING, as
21 * the floating point context is unconditionally saved/restored with every
22 * context switch.
23 *
24 * The floating point register sharing mechanism is designed for minimal
25 * intrusiveness. Floating point state saving is only performed for threads
26 * that explicitly indicate they are using FPU registers, to avoid impacting
27 * the stack size requirements of all other threads. Also, the SSE registers
28 * are only saved for threads that actually used them. For those threads that
29 * do require floating point state saving, a "lazy save/restore" mechanism
30 * is employed so that the FPU's register sets are only switched in and out
31 * when absolutely necessary; this avoids wasting effort preserving them when
32 * there is no risk that they will be altered, or when there is no need to
33 * preserve their contents.
34 *
35 * WARNING
36 * The use of floating point instructions by ISRs is not supported by the
37 * kernel.
38 *
39 * INTERNAL
40 * The kernel sets CR0[TS] to 0 only for threads that require FP register
41 * sharing. All other threads have CR0[TS] set to 1 so that an attempt
42 * to perform an FP operation will cause an exception, allowing the kernel
43 * to enable FP register sharing on its behalf.
44 */
45
46 #include <zephyr/kernel.h>
47 #include <kernel_internal.h>
48
49 /* SSE control/status register default value (used by assembler code) */
50 extern uint32_t _sse_mxcsr_default_value;
51
52 /**
53 * @brief Disallow use of floating point capabilities
54 *
55 * This routine sets CR0[TS] to 1, which disallows the use of FP instructions
56 * by the currently executing thread.
57 */
z_FpAccessDisable(void)58 static inline void z_FpAccessDisable(void)
59 {
60 void *tempReg;
61
62 __asm__ volatile(
63 "movl %%cr0, %0;\n\t"
64 "orl $0x8, %0;\n\t"
65 "movl %0, %%cr0;\n\t"
66 : "=r"(tempReg)
67 :
68 : "memory");
69 }
70
71
72 /**
73 * @brief Save non-integer context information
74 *
75 * This routine saves the system's "live" non-integer context into the
76 * specified area. If the specified thread supports SSE then
77 * x87/MMX/SSEx thread info is saved, otherwise only x87/MMX thread is saved.
78 * Function is invoked by FpCtxSave(struct k_thread *thread)
79 */
z_do_fp_regs_save(void * preemp_float_reg)80 static inline void z_do_fp_regs_save(void *preemp_float_reg)
81 {
82 __asm__ volatile("fnsave (%0);\n\t"
83 :
84 : "r"(preemp_float_reg)
85 : "memory");
86 }
87
88 /**
89 * @brief Save non-integer context information
90 *
91 * This routine saves the system's "live" non-integer context into the
92 * specified area. If the specified thread supports SSE then
93 * x87/MMX/SSEx thread info is saved, otherwise only x87/MMX thread is saved.
94 * Function is invoked by FpCtxSave(struct k_thread *thread)
95 */
z_do_fp_and_sse_regs_save(void * preemp_float_reg)96 static inline void z_do_fp_and_sse_regs_save(void *preemp_float_reg)
97 {
98 __asm__ volatile("fxsave (%0);\n\t"
99 :
100 : "r"(preemp_float_reg)
101 : "memory");
102 }
103
104 /**
105 * @brief Initialize floating point register context information.
106 *
107 * This routine initializes the system's "live" floating point registers.
108 */
z_do_fp_regs_init(void)109 static inline void z_do_fp_regs_init(void)
110 {
111 __asm__ volatile("fninit\n\t");
112 }
113
114 /**
115 * @brief Initialize SSE register context information.
116 *
117 * This routine initializes the system's "live" SSE registers.
118 */
z_do_sse_regs_init(void)119 static inline void z_do_sse_regs_init(void)
120 {
121 __asm__ volatile("ldmxcsr _sse_mxcsr_default_value\n\t");
122 }
123
124 /*
125 * Save a thread's floating point context information.
126 *
127 * This routine saves the system's "live" floating point context into the
128 * specified thread control block. The SSE registers are saved only if the
129 * thread is actually using them.
130 */
FpCtxSave(struct k_thread * thread)131 static void FpCtxSave(struct k_thread *thread)
132 {
133 #ifdef CONFIG_X86_SSE
134 if ((thread->base.user_options & K_SSE_REGS) != 0) {
135 z_do_fp_and_sse_regs_save(&thread->arch.preempFloatReg);
136 return;
137 }
138 #endif
139 z_do_fp_regs_save(&thread->arch.preempFloatReg);
140 }
141
142 /*
143 * Initialize a thread's floating point context information.
144 *
145 * This routine initializes the system's "live" floating point context.
146 * The SSE registers are initialized only if the thread is actually using them.
147 */
FpCtxInit(struct k_thread * thread)148 static inline void FpCtxInit(struct k_thread *thread)
149 {
150 z_do_fp_regs_init();
151 #ifdef CONFIG_X86_SSE
152 if ((thread->base.user_options & K_SSE_REGS) != 0) {
153 z_do_sse_regs_init();
154 }
155 #endif
156 }
157
158 /*
159 * Enable preservation of floating point context information.
160 *
161 * The transition from "non-FP supporting" to "FP supporting" must be done
162 * atomically to avoid confusing the floating point logic used by z_swap(), so
163 * this routine locks interrupts to ensure that a context switch does not occur.
164 * The locking isn't really needed when the routine is called by a cooperative
165 * thread (since context switching can't occur), but it is harmless.
166 */
z_float_enable(struct k_thread * thread,unsigned int options)167 void z_float_enable(struct k_thread *thread, unsigned int options)
168 {
169 unsigned int imask;
170 struct k_thread *fp_owner;
171
172 if (!thread) {
173 return;
174 }
175
176 /* Ensure a preemptive context switch does not occur */
177
178 imask = irq_lock();
179
180 /* Indicate thread requires floating point context saving */
181
182 thread->base.user_options |= (uint8_t)options;
183 /*
184 * The current thread might not allow FP instructions, so clear CR0[TS]
185 * so we can use them. (CR0[TS] gets restored later on, if necessary.)
186 */
187
188 __asm__ volatile("clts\n\t");
189
190 /*
191 * Save existing floating point context (since it is about to change),
192 * but only if the FPU is "owned" by an FP-capable task that is
193 * currently handling an interrupt or exception (meaning its FP context
194 * must be preserved).
195 */
196
197 fp_owner = _kernel.current_fp;
198 if (fp_owner != NULL) {
199 if ((fp_owner->arch.flags & X86_THREAD_FLAG_ALL) != 0) {
200 FpCtxSave(fp_owner);
201 }
202 }
203
204 /* Now create a virgin FP context */
205
206 FpCtxInit(thread);
207
208 /* Associate the new FP context with the specified thread */
209
210 if (thread == arch_current_thread()) {
211 /*
212 * When enabling FP support for the current thread, just claim
213 * ownership of the FPU and leave CR0[TS] unset.
214 *
215 * (The FP context is "live" in hardware, not saved in TCS.)
216 */
217
218 _kernel.current_fp = thread;
219 } else {
220 /*
221 * When enabling FP support for someone else, assign ownership
222 * of the FPU to them (unless we need it ourselves).
223 */
224
225 if ((arch_current_thread()->base.user_options & _FP_USER_MASK) == 0) {
226 /*
227 * We are not FP-capable, so mark FPU as owned by the
228 * thread we've just enabled FP support for, then
229 * disable our own FP access by setting CR0[TS] back
230 * to its original state.
231 */
232
233 _kernel.current_fp = thread;
234 z_FpAccessDisable();
235 } else {
236 /*
237 * We are FP-capable (and thus had FPU ownership on
238 * entry), so save the new FP context in their TCS,
239 * leave FPU ownership with self, and leave CR0[TS]
240 * unset.
241 *
242 * The saved FP context is needed in case the thread
243 * we enabled FP support for is currently pre-empted,
244 * since z_swap() uses it to restore FP context when
245 * the thread re-activates.
246 *
247 * Saving the FP context reinits the FPU, and thus
248 * our own FP context, but that's OK since it didn't
249 * need to be preserved. (i.e. We aren't currently
250 * handling an interrupt or exception.)
251 */
252
253 FpCtxSave(thread);
254 }
255 }
256
257 irq_unlock(imask);
258 }
259
260 /**
261 * Disable preservation of floating point context information.
262 *
263 * The transition from "FP supporting" to "non-FP supporting" must be done
264 * atomically to avoid confusing the floating point logic used by z_swap(), so
265 * this routine locks interrupts to ensure that a context switch does not occur.
266 * The locking isn't really needed when the routine is called by a cooperative
267 * thread (since context switching can't occur), but it is harmless.
268 */
z_float_disable(struct k_thread * thread)269 int z_float_disable(struct k_thread *thread)
270 {
271 unsigned int imask;
272
273 /* Ensure a preemptive context switch does not occur */
274
275 imask = irq_lock();
276
277 /* Disable all floating point capabilities for the thread */
278
279 thread->base.user_options &= ~_FP_USER_MASK;
280
281 if (thread == arch_current_thread()) {
282 z_FpAccessDisable();
283 _kernel.current_fp = (struct k_thread *)0;
284 } else {
285 if (_kernel.current_fp == thread) {
286 _kernel.current_fp = (struct k_thread *)0;
287 }
288 }
289
290 irq_unlock(imask);
291
292 return 0;
293 }
294
295 /*
296 * Handler for "device not available" exception.
297 *
298 * This routine is registered to handle the "device not available" exception
299 * (vector = 7).
300 *
301 * The processor will generate this exception if any x87 FPU, MMX, or SSEx
302 * instruction is executed while CR0[TS]=1. The handler then enables the
303 * current thread to use all supported floating point registers.
304 */
_FpNotAvailableExcHandler(struct arch_esf * pEsf)305 void _FpNotAvailableExcHandler(struct arch_esf *pEsf)
306 {
307 ARG_UNUSED(pEsf);
308
309 /*
310 * Assume the exception did not occur in an ISR.
311 * (In other words, CPU cycles will not be consumed to perform
312 * error checking to ensure the exception was not generated in an ISR.)
313 */
314
315 /* Enable highest level of FP capability configured into the kernel */
316
317 k_float_enable(arch_current_thread(), _FP_USER_MASK);
318 }
319 _EXCEPTION_CONNECT_NOCODE(_FpNotAvailableExcHandler,
320 IV_DEVICE_NOT_AVAILABLE, 0);
321