【驱动】linux设备驱动·字符设备驱动开发
Preface
前面对linux设备驱动的相应知识点进行了总结,现在进入实践阶段!
《linux设备驱动入门篇》:http://infohacker.blog.51cto.com/6751239/1218461
《linux设备驱动扫盲篇》:http://infohacker.blog.51cto.com/6751239/1218747
《fedora下的字符设备驱动开发》:http://infohacker.blog.51cto.com/6751239/1155153
开发一个基本的字符设备驱动
在Linux内核驱动中,字符设备是最基本的设备驱动。
字符设备包括了设备最基本的操作,如打开设备、关闭设备、I/O控制等。
功能:❶建立一个名为GlobalChar的虚拟设备,设备内部只有一个全局变量供用户操作。设备提供了❷读函数读取全局变量的值并且返回给用户,❸写函数把用户设定的值写入全局变量。
①代码如下:
//GlobalCharDev.c
#include <linux/fs.h>
#include <linux/module.h>
#include <asm/uaccess.h>
#include <linux/init.h>
MODULE_LICENSE("GPL");
MODULE_AUTHOR("mystery");
#define DEV_NAME "GlobalChar"
static ssize_t GlobalRead(struct file *, char *, size_t, loff_t *);
static ssize_t GlobalWrite(struct file *, const char *, size_t, loff_t *); static int char_major = 0;
static int GlobalData = 0; // "GlobalData" 设备的全局变量
//初始化字符设备驱动的file_operations结构体
static const struct file_operations globalchar_fops =
{
.read = GlobalRead,
.write = GlobalWrite
}; //注意分号啊!!!
//模块初始化函数
static int __init GlobalChar_init(void)
{
int ret;
ret = register_chrdev(char_major, DEV_NAME, &globalchar_fops); //注册设备驱动
if (ret < 0)
{
printk(KERN_ALERT "GlobalChar Reg Fail ! \n");
}
else
{
printk(KERN_ALERT "GlobalChar Reg Success ! \n");
char_major = ret;
printk(KERN_ALTER "Major = %d \n",char_major);
}
return ret;
}
//模块卸载函数
static void __exit GlobalChar_exit(void)
{
unregister_chrdev(char_major, DEV_NAME); //注销设备驱动
return;
}
//设备驱动读函数
static ssize_t GlobalRead(struct file *filp, char *buf, size_t len, loff_t *off)
{
if(copy_to_user(buf, &GlobalData, sizeof(int))) //从内核空间复制GlobalData到用户空间
{
return -EFAULT;
}
return sizeof(int);
}
//设备驱动写函数
static ssize_t GlobalWrite(struct file *filp, const char *buf, size_t len, loff_t *off)
{
if(copy_from_user(&GlobalData, buf, sizeof(int))) //从用户空间复制GlobalData到内核空间
{
return -EFAULT;
}
return sizeof(int);
}
module_init(GlobalChar_init);
module_exit(GlobalChar_exit);
在内核中操作数据要区分数据的来源,对于用户空间的数据要使用copy_from_user()函数复制,使用copy_to_user()函数回写,不能直接操作用户空间的数据,否则会产生内存访问错误。
②编写Makefile
obj-m := GlobalCharDev.o
KDIR := /lib/modules/$(shell uname -r)/build
SRCPWD := $(shell pwd)
all:
make -C $(KDIR) M=$(SRCPWD) modules
③编译并加载内核模块
④查看内核分配的主设备号
⑤使用mknod命令建立一个设备文件
mknod命令使用-m参数指定GlobalChar设备可以被所有用户访问。
249即上面查询的主设备号。
到这里,我们就已经正确地添加了一个字符设备到内核,下面需要测试一下驱动程序能否正常工作。
测试字符设备驱动
为了测试编写的字符设备是否能正常工作,我们编写一个应用程序测试一下能否正常读写字符设备。
测试代码:
//GlobalCharTest.c
#include <sys/types.h>
#include <sys/stat.h>
#include <stdio.h>
#include <fcntl.h>
#define DEV_NAME "/dev/GlobalChar"
int main()
{
int fd,num; fd = open(DEV_NAME, O_RDWR, S_IRUSR | S_IWUSR); //打开设备文件
if (fd < 0)
{
printf("Open Deivece Fail ! \n");
return -1;
}
read(fd, &num, sizeof(int));
printf("The GlobalChar is %d \n", num); //获取当前设备数值
printf("Please input a number written to GlobalChar: ");
scanf("%d", &num);
write(fd, &num, sizeof(int)); //写入新的数值
read(fd, &num, sizeof(int));
printf("The GlobalChar is %d \n", num); //重新读取设备数值
close(fd);
return 0;
}
程序首先使用open函数打开设备文件,然后使用read()函数读取字符设备的值,open()系统调用最终会被解释为字符设备注册的read调用。
测试结果:
从程序输出结果来看,最初从设备得到的数值是0,输入520后写入到字符设备,重新读出的数值也是520,与设置相同,表示设备驱动程序功能正确。
总结
linux字符设备驱动也不过如此嘛,嘿嘿,虽然只实现了read和write两个函数,不过其它函数也大同小异。
重点:实践再实践!!!
本文出自 “成鹏致远” 博客,请务必保留此出处http://infohacker.blog.51cto.com/6751239/1219217
【驱动】linux设备驱动·字符设备驱动开发的更多相关文章
- Linux块设备和字符设备
块设备:系统能够随机无序访问固定大小的数据片的设备,这些数据片称为块.块设备是以固定大小长度来传送资料的,它使用缓冲区暂存数据,时机成熟后从缓存中一次性写入到设备或者从设备中一次性放到缓存区.常见的块 ...
- Linux平台块设备到字符设备(裸设备)的三种映射方式(转载)
在Linux平台oracle rac的组建过程中,如果使用ASM+RAW的存储方式的话,由于asm不支持块设备,支持持字符访问设备,所以需要配置将Block Device Drive转变成Charac ...
- 002_linux驱动之_register_chrdev注册字符设备
(一)解析:register_chrdev函数和unregister_chrdev函数 (二)register_chrdev函数原型 int register_chrdev(unsigned int ...
- Linux学习 :字符设备框架
一.系统功能框架: U-boot : 启动内核 linux kernel: 启动应用 应用: open,read,write 都是通过C库实现,汇编就相当于swi val,引发中断,通过系统调用接口在 ...
- linux device driver —— 字符设备
现在对linux设备驱动还没有什么认识,跟着书上敲了一个字符驱动,这里把代码贴一下. 测试环境是 Ubuntu 16.04 64bit 驱动程序: #include <linux/fs.h> ...
- 韦东山驱动视频笔记——3.字符设备驱动程序之poll机制
linux内核版本:linux-2.6.30.4 目的:我们在中断方式的按键应用程序中,如果没有按键按下,read就会永远在那等待,所以如果在这个程序里还想做其他事就不可能了.因此我们这次改进它,让它 ...
- Linux 设备驱动程序 字符设备
已经无法再精简,适合入门. #include<linux/module.h> #include<linux/init.h> #include<asm/uaccess.h& ...
- (57)Linux驱动开发之三Linux字符设备驱动
1.一般情况下,对每一种设备驱动都会定义一个软件模块,这个工程模块包含.h和.c文件,前者定义该设备驱动的数据结构并声明外部函数,后者进行设备驱动的具体实现. 2.典型的无操作系统下的逻辑开发程序是: ...
- 《linux设备驱动开发详解》笔记——6字符设备驱动
6.1 字符设备驱动结构 先看看字符设备驱动的架构: 6.1.1 cdev cdev结构体是字符设备的核心数据结构,用于描述一个字符设备,cdev定义如下: #include <linux/cd ...
- Linux应用程序访问字符设备驱动详细过程【转】
本文转载自:http://blog.csdn.net/coding__madman/article/details/51346532 下面先通过一个编写好的内核驱动模块来体验以下字符设备驱动 可以暂时 ...
随机推荐
- C# 正则表达式过滤危险HTML
下面是两个过滤的方法 /// <summary> /// 此处过滤危险HTML方法 /// </summary> /// <param name="html&q ...
- ios中二维码的用法
网上的例子 zbar 下载地址 http://pan.baidu.com/share/link?shareid=2652605686&uk=9237761871.新建一个工程A:view-ba ...
- linux RPM包安装、更新、删除等操作命令简明总结, 如何查看yum安装的软件路径 ?
rpm -ivh package.rpm 安装一个rpm包rpm -ivh --nodeeps package.rpm 安装一个rpm包而忽略依赖关系警告rpm -U package.rpm 更新一个 ...
- UCP规模估算方法介绍 基于UCP方法的软件项目成本估计及其应用方法,软件,项目,UCP方法,应用,项目估算及软件及应用,软件估算,项目成本,软件项目
基于UCP方法的软件项目成本估计及其应用 UCP说明: UCP = 交易的UCP数 + Actor的UCP数,1.交易/Actor在估算时按复杂度分为简单.普通.复杂.主观类别,权重分别对应1.2.3 ...
- Intel Edison学习笔记(一)—— 刷系统
一.下载安装包 1.固件安装包:官网下载地址:http://downloadmirror.intel.com/ ... image-ww25.5-15.zip2 2.烧录工具下载地址:http://d ...
- Python学习笔记(九)—— 函数
一.函数调用: 1.python内部函数查询:http://docs.python.org/3/library/functions.html#abs 2.注意调用函数的参数个数和类型. 3.函数名其实 ...
- 【SqlServer】Sql Server 支持的数据类型
在计算机中数据有两种特征:类型和长度.所谓数据类型就是以数据的表现方式和存储方式来划分的数据的种类. 在SQL Server 中每个变量.参数.表达式等都有数据类型.系统提供的数据类型分为几大类 ...
- 【Linux】进程优先级、进程nice值和%nice
用top或者ps命令会输出PRI/PR.NI.%ni/%nice这三种指标值,这些到底是什么东西?先给出大概的解释如下: PRI :进程优先权,代表这个进程可被执行的优先级,其值越小,优先级就越高,越 ...
- golang学习笔记 ---数组与切片
数组: golang数组包含的每个数据称为数组元素(element),数组包含的元素个数被称为数组长度(length). golang数组的长度在定义后不可更改,并且在声明时可以是一个常量或常量表达式 ...
- Linux信号机制
Linux信号(signal) 机制分析 [摘要]本文分析了Linux内核对于信号的实现机制和应用层的相关处理.首先介绍了软中断信号的本质及信号的两种不同分类方法尤其是不可靠信号的原理.接着分析了内核 ...