当前位置: 首页 > news >正文

【Linux】【字符设备驱动】深入解析

在这里插入图片描述

Linux字符设备驱动程序用于控制不支持随机访问的硬件设备,如串行端口、打印机、调制解调器等。这类设备通常以字符流的形式与用户空间程序进行交互。本节将深入探讨字符设备驱动的设计原理、实现细节及其与内核其他组件的交互。

1. 引言

字符设备驱动程序是Linux内核的重要组成部分,它们负责处理来自用户空间应用程序的读写请求,并将这些请求转换为对硬件设备的实际操作。字符设备驱动程序通常用于控制串行端口、打印机、调制解调器等设备。

2. 字符设备的基本概念

2.1 设备号

在Linux系统中,每个字符设备都有一个唯一的设备号,设备号由主设备号和次设备号组成。主设备号用于标识一组相关设备,次设备号用于区分同一组中的不同设备。

2.1.1 设备号分配

设备号由内核动态分配或手动指定。例如:

sudo mknod /dev/my_char_dev c 240 0

这里,240 是主设备号,0 是次设备号。

2.2 设备文件

字符设备文件通常位于 /dev 目录下,它们代表了实际的硬件设备。设备文件可以通过 mknod 命令创建,也可以通过 udev 规则自动创建。

2.2.1 创建设备文件

使用 mknod 命令创建一个字符设备文件:

sudo mknod /dev/my_char_dev c 240 0

2.3 设备驱动程序

字符设备驱动程序负责处理来自用户空间应用程序的请求,并将这些请求转换为对硬件设备的操作。驱动程序通常包括初始化函数、读写操作函数等。

2.3.1 初始化函数

驱动程序初始化函数负责注册设备号、初始化 file_operations 结构体、注册字符设备等。

2.3.2 文件操作结构

struct file_operations 结构体包含了各种文件操作函数指针,如 readwriteopen 等。

3. 字符设备驱动程序的底层原理

3.1 内核与用户空间交互

字符设备驱动程序通过内核提供的接口与用户空间应用程序进行交互。当用户空间程序对设备文件进行读写操作时,内核会调用相应的驱动程序函数。

3.1.1 系统调用

系统调用是内核与用户空间交互的主要方式。当用户空间程序调用 readwrite 系统调用时,内核会调用对应的驱动程序函数。

3.1.2 文件操作结构

struct file_operations 结构体定义了一系列文件操作函数,这些函数由内核在适当的时机调用。

struct file_operations {int (*read)(struct file *, char __user *, size_t, loff_t *);int (*write)(struct file *, const char __user *, size_t, loff_t *);int (*open)(struct inode *, struct file *);int (*release)(struct inode *, struct file *);...
};

3.2 设备注册与管理

3.2.1 设备号注册

设备号注册是在内核中创建一个设备文件的步骤之一。设备号由主设备号和次设备号组成,通过 register_chrdev_region 函数注册。

register_chrdev_region(dev_num, 1, "my_char_dev");
3.2.2 设备文件操作结构初始化

设备文件操作结构初始化包括设置读写操作函数等。

cdev_init(&c_dev, &fops);
3.2.3 设备注册

设备注册是将设备文件操作结构添加到内核中,使设备可以被访问。

cdev_add(&c_dev, dev_num, 1);

3.3 设备操作函数

字符设备驱动程序需要实现一系列设备操作函数,如读、写、打开、关闭等。

3.3.1 读操作

读操作函数负责从设备中读取数据,并返回给用户空间程序。

static ssize_t dev_read(struct file *file, char __user *buf, size_t count, loff_t *ppos)
{if (*ppos >= PAGE_SIZE)return 0;if (copy_to_user(buf, &buf[*ppos], count))return -EFAULT;*ppos += count;return count;
}
3.3.2 写操作

写操作函数负责将用户空间程序的数据写入设备。

static ssize_t dev_write(struct file *file, const char __user *buf, size_t count, loff_t *ppos)
{if (*ppos >= PAGE_SIZE)return -ENOSPC;if (copy_from_user(&buf[*ppos], buf, count))return -EFAULT;*ppos += count;return count;
}
3.3.3 打开和关闭操作

打开操作函数负责初始化设备,关闭操作函数负责释放设备资源。

