Linux内核驱动开发核心问题全解


📖 推荐阅读:《Yocto项目实战教程:高效定制嵌入式Linux系统》
🎥 更多学习视频请关注 B 站:嵌入式Jerry


Linux内核驱动开发核心问题全解

本文系统梳理了 Linux 驱动开发、内核同步、中断处理、内存管理、进程通信、系统启动等典型场景中的高频核心问题,涵盖中断上下文与进程上下文、下半部机制、锁与内存分配、网络通信、init 启动流程等内容,配以关键代码片段,适合工程实践参考。


在这里插入图片描述

一、中断与驱动开发

1. 中断上下文与进程上下文的区别

  • 中断上下文:由CPU响应外设信号进入,没有进程调度能力,不能休眠(不能 sleep / schedule),用于处理紧急、耗时短的操作。
  • 进程上下文:运行在内核线程或用户进程中,可以主动调度和休眠,适合处理复杂或耗时任务。

2. 中断下半部机制(tasklet、softirq、workqueue)

  • tasklet/softirq:在中断返回后、软中断上下文中调度执行,不能睡眠。
  • workqueue:由内核线程执行,可休眠,适合需要阻塞或耗时的操作。

示例:tasklet 调度下半部

void tasklet_handler(unsigned long data) {// 执行实际下半部工作
}
DECLARE_TASKLET(my_tasklet, tasklet_handler, 0);
irqreturn_t my_irq_handler(int irq, void *dev_id) {tasklet_schedule(&my_tasklet); // 调度下半部return IRQ_HANDLED;
}

3. request_irq 注册格式及参数

int request_irq(unsigned int irq, irq_handler_t handler,unsigned long flags, const char *name, void *dev);
  • irq:中断号
  • handler:中断服务函数,签名为 irqreturn_t func(int, void*)
  • flags:如 IRQF_SHARED
  • name:用于 /proc/interrupts
  • dev:设备标识

4. 中断处理流程

  • 硬件触发中断
  • 汇编入口 (arch/x86/entry/entry_64.S 等)
  • 通用分发 (kernel/irq/handle.c)
  • 调用驱动注册的 handler
  • 调度下半部(如 tasklet、napi、workqueue)

二、同步、锁与多核并发

5. 多核系统缓存一致性与内存屏障

  • 多核缓存一致性:依赖硬件协议(如 MESI),软件无需手动刷新缓存。
  • 内存屏障(如 mb()):用于保证CPU/编译器的内存操作顺序,防止乱序。

代码示例:内存屏障

#include <asm/barrier.h>
void sync_example(void) {a = 1;mb(); // 确保a的写操作对其他核可见b = 2;
}

6. 进程间共享内存同步与死锁防范

  • 共享内存不是线程安全,需加锁同步
  • 推荐 pthread_mutex_t,需 PTHREAD_PROCESS_SHARED 属性
  • 死锁风险:如进程异常退出未解锁,可用健壮互斥锁 PTHREAD_MUTEX_ROBUST

代码片段:进程间共享互斥锁

pthread_mutexattr_t attr;
pthread_mutexattr_init(&attr);
pthread_mutexattr_setpshared(&attr, PTHREAD_PROCESS_SHARED);
pthread_mutex_init(&shared->lock, &attr);
// 使用时加锁/解锁
pthread_mutex_lock(&shared->lock); // 临界区
pthread_mutex_unlock(&shared->lock);

7. 自旋锁与互斥锁适用场景

  • 自旋锁:适合内核临界区/中断上下文,不能睡眠
  • 互斥锁:适合进程/线程上下文,允许休眠

三、内存管理机制

8. kmalloc 与 vmalloc 的区别

对比项kmallocvmalloc
连续性物理+虚拟连续虚拟连续物理不连续
分配效率
适用场景小块/需DMA大块/无需DMA
分配器slab+buddy伙伴分配多个物理页

9. slab 与 buddy 分配器

  • slab:管理常用小对象缓存(如 kmalloc-32、kmalloc-64)
  • buddy:以页为单位(2^n),用于大块内存,支持拆分/合并

10. kmalloc(32) 和 kmalloc(4096) 行为差异

  • kmalloc(32):分配小对象,来自 slab 缓存池(如 kmalloc-32),实际为一页切片
  • kmalloc(4096):直接分配整页,调用 buddy 分配器

伪代码逻辑:

kmalloc(size)-> slab 分配(若 size 小于页)-> buddy 分配(若 size 大于等于页)

11. malloc、页表与物理内存关系

  • 应用层 malloc 在用户空间分配虚拟地址,由 glibc 切块
  • 真正物理内存分配由内核通过页表管理,内存页来自 buddy 分配器

四、进程通信与网络

12. 进程间通信方式

  • 管道(pipe, fifo)
  • 消息队列
  • 共享内存(shmget/shmat)
  • 信号量
  • 本地/网络 socket
  • 信号、文件锁等

13. TCP 与 UDP 的区别

