字符设备的注册

#include<linux/module.h>
#include<linux/kernel.h>
#include<linux/init.h>
#include<linux/fs.h>
#include<linux/slab.h>
#include<linux/timer.h>
#include<linux/sched.h>
#include<linux/list.h>
#include<linux/interrupt.h>

#include<linux/cdev.h> //cdev_init() cdev_add() cdev_del()
#include<linux/types.h> //dev_t
#include<linux/kdev_t.h> //有两个宏获取主设备号和次设备号

MODULE_AUTHOR("Tan xujia");
MODULE_LICENSE("Dual BSD/GPL");

dev_t c_devt;  //设备的主次设备号


/*定义一个结构体,包含了cdev,类似与继承,把这个设备文件的属性给继承过来*/
struct char_dev {
    struct cdev c_dev;
    char c;
};

/*定义一个变量*/
struct char_dev *cdev_p;

/*设备的操作方法*/
int
char_open(struct inode *inode, struct file *filp)
{
    printk("char_open, %d, %d\n", iminor(inode), MINOR(inode->i_cdev->dev));
    return 0;
}

ssize_t
char_read(struct file *filp, char __user *buf, size_t count, loff_t *fpos)
{
    printk("char_read\n");
    return 0;
}

ssize_t
char_write(struct file *filp, const char __user *buf, size_t count, loff_t *fpos)
{
    printk("char_write\n");
    return count;
}

int
char_release(struct inode *inode, struct file *filp)
{
    printk("char_release\n");
    return 0;
}

struct file_operations char_fops = {
    .owner = THIS_MODULE,
    .read = char_read,
    .write = char_write,
    .open = char_open,
    .release = char_release,
};


static
int __init hello_init (void)
{
    printk("hello_init\n");
    int ret = 0, index = 0;
    ret = alloc_chrdev_region(&c_devt, 10, 2, "char_dev");
    if(ret)
         printk("alloc_chrdev_region fail\n");

    printk("major = %d, minor = %d\n", MAJOR(c_devt), MINOR(c_devt));

    cdev_p = kmalloc(sizeof(struct char_dev) * 2, GFP_KERNEL);
    if(!cdev_p)
        printk("alloc fail\n");

    for(index = 0; index < 2; index++) {
        cdev_init(&cdev_p[index].c_dev, &char_fops);
        cdev_p[index].c_dev.owner = THIS_MODULE;
        ret = cdev_add(&cdev_p[index].c_dev, 
            MKDEV(MAJOR(c_devt), MINOR(c_devt) + index), 1);
        if(ret) 
            printk("cdev_add fail\n");
    }
    return 0;
}

static
void __exit hello_exit (void)
{
    printk("hello_exit\n");
    int index = 0;
    for(index = 0; index < 2; index++) {
         cdev_del(&cdev_p[index].c_dev);
    }
    kfree(cdev_p);
    unregister_chrdev_region(c_devt, 2);
}

module_init(hello_init);
module_exit(hello_exit);


/*
MAJOR(dev_t dev) //获取主设备号
MINOR(dev_t dev) //获取次设备号

MKDEV(int major, int minor) //合成设备号

设备号的分配:
int register_chrdev_region(dev_t first, unsigned int count,
    char *name)
int alloc_chrdev_region(dev_t *dev, unsigned int firstminor,
    unsigned int count, char *name)

设备号的释放:
void unregister_chrdev_region(dev_t first, unsigned int count)
*/
KERNELDIR ?= /lib/modules/$(shell uname -r)/build
PWD       := $(shell pwd)

obj-m	:=cdev_register.o

all:
	make -C $(KERNELDIR) M=$(PWD) modules

clean:
	rm -rf *.o *~ core .depend .*.cmd *.ko *.mod.* .tmp_versions *.mod *.order *.symvers *.dwo

注意代码里面的count参数给的是10,在使用mknod创建设备文件以后,就可以对相应的设备进行读写了。

sudo mknod xxx c 主设备号 次设备号

这里的次设备号可以创建多个,但是这里是两个,并且是从10开始,也就是10和11.。。

iminor(inode):可以得到相应的设备文件号

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值