9.1前提

请按上面第七章使用GIT下载源码、使用repo下载工具链linux 驱动 开发,并配置了交叉编译工具链。

为何编译驱动程序之前要先编译内核?

①驱动程序要用到内核文件:

例如驱动程序中这样包含头文件:#include,其中的asm是一个链接文件,指向asm-arm或asm-mips,这须要先配置、编译内核就会生成asm这个链接文件。

②编译驱动时用的内核、开发板上运行到内核linux服务器配置与管理,要一致:

开发板上运行到内核是出厂时烧写的,你编译驱动时用到内核是你自己编译的,这两个内核不一致时会造成一些问题。

所以我们编译驱动程序前,要把自己编译下来到内核放在板子起来,取代原先的内核。

③更换板子上的内核后,板子上的其他驱动也要更换:

板子使用新编译下来的内核时,板子上原来的其他驱动也要更换为新编译下来的。

所以在编译我们自己的第1个驱动程序之前,要先编译内核、模块,但是放在板子起来。

9.2编译内核9.2.1STM32MP157开发板

内核的编译过程如下(编译内核前须要先配置好工具链等一些环境变量):

book@100ask:~/100ask_stm32mp157_pro-sdk$ cd Linux-5.4
book@100ask:~/100ask_stm32mp157_pro-sdk/Linux-5.4$ make 100ask_stm32mp157_pro_defconfig
book@100ask:~/100ask_stm32mp157_pro-sdk/Linux-5.4$ make uImage LOADADDR=0xC2000040 -j4
book@100ask:~/100ask_stm32mp157_pro-sdk/Linux-5.4$ make dtbs
book@100ask:~/100ask_stm32mp157_pro-sdk/Linux-5.4$ cp arch/arm/boot/uImage ~/nfs_rootfs
book@100ask:~/100ask_stm32mp157_pro-sdk/Linux-5.4$ cp arch/arm/boot/dts/stm32mp157c-100ask-512d-lcd-v1.dtb ~/nfs_rootfs

9.2.2IMX6ULL开发板

1.IMX6ULL全功能版

内核的编译过程如下(编译内核前须要先配置好工具链等一些环境变量):

book@100ask:~/100ask_imx6ull-sdk$ cd Linux-4.9.88

驱动开发是干什么的_驱动开发工程师_linux 驱动 开发

book@100ask:~/100ask_imx6ull-sdk/Linux-4.9.88$ make mrproper book@100ask:~/100ask_imx6ull-sdk/Linux-4.9.88$ make 100ask_imx6ull_defconfig book@100ask:~/100ask_imx6ull-sdk/Linux-4.9.88$ make zImage -j4 book@100ask:~/100ask_imx6ull-sdk/Linux-4.9.88$ make dtbs book@100ask:~/100ask_imx6ull-sdk/Linux-4.9.88$ cp arch/arm/boot/zImage ~/nfs_rootfs book@100ask:~/100ask_imx6ull-sdk/Linux-4.9.88$ cp arch/arm/boot/dts/100ask_imx6ull-14x14.dtb ~/nfs_rootfs

编译步骤参考如下,编译好zImage后再去编译设备树文件。

编译完成后生成的文件如右图所示

编译完成后,在arch/arm/boot目录下生成zImage内核文件,在arch/arm/boot/dts目录下生成设备树的二补码文件100ask_imx6ull-14×14.dtb。

把这2个文件复制到/home/book/nfs_rootfs目录下备用,如右图:

linux 驱动 开发_驱动开发是干什么的_驱动开发工程师

2.IMX6ULLmininand版

kernel的编译过程如下(编译内核前须要先配置好工具链等一些环境变量):

