驱动程序的helloworld

简介: 驱动程序的helloworld

我的开发环境:

内核版本4.1.15

开发板正点原子阿尔法IMX6ULL开发板

source insight写代码,搭建samba服务器与Ubuntu虚拟机共享代码所在目录,Ubuntu与开发板使用nfs共享代码目录。

最简单的Linux驱动结构

一个最简单的Linux驱动主要山以下几个部分组成。

(1)头文件(必须有)驱动需要包含内核相关头文件。必须包含<linux/module.h>和<linux/init.h>

(2)驱动加载函数。(必须有)当加载驱动的时候,驱动加载函数会自动被内核执行。

(3)驱动卸载函数(必须有)当卸载驱动的时候,驱动卸载函数会自动被内核执行。

(4)许可证声明(必须有)Linux内核是开源的,遵守GPL协议,驱动在加找的时候也要遵守相关的协议,可以接收的License有"GPL"、"GPLv2”、“GPLandadditionalrights”、"DuaI BSD/GPL’“Dual MIT/GPL”"Dual MPL/GPL"O内核驱动中最常见的是GPLv2

(5)模块参数(可选)模块参数是模块被加载的时候传递给内核模块的值。

(6)作者和版本信息(可选)可以声明驱动的作者信息和代码的版本信息。

helloworld

#include <linux/module.h>
#include <linux/init.h>
static int hello_init(void){
    printk("hello init!!!\n");
    return 0;
}
static void hello_exit(void){
    printk("hello exit!!!\n");
}
module_init(hello_init);
module_exit(hello_exit);
MODULE_AUTHOR("Paranoid");
MODULE_VERSION("V1.0");
MODULE_LICENSE("GPL");

Makefile

#make 编译项目
#make file 在存放.ko文件目录中创建对应项目的目录
#make install 将*.ko及其应用测试文件移动到根文件中
KERN_DIR = /home/alientek/linux/IMX6ULL/linux/temp/linux-imx-rel_imx_4.1.15_2.1.0_ga_alientek
ROOTFS_DIR = /home/alientek/linux/nfs/rootfs/experiment
#项目名字
PROJECT_NAME = hello
#各驱动名字,ko
DRIVER_NAME1 = hello
all:
    make -C $(KERN_DIR) M=`pwd` modules
clean:
    make -C $(KERN_DIR) M=`pwd` modules clean
    rm -rf modules.order
file:
    mkdir $(ROOTFS_DIR)/$(PROJECT_NAME)
install:
    cp *.ko $(ROOTFS_DIR)/$(PROJECT_NAME)
# 参考内核源码drivers/char/ipmi/Makefile
# 要想把a.c, b.c编译成ab.ko, 可以这样指定:
 obj-m += hello.o

KERNEL_DIR 表示开发板所使用的 Linux 内核源码目录,使用绝对路径,大家根据自己的实际情况填写即可。

M=pwd表示当前路径,直接通过运行“pwd”命令来获取当前所处路径。

obj-m 表示将 hello.c 这个文件编译为 hello.ko 模块。

make -C $(KERN_DIR) M=pwd modules具体的编译命令,后面的 modules 表示编译模块, -C 表示将当前的工作目录切换到指定目录中,也就是 KERNERLDIR 目录。 M 表示模块源码目录,“make modules”命令中加入 M=dir 以后程序会自动到指定的 dir 目录中读取模块的源码并将其编译为.ko 文件。进到KERNEL_DIR,使用PWD路径下源码和Makefile文件编译驱动模块。

make 编译项目

make file 在存放.ko文件目录中创建对应项目的目录

make install 将*.ko及其应用测试文件移动到根文件中

为什么在有的板子上需要,有的不需要设置ARCH和CROSS_COMPILE环境变量?

在Linux源码的顶层目录下,有一个Makefile文件,这个MakefiIe文件控制着Linux的编译流程。也叫做顶层MakefiIe文件。

在顶层Uakefile中有ARCH和CROSSCOMPILE变量。如果我们在顶层MakefiIe中固定了这俩个变量的值,就不用在编译ko文件的时候再次设置。如下IMX6ULL的内核源码的顶层Makefile中就固定了这俩个变量的值。如下图:

内核模块相关命令

模块安装与卸载

驱动编译完成以后扩展名为.ko,有两种命令可以加载驱动模块:insmod 和modprobe,insmod

是最简单的模块加载命令,此命令用于加载指定的.ko 模块,比如加载 drv.ko 这个驱动模块,命

令如下:

insmod drv.ko

insmod 命令不能解决模块的依赖关系,比如 drv.ko 依赖 first.ko 这个模块,就必须先使用

insmod 命令加载 first.ko 这个模块,然后再加载 drv.ko 这个模块。但是modprobe 就不会存在这

个问题,modprobe 会分析模块的依赖关系,然后会将所有的依赖模块都加载到内核中,因此

modprobe 命令相比 insmod 要智能一些。modprobe 命令主要智能在提供了模块的依赖性分析、

