当前位置:首页 > 电源 > 数字电源
[导读]Keil 的调试命令、在线汇编与断点设置 上一讲中我们学习了如何建立工程、汇编、连接工程,并获得目标代码,但是做到这一 步仅仅代表你的源程序没有语法错误,至于源程序中存在着的其它错误,必须通过调试才能 发现

Keil 的调试命令、在线汇编与断点设置
上一讲中我们学习了如何建立工程、汇编、连接工程,并获得目标代码,但是做到这一
步仅仅代表你的源程序没有语法错误,至于源程序中存在着的其它错误,必须通过调试才能
发现并解决,事实上,除了极简单的程序以外,绝大部份的程序都要通过反复调试才能得到
正确的结果,因此,调试是软件开发中重要的一个环节,这一讲将介绍常用的调试命令、利
用在线汇编、各种设置断点进行程序调试的方法,并通过实例介绍这些方法的使用。
一、常用调试命令
在对工程成功地进行汇编、连接以后,按Ctrl+F5 或者使用菜单Debug->Start/Stop Debug
Session 即可进入调试状态,Keil 内建了一个仿真CPU 用来模拟执行程序,该仿真CPU 功
能强大,可以在没有硬件和仿真机的情况下进行程序的调试,下面将要学的就是该模拟调试
功能。不过在学习之前必须明确,模拟毕竟只是模拟,与真实的硬件执行程序肯定还是有区
别的,其中最明显的就是时序,软件模拟是不可能和真实的硬件具有相同的时序的,具体的
表现就是程序执行的速度和各人使用的计算机有关,计算机性能越好,运行速度越快。
进入调试状态后,界面与编缉状态相比有明显的变化,Debug 菜单项中原来不能用的命
令现在已可以使用了,工具栏会多出一个用于运行和调试的工具条,如图1 所示,Debug 菜
单上的大部份命令可以在此找到对应的快捷按钮,从左到右依次是复位、运行、暂停、单步、
过程单步、执行完当前子程序、运行到当前行、下一状态、打开跟踪、观察跟踪、反汇编窗
口、观察窗口、代码作用范围分析、1#串行窗口、内存窗口、性能分析、工具按钮等命令。
学习程序调试,必须明确两个重要的概念,即单步执行与全速运行。全速执行是指一行程序执行完以后紧接着执行下一行程序,中间不停止,这样程序执行的速度很
快,并可以看到该段程序执行的总体效果,即最终结果正确
还是错误,但如果程序有错,则难以确认错误出现在哪些程
序行。单步执行是每次执行一行程序,执行完该行程序以后
即停止,等待命令执行下一行程序,此时可以观察该行程序
执行完以后得到的结果,是否与我们写该行程序所想要得到
的结果相同,借此可以找到程序中问题所在。程序调试中,
这两种运行方式都要用到。
使用菜单STEP 或相应的命令按钮或使用快捷键F11 可
以单步执行程序,使用菜单STEP OVER 或功能键F10 可以
以过程单步形式执行命令,所谓过程单步,是指将汇编语言
中的子程序或高级语言中的函数作为一个语句来全速执行。
按下F11 键,可以看到源程序窗口的左边出现了一个黄色调试箭头,指向源程序的第一
行,如图2 所示。每按一次F11,即执行该箭头所指程序行,然后箭头指向下一行,当箭头
指向LCALL DELAY 行时,再次按下F11,会发现,箭头指向了延时子程序DELAY 的第
一行。不断按F11 键,即可逐步执行延时子程序。
通过单步执行程序,可以找出一些问题的所在,但是仅依靠单步执行来查错有时是困难
的,或虽能查出错误但效率很低,为此必须辅之以其它的方法,如本例中的延时程序是通过

图1 调试工具条

