Linux内核编译与开发

简介:

一.Linux内核简介

linux kernel map:

linux kernel map

linux 系统体系结构:

linux 系统体系结构

linux kernel体系结构:

arm有7种工作模式,x86也实现了4个不同级别RING0-RING3,RING0级别最高,

这样linux用户代码运行在RING3下,内核运行在RING0,这样系统本身就得到了

充分的保护

用户空间(用户模式)转到内核空间(系统模式)方法:

  • 系统调用
  • 硬件中断

linux kernel 体系结构:

linux kernel 体系结构

虚拟文件系统VFS:

VFS(虚拟文件系统)隐藏各种文件系统的具体细节,为文件操作提供统一的接口

二.Linux内核源代码

linux内核下载www.kernel.org

目录结构:

解压linux kernel tar后目录

  • arch:根据cpu体系结构不同而分的代码
  • block:部分块设备驱动程序
  • crypto:加密,压缩,CRC校验算法
  • documentation:内核文档
  • drivers:设备驱动程序
  • fs(虚拟文件系统vfs):文件系统
  • include:内核所需的头文件,(与平台无关的头文件在include/linux中)
  • lib:库文件代码(与平台相关的)
  • mm:实现内存管理,与硬件体系结构无关的(与硬件体系结构相关的在arch中)
  • net:网络协议的代码
  • samples:一些内核编程的范例
  • scripts:配置内核的脚本
  • security:SElinux的模块
  • sound:音频设备的驱动程序
  • usr:cpio命令实现,用于制作根文件系统的命令(文件系统与内核放到一块的命令)
  • virt:内核虚拟机

linux DOC 编译生成:

linux源根目录/Documentation/00-INDEX:目录索引

linux源根目录/Documentation/HOWTO:指南

生成linux内核帮助文档:在linux源根目录(Documentation) 执行make htmldocs

ubuntu16下需要执行sudo apt-get install xmlto安装插件才可生成doc文档

后面开发中经常要改的是arch,drivers中的代码

三.Linux内核配置与编译

清理文件(在linux源码根目录):

  • make clean:只清理所有产生的文件
  • make mrproper:清理所有产生的文件与config配置文件
  • make distclean:清理所有产生的文件与config配置文件,并且编辑过的与补丁文件

配置(收集硬件信息如cpu型号,网卡等...):

  • make config:基于文本模式的交互配置
  • make menuconfig:基于文本模式的菜单模式(推荐使用)
  • make oldconfig:使用已有的.config,但会询问新增的配置项
  • make xconfig:图形化的配置(需要安装图形化系统)

配置方法:

1)使用make menuconfig操作方法:

1>按y:编译>连接>镜像文件

2>按m:编译

3>按n:什么都不做

4>按"空格键":y,n轮换

配置完并保存后会在linux源码根目录下生成一个.config文件

注意:在ubuntu11上要执行apt-get install libncurses5-dev来安装支持包

2)利用已有的配置文件模板(.config)

1>linux源码根目录/arch/<cpu架构>/configs/<具体某一的CPU文件>,把里面对应的文件copy并改名为.config至linux源码根目录下

2>利用当前运行已有的文件(要用ls /boot/ -a查看)把/boot/config-2.6.18-53.e15拷贝并改名为.config至linux源码根目录下执行以上操作就可以用make menuconfig在拷贝

.config文件上面修改文件了

编译内核:

1)make zImage

2)make bzImage

区别:在X86平台上,zimage只能用于小于512k的内核

获取详细编译信息:make zimage V=1 或 make bzimage V=1

编译好的内核在:arch/<cpu>/boot/目录下

注意:在把.config配置文件cp到根目录编译内核前,必须进入make menuconfig并保存退出(否则生不了效)

编译并安装模块:

1)编译内核模块:make modules

2)安装内核模块:make modules_install INSTALL_MOD_PATH=/lib/modules

更换本机器内核:将编译好的内核模块从内核源码目录copy至/lib/modules下

制作init ramdisk():输入执行命令mkinitrd initrd-2.6.39(任意) 2.6.39(可通过查询/lib/modules下的目录得到)

注意:

mkinitrd命令为redhat里面的,ubuntu的命令为:mkinitramfs -k /lib/modules/模块安装位置 -o initrd-2.6.39(任意) 2.6.39(可通过查询/lib/modules下的目录得到)

