原创 C语言嵌入式系统编程修炼之软件架构篇

2006-11-7 14:36 3221 4 5 分类: MCU/ 嵌入式


C语言嵌入式系统编程修炼之软件架构篇


 


模块划分



C语言模块化程序设计需理解如下概念:


  (1) 模块即是一个.c文件和一个.h文件的结合,头文件(.h)中是对于该模块接口的声明;


  (2) 某模块提供给其它模块调用的外部函数及数据需在.h中文件中冠以extern关键字声明;


  (3) 模块内的函数和全局变量需在.c文件开头冠以static关键字声明;


  (4) 永远不要在.h文件中定义变量!定义变量和声明变量的区别在于定义会产生内存分配的操作,是汇编阶段的概念;而声明则只是告诉包含该声明的模块在连接阶段从其它模块寻找外部函数和变量。


 


一个嵌入式系统通常包括两类模块:


  (1)硬件驱动模块,一种特定硬件对应一个模块;


  (2)软件功能模块,其模块的划分应满足低偶合、高内聚的要求。


多任务还是单任务


  所谓"单任务系统"是指该系统不能支持多任务并发操作,宏观串行地执行一个任务。而多任务系统则可以宏观并行(微观上可能串行)地"同时"执行多个任务。


  多任务的并发执行通常依赖于一个多任务操作系统(OS),多任务OS的核心是系统调度器,它使用任务控制块(TCB)来管理任务调度功能。TCB包括任务的当前状态、优先级、要等待的事件或资源、任务程序码的起始地址、初始堆栈指针等信息。调度器在任务被激活时,要用到这些信息。此外,TCB还被用来存放任务的"上下文"(context)。任务的上下文就是当一个执行中的任务被停止时,所要保存的所有信息。通常,上下文就是计算机当前的状态,也即各个寄存器的内容。当发生任务切换时,当前运行的任务的上下文被存入TCB,并将要被执行的任务的上下文从它的TCB中取出,放入各个寄存器中。


 


究竟选择多任务还是单任务方式,依赖于软件的体系是否庞大。例如,绝大多数手机程序都是多任务的,但也有一些小灵通的协议栈是单任务的,没有操作系统,它们的主程序轮流调用各个软件模块的处理程序,模拟多任务环境。


 


单任务程序典型架构


  (1)从CPU复位时的指定地址开始执行;


  (2)跳转至汇编代码startup处执行;


  (3)跳转至用户主程序main执行,在main中完成:


  a.初试化各硬件设备;


  b.初始化各软件模块;


  c.进入死循环(无限循环),调用各模块的处理函数


  用户主程序和各模块的处理函数都以C语言完成。用户主程序最后都进入了一个死循环,其首选方案是:


while(1)
{
}


 


中断服务程序


  中断是嵌入式系统中重要的组成部分,但是在标准C中不包含中断。许多编译开发商在标准C上增加了对中断的支持,提供新的关键字用于标示中断服务程序(ISR),类似于__interrupt、#program interrupt等。当一个函数被定义为ISR的时候,编译器会自动为该函数增加中断服务程序所需要的中断现场入栈和出栈代码。


中断服务程序需要满足如下要求:


  (1)不能返回值;


  (2)不能向ISR传递参数;


  (3) ISR应该尽可能的短小精悍;


  (4) printf(char * lpFormatString,…)函数会带来重入和性能问题,不能在ISR中采用。



硬件驱动模块


  一个硬件驱动模块通常应包括如下函数:


  (1)中断服务程序ISR


  (2)硬件初始化


  a.修改寄存器,设置硬件参数(如UART应设置其波特率,AD/DA设备应设置其采样速率等);


  b.将中断服务程序入口地址写入中断向量表:


