状态机——protothreads

状态机——Protothreads


宗旨:技术的学习是有限的,分享的精神是无限的。


一、prothreads的优缺点

优点:

    1. 以纯C语言实现,无硬件依靠性;因此不存在移植的困难。

    2. 极少的资源需求,每个Protothread仅需要2个额外的字节;

3. 支持阻塞操纵且没有栈的切换。

缺点:

    1. 函数中不具备可重入型,不能使用局部变量;

    2. 按顺序判断各任务条件是否满足,因此无优先级抢占;

3. 任务中的各条件也是按顺序判断的,因此要求任务中的条件必须是依次出现的 protothread的阻塞机制:在每个条件判断前,先将当前地址保存到某个变量中,再判断条件是否成立,若条件成立,则往下运行;若条件不成立,则返回。

 

二、注意事项

注意:
(1)任务中使用的变量应为静态变量
(2)线程内不能使用纯 while(1)--即含 PT_WAIT_UNTIL()等宏的 while(1)是可以的。
不能在 switch(){case…}中调用任务 Protothreads API 带有 case 的语句(即只能单向嵌套)。
(3)线程内可以使用:
for(){…}
switch(){case…}-- case 与 case 之间必须是一个完整的语句或者语句段
if(){…}else{…}
含宏的 while(1){…}
(4)ProtothreadS 系统可以仍然还是个大 while(1)循环。但也可设计为根据定时器产生的恒定间隔的中断来触发和管理任务
--时间触发方式的嵌入式系统,此时可更改 pt 结构体为(见《时间触发模式下的 ProtothreadS 设计应用》):
struct pt

{

lc_t lc;

unsigned short count; // 每次中断都减 1

unsigned short load; // 初始计数值

char ready; // 任务就绪标志

}
(5)在 ProtothreadS 系统中延时:
1)如果 ProtothreadS 系统是基于时间触发,则延时可基于该触发--即基于系统时钟。
2)如果 ProtothreadS 系统中无系统时钟,
(6)Protothreads 虽然提供了在各自线程内的条件阻塞机制,但对于在该线程内调用的其它函数,则无法阻塞其运行。所以,
如果要在线程内调用占用时间较多的函数,为保证各个线程的实时性要求,需要将这类函数进一步划分为更小的函数,分步执
行。
(7) Protothread 的精华:当 Protothread 程序运行到 PT_WAIT_UNTIL 时,判断其运行条件是否满足,若不满足,则阻塞。
Protothread 的阻塞其实质就是函数返回。
Protothread 仅能在程序员指定位置阻塞。

 

三、protothreads各函数简要介绍

函数

说明

PT_INIT(pt)

初始化任务变量,只在初始化函数中执行一次

PT_BEGIN(pt)

启动任务处理,放在函数开始处

PT_END(pt)

结束任务,放在函数的最后

PT_WAIT_UNTIL(pt,condition)

条件成立,执行下面的;否则退出,下次直接跳到此处执行

PT_WAIT_WHILE(pt, condition)

类似PT_WAIT_UNTIL,只是条件取反了

PT_WAIT_THREAD(pt, thread)

等待一个子任务执行完成

PT_SPAWN(pt, child, thread)

新建一个子任务,并等待其执行完退出

PT_RESTART(pt)

重新启动某个任务执行

PT_EXIT(pt)

任务后面的部分不执行,直接退出重新执行

PT_YIELD(pt)

锁死任务

PT_YIELD_UNTIL(pt, condition)

锁死任务等待条件成立重新执行

 

四、protothreads代码

