linux内核调试工具之kprobe
创始人
2024-01-16 19:13:05

目录

一、内核调试的痛点

二、kprobe的优点

三、kprobe探测点的要点

四、探测点的开销与优化

五、内核配置

六、API

七、程序架构

八、实例


一、内核调试的痛点


        内核调试,添加打印信息。在运行过程中想看某个函数的变量,需要重新编译内核。这样破坏了执行的过程。

二、kprobe的优点


        kprobe 可以在系统运行期间,自定义回调函数,动态插入探测点。当内核执行到探测函数时,会调用回调函数。同时,也可以动态移除探测函数。

   探测点的类型

  •  pre_handler:在被探测函数执行前回调
  •  post_handler:在被探测函数执行
  •  falut_handler:探测期间发生的错误,但在新内核中被删除

 

三、kprobe探测点的要点

  • 可以探测的函数,一般来说,可以探测任意函数,尤其是可以探测中断函数。
  • 在kernel/kprobes.c和arch/*/kernel/kprobes.c 用于实现自身的函数不能探测
  •  do_page_fault与 notifier_call_chain 会出现错误。在register_*probe 注册时会返回-EINVAL
  • 内联函数,不能保证都探测,因为gcc可能会优化某些函数
  • 回调函数,通过修改内核的数据结构或pt_regs结构体中的数据,可以修改被探测函数的环境。如在测试中安装bug修复信息或者注入错误代码。
  • kprobe会避免在处理探测点函数时 再次调用另一个探测点的回调函数。如在printk上注册的探测点,在它的回调函数中可能再次调用printk函数,此时将不再触发printk探测点的回调。
  • 在注册和销毁的过程中,不要使用mutexs或allocate memory函数
  • 多个回调函数可以同时在不同的CPU上触发。
  • 探测函数运行在禁用抢占或禁用中断的情况下运行,在这种情况下中断函数不能调用会放弃CPU的函数。
  • 如何函数的调用次数和返回次数不等,则在类似的函数上注册kretprobe不会达到预期的效果。如do_exit()
  • 在进入和退出一个函数时,CPU运行再非当前任务所有的栈上,往该函数上注册kretprobe可能会导致不可预料的后果。因此X86_64结构下为__switch_to()注册kretprobe会返回-EINVAL

四、探测点的开销与优化


2005年使用的典型CPU,kprobe命中需要0.5到1.0微秒来处理,返回探测命中的时间通常比kprobe命中的时间长50-75%

k = kprobe; r = return probe; kr = kprobe + return probe
on same function
i386: Intel Pentium M, 1495 MHz, 2957.31 bogomips
k = 0.57 usec; r = 0.92; kr = 0.99
x86_64: AMD Opteron 246, 1994 MHz, 3971.48 bogomips
k = 0.49 usec; r = 0.80; kr = 0.82
ppc64: POWER5 (gr), 1656 MHz (SMT disabled, 1 virtual CPU per physical CPU)
k = 0.77 usec; r = 1.26; kr = 1.45


优化的kprobe命中需要0.07到0.1微秒才能处理

k = unoptimized kprobe, b = boosted (single-step skipped), o = optimized kprobe,
r = unoptimized kretprobe, rb = boosted kretprobe, ro = optimized kretprobe.
i386: Intel(R) Xeon(R) E5410, 2.33GHz, 4656.90 bogomips
k = 0.80 usec; b = 0.33; o = 0.05; r = 1.10; rb = 0.61; ro = 0.33
x86-64: Intel(R) Xeon(R) E5410, 2.33GHz, 4656.90 bogomips
k = 0.99 usec; b = 0.43; o = 0.06; r = 1.24; rb = 0.68; ro = 0.30

五、内核配置

CONFIG_KPROBES
CONFIG_KALLSYMS
CONFIG_KALLSYMS_ALL
CONFIG_DEBUG_INFO

六、API

#include 
#include //注册
int register_kprobe(struct kprobe *kp);//探测前回调函数
int pre_handler(struct kprobe *p, struct pt_regs *regs);//探测后回调函数
void post_handler(struct kprobe *p, struct pt_regs *regs,unsigned long flags);int disable_kprobe(struct kprobe *kp);int enable_kprobe(struct kprobe *kp);void unregister_kprobes(struct kprobe **kps, int num);

七、程序架构

#include 
#include static int handler_pre(struct kprobe *p, struct pt_regs *regs)
{}static void handler_post(struct kprobe *p, struct pt_regs *regs, unsigned long flags)
{}static struct kprobe kpb;
static int __init kprobe_init(void)
{kpb.pre_handler = handler_pre;kpb.post_handler = handler_post;kpb.symbol_name = kprobe_func;if (register_kprobe(&kpb)) {pr_alert("register_kprobe failed!\n“);return -EINVAL;}
}static void __exit kprobe_exit(void)
{unregister_kprobe(&kpb);pr_info("bye, unregistering kernel probe @ '%s'\n", kprobe_func);
}module_init(kprobe_init);
module_exit(kprobe_exit);

八、实例

使用kprobe跟踪do_sys_openat2,并输出打开的文件名参数

内核中do_sys_openat2的原型

static long do_sys_openat2(int dfd, const char __user *filename,struct open_how *how)
{}

1、内核模块程序

kprobe.c源码


#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include #include 
#include MODULE_AUTHOR("wy");
MODULE_LICENSE("Dual MIT/GPL");
MODULE_VERSION("0.1");static spinlock_t lock;
static struct kprobe kpb;
static u64 tm_start, tm_end;
static char *fname;//接收脚本传递的参数 kprobe_func
#define MAX_FUNCNAME_LEN  64
static char kprobe_func[MAX_FUNCNAME_LEN];
module_param_string(kprobe_func, kprobe_func, sizeof(kprobe_func), 0);
MODULE_PARM_DESC(kprobe_func, "function name to attach a kprobe to");#define pr_fmt(fmt) "%s:%s(): " fmt, KBUILD_MODNAME, __func__//显示时间差
#define SHOW_DELTA(later, earlier)  do {    \if (time_after((unsigned long)later, (unsigned long)earlier)) { \s64 delta_ns = ktime_to_ns(ktime_sub(later, earlier));      \pr_info("delta: %lld ns", delta_ns);       \if (delta_ns/1000 >= 1)                    \pr_info(" %lld us", delta_ns/1000);    \if (delta_ns/1000000 >= 1)                 \pr_info(" %lld ms", delta_ns/1000000); \} else  \pr_warn("SHOW_DELTA(): *invalid* earlier > later?\n");  \
} while (0)//探测前的执行
static int handler_pre(struct kprobe *p, struct pt_regs *regs)
{char *param_fname_reg;param_fname_reg = (char __user *)regs->si;
#if 1//拷贝数据if (!strncpy_from_user(fname, param_fname_reg, PATH_MAX + 1))
#else/* 使用 copy_from_user() 会产生调度 导致CPU挂掉*/if (!copy_from_user(fname, (const char __user *)regs->si,strnlen_user((const char __user *)regs->si, PATH_MAX + 1)))
#endifreturn -EFAULT;pr_info("FILE being opened: reg:0x%px   fname:%s\n",(void *)param_fname_reg, fname);spin_lock(&lock);tm_start = ktime_get_real_ns();spin_unlock(&lock);return 0;
}//探测之后的输出
static void handler_post(struct kprobe *p, struct pt_regs *regs, unsigned long flags)
{spin_lock(&lock);tm_end = ktime_get_real_ns();//计算时间SHOW_DELTA(tm_end, tm_start);spin_unlock(&lock);
}static int __init kprobe_init(void)
{if (kprobe_func[0] == '\0') {pr_warn("expect a valid kprobe_func= module parameter");return -EINVAL;}//申请内存空间,用来存储do_sys_openat2参数名fname = kzalloc(PATH_MAX + 1, GFP_ATOMIC);if (unlikely(!fname))return -ENOMEM;kpb.pre_handler = handler_pre;kpb.post_handler = handler_post;kpb.symbol_name = kprobe_func;//注册if (register_kprobe(&kpb)) {pr_alert("kernel fun register_kprobe failed!\n", kprobe_func);return -EINVAL;}pr_info("registering kernel probe @ '%s'\n", kprobe_func);spin_lock_init(&lock);return 0;		/* success */
}static void __exit kprobe_exit(void)
{kfree(fname);unregister_kprobe(&kpb);pr_info("bye, unregistering kernel probe @ '%s'\n", kprobe_func);
}module_init(kprobe_init);
module_exit(kprobe_exit);