/* 设置中断向量表 */
m_myPtr = make_far_pointer(0l); /* 返回void far型指针void far * */
m_myPtr += ITYPE_UART; /* ITYPE_UART: uart中断服务程序 */
/* 相对于中断向量表首地址的偏移 */
*m_myPtr = &UART _Isr; /* UART _Isr:UART的中断服务程序 */


  (3)设置CPU针对该硬件的控制线


  a.如果控制线可作PIO(可编程I/O)和控制信号用,则设置CPU内部对应寄存器使其作为控制信号;


  b.设置CPU内部的针对该设备的中断屏蔽位,设置中断方式(电平触发还是边缘触发)。


  (4)提供一系列针对该设备的操作接口函数。例如,对于LCD,其驱动模块应提供绘制像素、画线、绘制矩阵、显示字符点阵等函数;而对于实时钟,其驱动模块则需提供获取时间、设置时间等函数。


  C的面向对象化


  在面向对象的语言里面,出现了类的概念。类是对特定数据的特定操作的集合体。类包含了两个范畴:数据和操作。而C语言中的struct仅仅是数据的集合,我们可以利用函数指针将struct模拟为一个包含数据和操作的"类"。下面的C程序模拟了一个最简单的"类":


#ifndef C_Class
#define C_Class struct
#endif
C_Class A
{
 C_Class A *A_this; /* this指针 */
 void (*Foo)(C_Class A *A_this); /* 行为:函数指针 */
 int a; /* 数据 */
 int b;
};


  我们可以利用C语言模拟出面向对象的三个特性:封装、继承和多态,但是更多的时候,我们只是需要将数据与行为封装以解决软件结构混乱的问题。C模拟面向对象思想的目的不在于模拟行为本身,而在于解决某些情况下使用C语言编程时程序整体框架结构分散、数据和函数脱节的问题。我们在后续章节会看到这样的例子。


  总结


  本篇介绍了嵌入式系统编程软件架构方面的知识,主要包括模块划分、多任务还是单任务选取、单任务程序典型架构、中断服务程序、硬件驱动模块设计等,从宏观上给出了一个嵌入式系统软件所包含的主要元素。


  请记住:软件结构是软件的灵魂!结构混乱的程序面目可憎,调试、测试、维护、升级都极度困难。


 


 


 


 


 


 


 


 


 


 


 

PARTNER CONTENT

文章评论1条评论)

登录后参与讨论

用户1053025 2006-11-7 16:58

最后一句话,深有感触。宁可花上三天把逻辑搞清楚,也不要一头扎进泥潭里面。
相关推荐阅读
用户479067 2008-03-31 10:03
资源网站
http://softdata.studa.com           //下载 http://ibook8.com/http://www.dzzl.cn/   电子资料城www.ithao123.c...
用户479067 2008-02-28 17:53
地的分类
1. 信号“地”;信号“地”又称参考“地”,就是零电位的参考点,也是构成电路信号回路的公共段,图形符号“⊥”。1) 直流地:直流电路“地”,零电位参考点。2) 交流地:交流电的零线。应与地线区别开。3...
用户479067 2008-02-28 17:52
0欧姆电阻的作用
0欧姆电阻的作用大概有以下几个功能:①做为跳线使用。这样既美观,安装也方便。②在数字和模拟等混合电路中,往往要求两个地分开,并且单点连接。我们可以用一个0欧的电阻来连接这两个地,而不是直接连在一起。这...
用户479067 2008-02-28 17:48
上拉电阻
上拉电阻:1、当TTL电路驱动COMS电路时,如果TTL电路输出的高电平低于COMS电路的最低高电平(一般为3.5V),这时就需要在TTL的输出端接上拉电阻,以提高输出高电平的值。2、OC门电路必须加...
用户479067 2008-02-28 17:46
电平转换方案
 1. 常用的电平转换方案(1) 晶体管+上拉电阻法    就是一个双极型三极管或 MOSFET,C/D极接一个上拉电阻到正电源,输入电平很灵活,输出电平大致就是正电源电平。(2) OC/OD 器件+...
用户479067 2008-02-28 17:40
字节对齐
字节对齐一、概念    对齐跟数据在内存中的位置有关。如果一个变量的内存地址正好位于它长度的整数倍,他就被称做自然对齐。比如在32位cpu下,假设一个整型变量的地址为0x00000004,那它就是自然...
我要评论
1
4
关闭 站长推荐上一条 /3 下一条