Linux 字符设备驱动模型
一。使用字符设备驱动程序
1. 编译/安装驱动
在Linux系统中,驱动程序通常采用内核模块的程序结构来进行编码。因此,编译/安装一个驱动程序,其实质就是编译/安装一个内核模块
2. 创建设备文件
通过字符设备文件,应用程序可以使用相应的字符设备驱动程序来控制字符设备。
创建字符设备文件的方法一般有两种:
1.使用mknod命令mknod/dev/文件名c 主设备号 次设备号
查询设备号的命令 cat /proc/devices
2. 使用函数在驱动程序中创建(后续课程介绍)
编写应用程序时,使用命令 arm-linux-readelf -d write_mem
3. 访问设备
编写应用程序访问设备。
编写了write_mem.c 和read_mem.c 访问文件为 /dev/memdev0 驱动为memdev
write_mem.c
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h> int main()
{
int fd = ;
int src = ;
fd = open("/dev/memdev0",O_RDWR); write(fd,&src,sizeof(int));
close(fd); return ; }
read_mem.c
int main()
{
int fd = ;
int dst = ; fd = open("dev/memdev0",O_RDWR); read(fd,&dst,sizeof(int)); printf("dst is %d\n",dst); }
二。 字符设备驱动模型
1. 设备描述结构 cdev
在任何一种驱动模型中,设备都会用内核中的一种结构来描述。
我们的字符设备在内核中使用struct cdev来描述。
struct cdev
{
structkobjectkobj;
structmodule *owner;
const structfile_operations*ops; //设备操作集
structlist_headlist;
dev_tdev; //设备号
unsigned intcount; //设备数
};
(1).字符设备文件与字符驱动程序如何建立起对应关系??
答案:主设备号
驱动程序什么来区分串口1和串2
答案:次设备号
(2).Linux内核中使用dev_t类型来定义设备号,dev_t这种类型其实质为32位的unsigned int,
其中高12位为主设备号,低20位为次设备号.
问1:如果知道主设备号,次设备号,怎么组合成dev_t类型
答:dev_t dev = MKDEV(主设备号,次设备号)
问2: 如何从dev_t中分解出主设备号?
答: 主设备号= MAJOR(dev_t dev)
问3: 如何从dev_t中分解出次设备号?
答: 次设备号=MINOR(dev_t dev)
(3).如何为设备分配一个主设备号?
#静态申请开发者自己选择一个数字作为主设备号,然后通过函数register_chrdev_region向内核申请使用。
缺点:如果申请使用的设备号已经被内核中的其他驱动使用了,则申请失败。使用的设备号已经被内核中的其他驱动使用了,则申请失败。
#动态分配使用alloc_chrdev_region由内核分配一个可用的主设备号。优点:因为内核知道哪些号已经被使用了,所以不会导致分配到已经被使用的号。
(4).不论使用何种方法分配设备号,都应该在驱动退出时,使用unregister_chrdev_region函数释放这些设备号
(5). Struct file_operations是一个函数指针的集合,定义能在设备上进行的操作。
结构中的函数指针指向驱动中的函数, 这些函数实现一个针对设备的操作, 对于不支持的操作则设置函数指针为NULL。
例如:struct file_operations dev_fops =
{
.llseek= NULL,
.read= dev_read,
.write= dev_write,
.ioctl= dev_ioctl,
.open= dev_open,
.release= dev_release,
};
2. 字符设备驱动模型
1. 驱动初始化
2.1.1 分配设备描述结构
cdev变量的定义可以采用静态和动态两种办法
•静态分配struct cdev mdev;
•动态分配struct cdev *pdev = cdev_alloc();
2.1.2 初始化设备描述结构
struct cdev的初始化使用cdev_init函数来完成。
cdev_init(struct cdev *cdev, const struct file_operations *fops)
参数:cdev: 待初始化的cdev结构
fops: 设备对应的操作函数集
2.1.3 注册设备描述结构
字符设备的注册使用cdev_add函数来完成。
cdev_add(struct cdev *p, dev_t dev, unsigned count)
参数:p: 待添加到内核的字符设备结构
dev: 设备号
count: 该类设备的设备个数
2.1.4 硬件初始化
2. 实现设备操作
1. open
int (*open) (struct inode *, struct file *)打开设备,响应open系统
open设备方法是驱动程序用来为以后的操作完成初始化准备工作的。
在大部分驱动程序中,open完成如下工作:
#标明次设备号
#启动设备
2. read
ssize_t (*read) (struct file *, char __user *, size_t, loff_t *)从设备读取数据,响应read系统调用
read设备方法通常完成2件事情:
#从设备中读取数据(属于硬件访问类操作)
#将读取到的数据返回给应用程序
ssize_t (*read) (struct file *filp, char __user *buff, size_t count, loff_t *offp)
参数分析:filp:与字符设备文件关联的file结构指针, 由内核创建.
buff : 从设备读取到的数据,需要保存到的位置。由read系统调用提供该参数。
count: 请求传输的数据量,由read系统调用提供该参数。
offp: 文件的读写位置,由内核从file结构中取出后,传递进来。
3. write
ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *)向设备写入数据,响应write系统调用
write设备方法通常完成2件事情:
#从应用程序提供的地址中取出数据
#将数据写入设备(属于硬件访问类操作)
4. lseek
loff_t (*llseek) (struct file *, loff_t, int)重定位读写指针,响应lseek系统调用
5. close
int (*release) (struct inode *, struct file *);关闭设备,响应close系统调用
release方法的作用正好与open相反。
这个设备方法有时也称为close,它应该:
#关闭设备
3. 驱动注销
当我们从内核中卸载驱动程序的时候,需要使用cdev_del函数来完成字符设备的注销。
3. 范例驱动分析
Linux 字符设备驱动模型的更多相关文章
- 深入理解Linux字符设备驱动
文章从上层应用访问字符设备驱动开始,一步步地深入分析Linux字符设备的软件层次.组成框架和交互.如何编写驱动.设备文件的创建和mdev原理,对Linux字符设备驱动有全面的讲解.本文整合之前发表的& ...
- Linux字符设备驱动结构(一)--cdev结构体、设备号相关知识机械【转】
本文转载自:http://blog.csdn.net/zqixiao_09/article/details/50839042 一.字符设备基础知识 1.设备驱动分类 linux系统将设备分为3类:字符 ...
- (57)Linux驱动开发之三Linux字符设备驱动
1.一般情况下,对每一种设备驱动都会定义一个软件模块,这个工程模块包含.h和.c文件,前者定义该设备驱动的数据结构并声明外部函数,后者进行设备驱动的具体实现. 2.典型的无操作系统下的逻辑开发程序是: ...
- Smart210学习记录----beep linux字符设备驱动
今天搞定了beep linux字符设备驱动,心里还是很开心的,哈哈...但在完成的过程中却遇到了一个非常棘手的问题,花费了我大量的时间,,,, 还是把问题描述一下吧,好像这个问题很普遍的,网上许多解决 ...
- Linux字符设备驱动实现
Linux字符设备驱动实现 要求 编写一个字符设备驱动,并利用对字符设备的同步操作,设计实现一个聊天程序.可以有一个读,一个写进程共享该字符设备,进行聊天:也可以由多个读和多个写进程共享该字符设备,进 ...
- Linux字符设备驱动基本结构
1.Linux字符设备驱动的基本结构 Linux系统下具有三种设备,分别是字符设备.块设备和网络设备,Linux下的字符设备是指只能一个字节一个字节读写的设备,不能随机读取设备内存中某一数据,读取数据 ...
- linux设备驱动之字符设备驱动模型(2)
在上一篇中我们已经了解了字符设备驱动的原理,也了解了应用层调用内核函数的机制,但是我们每次操作设备,都必须首先通过mknod命令创建一个设备文件名,比如说我们要打开u盘,硬盘等这些设备,难道我们还要自 ...
- linux设备驱动之字符设备驱动模型(1)
一:字符设备驱动 在linux下面,应用层看到的一切皆为文件(名字)所有的设备都是文件,都可以调用open,read,write来操作,而在内核中每个中每个设备有唯一的对应一个设备号: APP ( ...
- Linux字符设备驱动框架
字符设备是Linux三大设备之一(另外两种是块设备,网络设备),字符设备就是字节流形式通讯的I/O设备,绝大部分设备都是字符设备,常见的字符设备包括鼠标.键盘.显示器.串口等等,当我们执行ls -l ...
随机推荐
- 配置Redis主从复制
[构建高性能数据库缓存之redis主从复制][http://database.51cto.com/art/201407/444555.htm] 一.什么是redis主从复制? 主从复制,当用户往Mas ...
- WCF再学习小结
http://www.cnblogs.com/jillzhang/archive/2010/04/04/1704388.html http://leelei.blog.51cto.com/856755 ...
- ExtJs内的datefield控件选择日期过后的事件监听select
[摘要]: 选择时间过后我们为什么需要监听事件?一般有这样一种情况,那就是用于比较两个时间大小或者需要判断在哪个时间点上需要做什么样的操作.基于这样的种种情况,我们很有必要琢磨一下datefield控 ...
- 编写一个JavaScript函数 parseQueryString,把URL参数解析为一个对象
var url="http://www.taobao.com/index.php?key0=0&key1=1&key2=2"; function parseQuer ...
- hdu 4499 Cannon dfs
Cannon Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://acm.hdu.edu.cn/showproblem.php?pid=4499 D ...
- codeforces Gym 100187L L. Ministry of Truth 水题
L. Ministry of Truth Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/gym/100187/p ...
- C#操作Excel(1)Excel对象模型
Excel对象模型 (.Net Perspective) 本文主要针对在Visual Studio中使用C# 开发关于Excel的应用程序 本文的PDF下载地址:C#操作Excel2007.pdf ...
- SharedObject使用:在FluorineFx.net与Flex中使用共享对象维护在线用户列表实例【转】
一.添加一个新的FluorineFx的服务类项目OnLineService,删除原有的Sample.cs,并添加一个用户类定义与一个ApplicationAdpater类:如下: /*-- User. ...
- 为什么Wireshark无法解密HTTPS数据
为什么Wireshark无法解密HTTPS数据 导读 由于需要定位一个问题,在服务器上tcpdump抓取https数据包,然后下载到本地打开wireshark分析.然后我们下载域名私钥配置到wires ...
- POJ1651:Multiplication Puzzle(区间DP)
Description The multiplication puzzle is played with a row of cards, each containing a single positi ...