0

0

Linux驱动开发: 杂项字符设备

看不見的法師

看不見的法師

发布时间:2025-07-17 08:34:01

|

858人浏览过

|

来源于php中文网

原创

一、什么是杂项设备?

杂项设备(misc device)在嵌入式系统中应用广泛,是一种特殊的设备驱动。

在Linux内核的include/linux目录下,有一个Miscdevice.h文件,定义了misc设备及其相关的内核函数。

这些字符设备因为不符合预定的字符设备范畴,所以它们使用主设备号10,并统一归为misc device。misc_register函数实际上是使用主设备号10调用register_chrdev()的。

换句话说,misc设备就是特殊的字符设备。

在Linux驱动中,无法归类的各种设备被定义为杂项设备(使用miscdevice结构体来表示)。

miscdevice共享一个主设备号MISC_MAJOR(即10),但次设备号各不相同。

所有的miscdevice设备组成一个链表,内核在访问设备时会根据次设备号查找对应的miscdevice设备,并调用其file_operations结构中注册的文件操作接口进行操作。

内核中用struct miscdevice表示miscdevice设备,并通过调用其file_operations结构中注册的文件操作接口进行操作。

miscdevice的API实现位于drivers/char/misc.c中。

Linux驱动开发: 杂项字符设备

二、描述杂项设备的结构

内核使用struct miscdevice结构体来描述杂项设备:

struct miscdevice  {
    int minor;                       //次设备号,杂项设备的主设备号为10
    const char *name;                //设备的名称
    const struct file_operations *fops;   //文件操作
    /* 下面的成员是供内核使用,驱动编写不需要理会 */
    struct list_head list;      //misc_list的链表头
    struct device *parent;      //父设备
    struct device *this_device; //当前设备,是device_create的返回值
};

minor:次设备号,范围为0~255,当传递255时,内核会自动分配一个可用的次设备号。

name:设备名称,用于在/dev/目录下生成设备节点名称。

fops:文件操作方法指针。

特点:安装此类驱动后,会在系统的/dev目录下生成相应的设备节点文件。

三、内核提供编写杂项设备的API函数

3.1 注册函数

函数原型 int misc_register(struct miscdevice *misc)
头文件 #include
功能 注册一个杂项设备
参数 misc - 杂项设备的核心结构指针,至少已经实现minor、name、fops三个成员
返回值 0,表示注册成功;负数,表示注册失败

3.2 注销函数

函数原型 int misc_deregister(struct miscdevice *misc)
头文件 #include
功能 注销一个已存在的杂项设备
参数 misc - 杂项设备的核心结构指针,已注册的struct miscdevice结构
返回值 0,表示注销成功;负数,表示注销失败

四、杂项设备的设备号与特征

设备号用于标识设备,分为主设备号和次设备号。杂项设备的设备号如下:

Android linux内核介绍 中文WORD版
Android linux内核介绍 中文WORD版

本文档主要讲述的是Android linux内核介绍;Android内核是基于Linux 2.6内核的(目前最新开发版本是2.6.31),它是一个增强内核版本,除了修改部分Bug外,它提供了用于支持Android平台的设备驱动,希望本文档会给有需要的朋友带来帮助;感兴趣的朋友可以过来看看

下载
  • 主设备号:固定为10。
  • 次设备号:0~255。

a. 主设备号固定为10。

b. 注册后会自动在/dev/目录下生成设备文件。

c. 使用一个核心结构struct miscdevice封装。

五、编写驱动程序步骤如下:

1)编写一个基本的模块代码。

2)添加设备模型所需的头文件。

3)在模块的初始化函数中注册设备对应的结构体。

4)在模块的出口函数中注销设备对应的结构。

5)根据注册函数所需的参数,反向推导出应有的结构体成员。

针对杂项模型:

1)定义一个struct miscdevice,并填充minor、fops、name成员。

2)定义一个struct file_operations,并填充需要的成员。

3)在初始化函数中调用misc_register注册上一步实现的struct miscdevice结构变量。

4)在出口函数中调用misc_deregister注销上一步实现的struct miscdevice结构变量。

5)编写一个应用程序来测试驱动是否按预期运行,并对比结果。

最核心的工作是实现file_operations的接口函数,这些函数才是真正操作硬件的代码。其余部分属于模型代码。

六、杂项设备示例

6.1. 驱动程序代码清单

