一、 辅助工具
这部份功能并不是直接用来进行程序调试的,但可以帮助我们进行程序的调试、程序性 能的分析,同样是一些很有用的工具。
1、外围接口
为了能够比较直观地了解单片机中定时器、中断、
并行端口、串行端口等常用外设的使用情况,Keil 提 供了一些外围接口对话框,通过 Peripherals 菜单选择, 该菜单的下拉菜单内容与你建立项目时所选的 CPU 有关,如果是选择的 89C51 这一类“标准”的 51 机, 那么将会有 Interrupt(中断)、I/O Ports(并行 I/O 口)、 Serial(串行口)、Timer(定时/计数器)这四个外围设
图 1 外围设备之并行端口
备菜单。打开这些对话框,列出了外围设备的当前使用情况,各标志位的情况等,可以在这 些对话框中直观地观察和更改各外围设备的运行情况。
下面我们通过一个简单例子看一看并行端口的外围设备对话框的使用。例 4:
MOVA,#0FEH
LOOP:MOVP1,A
RLA
CALLDELAY ;延时 100 毫秒
JMPLOOP
其中延时 100 毫秒的子程序请自行编写。
编 译 、 连 接 进 入 调 试 后 , 点 击 Peripherals->I/O-Ports->Port 1 打开,如图 1 所示,全速运 行,可以看到代表各位的勾在不断变化(如果看不到变化, 请点击 View->Periodic Window Updata),这样可以形象地 看出程序执行的结果。
注:如果你看到的变化极快,甚至看不太清楚,那么 说明你的计算机性能好,模拟执行的速度快,你可以试着 将加长延时程序的时间以放慢速度。模拟运行速度与实际 运行的速度无法相同是软件模拟的一个固有弱点。
点击 Peripherals->I/O-Ports->Timer0 即出现图 2 所示 定时/计数器 0 的外围接口界面,可以直接选择 Mode 组中 的下拉列表以确定定时/计数工作方式,0-3 四种工作方式,
图 2 外围设备之定时器
设定定时初值等,点击选中 TR0,status 后的 stop 就变成了 run,如果全速运行程序,此时
th0,tl0 后的值也快速地开始变化(同样要求 Periodic Window Updata 处于选中状态),直观地 演示了定时/计数器的工作情况(当然,由于你的程序未对此写任何代码,所以程序不会对 此定时/计数器的工作进行处理)。
2、性能分析
Keil 提供了一个性能分析工具,利用该工具,我们可以了解程序中哪些部份的执行时间 最长,调用次数最多,从而了解影响整个程序中执行速度的瓶颈。下面通过一个实例来看一 看这个工具如何使用,例 5:
#include "reg51.h"
sbit P1_0=P1^0; //定义 P1.0
void mDelay(unsigned char DelayTime)
{ unsigned int j=0;
for(;DelayTime>0;DelayTime--)
{ for(j=0;j<125;j++) {;} }
}
void mDelay1(unsigned char DelayTime)
{ unsigned int j=0;
for(;DelayTime>0;DelayTime--)
{ for(j=0;j<125;j++) {;} }
}
void main()
{ unsigned int i;
for(;;){ mDelay(10); // 延时 10
毫秒
i++;
if(i==10)
{ P1_0=!P1_0; i=0; mDelay1(10);}
} }
编译连接。进入调试状态后使用菜单 View->Performance Analyzer Window,打开性能分 析对话框,进入该对话框后,只有一项 unspecified,点鼠标右键,在快捷菜单中选择 Setup PA 即打开性能分析设置对话框,对于 C 语言程序,该对话框右侧的“Function Symbol”下的 列表框给出函数符号,双击某一符号,该符号即出现在 Define Performance Analyzer 下的编 缉框中,每输入一个符号名字,点击 Define 按钮,即将该函数加入其上的分析列表框。对 于汇编语言源程序,Function Symbol 下的列表框中不会出现子程序名,可以直接在编缉框 中输入子程序名,点击 Close 关闭窗口,回到性能分析窗口,此时窗口共有 4 个选项。全速 执行程序,可以看到 mDelay 和 mDelay1 后出现一个蓝色指示条,配合上面的标尺可以直观 地看出每个函数占整个执行时间的比例,点击相应的函数名,可以在该窗口的状态栏看到更 详细的数据,其中各项的含义如下:
Min:该段程序执行所需的最短时间;Max:该段程序执行所需的最长时间;Avg:该 段程序执行所花平均时间;Total:该段程序到目前为目总共执行的时间;%:占整个执行时 间的百分比;count:被调用的次数。
本程序中,函数 mDelay 和 mDelay1 每次被调用都花费同样的时间,看不出 Min、Max、
和 Avg 的意义,实际上,由于条件的变化,某些函数执行的时间不一定是一个固定的值, 借助于这些信息,可以对程序有更详细的了解。下面将 mDelay1 函数略作修改作一演示。
void mDelay1(unsigned char DelayTime)
{ static unsigned char k;
unsigned int j=0;
for(;DelayTime>0;DelayTime--)
{ for(;j
{;}
} k++; }
程序中定义了一个静态变量 K,每次调用该变量加 1,而 j 的循环条件与 k 的大小有关,
这使每次执行该程序所花的时间不一样。编译、执行该程序,再次观察性能分析窗口,可以
看出 Min、Max、Avg 的意义。
3、变量来源浏览
该窗口用于观察程序中变量名的有关信息,如该变量名在那一个函数中被定义、在哪里 被调用,共出现多少次等。在 Source Browse 窗口中提供了完善的管理方法,如过滤器可以 分门别类地列出各种类别的变量名,可以对这些变量按 Class(组)、Type(类型)、Space
(所在空间)、Use(调用次数)排序,点击变量名,可以在窗口的右侧看到该变量名的更 详细的信息。
4、代码作用范围分析
在你写的程序中,有些代码可能永远不会被执行到(这是无效的代码),也有一些代码 必须在满足一定条件后才能被执行到,借助于代码范围分析工具,可以快速地了解代码的执 行情况。
进入调试后,全速运行,然后按停止按钮,停下来后,可以看到在源程序的左列有三种 颜色,灰、淡灰和绿,其中淡灰所指的行并不是可执行代码,如变量或函数定义、注释行等 等,而灰色行是可执行但从未执行过的代码,而绿色则是已执行过的程序行。使用调试工具 条上的 Code Coverage Window 可打开代码作用范围分析的对话框,里面有各个模块代码执 行情况的更详细的分析。如果你发现全速运行后有一些未被执行到的代码,那么就要仔细分 析,这些代码究竟是无效的代码还是因为条件没有满足而没有被执行到。