Chinaunix首页 | 论坛 | 博客
  • 博客访问: 118150
  • 博文数量: 6
  • 博客积分: 1849
  • 博客等级: 上尉
  • 技术积分: 121
  • 用 户 组: 普通用户
  • 注册时间: 2009-06-26 16:45
文章存档

2013年(1)

2012年(3)

2011年(2)

分类: LINUX

2011-04-24 11:48:46

一个简单的hello,world程序如下:

1#include <\stdio.h\>
2int main()
3{
4    printf("hello,world\n");
5    return 0;
6}

对于上面的程序,我们通常一步到位进行编译:

1gcc hello.c -o hello

这样的命令简单方便,不过通常会让我们忽略从.c文件到可执行文件的整个编译过程。通常在使用gcc编译程序时,编译过程通常会分为4个阶段,即预处理(pre-processing),编译(compiling),汇编(assembling),链接(linking)。

在预处理阶段,一般输入的是.c文件,而输出的是.i文件。在此阶段中通常会处理源文件中的预处理命令,比如#define,#include,#ifdef等命令。如果想要生成这种.i中间文件,那么可以使用下面命令:

1gcc -E hello.c -o hello.i

在编译阶段,输入的是.i中间文件,输出的是.s汇编语言文件。可以使用下面的命令:

1gcc -S hello.i -o test.s

在汇编阶段,输入的是上一步产生的.s文件,产生的是二进制机器代码.o文件。此阶段对应的命令如下:

1gcc -c test.s -o test.o

在链接阶段,输入的是.o二进制机器代码文件,连同其他的(如果有的话)机器代码文件和库文件一起汇集成一个可执行的二进制代码文件。

1gcc test.o -o test

以上是一个程序编译完整的四个阶段。一般来说我们会将前三个阶段一步搞定,那么整个编译过程可以用下面两条命令就完成:

1gcc -c test.c -o test.o
2gcc test.o -o test

也就是先生成目标文件,再将目标文件连接成可执行文件。当你熟悉了整个编译过程后,可以用一开始我们说的一条命令来完成。

了解普通文件的编译过程,我们现在就将hello.c中的代码转换成模块编程中的.c代码。首先我们要更改头文件:

1#include <\linux/kernel.h\>
2#include <\linux/module.h\>

这与我们一般的头文件不同。一般我们在用户态下编写C程序,头文件会放在:/usr/include/下,而我们模块编程时,它使用的是内核中的头文件,一般在:cd /usr/src/linux-headers-2.6.32-21/include/。特别的我们这里使用的是include/目录下linux/这个子目录中的头文件,因此模块编译的时候会自动在内核中的include/目录下找linux/kernel.h这样的头文件。

其次,printf到printk是一个典型的用户态下编程与内核模块编程的不同。可能我们一开始会比较奇怪,为什么我make成功,加载也成功,但是就是不能显示printk里面的语句呢?我们可以这么想printk就是专门为内核“服务”。它一般输出的语句都在内核的日志文件当中。


转载网址:http://edsionte.com/techblog/archives/1350

有段时间调试程序的时侯,也发现不能输出内容,可能是以上那们哥们讲的原因,就转上来,供大家学习!


阅读(1963) | 评论(1) | 转发(0) |
0

上一篇:没有了

下一篇:mini2440 移植linux2.6.30.10 及yaffs2文件系统

给主人留下些什么吧!~~

2007robot2011-04-24 17:53:07

这也是用户态和内核态编程的不同!我是这样理解的。