一个内核模块不是一个可以独立执行的文件,而是需要在运行时刻连接入内核的目标文件。所以,它们需要用-c 选项进行编译。而且,所有的内核模块都必须包含特定的标志:

  • __KERNEL__——这个标志告诉头文件此代码将在内核模块中运行,而不是作为用户进程。
  • MODULE——这个标志告诉头文件要给出适当的内核模块的定义。
  • LINUX——从技术上讲,这个标志不是必要的。但是,如果你希望写一个比较正规的内核模块,在多个操作系统上编译,这个标志将会使你感到方便。它可以允许你在独立于操作系统的部分进行常规的编译。
  • 还有其它的一些可被选择包含标志,取决于编译模块是的选项。如果你不能明确内核怎样被编译,可以在in/usr/include/linux/config.h 中查到。
  • __SMP__——对称多线程。在内核被编译成支持对称多线程(尽管在一台处理机上运行)是必须定义。如果是这样,还需要做一些别的事情(参见第12 章)。
  • CONFIG_MODVERSIONS——如果CONFIG_MODVERSIONS 被激活,你需要在编译是定义它并且包含文件/usr/include/linux/modversions.h。这可以有代码自动完成。
# Makefile for a basic kernel module
CC=gcc
MODCFLAGS := -Wall -DMODULE -D__KERNEL__ -DLINUX
hello.o: hello.c /usr/include/linux/version.h$(CC) $(MODCFLAGS) -c hello.cecho insmod hello.o to turn it onecho rmmod hello to turn if offechoecho X and kernel programming do not mix.echo Do the insmod and rmmod from outside

所以,并不是剩下的事情就是root(你没有把它编译成root,而是在边缘(注1.1)。对吗?),然后就在你的核心内容里插入或移出hello。当你这样做的时候,要注意到你的新模块在/proc/modules 里。

而且,编译文件不推荐从X 下插入的原因是内核有一条需要用printk 打印的消息,它把它送给了控制台。如果你不使用X,它就送到了你使用的虚拟终端(你用Alt-F<n>选择的哪个)并且你可以看到。相反的,如果你使用了X,就有两种可能性。如果用xterm –C 打开了一个控制台,输出将被送到哪里。如果没有,输出将被送到虚拟终端7——被X“覆盖”的那个。

如果你的内核变得不稳定,你可以在没有X 的情况下得到调试消息。在X 外,printk可以直接从内核中输出到控制台。而如果在X 里,printk 输出到一个用户态的进程(xterm–C)。当进程接收到CPU 时间,它会将其送到X 服务器进程。然后,当X 服务器进程接收到CPU 时间,它将会显示,但是一个不稳定的内核意味着系统将会崩溃或重起,所以你不希望显示错误的消息,然后可能被解释给你什么发生了错误,但是超出了正确的时间。

多文件内核模块

有些时候在几个源文件之间分出一个内核模块是很有意义的。在这种情况下,你需要做下面的事情:

  • 1. 在除了一个以外的所有源文件中,增加一行#define __NO_VERSION__。这是很重要的,因为module.h 一般包括kernel_version 的定义,这是一个全局变量,包含模块编译的内核版本。如果你需要version.h,你需要把自己把它包含进去,因为如果有__NO_VERSION__的话module.h 不会自动包含。
  • 2. 象通常一样编译源文件。
  • 3. 把所有目标文件联编成一个。在X86 下,用ld –m elf_i386 –r –o <name of module>.o<1st source file>

这里给出一个这样的内核模块的例子。

ex start.c

/* start.c
* Copyright (C) 1999 by Ori Pomerantz
*
* "Hello, world" - the kernel module version.
* This file includes just the start routine
*/
/* The necessary header files */
/* Standard in kernel modules */
#include <linux/kernel.h> /* We're doing kernel work */
#include <linux/module.h> /* Specifically, a module *//* Deal with CONFIG_MODVERSIONS */
#if CONFIG_MODVERSIONS==1
#define MODVERSIONS
#include <linux/modversions.h>
#endif/* Initialize the module */
int init_module()
{printk("Hello, world - this is the kernel speaking\n");/* If we return a non zero value, it means that* init_module failed and the kernel module* can't be loaded */return 0;
}

ex stop.c

/* stop.c
* Copyright (C) 1999 by Ori Pomerantz
*
* "Hello, world" - the kernel module version. This
* file includes just the stop routine.
*/
/* The necessary header files */
/* Standard in kernel modules */
#include <linux/kernel.h> /* We're doing kernel work */
#define __NO_VERSION__ /* This isn't "the" file of the kernel module *//* Specifically, a module */#include <linux/module.h>
/* Not included by module.h because of the __NO_VERSION__ */
#include <linux/version.h> /* Deal with CONFIG_MODVERSIONS */
#if CONFIG_MODVERSIONS==1
#define MODVERSIONS
#include <linux/modversions.h>
#endif/* Cleanup - undid whatever init_module did */
void cleanup_module()
{printk("Short is the life of a kernel module\n");
}

ex Makefile

