一个最简单的Linux驱动程序的编译运行

系统 Linux
模块(module)是一段可以被动态链接的目标代码(.ko),它可由insmod命令动态的装载并链接到正在运行的内核。链接后,它就成了内核的一部分,直到用rmmod命令解除链接并卸载。Linux驱动程序就是一种特殊的可装载内核模块。

[[207441]]

一、背景知识

可装载内核模块:为了使系统功能能够更灵活的扩充,Linux支持内核的动态扩展,即在系统运行时给内核增加新的功能(即模块module)。

模块:模块(module)是一段可以被动态链接的目标代码(.ko),它可由insmod命令动态的装载并链接到正在运行的内核。链接后,它就成了内核的一部分,直到用rmmod命令解除链接并卸载。

Linux驱动程序就是一种特殊的可装载内核模块。

说明:本代码在linux2.6以上测试运行通过。linux3.*也可。本人用ubuntu12.04,linux内核版本linux3.5。

二、编译内核模块

如何由源码(.c文件)生成动态链接的目标代码(即.ko文件)?

首先写一个最简单的驱动程序源码:hello.c

[html] view plain copy

 

  1. #include <linux/init.h> //声明头文件 
  2.  
  3. #include <linux/module.h> 
  4.  
  5. //模块加载函数 
  6.  
  7. static int hello_init(void) 
  8.  
  9.  
  10. printk(KERN_INFO " Hello World enter\n"); //系统调用打印函数,类似于用户调用的printf 
  11.  
  12. return 0; 
  13.  
  14.  
  15. //模块卸载函数 
  16.  
  17. static void hello_exit(void) 
  18.  
  19.  
  20. printk(KERN_INFO " Hello World exit\n "); 
  21.  
  22.  
  23. module_init(hello_init); //向系统注册模块加载函数 
  24.  
  25. module_exit(hello_exit); //向系统注册模块卸载函数 
  26.  
  27. MODULE_AUTHOR("Barry Song <21cnbao@gmail.com>"); //模块作者等信息声明,可选 
  28.  
  29. MODULE_LICENSE("Dual BSD/GPL"); //模块许可证声明 
  30.  
  31. MODULE_DESCRIPTION("A simple Hello World Module"); //模块描述声明,可选 
  32.  
  33. MODULE_ALIAS("a simplest module"); //模块别名的声明,可选 

 

将这段程序编译为hello.ko的步骤:

1、编写makefile文件

[html] view plain copy

  1. KVERS = $(shell uname -r) #变量KVERS为当前linux版本值 
  2.  
  3. obj-m += hello.o 
  4.  
  5. default
  6.  
  7. make -C /lib/modules/$(KVERS)/build M=$(CURDIR) modules 

 

-C后面是linux内核源码树目录,M=后面是源码目录

2、在终端下输入 cd /**/**,找到hello.c源码所在的目录

 

 

 

 

3、在终端输入 make命令。

 

 

 

 

4、如图所示即可生成hello.ko文件。

 

 

 

 

其他文件是生成的一些中间文件。

三、装载

加载:在终端输入insmod命令,模块加载完成。

[[207445]]

 

注意:如果终端没有打印printk语句,在linux系统日志文件/var/log/syslog中查看。

 

 

 

 

 

卸载:在终端输入remmod命令。

 

 

 

 

打印信息同在日志文件中。

四、一些理论知识

模块加载函数:用宏“module_init”指定,它返回整型值。若初始化成功则返回0,若失败则返回一个负值作为错误码。

模块卸载函数必须用宏“module_exit”指定,无返回值。

printk用法:

1、printk(KERN_INFO "Hello, world!/n");

2、printk("<6>Hello, world!/n");

***个参数指定输出级别:分为以下八种:

1、KERN_EMERG 用于紧急消息, 常常是那些崩溃前的消息.

2、KERN_ALERT 需要立刻动作的情形.

3、KERN_CRIT 严重情况, 常常与严重的硬件或者软件失效有关.

4、KERN_ERR 用来报告错误情况; 设备驱动常常使用 KERN_ERR 来报告硬件故障.

5、KERN_WARNING 有问题的情况的警告, 这些情况自己不会引起系统的严重问题.

6、KERN_NOTICE 正常情况, 但是仍然值得注意. 在这个级别一些安全相关的情况会报告.

7、KERN_INFO 信息型消息. 在这个级别, 很多驱动在启动时打印它们发现的硬件的信息.

8、KERN_DEBUG 用作调试消息. 

责任编辑:庞桂玉 来源: 嵌入式Linux中文站
相关推荐

2021-11-12 11:28:01

Linux 内核驱动Linux 系统

2011-05-19 15:14:49

PostgreSQL

2011-01-06 16:29:08

linuxtasklet机制

2013-10-31 16:29:10

Linux内核

2021-11-29 07:55:45

Linux GPIO Linux 系统

2021-05-26 07:53:58

Linux运维Linux系统

2011-01-10 18:21:38

linux编写程序

2010-01-07 13:27:22

Linux驱动程序

2016-11-08 18:53:08

编译器

2023-07-25 15:17:38

Linux操作系统开发

2009-12-07 09:39:04

Linux设备驱动硬件通信

2011-04-22 17:29:37

Linux网卡

2022-05-13 09:14:47

NVidia开源Linux

2009-07-06 18:17:46

JDBC驱动程序

2021-12-06 07:47:36

Linux 驱动程序Linux 系统

2020-12-21 20:25:24

UbuntuDebian LinuC 程序

2022-03-08 09:25:52

数据库系统MongoDB

2021-11-22 08:14:23

Linux Linux驱动Linux 系统

2022-05-23 13:17:32

Linux开源NVIDIA

2009-10-23 10:25:27

驱动程序技巧
点赞
收藏

51CTO技术栈公众号