错误检查、错误报告等功能,推荐使用 modprobe 命令来加载驱动。modprobe 命令默认会去

/lib/modules/目录中查找模块,比如本书使用的 Linux kernel 的版本号为 4.1.15,

因此 modprobe 命令默认会到/lib/modules/4.1.15 这个目录中查找相应的驱动模块,一般自己制

作的根文件系统中是不会有这个目录的,所以需要自己手动创建。

驱动模块的卸载使用命令“rmmod”即可,比如要卸载 drv.ko,使用如下命令即可:

rmmod drv.ko

也可以使用“modprobe -r”命令卸载驱动,比如要卸载 drv.ko,命令如下:

modprobe -r drv.ko

使用 modprobe 命令可以卸载掉驱动模块所依赖的其他模块,前提是这些依赖模块已经没

有被其他模块所使用,否则就不能使用 modprobe 来卸载驱动模块。所以对于模块的卸载,还是

推荐使用 rmmod 命令。

查看模块

lsmod

功能:列出己经载入Linux的内核模块

也可以使用命令cat/proc/m。du|es来查看模块是否加载成功。

modinfo

功能:查看内核模块信息

语法:modinfo模块名

举例:

modinfo helloworld.ko

在Makefile中删除编译出来的文件(排除ko)

#make 编译项目
#make file 在存放.ko文件目录中创建对应项目的目录
#make install 将*.ko及其应用测试文件移动到根文件中
KERN_DIR = /home/alientek/linux/IMX6ULL/linux/temp/linux-imx-rel_imx_4.1.15_2.1.0_ga_alientek
ROOTFS_DIR = /home/alientek/linux/nfs/rootfs/experiment
#项目名字
PROJECT_NAME = hello
#各驱动名字,ko
DRIVER_NAME1 = hello
all:
    make -C $(KERN_DIR) M=`pwd` modules
clean:
    rm -rf modules.order $(DRIVER_NAME1).mod.c *.o Module.symvers
file:
    mkdir $(ROOTFS_DIR)/$(PROJECT_NAME)
install:
    cp *.ko $(ROOTFS_DIR)/$(PROJECT_NAME)
# 参考内核源码drivers/char/ipmi/Makefile
# 要想把a.c, b.c编译成ab.ko, 可以这样指定:
 obj-m += $(DRIVER_NAME1).o


目录
相关文章
|
7月前
|
固态存储 内存技术 NoSQL
基础代码NVMe模块的实例helloworld代码
基础代码NVMe模块的实例helloworld代码
|
7月前
|
Java Linux 程序员
Linux 下 JNA 调用动态连接库文件 so
Linux 下 JNA 调用动态连接库文件 so
300 0
|
4月前
|
编译器 C# Windows
C#基础:手动编译一个.cs源代码文件并生成.exe可执行文件
通过上述步骤,应该能够高效准确地编译C#源代码并生成相应的可执行文件。此外,这一过程强调了对命令行编译器的理解,这在调试和自动化编译流程中是非常重要的。
353 2
|
7月前
|
C# C语言 C++
VS2010 Csharp调用非托管Cpp生成的DLL文件
VS2010 Csharp调用非托管Cpp生成的DLL文件
42 0
|
C++ Windows
C++ --- Dll文件的生成与调用(二)之动态库注入技术
C++ --- Dll文件的生成与调用(二)之动态库注入技术
164 0
|
7月前
|
并行计算 监控 Shell
openwrt编译模块demo练习
openwrt编译模块demo练习
150 0
字符块设备驱动程序框架---已测试程序hello为例
字符块设备驱动程序框架---已测试程序hello为例
79 0
|
Ubuntu 编译器 Linux
Linux驱动开发笔记(三):基于ubuntu的helloworld驱动源码编写、makefile编写以及驱动编译加载流程测试
前面学习了驱动的基础框架,上一篇编译了gcc7.3.0,那么为了方便很好的熟悉流程,本篇,将使用ubuntu18.04,直接编译ubuntu18.04的驱动,然后做好本篇文章的相关实战测试。
|
分布式计算 Java 大数据
Goland / Mac - 安装 & HelloWorld Demo
本文将安装 go 语言常用编辑器 GoLand,其与 idea、pycharm 同属JetBrains 旗下,由于突破试用的限制,下面教程主要安装 2019 版 Goland + Go 1.15.x 版本,有高版本编译器或高版本 Go 语言需求的同学可以忽略后续,如果只是入门熟悉操作可以参考下面教程。...
404 0
Goland / Mac - 安装 & HelloWorld Demo
|
C语言 iOS开发 MacOS
汇编语言版本的HelloWorld
平台 macOS 工具 nasm clang 文件 main.asm extern _printf ; 这里调用系统的一个系统调用函数, _printf, 使用extern告诉链接器该label在其他文件中有定义, _printf只接受一个string的地址, 和C语言层面上的printf是一样的 section .
1020 0

热门文章

最新文章