分享
分销 收藏 举报 申诉 / 15
播放页_导航下方通栏广告

类型linux 模块编译步骤(详解).doc

  • 上传人:xrp****65
  • 文档编号:7599588
  • 上传时间:2025-01-10
  • 格式:DOC
  • 页数:15
  • 大小:71KB
  • 下载积分:10 金币
  • 播放页_非在线预览资源立即下载上方广告
    配套讲稿:

    如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。

    特殊限制:

    部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。

    关 键  词:
    linux 模块编译步骤详解 模块 编译 步骤 详解
    资源描述:
    Linux 内核模块编程 Linux 内核模块编程是一个很重要的知识点。尤其是编写底层驱动程序时,一定会涉及到它。内核模块编程也是 Tiger哥学习 Linux 时第一节课所接触的知识。由此可以看出它的 important, 也可以看出其实它很 easy 。 一前言: 1. 什么是内核模块 1> 内核模块是具有独立功能的程序。它可以被单独编译,但是不能单独运行,它的运行必须被链接到内核作为内核的一部分在内核空间中运行。 2> 模块编程和内核版本密切相连,因为不同的内核版本中某些函数的函数名会有变化。因此模块编程也可以说是内核编程。 3> 特点: 模块本身不被编译进内核映像,从而控制了内核的大小;  模块一旦被加载,就和内核中的其他部分完全一样。 2 . 用户层编程和内核模块编程的区别   应用程序 内核模块程序 使用函数 libc 库 内核函数 运行空间 用户空间 内核空间 运行权限 普通用户 超级用户 入口函数 main() module_init 出口函数 exit() module_exit 编译 gcc makefile 链接 gcc insmod 运行 直接运行 insmod 调试 gdb kdbug 、 kdb 、 kgdb 二 . 说了这么多,那么怎么编写一个内核模块的程序呢? 1. 我们先来看两个最简单的函数实例,也是几乎所有程序员在学习一门新语言时都会编写的程序:输出 hello world! 现在我们分别用模块编程输出 hello world! ,和在用户层编程输出 hello wrold !。通过这两个程序我们来分析下如何来编写一个内核模块程序。 用户层编程: hello.c #include<stdio.h> int main(void) { printf("hello world/n"); } 内核编程 : module.c #include <linux/init.h> #include <linux/module.h> #include <linux/kernel.h> MODULE_LICENSE("Dual BSD/GPL"); static int hello_init(void) { printk(KERN_ALERT "hello,I am edsionte/n"); return 0; } static void hello_exit(void) { printk(KERN_ALERT "goodbye,kernel/n"); } module_init(hello_init); module_exit(hello_exit); // 可选  MODULE_AUTHOR("Tiger-John"); MODULE_DESCRIPTION("This is a simple example!/n"); MODULE_ALIAS("A simplest example"); Tiger-John 说明: 1.> 相信只要是学过 C 语言的同学对第一个程序都是没有问题的。但是也许大家看了第二个程序就有些不明白了。 可能有人会说: Tiger 哥你没疯吧,怎么会把 printf() 这么简单的函数错写成了 printk() 呢。 也有的人突然想起当年在大学学 C 编程时,老师告诉我们“一个 C 程序必须要有 main() 函数,并且系统会首先进入 main() 函数执行 " ,那么你的程序怎么没有 main() 函数呢?没有 main() 函数程序是怎么执行的呢? 可能也会有更仔细的人会发现:怎么两个程序头文件不一样呢?不是要用到输入和输出函数时,一定要用到 <stdio.h> 这个头文件,你怎么没有呢? -------------------------------------------------------------------------------------------- Tiger 哥很淡定的告诉大家其实第二个程序是正确的,现在我们就来看看到底如何来编写一个内核模块程序。 2. 内核模块编程的具体实现 第一步: 首先我们来看一下程序的头文件 #include<linux/kernel.h> #include<linux/module.h> #include<linux/init.h> 这三个头文件是编写内核模块程序所必须的 3 个头文件 。 Tiger-John 说明: 1> 由于内核编程和用户层编程所用的库函数不一样,所以它的头文件也和我们在用户层编写程序时所用的头文件也不一样。 2> 我们在来看看在 L inux 中又是在那块存放它们的头文件 a. 内核头文件的位置 : /usr/src/linux-2.6.x/include/ b. 用户层头文件的位置 : /usr/include/ 现在我们就明白了。其实我们在编写内核模块程序时所用的头文件和系统函数都和用层 编程时所用的头文件和系统函数是 不同的。 第二步: 编写内核模块时必须要有的两个函数 : 1> 加载 函数: static int init_fun(void) { // 初始化代码 }   函数实例: static int hello_init(void)// 不加 void 在调试时会出现报警 { printk("hello world!/n"); return 0; } 2> 卸载函数 无返回值 static void cleaup_fun(void) { // 释放代码 } 函数实例:   static void hello_exit(void)// 不加 void 会出现报警 , 若改为 static int 也会报错 , 因为出口函数是不能返会值的 { printk("bye,bye/n"); } 在模块编程中必须要有上面这两个函数; Tiger-John 补充: 注册函数和卸载函数还有另一中写法: 1> 模块加载 函数 static int __init init_fun(void) { // 初始化代码 } 函数实例: static int __init hello_init(void) { printk("hello tiger/n"); return 0; } 2> 卸载函数 无返回值 static void __exit cleaup_fun(void) { // 释放代码 } 函数实例: static void __exit exit(void) { printk("bye bye!/n"); } Tiger-John 补充: 通过比较我们可以发现第二中函数的写法与第一中函数的写法主要不同就是加了 __init 和 __exit 前缀。 (init 和 exit 前面都是两个下划线 ) 那么第二种方法比第一种有什么好处呢: _init 和 __exit 是 Linux 内核的一个宏定义,使系统在初始化完成后释放该函数,并释放其所占内存。因此它的优点是显而易见的。所以建议大家啊在编写入口函数和出口函数时采用第二中方法。 (1) 在 linux 内核中,所有标示为 __init 的函数在连接的时候都放在 .init.text 这个区段内,此外,所有的 __init 函数在区段 .initcall.init 中还保存了一份函数指针,在初始化时内核会通过这些函数指针调用这些 __init 函数,并在初始化完成后释放 init 区段(包括.init.text,.initcall.init 等)。 (2) 和 __init 一样, __exit 也可以使对应函数在运行完成后自动回收内存。   3 > 现在我们来看一下 printk() 函数 a. 上面已经说了,我们在内核 编程时所用的库函数和在用户态下的是不一样的。 printk 是内核态信息打印函数,功能和比准  C  库的printf  类似。  printk  还有信息打印级别。 b. 现在我们来看一下 printk() 函数的原型: int printk(const char *fmt, ...) 消息打印级别: fmt---- 消息级别: #define KERN_EMERG "<0>" /* 紧急事件消息,系统崩溃之前提示,表示系统不可用 */ #define KERN_ALERT "<1>" /* 报告消息,表示必须立即采取措施 */ #define KERN_CRIT "<2>" /* 临界条件,通常涉及严重的硬件或软件操作失败 */ #define KERN_ERR "<3>" /* 错误条件,驱动程序常用   KERN_ERR 来报告硬件的错误 */ #define KERN_WARNING "<4>" /* 警告条件,对可能出现问题的情况进行警告 */ #define KERN_NOTICE "<5>" /* 正常但又重要的条件,用于提醒。常用于与安全相关的消息 */ #define KERN_INFO "<6>" /* 提示信息,如驱动程序启动时,打印硬件信息   */ #define KERN_DEBUG "<7>" /* 调试级别的消息 */ Tiger-John 说明: 不同级别使用不同字符串表示,数字越小,级别越高。 c. 为什么内核态使用 printk() 函数,而在用户态使用 printf() 函数。 printk() 函数是直接使用了向终端写函数 tty_write() 。而 printf() 函数是调用 write() 系统调用函数向标准输出设备写。所以在用户态(如进程 0 )不能够直接使用 printk() 函数,而在内核态由于它已是特权级,所以无需系统调用来改变特权级,因而能够直接使用 printk() 函数。 printk 是内核输出,在终端是看不见的。我们可以看一下系统日志。 但是我们可以使用命令: cat /var/log/messages ,或者使用 dmesg 命令看一下输出的信息。 第三步: 加载模块和卸载模块 1>module_init(hello_init) a. 告诉内核你编写模块程序从那里开始执行。 b.module_init() 函数中的参数就是注册函数的函数名。 2>module_exit(hello_exit) a. 告诉内核你编写模块程序从那里离开。 b.module_exit() 中的参数名就是卸载函数的函数名。 Tiger-John 说明: 我们一般在注册函数里进行一些初始化比如申请内存空间注册设备号等 。那么我们就要在卸载函数进行释放我们所占有的资源。 (1) 若模块加载函数注册了 XXX, 则模块卸载函数应该注销 XXX (2) 若模块加载函数动态申请了内存,则模块卸载函数应该注销 XXX (3) 若模块加载函数申请了硬件资源(中断, DMA 通道)的占用,则模块卸载函数应该释放这些硬件资源。 (4) 若模块加载函数开启了硬件,则卸载函数中一般要关闭硬件。 第四步 : 许可权限的声明 1> 函数实例: MODULE_LICENSE("Dual BSD/GPL") ; 2> 此处可有可无,可以不加系统默认 ( 但是会报警   ) 模块声明描述内核模块的许可权限,如果不声明 LICENSE ,模块被加载时,将收到内核的警告。 在 Linux2.6 内核中,可接受的 LICENSE 包括" GPL","GPL v2","GPL and additional rights","Dual BSD/GPL","Dual MPL/GPL","Proprietary" 。 第五部:模块的声明与描述(可加可不加) MODULE_AUTHOR(“author”);// 作者 MODULE_DESCRIPTION(“description”);// 描述 MODULE_VERSION(”version_string“);// 版本 MODULE_DEVICE_TABLE(“table_info”);// 设备表 对于 USB , PCI 等设备驱动,通常会创建一个 MODULE_DEVICE_TABLE MODULE_ALIAS(”alternate_name“);// 别名 Tiger-John: 总结 经过以上五步(其实只要前四步)一个完整的模块编程就完成了。 第六步 : 常用的模块编程命令: 1> 在 Linux 系统中,使用 lsmod 命令可以获得系统中加载了的所有模块以及模块间的依赖关系 2> 也可以用 cat /proc/modules 来查看加载模块信息 3> 内核中已加载模块的信息也存在于 /sys/module 目录下,加载 hello.ko 后,内核中将包含 /sys/module/hello 目录,该目录下又包含一个 refcnt 文件和一个 sections 目录,在 /sys/module/hello 目录下运行 tree -a 可以看到他们之间的关系。 4> 使用 modinfo < 模块名 > 命令可以获得模块的信息,包括模块的作者,模块的说明,某块所支持的参数以及 vermagic. 但是,前面我们已经说过了。内核编程和用户层编程它们之间的编译 链接也不相同。那么我们 如何对模块程序进行编译,链接,运行呢? 现在我么继续深入来学习 Makefile 文件的编写: 三 .make 的使用以及 Makefile 的编写 1. 什么是 Makefile , make 1>Makefile 是一种脚本,这种脚本主要是用于多文件的编译 2> make 程序可以维护具有相互依赖性的源文件,但某些文件发生改变时,它能自动识别出, 并只对相应 文件进行自动编译 2.Makefile 的写法 Makefile 文件由五部分组成:显示规则 含规则 变量定义 makefile 指示符和注释 一条 Make 的规则原型为: 目标 ... :依赖 .. 命令 ... … makefile 中可以使用 Shell 命令,例如 pwd , uname 简单的 makefile 文件: obj-m := hello.o kernel_path=/usr/src/linux-headers-$(shell uname -r) all: make -C $(kernel_path) M=$(PWD) modules clean: make -C $(kernel_path) M=$(PWD) clean obj -m:= hello.o // 产生 hello 模块的目标 kernel_path // 定义内核源文件目录 all : make -C $(kernel_path) M=$(PWD) modules // 生成内核模块参数为内核源代码目录以及模块所在目录 clean: make -C $(kernel_path) M=$(PWD) clean // 清除生成的模块文件以及中间文件 Tiger-John 说明: 1> 在 all 和 clean 下面的一行,即 make 之前必须用 Table 符隔开,不能用空 格隔开,否则编译错误 。 2> 其中 -C 后指定的是 Linux 内核源代码的目录,而 M= 后指定的是 hello.c 和 Makefile 所在的目录 3.Makefile 实例:   1 obj-m:=module.o 2 3 4 CURRENT_PATH :=$(shell pwd) 5 VERSION_NUM :=$(shell uname -r) 6 LINUX_PATH :=/usr/src/linux-headers-$(VERSION_NUM) 7 8 all : 9 make -C $(LINUX_PATH) M=$(CURRENT_PATH) modules 10 clean : 11 make -C $(LINUX_PATH) M=$(CURRENT_PATH) clean   ---------------------------------------------------------------------- 经过上面的模块编程和 Makefile 的编程,我们就可以对我们的程序进行编译链接和运行了 四 . 内核模块的操作过程 1> 在控制台输入 make 进行编译链接 2> 正确后在控制台输入 sudo insmod module.ko (加载模块) 3> 在控制台输入 dmesg 查看结果 4> 在控制台输入 rmmod tiger( 卸载模块 ) 5> 输入 dmesg 查看结果 ( 或者用 cat /var/log/messages 查看系统日志文件) 6>make clean( 去除中间生成的文件) ---------------------------------------------------------------------- 现在我们就总体来实践一下 , 来体验一下。编写内核模块程序的乐趣   module.c   1 #include<linux/kernel.h> 2 #include<linux/init.h> 3 #include<linux/module.h> 4 MODULE_LICENSE("Dual BSD/GPL"); 5 6 static int __init hello_init(void) 7 { 8 printk("Hello world/n"); 9 return 0; 10 } 11 12 static void __exit hello_exit(void) 13 { 14 printk("Bye Corne/n"); 15 16 } 17 module_init(hello_init); 18 module_exit(hello_exit);   Makefile     1 obj-m:=module.o 2 3 4 CURRENT_PATH :=$(shell pwd) 5 VERSION_NUM :=$(shell uname -r) 6 LINUX_PATH :=/usr/src/linux-headers-$(VERSION_NUM) 7 8 all : 9 make -C $(LINUX_PATH) M=$(CURRENT_PATH) modules 10 clean : 11 make -C $(LINUX_PATH) M=$(CURRENT_PATH) clean 在终端输入 make think@ubuntu:~/work/moudule/mokua_biancheng$  make   make -C /usr/src/linux-headers-2.6.32-25-generic M=/home/think/work/moudule/mokua_biancheng modules make[1]: 正在进入目录 `/usr/src/linux-headers-2.6.32-25-generic'   Building modules, stage 2.   MODPOST 1 modules make[1]: 正在离开目录 `/usr/src/linux-headers-2.6.32-25-generic' think@ubuntu:~/work/moudule/mokua_biancheng$ think@ubuntu:~/work/moudule/mokua_biancheng$ sudo insmod module.ko think@ubuntu:~/work/moudule/mokua_biancheng$ dmesg [19011.002597] Hello world    Tiger-John :当程序没有错误时,当我们输入 dmesg 时就可以看到程序运行的结果了。  
    展开阅读全文
    提示  咨信网温馨提示:
    1、咨信平台为文档C2C交易模式,即用户上传的文档直接被用户下载,收益归上传人(含作者)所有;本站仅是提供信息存储空间和展示预览,仅对用户上传内容的表现方式做保护处理,对上载内容不做任何修改或编辑。所展示的作品文档包括内容和图片全部来源于网络用户和作者上传投稿,我们不确定上传用户享有完全著作权,根据《信息网络传播权保护条例》,如果侵犯了您的版权、权益或隐私,请联系我们,核实后会尽快下架及时删除,并可随时和客服了解处理情况,尊重保护知识产权我们共同努力。
    2、文档的总页数、文档格式和文档大小以系统显示为准(内容中显示的页数不一定正确),网站客服只以系统显示的页数、文件格式、文档大小作为仲裁依据,个别因单元格分列造成显示页码不一将协商解决,平台无法对文档的真实性、完整性、权威性、准确性、专业性及其观点立场做任何保证或承诺,下载前须认真查看,确认无误后再购买,务必慎重购买;若有违法违纪将进行移交司法处理,若涉侵权平台将进行基本处罚并下架。
    3、本站所有内容均由用户上传,付费前请自行鉴别,如您付费,意味着您已接受本站规则且自行承担风险,本站不进行额外附加服务,虚拟产品一经售出概不退款(未进行购买下载可退充值款),文档一经付费(服务费)、不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。
    4、如你看到网页展示的文档有www.zixin.com.cn水印,是因预览和防盗链等技术需要对页面进行转换压缩成图而已,我们并不对上传的文档进行任何编辑或修改,文档下载后都不会有水印标识(原文档上传前个别存留的除外),下载后原文更清晰;试题试卷类文档,如果标题没有明确说明有答案则都视为没有答案,请知晓;PPT和DOC文档可被视为“模板”,允许上传人保留章节、目录结构的情况下删减部份的内容;PDF文档不管是原文档转换或图片扫描而得,本站不作要求视为允许,下载前可先查看【教您几个在下载文档中可以更好的避免被坑】。
    5、本文档所展示的图片、画像、字体、音乐的版权可能需版权方额外授权,请谨慎使用;网站提供的党政主题相关内容(国旗、国徽、党徽--等)目的在于配合国家政策宣传,仅限个人学习分享使用,禁止用于任何广告和商用目的。
    6、文档遇到问题,请及时联系平台进行协调解决,联系【微信客服】、【QQ客服】,若有其他问题请点击或扫码反馈【服务填表】;文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“【版权申诉】”,意见反馈和侵权处理邮箱:1219186828@qq.com;也可以拔打客服电话:0574-28810668;投诉电话:18658249818。

    开通VIP折扣优惠下载文档

    自信AI创作助手
    关于本文
    本文标题:linux 模块编译步骤(详解).doc
    链接地址:https://www.zixin.com.cn/doc/7599588.html
    页脚通栏广告

    Copyright ©2010-2026   All Rights Reserved  宁波自信网络信息技术有限公司 版权所有   |  客服电话:0574-28810668    微信客服:咨信网客服    投诉电话:18658249818   

    违法和不良信息举报邮箱:help@zixin.com.cn    文档合作和网站合作邮箱:fuwu@zixin.com.cn    意见反馈和侵权处理邮箱:1219186828@qq.com   | 证照中心

    12321jubao.png12321网络举报中心 电话:010-12321  jubao.png中国互联网举报中心 电话:12377   gongan.png浙公网安备33021202000488号  icp.png浙ICP备2021020529号-1 浙B2-20240490   


    关注我们 :微信公众号  抖音  微博  LOFTER               

    自信网络  |  ZixinNetwork