perf top 显示大量 [unknown] 如何加载内核/用户符号表

11次阅读

perf top 显示大量[unknown]是因为缺少内核vmlinux和用户态调试符号;需指定vmlinux路径并安装debuginfo包或编译时加-g,JIT场景需额外配置如jvm参数或perf-map-agent。

perf top 显示大量 [unknown] 如何加载内核/用户符号表

perf top 显示大量 [unknown],说明 perf 无法解析函数符号——既找不到内核符号(vmlinux),也缺少用户态程序的调试信息(debug symbols)或动态链接映射。解决核心是让 perf 能访问到正确的符号表:内核用 vmlinux,用户态依赖 /proc/PID/maps + 对应二进制/so 的调试符号(如 .debug 段或 debuginfo 包)。

确认并加载内核符号(vmlinux)

perf 默认不自带内核符号,需手动指定系统编译好的 vmlinux 文件(非压缩的、带调试信息的内核镜像):

  • 检查当前内核版本:uname -r
  • 查找对应 vmlinux:通常在 /usr/lib/debug/boot/vmlinux-$(uname -r)(Fedora/RHEL)或 /usr/lib/debug/lib/modules/$(uname -r)/vmlinuxubuntu/debian 需先安装 linux-image-$(uname -r)-dbgsym 包(从 ddebs.ubuntu.com 获取)
  • 运行 perf 时显式指定:perf top -k /path/to/vmlinux;也可设环境变量 PERF_VMLINUX=/path/to/vmlinux 全局生效
  • 验证是否生效:执行 perf top 后按 Shift+H 查看帮助,若看到 “Kernel symbol resolution: ON” 即成功

补全用户态符号(二进制 + debuginfo)

用户空间函数显示为 [unknown],常见于未安装调试包、strip 过的二进制、或 JIT/动态生成代码(如 java/js):

  • 安装系统级 debuginfo:RHEL/centosdebuginfo-install $(rpm -qf /bin/bash);Fedora 用 dnf debuginfo-install glibc 等;Ubuntu 安装 *-dbgsym 包(需启用 -proposedddebs 源)
  • 对自研程序:编译时加 -g,避免 strip;若已部署,可将带调试信息的二进制复制到目标机同路径,perf 会自动读取
  • 验证用户符号:运行 perf record -g ./myapp 后用 perf report --no-children,观察是否出现具体函数名而非 [unknown]

处理 JIT 或动态代码(Java/python/node.js

这类运行时生成的代码默认无符号,需额外支持:

  • Java:启动 JVM 时添加 -XX:+PreserveFramePointer -XX:+UnlockDiagnosticVMOptions -XX:+DebugNonSafepoints,并确保 perf-map-agent 已 attach(或用 JDK 17+ 内置 AsyncProfiler 配合 perf)
  • node.js:启用 --perf-basic-prof(V8 选项),生成 isolate-*.log,再用 perf script -F +pid,+tid 关联
  • Python:需 py-spyperf + libpython debuginfo,并设置 export PYTHONFAULTHANDLER=1

其他关键检查项

即使符号文件存在,权限、路径或配置错误仍会导致失败:

  • 确认 perf 有读取 /proc/kallsyms 权限(需 root 或 cap_sys_admin
  • 检查 /proc/sys/kernel/kptr_restrict 是否为 0(否则内核地址被隐藏,影响符号解析)
  • 用户态程序若使用 ASLR,perf 依赖 /proc/PID/maps 定位模块基址;确保 perf 在目标进程运行中采样(而非事后分析 dump)
  • 临时调试可用 perf record -e cycles:u -g --call-graph dwarf ./app 强制用 DWARF 解析用户调用

不复杂但容易忽略。重点就是两件事:给 perf 找到带调试信息的 vmlinux,再让每个用户二进制都附带可读的符号——系统包走 debuginfo,自研程序留 -g,JIT 场景配专用 agent。做完这些,[unknown] 就会大幅减少。

text=ZqhQzanResources