news 2026/3/21 10:40:03

[Linux外设驱动详解]第一部分:基础篇 —— 驱动开发入门

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
[Linux外设驱动详解]第一部分:基础篇 —— 驱动开发入门

Linux 设备驱动开发入门指南

硬件平台: 瑞芯微 RK3588
源码路径:

  • U-Boot:./u-boot
  • Linux 内核:./kernel

目录

  1. 什么是 Linux 设备驱动
  2. Linux 内核模块机制
  3. 第一个字符设备驱动
  4. 源码分析实战
  5. 设备节点创建

1. 什么是 Linux 设备驱动

1.1 驱动程序的本质

设备驱动程序是连接 Linux 内核与硬件设备的桥梁,它为上层应用程序提供统一的访问接口,屏蔽底层硬件的具体实现细节。

┌─────────────────────────────────────────────────────────────┐ │ 用户空间 (User Space) │ │ ┌──────────────┐ ┌──────────────┐ ┌──────────────┐ │ │ │ 应用程序 │ │ 应用程序 │ │ Shell │ │ │ └──────┬───────┘ └──────┬───────┘ └──────┬───────┘ │ │ │ │ │ │ │ └─────────────────┼─────────────────┘ │ │ │ │ ├───────────────────────────┼───────────────────────────────────┤ │ 系统调用接口 │ (read, write, ioctl, etc.) │ ├───────────────────────────┼───────────────────────────────────┤ │ │ │ │ ┌────────────────────────▼─────────────────────────────┐ │ │ │ 内核空间 (Kernel Space) │ │ │ │ ┌────────────────────────────────────────────────┐ │ │ │ │ │ VFS (虚拟文件系统) │ │ │ │ │ └────────────────────────────────────────────────┘ │ │ │ │ ┌────────────────────────────────────────────────┐ │ │ │ │ │ 设备驱动程序 (Driver) │ │ │ │ │ │ ┌────────┐ ┌────────┐ ┌────────┐ │ │ │ │ │ │ │ 字符设备│ │ 块设备 │ │ 网络设备│ │ │ │ │ │ │ └────────┘ └────────┘ └────────┘ │ │ │ │ │ └────────────────────────────────────────────────┘ │ │ │ │ ┌────────────────────────────────────────────────┐ │ │ │ │ │ 硬件抽象层 │ │ │ │ │ └────────────────────────────────────────────────┘ │ │ │ └──────────────────────────────────────────────────────┘ │ │ │ │ ├───────────────────────────┼───────────────────────────────────┤ │ 硬件设备 │ │ ┌──────────┐ ┌──────────┐ ┌──────────┐ │ │ │ GPIO │ │ I2C │ │ SPI │ ... │ │ └──────────┘ └──────────┘ └──────────┘ │ └─────────────────────────────────────────────────────────────┘

关键点:

  • 用户空间程序通过系统调用访问设备
  • 驱动程序运行在内核空间,拥有最高权限
  • 设备文件(如/dev/gpio)是用户访问驱动的入口

1.2 设备驱动分类

Linux 系统将设备驱动分为三大类:

驱动类型特点访问方式典型设备源码位置
字符设备按字节流顺序访问,不支持随机访问read(),write(),ioctl()串口、GPIO、LED、键盘kernel/drivers/char/
块设备可随机访问,以数据块为单位文件系统接口硬盘、Flash、SD卡kernel/drivers/block/
网络设备处理网络数据包,不对应设备节点Socket 接口以太网、WiFi、蓝牙kernel/drivers/net/

字符设备 vs 块设备的核心区别:

// 字符设备:顺序访问,像水流一样/dev/tty → 只能顺序读写/dev/gpio → 按字节操作// 块设备:可随机访问任意位置/dev/sda → 可以跳到任意扇区/dev/mmcblk0 → 支持文件系统

2. Linux 内核模块机制

2.1 什么是内核模块

内核模块 (Kernel Module)是可以动态加载到内核中的代码,也称为 LKM (Loadable Kernel Module)。

优势:

  • 不需要重新编译整个内核
  • 按需加载,节省内存
  • 便于开发和调试

2.2 模块的生命周期

┌──────────────────────────────────────────────────────────────┐ │ 内核模块生命周期 │ ├──────────────────────────────────────────────────────────────┤ │ │ │ insmod/modprobe │ │ │ │ │ ▼ │ │ ┌─────────┐ module_init() ┌─────────────┐ │ │ │ .ko 文件│ ─────────────────► │ 运行中 │ │ │ │(磁盘上) │ │ (内核内存) │ │ │ └─────────┘ └──────┬──────┘ │ │ │ │ │ │ 提供服务 │ │ ▼ │ │ ┌─────────┐ │ │ │ 设备操作 │ │ │ └─────────┘ │ │ │ │ │ rmmod │ │ │ │ │ │ │ ▼ │ │ │ module_exit() ◄───────────────────────┘ │ │ │ │ │ ▼ │ │ ┌─────────┐ │ │ │ 卸载完成 │ │ │ └─────────┘ │ │ │ └──────────────────────────────────────────────────────────────┘

2.3 模块编程基础