Makefile

FNAME_C := kprobeKDIR ?= /lib/modules/$(shell uname -r)/buildCC     := $(CROSS_COMPILE)gcc
PWD            := $(shell pwd)
obj-m          += ${FNAME_C}.oall:make -C $(KDIR) M=$(PWD) modules
install:makesudo make -C $(KDIR) M=$(PWD) modules_install
clean:make -C $(KDIR) M=$(PWD) clean

2、用户态应用程序

测试程序打开/home/kprobe.c  测试在kprobe探测能否探测到

#include
#include void main()
{FILE *fp = NULL;while(1){fp = fopen("/home/kprobe.c","r+");if(fp == NULL)return;fclose(fp);usleep(10000);}
}

 gcc main.c  生成a.out

3、运行测试

运行用户程序

a.out

运行内核程序

insmod ./kprobe.ko kprobe_func=do_sys_openat2

运行输出截取,运行时很多的干扰可以使用 journalctl 

#journalctl -k >log.txt#cat log.txt | grep "/home/kprobe.c"

将内核输出都写到log.txt中(dmesg只能写部分),使用grep检索所需要查询的信息。

10月 24 21:24:58 wy-virtual-machine kernel: 3_kprobe:handler_pre(): FILE being opened: reg:0x00007ffd9a320820   fname:/sys/module/3_kprobe/uevent
10月 24 21:24:58 wy-virtual-machine kernel: 3_kprobe:handler_post(): delta: 37 ns
10月 24 21:24:58 wy-virtual-machine kernel: 3_kprobe:handler_pre(): FILE being opened: reg:0x00007ffd9a320890   fname:/run/udev/data/+module:3_kprobe
10月 24 21:24:58 wy-virtual-machine kernel: 3_kprobe:handler_post(): delta: 19 ns
10月 24 21:24:58 wy-virtual-machine kernel: 3_kprobe:handler_pre(): FILE being opened: reg:0x000055cd74eb8007   fname:/home/kprobe.c
10月 24 21:24:58 wy-virtual-machine kernel: 3_kprobe:handler_post(): delta: 40 ns

