Linux驱动入门——构建和运行模块
Hello world模块
本文介绍如何向内核中添加一个hello模块。该模块的功能是在模块加载时,向系统日志输出“hello world\n” 在模块卸载时输出“Good bye,cruel world!".
一个模块源代码一般有含有一个init函数(加载时调用)和一个exit函数(卸载时调用)。这两个函数由分别由宏module_init和module_exit调用。因而一个简单的Hello world模块源代码如下:
#include <linux/init.h><span style="white-space:pre"> </span>//这个头文件包含了你的模块初始化与清除的函数
#include <linux/module.h><span style="white-space:pre"> </span>//这个头文件包含了许多与模块加载有关的符号与函数的定义
<span style="white-space:pre"> </span>//如果模块需要参数传递,还可以参考moduleparam.h MODULE_LICENSE("GPL");<span style="white-space:pre"> </span>//表明本模块带有一个GPL的自由许可证,没有这行,加载模块时内核可能会抱怨
MODULE_AUTHOR("Windeal")<span style="white-space:pre"> </span>//author //模块加载时调用的函数
static int hello_init(void)
{
printk(KERN_ALERT "Hello world!\n");//printk是内核函数接口,会将内容输出到系统日志,KERN_ALERT表明输出优先级,printk也是行缓冲的
return 0;
} //模块卸载时调用的函数
static void hello_exit(void)
{
printk(KERN_ALERT "Good bye, cruel world!\n");
} module_init(hello_init);//宏,指定模块加载时调用的函数
module_exit(hello_exit);//宏,指定模块加载时调用的函数
关于MODULE_XXX宏做一个简单介绍:
MODULE_LICENSE<span style="white-space:pre"> </span>//许可证,具体许可证内容自行百度
MODULE_AUTHOR // 声明作者
MODULE_DESCRIPTION //对这个模块作一个简单的描述,这个描述是"human-readable"的
MODULE_VERSION // 这个模块的版本
MODULE_ALIAS // 这个模块的别名
MODULE_DEVICE_TABLE //告诉用户空间这个模块支持什么样的设备
编译、Makefile
前面我们已经写好了一个简单的hello模块,接下来要对模块进行编译。我们使用Makefile文件
# If KERNELRELEASE is defined, we've been invoked from the
# kernel build system and can use its language.
ifneq ($(KERNELRELEASE),)
obj-m := hello.o
# Otherwise we were called directly from the command
# line; invoke the kernel build system.
else
KERNELDIR ?= /lib/modules/$(shell uname -r)/build #KERNELDIR赋值,指代内核源代码目录 PWD := $(shell pwd) #当前目录
default:
$(MAKE) -C $(KERNELDIR) M=$(PWD) modules
endif
这是一个很有意思的Makefile脚本,当你在当前目录make的时候,它会执行两次。第一次执行的时候KERNELRELEASE未定义,因而会执行else分支的default目标的命令,即$(MAKE) -C $(KERNELDIR) M=$(PWD) modules .这条命令会进入内核源代码目录,执行内核源代码根目录下的Makefile,根目录下的Makefile会定义KERNELRELEASE,并切换到当前目录,再执行一遍当前目录的Makefile。此时KERNELRELEASE已被定义,因而会执行obj-m
:= hello.o 这条命令。从而整个依赖关系才完整,生成目标文件hello.ko。
ifeq ($(KERNELRELEASE),)目前,并无用处,它的由来是指在Linux源码根目录下的Makefile编译内核时,KERNELRELEASE宏会被定义,那么如果是从源码根目录开始的make则会将myhello.o模块编译进内核。
$(MAKE) -C $(KERNELDIR) M=$(PWD) modules 是Makefile的规则:这里的$(MAKE)就相当于make,-C 选项的作用是指将当前工作目录转移到你所指定的位置。“M=”选项的作用是,当用户需要以某个内核为基础编译一个外部模块的话,需要在make modules 命令中加入“M=dir”,程序会自动到你所指定的dir目录中查找模块源码,将其编译,生成ko文件。
模块加载:
编译好的模块hello.ko需要加载如内核才能使用。
sudo insmod XXX<span style="white-space:pre"> </span>#加载模块
sudo lsmod<span style="white-space:pre"> </span>#查看已加载的所有模块
sudo rmmod XXX<span style="white-space:pre"> </span>#卸载模块
而我们的prink输出会写入系统日志,可以通过dmesg查看。
因此我们的测试过程如下:
windeal@ubuntu:driver$ lsmod | grep hello #确认当前没有加载hello模块
windeal@ubuntu:driver$ sudo insmod hello.ko #加载hello.ko模块
windeal@ubuntu:driver$ dmesg<span style="white-space:pre"> </span>#查看系统日志
[190444.174802] Hello world!
windeal@ubuntu:driver$ lsmod | grep hello<span style="white-space:pre"> </span>#查看是否加载了hello模块
hello 12449 0
windeal@ubuntu:driver$ sudo rmmod hello<span style="white-space:pre"> </span>#卸载hello模块
windeal@ubuntu:driver$ dmesg<span style="white-space:pre"> </span>#查看系统日志
[190444.174802] Hello world!
[190487.913667] Good bye, cruel world!
windeal@ubuntu:driver$ lsmod | grep hello<span style="white-space:pre"> </span>#查看模块是否被卸载了
windeal@ubuntu:driver$
图:
Linux驱动入门——构建和运行模块的更多相关文章
- Linux驱动学习之常用的模块操作命令
1.常用的模块操作命令 (1)lsmod(list module,将模块列表显示),功能是打印出当前内核中已经安装的模块列表 (2)insmod(install module,安装模块),功能是向当前 ...
- Linux驱动之内核加载模块过程分析
Linux内核支持动态的加载模块运行:比如insmod first_drv.ko,这样就可以将模块加载到内核所在空间供应用程序调用.现在简单描述下insmod first_drv.ko的过程 1.in ...
- linux 驱动入门1
世事艰难,人生不易. 夜深人静时候,回顾过去,往事历历在目.创南京,混苏州,下上海.都付出了巨大的努力.多少个不眠的夜晚,在冥思苦想.天生愚钝.又不是学计算机的.一直没较为深刻的理解 编程什么东西,一 ...
- linux 驱动入门3
不吃苦中苦,难为人上人.努力,给老婆孩子提供个良好的生活居住环境. http://www.cnblogs.com/nan-jing/articles/5806399.html 上文提到.可以自动创建了 ...
- linux 驱动入门4
不吃苦中苦,难为人上人.努力,给老婆孩子提供个良好的生活居住环境.http://www.cnblogs.com/nan-jing/articles/5806399.html上文提到了如何创建proc节 ...
- linux 驱动入门5
慢慢的开始转驱动,目前比较有时间,一定要把驱动学会.哎.人生慢慢路,一回头.已经工作了八九年了.努力.在买套房.改退休了.学驱动.个人认为首先要熟悉驱动框架.慢慢来.心急吃不了热豆腐. 看网上都说的设 ...
- linux 驱动入门2
不吃苦中苦,难为人上人.努力,给老婆孩子提供个良好的生活居住环境. http://www.cnblogs.com/nan-jing/articles/5775038.html 这里提到.有这么多牛人. ...
- Linux驱动之建立一个hello模块
目标:在开发板上执行insmod hello.ko能在控制台打印出hello init:接着执行rmmod会在控制台打印出hello exit 建立一个hello模块的步骤如下: 1.建立一个hell ...
- linux 驱动入门6
看/sys目录经常看到bus device driver class. 这也是网上大量说的驱动驱动模型.这些的关系得熟悉得明白吧.是的.今天我先不整他们的关系.先逐个击破,然后再统一来理清楚他们之间的 ...
随机推荐
- python 打印 九九表
用Python 打印九九表. print 每打印一行默认会带有换行, 在print语句后加上,end = 't' 会变成tab. 排版会好点. def main(): for i in range(1 ...
- MVC中一个Form多个submit在controller中如何区分提交的是那个submit(如:登陆和注册)
1. 用Html.BeginForm(ActionName,ControllerName,Post)来实现controller-action的路由, 2. Form里的每个input的name值统一, ...
- php 与 c++ openssl 加密通信
$key = '1234567890123456'; $iv = '1234567890123456'; $enc = openssl_encrypt("hello wolrd!" ...
- js 的胖箭头问题
我们在声明函数的时候通常是 var foo function(a){ console.log(a) }; 用ES6 我们写成了这样 var foo = a =>{ console.log(a); ...
- DataStage系列教程 (Slowly Changing Dimension)缓慢变化维
BI中维表的增量更新一般有2种: Type 1:覆盖更改.记录的列值发生变化,直接update成最新记录. Type 2:历史跟踪更改.记录值发生变化,将该记录置为失效,再insert一条新的记录. ...
- Spring 集成rabbiatmq
pom 文件 <dependencies> <dependency> <groupId>com.rabbitmq</groupId> <artif ...
- perl I/O和缓存的关系
最近在查看日志时,突然发现信息没有及时写入日志,研究了很久,突然醒悟:原来是print的缓存原因. 顺着这个详细了解了下perl里的IO缓存机制: 1.正常情况下,操作系统的读写都有缓存(buffer ...
- LintCode刷题指南:字符串处理(C++,Python)
题目:两个字符串是变位词 题目难度:简单 题目描述: 写出一个函数 anagram(s, t) 判断两个字符串是否可以通过改变字母的顺序变成一样的字符串. 解题思路: C++:引入哈希的思维,这道题就 ...
- VMWare虚拟机网络配置
Bridged(桥接模式) 桥接模式相当于虚拟机和主机在同一个真实网段,VMWare充当一个集线器功能(一根网线连到主机相连的路由器上),所以如果电脑换了内网,静态分配的ip要更改.图如下: NAT( ...
- 9.深入理解AbstractQueuedSynchronizer(AQS)
1. AQS简介 在上一篇文章中我们对lock和AbstractQueuedSynchronizer(AQS)有了初步的认识.在同步组件的实现中,AQS是核心部分,同步组件的实现者通过使用AQS提供的 ...