打印

单片机开发中的一些技巧

[复制链接]
1658|5
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
liedc|  楼主 | 2012-4-13 15:01 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
很多朋友正在学习单片机开发技术,但开发中免不了要碰到这样、那样的问题,有些问题可能无碍大局,但有一些问题却直接影响到产品的成本、体积、性能。这里介绍笔者的几个技巧,希望对大家的工作有帮助。
    一.C语言中嵌入汇编语言
    单片机开发中,通常我们使用C语言编写主程序,这样可以充分借助C语言工具提供的运算库函数及强大的数据处理能力。但C语言的可控性不及汇编语言,在有些对时序要求严格的处理上,我们还需用灵活性更强的汇编语言来编写。这样就产生了C语言和汇编语言混合编程的问题,一般分成三种方式:1.汇编语言调用C语言函数;2. C语言调用汇编语言;3. C语言中嵌入汇编语言。这里我们主要介绍第3种,即C语言中嵌入汇编语言。
    下面的一段程序是主程序调用精确的205μS延时子程序并使P1.0交替输出高、低电平的方波。
/*------------程序名test.c------------*/
#include <AT89X51.H>//晶振频率12.000MHz
/****************/
void delay(void)//延时205μS
{
#pragma asm
MOV R0,#100
LOOP:
DJNZ R0,LOOP
#pragma endasm
}
/***************/
void main (void)//主函数,其功能使P1.0交替输出高、低电平的方波

{
while(1)
{P1_0=!P1_0;
delay();}
}

    具体实现过程为:
    1.先用汇编语言编制一段延时程序,在keil开发环境中编译,然后进行软件仿真,晶振频率的设置应和你的要求相符。仿真时注意观察左边寄存器窗口内的时间显示,调整延时程序的参数可得到我们需要的精确延时。
    2.用C51编写主程序及延时子程序的外壳(等待嵌入汇编语言),假定此程序名称为test.c。
    3.将第1步所得的汇编延时子程序放入C51编写的延时子程序外壳中。注意在开始及结束时分别加上#pragma asm、#pragma endasm语句,这种方法是通过asm与endasm告诉C51编译器,中间行不用编译为汇编行。
    4.按照Keil的使用方法,建立工程文件并添加源程序。
    5.点击含有汇编程序的C源程序后再右击,在弹出的下拉菜单中选中Options for File ‘test.c’(图1),这时出现图2所示的界面,勾选Generate Assembler SRC File(生成汇编SRC文件)及Assembler SRC File(封装汇编文件)使其有效。
    6. 根据项目的编译模式加载封装库文件,通常在Small模式时为C51S.LIB(该文件在C:\Keil\C51\Lib\C51S.LIB),具体见图3。
    7.点击Rebuild target(重建所有目标文件)即可得到编译结果(图4)。



图1


图2


图3


图4
    二.用软件扩展外部中断
    大家知道,51单片机的外部中断只有2个,书本上曾介绍了一种扩展外部中断源的方法,但是需增加硬件开销(见图5)。经或非门引入外中断源输入端(/INT0或/INT1),同时又连到某I/0口。这样,每个“源”都可能引起中断,在中断服务程序中通过软件查询便可确定哪一个是正在申请的中断源,其查询的次序则由中断源优先级决定,这就可实现多个外部中断源的扩展。


图5
这种方法尽管扩展了外部中断源,但也有不尽人意之处,如设计一个具有8个中断源的电路,则需一个8输入端的或非门(或门),显然,对体积与成本都不利。这里介绍笔者设计的扩展外部中断源的方法,由纯软件实现,不添加一个元件(见图6)。


