linux kernel之chardev_open()详解和从用户态open到内核驱动实现流程

linux kernel之chardev_open()详解和从用户态open到内核驱动实现流程

在我们目前的Linux系统中,我们大概共约300左右个系统调用,其中syscall_table.S列出了所有的系统调用表。

在本文件中记录了所有当前平台系统中所提供的系统调用表,其中第五项就包括:

.long sys_open /* 5 */


    查看sys_open() 函数,我们看到里面所完成的工作为:
    1、查看打开的是否是大文件,如果是的话,置大文件标志位:O_LARGEFILE
    2、做do_sys_open()函数调用。
    3、检查2的调用返回值ret是否有效。
    -----------------------------

    查看do_sys_open()函数所完成的工作为:
    调用getname() ,getname函数主要功能是在使用文件名之前将其拷贝到内核数据区,正常结束时返回内核分配的空间首地址,出错时返回错误代码。
    取得系统中可用的文件描述符fd。
    调用do_filp_open()函数,此函数使用了一个数据结构nameidata来描述与文件相关的文件操作。

struct nameidata {
struct dentry *dentry; // 目录数据
struct vfsmount *mnt; // 虚拟文件挂载点数据
struct qstr last; // hash值
unsigned int flags; // 文件操作标识
int last_type; // 类型
unsigned depth;
char *saved_names[MAX_NESTED_LINKS + 1];
union {
struct open_intent open;
} intent; // 专用数据
};
-----------------------------


    struct file *do_filp_open(const char * filename, int flags, int mode){
            int namei_flags, error;
            struct nameidata nd;
            namei_flags = flags;
            if ((namei_flags+1) & O_ACCMODE)
                    namei_flags++;        // 如果flags有O_WRONLY,则增加O_RDONLY

            error = open_namei(filename, namei_flags, mode, &nd);
                            // open_namei函数主要执行文件操作的inode部分的打开等操作。
            if (!error)
                    return nameidata_to_filp (nd, flags);
                            // 把文件的inod相关信息转换成文件结构。
            return ERR_PTR(error);        // 返回错误代码
    }
    -----------------------------

    我们下面来看这个比较关键的函数:nameidata_to_filp():
    struct file *(struct nameidata *nd, int flags)
    821 {
    822        struct file *filp;
    823 
    824        /* Pick up the filp from the open intent */
    825        filp = nd->intent.open.file;
                                            // 把相关 file结构的指针赋予 filp。 
    826        /* Has the filesystem initialised the file for us? */
    827        if (filp->f_path.dentry == NULL)
    828                filp = __dentry_open(nd->dentry, nd->mnt, flags, filp, NULL);
                                            // ***** 关键函数 ***** //
    829        else
    830                path_release(nd);
    831        return filp;
    832 }
    -----------------------------

    关键函数:__dentry_open():
    static struct file *__dentry_open(struct dentry *dentry, struct vfsmount *mnt,
                                            int flags, struct file *f,
                                            int (*open)(struct inode *, struct file *))
    {
                    ......
    695        f->f_pos = 0;
    696        f->f_op = fops_get(inode->i_fop);
                                            // 在这里进行赋值,f->f_op = &def_chr_fops,注意上文inode->i_fop中的赋值。
    697        file_move(f, &inode->i_sb->s_files);
    698 
    699        if (!open && f->f_op) 
                                            // 在调用__dentry_open时open赋值为空,所以!open为真。
    700                        open = f->f_op->open;
                                            // 在这里将open赋为chrdev_open。
    701        if (open) {
    702                        error = open(inode, f);
                                            // 这里调用chrdev_open, 参照下文。
    703                        if (error)
    704                                goto cleanup_all;
                    ......
    }
    -----------------------------

    在函数chrdev_open中(/fs/char_dev.v):
    int chrdev_open(struct inode * inode, struct file * filp)
    { 
            ......
            kobj = kobj_lookup(cdev_map, inode->i_rdev, &idx); 
                                            // 执行kobj_lookup函数,在cdev_map里寻找相应的inode->i_rdev设备。
                                            // cdev_map是一个256个probe结构组成的数组,用于查找具有相应设备号的设备。
                                            // inode->i_rdev为设备号。

    new = container_of(kobj, struct cdev, kobj); 
                                            //从kobj的位置倒算出cdev的内存地址,获得包含相应kobj的cdev。

    inode->i_cdev = p = new; 
                                            // 到这里p已经为我们要的设备cdev了。

    filp->f_op = fops_get(p->ops); 
                                            / /拿到 cdev操作集。
                                            // 至此以后read,write操作都通过file->f_op直接与我们要的设备操作集挂钩了。
            ......
    }
    -----------------------------

到此,系统通过file->f_op 就与我们在设备驱动里面的定义的相关操作联系起来了,我们之前在写驱动实现的功能操作就被系统通过应用层的open 一步一步的调用到我们自己的open跟相关其他的操作了。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
好的,我已经理解了您的问题。以下是一个简单的字符设备驱动程序的示例代码,包括设备打开、读、写、I/O控制和关闭等基本操作。请注意,这只是一个示例,您需要根据自己的需求进行适当的修改。 chardev.c: #include <linux/init.h> #include <linux/module.h> #include <linux/fs.h> #include <linux/uaccess.h> #define DEVICE_NAME "chardev" //设备名称 #define BUF_LEN 80 //缓冲区大小 static int major; //主设备号 static char msg[BUF_LEN]; //缓冲区 static char *msg_ptr; //缓冲区指针 //打开设备 static int device_open(struct inode *inode, struct file *file) { static int counter = 0; if (counter) return -EBUSY; counter++; msg_ptr = msg; try_module_get(THIS_MODULE); return 0; } //释放设备 static int device_release(struct inode *inode, struct file *file) { counter--; module_put(THIS_MODULE); return 0; } //读设备 static ssize_t device_read(struct file *filp, char *buffer, size_t length, loff_t *offset) { int bytes_read = 0; if (*msg_ptr == 0) return 0; while (length && *msg_ptr) { put_user(*(msg_ptr++), buffer++); length--; bytes_read++; } return bytes_read; } //写设备 static ssize_t device_write(struct file *filp, const char *buff, size_t length, loff_t *off) { int i; for (i = 0; i < length && i < BUF_LEN; i++) get_user(msg[i], buff + i); msg_ptr = msg; return i; } //I/O控制 static long device_ioctl(struct file *filp, unsigned int cmd, unsigned long arg) { switch (cmd) { case 1: printk(KERN_INFO "IOCTL command 1\n"); break; case 2: printk(KERN_INFO "IOCTL command 2\n"); break; default: return -EINVAL; } return 0; } //设备文件操作结构体 static struct file_operations fops = { .owner = THIS_MODULE, .read = device_read, .write = device_write, .unlocked_ioctl = device_ioctl, .open = device_open, .release = device_release }; //初始化字符设备 int init_chardev_module(void) { major = register_chrdev(0, DEVICE_NAME, &fops); if (major < 0) { printk(KERN_ALERT "Registering char device failed with %d\n", major); return major; } printk(KERN_INFO "Chardev module loaded with major number %d\n", major); return 0; } //关闭字符设备 void exit_chardev_module(void) { unregister_chrdev(major, DEVICE_NAME); printk(KERN_INFO "Chardev module unloaded\n"); } module_init(init_chardev_module); module_exit(exit_chardev_module); Makefile: obj-m := chardev.o all: make -C /lib/modules/$(shell uname -r)/build M=$(PWD) modules clean: make -C /lib/modules/$(shell uname -r)/build M=$(PWD) clean 在使用前,请确定您的系统已经安装了适当的编译工具链和内核头文件。可以使用以下命令来编译和加载模块: make sudo insmod chardev.ko 您可以使用以下命令来测试设备驱动程序: echo "Hello, world!" > /dev/chardev cat /dev/chardev sudo ioctl /dev/chardev 1 希望这能帮助您了解如何编写一个简单的字符设备驱动程序。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

xx-xx-xxx-xxx

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值