引言: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内核字符设备驱…的更多相关文章

  1. 深入浅出:Linux设备驱动之字符设备驱

    一.linux系统将设备分为3类:字符设备.块设备.网络设备.使用驱动程序: 字符设备:是指只能一个字节一个字节读写的设备,不能随机读取设备内存中的某一数据,读取数据需要按照先后数据.字符设备是面向流 ...

  2. 嵌入式Linux驱动学习之路(十)字符设备驱动-my_led

    首先贴上代码: 字符设备驱动代码: /** *file name: led.c */#include <linux/sched.h> #include <linux/signal.h ...

  3. linux驱动开发(四) 字符设备驱动框架(自动创建设备节点)

    代码如下 #include <linux/init.h> #include <linux/module.h> #include <linux/kernel.h> # ...

  4. linux驱动开发(三) 字符设备驱动框架

    还是老规矩先上代码 demo.c #include <linux/init.h> #include <linux/module.h> #include <linux/ke ...

  5. linux一切皆文件之tty字符设备(深入理解sshd创建pty的过程) (五)

    一.知识准备 1.在linux中,一切皆为文件,所有不同种类的类型都被抽象成文件(比如:块设备,socket套接字,pipe队列) 2.操作这些不同的类型就像操作文件一样,比如增删改查等 3.块设备支 ...

  6. register_chrdev_region/alloc_chrdev_region和cdev注册字符设备驱动

    内核提供了三个函数来注册一组字符设备编号,这三个函数分别是 register_chrdev_region().alloc_chrdev_region() 和 register_chrdev(). (1 ...

  7. Linux内核分析(六)----字符设备控制方法实现|揭秘系统调用本质

    原文:Linux内核分析(六)----字符设备控制方法实现|揭秘系统调用本质 Linux内核分析(六) 昨天我们对字符设备进行了初步的了解,并且实现了简单的字符设备驱动,今天我们继续对字符设备的某些方 ...

  8. Linux内核分析(五)----字符设备驱动实现

    原文:Linux内核分析(五)----字符设备驱动实现 Linux内核分析(五) 昨天我们对linux内核的子系统进行简单的认识,今天我们正式进入驱动的开发,我们今后的学习为了避免大家没有硬件的缺陷, ...

  9. 浅析Linux字符设备驱动程序内核机制

    前段时间在学习linux设备驱动的时候,看了陈学松著的<深入Linux设备驱动程序内核机制>一书. 说实话.这是一本非常好的书,作者不但给出了在设备驱动程序开发过程中的所须要的知识点(如对 ...

随机推荐

  1. 查bug受气了,反思

    昨天改bug了一天,上午出现bug的原因是Boolean和boolean使用BeanUtils等工具无法自动赋值.我们写的时候注意大小写. 下午查了一个下午没有找到任何原因. 但是我受了气,因为报错点 ...

  2. 手把手教你怎么用ArcgisOnline发布地图服务

    Arcgis推出了Arcgis Online,但是大家都不知道这是个什么东西,怎么用这个东西,今天这篇文章手把手的教你如何使用Arcgisonline发布地图服务. 一.ArcgisOnline简介 ...

  3. Leetcode 944. Delete Columns to Make Sorted

    class Solution: def minDeletionSize(self, A: List[str]) -> int: ans = 0 for j in range(len(A[0])) ...

  4. 前端之JavaScript 02

    一.函数 // 最基础的函数定义 function f1() { console.log('hello world!'); } f1(); // hello world! // 带参数的函数 func ...

  5. 通过设置debug_options调试squid

    debug_options用于设置输出到access.log中的调试信息的模块和级别,默认为ALL,1. ALL指所有模块,每个源文件被赋予一个唯一的模块号 1指级别,完全调试的级别为9.很显然,设置 ...

  6. [leetcode]_Interleaving String

    下午去蹭了一发新浪的笔试. 炒鸡多的网络基础知识,总共18道题,就写了8道左右吧,剩下的全是网络知识,这部分抽时间至少过一过. 其中一道算法题,回来跟嘟嘟商量,才发现是leetcode上的原题,连ex ...

  7. ZOJ-Big string(服气思维)

    个人心得:我在分治上看到的,但是感觉跟分治没关系,一眼想到斐波那契数可以找到此时n的字符串,但是无法精确到字母,题解的思路 真是令人佩服,以BA为基准,然后只要此时的长度大于7那么必然可以减去最大的斐 ...

  8. 【操作系统】总结五(I/O管理)

    输入输出管理本章主要内容: I/O管理概述(I/O控制方式.I/O软件层次结构)和I/O核心子系统(I/O调度概念.局速缓存与缓冲区.设备分配与回收.假脱机技术(SPOOLing)). 5.1 I/O ...

  9. django的表与表之间的关系详细讲解

    转:http://www.cnblogs.com/feixuelove1009/p/5855295.html

  10. 为什么中国出不了facebook和Twitter?

    我们坐拥全球最大基数的网民,我们拥有让人骄傲的四大发明,我们有有流传全世界的孙子兵法,可是在互联网时代,我们却落后了.互联网可以说是江山人才辈辈出,各领风骚三两年. 让我们来简单地回顾一下近几年的互联 ...