字符驱动设备的另一种写法

学号:19021211263

一、register_chrdev的缺点

之前注册字符设备用的如下函数注册字符设备驱动:

register_chrdev(unsigned int major, const char *name,const struct file_operations *fops);

但其实这个函数是linux版本2.4之前的注册方式,它的原理是:

  • (1)确定一个主设备号

  • (2)构造一个file_operations结构体, 然后放在chrdevs数组中

  • (3)注册:register_chrdev

然后当读写字符设备的时候,就会根据主设备号从chrdevs数组中取出相应的结构体,并调用相应的处理函数。

它会有个很大的缺点:

  • 每注册个字符设备,还会连续注册0~255个次设备号,使它们绑定在同一个file_operations操作方法结构体上,在大多数情况下,都只用极少的次设备号,所以会浪费很多资源。

  • 最多只能有255个设备

改进后的优点:

  • 通过一定区间长度的次设备号将file_operations操作方法结构体的数量限制住
  • 通过主设备号和次设备号来查找设备,其中次设备号20位,主设备号12位,理论上最大支持4G个驱动程序

二、全新字符驱动设备的注册方法

在2.4版本后,内核里就加入了以下几个函数也可以来实现注册字符设备:

1、register_chrdev_region

/*指定设备编号来静态注册一个字符设备*/
int register_chrdev_region(dev_t from, unsigned count, const char *name);   

from: 注册的指定起始设备编号,比如:MKDEV(100, 0),表示起始主设备号100, 起始次设备号为0

count:需要连续注册的次设备编号个数,比如: 起始次设备号为0,count=100,表示0~99的次设备号都要绑定在同一个file_operations操作方法结构体上

name:字符设备名称

当返回值小于0,表示注册失败

2、alloc_chrdev_region

/*动态分配一个字符设备,注册成功并将分配到的主次设备号放入*dev里*/
int alloc_chrdev_region(dev_t *dev, unsigned baseminor, unsigned count,const char *name);

dev: 存放起始设备编号的指针,当注册成功, *dev就会等于分配到的起始设备编号,可以通过MAJOR()和MINNOR()函数来提取主次设备号

baseminor:次设备号基地址,也就是起始次设备号

count:需要连续注册的次设备编号个数,比如: 起始次设备号(baseminor)为0,baseminor=2,表示0~1的此设备号都要绑定在同一个file_operations操作方法结构体上

name:字符设备名称

当返回值小于0,表示注册失败

3、cdev_init

 /*初始化cdev结构体,并将file_operations结构体放入cdev-> ops 里*/
void cdev_init(struct cdev *cdev, const struct file_operations *fops);

其中cdev结构体的成员,如下所示:

struct cdev {
    struct kobject    kobj;            // 内嵌的kobject对象 
    struct module   *owner;            //所属模块
    const struct file_operations  *ops;//操作方法结构体
    struct list_head  list;          //与 cdev 对应的字符设备文件的 inode->i_devices 的链表头
    dev_t dev;                  //起始设备编号,可以通过MAJOR(),MINOR()来提取主次设备号
    unsigned int count;                //连续注册的次设备号个数
};

其中可以通过MAJOR(cdev->dev)MINOR(cdev->dev)来提取主次设备号,亦可以通过MKDEV(major,minor)来将主设备号major和次设备号minor转换成dev_t类型变量dev

4、cdev_add

/*将cdev结构体添加到系统中,并将dev(注册好的设备编号)放入cdev-> dev里,  count(次设备编号个数)放入cdev->count里*/
int cdev_add(struct cdev *p, dev_t dev, unsigned count);

5、cdev_del

/*将系统中的cdev结构体删除掉*/
void cdev_del(struct cdev *p);

6、unregister_chrdev_region

 /*注销字符设备*/
void unregister_chrdev_region(dev_t from, unsigned count);

from: 注销的指定起始设备编号,比如:MKDEV(100, 0),表示起始主设备号100, 起始次设备号为0

count:需要连续注销的次设备编号个数,比如: 起始次设备号为0,baseminor=100,表示注销掉0~99的次设备号

三、编写全新的字符驱动设备

通过调用上面的函数,构造两个不同的file_operations操作结构体,

次设备号0~1对应第一个file_operations,

次设备号2~3对应第二个file_operations,

然后在/dev/下,通过次设备号(0~4)创建5个设备节点, 利用应用程序打开这5个文件,看有什么现象

驱动代码

#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <linux/init.h>
#include <linux/delay.h>
#include <asm/irq.h>
#include <asm/arch/regs-gpio.h>
#include <asm/hardware.h>
#include <asm/uaccess.h>
#include <asm/io.h>
#include <linux/list.h>
#include <linux/cdev.h>

