xtr.cpp 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. #include "all-headers.h"
  2. #define ENABLE_XTR_ASSERTIONS
  3. #ifdef ENABLE_XTR_ASSERTIONS
  4. #define XTR_ASSERT_NON_NULL_POINTER(ptr) assertNonNullPointer (ptr, __FILE__, __LINE__) ;
  5. #else
  6. #define XTR_ASSERT_NON_NULL_POINTER(ptr)
  7. #endif
  8. #ifdef ENABLE_XTR_ASSERTIONS
  9. #define XTR_ASSERT(condition,value) assertion (condition, value, __FILE__, __LINE__) ;
  10. #else
  11. #define XTR_ASSERT(condition,value)
  12. #endif
  13. // CORTEX-M4 EXCEPTION FRAME (WITHOUT FLOATING-POINT STORAGE) *
  14. // *
  15. // PSP+32 -> | | *
  16. // |----------------------------| \ *
  17. // PSP+28 -> | xPSR | | *
  18. // |----------------------------| | *
  19. // PSP+24 -> | PC (after SVC instruction) | | *
  20. // |----------------------------| | *
  21. // PSP+20 -> | LR | | *
  22. // |----------------------------| | *
  23. // PSP+16 -> | R12 | | Saved by interrupt response *
  24. // |----------------------------| | *
  25. // PSP+12 -> | R3 | | *
  26. // |----------------------------| | *
  27. // PSP+8 -> | R2 | | *
  28. // |----------------------------| | *
  29. // PSP+4 -> | R1 | | *
  30. // |----------------------------| | *
  31. // /---- PSP ----> | R0 | | *
  32. // | |----------------------------| / *
  33. // | *
  34. // | *---------------------*
  35. // | | LR return code | +36 [ 9]
  36. // | *---------------------*
  37. // \-----------------------------------------------| R13 (PSP) | +32 [ 8]
  38. // *---------------------*
  39. // | R11 | +28 [ 7]
  40. // *---------------------*
  41. // | R10 | +24 [ 6]
  42. // *---------------------*
  43. // | R9 | +20 [ 5]
  44. // *---------------------*
  45. // | R8 | +16 [ 4]
  46. // *---------------------*
  47. // | R7 | +12 [ 3]
  48. // *---------------------*
  49. // | R6 | + 8 [ 2]
  50. // *---------------------*
  51. // | R5 | + 4 [ 1]
  52. // *---------------------------------------* *---------------------*
  53. // | var.running.task.control.block.ptr +------> | R4 | + 0 [ 0]
  54. // *---------------------------------------* *---------------------*
  55. //
  56. //----------------------------------------------------------------------------------------------------------------------
  57. typedef struct {
  58. uint32_t mR0 ;
  59. uint32_t mR1 ;
  60. uint32_t mR2 ;
  61. uint32_t mR3 ;
  62. uint32_t mR12 ;
  63. uint32_t mLR ;
  64. uint32_t mPC ;
  65. uint32_t mXPSR ;
  66. } ExceptionFrame_without_floatingPointStorage ;
  67. typedef struct {
  68. uint32_t mR4 ;
  69. uint32_t mR5 ;
  70. uint32_t mR6 ;
  71. uint32_t mR7 ;
  72. uint32_t mR8 ;
  73. uint32_t mR9 ;
  74. uint32_t mR10 ;
  75. uint32_t mR11 ;
  76. ExceptionFrame_without_floatingPointStorage * mPSP ; // R13
  77. uint32_t mLR_RETURN_CODE ;
  78. } TaskContext ;
  79. static void kernel_set_task_context (INIT_MODE_
  80. TaskContext & ioTaskContext,
  81. const uint32_t inStackBufferAddress,
  82. const uint32_t inStackBufferSize,
  83. RoutineTaskType inTaskRoutine) {
  84. // Initialize LR
  85. ioTaskContext.mLR_RETURN_CODE = 0xFFFFFFFD ;
  86. // Stack Pointer initial value
  87. uint32_t initialTopOfStack = inStackBufferAddress + inStackBufferSize ;
  88. initialTopOfStack -= sizeof (ExceptionFrame_without_floatingPointStorage) ;
  89. // Initialize SP
  90. auto ptr = (ExceptionFrame_without_floatingPointStorage *) initialTopOfStack ;
  91. ioTaskContext.mPSP = ptr ;
  92. // Initialize PC
  93. ptr->mPC = (uint32_t) inTaskRoutine ;
  94. // Initialize CPSR
  95. ptr->mXPSR = 1 << 24 ; // Thumb bit
  96. }
  97. // T A S K C O N T R O L B L O C K *
  98. typedef struct TaskControlBlock {
  99. // Context buffer
  100. TaskContext mTaskContext ; // SHOULD BE THE FIRST FIELD
  101. // Task index
  102. uint8_t mTaskIndex ;
  103. } TaskControlBlock ;
  104. static TaskControlBlock gTaskDescriptorArray [TASK_COUNT] ;
  105. TaskControlBlock * descriptorPointerForTaskIndex (const uint8_t inTaskIndex) {
  106. XTR_ASSERT (inTaskIndex < TASK_COUNT, inTaskIndex) ;
  107. return & gTaskDescriptorArray [inTaskIndex] ;
  108. }
  109. uint8_t indexForDescriptorTask (const TaskControlBlock * inTaskPtr) { // should be not nullptr
  110. XTR_ASSERT_NON_NULL_POINTER (inTaskPtr) ;
  111. return inTaskPtr->mTaskIndex ;
  112. }
  113. // RUNNING TASK DESCRIPTOR POINTER
  114. TaskControlBlock * gRunningTaskControlBlockPtr asm ("var.running.task.control.block.ptr") ; // Shared with assembly code
  115. // SCHEDULER
  116. static TaskList gReadyTaskList ;
  117. static void kernel_makeTaskReady (IRQ_MODE_ TaskControlBlock * inTaskPtr) {
  118. XTR_ASSERT_NON_NULL_POINTER (inTaskPtr) ;
  119. gReadyTaskList.enterTask (MODE_ inTaskPtr) ;
  120. }
  121. void kernelSelectTaskToRun (IRQ_MODE) asm ("kernel.select.task.to.run") ; // Invoked from assembly
  122. void kernelSelectTaskToRun (IRQ_MODE) {
  123. if (gRunningTaskControlBlockPtr != nullptr) {
  124. gReadyTaskList.enterTask (MODE_ gRunningTaskControlBlockPtr) ;
  125. }
  126. gRunningTaskControlBlockPtr = gReadyTaskList.removeFirstTask (MODE) ;
  127. }
  128. // kernel_createTask
  129. static uint8_t gTaskIndex ;
  130. void kernel_createTask (INIT_MODE_
  131. uint64_t * inStackBufferAddress,
  132. uint32_t inStackBufferSize,
  133. RoutineTaskType inTaskRoutine) {
  134. XTR_ASSERT (gTaskIndex < TASK_COUNT, gTaskIndex) ;
  135. TaskControlBlock * taskControlBlockPtr = & gTaskDescriptorArray [gTaskIndex] ;
  136. taskControlBlockPtr->mTaskIndex = gTaskIndex ;
  137. // Initialize properties
  138. // As gTaskDescriptorArray is in bss, all properties are by default initialized to binary 0
  139. // taskControlBlockPtr->mDeadline = 0 ; // statically initialized to 0
  140. // taskControlBlockPtr->mUserResult = false ; // statically initialized to false
  141. // taskControlBlockPtr->mGuardState = GUARD_EVALUATING_OR_OUTSIDE ; // statically initialized to GUARD_EVALUATING_OR_OUTSIDE
  142. // taskControlBlockPtr->mGuardDescriptor.mCount = 0 ; // statically initialized to 0
  143. // Initialize Context
  144. kernel_set_task_context (MODE_
  145. taskControlBlockPtr->mTaskContext,
  146. (uint32_t) inStackBufferAddress,
  147. inStackBufferSize,
  148. inTaskRoutine) ;
  149. // Make task ready
  150. kernel_makeTaskReady (MODE_ taskControlBlockPtr) ;
  151. gTaskIndex += 1 ;
  152. }