为什么会有这个文章?在嵌入式的环境下,有需要执行多个并行的任务的需求,但是 RTOS 又太重了,比如控制 LED 闪烁的时候,同时控制蜂鸣器,同时又要解析串口数据,又有一堆需要等待运行的任务,如 GPS 对时,配置通信模块,但是远距离通信模块的速率又不能太快,容易爆缓存,发送一条以后需要等待一定的时间,如果使用简单的 HAL_Delay 会导致 CPU 空转,影响其他任务的执行,这个时候需要一种轻量级的多任务工具,在不同任务之间切换,Delay 的时候释放出 CPU 资源给其他任务使用,这就是协程。
首先思考一个问题,我们现在的需求是在单个任务中,在 delay 的时候,要将 cpu 释放出来,下次再进来的时候,可以恢复运行到现在的位置。在有操作系统的情况下,可以通过保存上下文(比如程序计数器)保存在堆栈上,然后任务调度器再去恢复别的任务,进行上下文切换,遗憾的是似乎只有更底层的汇编语言才能做到这一点。协程虽然也可以模仿这个去实现,大概意思是:传统意义上的函数调用是怎样的,协程也尝试去模仿、改造、封装。但是在嵌入式的单线程情况下,这虽然实现了协程的上下文切换,相较之下前者在应用上会产生相当的不确定性(比如不好封装),不希望引入太多的不确定性,稳定才是第一要素。
intfunction(void) { staticint i, state = 0; switch (state) { case0: goto LABEL0; case1: goto LABEL1; } LABEL0: /* start of function */ for (i = 0; i < 10; i++) { state = 1; /* so we will come back to LABEL1 */ return i; LABEL1:; /* resume control straight after the return */ } }
intfunction(void) { staticint i, state = 0; switch (state) { case0: /* start of function */ for (i = 0; i < 10; i++) { state = 1; /* so we will come back to "case 1" */ return i; case1:; /* resume control straight after the return */ } } }
到这里肯定很多人都一头雾水了,这 switch 还能这么用?写了十几年的 c 语言白写了?其实说白了 C 语言就是脱胎于汇编语言的,switch-case 跟 if-else 一样,无非就是汇编的条件跳转指令的另类实现而已。 还可以用 LINE 宏使其更加一般化:
1 2 3 4 5 6 7 8 9 10 11
intfunction(void) { staticint i, state = 0; switch (state) { case0: /* start of function */ for (i = 0; i < 10; i++) { state = __LINE__ + 2; /* so we will come back to "case __LINE__" */ return i; case __LINE__:; /* resume control straight after the return */ } } }
进一步使用宏提炼出一种范式,封装成组件:
1 2 3 4 5 6 7 8 9 10
#define Begin() static int state=0; switch(state) { case 0: #define Yield(x) do { state=__LINE__; return x; case __LINE__:; } while (0) #define End() } intfunction(void) { staticint i; Begin(); for (i = 0; i < 10; i++) Yield(i); End(); }
怎么样,看起来像不像发明了一种全新的语言?实际上我们利用了 switch-case 的分支跳转特性,以及预编译的 LINE 宏,实现了一种隐式状态机,最终实现了“yield 语义”。每次进入 function 以后,都会跳转到 Yield 处执行,然后释放出 CPU 的权限。
超级轻量级的无栈协程实现
下面的教程来自于一位 ARM 工程师、天才黑客 Simon Tatham 的开源 C 协程库 protothreads,这是一个全部用 ANSI C 写成的库,就是说,实现已经不能再精简了,几乎就是原语级别。事实上 protothreads 整个库不需要链接加载,因为所有源码都是头文件,类似于 STL 这样不依赖任何第三方库,在任何平台上可移植;总共也就 5 个头文件,有效代码量不足 100 行;API 都是宏定义的,所以不存在调用开销;最后,每个协程的空间开销是 2 个字节(是的,你没有看错,就是一个 short 单位的“栈”!)当然这种精简是要以使用上的局限为代价的,接下来的分析会说明这一点。
Protothreads 的原语和组件
1 2 3 4
#define LC_INIT(s) s = 0 #define LC_RESUME(s) switch (s) { case 0: #define LC_SET(s) s = __LINE__; case __LINE__: #define LC_END(s) }
但这种“原语”有个难以察觉的缺陷:就是你无法在 LC_RESUME 和 LC_END (或者包含它们的组件)之间的代码中使用 switch-case 语句,因为这会引起外围的 switch 跳转错误!为此,protothreads 又实现了基于 GNU C 的调度“原语”。在 GNU C 下还有一种语法糖叫做标签指针,就是在一个 label 前面加 &&(不是地址的地址,是 GNU 自定义的符号),可以用 void 指针类型保存,然后 goto 跳转:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
typedefvoid * lc_t; #define LC_INIT(s) s = NULL #define LC_RESUME(s) \ do { \ if (s != NULL) { \ goto *s; \ } } while (0) #define LC_CONCAT2(s1, s2) s1##s2 #define LC_CONCAT(s1, s2) LC_CONCAT2(s1, s2) #define LC_SET(s) \ do { \ LC_CONCAT(LC_LABEL, __LINE__): \ (s) = &&LC_CONCAT(LC_LABEL, __LINE__); \ } while (0)
staticint protothread1(struct pt *pt) { /* A protothread function must begin with PT_BEGIN() which takes a pointer to a struct pt. */ PT_BEGIN(pt);
/* We loop forever here. */ while (1) { delay_ticks = HAL_GetTick() + 1000; PT_WAIT_UNTIL(pt, HAL_GetTick() > delay_ticks); HAL_GPIO_Troggle(LED_Port,LED_Pin); }
PT_END(pt); }
staticint protothread2(struct pt *pt) { /* A protothread function must begin with PT_BEGIN() which takes a pointer to a struct pt. */ PT_BEGIN(pt);
/* We loop forever here. */ while (1) { delay_ticks = HAL_GetTick() + 1000; PT_WAIT_UNTIL(pt, HAL_GetTick() > delay_ticks); HAL_GPIO_Troggle(BEEP_Port,BEEP_Pin); }
PT_END(pt); }
/** * Finally, we have the main loop. Here is where the protothreads are * initialized and scheduled. First, however, we define the * protothread state variables pt1 and pt2, which hold the state of * the two protothreads. */ staticstructptpt1, pt2; int main(void) { /* Initialize the protothread state variables with PT_INIT(). */ PT_INIT(&pt1); PT_INIT(&pt2);
/* * Then we schedule the two protothreads by repeatedly calling their * protothread functions and passing a pointer to the protothread * state variables as arguments. */ while(1) { protothread1(&pt1); protothread2(&pt2); } }