static int dev_open(struct inode *inode, struct file *file)
{printk(KERN_INFO "Device opened.\n");return 0;
}static int dev_release(struct inode *inode, struct file *file)
{printk(KERN_INFO "Device closed.\n");return 0;
}

4. 字符设备驱动程序的编写

4.1 模块初始化和卸载

字符设备驱动程序通常是一个内核模块,模块需要定义初始化和卸载函数。

4.1.1 初始化函数

初始化函数通常包括注册设备号、初始化 file_operations 结构体、注册字符设备等。

static int __init dev_init(void)
{// 注册字符设备register_chrdev_region(dev_num, 1, "my_char_dev");// 初始化字符设备结构cdev_init(&c_dev, &fops);// 添加字符设备到设备类class = class_create(THIS_MODULE, "my_char_class");device = device_create(class, NULL, dev_num, NULL, "my_char_dev");// 注册字符设备cdev_add(&c_dev, dev_num, 1);return 0;
}
4.1.2 卸载函数

卸载函数负责注销字符设备、销毁设备类等。

static void __exit dev_exit(void)
{// 删除字符设备cdev_del(&c_dev);// 移除设备device_destroy(class, dev_num);// 销毁设备类class_unregister(class);// 注销字符设备区域unregister_chrdev_region(dev_num, 1);
}

4.2 设备操作函数

字符设备驱动程序需要实现一系列设备操作函数,如读、写、打开、关闭等。

4.2.1 读操作

读操作函数负责从设备中读取数据,并返回给用户空间程序。

static ssize_t dev_read(struct file *file, char __user *buf, size_t count, loff_t *ppos)
{if (*ppos >= PAGE_SIZE)return 0;if (copy_to_user(buf, &buf[*ppos], count))return -EFAULT;*ppos += count;return count;
}
4.2.2 写操作

写操作函数负责将用户空间程序的数据写入设备。

static ssize_t dev_write(struct file *file, const char __user *buf, size_t count, loff_t *ppos)
{if (*ppos >= PAGE_SIZE)return -ENOSPC;if (copy_from_user(&buf[*ppos], buf, count))return -EFAULT;*ppos += count;return count;
}
4.2.3 打开和关闭操作

打开操作函数负责初始化设备,关闭操作函数负责释放设备资源。

static int dev_open(struct inode *inode, struct file *file)
{printk(KERN_INFO "Device opened.\n");return 0;
}static int dev_release(struct inode *inode, struct file *file)
{printk(KERN_INFO "Device closed.\n");return 0;
}

5. 设备文件的创建

设备文件通常位于 /dev 目录下,可以通过 mknod 命令或 udev 规则创建。

5.1 使用 mknod 创建设备文件

sudo mknod /dev/my_char_dev c 240 0

5.2 使用 udev 规则自动创建设备文件

可以编写 udev 规则来自动创建设备文件:

# /etc/udev/rules.d/99-my-device.rulesKERNEL=="my_char_dev", MODE="0660", OWNER="root", GROUP="users", SYMLINK+="my_char_dev"

6. 用户空间程序

用户空间程序用于读写字符设备文件。

6.1 用户空间程序示例

编写一个简单的用户空间程序来读写设备文件:

#include <stdio.h>
#include <fcntl.h>
#include <unistd.h>int main(int argc, char *argv[])
{int fd;char buffer[256];ssize_t bytes_written, bytes_read;// 打开设备文件fd = open("/dev/my_char_dev", O_RDWR);if (fd == -1) {perror("Failed to open device");return 1;}// 写入数据bytes_written = write(fd, "Hello, World!", 13);if (bytes_written == -1) {perror("Failed to write to device");close(fd);return 1;}// 读取数据bytes_read = read(fd, buffer, sizeof(buffer));if (bytes_read == -1) {perror("Failed to read from device");close(fd);return 1;}printf("Read from device: %.*s\n", (int)bytes_read, buffer);// 关闭设备文件close(fd);return 0;
}

6.2 编译和运行用户空间程序

编译用户空间程序:

gcc -o my_prog my_prog.c

运行用户空间程序:

./my_prog

7. 字符设备驱动的调试

7.1 使用 printk 调试

printk 函数可以用于在内核模块中输出调试信息。