图2 调试窗口
将D2: DJNZ R6,D2 这一行程序执行六万多次来达到延时的目的,如果用按F11 六万多
次的方法来执行完该程序行,显然不合适,为此,可以采取以下一些方法,第一,用鼠标在
子程序的最后一行( ret)点一下,把光标定位于该行,然后用菜单Debug->Run to Cursor line
(执行到光标所在行),即可全速执行完黄色箭头与光标之间的程序行。第二,在进入该子
程序后,使用菜单Debug->Step Out of Current Function(单步执行到该函数外),使用该命令
后,即全速执行完调试光标所在的子程序或子函数并指向主程序中的下一行程序(这里是
JMP LOOP 行)。第三种方法,在开始调试的,按F10 而非F11,程序也将单步执行,不同
的是,执行到lcall delay 行时,按下F10 键,调试光标不进入子程序的内部,而是全速
执行完该子程序,然后直接指向下一行“JMP LOOP”。灵活应用这几种方法,可以大大提
高查错的效率。
二、在线汇编
在进入Keil 的调试环境以后,如果发现程序有错,可以直接对源程序进行修改,但是
要使修改后的代码起作用,必须先退出调试环境,重新进行编译、连接后再次进入调试,如
果只是需要对某些程序行进行测试,或仅需对源程序进行临时的修改,这样的过程未免有些
麻烦,为此Keil 软件提供了在线汇编的能力,将光标定位于需要修改的程序行上,用菜单
Debug->Inline Assambly… 即可出现如
图3 的对话框,在Enter New 后面的
编缉框内直接输入需更改的程序语
句,输入完后键入回车将自动指向下
一条语句,可以继续修改,如果不再
需要修改,可以点击右上角的关闭按
钮关闭窗口。
三、断点设置
程序调试时,一些程序行必须满足一定的条件才能被执行到(如程序中某变量达到一定
的值、按键被按下、串口接收到数据、有中断产生等),这些条件往往是异步发生或难以预
先设定的,这类问题使用单步执行的方法是很难调试的,这时就要使用到程序调试中的另一
种非常重要的方法——断点设置。断点设置的方法有多种,常用的是在某一程序行设置断点,
设置好断点后可以全速运行程序,一旦执行到该程序行即停止,可在此观察有关变量值,以
确定问题所在。在程序行设置/移除断点的方法是将光标定位于需要设置断点的程序行,使
用菜单Debug->Insert/Remove BreakPoint 设置或移除断点(也可以用鼠标在该行双击实现同
样的功能);Debug->Enable/Disable Breakpoint 是开启或暂停光标所在行的断点功能;
Debug->Disable All Breakpoint 暂停所有断点;Debug->Kill All BreakPoint 清除所有的断点设
置。这些功能也可以用工具条上的快捷按钮进行设置。
除了在某程序行设置断点这一基本方法以外,Keil 软件还提供了多种设置断点的方法,
按Debug->Breakpoints… 即出现一个对话框,该对话框用于对断点进行详细的设置,如图4
所示。
图4 中Expression 后的编缉框内用于输入表达式,该表达式用于确定程序停止运行的条
件,这里表达式的定义功能非常强大,涉及到Keil 内置的一套调试语法,这里不作详细说[!--empirenews.page--]
明,仅举若干实例,希望读者可以举一反三。
1) 在Experssion 中键入a==0xf7,再点击Define 即定义了一个断点, 注意,a 后有两
个等号,意即相等。该表达式的含义是:如果a 的值到达0xf7 则停止程序运行。除

图3 在线汇编窗口

使用相等符号之外,还可以使用>,>=,<,<=,!=(不等于),&(两值按位与),&&(两值相与)等运算符号。
2) 在Experssion 后中键入Delay 再点击Define,其含义是如果执行标号为Delay 的行则中断。
3) 在Experssion 后中键入Delay,按Count后的微调按钮,将值调到3,其意义是当第三次执行到Delay 时才停止程序运行。
4) 在Experssion 后键入Delay , 在
Command 后键入printf(“SubRoutine
‘Delay’ has been Calledn”)主程序每次
调用Delay 程序时并不停止运行,但会
在输出窗口Command 页输出一行字
符,即SubRoutine ‘Delay’ has been
Called。其中“n”的用途是回车换行,
使窗口输出的字符整齐。
5) 设置断点前先在输出窗口的Command
页中键入DEFINE int I,然后在断点设
置时同4),但是Command 后键入printf(“SubRoutine ‘Delay’ has been Called %d
timesn”,++I),则主程序每次调用Delay 时将会在Command 窗口输出该字符及被调
用的次数,如SubRoutine ‘Delay’ has been Called 10 times。
对于使用C 源程序语言的调试,表达式中可以直接使用变量名,但必须要注意,设置
时只能使用全局变量名和调试箭头所指模块中的局部变量名。
四、实例调试
为进行程序的调试,我们首先给源程序制造一个错误,将延时子程序的第三行“DJNZ
R6,$”后的$改为D1,然后重新编译,由于程序中并无语法错误,所以编译时不会有任何出
错提示,但由于转移目的地出错,所以子程序将陷入无限循环中。
进入调试状态后,按F10 以过程单步的形式执行程序,当执行到LCALL DELAY 行时,
程序不能继续往下执行,同时发现调试工具条上的Halt 按钮变成了红色,说明程序在此不
断地执行着,而我们预期这一行程序执行完后将停止,这个结果与预期不同,可以看出所调
用的子程序出了差错。为查明出错原因,按Halt 按钮使程序停止执行,然后按RST 按钮使
程序复位,再次按下F10 单步执行,但在执行到LCALL DELAY 行时,改按F11 键跟踪到
子程序内部(如果按下F11 键没有反应,请在源程序窗口中用鼠标点一下),单步执行程序,
可以发现在执行到“DJNZ R6,D1”行时,程序不断地从这一行转移到上一行,同时观察
左侧的寄存器的值,会发现R6 的值始终在FFH 和FEH 之间变化,不会减小,而我们的预
期是R6 的值不断减小,减到0 后往下执行,因此这个结果与预期不符,通过这样的观察,
不难发现问题是因为标号写错而产生的,发现问题即可以修改,为了验证即将进行的修改是
否正确,可以先使用在线汇编功能测试一下。把光标定位于程序行“DJNZ R6,D1”,打开
在线汇编的对话框,将程序改为“DJNZ R7,0EH”,即转回本条指令所在行继续执行,其中
0EH 是本条指令在程序存储器中的位置,这个值可以通过在线汇编窗口看到,如图3 所示。
然后关闭窗口,再进行调试,发现程序能够正确地执行了,这说明修改是正确的。注意,这
时候的源程序并没有修改,此时应该退出调试程序,将源程序更改过来,并重新编译连接,
以获得正确的目标代码。

