一、驱动相关概念
1.操作系统的功能
向下管理硬件,向上提供接口
操作系统向上提供的接口类型:
内存管理:内存申请(malloc) 内存释放(free)等
文件管理: 通过文件系统格式对文件ext2、ext3、 ext4格式进行管理
进程管理: 进程的创建/调度/消亡
网络管理: 通过网络栈协议,完成数据的收发
设备管理: 字符设备、块设备、网卡设备
2.驱动的概念
2.1.定义
驱动是能够让硬件实现某个特定功能的软件代码,根据驱动代码是否使用了系统内核提供的接口,将驱动分为裸机驱动和系统驱动
2.2.裸机驱动和系统驱动
裸机驱动:不基于操作系统提供的接口完成驱动的编写,这种驱动比较简单,由开发者独立即可编写完成,但是实现的功能比较单一
系统驱动(操作系统设备驱动):编写驱动时调用系统内核提供的接口,驱动程序会被加载到内核空间。系统驱动开发者无法独立完成,需要操作系统内核的辅助,但是基于系统驱动完成的硬件控制工作会更加完善和复杂。
3.系统驱动(操作系统设备驱动)在操作系统中的层次
设备驱动会被加载到系统内核空间,设备驱动主要完成对硬件功能的实现,相当于拓展了系统内核设备管理的功能
4.设备驱动的类型
设备驱动的类型时根据设备类型的不同进行划分的,一般将设备分为字符设备、块设备、和网卡设备 三类:
字符设备:以字节流的形式进行顺序访问的设备叫做字符设备。 90%的设备为字符设备,鼠标、键盘、摄像头...
块设备:以块为单位进行随机访问的设备叫做块设备。块设备一般是一些磁盘设备
网卡设备:用于进行网络数据传输的设备。网卡设备没有网卡设备文件,想要读取网卡设备的数据,需要基于socket套接字实现
二、内核模块编程
1.内核模块编程的意义
驱动属于内核的一部分,驱动资源要加载到内核中,所以要按照内核模块的编程去编写框架
2.内核模块的三要素
入口:主要进行资源的申请工作,安装内核模块时执行
出口:主要进行资源的释放工作,卸载内核模块时执行
许可证:声明当前内核模块遵循GPL协议
3.内核模块编写实例及代码解释
#inckude <linux/init.h>
#include <linux/module.h>
//入口函数:安装内核模块时执行
static int __init mycdev_init(void)
{
//static: 修饰当前函数只可在本文件使用
//int:函数返回值类型,如果函数规定了返回值但是没有加返回值,编译会报错
//mycdev_init:函数名,可以自己命名
//void:表示函数无参数,void不可以省略
//__init:告诉编译器当前函数保存在.init.text段 #define __init sectoin(".init.text")
//linux内核中有自己的链接脚本文件 vmlinux.lds,这个链接脚本文件规定了不同的内容存放在内存中的哪个位置
return 0;
}
//出口函数:卸载内核模块时执行
static void __exit mycdev_exit(void)
{
//__exit:告诉编译器当前函数保存在.exit.text段 #define __exit section(".exit.text")
}
//声明入口函数
module_init(mycdev_init);
//声明出口函数
module_exit(mycdev_exit);
//声明当前内核模块遵循GPL协议
MODULE_LICENSE("GPL");
4.内核模块的编译
编译内核镜像:make uImage
编译设备树:make dtbs
模块化编译:make modules
内核模块编译有两种方式:内部编译和外部编译
4.1内部编译
内部编译又称为静态编译,需要依赖于内核的源码树:
① 编写内核模块
② 将内核模块文件移动到内核指定路径下
③ 在内核指定路径下的Makefile文件中添加编译文件
④ 修改Kconfig文件,添加当前内核模块文件的菜单选项(选配项)
⑤ 在源码顶层目录下make menuconfig将当前文件的选配项选配为M,保存退出,此时.config被修改
⑥ make modules 进行模块化编译
4.2.外部编译
外部编译又称为动态编译,可以在内核路径外单独编译当前模块文件,外部编译需要我们自己手写Makefile
通用版本的Makefile
modname ?= demo #内核模块名称,询问赋值,若无对应的外部变量,则为demo,有则为外部变量
arch ?= arm #架构,询问赋值,若无对应的外部变量,则为arm,有则为外部变量
ifed ($(arch),arm) #通过命令行是否有对应的外部变量来判断采用什么架构进行编译
#KERNRLDIR保存开发板内核源码路径
KERNRLDIR := /home/ubuntu/linux.5.10.61/
else
#KERNRLDIR保存ubuntu内核源码路径
KERNRIDIR :=/lib/kernel/$(shell uname -r)/bulid #uname -r命令是查找当前操作系统使用的内核版本
endif
#PWD保存当前内核模块的路径
PWD := $(shell pwd)
all:
#make modules是模块化编译的命令
#-C $(KERNELDIR) 表示执行make前到$(KERNELDIR)所保存路径下,读取该目录下的Makefile文件执行make编译
#M=$(PWD) 指定模块化编译的路径,进行模块化编译的路径是PWD保存的路径
make -C $(KERNELDIR) M=$(PWD) modules
clean:
#编译删除
make -C $(KERNELDIR) m=$(PWD) clean
#将obj-m保存的文件单独链接生成内核模块
obj-m := $(modname).o #此处写.o不写.c,虽然当前目录下没有.o文件,但是Makefile有自动推导功
能,会自动寻找对应.c
5.内核模板操作相关命令
安装内核模块命令 insmod *.ko
卸载内核模块命令 rmmod *.ko (后缀名.ko可省略)
查看内核已经安装的内核模块 lsmod
查看某个内核模块的信息 modinfo *.ko
三、内核消息打印函数printk函数的使用
1.printk函数的使用格式
printk("格式控制符",输出列表)
这种格式消息按照默认输出级别进行输出
或
printk(消息级别 "格式控制符",输出列表)
2.printk函数消息队列打印级别
prink函数打印的内容属于内核消息。内核消息根据消息的轻重缓急给他们设置不同的消息级别。终端也会存在一个默认的消息级别,只有消息级别高于终端默认消息级别,消息才可以在终端进行输出,消息级别共分为0-7总共8级,数字越小代表消息级别越高,一般常用3-7级
#define KERN_EMERG KERN_SOH "0" /* system is unusable */
#define KERN_ALERT KERN_SOH "1" /* action must be taken immediately */
#define KERN_CRIT KERN_SOH "2" /* critical conditions */
#define KERN_ERR KERN_SOH "3" /* error conditions */
#define KERN_WARNING KERN_SOH "4" /* warning conditions */
#define KERN_NOTICE KERN_SOH "5" /* normal but significant condition */
#define KERN_INFO KERN_SOH "6" /* informational */
#define KERN_DEBUG KERN_SOH "7" /* debug-level messages */
3.查看消息默认级别
消息默认级别保存在/proc/sys/kernel/printk文件
查看cat /proc/sys/kernel/printk
4 4 1 7
4 终端默认消息级别
4 printk默认的消息级别
1 终端支持的消息最高级别
7 终端支持的消息最低级别
代码示例
#include <linux/init.h>
#include <linux/module.h>
// 入口函数,安装内核模块时执行
static int __init mycdev_init(void)
{
// static 修饰当前函数只能在本文件使用
// int 函数的返回值类型,如果函数规定返回值但是没有加返回值,编译会报错
// mycdev_init函数名,可以自己起名字
// void表示函数无参数,当没有参数时void一定要加,不然报错
//__init的作用是用来告诉编译器当前代码保存在.init.text段中
// #define __init __section(".init.text")
// linux内核也会有自己的链接脚本 vmlinux.lds,这个链接脚本里规定了不同的内容在
// 内存中的什么位置
printk(KERN_ERR "hello world\n");
int a=10;
printk(KERN_ERR "%d\n",a);
return 0;
}
// 出口函数,卸载内核模块时执行
static void __exit mycdev_exit(void)
{
// #define __exit __section(".exit.text")
//__exit指定出口函数保存在.exit.text段中
}
// 用于声明入口函数
module_init(mycdev_init);
// 用于声明出口函数
module_exit(mycdev_exit);
// 声明当前内核模块遵循GPL协议
MODULE_LICENSE("GPL");
5.如何修改消息默认级别
ubuntu:
先切换到管理员:sudo su
再输入以下内容:echo 4 3 1 7 > /proc/sys/kernel/printk
注:这种修改方式ubuntn重启后,级别恢复默认设置的4 4 1 7
开发板上linux
进入根文件系统下的etc下的init.d文件夹:cd ~/nfs/rootfs/etc/init.d
打开rcS脚本文件:vi rcS
在rcS中最后一行添加:echo 4 3 1 7 > /proc/sys/kernel/printk
6.ubuntu虚拟终端的使用
切换到虚拟终端:ctrl+alt+[f2-f6](fn)
退出虚拟终端:ctrl+alt+f1(fn)
7.dmesg命令的使用
dmesg:查看内核打印消息
dmesg -c:先将保存的内核消息打印在终端再清除
dmesg -C:直接清除