BUAA OS实验调试指南:从看懂到看开
一般的调试流程其实很简单:发现问题,稳定复现,确定临界条件,定位问题,修复问题,核查结果。迭代这个过程,形成一个闭环
老实说,OS的实验代码,开箱体验极差,程序跳来跳去,进了Lab4后还要考虑内核态切换,很难靠肉眼完成上述闭环。debug愉悦指数为负。
所以在几周的探索后,我大概总结整理了一些调试经验,主要是如何在当前体系下利用或构建调试工具,改善调试体验。
我们的口号是:没有蛀牙。
抛砖引玉。
从0开始:现在我们有什么
现在我们手里有:
- 一坨新鲜的系统代码。
- 上学期积累的老练的MIPS 32汇编开发与理论基础,虽然这个系统用的是MIPS 2
- 对高级语言特别是面向对象语言的美好眷恋
- 全宇宙最牛逼的编程语言:C,以及全宇宙最牛逼的C tool chain:GCC
除此之外我们还有:
- 凑活能用的
printf - 助教为我们封装好的
panic和assert,她们是海滩上最抢眼的宝贝儿 - 用户态的一些函数,但说真的它们不太好用
- 一颗勇敢的心
Level 0:静态调试
也就是所谓的“瞪眼法”。能瞪出来的Bug请直接瞪出来。
瞪眼法的升级版是小黄鸭调试法,进一步升级版是面向室友调试,终极形态是面向男朋友/女朋友/老妈调试
别漏了这一步,有时候瞪是最快的,但这也要求你对OS实验到底要干什么有完全清晰的把控。
记得调VIM配色,记得买珍视明。
Level 1: 断言与防御性编程
assert这个东西可能我们已经在各种check里面见过了。它的实现在include/mm.h中(很奇怪的位置),在我们的代码中是以宏的形式实现的:
#define assert(x) do {if(!(x)) panic("...%s", #x); } while(0)
作者十分老练,这里的trick是用拼接运算将参数x作为一个字符串传入panic,这样输出时我们就可以看到断言内的具体内容。
现在assert的功能就是这样的:如果传入的语句x布尔值为假,就陷入panic(顺带一提,panic其实就是在输出信息后陷入死循环)。熟悉其他语言的断言机制,比如python,的同学应该会感到十分亲切
这个东西有什么用呢?我们可以在代码中合适的位置加入断言,来显示地检查某段运行逻辑是否如我们的预期。
举个例子,page_insert后,插入的页面所对应的物理地址应该等于虚拟地址va所映射的物理地址(这正是这个函数的功能),因此在函数的末尾我们可以添加一句断言:
assert(va2pa(pgdir, va) == page2pa(pp));
当程序运行到这里时,如果没有出现错误,那么这句assert不会产生任何作用,如果出现了各种情况导致这两个物理地址不一致,那么程序就会自陷终止并输出信息。当我们调试时,如果看到了它自陷,我们就能知道这里的代码一定存在逻辑漏洞(或者,大灾难),在它引起更隐晦的错误前将其修复。
当然assert也不是完全没有副作用:它依旧会占用指令运行周期数,这个问题我们之后再讨论。
Level 2:把printf撒的满屋子都是
然后因为忘了删调试语句,喜提评测0/100
Level 3:更好用的输出调试信息:封装一个debug()
printf的问题在于,开关并不方便。很难做到只做少量的修改将所有调试信息全部关闭。因此这里开始我们需要自己造轮子
稍微优雅一些的做法是,声明一个调试宏或一个全局变量,来控制全局的调试开关
宏的写法类似于:
# ifndef __DEBUG__
# define __DEBUG__
# endif
...
# ifdef __DEBUG__
printf("SOME DEBUG INFO\n");
# endif
全局变量的写法则是
extern int is_debug_mode = 1;
...
if (is_debug_mode) {
printf("SOME DEBUG INFO\n");
}
宏的一个优势在于,如果我们关掉了调试,整个调试代码块不会被编译,执行的语句更少
直接用变量则当然更可读,更美观,并且可以更轻松地实现调试信息分级
无论哪一种,每次输出的时候敲三行代码太复杂了,因此终极形式是包装一个debug函数
void _debug(char * file, int line, char *fmt, ...){
if (is_debug_mode) printf("SAY SOMETHING I'M GIVING UPON YOU.\n");
}
# define debug(...) _debug(__FILE__, __LINE__, __VA__ARGS__)
这样我们就能像使用printf一样使用debug(支持可变参数),同时还能输出更多调试信息,比如这一句在哪个文件的哪一行
然后在init中控制debug的全局开闭就好
Level 3.5 汇编调试:相信聪明的MIPS一定帮我们搞定了
上面的各种方法都是针对C语言程序的,汇编则不太好办。大体上有两个思路:
- 为某一部分特定信息的输出单独封装一个函数,并在汇编中JAL跳转链接调用,比如上机时我们写过的
output_ov_info(我怎么可能记得它叫什么,反正差不多就这个东西)。这个东西使用场景太杂乱,不展开了 - 加断点,断点是好文明。
MIPS为我们提供了BREAK和一系列Trap指令,这里先只讨论Break的使用,Trap是类似的。
BREAK的作用是,抛出一个bp断点异常并使CPU切入内核态处理异常。bp的cause编码是9,为了让我们的小系统可以处理这个异常我们需要仿照课上的ov为其添加一个handler
- 修改
lib/trap.c,声明外链接extern handle_bp();,然后将其绑定在9号异常上 - 在
lib/genex.S中添加handle_bp的处理程序。推荐使用BUILD_HANDLER bp break_handler cli将异常处理移交给一个C语言函数break_handler(struct Trapframe *)完成
然后我们在异常处理中输出各种需要的信息,末尾死循环即可(也就是,panic)。最重要的信息或许是pc。
如果想要更好的体验,添加一个读入字符syscall(参考lab1某次课上),让我们可以向系统输入字符,然后把break的死循环换成等待读入。这么做的时候记得处理中断屏蔽
回过头看C程序的调试,用类似的技巧也可以实现断点调试。如果需要可以再封装一个bp()配合debug()使用。
再回过头看Trap(TEQ、TNE等一系列条件内陷语句),我们可以用类似的方法为其添加异常处理程序,这些指令抛出的异常都是TRAP(13),因此我们只需要把处理程序挂在13号异常上就可以令其运作。
Trap很像MIPS版的assert,当满足某个条件时让内核自陷。因此在汇编代码中善用这些指令也能起到尽早发现并规避错误的作用。
或许我们可以进一步挖掘MIPS的片上调试,但目前我还没有遇到这种粒度的需求。
Level 4
更多奇技淫巧,我自己还在探索与试用,如果体验不错的话之后再更新。
stay tuned
注意事项
- 自己实现的函数,声明尽量放在单独的头文件中,定义尽量放在单独的源文件中,所有东西尽可能封装,做到可插拔。万一课上把自己实现的某个关键文件替换掉导致CE(目前我还没遇到),沉着冷静处理,比如更换函数声明的位置
- 少造轮子,多看代码
- 舒服的才是好的,按自己的使用习惯改造自己的调试工具与调试流程
- 不要尝试学我的代码风格,我已经没有救了,你还有
BUAA OS实验调试指南:从看懂到看开的更多相关文章
- 4张图看懂delphi 10生成ipa和在iPhone虚拟器上调试(教程)
4张图看懂delphi 10生成ipa和在iPhone虚拟器上调试(教程) (2016-02-01 03:21:06) 转载▼ 标签: delphi ios delphi10 教程 编程 分类: 编程 ...
- 新手也能看懂的 SpringBoot 异步编程指南
本文已经收录自 springboot-guide : https://github.com/Snailclimb/springboot-guide (Spring Boot 核心知识点整理. 基于 S ...
- linux内核调试指南
linux内核调试指南 一些前言 作者前言 知识从哪里来 为什么撰写本文档 为什么需要汇编级调试 ***第一部分:基础知识*** 总纲:内核世界的陷阱 源码阅读的陷阱 代码调试的陷阱 原理理解的陷阱 ...
- Linux Kernel - Debug Guide (Linux内核调试指南 )
http://blog.csdn.net/blizmax6/article/details/6747601 linux内核调试指南 一些前言 作者前言 知识从哪里来 为什么撰写本文档 为什么需要汇编级 ...
- 小刻也能看懂的Unraid系统使用手册:基础篇
小刻也能看懂的Unraid系统使用手册 基础篇 Unraid系统简介 Unraid 的本体其实是 Linux,它主要安装在 NAS 和 All in One 服务器上,经常可以在 Linus 的视频里 ...
- PS2鼠标+LCD12864实验(调试未成功)
此试验我一人调试许久都未成功,但发送ff时,读出来的数据确是对的,一开始让我窃喜,但发送f4时,读出来的数据确是错的,哎让苦恼啊,能力有限,只能先暂时就这样吧,那位什么还要贴出来呢,有两个原因: 1. ...
- 看懂SqlServer查询计划【转】
原文链接:http://www.cnblogs.com/fish-li/archive/2011/06/06/2073626.html 开始 SQL Server 查找记录的方法 SQL Server ...
- 【转载】看懂SqlServer查询计划
看懂SqlServer查询计划 阅读目录 开始 SQL Server 查找记录的方法 SQL Server Join 方式 更具体执行过程 索引统计信息:查询计划的选择依据 优化视图查询 推荐阅读-M ...
- 看懂SqlServer查询计划
看懂SqlServer查询计划 阅读目录 开始 SQL Server 查找记录的方法 SQL Server Join 方式 更具体执行过程 索引统计信息:查询计划的选择依据 优化视图查询 推荐阅读-M ...
随机推荐
- visualvm工具远程对linux服务器上的JVM虚拟机进行监控与调优
文/朱季谦 最近在做了一些JVM监控与调优的事情,算是第一次实践,还比较陌生,故而先把这一次经验简单记下笔记,这样,对后面学习调优方面时,不至于又想不起来了.本文档主要总结在window本地环境远程对 ...
- (4)MySQL进阶篇SQL优化(常用SQL的优化)
1.概述 前面我们介绍了MySQL中怎么样通过索引来优化查询.日常开发中,除了使用查询外,我们还会使用一些其他的常用SQL,比如 INSERT.GROUP BY等.对于这些SQL语句,我们该怎么样进行 ...
- pytest进阶之fixture函数
fixture函数存在意义 与python自带的unitest测试框架中的setup.teardown类似,pytest提供了fixture函数用以在测试执行前和执行后进行必要的准备和清理工作.但是相 ...
- Java例题_30 在已经排好序的数组中插入值
1 /*30 [程序 30 插入数字] 2 题目:有一个已经排好序的数组.现输入一个数,要求按原来的规律将它插入数组中. 3 程序分析:首先判断此数是否大于最后一个数,然后再考虑插入中间的数的情况,插 ...
- 关于生产环境改用G1垃圾收集器的思考
背景 由于我们的业务量非常大,响应延迟要求高.目前沿用的老的ParNew+CMS已经不能支撑业务的需求.平均一台机器在1个月内有1次秒级别的stop the world.对系统来说是个巨大的隐患.所以 ...
- 关于在forEach中使用await的问题
先说需求,根据数组中的ID值,对每个ID发送请求,获取数据进行操作. 首先肯定考虑用forEach 或者 map对数组进行遍历,然后根据值进行操作,但是请求是个异步操作,forEach又是一个同步操作 ...
- Ansible-Playbook中的变量使用
变量名:仅能由字母.数字和下划线组成,且只能以字母开头 变量来源: 1.ansible all -m setup 远程主机的所有变量都可直接调用 #显示所有变量 ansible all -m setu ...
- [Fundamental of Power Electronics]-PART I-5.不连续导电模式-5.4 总结与重点
5.4 总结与重点 基本的buck,boost以及buck-boost电路的特点总结在表5.2中.其中给出了\(K_{crit}(D)\)的表达式,CCM和DCM下的变换比,以及DCM下二极管导通占空 ...
- python3使用myqr生成链接二维码
技术背景 二维码技术在各个领域中都已经有非常成熟的应用,比如随处可见的二维码支付,比如疫情期间的绿码,再比如工业领域中,可以使用二维码作为定位的标签,大大提升了室内定位技术的精确度.二维码的格式内容大 ...
- PAT A1025 考生排名问题
题目要求:有n个考场,每个考场有若干个考生,现给出各个考场中考生的准考证号与分数,按照分数从高到低排序,并按顺序输出所有考生的准考证号,排名,考场号以及所在的考场排名 #include<cstd ...