Linux 2.6.32内核字符设备驱…
引言:Linux驱动中,字符设备的设计一般会占产品驱动开发的90%以上,作者根据驱动开发的实际经验,总结了一个标准的字符设备驱动的模板,仅供参考。
//=======================字符设备驱动模板开始 ===========================//
#define
CHAR_DEV_DEVICE_NAME "char_dev" // 设备名
struct
class *char_dev _class; //
class结构用于自动创建设备结点
static int major = 0;
static
struct cdev char_dev_devs;// 定义一个cdev结构
// 设备建立子函数,被char_dev_init函数调用
static void char_dev_setup_cdev(struct cdev *dev, int minor, struct
file_operations *fops)
{
int err,
devno = MKDEV(major, minor);
cdev_init(dev, fops);
dev->owner = THIS_MODULE;
dev->ops
= fops;
err =
cdev_add(dev, devno, 1);
if( err
)
{
printk(KERN_NOTICE "Error %d adding char_dev
%d\n", err, minor);
}
}
// file_operations 结构体设置,该设备的所有对外接口在这里明确,此处只写出了几常用的
static struct file_operations char_dev_fops =
{
.owner =
THIS_MODULE,
.open =
char_dev_open, // 打开设备
.release =
char_dev_release, // 关闭设备
.read =
char_dev_read, // 实现设备读功能
.write =
char_dev_write, // 实现设备写功能
.ioctl =
char_dev_ioctl, //实现设备控制功能
};
// 进行初始化设置,打开设备,对应应用空间的open 系统调用
int char_dev_open (struct inode *inode, struct file
*filp)
{
... // 这里可以进行一些初始化
return
0;
}
// 释放设备,关闭设备,对应应用空间的close 系统调用
static
int char_dev_release (struct inode *node, struct file
*file)
{
... // 这里可以进行一些资源的释放
return
0;
}
// 实现读功能,读设备,对应应用空间的read 系统调用
ssize_t
char_dev_read (struct file *file,char __user *buff,size_t
count,loff_t *offp)
{
...
return
0;
}
// 实现写功能,写设备,对应应用空间的write 系统调用
ssize_t
char_dev_write(struct file *file,const char __user *buff,size_t
count,loff_t *offp)
{
...
return
0;
}
// 实现主要控制功能,控制设备,对应应用空间的ioctl 系统调用
static
int char_dev _ioctl(struct inode *inode,struct file *file,unsigned
int cmd,unsigned long arg)
{
...
return
0;
}
// 设备初始化
static int char_dev_init(void)
{
int
result;
dev_t dev =
MKDEV(major, 0);
if( major
)
{
// 给定设备号注册
result = register_chrdev_region(dev, 1,
CHAR_DEV_DEVICE_NAME);
}
else
{
// 动态分配设备号
result = alloc_chrdev_region(&dev, 0, 1,
CHAR_DEV_DEVICE_NAME);
major = MAJOR(dev);
}
char_dev_setup_cdev(&char_dev_devs, 0,
&char_dev_fops);
printk("The
major of the char_dev device is %d\n", major);
//==== 有中断的可以在此注册中断:request_irq,并要实现中断服务程序 ===//
// 创建设备结点
char_dev
_class = class_create(THIS_MODULE,"ad_class");
if
(IS_ERR(char_dev _class))
{
printk("Err:
failed in creating class.\n");
return
0;
}
device_create(char_dev_class, NULL, dev, NULL,
"char_dev");
return
0;
}
// 设备注销
static void char_dev_cleanup(void)
{
device_destroy(adc_class,dev);
class_destroy(adc_class);
cdev_del(&char_dev_devs);//字符设备的注销*/
unregister_chrdev_region(MKDEV(major, 0),
1);//设备号的注销
//======== 有中断的可以在此注销中断:free_irq ======//
printk("char_dev device
uninstalled\n");
}
module_init(char_dev_init);//模块初始化接口
module_exit(char_dev_cleanup);//模块注销接口
// 以下两句不能省略,否则编译不通过
MODULE_AUTHOR("www.embedhq.org");
MODULE_LICENSE("GPL");
//==================== 字符设备驱动模板结束 ========================//
用Makefile模板编译,Makefile如下:
//======================= Makefile开始 ===========================//
ifeq
($(KERNELRELEASE),)
#KERNELDIR ?= /your/target/source/directory/
KERNELDIR
?= /opt/kernal/linux-2.6.32.10/
PWD :=
$(shell pwd)
modules:
$(MAKE)
-C $(KERNELDIR) M=$(PWD) modules
modules_install:
$(MAKE)
-C $(KERNELDIR) M=$(PWD) modules_install
clean:
rm
-rf *.o *~ core .depend .*.cmd *.ko *.mod.c
.tmp_versions
.PHONY:
modules modules_install clean
//========================= Makefile结束 =============================//
make编译后生成char_dev.ko,控制台输入加载和卸载命令,还可以使用lsmod查看已经加载的模块信息。
insmod char_dev.ko #加载驱动,会执行module_init中的语句
rmmod char_dev #卸载驱动,会执行module_exit中的语句
(原文出处:http://www.embedhq.org/, 转载请注明出处)
Linux 2.6.32内核字符设备驱…的更多相关文章
- 深入浅出:Linux设备驱动之字符设备驱
一.linux系统将设备分为3类:字符设备.块设备.网络设备.使用驱动程序: 字符设备:是指只能一个字节一个字节读写的设备,不能随机读取设备内存中的某一数据,读取数据需要按照先后数据.字符设备是面向流 ...
- 嵌入式Linux驱动学习之路(十)字符设备驱动-my_led
首先贴上代码: 字符设备驱动代码: /** *file name: led.c */#include <linux/sched.h> #include <linux/signal.h ...
- linux驱动开发(四) 字符设备驱动框架(自动创建设备节点)
代码如下 #include <linux/init.h> #include <linux/module.h> #include <linux/kernel.h> # ...
- linux驱动开发(三) 字符设备驱动框架
还是老规矩先上代码 demo.c #include <linux/init.h> #include <linux/module.h> #include <linux/ke ...
- linux一切皆文件之tty字符设备(深入理解sshd创建pty的过程) (五)
一.知识准备 1.在linux中,一切皆为文件,所有不同种类的类型都被抽象成文件(比如:块设备,socket套接字,pipe队列) 2.操作这些不同的类型就像操作文件一样,比如增删改查等 3.块设备支 ...
- register_chrdev_region/alloc_chrdev_region和cdev注册字符设备驱动
内核提供了三个函数来注册一组字符设备编号,这三个函数分别是 register_chrdev_region().alloc_chrdev_region() 和 register_chrdev(). (1 ...
- Linux内核分析(六)----字符设备控制方法实现|揭秘系统调用本质
原文:Linux内核分析(六)----字符设备控制方法实现|揭秘系统调用本质 Linux内核分析(六) 昨天我们对字符设备进行了初步的了解,并且实现了简单的字符设备驱动,今天我们继续对字符设备的某些方 ...
- Linux内核分析(五)----字符设备驱动实现
原文:Linux内核分析(五)----字符设备驱动实现 Linux内核分析(五) 昨天我们对linux内核的子系统进行简单的认识,今天我们正式进入驱动的开发,我们今后的学习为了避免大家没有硬件的缺陷, ...
- 浅析Linux字符设备驱动程序内核机制
前段时间在学习linux设备驱动的时候,看了陈学松著的<深入Linux设备驱动程序内核机制>一书. 说实话.这是一本非常好的书,作者不但给出了在设备驱动程序开发过程中的所须要的知识点(如对 ...
随机推荐
- react:高阶组件wrappedComponent
什么是高阶组件? 高阶部件是一种用于复用组件逻辑的高级技术,它并不是 React API的一部分,而是从React 演化而来的一种模式. 具体地说,高阶组件就是一个接收一个组件并返回另外一个新组件的函 ...
- Java基础学习-IO流
package IObasics; import java.io.FileWriter; import java.io.IOException; /*IO流 * 通过数据流.序列化和文件系统提供系统输 ...
- http://www.cnblogs.com/peida/archive/2013/05/31/3070790.html深入理解Java:SimpleDateFormat安全的时间格式化
http://www.cnblogs.com/peida/archive/2013/05/31/3070790.html
- 理解 HTTP2.0
链接 HTTP/2 头部压缩技术介绍 我们知道,HTTP/2 协议由两个 RFC 组成: 一个是 RFC 7540,描述了 HTTP/2 协议本身:一个是 RFC 7541,描述了 HTTP/2 协议 ...
- 从云主机上下载文件到本地+获取外网地址(linux & Windows)
云主机上下载有集中方法,腾讯论坛有一遍文章:向云服务器上传下载文件方法汇总 货比三家,最终还是选择了rsync 下载代码如下 rsync ubuntu@123.207.251.217:/var/www ...
- SCARA——OpenGL入门学习四(颜色)
OpenGL入门学习[四] 本次学习的是颜色的选择.终于要走出黑白的世界了~~ OpenGL支持两种颜色模式:一种是RGBA,一种是颜色索引模式. 无论哪种颜色模式,计算机都必须为每一个像素保存一些数 ...
- 数据库SQL、SQLite语句单引号、双引号的用法
最近编程操作数据库语句的时候出现一些问题. 关于Insert字符串 ,在(单引号,双引号)这个方面发生了问题,其实主要是因为数据类型和变量在作怪. 下面我们就分别讲述,虽然说的是Insert语句, 但 ...
- Spring Boot系列—(一)入门
前言 因为项目组需要进行微服务改造,而微服务开发中需要以Spring Boot为基础.因此需要先弄懂SpringBoot. 我们先来看看SpringBoot的背景由来,SpringBoot是什么,一个 ...
- VS 2015 开发Android底部导航条----[实例代码,多图]
1.废话背景介绍 在Build 2016开发者大会上,微软宣布,Xamarin将被整合进所有版本的Visual Studio之中. 这也就是说,Xamarin将免费提供给所有购买了Visual ...
- 3 循环语句——《Swift3.0从入门到出家》
3 循环语句 当一段代码被多次重复利用的使用我们就使用循环 swift提供了三种形式的循环语句 1.while 循环 2.repeat — while 循环 3.for — in 循环 while 循 ...