历史上的今天
返回首页

历史上的今天

今天是:2025年03月20日(星期四)

正在发生

2018年03月20日 | 单片机多任务的实现方式

2018-03-20 来源:eefocus

    由于单片机具有价格低、运行要求低、易于开发、稳定可靠等优点,广泛应用于仪器仪表、家用电器、医用设备、航空航天、专用设备的智能化管理及过程控制等领域。但是,单片机的位数少、频率低、内存小、I/O口少等缺点限制了其加载操作系统的可能。因此,单片机不能像ARM等较高性能的处理器一样,利用加载的操作系统实现管理与配置内存、决定系统资源供需的优先次序、控制输入与输出设备、操作网络与管理文件系统等功能。
    但是,我们可以根据单片机所拥有的内存大小、CPU频率等因素,来为单片机量身定做一个小型的操作系统,以实现单片机的多任务运行。

1 微机实现多任务的方式
    微机实现多任务的方式一般是由加载的操作系统来实现的。通过操作系统提供的函数来创建多进程或者多线程来实现多任务方式。由于多进程耗费的资源多,而多线程的开销相对小的多,因此我们采用单片机模仿多线程的方式来实现。
    操作系统创建多个线程后,将管理各个线程占用CPU的时间。操作系统以轮换方式向线程提供CPU时间片,从而使多个线程看起来是同时运行的,而不是等待一个线程执行结束后再去执行下一个线程。
    PC(Program Counter,程序计数器)是用于存放下一条指令地址的地方。某个线程正在占用CPU时间,其实是PC值指向该线程所占的内存,并正在逐条取到CPU寄存器中进行运算。该时间片结束后,PC值要指向下一个线程所占用的内存中,进行类似的运算。其他线程都轮流一遍后,将又回到原来那个线程暂停的位置继续运算。所以,从一个线程转换到另外一个线程去执行时,要保存此线程的“现场”,包括此线程下一条指令的位置(PC值)、此线程所使用的各个寄存器值等。当此线程又拥有CPU时间时,将保存的PC值赋给PC寄存器,保存的各个寄存器值再赋给各个寄存器。
    除了保存“现场”与恢复“现场”外,另外关键的一点是,操作系统能够改变PC值——强制把使用CPU的权限从一个任务切换到另一个任务,这就用到了中断。微机是用操作系统来管理中断的,用户只能间接使用中断。

2 单片机实现多任务的思路
    由上面的介绍,我们知道微机中多线程轮流占用CPU时间,关键点在于:
    ①保存“现场”与恢复“现场”,即保存和恢复下一条指令的位置和通用寄存器的值。
    ②能够改变PC值,从而可以在多个线程中进行切换,以便同时运行。
    在51系列单片机中,如何实现上面的两个关键点呢?
    (1)保存此“现场”,恢复另一“现场”
    给每个任务开辟一个堆栈,各个任务的堆栈不能交叉。各个任务的对应堆栈用于实现以下功能:
    ①保存“现场”,在PC离开此任务前保存该任务所用到的通用寄存器值(寄存器A、B、Rn和位寄存器C等)。
    ②恢复“现场”,先获得下一个任务的堆栈地址,然后取出堆栈中所保存的通用寄存器值;
    ③在调用子函数时,用以保存下一条指令的地址。
    (2)每隔一段时间片,改变PC值几乎所有的处理器指令中,没有可以直接改变PC值的指令,但是系统发生中断时可以改变PC值,中断流程如图1所示。

a.JPG

    
    由图1可以看出,在倒数第二个步骤中,单片机会把栈顶的两个字节弹出给PC,由此来改变PC值,进而来改变程序的执行流程。所以,我们可以在出栈弹出字节给PC前改变栈顶的两个字节的内容,进而主动改变PC值。
    有了主动改变PC值的能力,我们就可以将这个中断设为定时器中断,每隔一段时间来切换PC值,进而实现多任务运行。

3 具体实现代码及注意事项
3.1 进入主循环前的工作
    根据上面的思路和技巧,进入主循环前的工作流程如图2所示。

