1 /*
2 * Copyright (c) 2019-2020 Cobham Gaisler AB
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #include <zephyr/kernel.h>
8 #include <ksched.h>
9
10 void z_thread_entry_wrapper(k_thread_entry_t thread,
11 void *arg1,
12 void *arg2,
13 void *arg3);
14
15 /*
16 * Frame used by _thread_entry_wrapper
17 *
18 * Allocate a 16 register window save area at bottom of the stack. This is
19 * required if we need to taken a trap (interrupt) in the thread entry wrapper.
20 */
21 struct init_stack_frame {
22 uint32_t window_save_area[16];
23 };
24
25 #if defined(CONFIG_FPU_SHARING)
26 #define USER_FP_MASK K_FP_REGS
27 #else
28 #define USER_FP_MASK 0
29 #endif
30
arch_new_thread(struct k_thread * thread,k_thread_stack_t * stack,char * stack_ptr,k_thread_entry_t entry,void * p1,void * p2,void * p3)31 void arch_new_thread(struct k_thread *thread, k_thread_stack_t *stack,
32 char *stack_ptr, k_thread_entry_t entry,
33 void *p1, void *p2, void *p3)
34 {
35 struct init_stack_frame *iframe;
36
37 /* Initial stack frame data, stored at base of the stack */
38 iframe = Z_STACK_PTR_TO_FRAME(struct init_stack_frame, stack_ptr);
39
40 thread->callee_saved.i0 = (uint32_t) entry;
41 thread->callee_saved.i1 = (uint32_t) p1;
42 thread->callee_saved.i2 = (uint32_t) p2;
43 thread->callee_saved.i3 = (uint32_t) p3;
44 thread->callee_saved.i6 = 0; /* frame pointer */
45 thread->callee_saved.o6 = (uint32_t) iframe; /* stack pointer */
46 thread->callee_saved.o7 = (uint32_t) z_thread_entry_wrapper - 8;
47 thread->callee_saved.psr = PSR_S | PSR_PS | PSR_ET;
48
49 if (IS_ENABLED(CONFIG_FPU_SHARING)) {
50 /* Selected threads can use the FPU */
51 if (thread->base.user_options & USER_FP_MASK) {
52 thread->callee_saved.psr |= PSR_EF;
53 }
54 } else if (IS_ENABLED(CONFIG_FPU)) {
55 /* Any thread can use the FPU */
56 thread->callee_saved.psr |= PSR_EF;
57 }
58
59 thread->switch_handle = thread;
60 }
61
z_arch_get_next_switch_handle(struct k_thread ** old_thread)62 void *z_arch_get_next_switch_handle(struct k_thread **old_thread)
63 {
64 *old_thread = arch_current_thread();
65
66 return z_get_next_switch_handle(*old_thread);
67 }
68
69 #if defined(CONFIG_FPU_SHARING)
arch_float_disable(struct k_thread * thread)70 int arch_float_disable(struct k_thread *thread)
71 {
72 return -ENOTSUP;
73 }
74
arch_float_enable(struct k_thread * thread,unsigned int options)75 int arch_float_enable(struct k_thread *thread, unsigned int options)
76 {
77 return -ENOTSUP;
78 }
79 #endif /* CONFIG_FPU_SHARING */
80