0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

基于事件驱动的有限状态机介绍

Q4MP_gh_c472c21 来源:Gitee 作者:Gitee 2021-11-16 15:29 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

一、介绍

EFSM(event finite state machine,事件驱动型有限状态机),是一个基于事件驱动的有限状态机,主要应用于嵌入式设备的软件系统中。

EFSM的设计原则是:简单!EFSM的使用者只需要关心:

  1. 当事件到来时,通过EFSM取得对应事件的处理方法
  2. 当特定事件到来,或者条件满足时,调用状态切换方法进行状态切换

由于EFSM的巧妙设计,避免了命名冲突的问题,你可以在一个程序中定义多个状态机;要是能对不同状态进行组织,还可以做出层次状态机的结构。

EFSM总共分为两个部分:

  • EFSM核心:由uthash.h、efsm.h和efsm_conf.h三个文件组成;他们构成了事件驱动型状态机的核心;使用的时候只需要包含efsm.h即可;
  • EFSM扩展:在EFSM核心的基础上,增加efsmt.h和efsmt.c两个文件,这两个文件会根据具体的状态机创建状态机线程,用于驱动状态机运转;使用的时候只需要包含efsmt.h即可;

二、接口总览

EFSM总共提供了两套接口,你只需要选择其中套用法即可。全部接口概述如下:

2.1 使用状态机工具集(EFSM核心)

若你想自己把控状态机的整个运转过程,可以直接使用EFSM核心,详细接口如下。

1. 状态操作接口

API 说明 参数
EFSM_SETS 用于创建某一状态下不同时间的处理集合 类型,无参数
EFSM_CREATE(state) 用于创建某一状态名 state是状态名
EFSM_DECLEAR(state) 当在其它地方需要使用到某个状态时,用于声明 state是状态名
EFSM_BIND(state, sets) 用于将状态state与处理集sets进行绑定 state是状态名,sets是处理集

2. 状态指针操作接口

API 说明 参数
EFSM_PTR_CREATE(name) 用于创建一个状态机指针 name是状态机指针名
EFSM_PTR_DECLEAR(name) 当在其它地方需要使用到某个状态指针时,用于声明 name是状态机指针名
EFSM_PTR_BIND(name, state) 用于为状态机指针name绑定到初始状态state,只调用一次 name是状态机指针名,state是状态名

3. 状态切换接口

API 说明 参数
EFSM_TRANSFER(name, state) 用于把状态机name切换到state状态 name是状态机指针名,state是状态名
EFSM_TRANSFER_ENABLE(name) 使能状态切换功能,在EFSM_TRANSFER()前调用 name是状态机指针名
EFSM_TRANSFER_DISABLE(name) 除能状态切换功能,在EFSM_TRANSFER()后调用 name是状态机指针名

4. 获取处理函数接口

API 说明 参数
EFSM_HANDLER(name, event) 用与当某个事件到来时,通过该方法获取到当前状态下的对应处理方法 name是状态机指针名,event是事件

2.2 使用状态机

若你不关心状态机的内部细节实现,需要一个可直接运转的状态机,那么请使用efsmt.h头文件,并将efsmt.c编译进你的源码。详细接口如下。

1. 状态操作接口

API 说明 参数
EFSM_SETS 用于创建某一状态下不同时间的处理集合 类型,无参数
EFSM_CREATE(state) 用于创建某一状态名 state是状态名
EFSM_DECLEAR(state) 当在其它地方需要使用到某个状态时,用于声明 state是状态名
EFSM_BIND(state, sets) 用于将状态state与处理集sets进行绑定 state是状态名,sets是处理集

2. 状态机操作接口

API 说明 参数
EFSMT_CREATE(name) 创建一个状态机 name是状态机名
EFSMT_DESTROY(name) 当不再使用状态机时,用于销毁 name是状态机名
EFSMT_DECLEAR(name) 当在其它地方需要使用到状态机时,用于声明 name是状态机名
EFSMT_BIND(name, state) 用于绑定状态机的初始状态 name是状态机名,state是状态名

