cb4785542e
Add an application agnostic State Machine Framework library to Zephyr that provides an easy way for developers to integrate state machines into their application. Twister passed: twister -T tests/lib/smf/ Signed-off-by: Sam Hurst <sbh1187@gmail.com>
155 lines
4.1 KiB
C
155 lines
4.1 KiB
C
/*
|
|
* Copyright 2021 The Chromium OS Authors
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
/* State Machine Framework */
|
|
|
|
#ifndef ZEPHYR_INCLUDE_SMF_H_
|
|
#define ZEPHYR_INCLUDE_SMF_H_
|
|
|
|
#ifdef CONFIG_SMF_ANCESTOR_SUPPORT
|
|
/**
|
|
* @brief Macro to create a hierarchical state.
|
|
*
|
|
* @param _entry State entry function
|
|
* @param _run State run function
|
|
* @param _exit State exit function
|
|
* @param _parent State parent object or NULL
|
|
*/
|
|
#define SMF_CREATE_STATE(_entry, _run, _exit, _parent) \
|
|
{ \
|
|
.entry = _entry, \
|
|
.run = _run, \
|
|
.exit = _exit, \
|
|
.parent = _parent \
|
|
}
|
|
|
|
#else
|
|
|
|
/**
|
|
* @brief Macro to create a flat state.
|
|
*
|
|
* @param _entry State entry function
|
|
* @param _run State run function
|
|
* @param _exit State exit function
|
|
*/
|
|
#define SMF_CREATE_STATE(_entry, _run, _exit) \
|
|
{ \
|
|
.entry = _entry, \
|
|
.run = _run, \
|
|
.exit = _exit \
|
|
}
|
|
|
|
#endif /* CONFIG_SMF_ANCESTOR_SUPPORT */
|
|
|
|
/**
|
|
* @brief Macro to cast user defined object to state machine
|
|
* context.
|
|
*
|
|
* @param o A pointer to the user defined object
|
|
*/
|
|
#define SMF_CTX(o) ((struct smf_ctx *)o)
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
#include <zephyr.h>
|
|
|
|
/**
|
|
* @brief Function pointer that implements a portion of a state
|
|
*
|
|
* @param obj pointer user defined object
|
|
*/
|
|
typedef void (*state_execution)(void *obj);
|
|
|
|
/** General state that can be used in multiple state machines. */
|
|
struct smf_state {
|
|
/** Optional method that will be run when this state is entered */
|
|
const state_execution entry;
|
|
/**
|
|
* Optional method that will be run repeatedly during state machine
|
|
* loop.
|
|
*/
|
|
const state_execution run;
|
|
/** Optional method that will be run when this state exists */
|
|
const state_execution exit;
|
|
/**
|
|
* Optional parent state that contains common entry/run/exit
|
|
* implementation among various child states.
|
|
* entry: Parent function executes BEFORE child function.
|
|
* run: Parent function executes AFTER child function.
|
|
* exit: Parent function executes AFTER child function.
|
|
*
|
|
* Note: When transitioning between two child states with a shared parent,
|
|
* that parent's exit and entry functions do not execute.
|
|
*/
|
|
const struct smf_state *parent;
|
|
};
|
|
|
|
/** Defines the current context of the state machine. */
|
|
struct smf_ctx {
|
|
/** Current state the state machine is executing. */
|
|
const struct smf_state *current;
|
|
/** Previous state the state machine executed */
|
|
const struct smf_state *previous;
|
|
/**
|
|
* This value is set by the set_terminate function and
|
|
* should terminate the state machine when its set to a
|
|
* value other than zero when it's returned by the
|
|
* run_state function.
|
|
*/
|
|
int32_t terminate_val;
|
|
/**
|
|
* The state machine casts this to a "struct internal_ctx" and it's
|
|
* used to track state machine context
|
|
*/
|
|
uint32_t internal;
|
|
};
|
|
|
|
/**
|
|
* @brief Initializes the state machine and sets its initial state.
|
|
*
|
|
* @param ctx State machine context
|
|
* @param init_state Initial state the state machine starts in.
|
|
*/
|
|
void smf_set_initial(struct smf_ctx *ctx, const struct smf_state *init_state);
|
|
|
|
/**
|
|
* @brief Changes a state machines state. This handles exiting the previous
|
|
* state and entering the target state. A common parent state will not
|
|
* exited nor be re-entered.
|
|
*
|
|
* @param ctx State machine context
|
|
* @param new_state State to transition to (NULL is valid and exits all states)
|
|
*/
|
|
void smf_set_state(struct smf_ctx *ctx, const struct smf_state *new_state);
|
|
|
|
/**
|
|
* @brief Termate a state machine
|
|
*
|
|
* @param ctx State machine context
|
|
* @param val Non-Zero termination value that's returned by the smf_run_state
|
|
* function.
|
|
*/
|
|
void smf_set_terminate(struct smf_ctx *ctx, int32_t val);
|
|
|
|
/**
|
|
* @brief Runs one iteration of a state machine (including any parent states)
|
|
*
|
|
* @param ctx State machine context
|
|
* @return A non-zero value should terminate the state machine. This
|
|
* non-zero value could represet a terminal state being reached
|
|
* or the detection of an error that should result in the
|
|
* termination of the state machine.
|
|
*/
|
|
int32_t smf_run_state(struct smf_ctx *ctx);
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#endif /* ZEPHYR_INCLUDE_SMF_H_ */
|