2.2 Android对Linux内核的改动

Android从多个方面对Linux内核进行了改动与增强,下面将对此进行详细介绍和分析。

2.2.1 Goldfish

Android模拟器通过运行一个Goldfish的虚拟CPU.Goldfish来运行arm926t指令集(arm926t属于armv5构架),并且仿真了输入/输出,比如键盘输入和LCD 输出。这个模拟器其实是在qemu之上开发的,输入/输出是基于libSDL的。既然Goldfish是被模拟器运行的虚拟CPU,那么当 Android 在真实的硬件设备上运行时,我们就需要去掉它,因此,只有知道 Google 对Goldfish做了哪些具体改动之后才能正确地去掉。据统计,Android 内核对Goldfish的改动主要涉及44个文件,具体汇总如下。

说明 本书中在被改动的文件前面加了Chg标记,在新增的文件前面加了New标记。

1      Chg     arch/arm/Makefile                       添加CONFIG_ARCH_GOLDFISH
2      New     arch/arm/configs/goldfish_defconfig           默认配置文件
3      New     arch/arm/mach-goldfish/Kconfig             为Goldfish CPU添加Kernel配置文件
4      New     arch/arm/mach-goldfish/Makefile             添加board-goldfish.o
5      New     arch/arm/mach-goldfish/Makefile.boot         为Goldfish CPU进行启动配置
6      New     arch/arm/mach-goldfish/audio.c Audio         的输入/输出
7      New     arch/arm/mach-goldfish/board-goldfish.c         中断请求、输入/输出等
8      New     arch/arm/mach-goldfish/pdev_bus.c           设备总线
9      New     arch/arm/mach-goldfish/pm.c                电源管理
10     New     arch/arm/mach-goldfish/switch.c             Switch控制
11     New     arch/arm/mach-goldfish/timer.c              获取和设置时间
12     Chg     arch/arm/mm/Kconfig                     添加ARCH_GOLDFISH到支持列表
13     Chg     drivers/char/Makefile                     添加goldfish_tty
14     New     drivers/char/goldfish_tty.c                  TTY驱动
15     Chg     drivers/input/keyboard/Kconfig              为Goldfish的键盘事件添加配置文件
16     Chg     drivers/input/keyboard/Makefile             添加goldfish_events事件
17     New     drivers/input/keyboard/goldfish_events.c         Goldfish键盘驱动
18     Chg     drivers/mmc/host/Kconfig                  添加Kernel配置选项Goldfish MMC卡
19     Chg     drivers/mmc/host/Makefile                 添加Goldfish MMC卡驱动
20     New     drivers/mmc/host/goldfish.c                 多媒体驱动
21     Chg     drivers/mtd/devices/Kconfig                 为Goldfish的NAND flash device添加Kernel配置选项
22     Chg     drivers/mtd/devices/Makefile                添加goldfish_nand
23     New     drivers/mtd/devices/goldfish_nand.c           NAND flash驱动
24     New     drivers/mtd/devices/goldfish_nand_reg.h         NAND flash驱动
25     Chg     drivers/power/Kconfig                    为Goldfish的battery(电池)驱动添加kernel配置选项
26     Chg     drivers/power/Makefile                    添加Goldfish电池
27     New     drivers/power/goldfish_battery.c             能源和电池状态驱动
28     Chg     drivers/rtc/Kconfig                       为Goldfish的rtc(时钟)驱动添加Kernel配置选项
29     Chg     drivers/rtc/Makefile                      添加rtc-goldfish
30     New     drivers/rtc/rtc-goldfish.c                   实时时钟驱动
31     Chg     drivers/video/Kconfig                     添加Goldfish的framebuffer
32     Chg     drivers/video/Makefile                    添加Goldfish的framebuffer
33     New     drivers/video/goldfishfb.c                  framebuffer驱动
34     New     include/asm-arm/arch-goldfish/dma.h
35     New     include/asm-arm/arch-goldfish/entry-macro.S
36     New     include/asm-arm/arch-goldfish/hardware.h
37     New     include/asm-arm/arch-goldfish/io.h
38     New     include/asm-arm/arch-goldfish/irqs.h
39     New     include/asm-arm/arch-goldfish/memory.h
40     New     include/asm-arm/arch-goldfish/system.h
41     New     include/asm-arm/arch-goldfish/timer.h
42     New     include/asm-arm/arch-goldfish/timex.h
43     New     include/asm-arm/arch-goldfish/uncompress.h
44     New     include/asm-arm/arch-goldfish/vmalloc.h

