微知-如何给某个文件打开linux内核的动态打印开关?(echo -n ‘file probe.c +p‘ > /sys/kernel/debug/dynamic_debug/control)

背景

printk 只能支持7个等级,并且是全局统一开关。粒度太粗。如何做到动态开关某个文件的debug?
linux内核提供了dynamic debug的机制(后文称dd)。
本文以pci驱动中的probe.c为例子介绍 开启内核中已使用了dd模块并且并已编译到内核的模块如何直接开启打印。

使用方法

在内核源码中driver/pci/probe.c中使用了`dev_dbg`的打印函数。该函数是dev.h提供的,封装的是`dynamic_dev_debug`。该函数是dynamic_debug.h提供的动态打印函数。类似的还有给`pr_debug`的`dynamic_pr_debug`和给netdev的`dynamic_netdev_debug`。

比如probe中有函数打印了`dev_debug("scanning bus"`,需要linux运行中动态打开它。操作方式

echo -n 'file probe.c +p' > /sys/kernel/debug/dynamic_debug/control

dmesg | grep scanning

这样就打开了
如果要开启某个模块的就用module xxx 替换其中的file xxx
开启后,在dmesg中就会打开

dd大概实现机制

接口依赖

dev_dbg
device.h
dynamic_dev_debug
dynamic_debug.h
pr_debug
dynamic_pr_debug
定义_ddebug的struct存储信息
根据config条件判断是否打印
调用printk

内核配置开关

linux内核提供了CONFIG_DYNAMIC_DEBUG的config配置(可以`cat /boot/config
*| grep -i dynamic`查看)
开启动态debug功能后,dd会在debugfs中提供control的配置文件,该文件就是前面开启某个文件debug功能的文件。
另外内核默认会将debugfs mount到/sys/kernel/debug下 如果没有`mount -t debugfs -o /sys/kernel/debug`然后在该目录下的dynamic_debu目录就是控制开关。

底层原理

dd将每一个打印都会调用一个宏定义,该宏定义会在调用打印函数的时候 将当前打印所在的模块 函数 行 fmt格式化字符串 存入一个struct _ddebug的实体中,每个函数都会定义一个。然后将这个变量指定___attribute 将其放入__verbose的section段 链接的时候自动放入。后期使用的时候当做数组一样取用
这个struct中有一个flags用来判断是否需要打印,如果开启就会调用到内部函数最终调用printk。control中的+p就是设置flags

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值