实现内核驱动程序模块

时间:2021-01-19 22:27:13

例子是从《Android系统源代码情景分析》第二章抄过来的,在学习的过程中还是遇到了不少的问题。

个人体会:在学习第二章之前应该把《Linux设备驱动程序》这本书至少前四章要读一遍,理解一些基础概念和背景知识,不过这本书还是略旧,比如sysfs就没有解释,多google吧。

本书第二章要理解透了再往下进行,因为这个简单的例子从下往上贯穿了Android系统各层,是理解后面各章的基础。

下面列出我遇到的问题和解决办法。

  • 三类文件系统接口的关系、作用分别是什么?

它实现了proc文件系统接口、传统的设备文件系统接口和devfs文件系统接口。

类Unix系统有一条基本的设计哲学:几乎所有的数据实体都被抽象成统一的接口——文件来看待。procfs、devfs和sysfs都是这种设计的体现。

/dev目录下每一个文件对应一个设备,通过操作这些文件可以实现与内核的交互,但devfs存在一些缺点,如命名不灵活,不能任意指定文件名,而且所有文件都在/dev/根目录下。进而演生出sysfs,它把实际连接到系统上的设备组织成一个分级的文件体系,比devfs更清晰更便于管理,sysfs挂载在/sys下面。

/proc文件系统中主要包含三大类内容:进程相关部分、系统信息部分和系统自信息部分。显然在本例中,procfs和sysfs部分的接口,作为入门第一个样例来说并不必要。

因此我打算只实现传统设备文件系统接口的部分,sysfs和procfs暂不实现。我认为依然可以完整地正常工作。

  • 编译问题

  • 每次编译和运行模拟器之前需要把各环境变量重设一遍:
$ cd WORKING_DIRECTORY
$ source build/envsetup.sh
$ lunch aosp_arm-eng
$ cd kernel
$ cd goldfish
$ export ARCH=arm
$ export SUBARCH=arm
$ export CROSS_COMPILE=arm-eabi-
$ export PATH=WORKING_DIRECTORY/prebuilts/gcc/darwin-x86/arm/arm-eabi-4.8/bin:$PATH
$ make goldfish_armv7_defconfig
$ make menuconfig
$ make -j8
$ emulator -kernel arch/arm/boot/zImage &
  • 在__freg_setup_dev函数中,对函数init_MUTEX的调用编译出错,这是因为该函数已经被启用,改成如下即可:

sema_init(&(dev->sem), 1);
  • 修改内核Kconfig文件。我编译的内核是3.4,需要在drivers/Kconfig中添加
source "drivers/freg/Kconfig"

而不是在arch/arm/Kconfig中。


 

  • 配置

  • 在Mac OSX,调用make menuconfig的时候出错,解决办法已经更新到《Android源码、内核编译》这篇博文里了。
  • 进入make menuconfig配置界面后,如果要设置Enable loadable module support,这块的交互有点坑爹。初次进入界面是这样的,Enable loadable module support没有被选中:

实现内核驱动程序模块

如果直接回车进入子菜单,发现前面是---,是不能被修改的:

实现内核驱动程序模块

我以为是自己的内核版本有问题,倒腾了半天才明白:应该在一级界面中先对Enable loadable module support按下Y,使之选中:

实现内核驱动程序模块

然后再回车进入才能看到并修改子项内容:

实现内核驱动程序模块

一万头*奔腾而过,害老子研究半天……


 

  • 调试

  • 查看内核log

我把模块编译到内核中,起初却在/dev下面找不到freg文件,必须要调试代码看在哪一步出了错。freg.c源码多出调用printk函数写了日志,我在每条日志字串头部加了[freg]的字样,如下:

printk(KERN_ALERT"[freg]Initializing freg device.\n");

以便在浩瀚的内核日志中筛选出freg的日志。

  • 方法一:可以使用dmesg查看内核打印信息,再配合grep如下:

$ adb shell
$ dmesg | grep "\[freg\]"
[freg]Initializing freg device.
[freg]Succeeded to initialize freg device.
  •  方法二:可以在运行emulator的时候加-show-kernel参数:
$ emulator -kernel arch/arm/boot/zImage -show-kernel &
  •  动态加载模块

当在make menuconfig中选择了动态加载模块,编译、启动emulator,需要通过adb把.ko模块push到emulator:

# 启动emulator
$ emulator -kernel arch/arm/boot/zImage &
# 把刚刚编译的freg.ko拷贝到emulator的/data下
$ adb push drivers/freg/freg.ko /data
$ adb shell ls -l /data/freg.ko
-rw-rw-rw- root     root         5281 2016-02-21 01:45 freg.ko
# 加载freg.ko模块
$ adb shell insmod /data/freg.ko