声明
以下都是我刚开始看驱动视频的个人强行解读,如果有误请指出,共同进步。
本节目标
静态方式申请主次设备号
动态申请主次设备号的函数
上节我们静态申请,这一节我们用动态申请
// 动态申请主次设备号(linux分配)
int alloc_chrdev_region(dev_t *dev,unsigned int -firstminor,unsigned int -count,char *name);
参数和静态的差不多
参数1: dev_t* 类型的指针
参数2:第一个次设备号的编号
参数3:申请的次设备号的数量
参数4:设备的名字
这一节几乎与上节相同,所以直接上代码,看代码就很清楚了
#include <linux/init.h>
#include <linux/module.h>
// 字符设备申请函数
#include <linux/fs.h>
// 设备号数据类型 dev_t
#include <linux/cdev.h>
// 处理宏定义 MKDEV
#include <linux/kdev_t.h>
MODULE_LICENSE("Dual BSD/GPL");
MODULE_AUTHOR("MrYang");
// 初始化主次设备号
int device_major = 0;
int device_minor = 0;
static int mryang_init(void)
{
int ret;
dev_t mryang_dev;
// 输出
printk(KERN_EMERG "HELLO MrYang\n");
// 申请主设备号
ret = alloc_chrdev_region(&mryang_dev, device_minor, 2, "mryang_acdev");
if(ret < 0)
printk(KERN_EMERG "failed!\n");
else
printk(KERN_EMERG "success!\n");
// 提取主、次设备号
device_major = MAJOR(mryang_dev);
device_minor = MINOR(mryang_dev);
printk(KERN_EMERG "major: %d, minor: %d\n", device_major, device_minor);
return 0;
}
static void mryang_exit(void)
{
dev_t mryang_dev;
// 输出
printk(KERN_EMERG "Bye MrYang\n");
// 注销
mryang_dev = MKDEV(device_major, device_minor);
unregister_chrdev_region(mryang_dev, 2);
printk(KERN_EMERG "over!\n");
}
module_init(mryang_init);
module_exit(mryang_exit);
编译完成后,
输入:
insmod probe_linux_module.ko
返回:
[ 40.115715] HELLO MrYang
[ 40.116806] success!
[ 40.118974] major: 248, minor: 0
结果正确,linux自动分配的主设备号是248,次设备号是0
输入:
cat /proc/devices
返回:
248 mryang_acdev
我们查看linux已注册的设备,反向我们的设备已经“登记在册”。
至于卸载是一样的,就不说了。
我理解的与上节的区别是:
- 静态方式需要自己指定主次设备号,然后MKDEV宏合成为一个dev_t类型的参数,再把这个值传给linux去注册。当然,你如果指定的0,那么linux会自动给你分配一个主次设备的号码,但本质还是你把号码传给linux去注册。
- 动态方式则是通过函数去申请,linux把申请好的主次设备号返回给你。你通过MAJOR和MINOR去解析主设备号和次设备号是多少。
虽然两者看上去好像差别不大,但是我们还是要注意本质区别,你传给他申请叫静态,linux传给你是动态。(说的是这个方向,虽然静态里的号是由linux分配的,但他仍然是 静态)
收尾
这两节主要讲的就是静态指定编号or动态分配编号,去注册设备。