b.JPG


    图2为进入主循环前的初始化工作。假定有3个任务,3个任务分别为Task1、Task2、Task3(这3个任务都应是死循环),如果开设每个堆栈大小为16字节,3个任务对应的堆栈范围为40H~4FH、50H~5FH、60H~6FH,则初始各个任务地址到对应堆栈如下:
   c.JPG
    sp1、sp2、sp3为定义的3个全局变量,用以存储各个任务的栈顶地址。


    初始化定时器后,要进入某个任务的死循环当中。假设我们要进入任务1中,则如下所示:
   d.JPG
    TaskIndex为全局变量,用以存储当前执行的任务序号;难点在于ret的妙用。ret一般用于子函数的最后一条,以回到调用函数前下一条指令的地址。ret的实质是取出此时堆栈中栈顶的两个字节赋给PC寄存器,以返回调用函数前的位置。所以,上述代码是先把任务1的地址放进堆栈中,然后调用ret来取出地址给PC,以重新跳到任务1中去执行。
3.2 多任务切换的主循环
    进入某个任务进行死循环后,程序的主循环流程如图3所示。当程序进入到某个任务进行死循环时,如上面的任务i,定时器中断周期发生,发生时意味着该任务的时间片结束,准备执行下一个任务。这些准备工作是在中断里做的,如图3所示。首先,应保存此时用到的各个寄存器值,以便下次轮到该任务时取出继续执行,还要保存栈顶的位置,以便下次能取出所保存的值;然后通过全局变量TaskIndex取得下一个任务的序号,通过任务序号,得到下一个任务的堆栈栈顶的地址,赋给栈顶寄存器SP;然后通过SP取出保存的各个通用寄存器值;最后,重设定时器值,使中断能够再次进行任务切换。

e.JPG

    
    这里重要的是整个思路,没有比较难的代码,故没有贴出代码。值得提醒的是,保存通用寄存器值时,并不需要保存所有的通用寄存器值,只需要保存任务中用到的就可以。这里解释前面程序中提及的45H、55H、65H:各个任务堆栈的开始处存储各个任务的地址,然后再把要保护的寄存器值入栈,栈顶抬高;而要恢复下一个任务时,需将上次保护寄存器后的栈顶值赋给SP寄存器,然后逐个出栈赋值给各个寄存器值,直到栈底处存储的上次任务暂停处的地址。因为本文的验证程序只保护了A、B、R0、R2 4个寄存器值,堆栈刚好到达45H、55H、65H。

总结
    单片机实现多任务的另一种常用方式是把任务切成小片,然后放在主循环里。这样,每个循环执行一次各个任务的一小片,从而看起来所有的任务都同时进行。切片的思想是把一个任务细分成多个步骤,而每次只执行其中一小步。如多段数码管的显示可以每次只显示一段,这是更常用的方式,但并不是每个任务都可以切片的。
    本文所讲的这种实现单片机多任务的方式要求程序员要有比较好的汇编基础,要求对中断的实现过程比较熟悉,对ret指令的实质要理解,能够根据任务来分配堆栈,对操作系统管理CPU时间片有大致理解,因此要求比较高。另一方面,时间片定多少需要程序员根据任务的不同来选择,需要测试多次来达到性能的最优化。


推荐阅读

史海拾趣

Black Box Corporation公司的发展小趣事

在成立初期,Black Box Corporation主要以提供语音通信解决方案为主。随着科技的快速发展,公司意识到单纯依赖语音通信已无法满足市场的多元化需求。于是,公司开始着手研发并推广全面的网络解决方案,包括数据传输、网络安全、云计算等。这一转型不仅使Black Box在行业内获得了更高的知名度,也为其带来了更多的商业机会。

EPIGAP公司的发展小趣事

EPIGAP公司非常重视创新文化的建设。公司鼓励员工提出新的想法和创意,并为员工提供充分的支持和资源。此外,EPIGAP还积极引进和培养人才,建立了一支高素质、专业化的研发团队。这支团队在公司的发展历程中发挥了至关重要的作用,不断推动公司产品的升级换代和技术创新。

DILABS公司的发展小趣事

为了进一步扩大市场份额,DILABS开始寻求与国内外知名企业的战略合作。他们与多家大型电子制造商签订了长期供货协议,同时也积极与高校和研究机构合作,共同研发新技术、新产品。这些战略合作不仅为DILABS带来了更多的订单,也促进了公司在技术和管理方面的不断进步。

