linux miscdevice 驱动程序

1  在Linux驱动中把无法归类的五花八门的设备定义为混杂设备(用miscdevice结构体表述)。miscdevice共享一个主设备号MISC_MAJOR(即10),但次设备号不同。 所有的miscdevice设备形成了一个链表,对设备访问时内核根据次设备号查找对应的miscdevice设备,然后调用其file_operations结构中注册的文件操作接口进行操作。 在内核中用struct miscdevice表示miscdevice设备,然后调用其file_operations结构中注册的文件操作接口进行操作。miscdevice的API实现在drivers/char/misc.c中。

struct miscdevice

{

int minor;

const char *name;

const struct file_operations *fops;

struct list_head list;

struct device *parent;

struct device *this_device;

const char *nodename;

mode_t mode;

};

minor是这个混杂设备的次设备号,若由系统自动配置,则可以设置为

MISC_DYNANIC_MINOR,name是设备名

misc->this_device = device_create(misc_class, misc->parent, dev, misc, "%s", misc->name);//说明了混杂设备驱动内部有自动设备驱动,可自动创建设备节点

#define MISC_DYNAMIC_MINOR      255  动态分配次设备号。

2

structfile_operations{

struct module *owner;

// 指向拥有该结构的模块的指针,避免正在操作时被卸载,一般为初始化为THIS_MODULES

loff_t (*llseek) (struct file *, loff_t, int);

// llseek用来修改文件当前的读写位置,返回新位置

// loff_t为一个"长偏移量"。当此函数指针为空,seek调用将会以不可预期的方式修改file结构中的位置计数器。

ssize_t (*read) (struct file *, char __user *, size_t, loff_t *);

// 从设备中同步读取数据。读取成功返回读取的字节数。设置为NULL,调用时返回-EINVAL

ssize_t (*aio_read) (struct kiocb *, char __user *, size_t, loff_t);

// 初始化一个异步的读取操作,为NULL时全部通过read处理

ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *);

// 向设备发送数据。

ssize_t (*aio_write) (struct kiocb *, const char __user *, size_t, loff_t);

// 初始化一个异步的写入操作。

int (*readdir) (struct file *, void *, filldir_t);

// 仅用于读取目录,对于设备文件,该字段为 NULL

unsigned int (*poll) (struct file *, struct poll_table_struct *);

// 返回一个位掩码,用来指出非阻塞的读取或写入是否可能。

// 将pool定义为 NULL,设备会被认为即可读也可写。

int (*ioctl) (struct inode *, struct file *, unsigned int, unsigned long);

// 提供一种执行设备特殊命令的方法。不设置入口点,返回-ENOTTY

long (*unlocked_ioctl) (struct file *, unsigned int, unsigned long);

// 不使用BLK的文件系统,将使用此种函数指针代替ioctl

long (*compat_ioctl) (struct file *, unsigned int, unsigned long);

// 在64位系统上,32位的ioctl调用,将使用此函数指针代替

int (*mmap) (struct file *, struct vm_area_struct *);

// 用于请求将设备内存映射到进程地址空间。如果无此方法,将访问-ENODEV。

int (*open) (struct inode *, struct file *);

// 如果为空,设备的打开操作永远成功,但系统不会通知驱动程序

// 由VFS调用,当VFS打开一个文件,即建立了一个新的"struct file",之后调用open方法分配文件结构。open属于struct inode_operations。

int (*flush) (struct file *);

// 发生在进程关闭设备文件描述符副本,执行并等待,若设置为NULL,内核将忽略用户应用程序的请求。

int (*release) (struct inode *, struct file *);

// file结构释放时,将调用此指针函数,release与open相同可设置为NULL

int (*fsync) (struct file *, struct dentry *, int datasync);

// 刷新待处理的数据,如果驱动程序没有实现,fsync调用将返回-EINVAL

int (*aio_fsync) (struct kiocb *, int datasync);

// 异步fsync

int (*fasync) (int, struct file *, int);

// 通知设备FASYNC标志发生变化,如果设备不支持异步通知,该字段可以为NULL

int (*lock) (struct file *, int, struct file_lock *);

