内核版本:3.0.8

open、close、read、write、ioctl等等都是类似。

==========================================================================================

驱动层:

struct file_operations fops =
{
.open = xxx_open,
.release = xxx_close,
.read = xxx_read,
.write = xxx_write,
...
};
struct cdev {
struct module *owner;
const struct file_operations *ops; //文件操作对象
struct list_head list;
dev_t dev; //设备号
...
};

// 申请设备号,创建文件操作结构体 struct file_operations fops

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

// 创建字符设备对象结构体 struct cdev,将 fops 赋值给 cdev

struct cdev *cdev;

cdev->ops = fops;

// 将设备号赋值给 cdev, 将 cdev 注册到链表cdev_map

int cdev_add(struct cdev *p, dev_t dev, unsigned count);

p->dev = dev;

kobj_map(cdev_map, dev, count, NULL, exact_match, exact_lock, p);

// cdev 结构体中包含 file_operations 和 设备号 dev_t。

// 创建字符设备类

struct class *__class_create(struct module *owner, const char *name, struct lock_class_key *key);

// 在类中创建一个设备, /dev 目录下会生成一个文件,名称为 led

// 系统自动为此文件分配 inode 节点(linux系统是通过 inode 节点来查找文件),节点中包含此字符设备的设备号 dev_t

struct device *device_create(struct class *class, struct device *parent, dev_t devt, void *drvdata,"led");

// 每一个文件都有一个inode
struct inode {
umode_t i_mode;//就是权限
uid_t i_uid; //用户
gid_t i_gid; //用户组
dev_t i_rdev; //设备号
loff_t i_size; //大小
... ...
}

==========================================================================================

应用层:

// 打开驱动中创建的设备,open 会调用VFS层中的 sys_open();

int open("/dev/led", int flags, mode_t mode);

// 调用 sys_open() 的过程,详见 http://blog.csdn.net/hxmhyp/article/details/22699669 ,感谢这位朋友细心讲解

SYSCALL_DEFINE3(open, const char __user *, filename, int, flags, int, mode);

==========================================================================================

VFS层:

// open会调用内核中的 sys_open(),这个函数是经过变换得来的,参数与open完全吻合

// 参数1:函数名,

// 参数2:文件名类型

// 参数3:文件名

// 参数4:打开标志类型

// 参数5:打开文件标志

// 参数6:文件权限类型

// 参数7:文件权限

int sys_open(open, const char __user *, filename, int, flags, int, mode)

// 每个文件都对应一个 struct file 结构体,打开一个文件,系统都会将 struct file 添加到 struct fdtable 的数组中

struct fdtable {
unsigned int max_fds;
struct file __rcu **fd; /* current fd array */
fd_set *close_on_exec;
fd_set *open_fds;
struct rcu_head rcu;
struct fdtable *next;
};

// 返回的文件描述符就是此文件的 struct file 在数组中的下标,从3开始(stdin-0 stdout-1 stderr-2)

struct file {
struct path f_path; // 文件路径
const struct file_operations *f_op; // 文件操作对象
unsigned int f_flags; //文件标志
fmode_t f_mode; // 文件权限
loff_t f_pos; // 文件当前偏移量
void *private_data; // 私有数据
... ...
}

// 通过 struct file 中的文件路径 f_path 得到文件的 inode 节点

long do_sys_open(int dfd, const char __user *filename, int flags, int mode)

static inline void fsnotify_open(struct file *file)

struct path *path = &file->f_path;

struct inode *inode = path->dentry->d_inode;

// 通过 inode 节点获取文件设备号

// 遍历 cdev 链表,获取设备号,与此文件的设备号相同表示匹配成功

// 将匹配成功的 cdev 结构体中的 file_operations 赋值给此文件的 struct file,从而关联到驱动层中的 xxx_open

==========================================================================================

// 应用层	--->   	驱动层
// open ---> xxx_open
// close ---> xxx_open
// read ---> xxx_open
// write ---> xxx_open

ps:以上的缩进都表示包含于上一个函数中之内。

