• 实现内核驱动程序模块


    例子是从《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.
    ");

    以便在浩瀚的内核日志中筛选出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
  • 相关阅读:
    Springboot Endpoint之二:Endpoint源码剖析
    Linux进程被杀掉(OOM killer),查看系统日志
    docker常用命令详解
    micrometer自定义metrics
    使有prometheus监控redis,mongodb,nginx,mysql,jmx
    Grafana+Prometheus打造springboot监控平台
    Grafana介绍
    Prometheus介绍
    Groovy与Java集成常见的坑
    ES之1:基本概念及原理
  • 原文地址:https://www.cnblogs.com/palance/p/5204316.html
Copyright © 2020-2023  润新知