static int hello_fops1_open(struct inode *inode, struct file *file)
{
    printk("open_hello1!!!\n");
    return 0;
}

static int hello_fops2_open (struct inode *inode, struct file *file)
{
    printk("open_hello2!!!\n");
    return 0;
}


/*  操作结构体1   */
static struct file_operations hello1_fops={
    .owner=THIS_MODULE,
    .open =hello_fops1_open,
};

/*  操作结构体2   */
static struct file_operations hello2_fops={
    .owner=THIS_MODULE,
    .open =hello_fops2_open,
};


static int major;              //主设备
static struct cdev hello1_cdev;//保存 hello1_fops操作结构体的字符设备 
static struct cdev hello2_cdev;//保存 hello2_fops操作结构体的字符设备 
static struct class *cls;

static int chrdev_ragion_init(void)
{
    dev_t  devid;  

    alloc_chrdev_region(&devid, 0, 4,"hello");//动态分配字符设备: (major,0)、(major,1)、(major,2)、(major,3)

    major=MAJOR(devid);

    cdev_init(&hello1_cdev, &hello1_fops);
    cdev_add(&hello1_cdev, MKDEV(major,0), 2);//(major,0) (major,1)


    cdev_init(&hello2_cdev, &hello2_fops);
    cdev_add(&hello2_cdev,MKDEV(major,2), 2);//(major,2) (major,3)     

    cls=class_create(THIS_MODULE, "hello");
    /*创建字符设备节点*/
    class_device_create(cls,0, MKDEV(major,0), 0, "hello0");//对应hello_fops1操作结构体
    class_device_create(cls,0, MKDEV(major,1), 0, "hello1");//对应hello_fops1操作结构体
    class_device_create(cls,0, MKDEV(major,2), 0, "hello2");//对应hello_fops2操作结构体
    class_device_create(cls,0, MKDEV(major,3), 0, "hello3");//对应hello_fops2操作结构体
    class_device_create(cls,0, MKDEV(major,4), 0, "hello4");//对应空
    return 0;
}

void chrdev_ragion_exit(void)
{
    class_device_destroy(cls, MKDEV(major,4));
    class_device_destroy(cls, MKDEV(major,3));
    class_device_destroy(cls, MKDEV(major,2));
    class_device_destroy(cls, MKDEV(major,1));
    class_device_destroy(cls, MKDEV(major,0));

    class_destroy(cls);


    cdev_del(&hello1_cdev);     
    cdev_del(&hello2_cdev); 
    unregister_chrdev_region(MKDEV(major,0), 4);//注销(major,0)~(major,3)
}

module_init(chrdev_ragion_init);
module_exit(chrdev_ragion_exit);
MODULE_LICENSE("GPL");

测试代码

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
//打印使用帮助信息
void print_useg(char arg[])
{
    printf("useg:  \n");
    printf("%s   [dev]\n",arg);
}

int main(int argc,char **argv)
{
    int fd;
    if(argc!=2)
    {
        print_useg(argv[0]);
        return -1;
    }

    fd=open(argv[1],O_RDWR);
    if(fd<0)
        printf("can't open %s \n",argv[1]);
    else
        printf("can open %s \n",argv[1]);
    return 0;
}

运行测试

如下所示,挂载驱动后,通过 ls /dev/hello* -l ,看到创建了5个字符设备节点

# ls /dev/hello* -l
crw-rw----    1 0     0     252, 0 Jan 1 00: 06/dev/hello0
crw-rw----    1 0     0     252, 1 Jan 1 00: 06/dev/hello1
crw-rw----    1 0     0     252, 2 Jan 1 00: 06/dev/hello2
crw-rw----    1 0     0     252, 3 Jan 1 00: 06/dev/hello3
crw-rw----    1 0     0     252, 4 Jan 1 00: 06/dev/hello4

接下来开始测试驱动,如下图所示,

打开/dev/hello0时,调用的是驱动代码的操作结构体hello1_fops里的.open(),

打开/dev/hello2时,调用的是驱动代码的操作结构体hello2_fops里的.open(),

打开/dev/hello4时,打开无效,因为在驱动代码里没有分配次设备号4的操作结构体,

# ./20th_chrdev_region_test /dev/hello0
open_hello1!!!
can open /dev/hello0
#
#
# ./20th_chrdev_region_test /dev/hello2
open_hello2!!!
can open /dev/hello2
#
#
# ./20th_chrdev_region_test /dev/hello4
can't open /dev/hello4

总结

使用register_chrdev_region()等函数来注册字符设备,里面可以存放多个不同的file_oprations操作结构体,实现各种不同的功能

更多有趣内容欢迎访问我的个人博客

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352