/*驱动代码 misc.c */
#include <linux/module.h>       /* Needed by all modules */
#include <linux/init.h>         /* Needed for the module-macros */
#include <linux/fs.h>
#include <linux/miscdevice.h>
<p>static ssize_t misc_read (struct file <em>pfile,
char __user </em>buff,
size_t size, loff_t *off)
{
printk(KERN_EMERG "line:%d,%s is call\n",<strong>LINE</strong>,<strong>FUNCTION</strong>);
return 0;
}</p><p>static ssize_t misc_write(struct file <em>pfile,
const char __user </em>buff,
size_t size, loff_t *off)
{
printk(KERN_EMERG "line:%d,%s is call\n",<strong>LINE</strong>,<strong>FUNCTION</strong>);
return 0;
}</p><p>static int  misc_open(struct inode <em>pinode, struct file </em>pfile)
{
printk(KERN_EMERG "line:%d,%s is call\n",<strong>LINE</strong>,<strong>FUNCTION</strong>);
return 0;
}</p><p>static int misc_release(struct inode <em>pinode, struct file </em>pfile)
{
printk(KERN_EMERG "line:%d,%s is call\n",<strong>LINE</strong>,<strong>FUNCTION</strong>);
return 0;
}</p><p>//文件操作方法:根据具体设备实现需要的功能
static const  struct file_operations misc_fops = {
.read     = misc_read,
.write    = misc_write,
.release  = misc_release,
.open     = misc_open,
};</p><h1>define DEV_NAME  "abc"</h1><h1>if 1  //C99 标准 新增加的结构体初始化方法,这种可以选其中部分进行初始化,初始化顺序不限</h1><p>static struct miscdevice  misc_dev =  {
.fops  =  &misc_fops,       /<em> 设备文件操作方法 </em>/
.minor = 255,                    /<em> 次设备号 </em>/
.name  =  DEV_NAME,           /<em> 设备名/dev/下的设备节点名   </em>/
};</p><h1>else  //c89标准风格的结构体成员初始,只能按顺序初始化成员(做了解)</h1><p>static struct miscdevice  misc_dev =  {
255,              /<em> 次设备号 </em>/
DEV_NAME,         /<em> 设备名/dev/下的设备节点名   </em>/
&misc_fops        /<em> 设备文件操作方法 </em>/
};</p><h1>endif</h1><p>static int __init hello_init(void)  {
misc_register(&misc_dev);
printk(KERN_EMERG "misc init \n");
return 0;
}</p><p>static void  __exit hello_exit(void){
misc_deregister(&misc_dev);
printk(KERN_EMERG "Goodbye,misc\n");
}</p><p>module_init(hello_init);
module_exit(hello_exit);
MODULE_LICENSE("GPL");

6.2. 应用程序代码

#include <stdio.h></p><h1>include <stdlib.h></h1><h1>include <string.h></h1><h1>include <unistd.h></h1><h1>include <sys/types.h></h1><h1>include <sys/stat.h></h1><h1>include <fcntl.h></h1><h1>define DEV_NAME  "/dev/abc"</h1><p>int main(void){
char buf[] = {1, 0, 0, 0};
int i = 0;
int fd;       //打开设备文件 O_RDWR,  O_RDONLY, O_WRONLY,
fd = open(DEV_NAME, O_RDWR);
if(fd < 0){
perror("open");
exit(1);
}</p><pre class="brush:php;toolbar:false;"><code>while(1){
    i++;
    if(i > 10000)
        break;
    write(fd, buf, 4);
    sleep(1);
}
close(fd);
return 0;

}

6.3. Makefile 代码

obj-m := misc.o
KDIR   := /home/work/linux-3.5</p><p>all:
make -C $(KDIR) M=$(PWD) modules
rm -f <em>.o </em>.mod.o <em>.mod.c </em>.symvers <em>.markers </em>.unsigned <em>.order </em>~
arm-linux-gcc app.c -o app
cp -f *.ko  app /home/work/rootfs/root</p><p>clean:
rm -f <em>.ko </em>.o <em>.mod.o </em>.mod.c <em>.symvers </em>.markers <em>.unsigned </em>.order *~

相关文章

驱动精灵
驱动精灵

驱动精灵基于驱动之家十余年的专业数据积累,驱动支持度高,已经为数亿用户解决了各种电脑驱动问题、系统故障,是目前有效的驱动软件,有需要的小伙伴快来保存下载体验吧!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

429

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

201

2025.07.04

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

930

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

603

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

294

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

212

2025.08.29

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1824

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

594

2025.10.17

Rust内存安全机制与所有权模型深度实践
Rust内存安全机制与所有权模型深度实践

本专题围绕 Rust 语言核心特性展开,深入讲解所有权机制、借用规则、生命周期管理以及智能指针等关键概念。通过系统级开发案例,分析内存安全保障原理与零成本抽象优势,并结合并发场景讲解 Send 与 Sync 特性实现机制。帮助开发者真正理解 Rust 的设计哲学,掌握在高性能与安全性并重场景中的工程实践能力。

4

2026.03.05

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 10.1万人学习

Git 教程
Git 教程

共21课时 | 4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号