Hello world 之Linux驱动程序

       貌似在学习一种新的编程语言的时候,通常会使用一个Hello world!作为一个入门,今天我们也用一个类似的程序,打开进入linux内核空间的大门。
       要在linux的内核空间中进行程序设计,需要准备哪些准备工作呢?首先你要有一个linux的开发环境:一个linux操作系统、一套linux内核源代码和编译环境。
       操作系统目前可选择的太多了,各种各样的发行版应有尽有,你可以在distrowatch.com看到各个版本的信息。目前最为流行的是ubuntu,采用debian包管理,安装配置软件非常简便,建议使用官方发布的版本,开发linux2.6内核下的驱动最好使用这个版本,如果是开发linux2.4下的驱动建议采用redhat企业版。
    linux内核源代码没有选择,你的驱动运行于那个linux系统下,你就要获得这个linux系统的内核源码。例如我们开发的驱动要运行在X86 ubuntu上,你就要获得这个ubuntu的源码,版本也要一致。驱动要运行在一个嵌入式linux单板上,你就要拿到这个单板所运行的linux的源码。绝对不能在kernel.org上随便下载一个内核,如果内核不匹配可能会出现意想不到的问题,例如什么printk overflow问题。
    编译环境问题比较复杂,跟开发环境有很大关系,你的驱动如果要运行在嵌入式平台上,你就需要一套交叉工具链,所谓的交叉工具链其实就是能在你的X86 CPU上编译出运行于mips、arm等其他架构上的程序的一套编译器。假定驱动运行在X86平台上,我们需要安装gcc编译器和make utility,如何安装这里就不一一赘述了,请google查一下。
    另外,还有一个使用什么工具编辑程序的问题,当然是vi或者gvim了,我在后续的文章中介绍如何使用vi。

    准备工作做好了,我们开始写第一个内核程序(假定开发环境是ubuntu linux2.6,以前一直在linux2.4下开发linux驱动,转到2.6下才发现很多东东都变了)。我们Hello world内核程序有两个文件组成,一个Makefile,另一个是test.c。
    test.c内容如下:

 1  #include  < linux / init.h >
 2  #include  < linux / kernel.h >
 3  #include  < linux / module.h >
 4 
 5  static   int  __init eth_init( void )
 6  {
 7      printk( " init module/n " );
 8       return   0 ;
 9  }
10  static   void  __exit eth_exit( void )
11  {
12      printk( " exit modules/n " );
13  }
14 
15  module_init(eth_init);
16  module_exit(eth_exit);

    Makefile内容如下:
PWD = $(shell pwd)
KERNEL_SRC = /usr/src/linux-source-2.6.15/
obj-m := test.o
module-objs := test.o
all:

    $(MAKE) -C $(KERNEL_SRC) M=$(PWD) modules
clean:
    rm *.ko
    rm *.o
注意千万不能从我这篇blog中通过复制粘贴
Makefile内容去创建Makefile文件!!!我要下载源码test.tgz.zip,原文件名是test.tgz,请去掉后面的zip后解压。
    在编译源码之前需要修改一下Makefile中的
KERNEL_SRC,使其指向你的linux内核源码。修改完成后,在test.c和Makefile所在的目录下运行make,如果看到类似输出
make -C /usr/src/linux-source-2.6.15/ M=/home/vmeth modules
make[1]: Entering directory `/usr/src/linux-source-2.6.15''''''''
  CC [M]  /home/vmeth/test.o
  Building modules, stage 2.
  MODPOST
  CC      /home/vmeth/test.mod.o
  LD [M]  /home/vmeth/test.ko
make[1]: Leaving directory `/usr/src/linux-source-2.6.15''''''''
    OK,一个内核程序编译出来了,目录下多了一个test.ko。出现/usr/bin/ld: crt1.o: No such file: No such file or directory.这个错误怎么办?这个问题的原因是usr/lib/目录下缺少crt1.o。解决方法:安装libc6-dev。另外在编译这个程序之前最好先把内核编译一下(在内核源码所在的目录下执行make menuconfig,什么都不要修改退出,退出时会提示你是否要保存配置,选择保存,然后执行make,整个make过程需要很长时间,----我在
make menuconfig时,好像提示缺少ncurses库,安装上libncurses就OK了)。
   
    下一步就是运行这个程序了,在root用户下执行insmod ./test.ko。没有错误提示的话,这个程序就被执行了,运行一下dmesg,最后一行是不是可以看到init module
字样。执行rmmod test;再运行dmesg,最后一行又出现了exit modules。这个内核程序目前只是打印了两行字,后续我将会扩充这个程序,完成一些有用的功能。
-------------------------------------------------------------------------------------------

Makefile文件分析:
obj-m := hello.o  代表了我们要构造的模块名为hell.ko,make 会在该目录下自动找到hell.c文件进行编译。如果 hello.o是由其他的源文件生成(比如file1.c和file2.c)的,则在下面加上(注意红色字体的对应关系):
hello-objs := file1.o file2.o ......

    $(MAKE) -C $(KERNELDIR) M=$(PWD) modules
其中
-C $(KERNELDIR) 指定了内核源代码的位置,其中保存有内核的顶层makefile文件。
   
M=$(PWD指定了模块源代码的位置
   
modules目标指向obj-m变量中设定的模块。
---------------------------------------------------------------------

几篇文章:

第一章 模块(Modules) http://greenlinux.blogcn.com/diary,103232026.shtml

从 2.4 到 2.6:Linux 内核可装载模块机制的改变对设备驱动的影响

http://www.ibm.com/developerworks/cn/linux/l-module26/

Linux2.6内核驱动移植参考

http://blog.chinaunix.net/u1/40912/showart_377391.html

---------------------------------------------------------------------

PS.

由于我用的是REDHAT9.0内核版本为2.4.20 编译hello的时候有N多的错误

没有办法,下载了2.6的内核作为原代码进行编译,成功!但是在insmod的时候出现了

hello.o: couldn''t find the kernel version the module was compiled for

错误,我感觉是由于编译时候使用是用2.6内核的,而却要运行在2.4的内核的原因;参考

http://www.linuxforum.net/forum/showflat.php?Cat=&Board=linuxK&Number=131413&view=collapsed&sb=5&o=all

上的说明解决了此问题.小记之.

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值