2015-08-12 10:53:41 +00:00
|
|
|
/**
|
|
|
|
* Definitions for the MicroBit Fiber scheduler.
|
|
|
|
*
|
|
|
|
* This lightweight, non-preemptive scheduler provides a simple threading mechanism for two main purposes:
|
|
|
|
*
|
|
|
|
* 1) To provide a clean abstraction for application languages to use when building async behaviour (callbacks).
|
|
|
|
* 2) To provide ISR decoupling for Messagebus events generted in an ISR context.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef MICROBIT_FIBER_H
|
|
|
|
#define MICROBIT_FIBER_H
|
|
|
|
|
|
|
|
#include "mbed.h"
|
2015-08-31 22:25:10 +00:00
|
|
|
#include "MicroBitConfig.h"
|
2015-09-11 15:39:38 +00:00
|
|
|
#include "MicroBitEvent.h"
|
2015-08-12 10:53:41 +00:00
|
|
|
|
|
|
|
// TODO: Consider a split mode scheduler, that monitors used stack size, and maintains a dedicated, persistent
|
2015-08-31 22:25:10 +00:00
|
|
|
// stack for any long lived fibers with large stack
|
2015-08-12 10:53:41 +00:00
|
|
|
|
2015-08-31 22:25:10 +00:00
|
|
|
// Fiber Scheduler Flags
|
|
|
|
#define MICROBIT_FLAG_DATA_READY 0x01
|
2015-08-19 22:27:26 +00:00
|
|
|
|
2015-08-31 22:25:10 +00:00
|
|
|
// Fiber Flags
|
2015-08-19 22:27:26 +00:00
|
|
|
#define MICROBIT_FIBER_FLAG_FOB 0x01
|
|
|
|
#define MICROBIT_FIBER_FLAG_PARENT 0x02
|
|
|
|
#define MICROBIT_FIBER_FLAG_CHILD 0x04
|
2015-09-19 10:45:45 +00:00
|
|
|
#define MICROBIT_FIBER_FLAG_RUN_IDLE_WITHIN 0x08
|
2015-08-12 10:53:41 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Thread Context for an ARM Cortex M0 core.
|
|
|
|
*
|
|
|
|
* This is probably overkill, but the ARMCC compiler uses a lot register optimisation
|
|
|
|
* in its calling conventions, so better safe than sorry. :-)
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
struct Cortex_M0_TCB
|
|
|
|
{
|
|
|
|
uint32_t R0;
|
|
|
|
uint32_t R1;
|
|
|
|
uint32_t R2;
|
|
|
|
uint32_t R3;
|
|
|
|
uint32_t R4;
|
|
|
|
uint32_t R5;
|
|
|
|
uint32_t R6;
|
|
|
|
uint32_t R7;
|
|
|
|
uint32_t R8;
|
|
|
|
uint32_t R9;
|
|
|
|
uint32_t R10;
|
|
|
|
uint32_t R11;
|
|
|
|
uint32_t R12;
|
|
|
|
uint32_t SP;
|
|
|
|
uint32_t LR;
|
2015-08-31 22:25:10 +00:00
|
|
|
uint32_t stack_base;
|
2015-08-12 10:53:41 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Representation of a single Fiber
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct Fiber
|
|
|
|
{
|
|
|
|
Cortex_M0_TCB tcb; // Thread context when last scheduled out.
|
2015-08-31 22:25:10 +00:00
|
|
|
uint32_t stack_bottom; // The start sddress of this Fiber's stack. Stack is heap allocated, and full descending.
|
|
|
|
uint32_t stack_top; // The end address of this Fiber's stack.
|
2015-08-12 10:53:41 +00:00
|
|
|
uint32_t context; // Context specific information.
|
2015-08-19 22:27:26 +00:00
|
|
|
uint32_t flags; // Information about this fiber.
|
2015-08-12 10:53:41 +00:00
|
|
|
Fiber **queue; // The queue this fiber is stored on.
|
|
|
|
Fiber *next, *prev; // Position of this Fiber on the run queues.
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Initialises the Fiber scheduler.
|
|
|
|
* Creates a Fiber context around the calling thread, and adds it to the run queue as the current thread.
|
|
|
|
*
|
|
|
|
* This function must be called once only from the main thread, and before any other Fiber operation.
|
|
|
|
*/
|
|
|
|
void scheduler_init();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Exit point for all fibers.
|
|
|
|
* Any fiber reaching the end of its entry function will return here for recycling.
|
|
|
|
*/
|
|
|
|
void release_fiber(void);
|
|
|
|
void release_fiber(void *param);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Launches a fiber
|
2015-08-31 22:25:10 +00:00
|
|
|
*/
|
|
|
|
void launch_new_fiber(void (*ep)(void), void (*cp)(void))
|
|
|
|
#ifdef __GCC__
|
|
|
|
__attribute__((naked))
|
|
|
|
#endif
|
|
|
|
;
|
|
|
|
|
|
|
|
void launch_new_fiber_param(void (*ep)(void *), void (*cp)(void *), void *pm)
|
2015-08-12 10:53:41 +00:00
|
|
|
#ifdef __GCC__
|
|
|
|
__attribute__((naked))
|
|
|
|
#endif
|
|
|
|
;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a new Fiber, and launches it.
|
|
|
|
*
|
|
|
|
* @param entry_fn The function the new Fiber will begin execution in.
|
|
|
|
* @param completion_fn The function called when the thread completes execution of entry_fn.
|
|
|
|
* @return The new Fiber.
|
|
|
|
*/
|
|
|
|
Fiber *create_fiber(void (*entry_fn)(void), void (*completion_fn)(void) = release_fiber);
|
|
|
|
|
2015-08-31 22:25:10 +00:00
|
|
|
|
2015-08-12 10:53:41 +00:00
|
|
|
/**
|
|
|
|
* Creates a new parameterised Fiber, and launches it.
|
|
|
|
*
|
|
|
|
* @param entry_fn The function the new Fiber will begin execution in.
|
|
|
|
* @param param an untyped parameter passed into the entry_fn anf completion_fn.
|
|
|
|
* @param completion_fn The function called when the thread completes execution of entry_fn.
|
|
|
|
* @return The new Fiber.
|
|
|
|
*/
|
|
|
|
Fiber *create_fiber(void (*entry_fn)(void *), void *param, void (*completion_fn)(void *) = release_fiber);
|
|
|
|
|
2015-09-19 10:45:45 +00:00
|
|
|
/**
|
|
|
|
* Allow the idle thread to run within the current thread's stack frame.
|
|
|
|
* This is useful to prevent paging of the thread's stack to the heap.
|
|
|
|
*/
|
|
|
|
void fiber_allow_run_idle_within();
|
|
|
|
|
2015-08-12 10:53:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Calls the Fiber scheduler.
|
|
|
|
* The calling Fiber will likely be blocked, and control given to another waiting fiber.
|
|
|
|
* Call this to yield control of the processor when you have nothing more to do.
|
|
|
|
*/
|
|
|
|
void schedule();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Blocks the calling thread for the given period of time.
|
|
|
|
* The calling thread will be immediatley descheduled, and placed onto a
|
|
|
|
* wait queue until the requested amount of time has elapsed.
|
|
|
|
*
|
|
|
|
* n.b. the fiber will not be be made runnable until after the elasped time, but there
|
|
|
|
* are no guarantees precisely when the fiber will next be scheduled.
|
|
|
|
*
|
|
|
|
* @param t The period of time to sleep, in milliseconds.
|
|
|
|
*/
|
|
|
|
void fiber_sleep(unsigned long t);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Timer callback. Called from interrupt context, once every FIBER_TICK_PERIOD_MS milliseconds.
|
|
|
|
* Simply checks to determine if any fibers blocked on the sleep queue need to be woken up
|
|
|
|
* and made runnable.
|
|
|
|
*/
|
|
|
|
void scheduler_tick();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Blocks the calling thread until the specified event is raised.
|
|
|
|
* The calling thread will be immediatley descheduled, and placed onto a
|
|
|
|
* wait queue until the requested event is received.
|
|
|
|
*
|
|
|
|
* n.b. the fiber will not be be made runnable until after the event is raised, but there
|
|
|
|
* are no guarantees precisely when the fiber will next be scheduled.
|
|
|
|
*
|
|
|
|
* @param id The ID field of the event to listen for (e.g. MICROBIT_ID_BUTTON_A)
|
|
|
|
* @param value The VALUE of the event to listen for (e.g. MICROBIT_BUTTON_EVT_CLICK)
|
|
|
|
*/
|
|
|
|
void fiber_wait_for_event(uint16_t id, uint16_t value);
|
|
|
|
|
2015-08-19 22:27:26 +00:00
|
|
|
/**
|
2015-08-31 22:25:10 +00:00
|
|
|
* Executes the given function asynchronously if necessary.
|
2015-08-19 22:27:26 +00:00
|
|
|
*
|
|
|
|
* Fibers are often used to run event handlers, however many of these event handlers are very simple functions
|
|
|
|
* that complete very quickly, bringing unecessary RAM overhead.
|
|
|
|
*
|
|
|
|
* This function takes a snapshot of the current processor context, then attempts to optimistically call the given function directly.
|
|
|
|
* We only create an additional fiber if that function performs a block operation.
|
|
|
|
*
|
|
|
|
* @param entry_fn The function to execute.
|
|
|
|
*/
|
2015-08-31 22:25:10 +00:00
|
|
|
void invoke(void (*entry_fn)(void));
|
2015-08-19 22:27:26 +00:00
|
|
|
|
|
|
|
/**
|
2015-08-31 22:25:10 +00:00
|
|
|
* Executes the given function asynchronously if necessary.
|
2015-08-19 22:27:26 +00:00
|
|
|
*
|
|
|
|
* Fibers are often used to run event handlers, however many of these event handlers are very simple functions
|
|
|
|
* that complete very quickly, bringing unecessary RAM. overhead
|
|
|
|
*
|
|
|
|
* This function takes a snapshot of the current fiber context, then attempt to optimistically call the given function directly.
|
|
|
|
* We only create an additional fiber if that function performs a block operation.
|
|
|
|
*
|
|
|
|
* @param entry_fn The function to execute.
|
|
|
|
* @param param an untyped parameter passed into the entry_fn anf completion_fn.
|
|
|
|
*/
|
2015-08-31 22:25:10 +00:00
|
|
|
void invoke(void (*entry_fn)(void *), void *param);
|
2015-08-19 22:27:26 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Resizes the stack allocation of the current fiber if necessary to hold the system stack.
|
|
|
|
*
|
|
|
|
* If the stack allocaiton is large enough to hold the current system stack, then this function does nothing.
|
|
|
|
* Otherwise, the the current allocation of the fiber is freed, and a larger block is allocated.
|
2015-08-31 22:25:10 +00:00
|
|
|
*
|
|
|
|
* @param f The fiber context to verify.
|
|
|
|
* @return The stack depth of the given fiber.
|
2015-08-19 22:27:26 +00:00
|
|
|
*/
|
|
|
|
inline void verify_stack_size(Fiber *f);
|
|
|
|
|
2015-08-12 10:53:41 +00:00
|
|
|
/**
|
|
|
|
* Event callback. Called from the message bus whenever an event is raised.
|
|
|
|
* Checks to determine if any fibers blocked on the wait queue need to be woken up
|
|
|
|
* and made runnable due to the event.
|
|
|
|
*/
|
|
|
|
void scheduler_event(MicroBitEvent evt);
|
|
|
|
|
2015-08-31 22:25:10 +00:00
|
|
|
/**
|
|
|
|
* Determines if any fibers are waiting to be scheduled.
|
|
|
|
* @return The number of fibers currently on the run queue
|
|
|
|
*/
|
|
|
|
int scheduler_runqueue_empty();
|
2015-08-12 10:53:41 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Utility function to add the currenty running fiber to the given queue.
|
|
|
|
* Perform a simple add at the head, to avoid complexity,
|
|
|
|
* Queues are normally very short, so maintaining a doubly linked, sorted list typically outweighs the cost of
|
|
|
|
* brute force searching.
|
|
|
|
*
|
|
|
|
* @param f The fiber to add to the queue
|
|
|
|
* @param queue The run queue to add the fiber to.
|
|
|
|
*/
|
|
|
|
void queue_fiber(Fiber *f, Fiber **queue);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Utility function to the given fiber from whichever queue it is currently stored on.
|
|
|
|
* @param f the fiber to remove.
|
|
|
|
*/
|
|
|
|
void dequeue_fiber(Fiber *f);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* IDLE task.
|
|
|
|
* Only scheduled for execution when the runqueue is empty.
|
|
|
|
* Performs a procressor sleep operation, then returns to the scheduler - most likely after a timer interrupt.
|
|
|
|
*/
|
|
|
|
void idle_task();
|
|
|
|
|
|
|
|
/**
|
2015-08-19 22:27:26 +00:00
|
|
|
* Determines if the processor is executing in interrupt context.
|
|
|
|
* @return true if any the processor is currently executing any interrupt service routine. False otherwise.
|
|
|
|
*/
|
|
|
|
inline int inInterruptContext()
|
|
|
|
{
|
|
|
|
return (((int)__get_IPSR()) & 0x003F) > 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Assembler Context switch routing.
|
2015-08-12 10:53:41 +00:00
|
|
|
* Defined in CortexContextSwitch.s
|
|
|
|
*/
|
|
|
|
extern "C" void swap_context(Cortex_M0_TCB *from, Cortex_M0_TCB *to, uint32_t from_stack, uint32_t to_stack);
|
|
|
|
extern "C" void save_context(Cortex_M0_TCB *tcb, uint32_t stack);
|
2015-08-19 22:27:26 +00:00
|
|
|
extern "C" void save_register_context(Cortex_M0_TCB *tcb);
|
|
|
|
extern "C" void restore_register_context(Cortex_M0_TCB *tcb);
|
2015-08-12 10:53:41 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Time since power on. Measured in milliseconds.
|
|
|
|
* When stored as an unsigned long, this gives us approx 50 days between rollover, which is ample. :-)
|
|
|
|
*/
|
|
|
|
extern unsigned long ticks;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This variable is used to prioritise the systems' idle fibre to execute essential tasks.
|
|
|
|
*/
|
|
|
|
extern uint8_t fiber_flags;
|
|
|
|
|
|
|
|
#endif
|