liunx驱动----构造和运行模块
以hello world模块为例
#include <linux/init.h>
#include <linux/module.h> //在执行 insmod hlello 的时候会被调用
static int hello_init(void)
{ printk(KERN_ALERT"hello_init\n");
return ;
} //在执行 rmmod hlello 的时候会被调用
static void hello_exit(void)
{
printk(KERN_ALERT"hello_exit\n");
} MODULE_LICENSE("GPL");//添加自由许可证
module_init(hello_init);
module_exit(hello_exit);
KERN_ALERT:是用来定义这条消息的优先级的。声明在:kernel.h include\linux\kernel.h中声明
#define KERN_EMERG "<0>" /* system is unusable (紧急情况,系统可能会崩溃) */
#define KERN_ALERT "<1>" /* action must be taken immediately(必须立即响应) */
#define KERN_CRIT "<2>" /* critical conditions (临界情况) */
#define KERN_ERR "<3>" /* error conditions (错误信息) */
#define KERN_WARNING "<4>" /* warning conditions (警告信息) */
#define KERN_NOTICE "<5>" /* normal but significant condition( 普通的但可能需要注意的信息) */
#define KERN_INFO "<6>" /* informational (提示性信息) */
#define KERN_DEBUG "<7>" /* debug-level messages (调试信息) */
Makfile:
KERN_DIR = /work/system/linux-2.6.22.6 //内核路径 all:
make -C $(KERN_DIR) M=`pwd` modules clean:
make -C $(KERN_DIR) M=`pwd` modules clean
rm -rf modules.order obj-m += helloworld.o //编译成helloworld.o
make输出信息:
make -C /work/system/linux-2.6.22.6 M=`pwd` modules
make[]: Entering directory `/work/system/linux-2.6.22.6'
CC [M] /home/book/work/nfs_root/first_fs/work/helloworld/helloworld.o
Building modules, stage .
MODPOST modules
CC /home/book/work/nfs_root/first_fs/work/helloworld/helloworld.mod.o
LD [M] /home/book/work/nfs_root/first_fs/work/helloworld/helloworld.ko
make[]: Leaving directory `/work/system/linux-2.6.22.6'
从输出信息中可以看出 ,最终的目的是生成了 helloworld.ko 这个文件
运行结果:
为什么需要使用printk 函数来打印了?
在模块运行过程中,不能依赖于c库,模块能够调用printk是因为在 insmod函数装入模块后,模块就链接到了内核中。所以就能访问公用的符号(printk)
为什么模块需要初始化?
模块初始化的目的就是为了以后调用模块中的函数做准备。
设备驱动程序如何引用当前进程?
在设备驱动程序中只需要包含<linux/sched.h>头文件即可引用当前进程。例如 通过访问 Staruc task_staruct 成员变量答应当前的命令名称。和进程ID
分别在装载和卸载调用中添加两条打印语句:printk(KERN_INFO"The process is \"%s\"(pid %i)\n",current->comm,current->pid)
打印出当前命令名称与进程ID
//在执行 insmod hlello 的时候会被调用
static int hello_init(void)
{ printk(KERN_ALERT"hello_init\n");
printk(KERN_INFO"The process is \"%s\"(pid %i)\n",current->comm,current->pid);
return ;
} //在执行 rmmod hlello 的时候会被调用
static void hello_exit(void)
{
printk(KERN_ALERT"hello_exit\n");
printk(KERN_INFO"The process is \"%s\"(pid %i)\n",current->comm,current->pid);
}
运行状态如下:
模块参数:
由于系统不同,所以驱动程序需要的参数也会发生变化,其中包括设备编号以及其他一些用来控制驱动程序操作方式的参数。
为了满足这个需求,内核允许对驱动程序指定参数,而这些参数可在装载驱动模块是发生改变。
在insmod改变模块参数之前,模块必须让这些参数对insmod命令可见(可操作)。必须使用module_param 宏来声明,在moduleparm.h中定义
module_param 需要三个参数: 变量名称 类型 用于sysfs入口项的访问许可掩码。
代码如下:
static char *param_string = "hello";//一个名字为param_string 的指针变量
static int print_cont =; //int 型变量 module_param(param_string, charp, S_IRUGO);//使用module_param宏来让param_string变量对insmod 命令可见
module_param(print_cont, int, S_IRUGO);//同上 //在执行 insmod hlello 的时候会被调用
static int hello_init(void)
{
int i;
printk(KERN_ALERT"hello_init\n");
printk(KERN_INFO"The process is \"%s\"(pid %i)\n",current->comm,current->pid);
for(i= ;i < print_cont; i++) //根据变量打印对应的次数
printk(KERN_ALERT"%s\n",param_string); //打印
return ;
} //在执行 rmmod hlello 的时候会被调用
static void hello_exit(void)
{
printk(KERN_ALERT"hello_exit\n");
printk(KERN_INFO"The process is \"%s\"(pid %i)\n",current->comm,current->pid);
} MODULE_LICENSE("GPL");//添加自由许可证
module_init(hello_init);
module_exit(hello_exit);
运行结果:
内核模块支持参数类型:
bool
invbool(反转其值,true变为false false 变为true)
charp 字符指针值,内核会给字符串分配内存,并相应设置指针。
int
long
short
uint
ulong
ushort
liunx驱动----构造和运行模块的更多相关文章
- linux设备驱动第二篇:构造和运行模块
上一篇介绍了Linux驱动的概念,以及linux下设备驱动的基本分类情况及其各个分类的依据和差异,这一篇我们来描述如何写一个类似hello world的简单测试驱动程序.而这个驱动的唯一功能就是输 ...
- 【Linux设备驱动程序】Chapter 2 - 构造和运行模块
Hello World 模块 #include <linux/init.h> #include <linux/module.h> MODULE_LICENSE("Du ...
- ldd3-2 构造和运行模块:环境搭建2
之前搭建了Ubuntu10.04驱动开发环境,但是那儿的内核版本是2.6.32.27,总感觉无从下手的感觉,因为书上的内核版本是2.6.10,作为初学者不知道差异在哪里,或许不应该纠结这个问题吧. 昨 ...
- 2018-05-17-OAA-一种mermaid脚本驱动的软件项目模块图形化表述思路
layout: post title: 2018-05-17-OAA-一种mermaid脚本驱动的软件项目模块图形化表述思路 key: 20180517 tags: OAA flow chart se ...
- Angular——配置模块与运行模块
配置模块 通过config方法实现对模块的配置,AngularJS中的服务大部分都对应一个“provider”,用来执行与对应服务相同的功能或对其进行配置.比如$log.$http.$location ...
- Appium学习之驱动真机运行
一.Appium工具的简单原理 Appium工具可以分为:客户端(appium-client).服务端(appium-server)和移动设备端(模拟器或者真机).客户端支持多语言,如:python- ...
- Linux驱动入门——构建和运行模块
Hello world模块 本文介绍如何向内核中添加一个hello模块.该模块的功能是在模块加载时,向系统日志输出"hello world\n" 在模块卸载时输出"Goo ...
- 【linux驱动笔记】linux模块机制浅析
1. 模块module 操作系统分微内核和宏内核,微内核优点,可以使操作系统仅作很少的事,其它事情如网络处理等都作为应用程序来实现,微内核精简的同时,必然带来性能的下降.而linux的宏内核设 ...
- liunx驱动之字符设备的注册
上一篇文章学习了如何编写linux驱动,通过能否正常加载模块进行验证是否成功,有做过liunx应用开发的小伙伴都知道驱动会在'/dev'目录下以文件的形式展现出来,所以只是能加载驱动模块不能算是完成驱 ...
随机推荐
- 2018-2019-1 20189210 《LInux内核原理与分析》第五周作业
一.系统调用实验 1.库函数将系统调用封装起来,大多数程序员使用库函数进行系统调用. 2.Linux操作系统的体系架构分为用户态和内核态.CPU的执行级别对应的就是内核态,所有指令都可以执行.用户态对 ...
- codecademy quiz——JavaScript Promise
Evernote Export What is the fulfilled value of Promise.all()? A Promise An object An array ...
- XML自学笔记
1. xml指可扩展标记语言,(extensible markup language),该语言被设计用来传输和存储数据,HTML被设计用来显示数据 XML标签没有被预定义,需要自定义标签:XML ...
- 01背包 || BZOJ 1606: [Usaco2008 Dec]Hay For Sale 购买干草 || Luogu P2925 [USACO08DEC]干草出售Hay For Sale
题面:P2925 [USACO08DEC]干草出售Hay For Sale 题解:无 代码: #include<cstdio> #include<cstring> #inclu ...
- 2018年年度总结 & 2019年计划
2018关键词 「探索」 引用以前作文最爱写的开头,时间如白驹过隙,回想上次写17年年度总结,仿佛也就过了几日光景. 首先回顾一下17年定下的目标, 18年我将关键字设为探索,目的有两个,一是 ...
- vue-cli3快速原型开发
先来讲一下,什么是快速原型开发. 当我们需要紧急或提前开发单独的一个页面时,有时候不需要在原项目中创建一个页面,再开发,我们可以单独的区开发这个项目,那么怎样单独的区开发这个项目呢,之前使用过vue- ...
- leetcode 之 two sum (easy)c++
1.数组的长度 length() .容器vector长度 size() .容器vector vector是C++标准模板库中的部分内容,它是一个多功能的,能够操作多种数据结构和算法的模板类和函数库. ...
- ui component 是一个前端 mvc 开发框架
- Module 3 - Azure - Web Apps
Module 3 - 微软云 Azure - Web Apps 1. Create new Web application in the Azure Portal Azure Portal -> ...
- JMeter接口自动化测试实例—JMeter引用javaScript
Jmeter提供了JSR223 PreProcessor前置处理器,通过该工具融合了Java 8 Nashorn 脚本引擎,可以执行js脚本以便对脚本进行前置处理.其中比较典型的应用就是通过执行js脚 ...