2.2.2 YAFFS2

不同于PC机(文件是存储在硬盘上的),手机使用FLASH作为存储介质。HTC的G1使用的是NANDFLASH——这种存储目前已经相当普及了,而且种类也颇多(如SLC、MLC等),存储密度也越来越高(已经出现几十GB大小的NANDFLASH),价格也越来越低。

YAFFS2是专门用在FLASH上的文件系统,YAFFS2是“Yet Another Flash File System, 2nd edition”的缩写。YAFFS2为Linux内核提供了一个高效访问NANDFLASH的接口。但是NANDFLASH 的支持并不包含在标准的 2.6.25 内核中,所以 Google 在其中添加了对NANDFLASH的支持。据统计,为了支持YAFFS2,Google一共改动和增加了以下35个文件:

1     Chg     fs/Kconfig                  添加YAFFS配置
2     Chg     fs/Makefile                 添加YAFFS

以下为新增的YAFFS2:

1     New    fs/yaffs2/Kconfig             18    New     fs/yaffs2/yaffs_mtddif2.h
2     New    fs/yaffs2/Makefile             19    New     fs/yaffs2/yaffs_nand.c
3     New    fs/yaffs2/devextras.h           20    New     fs/yaffs2/yaffs_nand.h
4     New    fs/yaffs2/moduleconfig.h        21    New     fs/yaffs2/yaffs_nandemul2k.h
5     New    fs/yaffs2/yaffs_checkptrw.c      22    New     fs/yaffs2/yaffs_packedtags1.c
6     New    fs/yaffs2/yaffs_checkprtw.h      23    New     fs/yaffs2/yaffs_packedtags1.h
7     New    fs/yaffs2/yaffs_ecc.c           24    New     fs/yaffs2/yaffs_packedtags2.c
8     New    fs/yaffs2/yaffs_ecc.h           25    New     fs/yaffs2/yaffs_packedtags2.h
9     New    fs/yaffs2/yaffs_fs.c            26    New     fs/yaffs2/yaffs_qsort.c
10    New    fs/yaffs2/yaffs_getblockinfo.h    27    New     fs/yaffs2/yaffs_qsort.h
11    New    fs/yaffs2/yaffs_guts.c          28    New     fs/yaffs2/yaffs_tagscompat.c
12    New    fs/yaffs2/yaffs_guts.h          29    New     fs/yaffs2/yaffs_tagscompat.h
13    New    fs/yaffs2/yaffs_mtdif.c          30    New     fs/yaffs2/yaffs_tagsvaliditiy.c
14    New    fs/yaffs2/yaffs_mtdif.h          31    New     fs/yaffs2/yaffs_tagsvalidity.h
15    New    fs/yaffs2/yaffs_mtddif1.c        32    New     fs/yaffs2/yaffsinterface.h
16    New    fs/yaffs2/yaffs_mtddif1.h        33    New     fs/yaffs2/yportenv.h
17    New    fs/yaffs2/yaffs_mtddif2.c

2.2.3 蓝牙

在蓝牙通信协议栈里Google修改了10个文件。这些改动修复了一些与蓝牙耳机相关的明显的Bug,以及一些与蓝牙调试和访问控制相关的函数,具体如下所示。

1     Chg     drivers/bluetooth/Kconfig         添加HCI UART Debug
2     Chg     drivers/bluetooth/hci_II.c          如果HCI UART Debug定义在Kernel配置中,则添加BT_DBG()宏
3     Chg     net/bluetooth/Kconfig            添加配置选项L2CAP,HCI_CORE,HCI_SOCK,以及通用接口和语音
4     Chg     net/bluetooth/af_bluetooth.c       如果CONFIG_ANDROID_PARANOID_NETWORK被定义,则添加蓝牙功能的安全检查
5     Chg     net/bluetooth/hci_event.c          修正蓝牙的加密Bug和增加语音的支持
6     Chg     net/bluetooth/rfcomm/core.c       修正Bug
7     Chg     net/bluetooth/rfcomm/sock.c       修复Bug
8     Chg     net/bluetooth/sco.c              禁用SCO链接
9     Chg     include/net/bluetooth/hci_core.h     禁用LMP_ESCO
10    Chg     include/net/bluetooth/rfcomm.h     在rfcomm_dlc中添加“out”参数

2.2.4 调度器(Scheduler)