# Makefile for a multifile kernel module
CC=gcc
MODCFLAGS := -Wall -DMODULE -D__KERNEL__ -DLINUXhello.o: start.o stop.old -m elf_i386 -r -o hello.o start.o stop.o
start.o: start.c /usr/include/linux/version.h$(CC) $(MODCFLAGS) -c start.c
stop.o: stop.c /usr/include/linux/version.h$(CC) $(MODCFLAGS) -c stop.c

linux内核模块的编译文件相关推荐

  1. linux 内核模块(驱动) 编译详解

    一.准备工作 准备工作如何做,这里就不详说了. a) 首先,你要有一台PC,装好了Linux. b) 安装好GCC(这个指的是host gcc,用于编译生成运行于pc机程序的).make.ncurse ...

  2. linux下g编译文件或目录,【转】在linux下使用gcc/g++编译多个.h文件

    博主写得很好 多个文件编译在linux下编译,下面有三个文件,分别是1.cpp 和 2.cpp 和myhead.h 文件. 1.cpp 2.cpp myhead.h 假如他们都在一个目录下面,那么编译 ...

  3. linux的as编译文件,Ubuntu Linux14 64位下在Android studio下用gradle编译Andrid项

    我在Ubuntu14 64为下安装了AS,但在用Gradle编译项目时总是报找不到 libz.so.1的错误. error while loading shared librarieserror wh ...

  4. 卸载自己编译linux的内核,Linux内核模块的编译、加载和卸载

    转载请注明出处:http://blog.csdn.net/zhangyang0402/archive/2010/07/04/5711502.aspx 一.hello.c #include#includ ...

  5. Linux内核模块反编译,内核oops错误调试学习笔记

    驱动开发中遇到的 oops 问题,导致内核崩溃,log 一般如下形式 Unable to handle kernel paging request at virtual address bfb10be ...

  6. LINUX内核模块上下文,飞凌嵌入式知识分享-Linux内核模块编译

    本文主要讲解什么是Linux内核,以及通过多张图片展示Linux内核的作用.功能及基本编程方法,以便于读者能快速理解什么是Linux内核,能看懂Linux内核. 拥有超过1300万行的代码,Linux ...

  7. Linux内核模块的概念和基本的编程方法

    Linux内核模块的概念和基本的编程方法 标签: Linux内核模块 2013-06-14 18:29 1864人阅读 评论(0) 收藏 举报 分类: linux内核(34) 版权声明:本文为博主原创 ...

  8. c需要实现安装卸载Linux模块,Linux内核模块编译与加载

    Linux内核的整体结构非常庞大,其包含的组件也非常多,如何使用所需要的组件? 方法一: 把所有的组件都编译进内核文件,即Zlmage或bzlmage,但会导致两个问题:一是生成的内核文件过大:二是如 ...

  9. c 远程编辑linux文件,makefile - 在远程Linux机器上编译C ++ - “检测到时钟偏差”警告...

    makefile - 在远程Linux机器上编译C ++ - "检测到时钟偏差"警告 我通过PuTTY和WinSCP连接到我大学的小型Linux集群,使用后者传输文件,并使用前者编 ...

最新文章

  1. Reactjs-JQuery-Omi-Extjs-Angularjs对比
  2. 第五周项目二-游戏中的角色类(2)
  3. mysql_connect 废弃_解决Deprecated: mysql_connect():
  4. Redis分布式锁,看完不懂你打我
  5. plsql(轻量版)_游标的使用1
  6. bootstrap项目实例_101个Python项目打包放送,工作学习必备(源码放送)
  7. verilog学习记(tinyriscv mcu设计)
  8. Atitit.收银系统pos 以及打印功能的行业标准
  9. 第三章 python流程控制
  10. Java+Swing+mysql实现学生选课管理系统
  11. 电脑录屏软件哪个好?高清流畅的录屏方法在这里!
  12. JavaCV的摄像头实战之五:推流
  13. error LNK1123: failure during conversion to COFF: file invalid or corrupt
  14. 用java编码图书查询系统_用java做的一个图书管理系统的源代码
  15. Y z推荐菜东家 易订货生鲜系统_新零售·新生鲜——易订货生鲜专版客户交流会(贵阳站)圆满结束!...
  16. python服务器性能测试工具locust使用指南
  17. matlab图像对折,Matlab下如何将一个索引图像进行对折小程序--原创
  18. 操作系统ahci aci ide模式
  19. app抓包分析sign
  20. 免费手机网站自助建站平台推荐

热门文章

  1. Spring框架----Spring的bean之三种创建bean对象的方式
  2. MNIST机器学习入门(二)
  3. java实现文件上传(使用FromData)
  4. oracle锁表与解表
  5. 微信公众号开发--.Net Core实现微信消息加解密
  6. OpenCV图像处理篇之边缘检測算子
  7. !!超级筹码理论总结
  8. 《learn objective-c on the Mac for OS X and IOS》译名《objective-C基础教程》
  9. 可编程模拟IC之考虑[zz]
  10. 演示JSP注释的使用