在输出信息中可以看到函数的地址、打开的文件名、函数执行的时间40ns等信息

参考

Kernel Probes (Kprobes) — The Linux Kernel documentation

相关内容

热门资讯

北京的名胜古迹 北京最著名的景... 北京从元代开始,逐渐走上帝国首都的道路,先是成为大辽朝五大首都之一的南京城,随着金灭辽,金代从海陵王...
苗族的传统节日 贵州苗族节日有... 【岜沙苗族芦笙节】岜沙,苗语叫“分送”,距从江县城7.5公里,是世界上最崇拜树木并以树为神的枪手部落...
长白山自助游攻略 吉林长白山游... 昨天介绍了西坡的景点详细请看链接:一个人的旅行,据说能看到长白山天池全凭运气,您的运气如何?今日介绍...
demo什么意思 demo版本... 618快到了,各位的小金库大概也在准备开闸放水了吧。没有小金库的,也该向老婆撒娇卖萌服个软了,一切只...
世界上最漂亮的人 世界上最漂亮... 此前在某网上,选出了全球265万颜值姣好的女性。从这些数量庞大的女性群体中,人们投票选出了心目中最美...
长白山自助游攻略 吉林长白山游... 昨天介绍了西坡的景点详细请看链接:一个人的旅行,据说能看到长白山天池全凭运气,您的运气如何?今日介绍...
脚上的穴位图 脚面经络图对应的... 人体穴位作用图解大全更清晰直观的标注了各个人体穴位的作用,包括头部穴位图、胸部穴位图、背部穴位图、胳...
猫咪吃了塑料袋怎么办 猫咪误食... 你知道吗?塑料袋放久了会长猫哦!要说猫咪对塑料袋的喜爱程度完完全全可以媲美纸箱家里只要一有塑料袋的响...
北京的名胜古迹 北京最著名的景... 北京从元代开始,逐渐走上帝国首都的道路,先是成为大辽朝五大首都之一的南京城,随着金灭辽,金代从海陵王...
苗族的传统节日 贵州苗族节日有... 【岜沙苗族芦笙节】岜沙,苗语叫“分送”,距从江县城7.5公里,是世界上最崇拜树木并以树为神的枪手部落...
世界上最漂亮的人 世界上最漂亮... 此前在某网上,选出了全球265万颜值姣好的女性。从这些数量庞大的女性群体中,人们投票选出了心目中最美...
应用未安装解决办法 平板应用未... ---IT小技术,每天Get一个小技能!一、前言描述苹果IPad2居然不能安装怎么办?与此IPad不...
阿西吧是什么意思 阿西吧相当于... 即使你没有受到过任何外语培训,你也懂四国语言。汉语:你好英语:Shit韩语:阿西吧(아,씨발! )日...
脚上的穴位图 脚面经络图对应的... 人体穴位作用图解大全更清晰直观的标注了各个人体穴位的作用,包括头部穴位图、胸部穴位图、背部穴位图、胳...
demo什么意思 demo版本... 618快到了,各位的小金库大概也在准备开闸放水了吧。没有小金库的,也该向老婆撒娇卖萌服个软了,一切只...
北京的名胜古迹 北京最著名的景... 北京从元代开始,逐渐走上帝国首都的道路,先是成为大辽朝五大首都之一的南京城,随着金灭辽,金代从海陵王...
苗族的传统节日 贵州苗族节日有... 【岜沙苗族芦笙节】岜沙,苗语叫“分送”,距从江县城7.5公里,是世界上最崇拜树木并以树为神的枪手部落...
长白山自助游攻略 吉林长白山游... 昨天介绍了西坡的景点详细请看链接:一个人的旅行,据说能看到长白山天池全凭运气,您的运气如何?今日介绍...