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

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

3天内不再提示

改进一个可实现多任务操作电表的代码程序

h1654155971.7688 2018-02-02 17:19 次阅读

2002年初,笔者着手写一个IC卡预付费电表的工作程序,该电表使用Philips公司的8位51扩展型单片机87LPC764,要求实现很多功能,包括熄显示、负荷计算与控制、指示闪烁以及电表各种参数的查询等,总之,要使用时间的单元很多。笔者当时使用ASM51完成了这个程序的编写,完成后的程序量是2KB多一点。后来,由于种种原因,这个程序并没有真正使用,只是作了一些改动之后用在一个老化设备上进行计时与负荷计算。约一年后,笔者又重新改写了这些代码。

1系统的改进

可以说,这个用ASM51实现的代码是没有什么组织性可言的,要什么功能就加入什么功能,弄得程序的结构非常松散,其实这也是导致笔者最终决定重新改写这些代码的原因。大家知道,87LPC764有4KB的FlashROM,而笔者的程序量只有2KB多点,因而第一个想法是改用C语言作为主要的开发语言,应该不至于导致代码空间不够用。其次,考虑到需要定时功能的模块(或称任务,以下统称任务)较多,有必要对这些任务进行有序的管理。笔者考虑使用时间片轮询方式,即给每个要求时间管理的任务以一个时间间隔,时间间隔一到,即运行其代码,达到合理使用系统定时器资源的目的。就51系统而言,一般至少一个定时器可用来进行时间片的轮询。基于以上的想法,构造了下述数据类型。

typedef unsigned char uInt8

typedef struct {

void (*proc)( void ); /* 处理程序 */

uInt8 ms_count; /* 时间片大小 */

} _op_;

数据结构定义好之后,接着就是实现代码,包括三部分,即初始化数据、时间片的刷新与时间到执行。

初始化数据。#defineproc_cnt0x08//定义过程或任务数量

//任务栈初始化

code _op_ Op[proc_cnt] = { { ic_check 10 } { disp_loop 100 }

{ calc_power 150 } { set_led 2 } … };

//设置时间片初始值

datauInt8time_val[proc_cnt]={10,100,150,2,…};

时间片刷新。

void time_int1( void ) interrupt 3

{

uInt8 cnt;

Time_Counter: = Time_Unit;

for ( cnt = 0; cnt < proc_cnt; cnt++ )

{

time_val[cnt]--;

}

}

任务的执行。

void main( void )

{

uInt8 cnt;

init(); /*程序初始化 */

interrupt_on(); /* 打开中断 */

do

{

for ( cnt = 0; cnt < proc_cnt; cnt++ )

{

if ( !time_val[cnt] )

{

time_val[cnt] = Op[cnt].ms_count;

Op[cnt].proc();

}

}

}

while ( 1 );

}

在上面的结构定义中,proc是不能带参数的,各任务之间的通信可以定义一个参数内存块,通过一种机制进行数据信息交互,如定义一个全局变量。对于小容量单片机系统而言,需要这样做的任务并不多,总任务量也不会太多,因而这种协调并不太难处理。也许大家都有这样的认识,即一个实时系统中,差不多所有的具体任务都是有时间属性的,即使是不需要定时的过程或任务,也不见得要时时进行查询与刷新。如IC卡介质检测,保证每秒一次就足够了。因而,这些任务也可以列入到这个结构中来。在以上的程序代码中,考虑到单片机系统的RAM限制,不能像一些实时OS那样将任务栈建立在RAM中。笔者将任务栈建立在代码空间,因而不能在程序运行时动态地加入任务,因此要求在程序编译时,任务栈已经确定。同时,定义一组计数值旗标time_val,记录程序运行时的时间量,并在一个定时器中断中对其进行刷新。改变时间片刷新中断过程语句Time_Counter:=Time_Unit;中的Time_Unit,可以改变系统时间片的刷新粒度,一般这个值由系统的最小时间度量值确定。同时,由任务的执行流程可知,此种系统构造并没有改变其前/后台系统的性质,只是对后台逻辑操作序列进行了有效管理。同时,如果将任务执行流程进行一些更改,并保证时间片小的任务前置,如下述程序。

do

{

for ( cnt = 0; cnt < proc_cnt; cnt++ )

{

if ( !time_val[cnt] )

{

time_val[cnt] = Op[cnt].ms_count;

Op[cnt].proc();

break; /* 执行完成后,重新进行优先调度 */

}

}

}

while ( 1 );

则系统变为一个以执行频率为优先级的任务调度系统。当然,设置此种方式得非常小心,并要注意时间片的分配,如果时间片过小,则可能导致执行频率较低的任务难以被执行;而如果存在两个同样的时间片,则更加危险,可能导致第二个具有相同时间片的任务不被执行,因而,时间片的分配要合理,并保证其唯一性。

2性能分析与任务拆分

以上两种任务管理方式,前一种按任务栈的顺序与时间片的大小依次进行调度,暂且称其为流水作业调度;而后一种,且称其为频率优先调度。两种方式各有优缺点。流水作业调度的各任务具有等同优先级,时间片一到即会被按序调用,时间片大小的次序与唯一性不作要求;缺点是可能导致时间片小的,即要求执行得较快的任务等待过长的时间。频率优先调度的各任务按其时间片的大小,即执行频率划分优先级,时间片小的任务,其执行频率高,总是具有较高的优先权,但时间片的分配得协调,否则可能会导致执行频率低的任务长时间等待。要特别注意的是,两种方式都有可能导致一些任务长时间等待,时间片所设定的时间也因此不能作为精确时间的依据,根据系统的要求或需要,甚至要在任务执行过程中进行某些保护工作,如中断屏蔽等,因而在进行任务规划时要注意。如果一个任务较繁琐或可能要等待很长时间,则应当考虑任务的拆分,把一个较大的任务细化为较小的任务,把一个费时长的任务划分为多个费时小的任务,协同完成其功能。如在等待时间长的情况下,可附加一个定时任务,定时任务到则发送一个消息旗标,主过程没有检测到消息旗标就马上返回,否则继续执行。下面是示例代码,假定该任务将等待很长时间,现将其拆分为两个任务proc1与proc2协同完成原来的工作,proc1每100个时间单位执行一次,而proc2每200个时间单位执行一次。

