近日尝试linux内核设备模块编程,使用《linux内核编译》一书,但在新版的2.6内核中,例子程序无法编译通过,在网上搜寻了很久,都没有找到一个完整的解决方案,最后终于在网站http://lwn.net/获得帮助,现总结如下
init和clear命名方式改变,makefile改变
用一个hello world程序说明
原版如下:
#include <linux/kernel.h>
#include <linux/module.h>
#if CONFIG_MODVERSIONS==1
#define MODVERSIONS
#include <linux/modversions.h>
#endif
int init_module()
{
printk("Hello, world - this is the kernel speaking\n");
return 0;
}
void cleanup_module()
{
printk("Short is the life of a kernel module\n");
}
Mvakefile如下
CC=gcc
MODCFLAGS := -Wall -DMODULE -D__KERNEL__ -DLINUX
hello.o: hello.c /usr/include/linux/version.h
$(CC) $(MODCFLAGS) -c hello.c
现在需要改成
#include <linux/init.h>
#include <linux/module.h>
#include <linux/kernel.h>
static int hello_init(void)
{
printk("<1> Hello, world\n");
return 0;
}
static void hello_exit(void)
{
printk("<1> Goodbye, cruel world\n");
}
module_init(hello_init);
module_exit(hello_exit);
Makefile 如下
ifneq ($(KERNELRELEASE),)
obj-m:= hello.o
else
KDIR:= /lib/modules/$(shell uname -r)/build
PWD:= $(shell pwd)
default:
$(MAKE) -C $(KDIR) SUBDIRS=$(PWD) modules
endif
有一点需要重点说明
$(MAKE) -C $(KDIR) SUBDIRS=$(PWD) modules
这个句子前面一定要加一个TAB,切记切记
这样生成的会有一个*.ko的文件,这个文件才能被insmod进去
不过我还遇到一个问题,就是我的GCC升级到3.4.1后编译出来的文件无法insmod进去,提示要用GCC 3.3,好郁闷,最终还是用GCC 3.3搞定了
另外就是这个东西insmod进去以后什么提示也没有,只是能在lsmod看到添加成功,rmmod时也是没有任何信息,那语句中的printk输出到哪里了?用的是多用户命令行方式,没有打开X
以上程序在内核2.6.5上编译通过
在 http://lwn.net/Articles/driver-porting/你会得到更多2.6内核改变的信息