1 /*
2 * FreeRTOS Kernel V11.1.0
3 * Copyright (C) 2021 Amazon.com, Inc. or its affiliates. All Rights Reserved.
4 *
5 * SPDX-License-Identifier: MIT
6 *
7 * Permission is hereby granted, free of charge, to any person obtaining a copy of
8 * this software and associated documentation files (the "Software"), to deal in
9 * the Software without restriction, including without limitation the rights to
10 * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
11 * the Software, and to permit persons to whom the Software is furnished to do so,
12 * subject to the following conditions:
13 *
14 * The above copyright notice and this permission notice shall be included in all
15 * copies or substantial portions of the Software.
16 *
17 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
19 * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
20 * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
21 * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
22 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
23 *
24 * https://www.FreeRTOS.org
25 * https://github.com/FreeRTOS
26 *
27 */
28
29 /*-----------------------------------------------------------
30 * Implementation of functions defined in portable.h for the RXv3 DPFPU port.
31 *----------------------------------------------------------*/
32
33 #warning Testing for DFPU support in this port is not yet complete
34
35 /* Scheduler includes. */
36 #include "FreeRTOS.h"
37 #include "task.h"
38
39 /* Library includes. */
40 #include "string.h"
41
42 /* Hardware specifics. */
43 #if ( configINCLUDE_PLATFORM_H_INSTEAD_OF_IODEFINE_H == 1 )
44
45 #include "platform.h"
46
47 #else /* configINCLUDE_PLATFORM_H_INSTEAD_OF_IODEFINE_H */
48
49 #include "iodefine.h"
50
51 #endif /* configINCLUDE_PLATFORM_H_INSTEAD_OF_IODEFINE_H */
52
53 /*-----------------------------------------------------------*/
54
55 /* Tasks should start with interrupts enabled and in Supervisor mode, therefore
56 * PSW is set with U and I set, and PM and IPL clear. */
57 #define portINITIAL_PSW ( ( StackType_t ) 0x00030000 )
58 #define portINITIAL_FPSW ( ( StackType_t ) 0x00000100 )
59 #define portINITIAL_DPSW ( ( StackType_t ) 0x00000100 )
60 #define portINITIAL_DCMR ( ( StackType_t ) 0x00000000 )
61 #define portINITIAL_DECNT ( ( StackType_t ) 0x00000001 )
62
63 /* Tasks are not created with a DPFPU context, but can be given a DPFPU context
64 * after they have been created. A variable is stored as part of the tasks context
65 * that holds portNO_DPFPU_CONTEXT if the task does not have a DPFPU context, or
66 * any other value if the task does have a DPFPU context. */
67 #define portNO_DPFPU_CONTEXT ( ( StackType_t ) 0 )
68 #define portHAS_DPFPU_CONTEXT ( ( StackType_t ) 1 )
69
70 /* The space on the stack required to hold the DPFPU data registers. This is 16
71 * 64-bit registers. */
72 #define portDPFPU_DATA_REGISTER_WORDS ( 16 * 2 )
73
74 /*-----------------------------------------------------------*/
75
76 /* The following lines are to ensure vSoftwareInterruptEntry can be referenced,
77 * and therefore installed in the vector table, when the FreeRTOS code is built
78 * as a library. */
79 extern BaseType_t vSoftwareInterruptEntry;
80 const BaseType_t * p_vSoftwareInterruptEntry = &vSoftwareInterruptEntry;
81
82 /*-----------------------------------------------------------*/
83
84 /*
85 * Function to start the first task executing - written in asm code as direct
86 * access to registers is required.
87 */
88 static void prvStartFirstTask( void );
89
90 /*
91 * Software interrupt handler. Performs the actual context switch (saving and
92 * restoring of registers). Written in asm code as direct register access is
93 * required.
94 */
95 static void prvYieldHandler( void );
96
97 /*
98 * The entry point for the software interrupt handler. This is the function
99 * that calls the inline asm function prvYieldHandler(). It is installed in
100 * the vector table, but the code that installs it is in prvYieldHandler rather
101 * than using a #pragma.
102 */
103 void vSoftwareInterruptISR( void );
104
105 /*
106 * The tick ISR handler. The peripheral used is configured by the application
107 * via a hook/callback function.
108 */
109 void vTickISR( void );
110
111 /*-----------------------------------------------------------*/
112
113 /* Saved as part of the task context. If ulPortTaskHasDPFPUContext is non-zero
114 * then a DPFPU context must be saved and restored for the task. */
115 #if ( configUSE_TASK_DPFPU_SUPPORT == 1 )
116
117 StackType_t ulPortTaskHasDPFPUContext = portNO_DPFPU_CONTEXT;
118
119 #endif /* configUSE_TASK_DPFPU_SUPPORT */
120
121 /* This is accessed by the inline assembler functions so is file scope for
122 * convenience. */
123 extern void * pxCurrentTCB;
124 extern void vTaskSwitchContext( void );
125
126 /*-----------------------------------------------------------*/
127
128 /*
129 * See header file for description.
130 */
pxPortInitialiseStack(StackType_t * pxTopOfStack,TaskFunction_t pxCode,void * pvParameters)131 StackType_t * pxPortInitialiseStack( StackType_t * pxTopOfStack,
132 TaskFunction_t pxCode,
133 void * pvParameters )
134 {
135 /* R0 is not included as it is the stack pointer. */
136
137 *pxTopOfStack = 0x00;
138 pxTopOfStack--;
139 *pxTopOfStack = portINITIAL_PSW;
140 pxTopOfStack--;
141 *pxTopOfStack = ( StackType_t ) pxCode;
142
143 /* When debugging it can be useful if every register is set to a known
144 * value. Otherwise code space can be saved by just setting the registers
145 * that need to be set. */
146 #ifdef USE_FULL_REGISTER_INITIALISATION
147 {
148 pxTopOfStack--;
149 *pxTopOfStack = 0xffffffff; /* r15. */
150 pxTopOfStack--;
151 *pxTopOfStack = 0xeeeeeeee;
152 pxTopOfStack--;
153 *pxTopOfStack = 0xdddddddd;
154 pxTopOfStack--;
155 *pxTopOfStack = 0xcccccccc;
156 pxTopOfStack--;
157 *pxTopOfStack = 0xbbbbbbbb;
158 pxTopOfStack--;
159 *pxTopOfStack = 0xaaaaaaaa;
160 pxTopOfStack--;
161 *pxTopOfStack = 0x99999999;
162 pxTopOfStack--;
163 *pxTopOfStack = 0x88888888;
164 pxTopOfStack--;
165 *pxTopOfStack = 0x77777777;
166 pxTopOfStack--;
167 *pxTopOfStack = 0x66666666;
168 pxTopOfStack--;
169 *pxTopOfStack = 0x55555555;
170 pxTopOfStack--;
171 *pxTopOfStack = 0x44444444;
172 pxTopOfStack--;
173 *pxTopOfStack = 0x33333333;
174 pxTopOfStack--;
175 *pxTopOfStack = 0x22222222;
176 pxTopOfStack--;
177 }
178 #else /* ifdef USE_FULL_REGISTER_INITIALISATION */
179 {
180 pxTopOfStack -= 15;
181 }
182 #endif /* ifdef USE_FULL_REGISTER_INITIALISATION */
183
184 *pxTopOfStack = ( StackType_t ) pvParameters; /* R1 */
185 pxTopOfStack--;
186 *pxTopOfStack = portINITIAL_FPSW;
187 pxTopOfStack--;
188 *pxTopOfStack = 0x11111111; /* Accumulator 1. */
189 pxTopOfStack--;
190 *pxTopOfStack = 0x22222222; /* Accumulator 1. */
191 pxTopOfStack--;
192 *pxTopOfStack = 0x33333333; /* Accumulator 1. */
193 pxTopOfStack--;
194 *pxTopOfStack = 0x44444444; /* Accumulator 0. */
195 pxTopOfStack--;
196 *pxTopOfStack = 0x55555555; /* Accumulator 0. */
197 pxTopOfStack--;
198 *pxTopOfStack = 0x66666666; /* Accumulator 0. */
199
200 #if ( configUSE_TASK_DPFPU_SUPPORT == 1 )
201 {
202 /* The task will start without a DPFPU context. A task that
203 * uses the DPFPU hardware must call vPortTaskUsesDPFPU() before
204 * executing any floating point instructions. */
205 pxTopOfStack--;
206 *pxTopOfStack = portNO_DPFPU_CONTEXT;
207 }
208 #elif ( configUSE_TASK_DPFPU_SUPPORT == 2 )
209 {
210 /* The task will start with a DPFPU context. Leave enough
211 * space for the registers - and ensure they are initialised if desired. */
212 #ifdef USE_FULL_REGISTER_INITIALISATION
213 {
214 pxTopOfStack -= 2;
215 *(double *)pxTopOfStack = 1515.1515; /* DR15. */
216 pxTopOfStack -= 2;
217 *(double *)pxTopOfStack = 1414.1414; /* DR14. */
218 pxTopOfStack -= 2;
219 *(double *)pxTopOfStack = 1313.1313; /* DR13. */
220 pxTopOfStack -= 2;
221 *(double *)pxTopOfStack = 1212.1212; /* DR12. */
222 pxTopOfStack -= 2;
223 *(double *)pxTopOfStack = 1111.1111; /* DR11. */
224 pxTopOfStack -= 2;
225 *(double *)pxTopOfStack = 1010.1010; /* DR10. */
226 pxTopOfStack -= 2;
227 *(double *)pxTopOfStack = 909.0909; /* DR9. */
228 pxTopOfStack -= 2;
229 *(double *)pxTopOfStack = 808.0808; /* DR8. */
230 pxTopOfStack -= 2;
231 *(double *)pxTopOfStack = 707.0707; /* DR7. */
232 pxTopOfStack -= 2;
233 *(double *)pxTopOfStack = 606.0606; /* DR6. */
234 pxTopOfStack -= 2;
235 *(double *)pxTopOfStack = 505.0505; /* DR5. */
236 pxTopOfStack -= 2;
237 *(double *)pxTopOfStack = 404.0404; /* DR4. */
238 pxTopOfStack -= 2;
239 *(double *)pxTopOfStack = 303.0303; /* DR3. */
240 pxTopOfStack -= 2;
241 *(double *)pxTopOfStack = 202.0202; /* DR2. */
242 pxTopOfStack -= 2;
243 *(double *)pxTopOfStack = 101.0101; /* DR1. */
244 pxTopOfStack -= 2;
245 *(double *)pxTopOfStack = 9876.54321;/* DR0. */
246 }
247 #else /* ifdef USE_FULL_REGISTER_INITIALISATION */
248 {
249 pxTopOfStack -= portDPFPU_DATA_REGISTER_WORDS;
250 memset( pxTopOfStack, 0x00, portDPFPU_DATA_REGISTER_WORDS * sizeof( StackType_t ) );
251 }
252 #endif /* ifdef USE_FULL_REGISTER_INITIALISATION */
253 pxTopOfStack--;
254 *pxTopOfStack = portINITIAL_DECNT; /* DECNT. */
255 pxTopOfStack--;
256 *pxTopOfStack = portINITIAL_DCMR; /* DCMR. */
257 pxTopOfStack--;
258 *pxTopOfStack = portINITIAL_DPSW; /* DPSW. */
259 }
260 #elif ( configUSE_TASK_DPFPU_SUPPORT == 0 )
261 {
262 /* Omit DPFPU support. */
263 }
264 #else /* if ( configUSE_TASK_DPFPU_SUPPORT == 1 ) */
265 {
266 #error Invalid configUSE_TASK_DPFPU_SUPPORT setting - configUSE_TASK_DPFPU_SUPPORT must be set to 0, 1, 2, or left undefined.
267 }
268 #endif /* if ( configUSE_TASK_DPFPU_SUPPORT == 1 ) */
269
270 return pxTopOfStack;
271 }
272 /*-----------------------------------------------------------*/
273
274 #if ( configUSE_TASK_DPFPU_SUPPORT == 1 )
275
vPortTaskUsesDPFPU(void)276 void vPortTaskUsesDPFPU( void )
277 {
278 /* A task is registering the fact that it needs a DPFPU context. Set the
279 * DPFPU flag (which is saved as part of the task context). */
280 ulPortTaskHasDPFPUContext = portHAS_DPFPU_CONTEXT;
281 }
282
283 #endif /* configUSE_TASK_DPFPU_SUPPORT */
284 /*-----------------------------------------------------------*/
285
xPortStartScheduler(void)286 BaseType_t xPortStartScheduler( void )
287 {
288 extern void vApplicationSetupTimerInterrupt( void );
289
290 /* Use pxCurrentTCB just so it does not get optimised away. */
291 if( pxCurrentTCB != NULL )
292 {
293 /* Call an application function to set up the timer that will generate the
294 * tick interrupt. This way the application can decide which peripheral to
295 * use. A demo application is provided to show a suitable example. */
296 vApplicationSetupTimerInterrupt();
297
298 /* Enable the software interrupt. */
299 _IEN( _ICU_SWINT ) = 1;
300
301 /* Ensure the software interrupt is clear. */
302 _IR( _ICU_SWINT ) = 0;
303
304 /* Ensure the software interrupt is set to the kernel priority. */
305 _IPR( _ICU_SWINT ) = configKERNEL_INTERRUPT_PRIORITY;
306
307 /* Start the first task. */
308 prvStartFirstTask();
309 }
310
311 /* Just to make sure the function is not optimised away. */
312 ( void ) vSoftwareInterruptISR();
313
314 /* Should not get here. */
315 return pdFAIL;
316 }
317 /*-----------------------------------------------------------*/
318
vPortEndScheduler(void)319 void vPortEndScheduler( void )
320 {
321 /* Not implemented in ports where there is nothing to return to.
322 * Artificially force an assert. */
323 configASSERT( pxCurrentTCB == NULL );
324
325 /* The following line is just to prevent the symbol getting optimised away. */
326 ( void ) vTaskSwitchContext();
327 }
328 /*-----------------------------------------------------------*/
329
330 #pragma inline_asm prvStartFirstTask
prvStartFirstTask(void)331 static void prvStartFirstTask( void )
332 {
333 #ifndef __CDT_PARSER__
334
335 /* When starting the scheduler there is nothing that needs moving to the
336 * interrupt stack because the function is not called from an interrupt.
337 * Just ensure the current stack is the user stack. */
338 SETPSW U
339
340
341 /* Obtain the location of the stack associated with which ever task
342 * pxCurrentTCB is currently pointing to. */
343 MOV.L # _pxCurrentTCB, R15
344 MOV.L [ R15 ], R15
345 MOV.L [ R15 ], R0
346
347
348 /* Restore the registers from the stack of the task pointed to by
349 * pxCurrentTCB. */
350
351 #if ( configUSE_TASK_DPFPU_SUPPORT == 1 )
352
353 /* The restored ulPortTaskHasDPFPUContext is to be zero here.
354 * So, it is never necessary to restore the DPFPU context here. */
355 POP R15
356 MOV.L # _ulPortTaskHasDPFPUContext, R14
357 MOV.L R15, [ R14 ]
358
359 #elif ( configUSE_TASK_DPFPU_SUPPORT == 2 )
360
361 /* Restore the DPFPU context. */
362 DPOPM.L DPSW-DECNT
363 DPOPM.D DR0-DR15
364
365 #endif /* if ( configUSE_TASK_DPFPU_SUPPORT == 1 ) */
366
367 POP R15
368
369 /* Accumulator low 32 bits. */
370 MVTACLO R15, A0
371 POP R15
372
373 /* Accumulator high 32 bits. */
374 MVTACHI R15, A0
375 POP R15
376
377 /* Accumulator guard. */
378 MVTACGU R15, A0
379 POP R15
380
381 /* Accumulator low 32 bits. */
382 MVTACLO R15, A1
383 POP R15
384
385 /* Accumulator high 32 bits. */
386 MVTACHI R15, A1
387 POP R15
388
389 /* Accumulator guard. */
390 MVTACGU R15, A1
391 POP R15
392
393 /* Floating point status word. */
394 MVTC R15, FPSW
395
396 /* R1 to R15 - R0 is not included as it is the SP. */
397 POPM R1-R15
398
399 /* This pops the remaining registers. */
400 RTE
401 NOP
402 NOP
403
404 #endif /* ifndef __CDT_PARSER__ */
405 }
406 /*-----------------------------------------------------------*/
407
vSoftwareInterruptISR(void)408 void vSoftwareInterruptISR( void )
409 {
410 prvYieldHandler();
411 }
412 /*-----------------------------------------------------------*/
413
414 #pragma inline_asm prvYieldHandler
prvYieldHandler(void)415 static void prvYieldHandler( void )
416 {
417 #ifndef __CDT_PARSER__
418
419 /* Re-enable interrupts. */
420 SETPSW I
421
422
423 /* Move the data that was automatically pushed onto the interrupt stack when
424 * the interrupt occurred from the interrupt stack to the user stack.
425 *
426 * R15 is saved before it is clobbered. */
427 PUSH.L R15
428
429 /* Read the user stack pointer. */
430 MVFC USP, R15
431
432 /* Move the address down to the data being moved. */
433 SUB # 12, R15
434 MVTC R15, USP
435
436 /* Copy the data across, R15, then PC, then PSW. */
437 MOV.L [ R0 ], [ R15 ]
438 MOV.L 4[ R0 ], 4[ R15 ]
439 MOV.L 8[ R0 ], 8[ R15 ]
440
441 /* Move the interrupt stack pointer to its new correct position. */
442 ADD # 12, R0
443
444 /* All the rest of the registers are saved directly to the user stack. */
445 SETPSW U
446
447 /* Save the rest of the general registers (R15 has been saved already). */
448 PUSHM R1-R14
449
450 /* Save the FPSW and accumulators. */
451 MVFC FPSW, R15
452 PUSH.L R15
453 MVFACGU # 0, A1, R15
454 PUSH.L R15
455 MVFACHI # 0, A1, R15
456 PUSH.L R15
457 MVFACLO # 0, A1, R15 /* Low order word. */
458 PUSH.L R15
459 MVFACGU # 0, A0, R15
460 PUSH.L R15
461 MVFACHI # 0, A0, R15
462 PUSH.L R15
463 MVFACLO # 0, A0, R15 /* Low order word. */
464 PUSH.L R15
465
466 #if ( configUSE_TASK_DPFPU_SUPPORT == 1 )
467
468 /* Does the task have a DPFPU context that needs saving? If
469 * ulPortTaskHasDPFPUContext is 0 then no. */
470 MOV.L # _ulPortTaskHasDPFPUContext, R15
471 MOV.L [ R15 ], R15
472 CMP # 0, R15
473
474 /* Save the DPFPU context, if any. */
475 BEQ.B ?+
476 DPUSHM.D DR0-DR15
477 DPUSHM.L DPSW-DECNT
478 ?:
479
480 /* Save ulPortTaskHasDPFPUContext itself. */
481 PUSH.L R15
482
483 #elif ( configUSE_TASK_DPFPU_SUPPORT == 2 )
484
485 /* Save the DPFPU context, always. */
486 DPUSHM.D DR0-DR15
487 DPUSHM.L DPSW-DECNT
488
489 #endif /* if ( configUSE_TASK_DPFPU_SUPPORT == 1 ) */
490
491
492 /* Save the stack pointer to the TCB. */
493 MOV.L # _pxCurrentTCB, R15
494 MOV.L [ R15 ], R15
495 MOV.L R0, [ R15 ]
496
497
498 /* Ensure the interrupt mask is set to the syscall priority while the kernel
499 * structures are being accessed. */
500 MVTIPL # configMAX_SYSCALL_INTERRUPT_PRIORITY
501
502 /* Select the next task to run. */
503 BSR.A _vTaskSwitchContext
504
505 /* Reset the interrupt mask as no more data structure access is required. */
506 MVTIPL # configKERNEL_INTERRUPT_PRIORITY
507
508
509 /* Load the stack pointer of the task that is now selected as the Running
510 * state task from its TCB. */
511 MOV.L # _pxCurrentTCB, R15
512 MOV.L [ R15 ], R15
513 MOV.L [ R15 ], R0
514
515
516 /* Restore the context of the new task. The PSW (Program Status Word) and
517 * PC will be popped by the RTE instruction. */
518
519 #if ( configUSE_TASK_DPFPU_SUPPORT == 1 )
520
521 /* Is there a DPFPU context to restore? If the restored
522 * ulPortTaskHasDPFPUContext is zero then no. */
523 POP R15
524 MOV.L # _ulPortTaskHasDPFPUContext, R14
525 MOV.L R15, [ R14 ]
526 CMP # 0, R15
527
528 /* Restore the DPFPU context, if any. */
529 BEQ.B ?+
530 DPOPM.L DPSW-DECNT
531 DPOPM.D DR0-DR15
532 ?:
533
534 #elif ( configUSE_TASK_DPFPU_SUPPORT == 2 )
535
536 /* Restore the DPFPU context, always. */
537 DPOPM.L DPSW-DECNT
538 DPOPM.D DR0-DR15
539
540 #endif /* if ( configUSE_TASK_DPFPU_SUPPORT == 1 ) */
541
542 POP R15
543
544 /* Accumulator low 32 bits. */
545 MVTACLO R15, A0
546 POP R15
547
548 /* Accumulator high 32 bits. */
549 MVTACHI R15, A0
550 POP R15
551
552 /* Accumulator guard. */
553 MVTACGU R15, A0
554 POP R15
555
556 /* Accumulator low 32 bits. */
557 MVTACLO R15, A1
558 POP R15
559
560 /* Accumulator high 32 bits. */
561 MVTACHI R15, A1
562 POP R15
563
564 /* Accumulator guard. */
565 MVTACGU R15, A1
566 POP R15
567 MVTC R15, FPSW
568 POPM R1-R15
569 RTE
570 NOP
571 NOP
572
573 #endif /* ifndef __CDT_PARSER__ */
574 }
575 /*-----------------------------------------------------------*/
576
577 #pragma interrupt ( vTickISR( vect = _VECT( configTICK_VECTOR ), enable ) )
vTickISR(void)578 void vTickISR( void )
579 {
580 /* Increment the tick, and perform any processing the new tick value
581 * necessitates. Ensure IPL is at the max syscall value first. */
582 set_ipl( configMAX_SYSCALL_INTERRUPT_PRIORITY );
583 {
584 if( xTaskIncrementTick() != pdFALSE )
585 {
586 taskYIELD();
587 }
588 }
589 set_ipl( configKERNEL_INTERRUPT_PRIORITY );
590 }
591 /*-----------------------------------------------------------*/
592