图4 断点设置对话框

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

近日某厂员工爆料,公司程序员因线上流量异常BUG事故,被公司进行处罚。处罚的结果是被要求将去年发的 4 万多年终奖归还给公司,否则就收取滞纳金并辞退,逾期将以每天万分之 5 的利息收取滞纳金。

关键字: 程序员 BUG 事故 年终奖 工程师

业内消息,近日有媒体报道一名女子将年度总结发到朋友圈并将状态设为私密,但随后她发现其好友可以看到她发了朋友圈,尽管无法看到具体内容。随即,微博话题#微信私密朋友圈被吐槽有BUG# 很快冲上热搜第一,许多网友也纷纷表示遇见...

关键字: 微信 BUG

从汇编角度来说,如果“test %al, %al”能改成“test %0x1, %al”就没有匪夷所思的问题了,如此一来应该会降低CPU的效率,毕竟执行指令还需要一个立即数,我没搞过编译器也没设计过CPU,纯属瞎猜,能搞...

关键字: 汇编 CPU 编译器

焕新玩乐与购物体验,为当地儿童、家庭和乐高爱好者带来更多创意玩乐机会 云南昆明2022年7月9日 /美通社/ -- 今日,全新乐高®授权专卖店于昆明恒隆广场正式开业,为昆明当地的儿童、家庭和乐高爱好者提供了近距...

关键字: 乐高 模型 AN BUG

“验证很多人都清楚,验证技术一直在发展,个人技术成长不进则退。于是采用最新的验证方法和趋势是很多验证牛人趋之若鹜的事情。一旦验证大佬尝试了某个事情,可能很快就会在团队传播起来,这就是偶像效应。这时候,这项技术仿佛就是经过...

关键字: BUG EDA VERILOG 编程语言

1、这是继承的代码,不需要验证。你100%确定这个代码经过完备验证么?你确定没有人后来修改过么?你确定和这个代码相关的一切都没有变化么? 2、我可以在5分钟内想出一个补丁。只要你确定你的验证环境不会成为一个屎山。问问你自...

关键字: 代码 BUG TESTBENCH 模型

学习曲线的定义为"在一定时间内获得的技能或知识的速率"。对于验证工程师,哪一种工作环境能提供最大的学习机会?1)IP验证2)SOC验证3)验证IP开发在工作时所具备的技能应该符合整个行业的要求,并且应该可以跨公司携带。假...

关键字: 工程师 VIP SOC验证 BUG

相信很多人都听说过这样的一个论断:“功能验证需要占用芯片设计周期的70%”。编写测试计划/可重用验证环境/设计断言、定位RTLbug、实现代码覆盖和功能覆盖目标以及门级仿真和调试,是功能验证工程师在tapeout前经历的...

关键字: 工程师 BUG 代码 编写

01这是经过硅验证的IP,无需进行完备的测试和覆盖你确定所有功能cross场景是否都经过硅验证么。你确定应用场景和外部约束都没有变化么?02tapeout出去的是RTL流片,无需检视验证环境,只要检视验证计划就足够了70...

关键字: 代码 BUG 工程师 测试平台

Windows11先后两个正式版本,都让AMD锐龙处理器的性能蒙上劣化阴影,好在解决方案终于下发。最新补丁KB5006746(Build22000.282)已经抵达Beta和发布预览通道,解决了影响AMD锐龙平台性能的处...

关键字: AMD BUG 锐龙
关闭
关闭