news 2026/7/5 22:58:50

20252820_中断和异常管理

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
20252820_中断和异常管理

20252820_中断和异常管理

实验内容:Tasklet / Workqueue / 信号捕获
实验平台:openEuler
内核版本:5.10.0+


1. 实验目的

  1. 理解 Linux 内核中下半部机制(Bottom Half)的基本思想与使用方式。
  2. 掌握tasklet的创建、调度与释放流程。
  3. 掌握workqueue(工作队列)的创建、延迟调度、周期执行与退出清理流程。
  4. 理解 Linux 用户态信号机制,能捕获并处理Ctrl+C / Ctrl+Z / Ctrl+\对应信号。

2. 实验环境与准备

2.1 软硬件环境

  • 操作系统:openEuler
  • 内核版本:5.10.0+
  • 编译工具:gcc / make
  • 内核构建目录:/lib/modules/$(uname -r)/build -> /usr/src/kernels/5.10.0+

2.2 实验目录结构

chapter4_interrupt_exception/ ├── task1_tasklet/ │ ├── tasklet_intertupt.c │ └── Makefile ├── task2_workqueue/ │ ├── workqueue_test.c │ └── Makefile └── task3_signal/ ├── catch_signal.c └── Makefile

2.3 建立目录

mkdir-p chapter4_interrupt_exception/{task1_tasklet,task2_workqueue,task3_signal}

3. 任务一:Tasklet 打印 HelloWorld

3.1 实验内容说明

本任务通过 tasklet 实现一个简单的下半部处理:模块加载时初始化 tasklet 并调度执行,在 tasklet 处理函数中打印Hello World!,模块卸载时使用tasklet_kill()确保 tasklet 安全退出。

3.2 核心代码要点(tasklet_intertupt.c)

文件位置:chapter4_interrupt_exception/task1_tasklet/tasklet_intertupt.c

  • tasklet_init(&tasklet, handler, data):初始化 tasklet
  • tasklet_schedule(&tasklet):调度 tasklet 执行
  • tasklet_kill(&tasklet):退出时确保 tasklet 不再运行

3.3 Makefile 配置(关键点:KERNELDIR)

文件位置:chapter4_interrupt_exception/task1_tasklet/Makefile

  • 外部模块编译:make -C $(KERNELDIR) M=$(PWD) modules
  • openEuler 推荐:KERNELDIR=/lib/modules/$(shell uname -r)/build

3.4 实验步骤

1)进入目录并编译:

cdchapter4_interrupt_exception/task1_taskletmakeclean&&make

2)加载模块并查看输出:

sudoinsmod tasklet_intertupt.kodmesg|tail-n30

3)卸载模块并查看输出:

sudormmod tasklet_intertuptdmesg|tail-n30

3.5 运行结果与截图

  • 截图1:make编译成功(生成.ko
    【截图:任务1-编译成功】


  • 截图2:insmoddmesg中出现Start...Hello World...
    【截图:任务1-加载与打印】

3.6 结果分析与总结

  • tasklet 属于下半部机制之一,适合执行短小、非阻塞的延后处理。
  • 退出时必须tasklet_kill(),避免模块卸载后 tasklet 仍可能访问已释放代码导致异常。

4. 任务二:Workqueue 周期打印 HelloWorld

4.1 实验内容说明

本任务使用工作队列实现“周期打印”效果:创建专用 workqueue,初始化 delayed_work,首次延迟入队;在 work 回调中打印信息并再次queue_delayed_work()自己,从而形成周期执行;卸载模块时取消 delayed_work 并销毁工作队列。

4.2 核心代码要点(workqueue_test.c)

文件位置:chapter4_interrupt_exception/task2_workqueue/workqueue_test.c

关键点:

  • create_singlethread_workqueue()创建队列
  • INIT_DELAYED_WORK(&mywork, work_handle)初始化延迟任务
  • queue_delayed_work(queue, &mywork, period*HZ)延迟调度
  • work_handle()中再次queue_delayed_work()实现周期循环
  • 退出时:cancel_delayed_work_sync()+destroy_workqueue()避免竞态与悬挂任务

4.3 Makefile 配置

文件位置:chapter4_interrupt_exception/task2_workqueue/Makefile

  • KERNELDIR ?= /lib/modules/$(shell uname -r)/build

4.4 实验步骤

1)进入目录并编译:

cdchapter4_interrupt_exception/task2_workqueuemakeclean&&make

2)加载模块(设置周期参数 5 秒):

sudoinsmod workqueue_test.koperiod=5

3)观察周期打印:

dmesg-w

4)卸载模块:

sudormmod workqueue_test

