2643|1

58

帖子

0

TA的资源

一粒金砂(中级)

楼主
 

从用户态的open到内核驱动实现流程 [复制链接]

作者:李强,华清远见嵌入式学院讲师。

转载自http://www.embedu.org/Column/Column338.htm

问题来源:

在讲授Linux初级驱动的时候,我发现困惑很多同学的是不真正理解从应用层到我们自己所写的驱动层的调用过程,所以写此文章来大概描述。

首先我们知道,在我们目前的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跟相关其他的操作了。

此帖出自ARM技术论坛

最新回复

恩恩,整理的很详细啊,之前对于从应用层到我们自己所写的驱动层的调用过程。还是很模糊。慢慢了解了  详情 回复 发表于 2012-4-23 15:37
点赞 关注
 

回复
举报

547

帖子

0

TA的资源

纯净的硅(初级)

沙发
 
恩恩,整理的很详细啊,之前对于从应用层到我们自己所写的驱动层的调用过程。还是很模糊。慢慢了解了
此帖出自ARM技术论坛
 
个人签名嵌入式培训www.51great.org
 
 

回复
您需要登录后才可以回帖 登录 | 注册

随便看看
查找数据手册?

EEWorld Datasheet 技术支持

相关文章 更多>>
关闭
站长推荐上一条 1/8 下一条

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

About Us 关于我们 客户服务 联系方式 器件索引 网站地图 最新更新 手机版

站点相关: 国产芯 安防电子 汽车电子 手机便携 工业控制 家用电子 医疗电子 测试测量 网络通信 物联网

北京市海淀区中关村大街18号B座15层1530室 电话:(010)82350740 邮编:100190

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2025 EEWORLD.com.cn, Inc. All rights reserved
快速回复 返回顶部 返回列表