首先是在systemtap脚本中定义能够抓取golang堆栈的功能函数:
function print_time_now() {
ms = gettimeofday_us()
sec = gettimeofday_s()
printf(“%s %d\(us\)\n”,ctime(sec),ms-sec*1000000 )
}
function print_x8664_gostack(){
goBaseName = “runtime.go exit”
rip = u_register(“rip”)
rbp_itr = u_register(“rbp”)
funcName = usymname(rip)
fileLine = usymline(rip)
printf(“%p: %s\n”,rip,funcName)
while(funcName != goBaseName && rbp_itr !=0){
funcName = usymname(user_long(rbp_itr+8))
fileLine = usymline(user_long(rbp_itr+8))
printf(“%p: %s \n”,user_long(rbp_itr+8),funcName)
rbp_itr = user_long(rbp_itr)
}
printf(“……….……..\n”)
}
function dumpProccessInfo(){
printf(“pid:%d, tid:%d, pid name:%s, ppid:%d, ppid name:%s, func:%s, cmd:%s \n”,pid(),tid(),pid2execname(pid()),ppid(), pid2execname(ppid()), ppfunc(), cmdline_str())
)
}
下面结合实际例子给出上述systemtap自定义函数的使用
probe process(“/usr/bin/docker-runc”),function(“github.com/opencontainers/runc/libcontainer.prepareRoot”) {
print_timeNow()
print_x8664_gostack()
}
如何通过systemtap修改程序中变量
systemtap除了可以抓取程序执行状态外,还可以修改程序的变量。只需要在stap命令中添加-g参数让其工作在guru模式下即可。
例如下面脚本可以修改dockerd程序的killwithSignal函数的入参sig的值
probe process ("/usr/bin/dockerd) .function( "github .cam/docker/docker/ daemon. (*Daemon) .killwithsignal"){
print_time()
print _process_info()
printf(“before signal %s”,$sig)
$sig=4;
print kstack()
}