/* * stack_avail - A linux kernel module to display available stack * If the kernel stack-poison patch has been applied, this * module also displays available stack at maximum usage * Copyright Frank Sorenson 2005 * * Permission is hereby granted to copy, modify and redistribute this code * in terms of the GNU Library General Public License, Version 2 or later, * at your option. * */ #include #include static int verbose = 0; static unsigned long sinittext; static unsigned long einittext; static unsigned long stext; static unsigned long etext; // task_struct - /UML/Source/Host/linux-2.6.15-rc5-mm3+fs3/include/linux/sched.h // thread_struct - /UML/Source/Host/linux-2.6.15-rc5-mm3+fs3/include/asm-i386/processor.h // thread_info - /UML/Source/Host/linux-2.6.15-rc5-mm3+fs3/include/asm-i386/thread_info.h static inline int valid_stack_ptr(struct thread_info *tinfo, void *p) { return p > (void *)tinfo && p < (void *)tinfo + THREAD_SIZE - 3; } static int core_kernel_text(unsigned long addr) { if (addr >= (unsigned long)stext && addr <= (unsigned long)etext) return 1; if (addr >= (unsigned long)sinittext && addr <= (unsigned long)einittext) return 1; return 0; } int my_kernel_text_address(unsigned long addr) { if (core_kernel_text(addr)) return 1; return 0; // return __module_text_address(addr) != NULL; } static inline unsigned long print_context_stack(struct thread_info *tinfo, unsigned long *stack, unsigned long ebp) { unsigned long addr; #ifdef CONFIG_FRAME_POINTER while (valid_stack_ptr(tinfo, (void *)ebp)) { addr = *(unsigned long *)(ebp + 4); printk(" [<%08lx>] ", addr); print_symbol("%s", addr); printk("\n"); ebp = *(unsigned long *)ebp; } #else while (valid_stack_ptr(tinfo, stack)) { addr = *stack++; if (my_kernel_text_address(addr)) { printk(" [<%08lx>]", addr); print_symbol(" %s", addr); printk("\n"); } } #endif return ebp; } void show_trace(struct task_struct *task, unsigned long * stack) { unsigned long ebp; ebp = *(unsigned long *) task->thread.esp; while (1) { struct thread_info *context; context = (struct thread_info *) ((unsigned long)stack & (~(THREAD_SIZE - 1))); ebp = print_context_stack(context, stack, ebp); stack = (unsigned long*)context->previous_esp; if (!stack) break; printk(" =======================\n"); } } static int stack_avail_load(void) { struct task_struct *task; unsigned char *base_addr; unsigned char *max_addr; unsigned char *ptr; unsigned long current_avail; unsigned long poisoned_avail; sinittext = kallsyms_lookup_name("_sinittext"); einittext = kallsyms_lookup_name("_einittext"); stext = kallsyms_lookup_name("_stext"); etext = kallsyms_lookup_name("_etext"); printk("Displaying stack space available:\n"); for_each_process(task) { printk("%d: %s", task->pid, task->comm); base_addr = (unsigned char *)(task->thread.esp & 0xFFFFF000); max_addr = base_addr + THREAD_SIZE - 1; ptr = base_addr + sizeof(struct thread_info); while (*ptr == 'Q') { ptr ++; } current_avail = (unsigned long)(task->thread.esp) - (unsigned long)(base_addr) - sizeof(struct thread_info); poisoned_avail = THREAD_SIZE - ((unsigned long)(max_addr - ptr)) - sizeof(struct thread_info) - 1; printk(" - free stack now: %lu", current_avail); if (poisoned_avail != 0) printk(", at max usage: %lu", poisoned_avail); if (verbose) { printk("\nCurrent call Trace:\n"); show_trace(task, (unsigned long *)(task->thread.esp)); } printk("\n"); } return 0; } static void stack_avail_unload(void) { printk("stack_avail module unloading\n"); } module_init(stack_avail_load); module_exit(stack_avail_unload); module_param(verbose, int, 0); MODULE_AUTHOR ("Frank Sorenson, frank@tuxrocks.com"); MODULE_DESCRIPTION ("Displays available stack space"); MODULE_LICENSE("GPL");