// 实现文件锁,设备驱动常不去实现此lock

ssize_t (*readv) (struct file *, const struct iovec *, unsigned long, loff_t *);

ssize_t (*writev) (struct file *, const struct iovec *, unsigned long, loff_t *);

// readv和writev 分散/聚集型的读写操作,实现进行涉及多个内存区域的单次读或写操作。

ssize_t (*sendfile) (struct file *, loff_t *, size_t, read_actor_t, void *);

// 实现sendfile调用的读取部分,将数据从一个文件描述符移到另一个,设备驱动通常将其设置为 NULL

ssize_t (*sendpage) (struct file *, struct page *, int, size_t, loff_t *, int);

// 实现sendfile调用的另一部分,内核调用将其数据发送到对应文件,每次一个数据页,设备驱动通常将其设置为NULL

unsigned long (*get_unmapped_area)(struct file *, unsigned long, unsigned long, unsigned long, unsigned long);

// 在进程地址空间找到一个合适的位置,以便将底层设备中的内存段映射到该位置。大部分驱动可将其设置为NULL

int (*check_flags)(int);

// 允许模块检查传递给fcntl(F_SETEL...)调用的标志

int (*dir_notify)(struct file *filp, unsigned long arg);

// 应用程序使用fcntl来请求目录改变通知时,调用该方法。仅对文件系统有效,驱动程序不必实现。

int (*flock) (struct file *, int, struct file_lock *);

// 实现文件锁

};

3  在linux内核源码miscdevice.h里杂项设备描述结构体定义:

struct miscdevice  {

int minor;

const char *name;

const struct file_operations *fops;

struct list_head list;

struct device *parent;

struct device *this_device;

const char *nodename;

mode_t mode;

};

extern int misc_register(struct miscdevice * misc);

extern int misc_deregister(struct miscdevice *misc);

miscdevice设备框架:

static ssize_t mydev_read(struct file *filp, char __user * buf, size_t count, loff_t *f_pos)

{

}

static int mydev_ioctl( struct inode *inode, struct file *file, unsigned int cmd, unsigned
long arg)

{

return 0;

}

static struct file_operations mydev_fops = {     //虚拟文件系统

.owner = THIS_MODULE,

.ioctl = mydev_ioctl,

.read = mydev_read,

};

static struct miscdevice misc = {

.minor = MISC_DYNAMIC_MINOR,

.name = DEVICE_NAME,

.fops = &mydev_fops,

};

static int mydev_init(void)

{

int ret;

ret = misc_register(&misc);

return ret;

}

static void mydev_exit(void)

{

misc_deregister(&misc);

}

module_init(mydev_init);   //模块加载函数

module_exit(mydev_exit);   //模块卸载函数

MODULE_LICENSE("Dual BSD/GPL");  //模块声明

杂项设备是特殊的字符设备,但是编写这类驱动程序省去了创建设备节点的麻烦,因此更加方便。

linux miscdevice 驱动程序

时间: 2024-10-10 04:43:00

linux miscdevice 驱动程序的相关文章

linux设备驱动程序该添加哪些头文件以及驱动常用头文件介绍(转)

原文链接:http://blog.chinaunix.net/uid-22609852-id-3506475.html 驱动常用头文件介绍 #include <linux/***.h> 是在linux-2.6.29/include/linux下面寻找源文件.#include <asm/***.h> 是在linux-2.6.29/arch/arm/include/asm下面寻找源文件.#include <mach/***.h> 是在linux-2.6.29/arch/ar

20150226 IMX257 混杂设备miscdevice驱动程序

20150226 IMX257 混杂设备miscdevice驱动程序 2015-02-26 16:00 李海沿 在Linux驱动中把无法归类的五花八门的设备定义为混杂设备(用miscdevice结构体表述).miscdevice共享一个主设备号MISC_MAJOR(即10),但次设备号不同. 所有的miscdevice设备形成了一个链表,对设备访问时内核根据次设备号查找对应的miscdevice设备,然后调用其file_operations结构中注册的文件操作接口进行操作. 在内核中用struc

Linux设备驱动程序学习笔记(一)