图6
#include <AT89X51.H>
static unsigned char data m;//m为全局变量
/*-------延时子程序-------*/
void delay(unsigned int k)
{
unsigned int i,j;
for(i=0;i<k;i++){
for(j=0;j<121;j++)
{;}}
}
/*---外部中断INT0子程序---*/
void init0()interrupt 0
{
delay(10);//延时10mS抗抖动干扰
if(P3_2==0)   
{
    EX0=0;//关INT0中断
    EA=0;//关总中断
    P3_2=0;//置P3.2为低电平
    P2=0xff;//置P2口为全1
    m=P2;//读取P2口状态至m
    P2=0x00;//恢复P2口为全0
    P3_2=1; //置P3.2为高电平
    IT0=1;//置INT0为边沿触发
    EX0=1; //开INT0中断
    EA=1;} //开总中断
}
/********主程序*********/
void main(void)
{
P2=0x00;// 置P2口为全0
P3_2=1;// 置P3.2为高电平
IT0=1;// 置INT0为边沿触发
EX0=1;// 开INT0中断
EA=1; //开总中断
while(1)//无限循环
{
P0=m;//将全局变量m中的内容输出至P0口
P3_0=!P3_0;//P3.0取反,指示程序状态
delay(500);//延时500mS
}
}
    程序解释:无按键按下时,P3.0的发光管闪亮,作程序状态显示。主程序初始化时,置P2口为全0,置P3.2为高电平,同时置INT0为边沿触发,并开放中断。8个按键的任一个按下时都会引起INT0中断,进入中断服务子程序后,首先关闭中断,然后置P3.2为低电平,置P2口为全1,再读取P2口状态至m,通过查询m的状态字即可知道正在申请的中断源。这里我们采用的方法是将m输出至P0口点亮LED作指示。退出中断时,重新开放中断。

相关帖子

沙发
liedc|  楼主 | 2012-4-13 15:01 | 只看该作者
一.库函数的生成
    当将自己开发的程序提供给他人使用但又不便公开源代码时,把源代码做成库函数是一种可行的办法,这样可以保护自己的知识产权及利益,这里我们介绍生成库函数的方法及使用。

/*------------程序名test1.c------------*/
void delay(unsigned int k)
{
unsigned int i,j;
for(i=0;i<k;i++){
for(j=0;j<121;j++)
{;}}
}
1.按照keil的使用方法,建立工程文件test1.uv2并添加上面的源程序test1.c。
2.点击工程,在弹出的下拉菜单中点Options for Target ‘Target 1’,在Output 页面中,选中“Create Library:”后进行编译,则在指定的路径上生成与项目同名的“Lib”文件(图1)。需注意的是,存储模式(Large或Small)应与所使用的系统设置相同。

图1
3. 建立另一个工程文件test2.uv2。
/*------------程序名test2.c------------*/
#include <AT89X51.H>//晶振频率12.000MHz
/****************/
extern void delay(void);
void main (void)//主函数,其功能使P1.0交替输出高、低电平的方波
{
while(1)
{P1_0=!P1_0;
delay();}
}
4.将包含主程序的test2.c及刚才生成的test1.LIB添加到工程中(图2)。在Output 页面中,勾选建立hex文件。


图2
5. 点击Rebuild target(重建所有目标文件)即可得到编译结果(图3)。



图3

二.修改Startup.a51起始代码
单片机运行过程中免不了受干扰,有时可能会造成死机,我们可以使用“看门狗”来复位并重启单片机。根据笔者的经验,这时的内存区数据可能不一定会全部冲毁,主要是PC指针错乱所为。但使用C51编写的程序在复位后会执行一段Startup.a51“起始代码”,导致内存全部清零,使正在运行的数据全部丢失。解决这一问题的办法是修改Startup.a51“起始代码”,本刊今年1月的**《谈谈C语言在单片机开发中的应用》也谈到这个问题,但许多读者在keil集成开发环境中不知怎么做?这里我们通过一个实验程序来详解一下,实验采用《手把手教你学单片机》讲座的S2试验板(S2板的电路原理见2003年2月号《电子制作》)。

/*------------程序名test3.c------------*/
#include <AT89X51.H>//晶振频率11.0592MHz
#define uchar unsigned char
#define uint unsigned int
uchar code DATA_7SEG[10]={0xC0,0xF9,0xA4,0xB0,0x99,//0~9数码管字形码
                     0x92,0x82,0xF8,0x80,0x90};
