| # | 
 | # 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)->thread_group.next) | 
 | 	set $init_t=&init_task | 
 | 	set $next_t=(((char *)($init_t->tasks).next) - $tasks_off) | 
 | 	set var $stacksize = sizeof(union thread_union) | 
 | 	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.sp | 
 | 		set var $stack_top = ($stackp & ~($stacksize - 1)) + $stacksize | 
 |  | 
 | 		while ($stackp < $stack_top) | 
 | 			if (*($stackp) > _stext && *($stackp) < _sinittext) | 
 | 				info symbol *($stackp) | 
 | 			end | 
 | 			set $stackp += 4 | 
 | 		end | 
 | 		set $next_th=(((char *)$next_t->thread_group.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.sp | 
 | 			set var $stack_top = ($stackp & ~($stacksize - 1)) + stacksize | 
 |  | 
 | 			while ($stackp < $stack_top) | 
 | 				if (*($stackp) > _stext && *($stackp) < _sinittext) | 
 | 					info symbol *($stackp) | 
 | 				end | 
 | 				set $stackp += 4 | 
 | 			end | 
 | 			set $next_th=(((char *)$next_th->thread_group.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 btthreadstack | 
 | 	set var $pid_task = $arg0 | 
 |  | 
 | 	printf "\npid %d; comm %s:\n", $pid_task.pid, $pid_task.comm | 
 | 	printf "task struct: " | 
 | 	print $pid_task | 
 | 	printf "===================\n" | 
 | 	set var $stackp = $pid_task.thread.sp | 
 | 	set var $stacksize = sizeof(union thread_union) | 
 | 	set var $stack_top = ($stackp & ~($stacksize - 1)) + $stacksize | 
 | 	set var $stack_bot = ($stackp & ~($stacksize - 1)) | 
 |  | 
 | 	set $stackp = *((unsigned long *) $stackp) | 
 | 	while (($stackp < $stack_top) && ($stackp > $stack_bot)) | 
 | 		set var $addr = *(((unsigned long *) $stackp) + 1) | 
 | 		info symbol $addr | 
 | 		set $stackp = *((unsigned long *) $stackp) | 
 | 	end | 
 | end | 
 | document btthreadstack | 
 | 	 dump a thread stack using the given task structure pointer | 
 | end | 
 |  | 
 |  | 
 | define btt | 
 | 	set $tasks_off=((size_t)&((struct task_struct *)0)->tasks) | 
 | 	set $pid_off=((size_t)&((struct task_struct *)0)->thread_group.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 | 
 | 		btthreadstack $next_t | 
 |  | 
 | 		set $next_th=(((char *)$next_t->thread_group.next) - $pid_off) | 
 | 		while ($next_th != $next_t) | 
 | 			set $next_th=(struct task_struct *)$next_th | 
 | 			btthreadstack $next_th | 
 | 			set $next_th=(((char *)$next_th->thread_group.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)->thread_group.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->thread_group.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->thread_group.next) - $pid_off) | 
 | 		end | 
 | 		set $next_t=(char *)($next_t->tasks.next) - $tasks_off | 
 | 	end | 
 |  | 
 | 	btthreadstack $pid_task | 
 | 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)->thread_group.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->thread_group.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->thread_group.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 dump_log_idx | 
 | 	set $idx = $arg0 | 
 | 	if ($argc > 1) | 
 | 		set $prev_flags = $arg1 | 
 | 	else | 
 | 		set $prev_flags = 0 | 
 | 	end | 
 | 	set $msg = ((struct printk_log *) (log_buf + $idx)) | 
 | 	set $prefix = 1 | 
 | 	set $newline = 1 | 
 | 	set $log = log_buf + $idx + sizeof(*$msg) | 
 |  | 
 | 	# prev & LOG_CONT && !(msg->flags & LOG_PREIX) | 
 | 	if (($prev_flags & 8) && !($msg->flags & 4)) | 
 | 		set $prefix = 0 | 
 | 	end | 
 |  | 
 | 	# msg->flags & LOG_CONT | 
 | 	if ($msg->flags & 8) | 
 | 		# (prev & LOG_CONT && !(prev & LOG_NEWLINE)) | 
 | 		if (($prev_flags & 8) && !($prev_flags & 2)) | 
 | 			set $prefix = 0 | 
 | 		end | 
 | 		# (!(msg->flags & LOG_NEWLINE)) | 
 | 		if (!($msg->flags & 2)) | 
 | 			set $newline = 0 | 
 | 		end | 
 | 	end | 
 |  | 
 | 	if ($prefix) | 
 | 		printf "[%5lu.%06lu] ", $msg->ts_nsec / 1000000000, $msg->ts_nsec % 1000000000 | 
 | 	end | 
 | 	if ($msg->text_len != 0) | 
 | 		eval "printf \"%%%d.%ds\", $log", $msg->text_len, $msg->text_len | 
 | 	end | 
 | 	if ($newline) | 
 | 		printf "\n" | 
 | 	end | 
 | 	if ($msg->dict_len > 0) | 
 | 		set $dict = $log + $msg->text_len | 
 | 		set $idx = 0 | 
 | 		set $line = 1 | 
 | 		while ($idx < $msg->dict_len) | 
 | 			if ($line) | 
 | 				printf " " | 
 | 				set $line = 0 | 
 | 			end | 
 | 			set $c = $dict[$idx] | 
 | 			if ($c == '\0') | 
 | 				printf "\n" | 
 | 				set $line = 1 | 
 | 			else | 
 | 				if ($c < ' ' || $c >= 127 || $c == '\\') | 
 | 					printf "\\x%02x", $c | 
 | 				else | 
 | 					printf "%c", $c | 
 | 				end | 
 | 			end | 
 | 			set $idx = $idx + 1 | 
 | 		end | 
 | 		printf "\n" | 
 | 	end | 
 | end | 
 | document dump_log_idx | 
 | 	Dump a single log given its index in the log buffer.  The first | 
 | 	parameter is the index into log_buf, the second is optional and | 
 | 	specified the previous log buffer's flags, used for properly | 
 | 	formatting continued lines. | 
 | end | 
 |  | 
 | define dmesg | 
 | 	set $i = log_first_idx | 
 | 	set $end_idx = log_first_idx | 
 | 	set $prev_flags = 0 | 
 |  | 
 | 	while (1) | 
 | 		set $msg = ((struct printk_log *) (log_buf + $i)) | 
 | 		if ($msg->len == 0) | 
 | 			set $i = 0 | 
 | 		else | 
 | 			dump_log_idx $i $prev_flags | 
 | 			set $i = $i + $msg->len | 
 | 			set $prev_flags = $msg->flags | 
 | 		end | 
 | 		if ($i == $end_idx) | 
 | 			loop_break | 
 | 		end | 
 | 	end | 
 | end | 
 | document dmesg | 
 | 	print the kernel ring buffer | 
 | end |