printk(KERN_INFO "Hello, World!\n");

7.2 使用 syslog 调试

syslog 函数可以用于将调试信息发送到系统日志。

#include <linux/syscalls.h>
...
syslog(KERN_INFO, "Hello, World!");

8. 字符设备驱动的优化

8.1 避免死锁

在多线程或多进程环境下,需要小心处理设备的读写操作,避免死锁。

8.1.1 互斥锁

使用互斥锁来保护共享资源。

static DEFINE_MUTEX(my_mutex);mutex_lock(&my_mutex);
// 执行关键操作
mutex_unlock(&my_mutex);

8.2 提高性能

通过优化读写操作,减少不必要的上下文切换,提高设备驱动程序的性能。

8.2.1 非阻塞 I/O

支持非阻塞 I/O 可以提高设备驱动程序的性能。

static int dev_poll(struct file *file, poll_table *wait)
{poll_wait(file, wait, POLLIN | POLLOUT);return POLLIN | POLLOUT;
}static const struct file_operations fops = {.owner          = THIS_MODULE,.read           = dev_read,.write          = dev_write,.open           = dev_open,.release        = dev_release,.poll           = dev_poll,
};

9. 字符设备驱动的应用案例

9.1 串行通信设备驱动

串行通信设备驱动用于控制串行端口,如 COM 端口。

// 串行通信设备驱动示例// 设备打开操作
static int dev_open(struct inode *inode, struct file *file)
{// 初始化串行端口init_serial_port();return 0;
}// 设备关闭操作
static int dev_release(struct inode *inode, struct file *file)
{// 释放串行端口资源release_serial_port();return 0;
}// 设备读操作
static ssize_t dev_read(struct file *file, char __user *buf, size_t count, loff_t *ppos)
{// 从串行端口读取数据read_from_serial_port(buf, count);return count;
}// 设备写操作
static ssize_t dev_write(struct file *file, const char __user *buf, size_t count, loff_t *ppos)
{// 将数据写入串行端口write_to_serial_port(buf, count);return count;
}

9.2 LED 控制设备驱动

LED 控制设备驱动用于控制 LED 灯。

// LED 控制设备驱动示例// 设备打开操作
static int dev_open(struct inode *inode, struct file *file)
{// 初始化 LEDinit_led();return 0;
}// 设备关闭操作
static int dev_release(struct inode *inode, struct file *file)
{// 释放 LED 资源release_led();return 0;
}// 设备读操作
static ssize_t dev_read(struct file *file, char __user *buf, size_t count, loff_t *ppos)
{// 读取 LED 状态read_led_status(buf, count);return count;
}// 设备写操作
static ssize_t dev_write(struct file *file, const char __user *buf, size_t count, loff_t *ppos)
{// 设置 LED 状态set_led_status(buf, count);return count;
}

10. 字符设备驱动的高级特性

10.1 设备属性

设备属性允许用户通过 /sys/class/<class>/device 目录下的文件来查询和修改设备状态。

10.1.1 添加设备属性

使用 sysfs API 添加设备属性:

static ssize_t show_attr(struct device *dev, struct device_attribute *attr, char *buf)
{return sprintf(buf, "%d\n", some_value);
}static ssize_t store_attr(struct device *dev, struct device_attribute *attr, const char *buf, size_t count)
{some_value = atoi(buf);return count;
}static DEVICE_ATTR(my_attr, 0644, show_attr, store_attr);static struct attribute *my_attrs[] = {&dev_attr_my_attr.attr,NULL,
};static struct device_type my_dev_type = {.name = "my_char_dev",.attrs = my_attrs,
};class_create(THIS_MODULE, "my_char_class");
device_create(class, NULL, dev_num, NULL, "my_char_dev");
device_create_file(device, &dev_attr_my_attr);

10.2 设备同步

为了保证数据的一致性和完整性,设备驱动程序需要处理同步问题。

10.2.1 原子操作

使用原子操作来保护数据的一致性。

static atomic_t my_counter;atomic_inc(&my_counter);
atomic_dec(&my_counter);

11. 字符设备驱动的错误处理

11.1 错误检测

在设备驱动程序中检测和处理错误是必要的。

11.1.1 返回值检查

