Linux 应用层open调用驱动层open过程
内核版本: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过程的更多相关文章
- DeviceIoControl 应用层如何和驱动层通信?
调用的方法之一的DeviceIoControl 驱动层提供设备名 例如filedisk 在驱动层 首先先是注册列表 用winObj查看 filedisk的驱动对象 但是 这八个对象时怎么生成的呢? 我 ...
- android和linux开发环境建立(驱动层)
流程:安装ubutu14.04操作系统==>安装各种库和应用程序并配置环境变量 1,install ubuntu14.04 为了完全释放PC机的资源,我们安装在主机上,就不用虚拟机来玩了.下面是 ...
- Linux的应用层到底层驱动的调用过程
应用层如何内核.md 1.从应用层打通内核:驱动 首先来说是设备号的引入,我们通过 cat/proc/kallsyms |grep mydevice 可以查看设备号,当然我们也是可以自己创建设备号,这 ...
- Linux 网卡驱动学习(六)(应用层、tcp 层、ip 层、设备层和驱动层作用解析)
本文将介绍网络连接建立的过程.收发包流程,以及当中应用层.tcp层.ip层.设备层和驱动层各层发挥的作用. 1.应用层 对于使用socket进行网络连接的server端程序.我们会先调用socket函 ...
- Minifilter微过滤框架:框架介绍以及驱动层和应用层的通讯
minifilter是sfilter后微软推出的过滤驱动框架.相比于sfilter,他更容易使用,需要程序员做的编码更简洁. 系统为minifilter专门制作了一个过滤管理器,这个管理器本身其实是一 ...
- 2、CC2541芯片中级教程-OSAL操作系统(进一步了解-OLED && 普通按键和5方向按键-中断!!!)这个系统驱动层和应用层不一样~
本文根据一周CC2541笔记汇总得来—— 适合概览和知识快速索引—— 全部链接: 中级教程-OSAL操作系统\OSAL操作系统-实验01 OSAL初探 [插入]SourceInsight-工程建立方法 ...
- Linux 驱动层实现阻塞和非阻塞
linux应用层的函数默认是阻塞型的,但是要想真正实现阻塞,还需要驱动的支持才行. 例:open().scanf().fgets().read().accept() 等 1.默认情形,驱动层不实现阻塞 ...
- 应用层open(read、write、close)怎样调用驱动open(read、write、close)函数的?
应用层open(read.write.close)怎样调用驱动open(read.write.close)函数的? 华清远见2014-09-29 北京海淀区 张俊浩 三大数据结构关系图
- 迅为4412开发板Linux驱动教程——编写简单应用调用驱动
Linux驱动教程:http://pan.baidu.com/s/1c0hljUS 编写简单应用调用驱动--头文件 • 打印头文件 – include <stdio.h>调用打印函数pri ...
随机推荐
- Effective C++ 条款02:尽量以const,enum,inline替换 #define
换一种说法就是宁可以编译器替换预处理器 举例 #define ASPECT_RATIO 1.653 记号ASPECT_RATIO也许从未被编译器看见:也许在编译起开始处理源码前它就被预处理器移走了,于 ...
- 浅谈MySQL中优化sql语句查询常用的30种方法 - 转载
浅谈MySQL中优化sql语句查询常用的30种方法 1.对查询进行优化,应尽量避免全表扫描,首先应考虑在 where 及 order by 涉及的列上建立索引. 2.应尽量避免在 where 子句中使 ...
- QSS设置字体不起作用
正确的QSS QLabel{ font-family: "Microsoft YaHei"; font-weight:bold; font-size:14px; color: #3 ...
- Entity Framework 6 预热、启动优化
虽然文章题目是针对EF的,但涉及的内容不仅仅是EF. 场景介绍 目前在做的一个项目,行业门户,项目部分站点按域名划分如下: user.xxx.com:用户登陆注册 owner.xxx.com:个人用户 ...
- Ajax-04 jQuery Ajax 常用操作
jQuery jQuery 其实就是一个JavaScript的类库,其将复杂的功能做了上层封装,使得开发者可以在其基础上写更少的代码实现更多的功能. jQuery Ajax a.概述 jQuery 不 ...
- jenkins构建中的除零错误
一. 除零错误(ZeroDivisionError) 今天在jenkins上运行接口自动化测试任务时,从控制台输出中看到了除零错误,大概是这样的 从上图中,通过分析,可以得出三个结论: 1. jenk ...
- IOS-将任意对象存进数据库
model // // HMShop.h // 将任意对象存进数据库 // // Created by apple on 14/11/20. // Copyright (c) 2014年 heima. ...
- 【51nod-1183】编辑距离
链接:https://www.51nod.com/onlineJudge/questionCode.html#!problemId=1183 #include <bits/stdc++.h> ...
- 你不知道的sticky
position:sticky,Chrome新版本已经做了支持.sticky的中文翻译是“粘性的”,position:sticky表现也符合这个粘性的表现.基本上,可以看出是position:rela ...
- android代码设置EditText只输入数字、字母
如何设置EditText,使得只能输入数字或者某些字母呢? 一.设置EditText,只输入数字: 方法1:直接生成DigitsKeyListener对象就可以了. et_1.setKeyLis ...