如果ubuntu里面没有mkinitramfs命令可以用apt-get install initrd-tools进行安装

安装内核模块:

1)手动

1>cp linux根目录/arch/x86/boot/bzImage /boot/mylinux-2.6.39

2>cp linux根目录/initrd-2.6.39 /boot/initrd-2.6.39

最后修改/etc/grub.conf或/etc/lilo.conf文件

2)自动

1>make install:这个命令会自动完成上面的操作(查看当前内核版本:uname -r)

-----------------------------------------------------------------------------

四.linux内核模块开发

描述:

linux内核组件非常庞大,内核ximage并不包含某组件,而是在该组件需要被使用的时候,动态的添加到正在运行的内核中(也可以卸载),这种机制叫做“内核模块”的机制。内核模块通常通过使用makefile文件对模块进行编译

模块安装与卸载:

1)加载:insmod hello.ko

2)卸载:rmmod hello

3)查看:lsmod

4)加载(自动寻找模块依赖):modprobe hello

modprobe会根据文件/lib/modules/version/modules.dep来查看要加载的模块,看它是否还依赖于其他模块,如果是,会先找到这些模块,把它们先加载到内核

实例分析:

1)moduleDep/1(一个模块的编译)


  
  
  1.  1 #include <linux/module.h> 2 #include <linux/init.h> 3  4 //模块入口函数 5 //__init:表示代码段中的子段,里面的内容只运行一次并且回收内存. 6 static int __init hello_init(void) 7 { 8     printk(KERN_EMERG "hello world!\n"); 9     return 0;10 }11 //模块卸载函数12 //__exit:13 static void __exit hello_exit(void)14 {15     printk(KERN_EMERG "hello exit!\n");16 }17 //内核符号导出 函数18 int add_integar(int a,int b)19 {20     return a+b;  
  2. 21 }22 int sub_integar(int a,int b)23 {24     return a-b;  
  3. 25 }26 27 module_init(hello_init);28 module_exit(hello_exit);29 //函数导出30 EXPORT_SYMBOL(add_integar);31 EXPORT_SYMBOL(sub_integar); 

makefile:


  
  
  1. #第一次执行KERNELRELEASE是空的,所以执行else里面的 
  2. ifneq ($(KERNELRELEASE),) 
  3.  
  4. obj-m :=hello.o 
  5.  
  6. #else块 
  7.  
  8. elseKDIR:= /lib/modules/2.6.18-53.el5/build 
  9.  
  10. all
  11. #KDIR    依赖内核模块源代码路径(内核编译安装路径) 
  12. #PWD     表示内核代码在哪(当前目录) 
  13. #modules 编译的是模块 
  14.     make -C $(KDIR) M=$(PWD) modules  
  15.  
  16. clean: 
  17.     rm -f *.ko *.o *.mod.o *.mod.c *.symvers *.order 
  18.  
  19. endif 

2)moduleDep/2(两个模块的编译)


  
  
  1. #include <linux/module.h>  
  2. #include <linux/init.h>  
  3. //模块可选信息  
  4. MODULE_LICENSE("GPL");//许可证声明  
  5. MODULE_AUTHOR("liyuan");//作者声明  
  6. MODULE_DESCRIPTION("This module is a param example.");//模块描述  
  7. MODULE_VERSION("V1.0");//模块别名  
  8. MODULE_ALIAS("a simple module");//模块别名  
  9.  
  10. //模块参数 
  11. static char *name = "liyuan arg"
  12. static int age = 30; 
  13. //S_IRUGO是参数权限,也可以用数字 
  14. module_param(age,int,S_IRUGO); 
  15. module_param(name,charp,S_IRUGO); 
  16.  
  17.  
  18. //使用外部文件函数 
  19. extern int add(int a,int b); 
  20.  
  21.  
  22. //声明 外部内核符号 函数 
  23. extern int add_integar(int a,int b); 
  24. extern int sub_integar(int a,int b); 
  25.  
  26. static int __init mains_init(void) 
  27.      //多文件编译 
  28.  
  29.     printk(KERN_EMERG"param hi"); 
  30.     int vle=add(1,2); 
  31.     printk(KERN_EMERG"add value:%d\n",vle); 
  32.     //模块参数 
  33.  
  34.      printk(KERN_EMERG" name : %s\n",name); 
  35.      printk(KERN_EMERG" age : %d\n",age); 
  36.  
  37.     //使用其他模块的函数(内核符号导出) 
  38.     int adds=add_integar(3,1); 
  39.     int subs=sub_integar(3,1); 
  40.     printk(KERN_EMERG" add_integar : %d\n",adds); 
  41.     printk(KERN_EMERG" sub_integar : %d\n",subs); 
  42.     return 0; 
  43.  
  44. static void __exit mains_exit(void) 
  45.     printk("param exit!"); 
  46.  
  47. module_init(mains_init);52 module_exit(mains_exit); 