3. 状态切换接口

API 说明 参数
EFSM_TRANSFER(name, state) 用于把状态机name切换到state状态 name是状态机指针名,state是状态名
EFSM_TRANSFER_ENABLE(name) 使能状态切换功能,在EFSM_TRANSFER()前调用 name是状态机指针名
EFSM_TRANSFER_DISABLE(name) 除能状态切换功能,在EFSM_TRANSFER()后调用 name是状态机指针名

4. 触发事件接口

API 说明 参数
EFSMT_INVOKE(name, event, arg) 当事件到来时,触发该事件,状态机会自动寻找并调用对应的处理事件 name是状态机名,event是事件,arg是事件参数

2.3 总结

从接口可以看出,创建处理集与状态集,和状态切换方法是完全一样的;两种方法的唯一差异就是:当事件来了之后,事件对应的处理是由你来控制,还是由状态机内部进行控制。

三、使用说明

要使用EFSM,非常简单,只需要如下三步:定义事件集、定义状态集、使用状态集。

3.1 定义事件集

在我们设计业务/功能时,首先对需要使用到的事件进行定义。具体实现方法是在efsm_event.h文件中,使用EFSM_EVENT()宏定义需要的事件。如果你需要定义多个状态机,那请将不同状态机的事件分块保存,建议使用enum进行管理。比如:

enum{
EVENT_PLAY=EFSM_EVENT(1),
EVENT_STOP=EFSM_EVENT(2),
EVENT_NEXT=EFSM_EVENT(3),
EVENT_PREV=EFSM_EVENT(4),
EVENT_START=EFSM_EVENT(7),//notrequirecontinuous
};

3.2 定义状态集

  1. 定义状态:使用EFSM_CREATE(state)创建一个状态state;在其它使用到它的地方用EFSM_DECLEAR(state)宏进行声明;
  2. 定义处理集:使用EFSM_SETS state_sets定义一个状态集合state_sets,其中每个事件可以对应一个处理函数,也可以对应多个函数,当然也可以为NULL;处理函数需满足如下格式:
typedefvoid(*EFSM_EVENT_HANDLER)(EFSM_EVENT_TYPEevent,void*arg);
  1. 绑定状态与处理集:在模块初始化函数中调用EFSM_BIND(state, sets)宏将状态state与处理集sets进行绑定;

3.3 使用状态集

当为模块/产品实现了所有的状态,那么编写业务应用程序来实现调度,让所有的状态机完美的运作起来。具体使用方法如下:

  1. 定义状态指针:使用EFSM_PTR_CREATE(name)定义一个状态指针name;当然你也可以定义多个状态指针,每个指针对应一个状态机。在其它使用到该状态指针的地方使用EFSM_PTR_DECLEAR(name)宏进行声明;
  2. 绑定初始状态:在整个状态机运作之前,需要使用EFSM_PTR_BIND(name, state)宏将状态指针name与状态state进行绑定;
  3. 获取处理函数指针:当接收到某个事件时,可以通过EFSM_HANDLER(name,event)宏从状态指针name中获取对该事件的处理方法,第一个参数是状态指针,第二个参数为事件;
  4. 状态切换:当遇到某个事件,或者在事件处理函数中条件满足,需要进行状态切换时,使用下面流程进行切换:
EFSM_TRANSFER_ENABLE(name);
EFSM_TRANSFER(name,state);
EFSM_TRANSFER_DISABLE(name);
  • EFSM_TRANSFER_ENABLE(name)宏使能状态指针name的切换能力;
  • EFSM_TRANSFER_DISABLE(name)宏除能状态指针name的切换能力;
  • EFSM_TRANSFER(name, state)宏执行状态指针name切换到state;

