一:linux内核裁剪:

  1:编译内核源码:

    今天的重点内容是内核驱动的编写,在编写驱动之前首先的了解linux内核源码,linux主要是由五个子系统组成:进程调度,内存管理,文件系统,网络接口以及进程间通信;下面是解压的linux内核源码文件:

  下面对linux内核里面的文件进行简单的说明:

  arch目录中包含于体系结构有关的子目录和文件,arm的相关平台信息在arch/arm目录下。

  scripts目录中存放着对核心配置的脚本文件。

  crypto目录中包含着常见的加密算法。

  drivers目录包含各种各样的驱动,包括字符型,快型,网络设备驱动程序。

  fs目录中包括了linux系统所支持的文件系统类型,比如ext4 nfs

  init目录中存放着与linux内核相关的启动代码。

  kernel目录中包含许多linux进程调度子系统相关的源代码。

  lib目录存放linux内核所用的库文件。

  mm目录存放linux内存管理的源代码。

  net目录存放有关网络协议的源代码。

  编译linux内核源码,最重要的是Makefile,在linux内核中每个文件都有一个Makefile,统一由最外层的Makefile来调用:Makefile这个文件中包含了许多linux内核配置的信息,我们Linux内核要编译的平台,交叉编译器的选择(如下图),编译链接Linux内核的参数;

  由于linux源代码非常庞大,我们找起来非常困难;在编译linux之前先介绍一个管理工具ctags

  在linux源码中我们使用命令[root@192 linux-3.5]# ctags -R ./

  等它执行完毕(要几分钟),我们可以用vim任意打开一个文件,在命令模式下输入:ta <需要找的函数名>回车就可以了 

   

  2:linux源码编译

  下面是linux源码编译步骤:

  (1)在编译之前首先使用make clean, make distclean清除原编译和配置文件

  (2)找到运行平台(arm)下面的配置文件拷贝到源码目录一命名为  .config,.config文件是在进行内核配置的时候,经过配置后生成的内核编译参考文件,命令如下

   [root@192 linux-3.5]# cp arch/arm/configs/exynos4_defconfig   .config

  (3)拷贝完成之后就可以运行make命令,这个过程要等很久:

   [root@192 linux-3.5]#make

  (4)编译完成之后就会在arch/armboot目录下生成zImage,生成的这个文件就是linux镜像

   [root@192 linux-3.5]# arch/arm/boot/zImage
   之后我们就可以使用linux的镜像文件了

  (2):linux内核裁剪

   以上编译的是没有裁剪过linux源码,如果我们不想使用源码里面的部分功能,我们就可以通过裁剪的方式阻止它编译:在linux源码包中我们可以使用make menuconfig,图形化界面很容易操作;就可以进行linux的源码的裁剪:

  (1)首先在源码包中运行make menuconfig这个命令

  如果不想使用它的某些功能,取消前面*即可,按空格键来控制选择

  (2)保存退出,执行make

  二:编译可加载模块

  内核模块是Linux内核向外部提供的一个插口,其全称为动态可加载内核模块(Loadable Kernel Module,LKM),我们简称为模块,模块是具有独立功能的程序,它可以被单独编译,但不能独立运行。它在运行时被链接到内核作为内核的一部分在内核空间运行,这与运行在用户空间的进程是不同的。模块通常由一组函数和数据结构组成,用来实现一种文件系统、一个驱动程序或其他内核上层的功能。

  总之,模块是一个为内核(从某种意义上来说,内核也是一个模块)或其他内核模块提供使用功能的代码块。

  Linux内核模块是一种可被动态加载和卸载的可执行程序。通过内核模块可以扩展内核功能,内核模块通常用于设备驱动、文件系统等。如果没有内核模块,需要向内核添加功能就需要自发代码、重新编译内核、安装新内核等步骤;

  下面我们实现一个简单的linux模块:

 #include <linux/init.h>
#include <linux/module.h> MODULE_LICENSE("GPL");//指定GPL协议
MODULE_AUTHOR("BUNFLY"); int bunfly_init()
{
printk("this is bunfly_init\n"); return ;
} void bunfly_exit()
{
printk("this is bunfly_exit\n");
} module_init(bunfly_init);
module_exit(bunfly_exit);

  需要注意的是:编写模块函数时,声明模块的授权协议,如果没有的话,编译器有警告的,如果在模块函数中调用的设备驱动模型的代码,就必须要指定为GPL协议,否则是不能加载到内核中,在linux内核中输出函数不再是printf(), 而是用的是printk();printk相当于printf的孪生姐妹,她们一个运行在用户态,另一个则在内核态。

  模块是动态加载内核中,是内核的一部分,它没有main函数,因此在编译的时候我们要需调用内核中的Makefile来编译模块;下面是Makefile文件内容:

 all:
make -C /home/bunfly/source_code/linux-3.5 M=`pwd` clean:
make -C /home/bunfly/source_code/linux-3.5 M=`pwd` clean obj-m += bunfly.o

在编辑Makefile的时候我们需要注意的是代码第8行:它分为三种模式:

(1)obj-不编入内核

(2)obj -y编入内核