对比项TCP(SOCK_STREAM)UDP(SOCK_DGRAM)
是否连接有连接无连接
可靠性可靠,顺序保证不可靠,可能丢包
速度
用途Web、SSH、文件传输DNS、直播、语音

14. socket() 返回值

  • 成功:返回文件描述符(>=0),不论 TCP 还是 UDP
  • 失败:返回 -1,需查 errno
  • 0/1/2 通常是标准输入/输出/错误

代码示例

int fd = socket(AF_INET, SOCK_STREAM, 0);
if (fd < 0) perror("socket error");

五、系统启动与 init 流程

15. init 进程的作用及根文件系统挂载

  • 内核挂载根文件系统(/)并启动 init 进程(如 /sbin/init, systemd, busybox init)
  • init 进程负责挂载 /proc、/sys、/dev 等虚拟文件系统,启动系统服务

真实内核片段:init 进程启动

// init/main.c (Linux 内核)
static const char * const init_paths[] = {"/sbin/init", "/etc/init", "/bin/init", "/bin/sh", NULL };
for (p = init_paths; *p; p++)if (!run_init_process(*p)) return 0;

结语

本文梳理了 Linux 驱动开发、同步机制、内存管理、通信协议、系统启动等多个核心环节及其典型代码实现,为深入理解与实践 Linux 内核提供参考。建议结合源码实际查阅、动手实验和知识串联,形成体系化认知。


📖 推荐阅读:《Yocto项目实战教程:高效定制嵌入式Linux系统》
🎥 更多学习视频请关注 B 站:嵌入式Jerry


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/pingmian/91410.shtml
繁体地址,请注明出处:http://hk.pswp.cn/pingmian/91410.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【C++篇】C++11入门:踏入C++新世界的大门

文章目录C11简介列表初始化1. {}初始化2. initializer_list容器initializer_list的使用场景声明1. auto2. decltype3. nullptrSTL中的变化1. 新容器array容器forward_list容器unordered_map和unordered_set容器2. 新接口C11简介 C98/03&#xff1a;在2003年C标准委员会曾经提交…

Java 日期时间处理:分类、用途与性能分析

Java提供了多种日期时间处理API&#xff0c;随着版本演进不断改进。以下是主要日期时间类的分类、用途和性能分析&#xff1a;一、Java日期时间API分类1. 传统日期时间API (Java 1.0/1.1)java.util.Date - 表示特定的瞬间&#xff0c;精确到毫秒java.util.Calendar - 抽象类&am…

[Linux]学习笔记系列 --GCC

文章目录属性__cleanup__attribute_malloc__ 用于标记函数返回一个新分配的内存块__attribute_alloc_size__ 用于指定分配的内存大小__attribute__((const)) 标记为纯函数(pure function)__attribute__((__externally_visible__)) 使其在编译器优化过程中保持对外部模块的可见性…

【龙泽科技】汽车维护与底盘拆装检修仿真教学软件【风光580】

产品简介汽车维护与底盘拆装检修仿真教学软件是依托《全国职业院校技能大赛》“汽车维修”赛项中“汽车维护与底盘拆装检修模块”竞赛模块&#xff0c;自主开发的一款仿真教学软件。软件采用仿真仿真技术模拟实际汽车维修工的岗位技能操作流程&#xff0c;操作内容主要包括&…

Spring之【循环引用】