book@100ask:~/100ask_myir_mini_imx6ull-sdk$ cd Linux-4.9.88
book@100ask:~/100ask_myir_mini_imx6ull-sdk/Linux-4.9.88$ make mrproper
book@100ask:~/100ask_myir_mini_imx6ull-sdk/Linux-4.9.88$ make 100ask_myir_imx6ull_mini_defconfig
book@100ask:~/100ask_myir_mini_imx6ull-sdk/Linux-4.9.88$ make zImage -jN //N 表示根据 CPU 个数,来加速编译系统
book@100ask:~/100ask_myir_mini_imx6ull-sdk/Linux-4.9.88$ make dtbs
book@100ask:~/100ask_myir_mini_imx6ull-sdk/Linux-4.9.88$ cp arch/arm/boot/zImage ~/nfs_rootfs
book@100ask:~/100ask_myir_mini_imx6ull-sdk/Linux-4.9.88$ cp arch/arm/boot/dts/100ask_myir_imx6ull_mini.dtb ~/nfs_rootfs

驱动开发是干什么的_驱动开发工程师_linux 驱动 开发

编译成功后,可以得到这些文件:内核文件arch/arm/boot/zImage,设备树文件arch/arm/boot/100ask_myir_imx6ull_mini.dtb。

把这2个文件复制到/home/book/nfs_rootfs目录下备用。

9.2.3其他开发板

使用GTI下载源码后,在以下目录有其他开发板的使用指南,请自行参考对应指南:

9.3编译内核模块9.3.1STM32MP157开发板

步入内核源码目录后,就可以编译内核模块了:

book@100ask:~$ cd 100ask_stm32mp157_pro-sdk/Linux-5.4
book@100ask:~/100ask_stm32mp157_pro-sdk/Linux-5.4$ make modules -j4
book@100ask:~/100ask_stm32mp157_pro-sdk/Linux-5.4$ sudo make INSTALL_MOD_PATH=/home/book/nfs_rootfs modules_install

最后一条命令是把模块安装到/home/book/nfs_rootfs目录下备用,会得到/home/book/nfs_rootfs/lib/modules目录。

9.3.2IMX6ULL开发板

无论是那个版本的IMX6ULL开发板,编译内核模块的命令是一样的。

1.IMX6ULL全功能版

步入内核源码目录后linux 驱动 开发,就可以编译内核模块了:

book@book-virtual-machine:~$ cd ~/100ask_imx6ull-sdk/Linux-4.9.88/
book@100ask:~/100ask_imx6ull-sdk/Linux-4.9.88$ make ARCH=arm CROSS_COMPILE=arm-linux-gnueabihf- modules
book@100ask:~/100ask_imx6ull-sdk/Linux-4.9.88$ sudo make ARCH=arm INSTALL_MOD_PATH=/home/book/nfs_rootfs modules_install

最后一条命令是把模块安装到/home/book/nfs_rootfs目录下备用,会得到/home/book/nfs_rootfs/lib/modules目录。

内核模块编译完成后截图示例

驱动开发工程师_驱动开发是干什么的_linux 驱动 开发

安装模块后的/home/book/nfs_rootfs/目录结构如右图所示。

注意:右图用到tree命令,假如提示没有该命令,须要执行“sudoaptinstalltree”命令安装tree工具(前提是Ubuntu能上网)。

2.IMX6ULLmininand版

步入内核源码目录后,就可以编译内核模块了:

book@book-virtual-machine:~$ cd ~/100ask_myir_mini_imx6ull-sdk/Linux-4.9.88
book@100ask:~/100ask_myir_mini_imx6ull-sdk/Linux-4.9.88$ make ARCH=arm CROSS_COMPILE=arm-linux-gnueabihf- modules
book@100ask:~/100ask_imx6ull-sdk/Linux-4.9.88$ sudo make ARCH=arm INSTALL_MOD_PATH=/home/book/nfs_rootfs modules_install

编译过程请参考前面IMX6ULL全功能版的截图,最后一条命令是把模块安装到/home/book/nfs_rootfs目录下备用,会得到/home/book/nfs_rootfs/lib/modules目录。

9.3.3其他开发板

使用GTI下载源码后,在以下目录有其他开发板的使用指南,请自行参考对应指南:

9.4安装内核和模块到开发板上

假定:执行上述命令后,在Ubuntu的/home/book/nfs_rootfs目录下早已有了zImage或uImage、dtb文件,而且有lib/modules子目录(上面富含各类模块)。