Android内核还修改了与进程调度和时钟相关的策略。只改动了5个文件,如下:

1     Chg     kernel/sched.c              添加NORMALIZED_SLEEPER
2     Chg     kernel/sched_fair.c           修改内核的调度方式
3     Chg     kernel/softirq.c             修改为CPU调度
4     Chg     kernel/time/tick-sched.c       修改为CPU调度
5     Chg     include/linux/tick.h          如果 CONFIG_NO_HZ 被定义,则添加 tick_nohz_ update_ stopped_sched_tick()

2.2.5 Android新增的驱动

Android在Linux的基础上新增了许多特有的驱动,如下所示。

1)IPC Binder 一种IPC(进程间通信)机制。它的进程能够为其他进程提供服务——通过标准的Linux系统调用API。IPC Binder的概念起源于一家名为Be.Inc的公司,在Google之前就已经被Palm软件采用了。

2)Low Memory Killer 其实内核里已经有一个类似的功能,名称为 oom killer(out of memory killer)。当内存不够的时候,该策略会试图结束一个进程。

3)Ashmem 匿名共享内存。该功能使得进程间能够共享大块的内存。比如说,系统可以使用 Ashmem 保存一些图标,多个应用程序可以访问这个共享内存来获取这些图标。Ashmem 为内核提供了一种回收这些使用完的共享内存块的方法,如果某个进程试图访问这些已经被回收的内存块,它将会得到错误的返回值,以便它重新进行内存块分配和数据初始化。

4)RAM Console and Log Device 为了调试方便,Android添加了一个功能,使调试信息可以输入到一个内存块中。此外,Android 还添加了一个独立的日志模块,这样用户空间的进程就能够读写日志消息,以及调试打印信息等。

5)Android Debug Bridge 嵌入式设备的调试的确比较麻烦,为了便于调试,Google设计了这个调试工具,可以简称为ADB,使用USB作为连接方式,ADB可以看做是连接Android设备和PC机的一套协议。

除了这些主要的功能之外,Android还增加了诸如 real-time clock、switch、timed GPIO等功能,所有这些改动和增加包含在以下28个文件之中。

1     Chg     drivers/Kconfig                进入配置文件
2     Chg     drivers/Makefile               添加switch,驱动等
3     New    drivers/android/Kconfig            添加 BINDER_IPC、POWER、POWER_STAT、POWER_ ALARM、LOGGER、RAM_CONSOLE、TIMED_GPIO、PARANOID_NETWORK到配置中
4     New    drivers/android/Makefile          添加binder.o、power.o、alarm.o、logger.o、ram_console.o、timed_gpio
5     New    drivers/android/alarm.c           系统硬件时钟和实时时钟管理
6     New    drivers/android/binder.c          IPC机制(Binder)
7     New    drivers/android/logger.c          Google的日志API
8     New    drivers/android/ram_console.c        RAM控制台和日志设备方便调试为了调试方便,Android 添加了一个功能,使得调试信息可以输入到一个内存块中。此外, Android 添加了一个独立的日志模块,这样用户空间的进程能够读写日志消息,调试打印信息等。
9     New    drivers/android/timed_gpio.c         Google的GPIO定时驱动
10    New    drivers/switch/Kconfig             为GPIO添加配置选项
11    New    drivers/switch/Makefile             引入GPIO驱动
12    New    drivers/switch/switch_class.c
13    New    drivers/switch/switch_gpio.c
14    Chg     drivers/usb/gadget/Kconfig          添加ADB配置选项
15    Chg     drivers/usb/gadget/Makefile          编译ADB所需的配置选项
16    New    drivers/usb/gadget/android_adb.c      ADB驱动
17    New    include/linux/android_aid.h          添加AIDs、INET、networking
18    New    include/linux/android_alarm.h        时钟功能设置
19    New    include/linux/android_timed_gpio.h    GPIO结构体
20    New    include/linux/ashmem.h             Android共享内存
21    New    include/linux/binder.h              Binder IPC API定义
22    New    include/linux/logger.h              Logger定义
23    New    include/linux/switch.h              GPIO switch接口
24    Chg     mm/Makefile                    添加ashmem.o
25    New    mm/ashmem.c                   内存共享实现
26    Chg     drivers/misc/Kconfig               添加LOW_MEMORY_KILLER配置选项
27    Chg     drivers/misc/Makefile              添加lowmemorykiller.c
28    New    drivers/misc/lowmemorykiller.c       当内存过低时,选择并结束进程

