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 ...
随机推荐
- Servlet容器的启动过程
[http://book.51cto.com/art/201408/448854.htm] Tomcat的启动逻辑是基于观察者模式设计的,所有的容器都会继承Lifecycle接口,它管理着容器的整 ...
- vs2013修改默认的开发环境
可能会有朋友和我一样,当安装完VS完成之后,没有选择默认的开发模板,在后面添加新项目时,总是不能选择默认的开发语言,下面给出个简单步骤,记录一下以备用. 看图吧 1.工具>导入和导出设置 2.选 ...
- Flex 日期和字符串之间转换
字符串转为日期: var dateTime:Date= DateField.stringToDate(deTime, "YYYY-MM-DD");//"YYYY-MM-D ...
- yum update Transaction Check Error
update系统时,发现其中一台server居然提示: Transaction Check Error:file /usr/lib/perl5/5.8.8/CGI.pm from install of ...
- javascript数字转汉字中文数字
/* 工具包 */ var Utils={ /* 单位 */ units:'个十百千万@#%亿^&~', /* 字符 */ chars:'零一二三四五六七八九', /* 数字转中文 @numb ...
- 番茄钟App(Pomodoro Tracker)
最近为了学习Swift编程语言,写了一个番茄钟的App(Pomodoro Tracker).刚上线的1.2版本增加了Apple Watch的支持. iPhone版 Apple Watch版 如果你跟我 ...
- BZOJ 1014: [JSOI2008]火星人prefix Splay+二分
1014: [JSOI2008]火星人prefix 题目连接: http://www.lydsy.com/JudgeOnline/problem.php?id=1014 Description 火星人 ...
- 战舰少女 黑暗炼钢 按键精灵 代码及apk下载
注: 该代码仅仅适用于1920*1080分辨率的android手机,因为我只有这个分辨率的手机TnT 代码其实蛮简单的,都是比较简单的模拟就好了…… 要改也比较轻松吧 APK下载地址:链接: http ...
- HTML 转义符
特殊字符 字符 十进制 转义字符 “ " " & & & < < < > > > 不断开空格(non-breaking ...
- Linux - wc统计文件行数、单词数或字节数
一 wc简单介绍 wc命令用来打印文件的文本行数.单词数.字节数等(print the number of newlines, words, and bytes in files).在Windows的 ...