struct pt {lc_t lc;
};#define PT_WAITING 0
#define PT_YIELDED 1
#define PT_EXITED 2
#define PT_ENDED  3#define PT_INIT(pt)  LC_INIT((pt)->lc)#define PT_THREAD(name_args) char name_args#define PT_BEGIN(pt) { char PT_YIELD_FLAG = 1;LC_RESUME((pt)->lc)#define PT_END(pt) LC_END((pt)->lc);PT_YIELD_FLAG = 0; \PT_INIT(pt); return PT_ENDED; }#define PT_WAIT_UNTIL(pt, condition)            \do {                                       \LC_SET((pt)->lc);                      \if(!(condition)) {                         \returnPT_WAITING;               \}                                        \} while(0)#define PT_WAIT_WHILE(pt, cond)  PT_WAIT_UNTIL((pt), !(cond))#define PT_WAIT_THREAD(pt, thread)PT_WAIT_WHILE((pt), PT_SCHEDULE(thread))#define PT_SPAWN(pt, child, thread)   \do {            \PT_INIT((child));       \PT_WAIT_THREAD((pt), (thread));  \} while(0)#define PT_RESTART(pt)        \do {            \PT_INIT(pt);        \returnPT_WAITING;      \} while(0)#define PT_EXIT(pt)       \do {            \PT_INIT(pt);        \returnPT_EXITED;     \} while(0)#define PT_SCHEDULE(f) ((f) < PT_EXITED)#define PT_YIELD(pt)        \do {            \PT_YIELD_FLAG = 0;        \LC_SET((pt)->lc);       \if(PT_YIELD_FLAG == 0) {      \returnPT_YIELDED;      \}           \} while(0)#define PT_YIELD_UNTIL(pt, cond)    \do {            \PT_YIELD_FLAG = 0;        \LC_SET((pt)->lc);       \if((PT_YIELD_FLAG == 0) || !(cond)) { \returnPT_YIELDED;      \}           \} while(0)

五、举例

//解码