AMERICASEMI [America Semiconductor, LLC]公司的发展小趣事

为了进一步扩大市场份额,AMERICASEMI积极寻求与其他企业的合作机会。公司与多家知名电子产品制造商建立了稳定的合作关系,为其提供优质的半导体产品和技术支持。同时,AMERICASEMI还积极开拓国际市场,将产品出口到全球多个国家和地区,实现了业务的快速增长。

ERP公司的发展小趣事

为了进一步拓展市场和提高竞争力,电子智链开始寻求与其他企业建立生态合作关系。公司与多家电子制造设备供应商、原材料供应商和物流服务商建立了战略合作关系,共同打造了一个覆盖电子产业全链条的生态圈。这一合作模式不仅为客户提供了更加全面和高效的解决方案,还促进了整个电子产业的协同发展。

安美通(APPCON)公司的发展小趣事

在发展过程中,安美通始终注重与合作伙伴的互利共赢。公司与多家知名企业建立了战略合作关系,共同开拓市场、分享资源。这些合作不仅为公司带来了更多的商业机会,也提高了其在行业内的知名度和影响力。

问答坊 | AI 解惑

modelsim下编译xilinx库的方法

建立ModelSim SE 的Xilinx仿真库方法,从网上搜到的有很多,实践过其中几种,其中一种较简单的方法如下:1)当然是要安装ModelSim 和ISE 。 2)将ModelSim根目录下的modelsim.ini文件的只读属性去掉。3)在ModelSim命令窗口输入:compxlib -s mti_s ...…

查看全部问答>

诚心求教, DeviceIoControl假死现象

小可刚接触USB(ez usb 68013)开发,遇到一个问题百思不得其解,想向大家讨教一下: 我用的是公板的驱动和固件,在应用程序里开双线程读数据(inBulkControl.pipeNum=1,),如果在读取的过程中再发送数据(outBulkControl.pipeNum=0),有时程序会停在读取数 ...…

查看全部问答>

s3c2440 Wince5.0 睡眠与唤醒

看了[url=http://www.hzlitai.com.cn/article/ARM9-article/1685.html][/url]尝试让系统睡眠 ; Sometimes it is not working in cache mode. So I modify to jump to ROM area.        ldr        ...…

查看全部问答>

Launch函数问题,高手请进

这一段是源代码 LEAF_ENTRY Launch                ldr            r2, = PhysicalStart                ldr&nbs ...…

查看全部问答>

USB Function总线驱动是在那里加载其DefaultClientDriver?

MSDN上说在[HKEY_LOCAL_MACHINE\\Drivers\\USB\\FunctionDrivers] 注册表项加个DefaultClientDriver注册表项,在机器启动的时候即可加载该Client Driver.但是我找遍了所有的代码,就是没找到是在那里启动DefaultClientDriver的?…

查看全部问答>

重载的问题

在一个动态连接库的.h文件里,有个结构体 #if defined DLL_EXPORT #define FVFIXED_API __declspec(dllexport) #else #define FVFIXED_API __declspec(dllimport) #endif struct FVFIXED_API CFvFixed { public:       &nbs ...…

查看全部问答>

想学习linux嵌入式开发,请教学习方法和教材

想学习linux嵌入式开发,请教学习方法和教材…

查看全部问答>

BENQ M23G调试出现奇怪的问题,大虾救救

出现的问题:能接电话,不能打电话,发短信正常,收短信的内容不对。 我的操作如下: AT+CFUN=1 AT+CFUN=1          AT+COPS=0          AT+IPR=115200     &nbs ...…

查看全部问答>

Driver studio 在VS2005下build出现问题

刚开始接触驱动程序的编写,想在winXP下用VS2005来写编写一些驱动,但是出现了一些问题。 我安装那三个文件的顺序是这样子的:     1。VS2005   在D盘       2。DDK (Windows XP SP1 DDK),装在F盘 ...…

查看全部问答>

复杂电路分析

拿到一张复杂的电路图,各位都会怎么分析呢??如何确定电路中电流的流向和信号的输入输出的呢??请前辈们介绍下自己的经验(最好能举出特定的例子进行说明),在这里万分感谢了。。。…

查看全部问答>