443 lines
13 KiB
C++
443 lines
13 KiB
C++
/**
|
|
* \file pros/rtos.h
|
|
*
|
|
* Contains declarations for the PROS RTOS kernel for use by typical VEX
|
|
* programmers.
|
|
*
|
|
* Visit https://pros.cs.purdue.edu/v5/tutorials/topical/multitasking.html to
|
|
* learn more.
|
|
*
|
|
* This file should not be modified by users, since it gets replaced whenever
|
|
* a kernel upgrade occurs.
|
|
*
|
|
* \copyright Copyright (c) 2017-2023, Purdue University ACM SIGBots.
|
|
* All rights reserved.
|
|
*
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
*/
|
|
|
|
#ifndef _PROS_RTOS_H_
|
|
#define _PROS_RTOS_H_
|
|
|
|
#include <stdbool.h>
|
|
#include <stdint.h>
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
namespace pros {
|
|
#endif
|
|
|
|
// The highest priority that can be assigned to a task. Beware of deadlock.
|
|
#define TASK_PRIORITY_MAX 16
|
|
|
|
// The lowest priority that can be assigned to a task.
|
|
// This may cause severe performance problems and is generally not recommended.
|
|
#define TASK_PRIORITY_MIN 1
|
|
|
|
// The default task priority, which should be used for most tasks.
|
|
// Default tasks such as autonomous() inherit this priority.
|
|
#define TASK_PRIORITY_DEFAULT 8
|
|
|
|
// The recommended stack size for a new task. This stack size is used for
|
|
// default tasks such as autonomous(). This equates to 32,768 bytes, or 128
|
|
// times the default stack size for a task in PROS 2.
|
|
#define TASK_STACK_DEPTH_DEFAULT 0x2000
|
|
|
|
// The minimal stack size for a task. This equates to 2048 bytes, or 8 times the
|
|
// default stack size for a task in PROS 2.
|
|
#define TASK_STACK_DEPTH_MIN 0x200
|
|
|
|
// The maximum number of characters allowed in a task's name.
|
|
#define TASK_NAME_MAX_LEN 32
|
|
|
|
// The maximum timeout value that can be given to, for instance, a mutex grab.
|
|
#define TIMEOUT_MAX ((uint32_t)0xffffffffUL)
|
|
|
|
typedef void* task_t;
|
|
typedef void (*task_fn_t)(void*);
|
|
|
|
typedef enum {
|
|
E_TASK_STATE_RUNNING = 0,
|
|
E_TASK_STATE_READY,
|
|
E_TASK_STATE_BLOCKED,
|
|
E_TASK_STATE_SUSPENDED,
|
|
E_TASK_STATE_DELETED,
|
|
E_TASK_STATE_INVALID
|
|
} task_state_e_t;
|
|
|
|
typedef enum {
|
|
E_NOTIFY_ACTION_NONE,
|
|
E_NOTIFY_ACTION_BITS,
|
|
E_NOTIFY_ACTION_INCR,
|
|
E_NOTIFY_ACTION_OWRITE,
|
|
E_NOTIFY_ACTION_NO_OWRITE
|
|
} notify_action_e_t;
|
|
|
|
#ifdef PROS_USE_SIMPLE_NAMES
|
|
#ifdef __cplusplus
|
|
#define TASK_STATE_RUNNING pros::E_TASK_STATE_RUNNING
|
|
#define TASK_STATE_READY pros::E_TASK_STATE_READY
|
|
#define TASK_STATE_BLOCKED pros::E_TASK_STATE_BLOCKED
|
|
#define TASK_STATE_SUSPENDED pros::E_TASK_STATE_SUSPENDED
|
|
#define TASK_STATE_DELETED pros::E_TASK_STATE_DELETED
|
|
#define TASK_STATE_INVALID pros::E_TASK_STATE_INVALID
|
|
#define NOTIFY_ACTION_NONE pros::E_NOTIFY_ACTION_NONE
|
|
#define NOTIFY_ACTION_BITS pros::E_NOTIFY_ACTION_BITS
|
|
#define NOTIFY_ACTION_INCR pros::E_NOTIFY_ACTION_INCR
|
|
#define NOTIFY_ACTION_OWRITE pros::E_NOTIFY_ACTION_OWRITE
|
|
#define NOTIFY_ACTION_NO_OWRITE pros::E_NOTIFY_ACTION_NO_OWRITE
|
|
#else
|
|
#define TASK_STATE_RUNNING E_TASK_STATE_RUNNING
|
|
#define TASK_STATE_READY E_TASK_STATE_READY
|
|
#define TASK_STATE_BLOCKED E_TASK_STATE_BLOCKED
|
|
#define TASK_STATE_SUSPENDED E_TASK_STATE_SUSPENDED
|
|
#define TASK_STATE_DELETED E_TASK_STATE_DELETED
|
|
#define TASK_STATE_INVALID E_TASK_STATE_INVALID
|
|
#define NOTIFY_ACTION_NONE E_NOTIFY_ACTION_NONE
|
|
#define NOTIFY_ACTION_BITS E_NOTIFY_ACTION_BITS
|
|
#define NOTIFY_ACTION_INCR E_NOTIFY_ACTION_INCR
|
|
#define NOTIFY_ACTION_OWRITE E_NOTIFY_ACTION_OWRITE
|
|
#define NOTIFY_ACTION_NO_OWRITE E_NOTIFY_ACTION_NO_OWRITE
|
|
#endif
|
|
#endif
|
|
|
|
typedef void* mutex_t;
|
|
|
|
/**
|
|
* Refers to the current task handle
|
|
*/
|
|
#ifdef __cplusplus
|
|
#define CURRENT_TASK ((pros::task_t)NULL)
|
|
#else
|
|
#define CURRENT_TASK ((task_t)NULL)
|
|
#endif
|
|
|
|
#ifdef __cplusplus
|
|
namespace c {
|
|
#endif
|
|
|
|
/**
|
|
* Gets the number of milliseconds since PROS initialized.
|
|
*
|
|
* \return The number of milliseconds since PROS initialized
|
|
*/
|
|
uint32_t millis(void);
|
|
|
|
/**
|
|
* Gets the number of microseconds since PROS initialized,
|
|
*
|
|
* \return The number of microseconds since PROS initialized
|
|
*/
|
|
uint64_t micros(void);
|
|
|
|
/**
|
|
* Creates a new task and add it to the list of tasks that are ready to run.
|
|
*
|
|
* This function uses the following values of errno when an error state is
|
|
* reached:
|
|
* ENOMEM - The stack cannot be used as the TCB was not created.
|
|
*
|
|
* \param function
|
|
* Pointer to the task entry function
|
|
* \param parameters
|
|
* Pointer to memory that will be used as a parameter for the task being
|
|
* created. This memory should not typically come from stack, but rather
|
|
* from dynamically (i.e., malloc'd) or statically allocated memory.
|
|
* \param prio
|
|
* The priority at which the task should run.
|
|
* TASK_PRIO_DEFAULT plus/minus 1 or 2 is typically used.
|
|
* \param stack_depth
|
|
* The number of words (i.e. 4 * stack_depth) available on the task's
|
|
* stack. TASK_STACK_DEPTH_DEFAULT is typically sufficienct.
|
|
* \param name
|
|
* A descriptive name for the task. This is mainly used to facilitate
|
|
* debugging. The name may be up to 32 characters long.
|
|
*
|
|
* \return A handle by which the newly created task can be referenced. If an
|
|
* error occurred, NULL will be returned and errno can be checked for hints as
|
|
* to why task_create failed.
|
|
*/
|
|
task_t task_create(task_fn_t function, void* const parameters, uint32_t prio, const uint16_t stack_depth,
|
|
const char* const name);
|
|
|
|
/**
|
|
* Removes a task from the RTOS real time kernel's management. The task being
|
|
* deleted will be removed from all ready, blocked, suspended and event lists.
|
|
*
|
|
* Memory dynamically allocated by the task is not automatically freed, and
|
|
* should be freed before the task is deleted.
|
|
*
|
|
* \param task
|
|
* The handle of the task to be deleted. Passing NULL will cause the
|
|
* calling task to be deleted.
|
|
*/
|
|
void task_delete(task_t task);
|
|
|
|
/**
|
|
* Delays a task for a given number of milliseconds.
|
|
*
|
|
* This is not the best method to have a task execute code at predefined
|
|
* intervals, as the delay time is measured from when the delay is requested.
|
|
* To delay cyclically, use task_delay_until().
|
|
*
|
|
* \param milliseconds
|
|
* The number of milliseconds to wait (1000 milliseconds per second)
|
|
*/
|
|
void task_delay(const uint32_t milliseconds);
|
|
|
|
void delay(const uint32_t milliseconds);
|
|
|
|
/**
|
|
* Delays a task until a specified time. This function can be used by periodic
|
|
* tasks to ensure a constant execution frequency.
|
|
*
|
|
* The task will be woken up at the time *prev_time + delta, and *prev_time will
|
|
* be updated to reflect the time at which the task will unblock.
|
|
*
|
|
* \param prev_time
|
|
* A pointer to the location storing the setpoint time. This should
|
|
* typically be initialized to the return value of millis().
|
|
* \param delta
|
|
* The number of milliseconds to wait (1000 milliseconds per second)
|
|
*/
|
|
void task_delay_until(uint32_t* const prev_time, const uint32_t delta);
|
|
|
|
/**
|
|
* Gets the priority of the specified task.
|
|
*
|
|
* \param task
|
|
* The task to check
|
|
*
|
|
* \return The priority of the task
|
|
*/
|
|
uint32_t task_get_priority(task_t task);
|
|
|
|
/**
|
|
* Sets the priority of the specified task.
|
|
*
|
|
* If the specified task's state is available to be scheduled (e.g. not blocked)
|
|
* and new priority is higher than the currently running task, a context switch
|
|
* may occur.
|
|
*
|
|
* \param task
|
|
* The task to set
|
|
* \param prio
|
|
* The new priority of the task
|
|
*/
|
|
void task_set_priority(task_t task, uint32_t prio);
|
|
|
|
/**
|
|
* Gets the state of the specified task.
|
|
*
|
|
* \param task
|
|
* The task to check
|
|
*
|
|
* \return The state of the task
|
|
*/
|
|
task_state_e_t task_get_state(task_t task);
|
|
|
|
/**
|
|
* Suspends the specified task, making it ineligible to be scheduled.
|
|
*
|
|
* \param task
|
|
* The task to suspend
|
|
*/
|
|
void task_suspend(task_t task);
|
|
|
|
/**
|
|
* Resumes the specified task, making it eligible to be scheduled.
|
|
*
|
|
* \param task
|
|
* The task to resume
|
|
*/
|
|
void task_resume(task_t task);
|
|
|
|
/**
|
|
* Gets the number of tasks the kernel is currently managing, including all
|
|
* ready, blocked, or suspended tasks. A task that has been deleted, but not yet
|
|
* reaped by the idle task will also be included in the count. Tasks recently
|
|
* created may take one context switch to be counted.
|
|
*
|
|
* \return The number of tasks that are currently being managed by the kernel.
|
|
*/
|
|
uint32_t task_get_count(void);
|
|
|
|
/**
|
|
* Gets the name of the specified task.
|
|
*
|
|
* \param task
|
|
* The task to check
|
|
*
|
|
* \return A pointer to the name of the task
|
|
*/
|
|
char* task_get_name(task_t task);
|
|
|
|
/**
|
|
* Gets a task handle from the specified name
|
|
*
|
|
* The operation takes a relatively long time and should be used sparingly.
|
|
*
|
|
* \param name
|
|
* The name to query
|
|
*
|
|
* \return A task handle with a matching name, or NULL if none were found.
|
|
*/
|
|
task_t task_get_by_name(const char* name);
|
|
|
|
/**
|
|
* Get the currently running task handle. This could be useful if a task
|
|
* wants to tell another task about itself.
|
|
*
|
|
* \return The currently running task handle.
|
|
*/
|
|
task_t task_get_current();
|
|
|
|
/**
|
|
* Sends a simple notification to task and increments the notification counter.
|
|
*
|
|
* See https://pros.cs.purdue.edu/v5/tutorials/topical/notifications.html for
|
|
* details.
|
|
*
|
|
* \param task
|
|
* The task to notify
|
|
*
|
|
* \return Always returns true.
|
|
*/
|
|
uint32_t task_notify(task_t task);
|
|
|
|
/**
|
|
*
|
|
* Utilizes task notifications to wait until specified task is complete and deleted,
|
|
* then continues to execute the program. Analogous to std::thread::join in C++.
|
|
*
|
|
* See https://pros.cs.purdue.edu/v5/tutorials/topical/notifications.html for
|
|
* details.
|
|
*
|
|
* \param task
|
|
* The task to wait on.
|
|
*
|
|
* \return void
|
|
*/
|
|
void task_join(task_t task);
|
|
|
|
/**
|
|
* Sends a notification to a task, optionally performing some action. Will also
|
|
* retrieve the value of the notification in the target task before modifying
|
|
* the notification value.
|
|
*
|
|
* See https://pros.cs.purdue.edu/v5/tutorials/topical/notifications.html for
|
|
* details.
|
|
*
|
|
* \param task
|
|
* The task to notify
|
|
* \param value
|
|
* The value used in performing the action
|
|
* \param action
|
|
* An action to optionally perform on the receiving task's notification
|
|
* value
|
|
* \param prev_value
|
|
* A pointer to store the previous value of the target task's
|
|
* notification, may be NULL
|
|
*
|
|
* \return Dependent on the notification action.
|
|
* For NOTIFY_ACTION_NO_WRITE: return 0 if the value could be written without
|
|
* needing to overwrite, 1 otherwise.
|
|
* For all other NOTIFY_ACTION values: always return 0
|
|
*/
|
|
uint32_t task_notify_ext(task_t task, uint32_t value, notify_action_e_t action, uint32_t* prev_value);
|
|
|
|
/**
|
|
* Waits for a notification to be nonzero.
|
|
*
|
|
* See https://pros.cs.purdue.edu/v5/tutorials/topical/notifications.html for
|
|
* details.
|
|
*
|
|
* \param clear_on_exit
|
|
* If true (1), then the notification value is cleared.
|
|
* If false (0), then the notification value is decremented.
|
|
* \param timeout
|
|
* Specifies the amount of time to be spent waiting for a notification
|
|
* to occur.
|
|
*
|
|
* \return The value of the task's notification value before it is decremented
|
|
* or cleared
|
|
*/
|
|
uint32_t task_notify_take(bool clear_on_exit, uint32_t timeout);
|
|
|
|
/**
|
|
* Clears the notification for a task.
|
|
*
|
|
* See https://pros.cs.purdue.edu/v5/tutorials/topical/notifications.html for
|
|
* details.
|
|
*
|
|
* \param task
|
|
* The task to clear
|
|
*
|
|
* \return False if there was not a notification waiting, true if there was
|
|
*/
|
|
bool task_notify_clear(task_t task);
|
|
|
|
/**
|
|
* Creates a mutex.
|
|
*
|
|
* See https://pros.cs.purdue.edu/v5/tutorials/topical/multitasking.html#mutexes
|
|
* for details.
|
|
*
|
|
* \return A handle to a newly created mutex. If an error occurred, NULL will be
|
|
* returned and errno can be checked for hints as to why mutex_create failed.
|
|
*/
|
|
mutex_t mutex_create(void);
|
|
|
|
/**
|
|
* Takes and locks a mutex, waiting for up to a certain number of milliseconds
|
|
* before timing out.
|
|
*
|
|
* See https://pros.cs.purdue.edu/v5/tutorials/topical/multitasking.html#mutexes
|
|
* for details.
|
|
*
|
|
* \param mutex
|
|
* Mutex to attempt to lock.
|
|
* \param timeout
|
|
* Time to wait before the mutex becomes available. A timeout of 0 can
|
|
* be used to poll the mutex. TIMEOUT_MAX can be used to block
|
|
* indefinitely.
|
|
*
|
|
* \return True if the mutex was successfully taken, false otherwise. If false
|
|
* is returned, then errno is set with a hint about why the the mutex
|
|
* couldn't be taken.
|
|
*/
|
|
bool mutex_take(mutex_t mutex, uint32_t timeout);
|
|
|
|
/**
|
|
* Unlocks a mutex.
|
|
*
|
|
* See https://pros.cs.purdue.edu/v5/tutorials/topical/multitasking.html#mutexes
|
|
* for details.
|
|
*
|
|
* \param mutex
|
|
* Mutex to unlock.
|
|
*
|
|
* \return True if the mutex was successfully returned, false otherwise. If
|
|
* false is returned, then errno is set with a hint about why the mutex
|
|
* couldn't be returned.
|
|
*/
|
|
bool mutex_give(mutex_t mutex);
|
|
|
|
/**
|
|
* Deletes a mutex
|
|
*
|
|
* \param mutex
|
|
* Mutex to unlock.
|
|
*/
|
|
void mutex_delete(mutex_t mutex);
|
|
|
|
#ifdef __cplusplus
|
|
} // namespace c
|
|
} // namespace pros
|
|
}
|
|
#endif
|
|
|
|
#endif // _PROS_RTOS_H_
|