(3)obj-m编译成模块

  如果要将自己的模块写到make menuconfig菜单中去,就要用到另一个配置文件Kconfig,在Linux 内核源码树的每个目录下都有两个文档Kconfig和Makefile。分布到各目录的Kconfig构成了一个分布式的内核配置数据库,每个 Kconfig分别描述了所属目录源文档相关的内核配置菜单。在执行内核配置make menuconfig时,从Kconfig中读出菜单,用户选择后保存到.config的内核配置文档中。在内核编译时,主Makefile调用这 个.config,就知道了用户的选择。这个内容说明了,Kconfig就是对应着内核的每级配置菜单。

linux内核裁剪及编译可加载模块的更多相关文章

  1. linux内核启动以及文件系统的加载过程

    Linux 内核启动及文件系统加载过程 当u-boot 开始执行 bootcmd 命令,就进入 Linux 内核启动阶段.普通 Linux 内核的启动过程也可以分为两个阶段.本文以项目中使用的 lin ...

  2. Linux内核和根文件系统引导加载程序

    续博文<u-boot之u-boot-2009.11启动过程分析> Linux内核启动及文件系统载入过程 当u-boot開始运行bootcmd命令.就进入Linux内核启动阶段,与u-boo ...

  3. Linux驱动之内核加载模块过程分析

    Linux内核支持动态的加载模块运行:比如insmod first_drv.ko,这样就可以将模块加载到内核所在空间供应用程序调用.现在简单描述下insmod first_drv.ko的过程 1.in ...

  4. [driver]linux内核动态加载模块

    问题: 1. 把编译好的模块放到板子/lib/modules对应文件夹下,并且执行了depmod -a, 比如pl2303.ko, 那么下一次插入pl2303的串口线,是否可以识别,也就是自动加载pl ...

  5. 嵌入式 Linux 与linux启动时自动加载模块

    一.在ARM linux 下,一般而言,产品在启动的过程中应该加载模块,最简单的方法是修改启动过程的rc脚本(/etc/init.d/rcS),增加ismod /../xxx.ko这个命令.例如:加载 ...

  6. Linux中实现在系统启动时自动加载模块

    下面是以前学习Linux时写的,后来仔细研究rc.sysinit后发现,只需要修改下列地方就可以了,不必这么麻烦的: rc.sysinit中有这样的一段代码: # Load other user-de ...

  7. Linux内核裁剪的具体步骤

    在menuconfig中配置: 详细介绍内核配置选项及删改情况 第一部分:全部删除 Code maturity level options ---> 代码成熟等级选项 []Prompt for  ...

  8. Linux下c函数dlopen实现加载动态库so文件代码举例

    dlopen()是一个强大的库函数.该函数将打开一个新库,并把它装入内存.该函数主要用来加载库中的符号,这些符号在编译的时候是不知道的.这种机制使得在系统中添加或者删除一个模块时,都不需要重新编译了. ...

  9. Java类编译、加载、和执行机制

    Java类编译.加载.和执行机制 标签: java 类加载 类编译 类执行 机制 0.前言 个人认为,对于JVM的理解,主要是两大方面内容: Java类的编译.加载和执行. JVM的内存管理和垃圾回收 ...

随机推荐

  1. hadoop学习大纲

  2. 证书,CSP与Openssl

    证书,CSP与Openssl 起因 最近在研究更安全的交互体系,自然想到的就是提供证书的交互方式.给用户分配一对公私钥,然后将私钥交给用户保管,用户在登录或者一些关键操作的时候通过私钥签名,从而保证其 ...

  3. 如何用代码禁用SpriteBuilder中创建的关节

    这个目标是临时的禁用距离关节(distance joint). 不幸的是,你只可以无效化(通过删除的方式)一个关节. 所以,你必须通过代码创建一个新的距离关节实例并且赋予它之前删除关节(在Sprite ...

  4. 三消游戏FSM状态机设计图

    三消游戏FSM状态机设计图 1) 设计FSM图 2) smc配置文件 ///////////////////////////////////////////////////////////////// ...

  5. 中文注释 MariaDB my.cnf 大型服务器配置模板

    文件如下所示,请根据需要进行修改: 翻译日期: 2014年5月22日 翻译人员: 铁锚 # /usr/share/mysql/my-huge.cnf # MariaDB 配置文件 示例模板(huge, ...

  6. InvocationTargetException异常解析

    InvocationTargetException异常由Method.invoke(obj, args...)方法抛出.) { throw new ZeroException("参数不能小于 ...

  7. Hadoop Bloom Filter 使用

    1.Bloom Filter  默认的 BloomFilter filter =new BloomFilter(10,2,1); // 过滤器长度为10 ,用2哈希函数,MURMUR_HASH (1) ...

  8. 配置使用dwr完成收邮件提示

    DWR(Direct Web Remoting)是一个用于改善web页面与Java类交互的远程服务器端Ajax开源框架,可以帮助开发人员开发包含AJAX技术的网站.它可以允许在浏览器里的代码使用运行在 ...

  9. Dubbo性能调优参数及原理

    本文是针对 Dubbo 协议调用的调优指导,详细说明常用调优参数的作用域及源码. Dubbo调用模型 常用性能调优参数 参数名 作用范围 默认值 说明 备注 threads provider 200 ...

  10. jQuery插件之-----弹性运动

    <!doctype html><html><head><meta charset="utf-8"><title>弹性运动 ...