嵌入式驱动学习第六周——内核函数调用(堆栈打印)

本文主要是介绍嵌入式驱动学习第六周——内核函数调用(堆栈打印),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

   在内核中,函数调用堆栈非常重要,因为它可以帮助开发人员理解代码是如何执行的,从而进行调试、性能优化或问题排查。堆栈可以显示当前执行的函数以及导致该函数调用的先前函数,从而形成一个函数调用链。本篇博客就介绍堆栈打印内核函数的调用。

   嵌入式驱动学习专栏将详细记录博主学习驱动的详细过程,未来预计四个月将高强度更新本专栏,喜欢的可以关注本博主并订阅本专栏,一起讨论一起学习。现在关注就是老粉啦!

目录

  • 前言
  • dump_stack函数
  • WARN_ON函数
  • BUG_ON函数
  • panic函数
  • 参考资料

dump_stack函数

   该函数作用是打印内核调用堆栈,并打印函数的调用关系。

   下面给出一段实验代码,在该内核模块中,我们定义四个函数aaabbbcccddd,然后bbb中调用aaaccc中调用bbbddd函数谁都不调用。在入口函数中,我们调用cccddd函数。

   在aaa函数中使用dump_stack函数,查看aaa函数的调用栈

#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/delay.h>void aaa(void) {printk(KERN_EMERG "aaa\n");dump_stack();msleep(100);
}void bbb(void) {printk(KERN_EMERG "bbb\n");aaa();msleep(100);
}void ccc(void) {printk(KERN_EMERG "ccc\n");bbb();msleep(100);
}void ddd(void) {printk(KERN_EMERG "ddd\n");msleep(100);
}static int __init chrdevTest_init(void) {printk(KERN_EMERG "INIT func\r\n");ccc();ddd();return 0;
}static void __exit chrdevTest_exit(void) {printk(KERN_EMERG "EXIT func\r\n");
}module_init(chrdevTest_init);
module_exit(chrdevTest_exit);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("wp");

   加载驱动后,显示信息如下所示,可以看到先打印INIT func,然后按照调用关系分别打印ccc,bbb,aaa。

   之后就是aaa的调用栈,框中就是调用关系,因为是chrdevTest_init调用cccccc调用bbbbbb调用aaa,由于Cortex-A处理器的堆栈是向下生长的,因此先压入chrdevTest_init函数(地址较大),在压入ccc函数(地址递减),以此类推到aaa函数。

   最后打印了一下谁都不调用的ddd函数。
在这里插入图片描述

   dump_stack()函数有助于我们调试,查看目标函数的调用关系

   如果信息消失了,可以使用dmesg指令重新再控制台打印出来

WARN_ON函数

   WARN_ON(condition)函数的作用是,在括号中的条件成立时,内核会抛出栈回溯,打印函数的调用关系,通常用于内核抛出一个警告,暗示某种不太合理的事情发生了。

   WARN_ON实际上也是调用了dump_stack,只是多了一个条件判断是否成立。

   在刚刚dump_stack函数的实验代码基础上,我们将aaa函数中dump_stack函数调用的位置改为WARN_ON(1)

void aaa(void) {printk(KERN_EMERG "aaa\n");WARN_ON(1);						// 条件为真,打印调用信息msleep(100);
}

   加载模块后,基本和上一个dump_stack的结果一样,但是可以看到,dump_stack函数也被压进调用栈了,因此可以确定WARN_ON是调用的dump_stack函数。
在这里插入图片描述

   现在我们将WARN_ON中的条件改为false,再看看结果:

void aaa(void) {printk(KERN_EMERG "aaa\n");WARN_ON(0);						// 条件为假,不打印调用信息msleep(100);
}

   可以看到控制台并没有输出调用栈。

在这里插入图片描述

BUG_ON函数

   内核中也有许多地方用到了BUG_ON函数,这个函数就像一个内核运行时的断言,意味着本来不该执行到BUG_ON这句,一旦执行就会抛出oops,导致栈的回溯和错误信息的打印,大部分体系结构把BUG()BUG_ON()定义成某种非法操作,这样自然会产生需要的oops

   在上面代码的基础上,将aaa函数中改为BUG_ON函数

void aaa(void) {printk(KERN_EMERG "aaa\n");BUG_ON(1);msleep(100);
}

   加载模块后,打印出来的信息如下所示,可以看到其中抛出了oops,并且最后并没有打印ddd函数的信息。然后函数调用关系也打印出来了,寄存器值也都打印出来了。

在这里插入图片描述

   我们将BUG_ON中的条件改为false

void aaa(void) {printk(KERN_EMERG "aaa\n");BUG_ON(0);msleep(100);
}

   加载驱动后如下所示,可以看到不打印任何堆栈信息,并且ddd函数可以顺利执行。

