【Linux】驱动
驱动
驱动程序过程
系统调用
用户空间
内核空间
添加驱动和调用驱动
驱动程序是如何调用设备硬件
驱动
在计算机领域,驱动(Driver)是一种软件,它充当硬件设备与操作系统之间的桥梁,允许它们进行通信和协同工作。驱动程序的主要功能是向操作系统提供一种标准化的接口,使得操作系统可以与硬件设备进行交互,而无需了解设备的具体实现细节。
具体而言,驱动程序通常包括以下方面的功能:
-
设备控制: 驱动程序负责向硬件设备发送命令和控制信息,以执行特定的操作,如读取数据、写入数据、初始化设备等。
-
中断处理: 驱动程序能够处理硬件设备生成的中断信号,从而及时响应设备状态的变化。
-
资源管理: 驱动程序管理设备所需的资源,如内存、输入输出端口等,以确保不同设备之间的资源冲突得到解决。
-
提供接口: 驱动程序通过向操作系统提供标准接口,使得应用程序能够通过操作系统来访问和控制硬件设备。
-
与操作系统交互: 驱动程序与操作系统内核进行交互,通过系统调用、中断服务例程等机制实现与操作系统的协同工作。
驱动程序在操作系统层次结构中处于内核空间,与硬件直接交互。不同操作系统有不同的驱动程序模型,例如在Linux系统中,驱动程序通常作为内核模块加载,而在Windows系统中,驱动程序以.sys文件的形式存在。
总的来说,驱动程序是连接操作系统和硬件设备之间的软件层,使得它们能够协同工作,实现计算机系统的各种功能。
驱动程序过程
驱动程序的运行流程涉及到多个层次,从用户空间到内核态,再到硬件层。
用户空间的应用程序、内核空间的系统调用、VFS、设备驱动程序以及硬件层的交互。
下面是一个简要的概述:
用户空间
-
应用程序: 用户编写的应用程序需要访问文件或设备。
-
C库(libc): 应用程序通过C库中的函数(例如
open
、read
、write
)来实现文件和设备的访问。 -
系统调用: C库中的函数最终会导致系统调用。这个过程通常包括:
- 应用程序通过C库函数调用触发相应的系统调用,如
open
。 - C库中的系统调用包装器将参数放入寄存器中,触发软中断。
- 应用程序通过C库函数调用触发相应的系统调用,如
-
软中断: 触发软中断时,操作系统会切换到内核态执行中断服务例程。在 x86 架构中,通过
int 0x80
指令触发软中断,中断号为0x80
。
内核空间
-
中断服务例程: 操作系统中断服务例程处理软中断,执行相应的系统调用服务例程。对于
int 0x80
,会执行相应的中断服务例程。 -
系统调用服务例程: 系统调用服务例程根据中断号调用相应的系统调用处理函数。例如,
0x80
对应于sys_call
,这是一个中央的系统调用处理函数。 -
VFS(虚拟文件系统): 在系统调用中,VFS提供了对文件系统的抽象。例如,对于
open
系统调用,VFS将根据路径名找到相应的文件系统,然后调用该文件系统的open
函数。 -
sys_open: 在 VFS 中,
sys_open
是open
系统调用的具体实现。它会检查设备名和路径,并通过文件系统的驱动程序找到相应的文件。
设备层
-
设备驱动程序: 当 VFS 需要访问硬件设备时,它会调用相应设备文件对应的设备驱动程序。
- 设备驱动程序负责管理硬件设备的底层细节,如与设备的通信、中断处理等。
- 驱动程序通过与硬件设备的接口进行交互,执行读写等操作。
-
硬件层: 驱动程序与硬件层进行通信,实现对硬件设备的具体控制。
整个过程中,VFS起到了一个桥梁的作用,使得用户空间应用程序无需关心底层硬件和文件系统的细节。具体的系统调用、VFS、设备驱动程序的实现会依赖于操作系统和硬件平台。
驱动程序的编写涉及到内核模块的开发,需要熟悉设备文件、系统调用、VFS等相关概念和接口。
系统调用
系统调用是操作系统提供给用户程序或软件的一组接口,用于访问操作系统的服务和资源。通过系统调用,用户程序可以请求操作系统执行特权指令、访问硬件设备、进行文件操作等。
在 Linux 中,系统调用是用户程序与内核之间的接口,用于执行一些只有内核才能执行的特权操作。以下是一些常见的 Linux 系统调用:
-
open
: 用于打开文件,返回文件描述符。 -
close
: 用于关闭文件。 -
read
: 用于从文件中读取数据。 -
write
: 用于向文件中写入数据。 -
ioctl
: 用于设备的控制操作。 -
fork
: 用于创建新的进程。 -
exec
: 用于加载新的程序到当前进程中。 -
exit
: 用于退出当前进程。 -
kill
: 用于向进程发送信号。 -
wait
: 用于等待子进程退出。 -
stat
: 用于获取文件状态信息。 -
mmap
: 用于在进程的地址空间中映射文件或设备。
这些系统调用是通过中断(软中断)来实现的。当用户程序执行系统调用时,会触发一个软中断,将控制权转移到内核态,内核会根据系统调用号来执行相应的功能。系统调用提供了一种用户程序与内核之间的标准接口,使用户程序能够安全而受控地访问底层操作系统的功能。
在 C 语言中,可以使用 syscall
函数或者直接调用包装好的库函数来进行系统调用。例如,open
系统调用可以通过 open
函数在 C 语言中调用。
用户空间
用户空间(User Space)是指操作系统中划分给用户进程运行的地址空间部分。在计算机系统中,操作系统内核和用户应用程序是两个主要的运行空间,它们各自拥有独立的内存空间。
以下是用户空间的一些关键特点和组成部分:
-
地址空间: 用户空间是指分配给用户进程的地址范围,通常从0开始,到系统的最大地址。在32位系统中,用户空间通常从0到4GB,而在64位系统中,用户空间范围更大。
-
用户进程: 所有运行在用户空间的应用程序都是用户进程的一部分。这些进程通过系统调用等方式与操作系统进行通信,请求服务或执行特权操作。
-
用户程序: 用户空间包含了用户程序的执行代码、数据段和堆栈。用户程序是用户进程的核心,它们通过调用系统提供的服务来执行特定的任务。
-
动态链接库: 用户空间中还包含了一些动态链接库(Dynamic Link Libraries,DLL)或共享库,这些库包含了一些通用的功能和程序模块,可以被多个应用程序共享使用,减少了代码冗余。
-
进程间通信(IPC): 不同的用户进程之间需要进行通信,而用户空间提供了各种IPC机制,例如管道、消息队列、共享内存等,以实现进程之间的数据交换。
-
文件系统: 用户进程通常需要访问文件系统中的文件,用户空间提供了对文件系统的访问接口,使得用户程序可以读写文件。
-
系统调用: 为了执行一些特权操作,用户进程需要通过系统调用(System Call)请求操作系统的协助。系统调用是用户空间与内核空间之间的桥梁,允许用户程序访问底层操作系统服务。
-
安全性: 用户空间的进程受到操作系统的保护,各个进程之间相互隔离,一个进程无法直接访问另一个进程的内存空间。
总体而言,用户空间是用户程序运行的环境,提供了访问系统资源的接口,使得应用程序能够在计算机系统中执行各种任务。操作系统通过对用户空间的管理和保护,确保系统的稳定性、安全性和可维护性。
内核空间
内核空间(Kernel Space)是操作系统中的一个重要部分,用于执行操作系统内核的代码和管理系统资源。与用户空间相对,内核空间拥有更高的权限和更广泛的系统访问权限。以下是内核空间的一些关键方面:
-
设备驱动: 内核空间包含设备驱动程序,用于与硬件设备进行通信。设备驱动允许内核控制和管理与计算机系统连接的各种外部设备,例如磁盘驱动器、网卡、显卡等。
-
内存管理: 内核负责系统内存的管理,包括内存分配、释放、虚拟内存管理、页表维护等。它确保不同进程之间不会互相干扰,同时有效地利用系统的物理内存。
-
进程和线程管理: 内核负责创建、终止和调度进程和线程。它管理进程的状态、优先级、调度顺序等,确保系统中的多个任务能够协调运行。
-
系统调用: 内核提供了系统调用接口,允许用户空间的程序请求内核执行特权操作。这些系统调用涉及文件操作、进程控制、网络通信等,是用户程序与内核之间的桥梁。
-
文件系统: 内核管理文件系统,包括文件的创建、删除、读取、写入等操作。它确保文件系统的一致性和安全性,并提供对文件和目录的访问权限控制。
-
网络协议栈: 内核中包含网络协议栈,负责处理网络通信。它管理网络连接、数据包传输、协议处理等,为用户空间提供网络服务。
-
中断处理: 内核负责处理硬件和软件引发的中断。中断是一种异步事件,可能来自硬件设备、时钟等。内核必须能够适时地响应中断,执行相应的中断处理程序。
-
锁和同步: 内核提供了各种同步机制,如锁、信号量等,以确保多个进程或线程之间的互斥和同步。
-
安全性: 内核对系统的安全性负有重要责任,包括对用户空间的访问权限、系统资源的保护、防止恶意代码执行等。
总体而言,内核空间是操作系统的核心,负责管理和控制系统的各个方面,以确保系统能够高效、稳定、安全地运行。内核空间的代码运行在特权级别最高的CPU模式,能够执行一些用户空间不可执行的指令,具有更高的系统访问权限。
添加驱动和调用驱动
添加驱动和调用驱动是 Linux 系统中涉及设备驱动的两个主要方面。让我们更详细地看一下这两个步骤。
1. 添加驱动
在 Linux 中,添加设备驱动通常包括以下步骤:
编写完驱动程序:
-
实现设备驱动函数: 编写设备驱动函数,该函数定义了设备的各种操作,如读、写、控制等。
-
指定设备号: 在加载驱动时,需要为设备分配一个设备号,该设备号在驱动中使用。
-
操作寄存器来驱动 IO 口: 如果设备与 IO 口通信,需要编写代码来读写寄存器,与设备进行交互。
-
注册字符设备驱动: 在模块初始化中,通过
register_chrdev
等函数注册字符设备驱动。
操作寄存器来驱动 IO 口的例子:
// 读取寄存器
unsigned char read_register(unsigned int addr) {return inb(addr);
}// 写入寄存器
void write_register(unsigned int addr, unsigned char value) {outb(value, addr);
}
注册字符设备驱动的例子:
#include <linux/fs.h>
#include <linux/cdev.h>static dev_t device_number;
static struct cdev my_cdev;// 在模块初始化中注册字符设备
static int __init my_module_init(void) {int err;// 分配设备号err = alloc_chrdev_region(&device_number, 0, 1, "my_device");if (err) {printk(KERN_ERR "Failed to allocate device number\n");return err;}// 初始化字符设备cdev_init(&my_cdev, &fops);// 注册字符设备err = cdev_add(&my_cdev, device_number, 1);if (err) {printk(KERN_ERR "Failed to add character device\n");unregister_chrdev_region(device_number, 1);return err;}printk(KERN_INFO "Device registered successfully\n");return 0;
}// 在模块卸载时注销字符设备
static void __exit my_module_exit(void) {cdev_del(&my_cdev);unregister_chrdev_region(device_number, 1);printk(KERN_INFO "Device unregistered\n");
}
2. 调用驱动
在用户空间调用设备驱动通常包括以下步骤:
-
打开设备文件: 使用
open
系统调用打开设备文件。 -
执行设备操作: 使用
read
、write
、ioctl
等系统调用执行设备操作。 -
关闭设备文件: 使用
close
系统调用关闭设备文件。
例子:
#include <fcntl.h>
#include <unistd.h>int main() {int fd = open("/dev/my_device", O_RDWR);if (fd < 0) {perror("Failed to open device");return -1;}// Perform device operations using read, write, ioctl, etc.close(fd);return 0;
}
在这个例子中,/dev/my_device
是设备文件的路径,用户程序通过打开这个文件来与设备驱动进行通信。
总体而言,添加设备驱动和调用设备驱动是 Linux 设备驱动开发的两个基本方面,其中添加驱动需要编写内核模块代码,而调用驱动则是用户空间程序通过系统调用与设备驱动进行交互。
驱动程序是如何调用设备硬件
在 Linux 系统中,驱动程序通过实现一组特定的操作函数(如 read
、write
、ioctl
等)来与设备硬件进行交互。这些操作函数定义了用户空间程序与驱动程序之间的接口。当用户程序调用系统调用(如 open
、read
、write
)时,这些调用最终会触发相应的设备操作函数,从而完成与硬件的通信。
让我们以字符设备为例,来说明驱动程序是如何调用设备硬件的:
-
打开设备文件: 用户空间程序通过
open
系统调用打开设备文件。在驱动中,open
操作会触发相应的操作函数my_open
。static int my_open(struct inode *inode, struct file *file) {// 打开设备时的操作,可以包括初始化硬件等return 0; }
-
读写设备文件: 用户空间程序通过
read
和write
等系统调用来进行读写操作。这些调用会触发相应的操作函数my_read
和my_write
。static ssize_t my_read(struct file *file, char __user *buf, size_t count, loff_t *offset) {// 读取设备的操作,向用户空间传递数据return 0; }static ssize_t my_write(struct file *file, const char __user *buf, size_t count, loff_t *offset) {// 写入设备的操作,从用户空间接收数据return 0; }
-
控制设备: 用户空间程序通过
ioctl
等系统调用来进行控制操作。这些调用会触发相应的操作函数my_ioctl
。static long my_ioctl(struct file *file, unsigned int cmd, unsigned long arg) {// 控制设备的操作,可以包括配置硬件参数等return 0; }
这些操作函数在设备的操作过程中,可以通过访问设备的寄存器、执行硬件指令等方式与硬件进行通信。驱动程序的实现要根据具体的硬件设备和需求而定。
总体而言,设备驱动程序通过实现操作函数,提供给用户空间的系统调用来与硬件设备进行通信。这种方式使得用户空间程序不需要了解底层硬件细节,而是通过抽象的接口与设备进行交互。
相关文章:

【Linux】驱动
驱动 驱动程序过程 系统调用 用户空间 内核空间 添加驱动和调用驱动 驱动程序是如何调用设备硬件 驱动 在计算机领域,驱动(Driver)是一种软件,它充当硬件设备与操作系统之间的桥梁,允许它们进行通信和协同工作。驱动程…...
Java研学-HTML
HTML 1 介绍 HTML(Hypertext Markup Language) 超文本标记语言。静态网页,用于在浏览器上显示数据 超文本: 指页面内可以包含图片、链接,甚至音乐、程序等非文字元素。 标记语言: 使用 < > 括起来的语言 超文本标记语言的结构, 包括“头”部分&am…...

SpringBoot之响应的详细解析
2. 响应 前面我们学习过HTTL协议的交互方式:请求响应模式(有请求就有响应) 那么Controller程序呢,除了接收请求外,还可以进行响应。 2.1 ResponseBody 在我们前面所编写的controller方法中,都已经设置了…...

redis:四、双写一致性的原理和解决方案(延时双删、分布式锁、异步通知MQ/canal)、面试回答模板
双写一致性 场景导入 如果现在有个数据要更新,是先删除缓存,还是先操作数据库呢?当多个线程同时进行访问数据的操作,又是什么情况呢? 以先删除缓存,再操作数据库为例 多个线程运行的正常的流程应该如下…...

智能优化算法应用:基于动物迁徙算法3D无线传感器网络(WSN)覆盖优化 - 附代码
智能优化算法应用:基于动物迁徙算法3D无线传感器网络(WSN)覆盖优化 - 附代码 文章目录 智能优化算法应用:基于动物迁徙算法3D无线传感器网络(WSN)覆盖优化 - 附代码1.无线传感网络节点模型2.覆盖数学模型及分析3.动物迁徙算法4.实验参数设定5.算法结果6.…...
illuminate/database 使用 五
之前文章: illuminate/database 使用 一-CSDN博客 illuminate/database 使用 二-CSDN博客 illuminate/database 使用 三-CSDN博客 illuminate/database 使用 四-CSDN博客 一、原生查询 1.1 原理 根据之前内容调用执行的静态类为Illuminate\Database\Capsule\M…...
武汉灰京文化:益智游戏的教育与娱乐完美结合
随着游戏技术的不断发展,益智类游戏正经历着一场革命性的变革,逐渐融合了教育和娱乐的元素。创新的设计和互动方式使得许多益智游戏成为了知识传递和技能训练的有效工具,同时保持了娱乐体验的趣味性。这种教育与娱乐的完美结合不仅使益智游戏…...
arcgis api for js 中的query实现数据查询
相当于服务地址中的query查询 获取图层范围内的数据4.24 import Query from arcgis/core/rest/support/Query; import * as QueryTask from "arcgis/core/rest/query";//获取图层范围内的数据4.24 _returnFeatureFromWhere(url, where, geo) {const self thisretu…...

AcWing 1250. 格子游戏(并查集)
题目链接 活动 - AcWing本课程系统讲解常用算法与数据结构的应用方式与技巧。https://www.acwing.com/problem/content/1252/ 题解 当两个点已经是在同一个连通块中,再连一条边,就围成一个封闭的圈。一般用x * n y的形式将(x, y࿰…...

CSS对文本的简单修饰
CSS格式: 格式一:在head中的style标签范围内。 < style> 在style内的只写名字不写 : < > 选择器 { 属性的名称 : 样式; 属性的名称:样式; } < style> style中的注释用/* *…...

C++17中if和switch语句的新特性
1.从C17开始,if语句允许在条件表达式里添加一条初始化语句。当仅在if语句范围内需要变量时,使用这种形式的if语句。在if语句的条件表达式里定义的变量将在整个if语句中有效,包括else部分。 std::mutex mx; bool shared_flag true; // guard…...

极坐标下的牛拉法潮流计算57节点MATLAB程序
微❤关注“电气仔推送”获得资料(专享优惠) 潮流计算: 潮流计算是根据给定的电网结构、参数和发电机、负荷等元件的运行条件,确定电力系统各部分稳态运行状态参数的计算。通常给定的运行条件有系统中各电源和负荷点的功率、枢纽…...

软件设计师——计算机网络(三)
📑前言 本文主要是【计算机网络】——软件设计师——计算机网络的文章,如果有什么需要改进的地方还请大佬指出⛺️ 🎬作者简介:大家好,我是听风与他🥇 ☁️博客首页:CSDN主页听风与他 …...
【ArkTS】循环控制与List的使用
ArkTS如何进行循环渲染 现有数据如下 class Item{name:stringimage:Resourceprice:numberdicount:numberconstructor(name:string,image:Resource,price:number,dicount?:number) {this.name namethis.image imagethis.price pricethis.dicount dicount} }private items…...
条款3:尽量使用const
文章目录 const指针和函数声明const修饰指针const修饰函数const修饰容器const应用在函数中 const限定成员函数避免const重载的代码重复总结 const指针和函数声明 const修饰指针 char greeting[] "Hello"; char* p greeting; // non-const 指针,// non-const 数据…...

Chromadb词向量数据库总结
简介 Chroma 词向量数据库是一个用于自然语言处理(NLP)和机器学习的工具,它主要用于词嵌入(word embeddings)。词向量是将单词转换为向量表示的技术,可以捕获单词之间的语义和语法关系,使得计算…...

Gin之GORM 操作数据库(MySQL)
GORM 简单介绍 GORM 是 Golang 的一个 orm 框架。简单说,ORM 就是通过实例对象的语法,完成关系型数据库的操作的技术,是"对象-关系映射"(Object/Relational Mapping) 的缩写。使用 ORM框架可以让我们更方便…...

二十七、读写文件
二十七、读写文件 27.1 文件类QFile #include <QCoreApplication>#include<QFile> #include<QDebug>int main(int argc, char *argv[]) {QCoreApplication a(argc, argv);QFile file("D:/main.txt");if(!file.open(QIODevice::WriteOnly | QIODe…...

flutter 代码混淆
Flutter 应用混淆: Flutter 应用的混淆非常简单,只需要在构建 release 版应用时结合使用 --obfuscate 和 --split-debug-info 这两个参数即可。 –obfuscate --split-debug-info 用来指定输出调试文件的位置,该命令会生成一个符号映射表。目前…...
05 Vue中常用的指令
概述 All Vue-based directives start with a v-* prefix as a Vue-specific attribute. 所有基于 Vue 的指令都以 v-* 前缀作为 Vue 特有的属性。 v-text The v-text directive has the same reactivity as with interpolation. Interpolation with {{ }} is more perform…...
Spring Boot 实现流式响应(兼容 2.7.x)
在实际开发中,我们可能会遇到一些流式数据处理的场景,比如接收来自上游接口的 Server-Sent Events(SSE) 或 流式 JSON 内容,并将其原样中转给前端页面或客户端。这种情况下,传统的 RestTemplate 缓存机制会…...
django filter 统计数量 按属性去重
在Django中,如果你想要根据某个属性对查询集进行去重并统计数量,你可以使用values()方法配合annotate()方法来实现。这里有两种常见的方法来完成这个需求: 方法1:使用annotate()和Count 假设你有一个模型Item,并且你想…...
OkHttp 中实现断点续传 demo
在 OkHttp 中实现断点续传主要通过以下步骤完成,核心是利用 HTTP 协议的 Range 请求头指定下载范围: 实现原理 Range 请求头:向服务器请求文件的特定字节范围(如 Range: bytes1024-) 本地文件记录:保存已…...
sqlserver 根据指定字符 解析拼接字符串
DECLARE LotNo NVARCHAR(50)A,B,C DECLARE xml XML ( SELECT <x> REPLACE(LotNo, ,, </x><x>) </x> ) DECLARE ErrorCode NVARCHAR(50) -- 提取 XML 中的值 SELECT value x.value(., VARCHAR(MAX))…...
【C语言练习】080. 使用C语言实现简单的数据库操作
080. 使用C语言实现简单的数据库操作 080. 使用C语言实现简单的数据库操作使用原生APIODBC接口第三方库ORM框架文件模拟1. 安装SQLite2. 示例代码:使用SQLite创建数据库、表和插入数据3. 编译和运行4. 示例运行输出:5. 注意事项6. 总结080. 使用C语言实现简单的数据库操作 在…...
拉力测试cuda pytorch 把 4070显卡拉满
import torch import timedef stress_test_gpu(matrix_size16384, duration300):"""对GPU进行压力测试,通过持续的矩阵乘法来最大化GPU利用率参数:matrix_size: 矩阵维度大小,增大可提高计算复杂度duration: 测试持续时间(秒&…...
Element Plus 表单(el-form)中关于正整数输入的校验规则
目录 1 单个正整数输入1.1 模板1.2 校验规则 2 两个正整数输入(联动)2.1 模板2.2 校验规则2.3 CSS 1 单个正整数输入 1.1 模板 <el-formref"formRef":model"formData":rules"formRules"label-width"150px"…...
今日学习:Spring线程池|并发修改异常|链路丢失|登录续期|VIP过期策略|数值类缓存
文章目录 优雅版线程池ThreadPoolTaskExecutor和ThreadPoolTaskExecutor的装饰器并发修改异常并发修改异常简介实现机制设计原因及意义 使用线程池造成的链路丢失问题线程池导致的链路丢失问题发生原因 常见解决方法更好的解决方法设计精妙之处 登录续期登录续期常见实现方式特…...
2023赣州旅游投资集团
单选题 1.“不登高山,不知天之高也;不临深溪,不知地之厚也。”这句话说明_____。 A、人的意识具有创造性 B、人的认识是独立于实践之外的 C、实践在认识过程中具有决定作用 D、人的一切知识都是从直接经验中获得的 参考答案: C 本题解…...

HarmonyOS运动开发:如何用mpchart绘制运动配速图表
##鸿蒙核心技术##运动开发##Sensor Service Kit(传感器服务)# 前言 在运动类应用中,运动数据的可视化是提升用户体验的重要环节。通过直观的图表展示运动过程中的关键数据,如配速、距离、卡路里消耗等,用户可以更清晰…...