过了个年,好久没碰专业内的东西了,之前做的JOS相关的东西都快忘了,还是看了前面两篇日志才想起来。
当进入内核后基本都是比较简单的代码了,我也并没有全部分析,根据讲义要求只分析了一下printf函数和堆栈的backtrace,所以这篇日志也就写这两个方面吧。
1、printf函数。
进入kernel后从i386_init函数开始,首先做一些初始化工作,包括部分内存的清零,初始化显示器串口等(无非是判断一下地址使光标闪动正确的位置等),然后调用了cprintf,尝试讲一个10进制的数字用8进制来表示,而这个函数是需要我们完成的。
进入cprintf函数(printf.c文件)后,首先是使用va_前缀的函数(也许是宏)来取出参数,这是标准的c语言可变长度参数的实现形式。这个函数其实是一些_buildin前缀的函数的别名,而_buildin函数则是gcc内置的函数,并不在任何的JOS代码中有定义,当GCC进行编译的时候会自动将这些函数名与相应的函数体连接。之后则调用vcprintf函数。
vcprintf函数定义在同一个文件里,直接调用vprintfmt函数,值得注意的是传入一个函数指针,指向了本文件的putch函数,这个函数之后再讲。
进入vprintfmt(printfmt.c文件)函数,发现这里实现的控制打印格式的逻辑,然后调用传进来的函数指针,进行具体的打印工作。找到含有case 'o'的代码,仿照case 'd'的代码完成8进制数字显示即可,也就是原封不动的复制过来,将base改成8,如下:</