在这里插入图片描述

panic函数

   可以用panic()引发更严重的错误。调用panic()不但会打印错误消息(Oops)而且还会挂起整个系统。显然,你只应该在极端恶劣的情况下使用它。

   将同样的位置换为panic()函数

void aaa(void) {printk(KERN_EMERG "aaa\n");panic("###########################################wpwpwpwp");msleep(100);
}

   可以看到,打印完aaa函数后,控制台打印出panic中的字符串,然后整个进程进入到了阻塞状态。

在这里插入图片描述

参考资料

[1] Linux打印内核函数调用栈(dump_stack)

[2] Linux内核之BUG_ON()和WARN_ON()

[3] linux 内核态调试函数BUG_ON()

这篇关于嵌入式驱动学习第六周——内核函数调用(堆栈打印)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/880684

相关文章

使用docker搭建嵌入式Linux开发环境

《使用docker搭建嵌入式Linux开发环境》本文主要介绍了使用docker搭建嵌入式Linux开发环境,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面... 目录1、前言2、安装docker3、编写容器管理脚本4、创建容器1、前言在日常开发全志、rk等不同

Java+AI驱动实现PDF文件数据提取与解析

《Java+AI驱动实现PDF文件数据提取与解析》本文将和大家分享一套基于AI的体检报告智能评估方案,详细介绍从PDF上传、内容提取到AI分析、数据存储的全流程自动化实现方法,感兴趣的可以了解下... 目录一、核心流程:从上传到评估的完整链路二、第一步:解析 PDF,提取体检报告内容1. 引入依赖2. 封装

Unity新手入门学习殿堂级知识详细讲解(图文)

《Unity新手入门学习殿堂级知识详细讲解(图文)》Unity是一款跨平台游戏引擎,支持2D/3D及VR/AR开发,核心功能模块包括图形、音频、物理等,通过可视化编辑器与脚本扩展实现开发,项目结构含A... 目录入门概述什么是 UnityUnity引擎基础认知编辑器核心操作Unity 编辑器项目模式分类工程

Python学习笔记之getattr和hasattr用法示例详解

《Python学习笔记之getattr和hasattr用法示例详解》在Python中,hasattr()、getattr()和setattr()是一组内置函数,用于对对象的属性进行操作和查询,这篇文章... 目录1.getattr用法详解1.1 基本作用1.2 示例1.3 原理2.hasattr用法详解2.

Linux之platform平台设备驱动详解

《Linux之platform平台设备驱动详解》Linux设备驱动模型中,Platform总线作为虚拟总线统一管理无物理总线依赖的嵌入式设备,通过platform_driver和platform_de... 目录platform驱动注册platform设备注册设备树Platform驱动和设备的关系总结在 l

Java实现预览与打印功能详解

《Java实现预览与打印功能详解》在Java中,打印功能主要依赖java.awt.print包,该包提供了与打印相关的一些关键类,比如PrinterJob和PageFormat,它们构成... 目录Java 打印系统概述打印预览与设置使用 PageFormat 和 PrinterJob 类设置页面格式与纸张

Java通过驱动包(jar包)连接MySQL数据库的步骤总结及验证方式

《Java通过驱动包(jar包)连接MySQL数据库的步骤总结及验证方式》本文详细介绍如何使用Java通过JDBC连接MySQL数据库,包括下载驱动、配置Eclipse环境、检测数据库连接等关键步骤,... 目录一、下载驱动包二、放jar包三、检测数据库连接JavaJava 如何使用 JDBC 连接 mys

SQLite3 在嵌入式C环境中存储音频/视频文件的最优方案

《SQLite3在嵌入式C环境中存储音频/视频文件的最优方案》本文探讨了SQLite3在嵌入式C环境中存储音视频文件的优化方案,推荐采用文件路径存储结合元数据管理,兼顾效率与资源限制,小文件可使用B... 目录SQLite3 在嵌入式C环境中存储音频/视频文件的专业方案一、存储策略选择1. 直接存储 vs

嵌入式数据库SQLite 3配置使用讲解

《嵌入式数据库SQLite3配置使用讲解》本文强调嵌入式项目中SQLite3数据库的重要性,因其零配置、轻量级、跨平台及事务处理特性,可保障数据溯源与责任明确,详细讲解安装配置、基础语法及SQLit... 目录0、惨痛教训1、SQLite3环境配置(1)、下载安装SQLite库(2)、解压下载的文件(3)、

Python打印对象所有属性和值的方法小结

《Python打印对象所有属性和值的方法小结》在Python开发过程中,调试代码时经常需要查看对象的当前状态,也就是对象的所有属性和对应的值,然而,Python并没有像PHP的print_r那样直接提... 目录python中打印对象所有属性和值的方法实现步骤1. 使用vars()和pprint()2. 使