add.c


  
  
  1. int add(int a,int b) 
  2.      return a+b; 

makefile


  
  
  1. ifneq ($(KERNELRELEASE),) 
  2. #两个以上内核源文件 生成单独的内核模块名ma 
  3.  
  4. #内核ma 
  5. obj-m :=ma.o 
  6. #下面的ma-objs前面必须和上面一样为ma 
  7. ma-objs := mains.o add.oelseKDIR:= /lib/modules/2.6.18-53.el5/build 
  8.  
  9. all
  10.         make -C $(KDIR) M=$(PWD) modules  
  11. clean: 
  12.     rm -f *.ko *.o *.mod.o *.mod.c *.symvers *.order 
  13.  
  14. endif 

运行带参模块:insmod hello.ko name=yuan age=12

内核符号导出(/proc/kallsyms记录了内核中所有导出的符号的名字与地址):

一个内核模块的运行依赖另一个内核模块的函数实现,必须先运行第一个内核模块,这样就需要进行内核符号导出。

注意:

错误信息:disagrees about version of symbol struct_module insmod:error inserting ...

开发内核模块时会出现,内核模块不匹配的情况.是你当前运行的linux内核与编译连接所依赖的

内核版本不匹配,解决方法:

  • 使用modprobe --force-modversion强行插入
  • 可使用uname -r进行查看当前运行的内核版本

printk内核打印:

在<linux/kernel.h>中printk有8个优先级,按优先级递减的是:

  • KERN_EMERG 0

用于紧急的消息,常常是那些崩溃的消息

  • KERN_ALERT 1

需要立刻行动的消息

  • KERN_CRIT 2

严重情况

  • KERN_ERR 3

错误情况

  • KERN_WARNING(printk默认级别) 4

有问题的警告

  • KERN_NOTICE 5

正常情况,但是仍然值得注意

  • KERN_INFO 6

信息消息

  • KERN_DEBUG 7

用作调试消息

不管是哪个级别的都会在/var/log/messages里面打印出来(messages可以删除后,运行内核进行测试内核打印情况)控制台打印(优先级配置/proc/sys/kernel/printk)

6 4 1 7

  • Console_loglevel
  • Default_message_loglevel
  • Minimum_console_level
  • Default_console_loglevel

在vm+redhat安装2.6.39内核时出现的错误

启动时报could not find filesystem '/dev/root'

解决方法

a.通过make menuconfig选中以下对应的选项

General setup -->

[*] enable deprecated sysfs features to support old userspace tools

成功时下面那个也*了的

b.修改.config文件

修改.config文件中CONFIG_SYSFS_DEPRECATED_V2,将原本被注释掉的

CONFIG_SYSFS_DEPRECATED_V2 改成CONFIG_SYSFS_DEPRECATED_V2=y