2.2.6 电源管理

电源管理(Power Management)对于移动设备来说相当重要,也是最为复杂和开发难度最高的一个功能。Google添加了一个新的电源管理系统,不包含原有的apm和dpm等。这项改动主要涉及以下5个文件:

1     New    include/linux/android_power.h      定义电源管理API
2     New    drivers/android/power.c           电源管理API实现
3     Chg     drivers/input/evdev.c             修改Android电源处理方式
4     Chg     fs/inotify_user.c                修改Android电源处理方式
5     Chg     kernel/power/process.c           修改Android电源处理方式

2.2.7 杂项

除了上述改动之外,还有一些小改动,如新增的额外调试功能、键盘背光控制、TCP 网络管理等,共涉及36个文件,如下所示。

1    New    Documentation/vm/pagemap.txt
2    Chg    arch/arm/Kconfig              添加HAVE_LATENCYTOP_SUPPORT和ARCH_GOLDFISH
3    Chg    arch/arm/kernel/process.c         添加dump_task_regs方法
4    Chg    arch/arm/kernel/signal.c          解决系统无法重新启动的问题
5    Chg    arch/arm/kernel/stacktrace.c       改进调试栈跟踪
6    Chg    arch/arm/mm/abort-ev6.S
7    Chg    drivers/char/Kconfig            添加Memory device driver和Goldfish TTY driver
8    Chg    drivers/char/mem.c             使编译结果输出到/dev/kmem and/dev/mem
9    Chg    drivers/leds/Kconfig            当CPU运行时打开LEDS,但是屏幕是关闭的
10   Chg    drivers/leds/Makefile            添加编译ledtrig-sleep.o
11    New    drivers/leds/ledtrig-sleep.c        睡眠(当关闭屏幕后CPU仍然运行)
12   Chg    drivers/rtc/class.c              修正实时时钟误差的Bug
13   Chg    fs/fat/dir.c                    添加VFAT_IOCTL_GET_VOLUME_ID到fat_dir_ioctl()
14   Chg    fs/fat/inode.c
15   Chg    fs/proc/base.c                 当内存不足时调整/proc文件
16   Chg    fs/proc/proc_misc.c             修正kpagecount_read和kpageflags_read返回的一些错误
17   Chg    fs/proc/task_mmu.c             简化add_to_pagemap中的错误检查
18   Chg    include/asm-arm/elf.h           添加ELF_CORE_COPY_TASK_REGS()宏调用dump_task_regs(...)
19   Chg    include/linux/mm.h             添加shmem_set_file(...)函数原型
20   Chg    include/linux/msdos_fs.h         添加VFAT_IOCTL_GET_VOLUME_ID宏
21   Chg    kernel/hrtimer.c                修复run_hrtimer_pending错误
22   Chg    init/Kconfig                  添加PANIC_TIMEOUT默认为0
23   Chg    kernel/panic.c                 设置默认的panic_timeout:从kernel配置到PANIC_TIMEOUT
24   Chg    kernel/power/console.c           修复虚拟控制台的错误信息
25   Chg    kernel/printk.c                 修复printk错误
26   Chg    mm/filemap.c                 修正filemap_fault
27   Chg    mm/shmmem.c                重构shmem_zero_setup
28   Chg    mm/tiny-shmem.c              重构shmem_zero_setup
29   Chg    include/linux/sockios.h          添加SIOCKILLADDR控制
30   Chg    include/net/tcp.h               添加tcp_v4_nuke_addr函数
31   Chg    net/ipv4/Makefile              如果设置CONFIG_SYSFS,编译sysfs_net_ipv4
32   Chg    net/ipv4/af_inet_c              如果定义CONFIG_ANDROID_PARANOID_NETWORK,则添加安全检查
33   Chg    net/ipv4/devinet.c              添加SIOCKILLADDR
34   Chg    net/ipv4/sysfs_net_ipv4.c         控制TCP窗口长度
35   Chg    net/ipv4/tcp_ipv4.c             添加tcp_v4_nuke_addr函数
36   Chg    net/ipv6/af_inet6.c              如果定义CONFIG_ANDROID_PARANOID_NETWORK,则添加安全检查

上面这些看似简单,但是非常重要,当大家进行系统级应用开发和程序移植时都需要研究这些文件。对于每个文件的具体改动方式和实现,我们需要进一步查看Android的内核源代码,这是后面将要详细讲解的内容。