1.设备驱动程序扮演的角色:       设备程序是一个独立的“黑盒子”,使其某个特定硬件响应一个定义良好的内部编程接口,这些接口完全隐藏了设备的工作细节.用户的操作通过一组标准化的调用执行,而这些调用独立于特定的驱动程序.将这些调用映射到作用于实际硬件的设备特有操作上,则是设备驱动程序的任务.2.驱动程序的作用:        驱动程序应该处理如何使用硬件可用的问题,而将怎样使用硬件的问题留给上层应用.(提供机制,而不是策略)3.内核功能划分:        进程管理    内存管理    文

LINUX设备驱动程序笔记(一)设备驱动程序简介

<一>:设备驱动程序的作用 从一个角度看,设备驱动程序的作用在于提供机制,而不是策略.在编写驱动程序时,程序员应该特别注意下面这个基本概念:编写访问硬件的内核代码时,不要给用户强加任何特定策略.因为不同的用户有不同的需求,驱动程序应该处理如何使硬件可用的问题,而将怎样使用硬件的问题留给上层应用程序. 从另一个角度来看驱动程序,它还可以看作是应用程序和实际设备之间的一个软件层. 总的来说,驱动程序设计主要还是综合考虑下面三个方面的因素:提供给用户尽量多的选项.编写驱动程序要占用的时间以及尽量保持

LINUX设备驱动程序笔记(二)构造和运行模块

         <一>:设置测试系统 首先准备好一个内核源码树,构造一个新内核,然后安装到自己的系统中.           <二>:HelloWorld模块 #include <linux/init.h> //定义了驱动的初始化和退出相关的函数 #include <linux/module.h> //定义了内核模块相关的函数.变量及宏 MODULE_LICENSE("Dual BSD/GPL"); //该宏告诉内核,该模块采用自由许可

LINUX设备驱动程序笔记(三)字符设备驱动程序

      <一>.主设备号和次设备号        对字符设备的访问时通过文件系统内的设备名称进行的.那些设备名称简单称之为文件系统树的节点,它们通常位于/dev目录.字符设备驱动程序的设备文件可通过ls -l命令输出的第一列中的'c'来识别.块设备同样位于/dev下,由字符'b'标识 crw-rw----  1 root root    253,   0 2013-09-11 20:33 usbmon0 crw-rw----  1 root root    253,   1 2013-09

LINUX设备驱动程序笔记(四)并发和竞态

       <一>.并发及其管理 大部分竞态可通过使用内核的并发控制原语,并应用几个基本的原理来避免.第一个规则是,只要可能,就应该避免资源的共享,这种思想的明显应用就是避免使用全局变量.但硬件资源本质上就是共享的,软件资源经常需要对其他执行线程可用.全局变量并不是共享数据的唯一途径,只要我们的代码将一个指针传递给了内核的其他部分,一个新的共享就可能建立.在单个执行线程之外共享硬件或软件资源的任何时候,因为另外一个线程可能产生对该资源的不一致观察,因此必须显示地管理对该资源的访问.访问管理的

《 转》Linux 网卡驱动程序对 ethtool 的支持和实现

Linux 的一个显著特点就是其强大的网络功能,Linux 几乎支持所有的网络协议,并在这些协议基础上提供了丰富的应用.对 Linux 网络管理的重要性不言而喻,这些管理依赖于网络工具,比如最常用的 ifconfig,route,ip,ethtool 等,其中 ethtool 提供了强大的网卡及网卡驱动管理能力,其具体的实现框架和网络驱动程序及网络硬件关系紧密,容易修改和扩展,能够为 Linux 网络开发人员和管理人员提供对网卡硬件,驱动程序和网络协议栈的设置,查看以及及调试等功能. 从典型的以

Linux 设备驱动程序 字符设备

已经无法再精简,适合入门. 1 #include<linux/module.h> 2 #include<linux/init.h> 3 4 #include<asm/uaccess.h> 5 #include <linux/types.h> 6 #include<linux/fs.h> 7 #include<linux/cdev.h> 8 struct mengc_dev{ 9 char data[64]; 10 struct cde