实用的嵌入式系统我们一般采用分层的架构,可以分成3层(硬件层、驱动层和应用层)或4层(硬件层、驱动层、操作系统层和应用层),这在例解嵌入式系统分层结构已有说明,但是应用程序本身是非常复杂的,尤其是可能涉及到各种人机交互、机机交互的问题,使得系统更加复杂,所以对系统程序的架构做进一步的分析是很有必要的。
下面的分析我们都假定系统在大的层次上分为3层,设备驱动已准备好,下面的代码为伪代码,看懂程序结构即可。
例:流水灯
/*exe-1*/
#include"config.h"
int main(void){unsigned char i=0;//leds_init();//while(1){for(i=0;i<8;i++){leds_on(i);delay(1000);leds_off();}}return 0;
}
例:按键控制数码管,数码管显示按下的按键名称1、2、3
1)采用查询方式
/*exe-2*/
#include "config.h"
int main(){unsigned char key=0;//初始化segment_init();key_init();//主循环while(1){key=getKey();switch(key){case 1:segment_off();segment_on(0,1);break;case 2:segment_off();segment_on(0,2);break;case 3:segment_off();segment_on(0,3);break;default::segment_off();segment_on(0,_OFF_);break;}delay(100);
}
2)采用中断方式
按键触发外部中断,在中断服务程序中处理按键,并显示键值
/*exe-3*/
#include "config.h"
void eint_isr(void){key=getKey();switch(key){case 1:segment_off();segment_on(0,1);break;case 2:segment_off();segment_on(0,2);break;case 3:segment_off();segment_on(0,3);break;default::segment_off();segment_on(0,_OFF_);break;}
}
int main(){unsigned char key=0;//初始化segment_init();key_init();VIC_init();//主循环while(1);
}
用上了中断后你就会爱上它,舍不得不用了。
例:处理按键、显示、控制、测量电机转速
按键三个,分别为加速,减速,停止,属于输入。显示属于输出,电机属于输出,电机测试属于输入。
完成上述工作需要实现的功能很多,不用操作系统的情况下,我们一般采用中断驱动的前后台结构
主函数的主循环中处理各种交互及业务,适合于简单任务
/*exe-4*/
#include "config.h"
void handle_key(){...
}
void handle_disp(){...
}
void handle_motor(){...
}int main(){unsigned char key=0;//初始化segment_init();key_init();motor_init();//主循环while(1){handle_key();handle_disp();handle_motor();}
}
代码和上述exe-4相似,将按键、显示等都放在中断程序中,主循环空转
中断驱动结构的主要问题是将业务逻辑也放在了中断服务程序中,但业务逻辑比较复杂时,中断响应可能不及时,并且结构复杂。
在各种任务中,有些任务是必然存在的:数据的的显示、按键的处理等,一般来说业务逻辑负责处理和产生数据。这时候我们可以采用所谓的MVC架构,M即模型(model),可以看作就是数据,V即视图(view),如何显示数据;而C即控制(control),也就是处理、产生数据;业务逻辑(C)处理数据、模型保存数据、视图显示数据,对于有交互的系统来说这是一种非常有效的架构,通常我们定时器中断周期性的处理显示问题、按键扫描问题。下面的结构仔细体会:
/*exe-5*/
#include "config.h"
//处理和显示的数据,相当于model
int keyValue=0;
int motor_cnt=0;//记录电机转速产生的脉冲数
int motorSpeed=0;
unsigned char pattern[]={_OFF_,_OFF_,_OFF_,_OFF_,_OFF_,_OFF_,_OFF_,_OFF_};
//同步变量
int motorCntrl_flag=0;
int speedDisp_flag=1;
int keyPress_flag=0;void sysTick_isr(void){//按键处理,获得键值--getKey()//置处理按键标志if(keyValue!=0){keyPress_flag=1;}//显示速度值----MVC中的viewsegment_disp(pattern,5);//处理电机速度if(time==1S){motorSpeed=motor_cnt*60;motor_cnt=0;speedDisp_flag=1;}
}
//下面都是业务逻辑了
void eint1_isr(void){//外部中断motor_cnt++;//测量发生中断的次数,通过一秒钟发生的中断次数计算电机转速}
void handle_key(){switch(keyValue){case 1:motorCntrlFlag=1;break;case 2:motorCntrlFlag=2;break;case 3:motorCntrlFlag=3;break;case 4:motorCntrlFlag=4;break;}
}
void handle_motor(){switch(motorCntrlFlag){case 1://电机加速...;break;case 2://电机减速...;break;case 3://电机停止...;break;case 4://电机启动...;break;}motorCntrlFlag=0;
}
void handle_disp(){int speed=motorSpeed;for(i=7;i>=0;i--){pattern[i]=speed%10;speed/=10;}
}
int main(){//初始化segment_init();key_init();motor_init();//主循环while(1){//主循环中主要放业务逻辑或需要处理时间很长的任务if(keyPress==1){handle_key();keyPress=0;}if(speedDisp_flag==1){handle_disp();speedDisp_flag=0;}if(motorCntrlFlag!=0){handle_motor();motorCntrlFlag=0;} }
}
MVC架构实现例1-4:
/*exe-1的改进*/
#include"config.h"
//显示的数据,相当于MVC中的model;
unsigned char pattern[]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80};
int main(void){unsigned char i=0;//leds_init();//while(1){for(i=0;i<8;i++){//显示任务,只复杂显示,与显示的内容无关,相当于viewleds_on(pattern[i]);delay(1000);leds_off();}}return 0;
}
/*exe-2的改进*/
#include "config.h"
//数据,相当于model
unsigned char pattern[]={_OFF_,};
int main(){unsigned char key=0;//初始化segment_init();key_init();//主循环while(1){//按键处理key=getKey();switch(key){case 1:pattern[0]=1;break;case 2:pattern[0]=2;break;case 3:pattern[0]=3;break;default::pattern[0]=_OFF_;break;}//显示处理segment_off();//消影segment_on(0,pattern[0]);//在0位显示按键值dealy(100);}
}
/*exe-3的改进*/
#include "config.h"
unsigned char pattern[]={_OFF_,_OFF_,_OFF_,_OFF_,_OFF_,_OFF_,_OFF_,_OFF_,};
void sysTick_isr(void){//按键处理key=getKey();if(key)pattern[0]=key;//显示segment_disp(*pattern,10);
}
int main(){unsigned char key=0;//初始化segment_init();key_init();VIC_init();//主循环while(1);
}
从上述分析可以看出,对于嵌入式系统来说,定时器是非常重要的。但其重要性还不仅如此,我们常常会遇到系统要求在什么样的时间条件下做什么事情、又在什么样的时间条件下做什么事情,这时定时器中断服务程序可能要控制多个同步变量、或进行多种处理、不仅仅计时,为处理效率考虑,我们常常发现定时器不够用了!
从这里我们也可以看出,嵌入式系统中有三个设备非常重要:GPIO、中断、定时器!一定要掌握。
这里还有几个问题需要解决:
随着系统复杂程度的增加,同步变量的数量越来越多,如何管理这些同步变量?----消息事件机制,统一管理
同一个消息,同一个任务,在不同的状态下需要执行不同的操作,业务逻辑复杂如何解决?---有限状态机
某轮询任务复杂、处理时间较长,影响其他任务的及时处理,如何解决?----大任务拆成小任务(部分解决问题,不是长久之计);多任务操作系统
这是另外一个比较复杂的话题,暂略。