本文作者:佚名
来源:51CTO
目录
相关文章
|
2月前
|
缓存 Linux 开发者
Linux内核中的并发控制机制
本文深入探讨了Linux操作系统中用于管理多线程和进程的并发控制的关键技术,包括原子操作、锁机制、自旋锁、互斥量以及信号量。通过详细分析这些技术的原理和应用,旨在为读者提供一个关于如何有效利用Linux内核提供的并发控制工具以优化系统性能和稳定性的综合视角。
|
2月前
|
缓存 负载均衡 算法
深入探索Linux内核的调度机制
本文旨在揭示Linux操作系统核心的心脏——进程调度机制。我们将从Linux内核的架构出发,深入剖析其调度策略、算法以及它们如何共同作用于系统性能优化和资源管理。不同于常规摘要提供文章概览的方式,本摘要将直接带领读者进入Linux调度机制的世界,通过对其工作原理的解析,展现这一复杂系统的精妙设计与实现。
107 8
|
2月前
|
算法 Linux 调度
深入理解Linux内核调度器:从基础到优化####
本文旨在通过剖析Linux操作系统的心脏——内核调度器,为读者揭开其高效管理CPU资源的神秘面纱。不同于传统的摘要概述,本文将直接以一段精简代码片段作为引子,展示一个简化版的任务调度逻辑,随后逐步深入,详细探讨Linux内核调度器的工作原理、关键数据结构、调度算法演变以及性能调优策略,旨在为开发者与系统管理员提供一份实用的技术指南。 ####
84 4
|
6天前
|
Ubuntu Linux 开发者
Ubuntu20.04搭建嵌入式linux网络加载内核、设备树和根文件系统
使用上述U-Boot命令配置并启动嵌入式设备。如果配置正确,设备将通过TFTP加载内核和设备树,并通过NFS挂载根文件系统。
36 15
|
1月前
|
算法 Linux
深入探索Linux内核的内存管理机制
本文旨在为读者提供对Linux操作系统内核中内存管理机制的深入理解。通过探讨Linux内核如何高效地分配、回收和优化内存资源,我们揭示了这一复杂系统背后的原理及其对系统性能的影响。不同于常规的摘要,本文将直接进入主题,不包含背景信息或研究目的等标准部分,而是专注于技术细节和实际操作。
|
1月前
|
存储 缓存 网络协议
Linux操作系统的内核优化与性能调优####
本文深入探讨了Linux操作系统内核的优化策略与性能调优方法,旨在为系统管理员和高级用户提供一套实用的指南。通过分析内核参数调整、文件系统选择、内存管理及网络配置等关键方面,本文揭示了如何有效提升Linux系统的稳定性和运行效率。不同于常规摘要仅概述内容的做法,本摘要直接指出文章的核心价值——提供具体可行的优化措施,助力读者实现系统性能的飞跃。 ####
|
1月前
|
监控 算法 Linux
Linux内核锁机制深度剖析与实践优化####
本文作为一篇技术性文章,深入探讨了Linux操作系统内核中锁机制的工作原理、类型及其在并发控制中的应用,旨在为开发者提供关于如何有效利用这些工具来提升系统性能和稳定性的见解。不同于常规摘要的概述性质,本文将直接通过具体案例分析,展示在不同场景下选择合适的锁策略对于解决竞争条件、死锁问题的重要性,以及如何根据实际需求调整锁的粒度以达到最佳效果,为读者呈现一份实用性强的实践指南。 ####
|
1月前
|
缓存 监控 网络协议
Linux操作系统的内核优化与实践####
本文旨在探讨Linux操作系统内核的优化策略与实际应用案例,深入分析内核参数调优、编译选项配置及实时性能监控的方法。通过具体实例讲解如何根据不同应用场景调整内核设置,以提升系统性能和稳定性,为系统管理员和技术爱好者提供实用的优化指南。 ####
|
1月前
|
负载均衡 算法 Linux
深入探索Linux内核调度机制:公平与效率的平衡####
本文旨在剖析Linux操作系统内核中的进程调度机制,特别是其如何通过CFS(完全公平调度器)算法实现多任务环境下资源分配的公平性与系统响应速度之间的微妙平衡。不同于传统摘要的概览性质,本文摘要将直接聚焦于CFS的核心原理、设计目标及面临的挑战,为读者揭开Linux高效调度的秘密。 ####
37 3
|
2月前
|
负载均衡 算法 Linux
深入探索Linux内核调度器:公平与效率的平衡####
本文通过剖析Linux内核调度器的工作机制,揭示了其在多任务处理环境中如何实现时间片轮转、优先级调整及完全公平调度算法(CFS),以达到既公平又高效地分配CPU资源的目标。通过对比FIFO和RR等传统调度策略,本文展示了Linux调度器如何在复杂的计算场景下优化性能,为系统设计师和开发者提供了宝贵的设计思路。 ####
43 6

热门文章

最新文章