-
Notifications
You must be signed in to change notification settings - Fork 8
/
piccolo_os.c
157 lines (135 loc) · 5.16 KB
/
piccolo_os.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
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
/*
* Copyright (C) 2021-2022 Gary Sims
* Copyright (C) 2022 Keith Standiford
* All rights reserved.
*
* Portions copyright (C) 2017 Scott Nelson
* Portions copyright (C) 2015-2018 National Cheng Kung University, Taiwan
* Portions copyright (C) 2014-2017 Chris Stones
*
* SPDX-License-Identifier: BSD-3-Clause
*/
#include "pico/stdlib.h"
#include <stdio.h>
//#include "hardware/clocks.h"
//#include "hardware/structs/clocks.h"
#include "hardware/structs/systick.h"
#include "hardware/sync.h"
#include "piccolo_os.h"
unsigned int *__piccolo_os_create_task(unsigned int *stack,
void (*start)(void));
void __piccolo_task_init(void);
unsigned int *__piccolo_pre_switch(unsigned int *stack);
void __piccolo_task_init_stack(unsigned int *stack);
piccolo_os_internals_t piccolo_ctx;
/* Initialize user task stack and execute it once only
* We set THREAD_PSP to LR so that exception return
* works correctly.
* THREAD_PSP means: Return to Thread mode.
* Exception return gets state from the process stack.
* Execution uses PSP after return.
* See:
* http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.dui0552a/Babefdjc.html
*/
unsigned int *__piccolo_os_create_task(unsigned int *task_stack,
void (*pointer_to_task_function)(void)) {
/* This task_stack frame needs to mimic would be saved by hardware and by the
* software (in isr_svcall) */
/*
Exception frame saved by the hardware onto task_stack:
+------+
| xPSR | task_stack[16] 0x01000000 i.e. PSR Thumb bit
| PC | task_stack[15] pointer_to_task_function
| LR | task_stack[14]
| R12 | task_stack[13]
| R3 | task_stack[12]
| R2 | task_stack[11]
| R1 | task_stack[10]
| R0 | task_stack[9]
+------+
Registers saved by the software (isr_svcall):
+------+
| LR | task_stack[8] (THREAD_PSP i.e. 0xFFFFFFFD)
| R7 | task_stack[7]
| R6 | task_stack[6]
| R5 | task_stack[5]
| R4 | task_stack[4]
| R11 | task_stack[3]
| R10 | task_stack[2]
| R9 | task_stack[1]
| R8 | task_stack[0]
+------+
*/
task_stack += PICCOLO_OS_STACK_SIZE - 17; /* End of task_stack, minus what we are about to push */
task_stack[8] = (unsigned int) PICCOLO_OS_THREAD_PSP;
task_stack[15] = (unsigned int) pointer_to_task_function;
task_stack[16] = (unsigned int) 0x01000000; /* PSR Thumb bit */
//task_stack = __piccolo_pre_switch(task_stack);
return task_stack;
}
int piccolo_create_task(void (*pointer_to_task_function)(void)) {
if (piccolo_ctx.task_count >= PICCOLO_OS_TASK_LIMIT)
return -1;
int tc = piccolo_ctx.task_count; // Just for readability
piccolo_ctx.the_tasks[tc] =
__piccolo_os_create_task(piccolo_ctx.task_stacks[tc], pointer_to_task_function);
piccolo_ctx.task_count++;
return piccolo_ctx.task_count - 1;
}
void piccolo_sleep(piccolo_sleep_t *start, int ticks) {
*start = to_ms_since_boot(get_absolute_time());
while (to_ms_since_boot(get_absolute_time()) < *start + ticks) {
piccolo_yield();
}
}
void piccolo_sleep_ms(int ticks) {
piccolo_sleep_t start;
start = to_ms_since_boot(get_absolute_time());
while (to_ms_since_boot(get_absolute_time()) < start + ticks) {
piccolo_yield();
}
}
/* After a reset, processor is in thread mode
* Switch to handler mode, so that the Piccolo OS kernel runs in handler mode,
* and to ensure the correct return from an exception/interrupt later
* when switching to a task
*/
void __piccolo_task_init(void) {
unsigned int dummy[32];
__piccolo_task_init_stack(dummy + 32);
}
void piccolo_init() {
piccolo_ctx.task_count = 0;
stdio_init_all();
/*
* set interrupt priority for SVC, PENDSV and Systick to 'all bits on'
* for LOWEST interrupt priority. We do not want ANY of them to preempt
* any other irq or each other.
*/
hw_set_bits((io_rw_32 *)(PPB_BASE + M0PLUS_SHPR2_OFFSET), M0PLUS_SHPR2_BITS);
hw_set_bits((io_rw_32 *)(PPB_BASE + M0PLUS_SHPR3_OFFSET), M0PLUS_SHPR3_BITS);
}
void __piccolo_systick_config(unsigned int n) {
/* Stop systick and cancel it if it is pending */
systick_hw->csr = 0; // Disable timer and IRQ
__dsb(); // make sure systick is disabled
__isb(); // and it is really off
// clear the systick exception pending bit if it got set
hw_set_bits ((io_rw_32 *)(PPB_BASE + M0PLUS_ICSR_OFFSET),M0PLUS_ICSR_PENDSTCLR_BITS);
systick_hw->rvr = (n) - 1UL; // set the reload value
systick_hw->cvr = 0; // clear counter to force reload
systick_hw->csr = 0x03; // arm IRQ, start counter with 1 usec clock
}
void piccolo_start() {
piccolo_ctx.current_task = 0;
__piccolo_task_init();
while (1) {
__piccolo_systick_config(PICCOLO_OS_TIME_SLICE);
piccolo_ctx.the_tasks[piccolo_ctx.current_task] =
__piccolo_pre_switch(piccolo_ctx.the_tasks[piccolo_ctx.current_task]);
piccolo_ctx.current_task++;
if (piccolo_ctx.current_task >=
piccolo_ctx.task_count)
piccolo_ctx.current_task = 0;
}
}