static char
camera_rs485_rx_decode(struct pt *pt, uint8_t c)
{PT_BEGIN(pt);PT_WAIT_UNTIL(pt, 0x7E == c);camera_rs485.rx_count = 0;while(1){PT_YIELD(pt);if (0x7E== c){if(camera_rs485.rx_count){camera_rs485_dispatch(camera_rs485.rx_buf,camera_rs485.rx_count);PT_EXIT(pt);}else{continue;}}if(0x1B ==c){PT_YIELD(pt);if (0x00== c){c =0x1B;}else if (0x65 == c){c =0x7E;}else{//c ^=0x20;}}if(camera_rs485.rx_count < CAMERA_RS485_RX_BUF_SIZE){camera_rs485.rx_buf[camera_rs485.rx_count++] = c;}else{PT_EXIT(pt);}}PT_END(pt);
}

// 模块类型识别——定时器
static char
gps_probe(struct pt *pt, uint32_t ms)
{staticuint32_t tmo = 0;static inti;const char*cmd;if (tmo >ms){tmo -= ms;}else{tmo = 0;}PT_BEGIN(pt);tmo = 3000;PT_WAIT_UNTIL(pt, (0 == tmo));if(GPS_MODULE_TYPE_UNKNOWN == gps.module_type){for(i =GPS_MODULE_TYPE_UNKNOWN + 1; i < GPS_MODULE_TYPE_COUNT; ++i){if(gps_cmds[i].query_version){debug("gps_probe type: %d cmd: %s", i,gps_cmds[i].query_version);// 发送两次,确保模块接收到正确的命令cmd =gps_cmds[i].query_version;UartSend(gps_uart, cmd, strlen(cmd));PT_YIELD(pt);cmd =gps_cmds[i].query_version;UartSend(gps_uart, cmd, strlen(cmd));// 等待模块输出版本信息tmo =3000;PT_WAIT_UNTIL(pt, (0 == tmo));}if(GPS_MODULE_TYPE_UNKNOWN != gps.module_type){break;}}}// stopPT_WAIT_UNTIL(pt, FALSE);PT_END(pt);
}



本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/466810.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

南橘北枳

春秋战国时期&#xff0c;智者晏子曾经说过一句话&#xff1a;“桔生淮南为橘&#xff0c;生于淮北为枳”&#xff0c;说的是淮南香甜的橘子移植到淮北就变成苦涩难吃的枳。因为事物的条件和环境等变了&#xff0c;所以才会这样。这句话现在大多是针对人品说的&#xff0c;大概…

CentOS经常使用文件操作命令[百度博客搬家]

路径操作的CentOS经常使用命令 如今整理例如以下&#xff08;百度博客搬家) cd pwd NO1. 显示当前路径 [rootrehat root]# pwd NO2. 返回用户主文件夹 [rootrehat root]# cd NO3. 改变到其他路径 [rootrehat root]# cd /etc NO4. 返回到上一级文件夹 [rootrehat root]…

Apache配置SSL证书服务器傻瓜步骤

在LinuxApacheOpenSSL中配置SSL安全证书认证是不难的&#xff0c;我的另一篇工作随记中曾提到[url]http://www.host01.com/article/server/00070002/0621409075193649.htm[/url]中有所有步骤&#xff0c;不过其中的一些描述有点复杂&#xff0c;并且依赖关系也没有这些严格。这…

用C语言字符画圆

用C语言画圆&#xff0c;这是一是算法的问题&#xff0c;我们知道&#xff0c;程序是由数据结构加算法组成的&#xff0c;研究这类东西&#xff0c;其实也是研究算法的问题&#xff0c;是比较有意思的。圆心为0的时候&#xff0c;圆的公式公式&#xff1a;那我们要如何写代码呢…

Datawhale组队-pandas(上)基础(自学)

Pandas是Python的核心数据分析支持库&#xff0c;提供了快速灵活、明确的数据节后&#xff0c;旨在简单、直观地处理关系型、标记型数据。Pandas的目标是称为Python数据分析时间与实战的必备高级工具&#xff0c;其长远目标是称为最强大、最灵活&#xff0c;可以支持任何语言的…

Linux C目标文件

LinuxC目标文件 宗旨&#xff1a;技术的学习是有限的&#xff0c;分享的精神是无限的。 一、目标文件格式&#xff08;ELF格式&#xff09; 编译器编译源代码后生成的文件叫做目标文件。目标文件是已经编译后的可执行文件&#xff0c;只是还没有经过链接的过程。 PC平台流行…

Win2003下Exchange2003部署图解之七

Exchange 2003 基本配置在邮件服务器SERVERVM下打开“第一个存储组”&#xff0c;&#xff08;这里我们只看到一个缺省的存储组&#xff0c;在Exchange2003标准版中我们只能建立一个存储组&#xff0c;而企业版中我们可以建立4个存储组&#xff09;选择“邮箱存储”&#xff0c…

嵌入式Linux操作UART实例

1引言串口是我们实际工作中经常使用的一个接口&#xff0c;比如我们在Linux下使用的debug串口&#xff0c;它用来登录Linux系统&#xff0c;输出log。另外我们也会使用串口和外部的一些模块通信&#xff0c;比如GPS模块、RS485等。这里对Linux下串口使用做个总结&#xff0c;希…

Datawhale组队-Pandas(下)文本数据(打卡)

一、string类型的性质 1.string和object的区别 string类型和object不同之处有三&#xff1a; 字符存取方法&#xff08;string accessor methods&#xff0c;如str.count&#xff09;会返回相应数据的Nullable类型&#xff0c;而object会随缺失值的存在而改变返回类型某些Se…

sql语句遇到的问题

分别统计所有男同学的平均分&#xff0c;所有女同学的平均分及总平均分 //SELECT AVG(score),name,(SELECT AVG(score) FROM student) as 总平均分 from student GROUP BY sex; 按照分数从小到大的顺序打印分数大于总平均分的学员信息(id-name-sex-score),并将分数大于总平均分…

新唐单片机代码评审总结

昨晚上&#xff0c;我们一个同事组织了一个小会议&#xff0c;大家一起讨论了一个项目的单片机代码&#xff0c;这个单片机用的是新唐单片机&#xff0c;期间大家也讨论了一些问题&#xff0c;总结一下&#xff0c;希望对写单片机的同学们有帮助。我这个同事写的代码非常优秀&a…

Linux直接与编译安装Vsftpd服务器

1、VSFTPD简述如果试问哪种FTP服务器最安全&#xff1f;那么在UNIX和Linux中&#xff0c;首推的就是VSFTP&#xff08;Very Secure FTP Daemon&#xff0c;非常安全的FTP服务器&#xff09;。顾名思义&#xff0c;VSFTPD设计的出发点就是安全性。同时随着版本的不断升级&#x…

main函数

main函数 宗旨&#xff1a;技术的学习是有限的&#xff0c;分享的精神是无限的。 Linux下运行的程序多数是带参数的&#xff0c;例如ls –l等。一般来说&#xff0c;所有程序代码都是从main函数开始执行的。 main函数 &#xff08;1&#xff09;函数原型 int main(int argc,…

Datawhale组队-Pandas(下)分类数据(打卡)

Categoricals是pandas的一种数据类型&#xff0c;对应于统计学中的Categorical variables&#xff08;分类变量&#xff09;&#xff0c;分类变量是有限且固定的可能值&#xff0c;例如&#xff1a;gender&#xff08;性别&#xff09;、血型、国籍等&#xff0c;与统计学的Cat…

【分享】一个集成tracert和ping的网络监测工具

最近接到一个需求&#xff0c;需求背景是这样的&#xff1a;目前Windows平台下本身都有tracert和ping的实现&#xff0c;而且可以直接在cmd下使用。 需求中有两个要求&#xff1a; 1. Windows平台中的tracert执行速度太慢&#xff0c;一次tracert可能要花十几分钟。所以&#x…

秀操作 | 函数宏的三种封装方式

作者&#xff1a;☆星轨★ 链接&#xff1a;https://blog.csdn.net/qq_35692077/article/details/1029949591. 函数宏介绍函数宏&#xff0c;即包含多条语句的宏定义&#xff0c;其通常为某一被频繁调用的功能的语句封装&#xff0c;且不想通过函数方式封装来降低额外的弹栈压…

Sun为何一掷千金拿下MySQL?

标题&#xff1a;深度解析 Sun为何一掷千金拿下MySQL&#xff1f;作者&#xff1a;IT168 东方蜘蛛链接&#xff1a;[url]http://tech.it168.com/j/2008-01-21/200801211529940.shtml[/url]摘要&#xff1a;近日美国著名软件开发商Sun宣布&#xff0c;将以10亿美元的天价收购开源…

Datawhale组队-Pandas(下)时序数据(打卡)

Pandas可以处理任何领域的时序数据&#xff08;time series&#xff09;&#xff0c;使用Numpy的datetime64 和timedelta64 类型&#xff0c;Pandas整合了来自其他Python库的大量功能&#xff0c;如Scikits.TimeSeries&#xff0c;并为处理时间序列数据创建了大量新功能。 一、…

Linux C高级编程——文件操作之系统调用

Linux C高级编程文件操作之系统调用 宗旨&#xff1a;技术的学习是有限的&#xff0c;分享的精神是无限的&#xff01; 库函数是一些完成特定功能的函数&#xff0c;一般由某个标准组织制作发布&#xff0c;并形成一定的标准。使用库函数编写的函数一般可以应用于不同的平台而…

解析一个C语言俄罗斯方块游戏,包你看了就会

最近在看俄罗斯方块的游戏&#xff0c;看到一个大神写的俄罗斯方块代码&#xff0c;非常非常优秀&#xff0c;拿出来解析给大家看看&#xff0c;也希望大家自己尝试运行试试&#xff0c;从中能得到一些启发。#先了解下俄罗斯方块的几个形状一共分成 7 形状&#xff0c;有的形状…