/* 定义两个任务,并将其加入到任务栈中。 */

code _op_Op[proc_cnt] = { …{ proc1 100 } { proc2 200 } };

data inttime1_Seg; /* 定义一个全局旗标 */

/* 任务实现 */

void proc1( void )

{

if ( time1_Seg )

exit;

else

time1_Seg = const_Time1; /* 如果时间到了,则恢复初值并 */

/* 接着执行下列代码。 */

… /* 任务实际执行代码 */

}

void proc2( void )

{

if ( time1_Seg )

time1_Seg--;

}

由上例可以看出,任务拆分后,几乎不占过多的CPU时间,使得任务的等待时间大减,让CPU有足够的时间进行任务管理与调度。同时也让程序的结构性与可读性大为加强。

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

    关注

    6037

    文章

    44569

    浏览量

    636179
  • IC
    IC
    +关注

    关注

    36

    文章

    5957

    浏览量

    175753
  • cpu
    cpu
    +关注

    关注

    68

    文章

    10873

    浏览量

    212093
  • 电表
    +关注

    关注

    12

    文章

    566

    浏览量

    35362

原文标题:如何用单片机完成多任务操作

文章出处:【微信号:weixin21ic,微信公众号:21ic电子网】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    实时多任务操作系统

    大多数嵌入式软件开发还是基于处理器直接编写,没有采用商品化的RTOS,不能将系统软件和应用软件分开处理。RTOS是段嵌入在目标代码中的软件,用户的其它应用程序都建立在RTOS之上。不但如此,RTOS还是
    发表于 08-12 14:31

    setjmp构建简单协作式多任务系统

    在实际应用中,抢中式的多任务某种程序上带来了用户程序设计时数据保护的困难,并且,具备抢占功能的多任务内核设计时困难也比较多,这会增加操作系统
    发表于 12-07 15:55

    实时多任务操作系统(RTOS)

    多任务处理是指用户可以在同时间内运行多个应用程序,每个应用程序被称作
    发表于 08-24 06:34

    如何去实现基于stm32的FreeRTOS多任务程序

    基于 stm32 的 FreeRTOS多任务程序实现、什么是FreeRTOS二、AHT20数据采集原理电路绘制三、FreeRTOS的多任务
    发表于 08-24 07:45

    多任务编程多任务处理是指什么

    嵌入式Linux应用编程-多任务编程多任务处理是指用户可在同时间内运行多个应用程序,每个正在执行的应用程序被称为
    发表于 12-22 08:30

    如何在STM32下完成基于FreeRTOS的多任务程序

    任务要求:在STM32下完成基于FreeRTOS的多任务程序,执行3周期性task;目录
    发表于 01-17 07:10

    VxWorks操作系统及实时多任务程序设计

    VxWorks操作系统及实时多任务程序设计
    发表于 03-25 10:38 13次下载

    VxWorks下实时多任务程序实现

    本文介绍了VxWorks操作系统的技术特点,给出了种VxWorks操作系统下实现实时多任务程序
    发表于 09-07 08:22 9次下载

    基于消息驱动的多任务操作机制

    本文对多任务操作机制与传统的单任务操作机制进行了比较,分析了多任务系统中数据交换的方法,特别是目前最为流行的用消息驱动方式
    发表于 05-14 16:56 0次下载

    MSP430单片机实时多任务操作系统c源代码

    : 资料-【C】嵌入系统-【C0】嵌入式综合-【2】单片机编程-【参考程序】-MSP430单片机的实时多任务操作系统c源代码.zip
    发表于 09-18 10:18 91次下载
    MSP430单片机实时<b class='flag-5'>多任务</b><b class='flag-5'>操作</b>系统c源<b class='flag-5'>代码</b>

    DSP实时多任务操作系统设计与实现

    DSP实时多任务操作系统设计与实现又需要的下来看看。
    发表于 05-09 10:15 0次下载

    51单片机多任务操作系统的原理与实现

    51单片机多任务操作系统的原理与实现,在51单片机上模拟操作系统
    发表于 02-28 15:09 16次下载

    基于实时操作系统的单片机多任务应用程序设计

    基于实时操作系统的单片机多任务应用程序设计
    发表于 10-27 15:21 49次下载
    基于实时<b class='flag-5'>操作</b>系统的单片机<b class='flag-5'>多任务</b>应用<b class='flag-5'>程序</b>设计

    种实时嵌入式多任务微内核的分析与改进

    种实时嵌入式多任务微内核的分析与改进(嵌入式开发系统)-种实时嵌入式多任务微内核的分析与改进
    发表于 07-30 13:49 11次下载
    <b class='flag-5'>一</b>种实时嵌入式<b class='flag-5'>多任务</b>微内核的分析与<b class='flag-5'>改进</b>

    STM32 简单多任务调度的方法与程序例程

    多任务处理是指用户可以在同时间内运行多个应用程序,每个应用程序被称作
    发表于 12-22 19:04 13次下载
    STM32 简单<b class='flag-5'>多任务</b>调度的方法与<b class='flag-5'>程序</b>例程