使用返回值来检查函数是否成功执行。

if (ioctl(fd, MY_IOCTL_CMD, &arg) < 0) {perror("ioctl failed");return -1;
}

11.2 错误报告

在设备驱动程序中报告错误信息。

11.2.1 日志记录

使用 printksyslog 记录错误信息。

printk(KERN_ERR "Error occurred in driver.\n");

12. 总结

Linux字符设备驱动程序用于控制不支持随机访问的硬件设备,如串行端口、打印机等。通过编写字符设备驱动程序,可以实现对这些设备的高效控制。希望本文能帮助读者更好地理解和掌握Linux字符设备驱动程序的开发技巧,并深入了解其底层原理。

相关文章:

【Linux】【字符设备驱动】深入解析

Linux字符设备驱动程序用于控制不支持随机访问的硬件设备&#xff0c;如串行端口、打印机、调制解调器等。这类设备通常以字符流的形式与用户空间程序进行交互。本节将深入探讨字符设备驱动的设计原理、实现细节及其与内核其他组件的交互。 1. 引言 字符设备驱动程序是Linux内…...

【JavaEE】多线程(2)

一、线程安全 1.1 线程安全的概念 线程是随机调度执行的&#xff0c;如果多线程环境下的程序运行的结果符合我们预期则说明线程安全&#xff0c;反之&#xff0c;如果遇到其他结果甚至引起了bug则说明线程不安全 1.2 经典例子与解释 下面举一个经典的线程不安全的例子&…...

mac下Gpt Chrome升级成GptBrowser书签和保存的密码恢复

cd /Users/自己的用户名/Library/Application\ Support/ 目录下有 GPT\ Chrome/ Google/ GptBrowser/ GPT\ Chrome 为原来的chrome浏览器的文件存储目录. GptBrowser 为升级后chrome浏览器存储目录 书签所在的文件 Bookmarks 登录账号Login 相关的文件 拷贝到GptBrow…...

使用Grafana K6来测测你的系统负载能力

背景 近期我们有个号称会有很高很高并发的系统要上线&#xff0c;为了测试一下自己开发的系统的负载能力&#xff0c;准备了点海克斯科技&#xff0c;来看看抗不抗的住。 之前笔者写过用Apache JMeter进行压力测试的文章&#xff08;传送门&#x1f449;&#xff1a;https://…...

【论文复现】基于BERT的语义分析实现

&#x1f4dd;个人主页&#x1f339;&#xff1a;Eternity._ &#x1f339;&#x1f339;期待您的关注 &#x1f339;&#x1f339; ❀ WRN: 宽度残差网络 概述语义分类文本分类情感分类 实现原理 核心逻辑pre_deal.pytrain.pytest_demo.py 实现方式&演示效果训练阶段测试阶…...

CTF-RE: STL逆向 [NewStarCTF 2023 公开赛道 STL] WP

