第一次在android平台上单独编译内核模块并且insmod,过程比较难受,毕竟啥事第一次做都不怎么顺畅!

本文基于Android5.1 msm8909

因为笔者使用的平台物理串口只有两个,一个已经作为系统的debug调试串口,那就相当于只有一个串口供使用了,这样因为串口个数需求远大于一个,需要添加虚拟串口的驱动。

采取如下方法实现:

编写虚拟串口驱动

单独编译的时候,

1.创建驱动目录vcom

因为是单独编译,因此文件夹的位置可在任意处,笔者放在android源码的根目录下

2.准备.c .h Makefile 文件

.c 和 Makefile 文件是必须的

这里说下Makefile文件,因为笔者在这吃了个难受的亏

Makefile的编写语法,格式什么的,我这里就不多说了,不知道的朋友可以去查查相关资料

Makefile中有几个重要的点需要注意:第一个是编译采用的ARCH平台,如arm,arm64等;第二个是编译器的路径,这里需要知道绝对路径和相对路径的概念,为了稳保路径正确,我这里采用了绝对路径。第三个就是内核目录:这里我才用android源码编译过后的内核目录。Makefile内容如下所示,几个重要的点用箭头指明了:

android 源码单独编译frameworks模块 安卓单独编译kernel_AndroidDriver

ifneq ($(KERNELRELEASE),)
MODULE_NAME = hogovcom
 $(MODULE_NAME)-objs := extport.o vcom.o
 obj-m := $(MODULE_NAME).oelse
    KERNELDIR ?= /home/adr/workspace/android_code/android5_1/code_7/LINUX/android/out/target/product/msm8909/obj/KERNEL_OBJ
     PWD ?= $(shell pwd)
     ARCH = arm
     CROSS_COMPILE= /home/adr/workspace/android_code/android5_1/code_7/LINUX/android/prebuilts/gcc/linux-x86/arm/arm-eabi-4.8/bin/arm-eabi-4.8-.PHONY: modules clean
modules:
         $(MAKE) ARCH=$(ARCH) CROSS_COMPILE=$(CROSS_COMPILE) -C $(KERNELDIR) M=$(PWD) modules
 clean:
         rm -f *.o *.mod.c .*.*.cmd *.ko *.order *.symvers
         rm -rf .tmp_versionsendif

 

3.执行make指令

执行make的时候,可能会遇到一些错误,出错时可以阅读出错信息来进行纠错,笔者在make时遇到了如下错误,可供参考:

在编译完成并生成ko文件后,在android板上insmod发现出现了如下错误,这可以看到图中有两处类型的错误:

android 源码单独编译frameworks模块 安卓单独编译kernel_串口_02

第一处 Unknown symbol _GLOBAL_OFFSET_TABLE_ (err 0)

这个可能是你编译器不对的原因,笔者开始使用的并不是上图makefile中所示的编译器,而是同级目录下的带android字符的编译器,然后因为出错,我换成了不带android字符的编译器,结果重新make insmod,该错误解决


第二处:咋的一看,这么多方法没找到,但不应该啊,make都过去了,为啥insmod的时候会有方法找不到,结果把多个文件整合成一个文件,在make insmod ,问题解决。 但~ 这也不是个办法啊,大家都知道,在C语言中,不能把所有的代码都放在一个.c文件里面,这样这个.c文件会很大,而且代码维护起来会很麻烦。那么开始思考,makefile多文件生成一个ko这个该咋做,百度到个链接得到一个思路,然后,调换 vcom.o 和extport.o的位置,然后make insmod,问题解决,卧槽,还真是这个问题。对于这个问题,我理解的是因为我vcom.o 里面我是需要调用extport里面的函数的,如果我把extport放后面,这样ld链接的时候,先编译vcom链接,这个时候extport还没有编译链接,那么当然找不到相关函数,这里可能说的不清楚,有错误的地方请大家指点!