Linux 应用层open调用驱动层open过程的更多相关文章

  1. DeviceIoControl 应用层如何和驱动层通信?

    调用的方法之一的DeviceIoControl 驱动层提供设备名 例如filedisk 在驱动层 首先先是注册列表 用winObj查看 filedisk的驱动对象 但是 这八个对象时怎么生成的呢? 我 ...

  2. android和linux开发环境建立(驱动层)

    流程:安装ubutu14.04操作系统==>安装各种库和应用程序并配置环境变量 1,install ubuntu14.04 为了完全释放PC机的资源,我们安装在主机上,就不用虚拟机来玩了.下面是 ...

  3. Linux的应用层到底层驱动的调用过程

    应用层如何内核.md 1.从应用层打通内核:驱动 首先来说是设备号的引入,我们通过 cat/proc/kallsyms |grep mydevice 可以查看设备号,当然我们也是可以自己创建设备号,这 ...

  4. Linux 网卡驱动学习(六)(应用层、tcp 层、ip 层、设备层和驱动层作用解析)

    本文将介绍网络连接建立的过程.收发包流程,以及当中应用层.tcp层.ip层.设备层和驱动层各层发挥的作用. 1.应用层 对于使用socket进行网络连接的server端程序.我们会先调用socket函 ...

  5. Minifilter微过滤框架:框架介绍以及驱动层和应用层的通讯

    minifilter是sfilter后微软推出的过滤驱动框架.相比于sfilter,他更容易使用,需要程序员做的编码更简洁. 系统为minifilter专门制作了一个过滤管理器,这个管理器本身其实是一 ...

  6. 2、CC2541芯片中级教程-OSAL操作系统(进一步了解-OLED && 普通按键和5方向按键-中断!!!)这个系统驱动层和应用层不一样~

    本文根据一周CC2541笔记汇总得来—— 适合概览和知识快速索引—— 全部链接: 中级教程-OSAL操作系统\OSAL操作系统-实验01 OSAL初探 [插入]SourceInsight-工程建立方法 ...

  7. Linux 驱动层实现阻塞和非阻塞

    linux应用层的函数默认是阻塞型的,但是要想真正实现阻塞,还需要驱动的支持才行. 例:open().scanf().fgets().read().accept() 等 1.默认情形,驱动层不实现阻塞 ...

  8. 应用层open(read、write、close)怎样调用驱动open(read、write、close)函数的?

    应用层open(read.write.close)怎样调用驱动open(read.write.close)函数的? 华清远见2014-09-29   北京海淀区 张俊浩 三大数据结构关系图

  9. 迅为4412开发板Linux驱动教程——编写简单应用调用驱动

    Linux驱动教程:http://pan.baidu.com/s/1c0hljUS 编写简单应用调用驱动--头文件 • 打印头文件 – include <stdio.h>调用打印函数pri ...

随机推荐

  1. 用maven创建web工程

    1.打开eclipse,选择File->New->Other菜单,弹出下面的对话框,在Wizards中输入maven,会过滤出和maven相关的菜单,选中Maven Project菜单,然 ...

  2. HTML5 canvas之基础篇(一)

    一.检测浏览器是否支持canvas if( !canvas || !canvas.getContext){ return; } 也可以使用modernizr.js库,Modernizr是一个易用的轻量 ...

  3. xpath选择器简介及如何使用

    xpath选择器简介及如何使用 一.总结 一句话总结:XPath 的全称是 XML Path Language,即 XML 路径语言,它是一种在结构化文档(比如 XML 和 HTML 文档)中定位信息 ...

  4. python下的复杂网络编程包networkx的使用(摘抄)

    原文:http://blog.sciencenet.cn/home.php?mod=space&uid=404069&do=blog&classid=141080&vi ...

  5. Linux命令详解-ls

    Ls(list)命令是linux下最常用的命令.ls命令就是list的缩写缺省下ls用来打印出当前目录的清单如果ls指定其他目录那么就会显示指定目录里的文件及文件夹清单.通过ls 命令不仅可以查 ...

  6. Mybatis输入和输出映射(#{}和${}的区别)

    #{}占位符                                                                                              ...

  7. split方法的使用

    // 分隔竖线 String[] param = text.split("\\|");   //分隔问号 String name = singleResource.get(&quo ...

  8. 配置管理puppet

    目录: 1.功能 2.服务器 3.管理对应配置文件 4.脚本相关 5.puppet配置相关 6.puppet客户端安装脚本 7.主机配置文件新增节点详细 8.新增文件项目 9.puppet管理命令   ...

  9. 《The Cg Tutorial》阅读笔记——动画 Animation

    这段时间阅读了英文版的NVidia官方的<The Cg Tutorial>,借此来学习基本的图形学知识和着色器编程. 在此做一个阅读笔记. 本文为大便一箩筐的原创内容,转载请注明出处,谢谢 ...

  10. Oracle中的填充函数lpad和rpad的用法(转)

    原文链接:http://blog.csdn.net/myzhanglt/article/details/7392999 今日学习遇到一个不熟悉的函数LPAD,查了一下文档,将其用法总结如下: Lpad ...