注意:做状态切换时,必须满足ENABLE()->TRANSFER()->DISABLE()的流程。这么做的目的,是为了让编程者思考:状态设计与状态的跳转是否必要与合理

四、常见问题

  1. 使用过程中若遇到如下错误,是因为在正式使用前,没有把状态指针绑定到具体的状态:
EFSM:cur-state-ptrhave'tbindastate:%xxx!!!
  1. 使用过程中若遇到如下错误,是因为状态切换动作不满足ENABLE()->TRANSFER()->DISABLE()的流程:
EFSM:'xxx'switchto'xxx'failed!!!
  1. 使用过程中,若遇到了死锁或卡死,是因为状态切换动作不满足ENABLE()->TRANSFER()->DISABLE()的流程:
  2. 命名定义了处理函数,为什么每次EFSM_HANDLER()得到的都是NULL?答:因为你没有将状态与事件集进行绑定。
  3. 对于某个状态没有使用到的事件,我是否可以不对其定义?答:完全可以,这样还可以加快EFSM的处理速度。不过不建议直接删除,而采用注释的形式,比如:
EFSM_SETSonline[]={{EVENT_PLAY,online_play},
/*{EVENT_STOP,NULL},*/
{EVENT_NEXT,online_next},
/*{EVENT_PREV,NULL},*/
{EVENT_START,online_start},
};

END 原文地址:https://gitee.com/simpost/EFSM/tree/master/ 责任编辑:haq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 嵌入式
    +关注

    关注

    5212

    文章

    20801

    浏览量

    339059
  • 驱动
    +关注

    关注

    12

    文章

    2002

    浏览量

    88823

原文标题:分享一个基于事件驱动的有限状态机

文章出处:【微信号:gh_c472c2199c88,微信公众号:嵌入式微处理器】欢迎添加关注!文章转载请注明出处。