4.5 运行结果与截图

  • 截图1:make编译成功(生成.ko
    【截图:任务2-编译成功】


  • 截图2:insmoddmesg连续多次输出(体现周期性)
    【截图:任务2-周期打印】

4.6 结果分析与总结

  • workqueue 在进程上下文中执行,允许睡眠/阻塞,适合执行相对更复杂的任务。
  • 周期执行推荐在 work 回调中自我重新入队,而不是在module_init()sleep,避免insmod阻塞。
  • 退出必须同步取消 delayed_work,防止模块卸载后工作仍被调度。

5. 任务三:捕获 Ctrl+C / Ctrl+Z / Ctrl+\ 信号

5.1 实验内容说明

本任务在用户态编写程序捕获三类终端按键信号:

  • Ctrl + CSIGINT
  • Ctrl + ZSIGTSTP
  • Ctrl + \SIGQUIT

程序打印当前进程 PID,注册信号处理函数,在捕获到信号后输出对应提示并退出。

5.2 核心代码要点(catch_signal.c)

文件位置:chapter4_interrupt_exception/task3_signal/catch_signal.c

  • 使用signal(SIGxxx, handler)注册处理器
  • 使用pause()阻塞等待信号到来
  • 在 handler 中区分不同sig并输出提示

5.3 编译与运行步骤

1)进入目录并编译:

cdchapter4_interrupt_exception/task3_signalmakeclean&&make

2)运行程序:

./catch_signal

3)分别测试三种按键(每次测试重新运行一次程序):

  • Ctrl + C(SIGINT)
  • Ctrl + Z(SIGTSTP)
  • Ctrl + \(SIGQUIT)

5.4 运行结果与截图

  • 截图1:程序启动输出 PID
    【截图:任务3-启动输出PID】

  • 截图2:按下Ctrl+C捕获 SIGINT 输出,按下Ctrl+Z捕获 SIGTSTP 输出,按下Ctrl+\捕获 SIGQUIT 输出

5.5 结果分析与总结

  • 信号是 Linux 用户态重要的异步事件通知机制,终端按键会向前台进程组发送特定信号。
  • 通过注册 handler 可以实现自定义行为(打印、资源释放、优雅退出等)。
  • 本实验通过 handler 拦截默认行为,实现“捕获并提示”的效果。

6. 实验中遇到的问题与解决方案

6.1 问题:内核源码目录找不到导致编译失败

现象:make[1]: *** /root/raspberrypi-kernel: No such file or directory
原因:Makefile 中KERNELDIR指向了不存在路径。
解决:openEuler 使用:/lib/modules/$(uname -r)/build(本机链接到/usr/src/kernels/5.10.0+)。


7. 实验结论

完成 tasklet、workqueue 与用户态信号捕获三个任务,理解了 Linux 中断下半部与异步事件处理的基本机制,并掌握了模块编译加载、日志验证、退出清理等关键流程。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/7/4 2:57:45

25、文档管理与超文本应用全解析

文档管理与超文本应用全解析 在文档处理过程中,我们常常会遇到各种复杂的情况,比如条件文本的管理、文档更改的跟踪以及超文本的应用等。下面将详细介绍这些方面的相关知识和操作方法。 条件文本管理 条件文本在概念上相对容易理解,但在复杂文档中管理起来却颇具挑战。如…

作者头像 李华
网站建设 2026/7/1 20:58:27

Actix Web终极实战指南:从零构建高性能Rust微服务

Actix Web终极实战指南:从零构建高性能Rust微服务 【免费下载链接】actix-web Actix Web is a powerful, pragmatic, and extremely fast web framework for Rust. 项目地址: https://gitcode.com/gh_mirrors/ac/actix-web 你是否正在寻找一个既能提供极致性…

作者头像 李华
网站建设 2026/7/1 17:56:38

基于STM32的平衡车开发计划-1-PWM设置

我们使用的是STM32F103C8T6进行编程和实现。一.配置一下PWM输出1.PWM设置HAL_TIM_PWM_Start(&htim1,TIM_CHANNEL_1);启动PWM使用的是TIM1的ch1通道,也就是PA8端口,cubemx会自动帮我们配置好。预分频为72,自从重装值1000,采用向…

作者头像 李华
网站建设 2026/7/2 0:13:11

在STM32函数指针是什么,怎么使用还有典型应用场景。

函数指针听起来复杂,但其实你可以把它理解成一个“遥控器”。它本身不干活,但按一下(调用它),就能遥控执行另一个函数。在STM32开发里,这个“遥控”的特性,正好能解决硬件和应用之间灵活联动的问…

作者头像 李华
网站建设 2026/7/2 0:12:56

进销存最新推荐 | 哪个进销存软件性价比最好?- 象过河软件

在数字化管理工具选型中,进销存软件的 “性价比” 从来不是单纯看价格,而是 “功能实用性 投入成本” 的综合考量。市面上有的软件低价但功能残缺,有的功能齐全却年费高昂,还有的隐藏插件收费,让企业陷入 “两难”。2…

作者头像 李华
网站建设 2026/7/5 0:37:45

5个关键技巧:高效使用baseimage-docker构建企业级容器环境

5个关键技巧:高效使用baseimage-docker构建企业级容器环境 【免费下载链接】baseimage-docker A minimal Ubuntu base image modified for Docker-friendliness 项目地址: https://gitcode.com/gh_mirrors/ba/baseimage-docker 在容器化技术日益普及的今天&a…

作者头像 李华