linux0.12 链接过程
终于编译OK了。。可链接就是一大堆错误
问题1:
boot/head.o: In function `startup_32': (.text+0x10): undefined reference to `_stack_start' boot/head.o: In function `startup_32': (.text+0x2e): undefined reference to `_stack_start' boot/head.o: In function `after_page_tables': (.text+0x540c): undefined reference to `_main' boot/head.o: In function `ignore_int': (.text+0x5440): undefined reference to `_printk'
一堆的undefined reference to xxxxx,stack_start 定义在kernel/sched.c中,main 定义在init/main.c中,实际上这些符号是存在的,可为什么链接的时候找不到呢。最终找到原因,是由于现在GCC在编译C时不会在函数前加"_"了,即以前GCC编译一个C函数,例如:fun() 编译后会变成_fun(), 而现在还是fun(),所以就导致了符号找不到,这个叫name magling,有兴趣的可以自行google下。
解决的思路有两种:
1. 网上的是把把有引用C符号的.s 和 .S中的变量前的"_"全部删掉,这个用正则不太好弄,得一个个删。
2. 在查name magling的时候,有幸让我看到了-fleading-underscore这个option, 可以让让GCC在编译的时候强制加上"_",这种只需要在CFLAGS中加-fleading-underscore即可,第二种方法更省事。
方案2具体解决方法
msed -O -O\ -fleading-underscore
请记得在改完后,make clean,后再make。
问题2:
main.c:(.text+0x6c): undefined reference to `_printf' main.c:(.text+0x87): undefined reference to `_printf' main.c:(.text+0x142): undefined reference to `_puts' main.c:(.text+0x1e9): undefined reference to `_printf'
额,之前编译的时候似乎正则没全部匹配到,只好手动改下了,把init/main.c 里的printf 全改成 printw
问题3:
vsprintf.c:(.text+0x18cb): undefined reference to `___stack_chk_fail' fs/fs.o: In function `_do_execve': (.text+0x3554): undefined reference to `___stack_chk_fail' kernel/chr_drv/chr_drv.a(tty_ioctl.o): In function `_tty_ioctl': tty_ioctl.c:(.text+0x51d): undefined reference to `___stack_chk_fail'
直接google到的答案,这是由于ubuntu的GCC默认是加-fstack-protector的,这里在CFLAGS中去掉这个option,加入-fno-stack-protector,至于这个option到底有什么作用,请有兴趣的自行google
msed -O -O\ -fno-stack-protector
问题4:
build.c:(.text+0x1c): undefined reference to `_stderr' build.c:(.text+0x24): undefined reference to `___fprintf_chk' /tmp/cctqTbXN.o: In function `_main': build.c:(.text+0x7d): undefined reference to `_strcmp' build.c:(.text+0x99): undefined reference to `___xstat' build.c:(.text+0xab): undefined reference to `_perror'
这个问题是由问题1的解决方案引入的,build.c链接时用的是glibc,而glibc在是没-fleading-underscore这个option的,所以在编译build.c时我们 把这个option去掉。。。,目前没找到什么好方法,我只好手动把CFLAGS展开,然后去掉-fleading-underscore了
修改linux0.12根目录下的Makefile第50行
tools/build: tools/build.c
$(CC) $(CFLAGS) \
-o tools/build tools/build.c
上面是原来的,下面是展开后的
tools/build: tools/build.c
$(CC) -w -O -fno-stack-protecto -m32 -fstrength-reduce -fomit-frame-pointer \
-o tools/build tools/build.c
问题5:
build.c:(.text+0xcb): undefined reference to `MAJOR' build.c:(.text+0xe3): undefined reference to `MINOR' build.c:(.text+0x16c): undefined reference to `MAJOR' build.c:(.text+0x184): undefined reference to `MINOR'
缺少两个宏定义,将以下两个定义加入到build.c中即可
#define MAJOR(x) (((unsigned)(x))>>8) #define MINOR(x) ((x)&0xff)
问题6:
/dev/hd6: No such file or directory Couldn't stat root device.
这个是在linux0.12根目录下的Makefile中定义的
ROOT_DEV=/dev/hd6 SWAP_DEV=/dev/hd2
但我们没这东西,阅读build.c 代码,了解到这里可以写成FLOPPY代替/dev/hd6
) {
], "FLOPPY")) {
], &sb)) {
perror(argv[]);
die("Couldn't stat root device.");
}
major_root = MAJOR(sb.st_rdev);
minor_root = MINOR(sb.st_rdev);
} else {
major_root = ;
minor_root = ;
}
} else {
major_root = DEFAULT_MAJOR_ROOT;
minor_root = DEFAULT_MINOR_ROOT;
}
问题6:
/dev/hd2: No such file or directory Couldn't stat root device.
与上个问题类似,相同的地方,阅读build.c,可以用NONE代替/dev/hd2
) {
], "NONE")) {
], &sb)) {
perror(argv[]);
die("Couldn't stat root device.");
}
major_swap = MAJOR(sb.st_rdev);
minor_swap = MINOR(sb.st_rdev);
} else {
major_swap = ;
minor_swap = ;
}
} else {
major_swap = DEFAULT_MAJOR_SWAP;
minor_swap = DEFAULT_MINOR_SWAP;
}
问题7:
Non-GCC header of 'system'
我们先看代码
],O_RDONLY,))<)
die("Unable to open 'system'");
if (read(id,buf,GCC_HEADER) != GCC_HEADER)
die("Unable to read header of 'system'");
] != )
die("Non-GCC header of 'system'");
; (c=read(id,buf, ; i+=c )
,buf,c)!=c)
die("Write call failed");
大致分4步,1. 打开第4个参数的文件,即system,2. 读system的头1024个字节;3. 判断程序入口地址是不是0x0;4. 读取剩下的部分写入Image中。错在第三步。我们用readelf -h system来看下
~/workspace/blog/linux-0.12/tools$ readelf -h system ELF Header: Magic: 7f 4c Class: ELF32 Data: 's complement, little endian Version: (current) OS/ABI: UNIX - System V ABI Version: Type: EXEC (Executable file) Machine: Intel Version: 0x1 Entry point address: 0x80480a0 Start of program headers: (bytes into file) Start of section headers: (bytes into file) Flags: 0x0 Size of (bytes) Size of program headers: (bytes) Number of program headers: Size of section headers: (bytes) Number of section headers: Section header
这里可以看到程序的入口是0x80480a0,不是0x0,所以要告诉链接器,把text段放到0x0上,在linux0.12根目录下面的Makefile,修改ld命令,在其后面加上-Ttext 0x0 -e startup_32, 同时在head.s 的.global中加入startup_32以便外部访问。除此而外,GCC头的第24个字节是version,第26个字节开始才是程序入口。。。额,索性直接把判断GCC头那段给干掉。。。
然后链接也OK了,生成了Image 进入调试阶段。。
linux0.12 链接过程的更多相关文章
- linux0.12 编译过程
感谢这篇文章的作者: http://www.cnblogs.com/strugglesometimes/p/4231359.html 编译是个很蛋疼的事情,本想把linux0.12在bochs上 ...
- linux0.12 学习总序(不断更新状态中)
最近有空闲时间,想静下心来学点东西.一直对kernel有兴趣,又苦于无从下手,就拿linux0.12练手.尝试了解并熟悉kernel各模块工作原理. 接下来的博客主要用来记录自己所遇到的问题和解决的方 ...
- 第一次作业:基于Linux-0.12的进程分析
这次作业主要基于Linux-0.12的源代码,分析Linux是如何组织进程,进程的状态之间是如何转换,以及进程是如何调度的. 一. 进程的概念: 1.进程就是:程序在数据集合上的一次运行过程,是系统进 ...
- GCC编译链接过程
编译链接过程 代码 #cat main.c #include <stdio.h> int add(int x, int y); int sub(int x, int y); int mul ...
- 鸿蒙内核源码分析(静态链接篇) | 完整小项目看透静态链接过程 | 百篇博客分析OpenHarmony源码 | v54.01
百篇博客系列篇.本篇为: v54.xx 鸿蒙内核源码分析(静态链接篇) | 完整小项目看透静态链接过程 | 51.c.h.o 下图是一个可执行文件编译,链接的过程. 本篇将通过一个完整的小工程来阐述E ...
- 【图片+代码】:GCC 链接过程中的【重定位】过程分析
作 者:道哥,10+年嵌入式开发老兵,专注于:C/C++.嵌入式.Linux. 关注下方公众号,回复[书籍],获取 Linux.嵌入式领域经典书籍:回复[PDF],获取所有原创文章( PDF 格式). ...
- [转]C++编译链接过程详解
C语言的编译链接过程要把我们编写的一个c程序(源代码)转换成可以在硬件上运行的程序(可执行代码),需要进行编译和链接.编译就是把文本形式源代码翻译为机器语言形式的目标文件的过程.链接是把目标文件.操作 ...
- mysql学习(2)-Navicat Premium 12 链接MySQL8.0.11数据库报2059错误
Navicat Premium 12 链接MySQL8.0.11数据库报2059错误 1,问题现象 安装完MySQL8.0.11和Navicat Premium12后,我们会用Navicat去测试连接 ...
- C-从源文件到可执行文件的详细编译链接过程
一直用windows一键搞定, 没有去了解详细的编译链接过程, 今天看了一篇文章, 顺便实验和记录在Linux下逐步生成的步骤. 预处理: 执行#include, #define, #if, #ifd ...
随机推荐
- 苹果Swift编程语言新手教程【中国版】
Swift代码语言教程:在刚刚过去的WWDC2014大会上,苹果公司新公布了一种编程语言Swift.据悉.Swift语言继承了C语言以及Objective-C的特性,且克服了C语言的兼容性问题.对于广 ...
- MySQL创建新用户、增加账户的2种方法及使用实例
可以用两种方式创建MySQL账户:1.使用GRANT语句2.直接操作MySQL授权表最好的方法是使用GRANT语句,因为这样更精确,错误少.创建超级用户: mysql> GRANT ALL PR ...
- Android - ContentProvider机制
以下资料摘录整理自老罗的Android之旅博客,是对老罗的博客关于Android底层原理的一个抽象的知识概括总结(如有错误欢迎指出)(侵删):http://blog.csdn.net/luosheng ...
- yii console.php 报错 Property "CConsoleApplication.theme" is not defined.
默认配置的话,是不会出现这个错误的,应该是有人为修改了 yiic.php 这个文件,本来是 $config 载入的应该是 console.php ,人为修改后载入了 main.php 这个配置文件了 ...
- DELL RACADM 批量升级戴尔IDRAC固件
需求:通过服务器远程管理IP批量升级戴尔IDRAC固件 工具:racadm.ipmitool.Remote Access Configuration Tool 下载: 第一步,将要更新BMC IP写入 ...
- 线段树---HDU1754 I hate it
这个题也是线段树的基础题,有了上一个题的基础,在做这个题就显得比较轻松了,大体都是一样的,那个是求和,这个改成求最大值,基本上思路差不多,下面是代码的实现 #include <cstdio> ...
- mongodb入门教程二
title: mongodb入门教程二 date: 2016-04-07 10:33:02 tags: --- 上一篇文章说了mongodb最基本的东西,这边博文就在深入一点,说一下mongo的一些高 ...
- 画板社交工具开发分享——HTML5 canvas控件、PHP、社交分享学习(一)
不谈国内社交网站(人人.微博等)目前的盈利状况如何,facebook.twitter都已经取得了很大的成功.这一定程度上都得益于人们对社交的喜爱和投入. 目前的情况来看,也有很多人已经对直接的文字分享 ...
- deb文件安装命令
一般在此类发行版中可以直接双击安装 手动安装.如果您喜欢使用终端,您需要管理员权限来安装一个 .deb 文件. 打开终端后,输入: sudo dpkg -i package_file.deb 要卸载一 ...
- 转载:/etc/resolv.conf的作用
转载网址:http://jiao-zhong.blog.sohu.com/97976004.html 该文件是DNS域名解析的配置文件,它的格式很简单,每行以一个关键字开头,后接配置参数.resolv ...