多看看STL题就会了,很简单 int __fastcall main(int argc, const char **argv, const char **envp) {__int64 v3; // rbx__int64 v4; // raxchar v5; // bl_BYTE *v6; // rax_QWORD *v7; // rax__int64 v8; // rax__int64 v9; // raxint i; // [rsp0h] [rbp-250h]int j; // [r…...

实习冲刺第三十六天

46.全排列 给定一个不含重复数字的数组 nums &#xff0c;返回其 所有可能的全排列 。你可以 按任意顺序 返回答案。 示例 1&#xff1a; 输入&#xff1a;nums [1,2,3] 输出&#xff1a;[[1,2,3],[1,3,2],[2,1,3],[2,3,1],[3,1,2],[3,2,1]]示例 2&#xff1a; 输入&#…...

【Zemax光学设计实训三】---激光缩束镜的设计优化

前言与目录 技术设计要求&#xff1a; 设计一个激光扩束镜&#xff0c;使用的波长为1064nm&#xff0c;输入光束直径为10mm&#xff0c;输出光束的直径为2mm&#xff0c;且输入光束和输出光束平行&#xff08;即平行光入射&#xff0c;平行光出射&#xff09;。要求只使用两片…...

TCP/IP协议簇自学笔记

摘抄于大学期间记录在QQ空间的一篇自学笔记&#xff0c;当前清理空间&#xff0c;本来想直接删除掉的&#xff0c;但是感觉有些舍不得&#xff0c;因此先搬移过来。 曾经&#xff0c;我只知道socket函数能进行网络间数据的通信&#xff0c;知道tcp/ip协议也是用来进行网络数据…...

Spring Boot教程之十一:获取Request 请求 和 Put请求

如何在 Spring Boot 中获取Request Body&#xff1f; Java 语言是所有编程语言中最流行的语言之一。使用 Java 编程语言有几个优点&#xff0c;无论是出于安全目的还是构建大型分发项目。使用 Java 的优点之一是 Java 试图借助类、继承、多态等概念将语言中的每个概念与现实世…...

计算机网络(二)

ip地址&#xff1a;11010010&#xff1a;01011110:00100100:00010100 子网掩码:11111111:11111111:11111111:11000000 and &#xff1a;11010010&#xff1a;01011110&#xff1a;00100100&#xff1a;00000000 210.94.36.0的下一站为R1 因为255为11111111 192为&#xff…...

如何在Python中进行数学建模?

数学建模是数据科学中使用的强大工具&#xff0c;通过数学方程和算法来表示真实世界的系统和现象。Python拥有丰富的库生态系统&#xff0c;为开发和实现数学模型提供了一个很好的平台。本文将指导您完成Python中的数学建模过程&#xff0c;重点关注数据科学中的应用。 数学建…...

JavaSE——类与对象(5)

一、抽象类 1.1为什么需要抽象类 父类的某些方法&#xff0c;不确定怎么实现&#xff0c;也不需要实现。 class Animal{public String name;public Animal(String name){this.name name;}public void eat()//这里实现了也没有意义{System.out.println("这是一个动物&am…...

Istio笔记01--快速体验Istio

Istio笔记01--快速体验Istio 介绍部署与测试部署k8s安装istio测试istio 注意事项说明 介绍 Istio是当前最热门的服务网格产品&#xff0c;已经被广泛应用于各个云厂商和IT互联网公司。企业可以基于Istio轻松构建服务网格&#xff0c;在接入过程中应用代码无需更改&#xff0c;…...

面试小札:Java如何实现并发编程

多线程基础 继承Thread类 定义一个类继承自 Thread 类&#xff0c;重写 run 方法。在 run 方法中编写线程要执行的任务逻辑。例如&#xff1a; java class MyThread extends Thread { Override public void run() { System.out.println("线程执行的任务…...

java-a+b 开启java语法学习

代码 &#xff08;ab) import java.util.Scanner; //导入 java.util包中的Scanner 类&#xff0c;允许读取键盘输入数据public class Main { // 创建一个公共类 Mainpublic static void main(String[] args) {//程序入口点&#xff0c;main方法Scanner scanner new Scanner(…...

RNN模型文本预处理--数据增强方法

数据增强方法 数据增强是自然语言处理&#xff08;NLP&#xff09;中常用的一种技术&#xff0c;通过生成新的训练样本来扩充数据集&#xff0c;从而提高模型的泛化能力和性能。回译数据增强法是一种常见的数据增强方法&#xff0c;特别适用于文本数据。 回译数据增强法 定义…...

maven 中<packaging>pom</packaging>配置使用

在 Maven 项目的 pom.xml 文件中&#xff0c; 元素用于指定项目的打包类型。默认情况下&#xff0c;如果 元素没有被显式定义&#xff0c;Maven 会假设其值为 jar。但是&#xff0c;当您设置 pom 时&#xff0c;这意味着该项目是一个 POM&#xff08;Project Object Model&…...

【Python中while循环】

一、深拷贝、浅拷贝 1、需求 1&#xff09;拷贝原列表产生一个新列表 2&#xff09;想让两个列表完全独立开&#xff08;针对改操作&#xff0c;读的操作不改变&#xff09; 要满足上述的条件&#xff0c;只能使用深拷贝 2、如何拷贝列表 1&#xff09;直接赋值 # 定义一个…...

【深度学习】服务器常见命令

1、虚拟环境的安装位置 先进入虚拟环境 which python2、升序查看文件内容 ls -ltr3、查看服务器主机空间使用情况 df -hdf -h .4、查看本地空间使用情况 du -sh ./*du -sh * | sort -nr5、查找并删除进程 # 查找 ps aux# 删除 kill -KILL pid6、查看服务器配置 lscpuuna…...

【网络】每天掌握一个Linux命令 - iftop

在Linux系统中&#xff0c;iftop是网络管理的得力助手&#xff0c;能实时监控网络流量、连接情况等&#xff0c;帮助排查网络异常。接下来从多方面详细介绍它。 目录 【网络】每天掌握一个Linux命令 - iftop工具概述安装方式核心功能基础用法进阶操作实战案例面试题场景生产场景…...

Ubuntu系统下交叉编译openssl

一、参考资料 OpenSSL&&libcurl库的交叉编译 - hesetone - 博客园 二、准备工作 1. 编译环境 宿主机&#xff1a;Ubuntu 20.04.6 LTSHost&#xff1a;ARM32位交叉编译器&#xff1a;arm-linux-gnueabihf-gcc-11.1.0 2. 设置交叉编译工具链 在交叉编译之前&#x…...

基于FPGA的PID算法学习———实现PID比例控制算法

基于FPGA的PID算法学习 前言一、PID算法分析二、PID仿真分析1. PID代码2.PI代码3.P代码4.顶层5.测试文件6.仿真波形 总结 前言 学习内容&#xff1a;参考网站&#xff1a; PID算法控制 PID即&#xff1a;Proportional&#xff08;比例&#xff09;、Integral&#xff08;积分&…...

ssc377d修改flash分区大小

1、flash的分区默认分配16M、 / # df -h Filesystem Size Used Available Use% Mounted on /dev/root 1.9M 1.9M 0 100% / /dev/mtdblock4 3.0M...

oracle与MySQL数据库之间数据同步的技术要点

Oracle与MySQL数据库之间的数据同步是一个涉及多个技术要点的复杂任务。由于Oracle和MySQL的架构差异&#xff0c;它们的数据同步要求既要保持数据的准确性和一致性&#xff0c;又要处理好性能问题。以下是一些主要的技术要点&#xff1a; 数据结构差异 数据类型差异&#xff…...

Qwen3-Embedding-0.6B深度解析:多语言语义检索的轻量级利器

第一章 引言&#xff1a;语义表示的新时代挑战与Qwen3的破局之路 1.1 文本嵌入的核心价值与技术演进 在人工智能领域&#xff0c;文本嵌入技术如同连接自然语言与机器理解的“神经突触”——它将人类语言转化为计算机可计算的语义向量&#xff0c;支撑着搜索引擎、推荐系统、…...

Keil 中设置 STM32 Flash 和 RAM 地址详解

文章目录 Keil 中设置 STM32 Flash 和 RAM 地址详解一、Flash 和 RAM 配置界面(Target 选项卡)1. IROM1(用于配置 Flash)2. IRAM1(用于配置 RAM)二、链接器设置界面(Linker 选项卡)1. 勾选“Use Memory Layout from Target Dialog”2. 查看链接器参数(如果没有勾选上面…...

Ascend NPU上适配Step-Audio模型

1 概述 1.1 简述 Step-Audio 是业界首个集语音理解与生成控制一体化的产品级开源实时语音对话系统&#xff0c;支持多语言对话&#xff08;如 中文&#xff0c;英文&#xff0c;日语&#xff09;&#xff0c;语音情感&#xff08;如 开心&#xff0c;悲伤&#xff09;&#x…...

【JavaWeb】Docker项目部署

引言 之前学习了Linux操作系统的常见命令&#xff0c;在Linux上安装软件&#xff0c;以及如何在Linux上部署一个单体项目&#xff0c;大多数同学都会有相同的感受&#xff0c;那就是麻烦。 核心体现在三点&#xff1a; 命令太多了&#xff0c;记不住 软件安装包名字复杂&…...

tree 树组件大数据卡顿问题优化

问题背景 项目中有用到树组件用来做文件目录&#xff0c;但是由于这个树组件的节点越来越多&#xff0c;导致页面在滚动这个树组件的时候浏览器就很容易卡死。这种问题基本上都是因为dom节点太多&#xff0c;导致的浏览器卡顿&#xff0c;这里很明显就需要用到虚拟列表的技术&…...