Kei软件实例教程(四) Keil的辅助工具和部份高级技巧 在前面的几讲中我们介绍了工程的建立方法,常用的调试方法,除此之外,Keli还提供 了一些辅助工具如外围接口、性能分析、变量来源分析、代码作用分析等,帮助我们了解程 的性能、查找程序中的隐藏错误,快速查看程序变量名信息等,这一讲中将对这些功工具作 一介绍,另外还将介绍Keil的部份高级调试技巧 辅助工具 这部份功能并不是直接用来进行程序调试的,但可以帮助我们进行程序的调试、程序性 能的分析,同样是一些很有用的工具 1、外围接口 为了能够比较直观地了解单片机中定时器、中断、 并行端口、串行端口等常用外设的使用情况,Kei提rart1 供了一些外围接口对话框,通过Pepe菜单选择,r:1frpv 该菜单的下拉菜单内容与你建立项目时所选的 CPU'ins: OxFT FENVVMM 有关,如果是选择的89C51这一类“标准”的51机 那么将会有 Interrupt(中断)、 1/O Ports(并行LO口)、 Serial(串行口)、 Timer(定时计数器)这四个外围设 图1外围设备之并行端口 备菜单。打开这些对话框,列出了外围设备的当前使用情况,各标志位的情况等,可以在这 些对话框中直观地观察和更改各外围设备的运行情况 下面我们通过一个简单例子看一看并行端口的外围设备对话框的使用。例4 MOVA,并#OFEH LOOP. MOV PlA RL ALL DELAY;延时100毫秒 JMP LOOP 其中延时100毫秒的子程序请自行编写。 编译、连接进入调试后,点击 Timer/Counter 0 Peripherals-> 1/O-Ports>Port1打开,如图1所示,全速运rmer/ Counter 0 行,可以看到代表各位的勾在不断变化(如果看不到变化, 请点击Ⅴ iew->Periodic Window Updata),这样可以形象地013 Bit Timer/Counte 看出程序执行的结果 Timer 注:如果你看到的变化极快,甚至看不太清楚,那么r:120mn 说明你的计算机性能好,模拟执行的速度快,你可以试着 THO: 0x48 将加长延时程序的时间以放慢速度。模拟运行速度与实际 Con:. M TO Pin M TPO 运行的速度无法相同是软件模拟的一个固有弱点 tatus: Stop 点击 Peripheral>/ O-Ports-> Timer0即出现图2所示 厂TE0厂 GATE M II0# 定时/计数器0的外围接口界面,可以直接选择Mode组中 的下拉列表以确定定时/计数工作方式,0-3四种工作方式, 图2外围设备之定时器
Keil 软件实例教程(四) Keil 的辅助工具和部份高级技巧 在前面的几讲中我们介绍了工程的建立方法,常用的调试方法,除此之外,Keil 还提供 了一些辅助工具如外围接口、性能分析、变量来源分析、代码作用分析等,帮助我们了解程 的性能、查找程序中的隐藏错误,快速查看程序变量名信息等,这一讲中将对这些功工具作 一介绍,另外还将介绍 Keil 的部份高级调试技巧。 一、 辅助工具 这部份功能并不是直接用来进行程序调试的,但可以帮助我们进行程序的调试、程序性 能的分析,同样是一些很有用的工具。 1、外围接口 为了能够比较直观地了解单片机中定时器、中断、 并行端口、串行端口等常用外设的使用情况,Keil 提 供了一些外围接口对话框,通过 Peripherals 菜单选择, 该菜单的下拉菜单内容与你建立项目时所选的 CPU 有关,如果是选择的 89C51 这一类“标准”的 51 机, 那么将会有 Interrupt(中断)、I/O Ports(并行 I/O 口)、 Serial(串行口)、Timer(定时/计数器)这四个外围设 备菜单。打开这些对话框,列出了外围设备的当前使用情况,各标志位的情况等,可以在这 些对话框中直观地观察和更改各外围设备的运行情况。 下面我们通过一个简单例子看一看并行端口的外围设备对话框的使用。例 4: MOV A,#0FEH LOOP: MOV P1,A RL A CALL DELAY ;延时 100 毫秒 JMP LOOP 其中延时 100 毫秒的子程序请自行编写。 编译、连接进入调试后,点击 Peripherals->I/O-Ports->Port 1 打开,如图 1 所示,全速运 行,可以看到代表各位的勾在不断变化(如果看不到变化, 请点击 View->Periodic Window Updata),这样可以形象地 看出程序执行的结果。 注:如果你看到的变化极快,甚至看不太清楚,那么 说明你的计算机性能好,模拟执行的速度快,你可以试着 将加长延时程序的时间以放慢速度。模拟运行速度与实际 运行的速度无法相同是软件模拟的一个固有弱点。 点击 Peripherals->I/O-Ports->Timer0 即出现图 2 所示 定时/计数器 0 的外围接口界面,可以直接选择 Mode 组中 的下拉列表以确定定时/计数工作方式,0-3 四种工作方式, 图 1 外围设备之并行端口 图 2 外围设备之定时器
Kei软件实例教程(四) 设定定时初值等,点击选中TRO, status后的stop就变成了run,如果全速运行程序,此时 tho,t0后的值也快速地开始变化(同样要求 Periodic Window Updata处于选中状态),直观地 演示了定时/计数器的工作情况(当然,由于你的程序未对此写任何代码,所以程序不会对 此定时计数器的工作进行处理)。 2、性能分析 Kei提供了一个性能分析工具,利用该工具,我们可以了解程序中哪些部份的执行时间 最长,调用次数最多,从而了解影响整个程序中执行速度的瓶颈。下面通过一个实例来看 看这个工具如何使用,例5: #include"reg51h sbit p10=P1^0,∥定义P void m Delay (unsigned char Delay Time i unsigned int i; i unsigned int j=0 for(;){ mDelay(10,∥延时10 for( Delay Time>0; Delay Time--) 毫秒 for(=0j0; Delay Time--) mDelay 1(10); j for(j=0j Performance Analyzer Window,打开性能分 析对话框,进入该对话框后,只有一项 unspecified,点鼠标右键,在快捷菜单中选择 Setup PA 即打开性能分析设置对话框,对于C语言程序,该对话框右侧的“ Function Symbol”下的 列表框给出函数符号,双击某一符号,该符号即出现在 Define Performance Analyzer下的编 缉框中,每输入一个符号名字,点击 Define按钮,即将该函数加入其上的分析列表框。对 于汇编语言源程序, Function Symbol下的列表框中不会出现子程序名,可以直接在编缉框 中输入子程序名,点击 Close关闭窗口,回到性能分析窗口,此时窗口共有4个选项。全速 执行程序,可以看到 mDelay和 mDelay I后出现一个蓝色指示条,配合上面的标尺可以直观 地看出每个函数占整个执行时间的比例,点击相应的函数名,可以在该窗口的状态栏看到更 详细的数据,其中各项的含义如下 Min:该段程序执行所需的最短时间;Max:该段程序执行所需的最长时间;Avg:该 段程序执行所花平均时间:Totl:该段程序到目前为目总共执行的时间:%:占整个执行时 间的百分比; count:被调用的次数。 本程序中,函数 mDelay和 mDelay I每次被调用都花费同样的时间,看不出Mn、Max 和Avg的意义,实际上,由于条件的变化,某些函数执行的时间不一定是一个固定的值 借助于这些信息,可以对程序有更详细的了解。下面将 mDelayl函数略作修改作一演示 void m Delay l(unsigned char Delay Time) i static unsigned char k unsigned int j =0 for( Delay Time>0; Delay Time--) i forGe<k:j++) }k++,} 程序中定义了一个静态变量K,每次调用该变量加1,而j的循环条件与k的大小有关
Keil 软件实例教程(四) 设定定时初值等,点击选中 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;j0;DelayTime--) { for(j=0;jPerformance 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;j++) {;} } k++; } 程序中定义了一个静态变量 K,每次调用该变量加 1,而 j 的循环条件与 k 的大小有关
Kei软件实例教程(四) 这使每次执行该程序所花的时间不一样。编译、执行该程序,再次观察性能分析窗口,可以 看出Min、Max、Awg的意义。 3、变量来源浏览 该窗口用于观察程序中变量名的有关信息,如该变量名在那一个函数中被定义、在哪里 被调用,共出现多少次等。在 Source browse窗口中提供了完善的管理方法,如过滤器可以 分门别类地列出各种类别的变量名,可以对这些变量按 Class(组)、Type(类型)、 Space (所在空间)、Use(调用次数)排序,点击变量名,可以在窗口的右侧看到该变量名的更 详细的信息 4、代码作用范围分析 在你写的程序中,有些代码可能永远不会被执行到(这是无效的代码),也有一些代码 必须在满足一定条件后才能被执行到,借助于代码范围分析工具,可以快速地了解代码的执 行情况。 进入调试后,全速运行,然后按停止按钮,停下来后,可以看到在源程序的左列有三种 颜色,灰、淡灰和绿,其中淡灰所指的行并不是可执行代码,如变量或函数定义、注释行等 等,而灰色行是可执行但从未执行过的代码,而绿色则是已执行过的程序行。使用调试工具 条上的 Code Coverage window可打开代码作用范围分析的对话框,里面有各个模块代码执 行情况的更详细的分析。如果你发现全速运行后有一些未被执行到的代码,那么就要仔细分 析,这些代码究竞是无效的代码还是因为条件没有满足而没有被执行到 二、部份高级调试技巧 Kei内置了一套调试语言,很多高级调试技巧与此有关,但是全面学习这套语言并不现 实,这不是这么几期连载可以胜任的,这里仅介绍部份较为实用的功能,如要获得更详细的 信息,请参考Kei自带的帮助文件GS51PDF。 1、串行窗口与实际硬件相连 Kei的串行窗口除可以模拟串行口的输入和输出功能外还可以与PC机上实际的串口相 连,接受串口输入的内容,并将输出送到串口。这需要在Kel中进行设置。方法是首先在 输出窗口的 Command页用MODE命令设置串口的工作方式,然后用 ASSIGN命令将串行 窗口与实际的串口相关联,下面我们通过一个实例来说明如何操作。例6: ORG 0000H SETB ES JMP START JMP$;主程序到此结束 ORG3+4*8;串行中断入口 SER INT. JMP SER INT JBC RINEXT;如果串口接收到字 START 符,转 MOV SP#5FH:堆栈初始化 JMP SEND 否则转发送处理 CALL SER INIT串行口初始化A NEXT SETB EA MOV ASBUF;从SBUF中取字符
Keil 软件实例教程(四) 这使每次执行该程序所花的时间不一样。编译、执行该程序,再次观察性能分析窗口,可以 看出 Min、Max、Avg 的意义。 3、变量来源浏览 该窗口用于观察程序中变量名的有关信息,如该变量名在那一个函数中被定义、在哪里 被调用,共出现多少次等。在 Source Browse 窗口中提供了完善的管理方法,如过滤器可以 分门别类地列出各种类别的变量名,可以对这些变量按 Class(组)、Type(类型)、Space (所在空间)、Use(调用次数)排序,点击变量名,可以在窗口的右侧看到该变量名的更 详细的信息。 4、代码作用范围分析 在你写的程序中,有些代码可能永远不会被执行到(这是无效的代码),也有一些代码 必须在满足一定条件后才能被执行到,借助于代码范围分析工具,可以快速地了解代码的执 行情况。 进入调试后,全速运行,然后按停止按钮,停下来后,可以看到在源程序的左列有三种 颜色,灰、淡灰和绿,其中淡灰所指的行并不是可执行代码,如变量或函数定义、注释行等 等,而灰色行是可执行但从未执行过的代码,而绿色则是已执行过的程序行。使用调试工具 条上的 Code Coverage Window 可打开代码作用范围分析的对话框,里面有各个模块代码执 行情况的更详细的分析。如果你发现全速运行后有一些未被执行到的代码,那么就要仔细分 析,这些代码究竟是无效的代码还是因为条件没有满足而没有被执行到。 二、部份高级调试技巧 Keil 内置了一套调试语言,很多高级调试技巧与此有关,但是全面学习这套语言并不现 实,这不是这么几期连载可以胜任的,这里仅介绍部份较为实用的功能,如要获得更详细的 信息,请参考 Keil 自带的帮助文件 GS51.PDF。 1、串行窗口与实际硬件相连 Keil 的串行窗口除可以模拟串行口的输入和输出功能外还可以与 PC 机上实际的串口相 连,接受串口输入的内容,并将输出送到串口。这需要在 Keil 中进行设置。方法是首先在 输出窗口的 Command 页用 MODE 命令设置串口的工作方式,然后用 ASSIGN 命令将串行 窗口与实际的串口相关联,下面我们通过一个实例来说明如何操作。例 6: ORG 0000H JMP START ORG 3+4*8 ;串行中断入口 JMP SER_INT START: MOV SP,#5FH ;堆栈初始化 CALL SER_INIT ;串行口初始化 A SETB EA ; SETB ES ; JMP $ ;主程序到此结束 SER_INT: JBC RI,NEXT ;如果串口接收到字 符,转 JMP SEND ;否则转发送处理 NEXT: MOV A,SBUF ;从 SBUF 中取字符
Kei软件实例教程(四) MOV SBUFA回送到发送SBUF中 ORL TMOD#20H JMP OVER ORL PCON #80H SEND MOV TH1,#OFDH;设定波特率 SETB TR1;定时器1开始运行 OVER. SETB REN;允许接收 SETB SM2 RET SER INIT. ;中断初始化 END MOV SCON.#50H 这个程序使用了中断方式编写串行口输入输出程序,它的功能是将接串行口收到的字 符回送,即再通过串行口发送出去 正确输入源文件、建立工程、编译连接没有错后,可进行调试,使用Kei自带的串行 窗口测试功能是否正确,如果正确,可以进行下一步的连机试验 为简单实用,我们不借助于其它的硬件,而是让PC机上的两个串口互换数据,即COM1 发送COM2接收,而COM2发送则由COM1接收,为此,需要做一根连接线将这两个串口 连起来,做法很简单,找两个可以插入PC机串口的DN9插座(母),然后用一根3芯线将 它们连起来,连线的方法是 3-—2 接好线把两个插头分别插入PC机上的串口1与串口2。找一个PC机上的串口终端调 试软件,如串口精灵之类,运行该软件,设置好串口参数,其中串口选择2,串口参数设置 19200,n,8,1其含义是波特率为19200,无奇偶校验,8位数据,1位停止位 在Kei调试窗口的 command页中输入: > mode coml19200.0.8.1 >assign coml sout 注意两行最前面的“>”是提示符,不要输入,第二行中的“”即“小于”和 大于”符号,中间的是字母“s”和“ Input”的前两个字母,最后是字母“s”和“ output 的前三个字母。 第一行命令定义串口1的波特率为19200,无奇偶校验,8位数据,1位停止位。第二 行是将串口1(com1)分配给串行窗口。 全速运行程序,然后切换串口精灵,开始发送,会看到发送后的数据会立即回显到窗口 中,说明已接收到了发送过来的数据。切换到 u Vison,查看串行窗口1,会看到这里的确接 收到了串口精灵送来的内容 2、从端口送入信号 程序调试中如果需要有信号输入,比如数据采集类程序,需要从外界获得数据,由于 Kei的调试完全是一个软件调试工具,没有硬件与之相连,所以不可能直接获得数据,为此 必须采用一些替代的方法,例如,某电路用P1口作为数据采集口,那么可以使用的一种方 法是利用外围接口,打开PORT1,用鼠标在点击相应端口位,使其变为高电平或低电平 就能输入数据。显然,这种方法对于要输获得数据而不是作位处理来说太麻烦了,另一种方 法是直接在 command页输入 portI=数值,以下是一个小小的验证程序。例7: LOOP. MOV APl
Keil 软件实例教程(四) MOV SBUF,A ;回送到发送SBUF中 JMP OVER SEND: clr ti OVER: reti SER_INIT: ;中断初始化 MOV SCON,#50H ORL TMOD,#20H ORL PCON,#80H MOV TH1,#0FDH ;设定波特率 SETB TR1 ;定时器 1 开始运行 SETB REN ;允许接收 SETB SM2 RET END 这个程序使用了中断方式编写串行口输入/输出程序,它的功能是将接串行口收到的字 符回送,即再通过串行口发送出去。 正确输入源文件、建立工程、编译连接没有错后,可进行调试,使用 Keil 自带的串行 窗口测试功能是否正确,如果正确,可以进行下一步的连机试验。 为简单实用,我们不借助于其它的硬件,而是让 PC 机上的两个串口互换数据,即 COM1 发送 COM2 接收,而 COM2 发送则由 COM1 接收,为此,需要做一根连接线将这两个串口 连起来,做法很简单,找两个可以插入 PC 机串口的 DIN9 插座(母),然后用一根 3 芯线将 它们连起来,连线的方法是: 2——3 3——2 5——5 接好线把两个插头分别插入 PC 机上的串口 1 与串口 2。找一个 PC 机上的串口终端调 试软件,如串口精灵之类,运行该软件,设置好串口参数,其中串口选择 2,串口参数设置 为: 19200,n,8,1 其含义是波特率为 19200,无奇偶校验,8 位数据,1 位停止位。 在 Keil 调试窗口的 command 页中输入: >mode com1 19200,0,8,1 >assign com1 sout 注意两行最前面的“>”是提示符,不要输入,第二行中的“”即“小于”和 “大于”符号,中间的是字母“s”和“input”的前两个字母,最后是字母“s”和“output” 的前三个字母。 第一行命令定义串口 1 的波特率为 19200,无奇偶校验,8 位数据,1 位停止位。第二 行是将串口 1(com1)分配给串行窗口。 全速运行程序,然后切换串口精灵,开始发送,会看到发送后的数据会立即回显到窗口 中,说明已接收到了发送过来的数据。切换到 uVison,查看串行窗口 1,会看到这里的确接 收到了串口精灵送来的内容。 2、从端口送入信号 程序调试中如果需要有信号输入,比如数据采集类程序,需要从外界获得数据,由于 Keil 的调试完全是一个软件调试工具,没有硬件与之相连,所以不可能直接获得数据,为此 必须采用一些替代的方法,例如,某电路用 P1 口作为数据采集口,那么可以使用的一种方 法是利用外围接口,打开 PORT 1,用鼠标在点击相应端口位,使其变为高电平或低电平, 就能输入数据。显然,这种方法对于要输获得数据而不是作位处理来说太麻烦了,另一种方 法是直接在 command 页输入 port1=数值,以下是一个小小的验证程序。例 7: LOOP: MOV A,P1
Kei软件实例教程(四) NEXT MOV RO.#55H JMP LOOP NEXT: MOV RO#OAAH JMP LOOP END 该程序从P1口获得数据,如果P1口的值是0,那么就让R0的值为0AAH,否则让R0 的值为55H。输入源程序并建立工程,进入调试后,在观察窗口加入RO,然后全速运行程 序,注意确保View-> Periodic Window Updata处于选中状态,然后在 Command后输入 PORT1=0回车后可以发现观察窗口中的R0的值变成了0AAH,然后再输入PORT1=1或其 它非零值,则R0的值会变为55H 同样的道理,可以用poro、port2、port3分别向端口0、2、3输入信号 3、直接更改内存值 在程序运行中,另一种输入数据的方法是直接更改相应的内存单元的值,例如,某数据 采集程序,使用30H和3H作为存储单元,采入的数据由这两个单元保存,那么我们更改 了30H和31H单元的值就相当于这个数据采集程序采集到了数据,这可以在内存窗口中直 接修改(参考上一讲),也可以通过命令进行修改,命令的形式是: WBYTE(地址数据), 其中地址是指待写入内存单元的地址,而数据则是待写入该地址的数据。例如 WBYTE(0x30,11)会将值11写入内存地址十六进制30H单元中
Keil 软件实例教程(四) JZ NEXT MOV R0,#55H JMP LOOP NEXT: MOV R0,#0AAH JMP LOOP END 该程序从 P1 口获得数据,如果 P1 口的值是 0,那么就让 R0 的值为 0AAH,否则让 R0 的值为 55H。输入源程序并建立工程,进入调试后,在观察窗口加入 R0,然后全速运行程 序,注意确保 View->Periodic Window Updata 处于选中状态,然后在 Command 后输入 PORT1=0 回车后可以发现观察窗口中的 R0 的值变成了 0AAH,然后再输入 PORT1=1 或其 它非零值,则 R0 的值会变为 55H。 同样的道理,可以用 port0、port2、port3 分别向端口 0、2、3 输入信号。 3、直接更改内存值 在程序运行中,另一种输入数据的方法是直接更改相应的内存单元的值,例如,某数据 采集程序,使用 30H 和 31H 作为存储单元,采入的数据由这两个单元保存,那么我们更改 了 30H 和 31H 单元的值就相当于这个数据采集程序采集到了数据,这可以在内存窗口中直 接修改(参考上一讲),也可以通过命令进行修改,命令的形式是: _WBYTE (地址,数据), 其中地址是指待写入内存单元的地址,而数据则是待写入该地址的数据。例如 _WBYTE(0x30,11)会将值 11 写入内存地址十六进制 30H 单元中