uchar data counter1, counter2;//定义两个软件计数器
void delay(uint k)      //延时子程序
{
uint i,j;
for(i=0;i<k;i++){
for(j=0;j<121;j++)
{;}}
}
void main(void)     //主程序
{ delay(1);         //延时1mS
while(1)           //无限循环
{
if(counter1==counter2)//如两个计数值相等
{P0= DATA_7SEG[counter1];//输出至P0口显示
delay(500);          //延时500mS
counter1++;counter2++;//计数值递增
if(counter1>=10){ counter1=0;counter2=0;}//计数值在0~9循环
}
else
{ counter1=0xff;counter2=0xff;//否则计数值置0xff
//…………出错处理
}
}
}
1.按照keil的使用方法,建立工程文件test3.uv2并添加上面的源程序test3.c。在Output 页面中,勾选建立hex文件。
2.点击Rebuild target(重建所有目标文件)可得到编译结果。
3. 编译通过后,将生成的test3.hex文件烧录到单片机89C51中,将89C51芯片插入到S2型试验板上,通电运行后,右边的数码管从0至9开始循环显示。显示到某个数(例如5)时,按一下RESET键,右边的数码管又从0至9开始循环显示。 这是因为带电复位(热启动)时,C51执行了一段“起始代码”,将内存的128个单元全部清零,导致计数值(例如5)丢失。
解决的步骤如下:
4.点击“文件”,在下拉菜单中选择“打开”,在弹出的搜寻路径中,选择C:\Keil\C51\Lib\Startup.a51后打开,可见到如下代码:
………………………………………………………………………………………………
………………………………………………………………………………………………
IDATALEN        EQU     80H     ; the length of IDATA memory in bytes.
;
XDATASTART      EQU     0H      ; the absolute start-address of XDATA memory
XDATALEN        EQU     0H      ; the length of XDATA memory in bytes.
;
PDATASTART      EQU     0H      ; the absolute start-address of PDATA memory
PDATALEN        EQU     0H      ; the length of PDATA memory in bytes.
………………………………………………………………………………………………
………………………………………………………………………………………………
我们将IDATALEN        EQU     80H     ; the length of IDATA memory in bytes.改为IDATALEN        EQU     00H     ; the length of IDATA memory in bytes.然后保存关闭。
5. 将Startup.a51添加到test3.uv2工程中(图4)。

图4
6. 点击Rebuild target(重建所有目标文件)可得到编译结果。
7. 将生成的test3.hex文件再烧录到单片机89C51中,将89C51芯片插入到S2型试验板上,通电运行后,右边的数码管从0至9开始循环显示。显示到5时,按一下RESET键,右边的数码管从5起继续计数显示(注意:这次不是从0开始),实现了热启动后的继续计数功能。
这种技术非常有用,如因干扰等因素导致“看门狗”动作后(即热启动),不会将原来正在处理的数据丢失,从而可继续工作下去。可能有的读者会问,一旦干扰冲毁了数据,那么继续工作的这些数据可能是错误的,岂不是错上加错。对于这个问题,我们可采取数据冗余的办法,如正在计数的值由两个内存单元保存(例如本例中的counter1与counter2),使用时两个内存单元数据进行对比,一旦不等说明干扰破坏了数据,可进行出错处理,否则可认为数据正确有效。

使用特权

评论回复
板凳
okyouwin| | 2012-4-13 17:03 | 只看该作者
好贴。顶个。沙发

使用特权

评论回复
地板
jxmzzr| | 2012-4-15 21:21 | 只看该作者
很好的心得体会,谢谢分享经验。顶一个

使用特权

评论回复
5
tee.| | 2012-4-16 15:36 | 只看该作者
顶个。多谢分享了

使用特权

评论回复
6
职场新鲜人| | 2012-4-17 11:15 | 只看该作者
我觉得做好单片机系统的可靠性测试也很关键

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

0

主题

205

帖子

0

粉丝