|  | # | 
|  | # This file contains a few gdb macros (user defined commands) to extract | 
|  | # useful information from kernel crashdump (kdump) like stack traces of | 
|  | # all the processes or a particular process and trapinfo. | 
|  | # | 
|  | # These macros can be used by copying this file in .gdbinit (put in home | 
|  | # directory or current directory) or by invoking gdb command with | 
|  | # --command=<command-file-name> option | 
|  | # | 
|  | # Credits: | 
|  | # Alexander Nyberg <alexn@telia.com> | 
|  | # V Srivatsa <vatsa@in.ibm.com> | 
|  | # Maneesh Soni <maneesh@in.ibm.com> | 
|  | # | 
|  |  | 
|  | define bttnobp | 
|  | set $tasks_off=((size_t)&((struct task_struct *)0)->tasks) | 
|  | set $pid_off=((size_t)&((struct task_struct *)0)->pids[1].pid_list.next) | 
|  | set $init_t=&init_task | 
|  | set $next_t=(((char *)($init_t->tasks).next) - $tasks_off) | 
|  | while ($next_t != $init_t) | 
|  | set $next_t=(struct task_struct *)$next_t | 
|  | printf "\npid %d; comm %s:\n", $next_t.pid, $next_t.comm | 
|  | printf "===================\n" | 
|  | set var $stackp = $next_t.thread.esp | 
|  | set var $stack_top = ($stackp & ~4095) + 4096 | 
|  |  | 
|  | while ($stackp < $stack_top) | 
|  | if (*($stackp) > _stext && *($stackp) < _sinittext) | 
|  | info symbol *($stackp) | 
|  | end | 
|  | set $stackp += 4 | 
|  | end | 
|  | set $next_th=(((char *)$next_t->pids[1].pid_list.next) - $pid_off) | 
|  | while ($next_th != $next_t) | 
|  | set $next_th=(struct task_struct *)$next_th | 
|  | printf "\npid %d; comm %s:\n", $next_t.pid, $next_t.comm | 
|  | printf "===================\n" | 
|  | set var $stackp = $next_t.thread.esp | 
|  | set var $stack_top = ($stackp & ~4095) + 4096 | 
|  |  | 
|  | while ($stackp < $stack_top) | 
|  | if (*($stackp) > _stext && *($stackp) < _sinittext) | 
|  | info symbol *($stackp) | 
|  | end | 
|  | set $stackp += 4 | 
|  | end | 
|  | set $next_th=(((char *)$next_th->pids[1].pid_list.next) - $pid_off) | 
|  | end | 
|  | set $next_t=(char *)($next_t->tasks.next) - $tasks_off | 
|  | end | 
|  | end | 
|  | document bttnobp | 
|  | dump all thread stack traces on a kernel compiled with !CONFIG_FRAME_POINTER | 
|  | end | 
|  |  | 
|  | define btt | 
|  | set $tasks_off=((size_t)&((struct task_struct *)0)->tasks) | 
|  | set $pid_off=((size_t)&((struct task_struct *)0)->pids[1].pid_list.next) | 
|  | set $init_t=&init_task | 
|  | set $next_t=(((char *)($init_t->tasks).next) - $tasks_off) | 
|  | while ($next_t != $init_t) | 
|  | set $next_t=(struct task_struct *)$next_t | 
|  | printf "\npid %d; comm %s:\n", $next_t.pid, $next_t.comm | 
|  | printf "===================\n" | 
|  | set var $stackp = $next_t.thread.esp | 
|  | set var $stack_top = ($stackp & ~4095) + 4096 | 
|  | set var $stack_bot = ($stackp & ~4095) | 
|  |  | 
|  | set $stackp = *($stackp) | 
|  | while (($stackp < $stack_top) && ($stackp > $stack_bot)) | 
|  | set var $addr = *($stackp + 4) | 
|  | info symbol $addr | 
|  | set $stackp = *($stackp) | 
|  | end | 
|  |  | 
|  | set $next_th=(((char *)$next_t->pids[1].pid_list.next) - $pid_off) | 
|  | while ($next_th != $next_t) | 
|  | set $next_th=(struct task_struct *)$next_th | 
|  | printf "\npid %d; comm %s:\n", $next_t.pid, $next_t.comm | 
|  | printf "===================\n" | 
|  | set var $stackp = $next_t.thread.esp | 
|  | set var $stack_top = ($stackp & ~4095) + 4096 | 
|  | set var $stack_bot = ($stackp & ~4095) | 
|  |  | 
|  | set $stackp = *($stackp) | 
|  | while (($stackp < $stack_top) && ($stackp > $stack_bot)) | 
|  | set var $addr = *($stackp + 4) | 
|  | info symbol $addr | 
|  | set $stackp = *($stackp) | 
|  | end | 
|  | set $next_th=(((char *)$next_th->pids[1].pid_list.next) - $pid_off) | 
|  | end | 
|  | set $next_t=(char *)($next_t->tasks.next) - $tasks_off | 
|  | end | 
|  | end | 
|  | document btt | 
|  | dump all thread stack traces on a kernel compiled with CONFIG_FRAME_POINTER | 
|  | end | 
|  |  | 
|  | define btpid | 
|  | set var $pid = $arg0 | 
|  | set $tasks_off=((size_t)&((struct task_struct *)0)->tasks) | 
|  | set $pid_off=((size_t)&((struct task_struct *)0)->pids[1].pid_list.next) | 
|  | set $init_t=&init_task | 
|  | set $next_t=(((char *)($init_t->tasks).next) - $tasks_off) | 
|  | set var $pid_task = 0 | 
|  |  | 
|  | while ($next_t != $init_t) | 
|  | set $next_t=(struct task_struct *)$next_t | 
|  |  | 
|  | if ($next_t.pid == $pid) | 
|  | set $pid_task = $next_t | 
|  | end | 
|  |  | 
|  | set $next_th=(((char *)$next_t->pids[1].pid_list.next) - $pid_off) | 
|  | while ($next_th != $next_t) | 
|  | set $next_th=(struct task_struct *)$next_th | 
|  | if ($next_th.pid == $pid) | 
|  | set $pid_task = $next_th | 
|  | end | 
|  | set $next_th=(((char *)$next_th->pids[1].pid_list.next) - $pid_off) | 
|  | end | 
|  | set $next_t=(char *)($next_t->tasks.next) - $tasks_off | 
|  | end | 
|  |  | 
|  | printf "\npid %d; comm %s:\n", $pid_task.pid, $pid_task.comm | 
|  | printf "===================\n" | 
|  | set var $stackp = $pid_task.thread.esp | 
|  | set var $stack_top = ($stackp & ~4095) + 4096 | 
|  | set var $stack_bot = ($stackp & ~4095) | 
|  |  | 
|  | set $stackp = *($stackp) | 
|  | while (($stackp < $stack_top) && ($stackp > $stack_bot)) | 
|  | set var $addr = *($stackp + 4) | 
|  | info symbol $addr | 
|  | set $stackp = *($stackp) | 
|  | end | 
|  | end | 
|  | document btpid | 
|  | backtrace of pid | 
|  | end | 
|  |  | 
|  |  | 
|  | define trapinfo | 
|  | set var $pid = $arg0 | 
|  | set $tasks_off=((size_t)&((struct task_struct *)0)->tasks) | 
|  | set $pid_off=((size_t)&((struct task_struct *)0)->pids[1].pid_list.next) | 
|  | set $init_t=&init_task | 
|  | set $next_t=(((char *)($init_t->tasks).next) - $tasks_off) | 
|  | set var $pid_task = 0 | 
|  |  | 
|  | while ($next_t != $init_t) | 
|  | set $next_t=(struct task_struct *)$next_t | 
|  |  | 
|  | if ($next_t.pid == $pid) | 
|  | set $pid_task = $next_t | 
|  | end | 
|  |  | 
|  | set $next_th=(((char *)$next_t->pids[1].pid_list.next) - $pid_off) | 
|  | while ($next_th != $next_t) | 
|  | set $next_th=(struct task_struct *)$next_th | 
|  | if ($next_th.pid == $pid) | 
|  | set $pid_task = $next_th | 
|  | end | 
|  | set $next_th=(((char *)$next_th->pids[1].pid_list.next) - $pid_off) | 
|  | end | 
|  | set $next_t=(char *)($next_t->tasks.next) - $tasks_off | 
|  | end | 
|  |  | 
|  | printf "Trapno %ld, cr2 0x%lx, error_code %ld\n", $pid_task.thread.trap_no, \ | 
|  | $pid_task.thread.cr2, $pid_task.thread.error_code | 
|  |  | 
|  | end | 
|  | document trapinfo | 
|  | Run info threads and lookup pid of thread #1 | 
|  | 'trapinfo <pid>' will tell you by which trap & possibly | 
|  | address the kernel panicked. | 
|  | end | 
|  |  | 
|  |  | 
|  | define dmesg | 
|  | set $i = 0 | 
|  | set $end_idx = (log_end - 1) & (log_buf_len - 1) | 
|  |  | 
|  | while ($i < logged_chars) | 
|  | set $idx = (log_end - 1 - logged_chars + $i) & (log_buf_len - 1) | 
|  |  | 
|  | if ($idx + 100 <= $end_idx) || \ | 
|  | ($end_idx <= $idx && $idx + 100 < log_buf_len) | 
|  | printf "%.100s", &log_buf[$idx] | 
|  | set $i = $i + 100 | 
|  | else | 
|  | printf "%c", log_buf[$idx] | 
|  | set $i = $i + 1 | 
|  | end | 
|  | end | 
|  | end | 
|  | document dmesg | 
|  | print the kernel ring buffer | 
|  | end |