Linux内核基础篇——常用调试技巧汇总

简介: Linux内核基础篇——常用调试技巧汇总

printk

printk共有8个等级,从0-7,等级依次降低。

打印等级可以通过修改/proc/sys/kernel/printk来改变。

查看printk等级:

cat /proc/sys/kernel/printk
7 4 1 7

打开内核所有打印:

echo 8 > /proc/sys/kernel/printk

具体内容:

Linux内核基础篇——printk

动态输出

printk打印是全局的,使用动态输出则可以有选择地输出某个模块或某个子系统的打印,pr_debug()就是使用了动态输出。

打开svcsock.c文件中所有的动态输出语句

# echo 'file svcsock.c +p' > /sys/kernel/debug/dynamic_debug/control

打开usbcore模块中所有的动态输出语句

# echo 'module usbcore +p' > /sys/kernel/debug/dynamic_debug/control

打开svc_process()函数中所有的动态输出语句

# echo 'func svc_process() +p' > /sys/kernel/debug/dynamic_debug/control

打开文件路径包含usb的文件里所有的动态输出语句

# echo -n '*usb* +p' > /sys/kernel/debug/dynamic_debug/control

打开系统所有的动态输出语句

# echo -n '+p' > /sys/kernel/debug/dynamic_debug/control

具体内容:

Linux内核基础篇——动态输出调试

BUG()和BUG_ON()

在内核中经常看到BUG()和BUG_ON()宏,这也是内核调试常用的技巧之一。

#<include/asm-generic/bug.h>
#define BUG_ON(condition) do { if (unlikely(condition)) BUG();} while(0)
#define BUG() do { \
 printk("BUG:failure at %s:%d/%s()!\n",__FILE__, __LINE__, __func__); \
    panic("BUG!"); \
}while(0)

对于BUG_ON()宏来说,满足条件(condition)就会触发BUG()宏,它会使用panic()函数来主动让系统宕机。通常只有一些内核的bug才会触发BUG_ON()宏,在实际产品中使用该宏需要小心谨慎。

WARN_ON()宏相对会好一些,不会触发panic()函数,使系统主动宕机,但会输出函数调用栈信息,提示开发者可能发生了一些不好的事情。

dump_stack()

dump_stack()用于打印函数的调用关系,例如,在驱动的入口处添加一句dump_stack():

static int __init hello_init(void)
{
......
 dump_stack();
......
 return 0;
}

dump_stack()打印的信息:

dump_stack()打印的函数调用关系,需要从下往上看:

ret_fast_syscall->sys_finit_module->load_module->do_init_module->do_one_initcall->hello_init

通过调用关系看出,驱动入口函数hello_init,是由do_init_module调用do_one_initcall,然后再调用到了hello_init。

devmem

通常在Linux驱动中操作寄存器,需要先通过ioremap映射寄存器基地址,转为虚拟地址后才可进行访问。

而devmem可以直接对寄存器进行读写操作,而不需要专门写一个驱动这么麻烦,这在调试的时候很有用。

devmem命令格式:

Usage: devmem ADDRESS [WIDTH [VALUE]]
Read/write from physical address
 ADDRESS Address to act upon
 WIDTH Width (8/16/...)
 VALUE Data to be written

ADDRESS:物理地址

WIDTH:位宽,32位、64位等等

VALUE:要写入的值

例如,读取32位寄存器0x40200000的值:

devmem 0x40200000 32

向32位寄存器0x40200000写入0x12345678

devmem 0x40200000 32 0x12345678

具体内容:

内核调试之devmem直接读写寄存器

end

猜你喜欢

Linux内核基础篇——动态输出调试

Linux内核基础篇——printk调试

Linux内核基础篇——initcall

RISC-V SiFive U64内核——HPM硬件性能监视器

RISC-V SiFive U64内核——L2 Prefetcher预取器

RISC-V SiFive U54内核——PMP物理内存保护

RISC-V SiFive U54内核——PLIC平台级中断控制器

RISC-V SiFive U54内核——CLINT中断控制器

RISC-V SiFive U54内核——中断和异常详解

实战 | RISC-V Linux入口地址2M预留内存优化

RISC-V Linux启动之页表创建分析

RISC-V Linux汇编启动过程分析

RISC-V 入门笔记(新手必看!)

教你在QEMU上运行RISC-V Linux

OpenSBI三种固件的区别

写给新手的MMU工作原理

相关文章
|
9天前
|
Linux
Linux(5)WIFI/BT调试笔记
Linux(5)WIFI/BT调试笔记
31 0
|
21天前
|
Linux C语言
Linux内核队列queue.h
Linux内核队列queue.h
|
2月前
|
Linux 编译器 程序员
【Linux 调试秘籍】深入探索 C++:运行时获取堆栈信息和源代码行数的终极指南
【Linux 调试秘籍】深入探索 C++:运行时获取堆栈信息和源代码行数的终极指南
69 0
|
4天前
|
网络协议 算法 Linux
【Linux】深入探索:Linux网络调试、追踪与优化
【Linux】深入探索:Linux网络调试、追踪与优化
|
21天前
|
存储 Linux
linux查看系统版本、内核信息、操作系统类型版本
linux查看系统版本、内核信息、操作系统类型版本
56 9
|
1月前
|
Ubuntu Linux
linux查看系统版本及内核信息
在Linux中检查系统版本和内核信息,可使用`uname -r`查看内核版本,`uname -a`获取详细信息,或者查看`/proc/version`。要了解发行版版本,尝试`lsb_release -a`(如果安装了)或查阅`/etc/os-release`。Red Hat家族用`/etc/redhat-release`,Debian和Ubuntu系用`/etc/issue`及相关文件。不同发行版可能需不同命令。
32 3
|
1天前
|
算法 大数据 Linux
深入理解Linux内核的进程调度机制
【4月更文挑战第30天】操作系统的核心职能之一是有效地管理和调度进程,确保系统资源的合理分配和高效利用。在众多操作系统中,Linux因其开源和高度可定制的特点,在进程调度机制上展现出独特优势。本文将深入探讨Linux内核中的进程调度器——完全公平调度器(CFS),分析其设计理念、实现原理及面临的挑战,并探索未来可能的改进方向。
|
1天前
|
算法 安全 Linux
深度解析:Linux内核内存管理机制
【4月更文挑战第30天】 在操作系统领域,内存管理是核心功能之一,尤其对于多任务操作系统来说更是如此。本文将深入探讨Linux操作系统的内核内存管理机制,包括物理内存的分配与回收、虚拟内存的映射以及页面替换算法等关键技术。通过对这些技术的详细剖析,我们不仅能够理解操作系统如何高效地利用有限的硬件资源,还能领会到系统设计中的性能与复杂度之间的权衡。
|
2天前
|
弹性计算 网络协议 Shell
自动优化Linux 内核参数
【4月更文挑战第29天】
5 1
|
3天前
|
弹性计算 网络协议 Linux
自动优化 Linux 内核参数
【4月更文挑战第28天】
10 0

热门文章

最新文章