2024年3月11日发(作者:)

本系列文章包括两篇,他们文周详地地介绍了Linux系统下用户空间和内核空间数据交换的九

种方式,包括内核启动参数、模块参数和sysfs、

sysctl、系统调用、netlink、procfs、seq_file、debugfs和relayfs,并给出具体的例子帮助读者

掌控这些技术的使

用。

本文是该系列文章的第二篇,他介绍了procfs、seq_file、debugfs和relayfs,并结合给出的例

子程式周详地说明了他们怎么使用。

1、内核启动参数

Linux 提供了一种通过 bootloader 向其传输启动参数的功能,内核研发者能通过这种方式来向

内核传输数据,从而控制内核启动行为。

通常的使用方式是,定义一个分析参数的函数,而后使用内核提供的宏 __setup把他注册到内

核中,该宏定义在 linux/init.h 中,因此要使用他必须包含该头文件:

__setup("para_name=", parse_func)

para_name 为参数名,parse_func

为分析参数值的函数,他负责把该参数的值转换成相应的内核变量的值并设置那个内核变量。

内核为整数参数值的分析提供了函数 get_option 和

get_options,前者用于分析参数值为一个整数的情况,而后者用于分析参数值为逗号分割的一

系列整数的情况,对于参数值为字符串的情况,需要研发

者自定义相应的分析函数。在原始码包中的内核程式kern-boot-params.c

说明了三种情况的使用。该程式列举了参数为一个整数、逗号分割的整数串及字符串三种情况,

读者要想测试该程式,需要把该程式拷贝到要使用的内核的源码目

录树的一个目录下,为了避免和内核其他部分混淆,作者建议在内核源码树的根目录下创建一

个新目录,如 examples,然后把该程式拷贝到

examples 目录下并重新命名为 setup_example.c,并且为该目录创建一个 Makefile 文件:

obj-y = setup_example.o

Makefile 仅许这一行就足够了,然后需要修改源码树的根目录下的 Makefile文件的一行,把下

面行

core-y := usr/

修改为

core-y := usr/ examples/

注意:如果读者创建的新目录和重新命名的文件名和上面不同,需要修改上面所说 Makefile 文

件相应的位置。

做完以上工作就能按照内核构建步骤去构建新的内核,在构建好内核并设置好lilo或grub为该

内核的启动条目后,就能启动该内核,然后使用lilo或grub的编辑功能为该内核的启动参数行

增加如下参数串:

setup_example_int=1234 setup_example_int_array=100,200,300,400

setup_example_string=Thisisatest

当然,该参数串也能直接写入到lilo或grub的设置文件中对应于该新内核的内核命令行参数串

中。读者能使用其他参数值来测试该功能。

下面是作者系统上使用上面参数行的输出:

setup_example_int=1234

setup_example_int_array=100,200,300,400

setup_example_int_array includes 4 intergers

setup_example_string=Thisisatest

读者能使用

dmesg | grep setup

来查看该程式的输出。

2、模块参数和sysfs

内核子系统或设备驱动能直接编译到内核,也能编译成模块,如果编译到内核,能使用前一节

介绍的方法通过内核启动参数来向他们传递参数,如果编译成模块,则能通过命令行在插入模

块时传递参数,或在运行时,通过sysfs来设置或读取模块数据。

Sysfs是个基于内存的文件系统,实际上他基于ramfs,sysfs提供了一种把内核数据结构,他

们的属性及属性和数据结构的联系开放给用

户态的方式,他和kobject子系统紧密地结合在一起,因此内核研发者不必直接使用他,而是内

核的各个子系统使用他。用户要想使用 sysfs

读取和设置内核参数,仅需装载 sysfs 就能通过文件操作应用来读取和设置内核通过 sysfs 开

放给用户的各个参数:

$ mkdir -p /sysfs

$ mount -t sysfs sysfs /sysfs

注意,不要把 sysfs 和 sysctl 混淆,sysctl 是内核的一些控制参数,其目的是方便用户对内

核的行为进行控制,而

sysfs 仅仅是把内核的 kobject 对象的层次关系和属性开放给用户查看,因此 sysfs 的绝大部

分是只读的,模块作为一个

kobject 也被出口到 sysfs,模块参数则是作为模块属性出口的,内核实现者为模块的使用提供

了更灵活的方式,允许用户设置模块参数在

sysfs 的可见性并允许用户在编写模块时设置这些参数在 sysfs 下的访问权限,然后用户就能

通过sysfs