下边linux基础教程,要把那些文件复制到开发板上。

假如你使用的是VMwareNAT形式,假定WindowsIP为192.168.1.100,在开发板启动步入Linux后,输入root登陆,之后执行以下命令(注意:必须指定port为2049、mountport为9999):

mount -t nfs -o nolock,vers=3,port=2049,mountport=9999 192.168.1.100:/home/book/nfs_rootfs /mnt
cp /mnt/zImage /boot 或 cp /mnt/uImage /boot 
cp /mnt/*.dtb /boot
cp /mnt/lib/modules /lib -rfd
sync
reboot

假如你使用的是VMware桥接方法,假定UbuntuIP为192.168.1.100,在开发板上执行以下命令:

mount -t nfs -o nolock,vers=3 192.168.1.100:/home/book/nfs_rootfs /mnt
cp /mnt/zImage /boot 或 cp /mnt/uImage /boot 
cp /mnt/*.dtb /boot
cp /mnt/lib/modules /lib -rfd
sync
reboot

最后重启开发板,它就使用新的zImage或uImage、dtb、模块了。

9.5体验第1个驱动程序9.5.1更改Makefile指定内核目录

把第1个驱动程序01_hello_drv上传到Ubuntu后,更改它的Makefile,设置其中的KERN_DIR变量为内核的源码目录,以IMX6ULL为例,如下:

KERN_DIR = /home/book/100ask_imx6ull-sdk/Linux-4.9.88

9.5.2编译

设置好工具链、配置、编译过内核以后,在01_hello_drv目录下执行make命令即可编译驱动程序及测试程序,如下:

驱动开发工程师_驱动开发是干什么的_linux 驱动 开发

9.5.3在开发板上运行

开发板启动后通过nfs挂载Ubuntu目录的方法,将相应的文件拷贝到开发板上。假如你使用的是VMwareNAT形式,假定WindowsIP为192.168.1.100,在开发板上执行以下命令(注意:

必须指定port为2049、mountport为9999):

[root@board:~]# mount -t nfs -o nolock,vers=3,port=2049,mountport=9999 192.168.1.100:/home/book/nfs_rootfs /mnt

假如你使用的是VMware桥接方法,假定UbuntuIP为192.168.1.100,在开发板上执行以下命令:

[root@board:~]# mount -t nfs -o nolock,vers=3 192.168.1.100:/home/book/nfs_rootfs /mnt

挂载NFS成功后,把驱动和测试程序复制到开发板上:

[root@board:~]# cp /mnt/hello_drv.ko ./
[root@board:~]# cp /mnt/hello_drv_test ./

之后安装驱动程序并验证是否成功:

[root@board:~]# insmod hello_drv.ko

执行“lsmod”可以看见hello_drv驱动,如下:

执行“cat/proc/devices”:

执行“ls-l/dev/hello”,可以发觉有这个设备节点,但是它的主设备号跟上图一样:

linux 驱动 开发_驱动开发是干什么的_驱动开发工程师

最后执行测试程序:

./hello_drv_test -w www.100ask.net
./hello_drv_test -r

倘若不想见到驱动调试信息,可以先执行以下命令拿来关掉内核的复印信息:

echo "1 4 1 7" > /proc/sys/kernel/printk

9.6常见问题

安装驱动程序时,假若有以下提示信息,缘由就是板子上运行的内核太老了,解决方式就是先编译内核、

替换板能上的内核,再重新编译、安装驱动程序:

disagrees about version of symbol device_create
[ 2098.200219] hello_drv: Unknown symbol device_create (err -22)
[ 2098.208445] hello_drv: disagrees about version of symbol device_destroy
[ 2098.215871] hello_drv: Unknown symbol device_destroy (err -22)
insmod: ERROR: could not insert module hello_drv.ko: Invalid parameters

全文下载:嵌入式Linux系统开发完全指南第二版

Tagged:
Author

这篇优质的内容由TA贡献而来

刘遄

《Linux就该这么学》书籍作者,RHCA认证架构师,教育学(计算机专业硕士)。

发表回复