目录前置知识SingletonBeanRegistryDefaultSingletonBeanRegistrySpring中处理循环引用的流程分析定义两个具有循环引用特点的Bean执行A的实例化执行A的属性填充(执行过程中发现A依赖B&#xff0c;就去执行B的实例化逻辑)执行B的实例化执行B的属性填充执行B的初始化执行A的属性…

LRU缓存淘汰算法的详细介绍与具体实现

LRU&#xff08;Least Recently Used&#xff0c;最近最少使用&#xff09;是一种基于时间局部性原理的缓存淘汰策略。其核心思想是&#xff1a;最近被访问的数据在未来更可能被再次使用&#xff0c;而最久未被访问的数据应优先被淘汰&#xff0c;从而在有限的缓存空间内保留高…

JS-第十九天-事件(一)

一、事件基础概念1.1 事件三要素事件源&#xff1a;触发事件的元素事件类型&#xff1a;事件的种类&#xff08;如click、mouseover等&#xff09;事件处理程序&#xff1a;响应事件的函数1.2 事件流机制事件传播分为三个阶段&#xff1a;捕获阶段&#xff1a;事件从顶层开始&a…

Matplotlib(三)- 图表辅助元素

文章目录一、图表辅助元素简介二、坐标轴的标签、刻度范围和刻度标签1. 坐标轴标签1.1 x轴标签1.2 y轴标签1.3 示例&#xff1a;绘制天气气温折线图2. 刻度范围和刻度标签2.1 刻度范围2.1.1 x轴刻度范围2.1.2 y轴刻度范围2.2 刻度标签2.2.1 x轴刻度标签2.2.2 y轴刻度标签2.3 示…

【Linux基础知识系列】第七十八篇 - 初识Nmap:网络扫描工具

在网络管理和安全领域&#xff0c;网络扫描是一个不可或缺的工具。它可以帮助网络管理员了解网络中的设备、服务以及潜在的安全漏洞。Nmap&#xff08;Network Mapper&#xff09;是一个功能强大的开源网络扫描工具&#xff0c;它能够快速发现网络中的主机、端口和服务&#xf…

EasyGBS的两种录像回看

EasyGBS 支持两种录像回看&#xff0c;即“平台端”的录像回看和“设备端”的录像回看。本期我们来介绍两者的区别和使用方法。一、平台端录像1、什么是平台端录像平台端录像是指由 EasyGBS 平台直接录制并存储。2、配置平台端录像进入平台&#xff0c;依次点击【录像回放】→【…

大模型学习思路推荐!

为进一步贯彻落实中共中央印发《关于深化人才发展体制机制改革的意见》和国务院印发《关于“十四五”数字经济发展规划》等有关工作的部署要求&#xff0c;深入实施人才强国战略和创新驱动发展战略&#xff0c;加强全国数字化人才队伍建设&#xff0c;持续推进人工智能从业人员…

数据库连接池性能优化实战

背景我们公司正在处于某个项目的维护阶段&#xff0c;领导对资源告警比较重视&#xff0c;服务器资源告警的就不说了&#xff0c;运维同学每隔一小时都会检测线上环境的应用服务信息&#xff0c;例如&#xff1a;网关日志响应时间告警/nginx日志接口响应时间告警/日志关键字异常…

Excel常用函数大全,非常实用

一、数学与统计函数1. SUM作用&#xff1a;求和SUM(number1, [number2], ...)SUM(A1:A10) ➔ 计算A1到A10单元格的总和注意&#xff1a;自动忽略文本和空单元格2. AVERAGE作用&#xff1a;计算平均值AVERAGE(number1, [number2], ...)AVERAGE(B2:B20) ➔ 计算B列20个数据的平均…

性能优化(一):时间分片(Time Slicing):让你的应用在高负载下“永不卡顿”的秘密

性能优化(一)&#xff1a;时间分片&#xff08;Time Slicing&#xff09;&#xff1a;让你的应用在高负载下“永不卡顿”的秘密 引子&#xff1a;那张让你浏览器崩溃的“无限列表” 想象一个场景&#xff1a;你需要渲染一个包含一万个项目的列表。在我们的“看不见”的应用中&a…

《C++》STL--list容器详解

在 C 标准模板库(STL)中&#xff0c;list 是一个非常重要的序列容器&#xff0c;它实现了双向链表的数据结构。与 vector 和 deque 不同&#xff0c;list 提供了高效的插入和删除操作&#xff0c;特别是在任意位置。本文将深入探讨 list 容器的特性、使用方法以及常见操作。 文…

Day 28:类的定义和方法

DAY 28 类的定义和方法 知识点学习 1. 类的定义 在Python中&#xff0c;类是创建对象的模板。使用class关键字来定义一个类。类名通常采用首字母大写的命名方式&#xff08;PascalCase&#xff09;。 # 最简单的类定义 class MyClass:pass # 使用pass占位符类的定义就像是…

OSPF综合实验报告册

一、实验拓扑二、实验要求1、R4为ISP&#xff0c;其上只配置IP地址&#xff1b;R4与其他所直连设备间均使用公有IP&#xff1b; 2、R3-R5、R6、R7为MGRE环境&#xff0c;R3为中心站点&#xff1b; 3、整个OSPF环境IP基于172.16.0.0/16划分&#xff1b;除了R12有两个环回&#x…

网络层6——内部网关协议RIP、OSPF(重点)

目录 一、基本概念 1、理想的路由算法应具备的特点 2、分层次的路由选择协议 二、内部网关协议RIP 1、特点 2、路由交换信息 3、距离向量算法 4、坏消息传送慢问题 5、RIP报文格式 三、内部网关协议OSPF 1、特点 2、其他特点 3、自治系统区域划分 4、OSPF的5中分…

同品牌的系列广告要如何保证宣传的连贯性?

对于品牌的系列广告而言&#xff0c;内容的连贯性十分重要。如果系列广告之间缺乏内在联系&#xff0c;不仅会削弱品牌形象的统一性&#xff0c;还可能导致用户的认知混乱。保证宣传内容的连贯性不是让每则广告完全相同&#xff0c;而是在变化中保持核心要素的一致性。我们该如…

深度学习:激活函数Activaton Function

一、为什么需要激活函数&#xff1f;神经网络本质上是多个线性变换&#xff08;矩阵乘法&#xff09;叠加。如果没有激活函数&#xff0c;即使叠加多层&#xff0c;整体仍等价于一个线性函数&#xff1a;这样的网络无法学习和拟合现实世界中复杂的非线性关系。激活函数的作用&a…