收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    如何理解Linux内核中的PCIe驱动

    我们习惯了用 Verilog 去死磕 PCIe 的底层协议状态机。但一旦越过硬件边界来到操作系统层面,Linux 内核是如何接管并驱动这些 PCI/PCIe 设备的呢?由于不同的 CPU 架构实现了
    的头像 发表于 04-11 17:22 1431次阅读

    为什么在FPGA设计中使用MicroBlaze V处理器

    在各类行业与应用中,经常能看到许多 FPGA 设计。一个非常常见的现象是:设计者常常用复杂的有限状态机(FSM)来实现 I²C、SPI、GPIO 时序控制等功能。
    的头像 发表于 12-19 15:29 8666次阅读
    为什么在FPGA设计中使用MicroBlaze V处理器

    一个面向单片、事件驱动的嵌入式开发平台介绍

    易用,是进行线程(状态机)间同步或者通信的主要手段,也是对EventOS分布式特性和跨平台开发进行支持的唯一手段。事件支持广播发送,或者发布-订阅机制两种方式(二选一)。 2、全局事件队列,摒弃每个
    发表于 12-05 06:26

    睿远研究院丨IO-Link规范解读(十一):ISDU状态机与EVENT事件

    上篇我们介绍了ISDU的典型编码格式和应用案例,本篇我们就来详细介绍下,ISDU的状态机,并把EVENT事件的逻辑,给大家好好解析下。 1主站ISDU状态机 如上图所示,ISDU的
    的头像 发表于 11-29 18:28 4985次阅读
    睿远研究院丨IO-Link规范解读(十一):ISDU<b class='flag-5'>状态机</b>与EVENT事件

    什么是状态机

    什么是状态机状态机(state machine)有5个要素: 状态(state) 迁移(transition) 事件(event) 动作(action) 条件(guard) 状态
    发表于 11-27 08:15

    嵌入式开发为何经常用到状态机架构

    一个管脚电平跳变或者一个串口数据,让整个程序都不动的情况也让我非常纠结,如果事件一直不发生电平跳变,你要等到世界末日么? 如果应用状态机编程思想,程序只需要用全局变量记录下工作状态,就可以转头去干别的
    发表于 11-25 07:08

    睿远研究院丨IO-Link规范解读(八):M-Sequence Type 与消息处理状态机

    丢弃,减少后端工作量。 那么今天我们就来深入研究下,消息处理模块中提到的各类M-Sequence,以及主从站的消息处理状态机!   1 M-Sequence Type 上回讲到主站发出来的MC和CKT
    的头像 发表于 11-07 16:18 6989次阅读
    睿远研究院丨IO-Link规范解读(八):M-Sequence Type 与消息处理<b class='flag-5'>状态机</b>

    睿远研究院丨IO-Link规范解读(六):主从站状态机解析

    前言 书接上文,今天我们就来好好聊聊主从站的DL-Mode状态机,还请各位童鞋前排坐好! 1主站状态机解析 主站的DL-Mode状态机有5个大状态,也是我们很熟悉的 建立通信、开始、预
    的头像 发表于 10-28 17:34 6492次阅读
    睿远研究院丨IO-Link规范解读(六):主从站<b class='flag-5'>状态机</b>解析

    LSM6DSV iNEMO™惯性模块:为下一代智能设备提供高性能运动感知

    的配置、滤波和处理功能。LSM6DSV在高性能模式下提升性能(0.65mA时),具有不间断低功耗特性,实现最佳的消费者运动体验。该器件嵌入了高级专用功能,例如有限状态机和数据滤波,用于OIS、EIS和运动处理。
    的头像 发表于 10-28 11:25 1108次阅读
    LSM6DSV iNEMO™惯性模块:为下一代智能设备提供高性能运动感知

    掌握数字设计基础:迈向芯片设计的第一步

    寄存器、计数器。 两者结合,就能实现强大的数据处理与控制能力。 4、有限状态机(FSM):数字系统的大脑 在更复杂的设计中,有限状态机(FSM) 就像系统的“大脑”,通过状态与转移逻辑来驱动
    发表于 10-09 21:11

    JTAG标准的状态机实现

    JTAG作为一项国际标准测试协议(IEEE1149.1兼容),主要用于芯片内部测试和调试。目前的主流芯片均支持JTAG协议,如DSP、FPGA、ARM、部分单片等。标准的JTAG接口是20Pin,但JTAG实际使用的只有4根信号线,再配合电源、地。
    的头像 发表于 08-21 15:12 3162次阅读
    JTAG标准的<b class='flag-5'>状态机</b>实现

    洲光源彩色光传感器产品介绍

    ZCS-2016C-08D-Z4 设备可进行色温测量、亮度传感。内部状态机提供了将设备置于色温测量之间的低功率状态的能力,非常低的平均功耗。
    的头像 发表于 07-29 09:17 986次阅读
    洲光源彩色光传感器产品<b class='flag-5'>介绍</b>

    请问如何在FX10上使用GPIF III状态机 *.h 文件?

    LVCMOS 2 位 SlaveFIFO GPIF III 状态机的演示中有一个 cy_gpif_header_lvcmos.h 文件。 我想知道如何使用.h文件,只需放入.h文件放入 FX10 项目? 您有它的用户指南文档吗?
    发表于 07-16 08:17

    NVMe高速传输之摆脱XDMA设计之十:NVMe初始化状态机设计

    1为NVMe配置初始化状态机状态转移图。各状态的说明如下: 图1NVMe初始化状态转移图 IDLE:空闲状态,复位后的初始
    发表于 07-05 22:03

    NVMe高速传输之摆脱XDMA设计之八:PCIe初始化状态机设计

    PCIe配置初始化状态机实现PCIe设备枚举和配置空间初始化过程,在完成链路训练后,使用DFS(深度优先搜索)算法枚举PCIe总线上的设备,完成PCIe总线域的地址分配和设备的初始化。PCIe配置
    发表于 07-05 22:00