最简单的内核模块
#include<linux/init.h>// 宏定义#include<linux/module.h>// 模块核心头文件#include<linux/kernel.h>// 内核函数// 模块初始化函数 - 加载时执行staticint__inithello_init(void){printk(KERN_INFO"Hello: 模块已加载\n");return0;// 返回 0 表示成功}// 模块退出函数 - 卸载时执行staticvoid__exithello_exit(void){printk(KERN_INFO"Hello: 模块已卸载\n");}// 注册初始化和退出函数module_init(hello_init);module_exit(hello_exit);// 模块元信息MODULE_LICENSE("GPL");// 许可证声明 (必须)MODULE_AUTHOR("Your Name");// 作者MODULE_DESCRIPTION("简单模块");// 描述MODULE_VERSION("1.0");// 版本

宏说明:

  • __init: 函数在初始化后可被丢弃以节省内存
  • __exit: 函数仅在模块卸载时使用
  • module_init(): 指定模块加载入口
  • module_exit(): 指定模块卸载出口
模块加载与卸载命令
# 加载模块insmod hello.ko# 简单加载modprobe hello.ko# 智能加载(处理依赖)# 查看模块lsmod|grephello# 列出已加载模块modinfo hello.ko# 查看模块信息# 卸载模块rmmod hello# 移除模块# 查看内核日志dmesg|tail# 查看最近的内核消息dmesg|grepHello# 查看特定消息

2.4 printk 与内核日志

printk 是内核空间的 printf,用于输出调试信息。

日志级别
#defineKERN_EMERG"0"// 紧急情况:系统崩溃#defineKERN_ALERT"1"// 需要立即处理#defineKERN_CRIT"2"// 严重情况#defineKERN_ERR"3"// 错误#defineKERN_WARNING"4"// 警告#defineKERN_NOTICE"5"// 注意#defineKERN_INFO"6"// 信息#defineKERN_DEBUG"7"// 调试

使用示例:

printk(KERN_INFO"设备号: %d\n",major);// 信息级别printk(KERN_ERR"设备初始化失败: %d\n",ret);// 错误级别printk(KERN_DEBUG"调试信息: count=%u\n",count);// 调试级别// 简写形式(需要包含 <linux/printk.h>)pr_info("设备号: %d\n",major);pr_err("初始化失败: %d\n",ret);pr_debug("count=%u\n",count);
查看内核日志
# 实时查看内核日志dmesg-w# 持续监控dmesg-w|grepmy_driver# 过滤特定驱动# 控制台日志级别cat/proc/sys/kernel/printk# 查看当前级别echo8>/proc/sys/kernel/printk# 设置为调试级别

3. 第一个字符设备驱动

3.1 字符设备架构概览

┌───────────────────────────────────────────────────────────────┐ │ 字符设备驱动架构 │ ├───────────────────────────────────────────────────────────────┤ │ │ │ 用户空间 │ │ ┌─────────────┐ open("/dev/mydev") ┌──────────────┐ │ │ │ 应用程序 │ ─────────────────────► │ VFS 层 │ │ │ │ │ │ │ │ │ │ fd = open()│ read(fd, buf, len) │ 文件系统 │ │ │ │ read(fd) │ ─────────────────────► │ │ │ │ │ write(fd) │ └─
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/3/15 18:23:59

AUTOSAR开发新革命:AI如何加速汽车软件架构设计

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 开发一个基于AUTOSAR标准的AI辅助工具&#xff0c;能够自动解析ARXML文件并生成优化的BSW模块配置代码。要求支持以下功能&#xff1a;1. 智能识别ECU硬件资源需求 2. 自动生成符合…

作者头像 李华
网站建设 2026/3/15 18:23:58

League Akari终极教程:3步配置法显著提升游戏效率

League Akari终极教程&#xff1a;3步配置法显著提升游戏效率 【免费下载链接】LeagueAkari ✨兴趣使然的&#xff0c;功能全面的英雄联盟工具集。支持战绩查询、自动秒选等功能。基于 LCU API。 项目地址: https://gitcode.com/gh_mirrors/le/LeagueAkari 英雄联盟辅助…

作者头像 李华
网站建设 2026/3/15 18:09:54

AI助力STM32开发:如何用快马平台自动生成CubeProgrammer脚本

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 请生成一个STM32CubeProgrammer的Python脚本&#xff0c;实现以下功能&#xff1a;1.自动连接STM32开发板&#xff1b;2.擦除指定扇区&#xff1b;3.烧录提供的hex文件&#xff1b…

作者头像 李华
网站建设 2026/3/16 2:45:15

核心要点:掌握半加器的两个关键输出信号

从0和1开始&#xff1a;拆解半加器的“和”与“进位”&#xff0c;看懂数字电路的第一块积木你有没有想过&#xff0c;计算机是怎么做加法的&#xff1f;不是掏出计算器&#xff0c;也不是列竖式——它靠的是成千上万个微小逻辑门的协同工作。而这一切的起点&#xff0c;正是一…

作者头像 李华
网站建设 2026/3/16 2:45:17

GLM-4.6V-Flash-WEB在考古现场图像记录中的分类归档作用

GLM-4.6V-Flash-WEB在考古现场图像记录中的分类归档作用 在一次河南某新石器时代遗址的发掘现场&#xff0c;考古队每天拍摄超过300张照片——从探方剖面到陶片分布&#xff0c;从碳化种子到工具痕迹。这些图像承载着关键信息&#xff0c;但整理它们却成了最耗时的工作&#xf…

作者头像 李华
网站建设 2026/3/15 14:40:18

AWVS扫描效率提升300%的10个技巧

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 创建一个AWVS效率优化工具包&#xff0c;包含&#xff1a;1. 最优扫描配置生成器 2. 目标网站预分析脚本 3. 扫描结果快速过滤器 4. 并行扫描管理器。工具应能自动分析目标网站结构…

作者头像 李华