1*1da177e4SLinus Torvalds /* 2*1da177e4SLinus Torvalds * Kernel unwinding support 3*1da177e4SLinus Torvalds * 4*1da177e4SLinus Torvalds * (c) 2002-2004 Randolph Chung <tausq@debian.org> 5*1da177e4SLinus Torvalds * 6*1da177e4SLinus Torvalds * Derived partially from the IA64 implementation. The PA-RISC 7*1da177e4SLinus Torvalds * Runtime Architecture Document is also a useful reference to 8*1da177e4SLinus Torvalds * understand what is happening here 9*1da177e4SLinus Torvalds */ 10*1da177e4SLinus Torvalds 11*1da177e4SLinus Torvalds #include <linux/config.h> 12*1da177e4SLinus Torvalds #include <linux/kernel.h> 13*1da177e4SLinus Torvalds #include <linux/init.h> 14*1da177e4SLinus Torvalds #include <linux/slab.h> 15*1da177e4SLinus Torvalds #include <linux/kallsyms.h> 16*1da177e4SLinus Torvalds 17*1da177e4SLinus Torvalds #include <asm/uaccess.h> 18*1da177e4SLinus Torvalds #include <asm/assembly.h> 19*1da177e4SLinus Torvalds 20*1da177e4SLinus Torvalds #include <asm/unwind.h> 21*1da177e4SLinus Torvalds 22*1da177e4SLinus Torvalds /* #define DEBUG 1 */ 23*1da177e4SLinus Torvalds #ifdef DEBUG 24*1da177e4SLinus Torvalds #define dbg(x...) printk(x) 25*1da177e4SLinus Torvalds #else 26*1da177e4SLinus Torvalds #define dbg(x...) 27*1da177e4SLinus Torvalds #endif 28*1da177e4SLinus Torvalds 29*1da177e4SLinus Torvalds extern struct unwind_table_entry __start___unwind[]; 30*1da177e4SLinus Torvalds extern struct unwind_table_entry __stop___unwind[]; 31*1da177e4SLinus Torvalds 32*1da177e4SLinus Torvalds static spinlock_t unwind_lock; 33*1da177e4SLinus Torvalds /* 34*1da177e4SLinus Torvalds * the kernel unwind block is not dynamically allocated so that 35*1da177e4SLinus Torvalds * we can call unwind_init as early in the bootup process as 36*1da177e4SLinus Torvalds * possible (before the slab allocator is initialized) 37*1da177e4SLinus Torvalds */ 38*1da177e4SLinus Torvalds static struct unwind_table kernel_unwind_table; 39*1da177e4SLinus Torvalds static LIST_HEAD(unwind_tables); 40*1da177e4SLinus Torvalds 41*1da177e4SLinus Torvalds static inline const struct unwind_table_entry * 42*1da177e4SLinus Torvalds find_unwind_entry_in_table(const struct unwind_table *table, unsigned long addr) 43*1da177e4SLinus Torvalds { 44*1da177e4SLinus Torvalds const struct unwind_table_entry *e = NULL; 45*1da177e4SLinus Torvalds unsigned long lo, hi, mid; 46*1da177e4SLinus Torvalds 47*1da177e4SLinus Torvalds lo = 0; 48*1da177e4SLinus Torvalds hi = table->length - 1; 49*1da177e4SLinus Torvalds 50*1da177e4SLinus Torvalds while (lo <= hi) { 51*1da177e4SLinus Torvalds mid = (hi - lo) / 2 + lo; 52*1da177e4SLinus Torvalds e = &table->table[mid]; 53*1da177e4SLinus Torvalds if (addr < e->region_start) 54*1da177e4SLinus Torvalds hi = mid - 1; 55*1da177e4SLinus Torvalds else if (addr > e->region_end) 56*1da177e4SLinus Torvalds lo = mid + 1; 57*1da177e4SLinus Torvalds else 58*1da177e4SLinus Torvalds return e; 59*1da177e4SLinus Torvalds } 60*1da177e4SLinus Torvalds 61*1da177e4SLinus Torvalds return NULL; 62*1da177e4SLinus Torvalds } 63*1da177e4SLinus Torvalds 64*1da177e4SLinus Torvalds static const struct unwind_table_entry * 65*1da177e4SLinus Torvalds find_unwind_entry(unsigned long addr) 66*1da177e4SLinus Torvalds { 67*1da177e4SLinus Torvalds struct unwind_table *table; 68*1da177e4SLinus Torvalds const struct unwind_table_entry *e = NULL; 69*1da177e4SLinus Torvalds 70*1da177e4SLinus Torvalds if (addr >= kernel_unwind_table.start && 71*1da177e4SLinus Torvalds addr <= kernel_unwind_table.end) 72*1da177e4SLinus Torvalds e = find_unwind_entry_in_table(&kernel_unwind_table, addr); 73*1da177e4SLinus Torvalds else 74*1da177e4SLinus Torvalds list_for_each_entry(table, &unwind_tables, list) { 75*1da177e4SLinus Torvalds if (addr >= table->start && 76*1da177e4SLinus Torvalds addr <= table->end) 77*1da177e4SLinus Torvalds e = find_unwind_entry_in_table(table, addr); 78*1da177e4SLinus Torvalds if (e) 79*1da177e4SLinus Torvalds break; 80*1da177e4SLinus Torvalds } 81*1da177e4SLinus Torvalds 82*1da177e4SLinus Torvalds return e; 83*1da177e4SLinus Torvalds } 84*1da177e4SLinus Torvalds 85*1da177e4SLinus Torvalds static void 86*1da177e4SLinus Torvalds unwind_table_init(struct unwind_table *table, const char *name, 87*1da177e4SLinus Torvalds unsigned long base_addr, unsigned long gp, 88*1da177e4SLinus Torvalds void *table_start, void *table_end) 89*1da177e4SLinus Torvalds { 90*1da177e4SLinus Torvalds struct unwind_table_entry *start = table_start; 91*1da177e4SLinus Torvalds struct unwind_table_entry *end = 92*1da177e4SLinus Torvalds (struct unwind_table_entry *)table_end - 1; 93*1da177e4SLinus Torvalds 94*1da177e4SLinus Torvalds table->name = name; 95*1da177e4SLinus Torvalds table->base_addr = base_addr; 96*1da177e4SLinus Torvalds table->gp = gp; 97*1da177e4SLinus Torvalds table->start = base_addr + start->region_start; 98*1da177e4SLinus Torvalds table->end = base_addr + end->region_end; 99*1da177e4SLinus Torvalds table->table = (struct unwind_table_entry *)table_start; 100*1da177e4SLinus Torvalds table->length = end - start + 1; 101*1da177e4SLinus Torvalds INIT_LIST_HEAD(&table->list); 102*1da177e4SLinus Torvalds 103*1da177e4SLinus Torvalds for (; start <= end; start++) { 104*1da177e4SLinus Torvalds if (start < end && 105*1da177e4SLinus Torvalds start->region_end > (start+1)->region_start) { 106*1da177e4SLinus Torvalds printk("WARNING: Out of order unwind entry! %p and %p\n", start, start+1); 107*1da177e4SLinus Torvalds } 108*1da177e4SLinus Torvalds 109*1da177e4SLinus Torvalds start->region_start += base_addr; 110*1da177e4SLinus Torvalds start->region_end += base_addr; 111*1da177e4SLinus Torvalds } 112*1da177e4SLinus Torvalds } 113*1da177e4SLinus Torvalds 114*1da177e4SLinus Torvalds static void 115*1da177e4SLinus Torvalds unwind_table_sort(struct unwind_table_entry *start, 116*1da177e4SLinus Torvalds struct unwind_table_entry *finish) 117*1da177e4SLinus Torvalds { 118*1da177e4SLinus Torvalds struct unwind_table_entry el, *p, *q; 119*1da177e4SLinus Torvalds 120*1da177e4SLinus Torvalds for (p = start + 1; p < finish; ++p) { 121*1da177e4SLinus Torvalds if (p[0].region_start < p[-1].region_start) { 122*1da177e4SLinus Torvalds el = *p; 123*1da177e4SLinus Torvalds q = p; 124*1da177e4SLinus Torvalds do { 125*1da177e4SLinus Torvalds q[0] = q[-1]; 126*1da177e4SLinus Torvalds --q; 127*1da177e4SLinus Torvalds } while (q > start && 128*1da177e4SLinus Torvalds el.region_start < q[-1].region_start); 129*1da177e4SLinus Torvalds *q = el; 130*1da177e4SLinus Torvalds } 131*1da177e4SLinus Torvalds } 132*1da177e4SLinus Torvalds } 133*1da177e4SLinus Torvalds 134*1da177e4SLinus Torvalds struct unwind_table * 135*1da177e4SLinus Torvalds unwind_table_add(const char *name, unsigned long base_addr, 136*1da177e4SLinus Torvalds unsigned long gp, 137*1da177e4SLinus Torvalds void *start, void *end) 138*1da177e4SLinus Torvalds { 139*1da177e4SLinus Torvalds struct unwind_table *table; 140*1da177e4SLinus Torvalds unsigned long flags; 141*1da177e4SLinus Torvalds struct unwind_table_entry *s = (struct unwind_table_entry *)start; 142*1da177e4SLinus Torvalds struct unwind_table_entry *e = (struct unwind_table_entry *)end; 143*1da177e4SLinus Torvalds 144*1da177e4SLinus Torvalds unwind_table_sort(s, e); 145*1da177e4SLinus Torvalds 146*1da177e4SLinus Torvalds table = kmalloc(sizeof(struct unwind_table), GFP_USER); 147*1da177e4SLinus Torvalds if (table == NULL) 148*1da177e4SLinus Torvalds return NULL; 149*1da177e4SLinus Torvalds unwind_table_init(table, name, base_addr, gp, start, end); 150*1da177e4SLinus Torvalds spin_lock_irqsave(&unwind_lock, flags); 151*1da177e4SLinus Torvalds list_add_tail(&table->list, &unwind_tables); 152*1da177e4SLinus Torvalds spin_unlock_irqrestore(&unwind_lock, flags); 153*1da177e4SLinus Torvalds 154*1da177e4SLinus Torvalds return table; 155*1da177e4SLinus Torvalds } 156*1da177e4SLinus Torvalds 157*1da177e4SLinus Torvalds void unwind_table_remove(struct unwind_table *table) 158*1da177e4SLinus Torvalds { 159*1da177e4SLinus Torvalds unsigned long flags; 160*1da177e4SLinus Torvalds 161*1da177e4SLinus Torvalds spin_lock_irqsave(&unwind_lock, flags); 162*1da177e4SLinus Torvalds list_del(&table->list); 163*1da177e4SLinus Torvalds spin_unlock_irqrestore(&unwind_lock, flags); 164*1da177e4SLinus Torvalds 165*1da177e4SLinus Torvalds kfree(table); 166*1da177e4SLinus Torvalds } 167*1da177e4SLinus Torvalds 168*1da177e4SLinus Torvalds /* Called from setup_arch to import the kernel unwind info */ 169*1da177e4SLinus Torvalds static int unwind_init(void) 170*1da177e4SLinus Torvalds { 171*1da177e4SLinus Torvalds long start, stop; 172*1da177e4SLinus Torvalds register unsigned long gp __asm__ ("r27"); 173*1da177e4SLinus Torvalds 174*1da177e4SLinus Torvalds start = (long)&__start___unwind[0]; 175*1da177e4SLinus Torvalds stop = (long)&__stop___unwind[0]; 176*1da177e4SLinus Torvalds 177*1da177e4SLinus Torvalds spin_lock_init(&unwind_lock); 178*1da177e4SLinus Torvalds 179*1da177e4SLinus Torvalds printk("unwind_init: start = 0x%lx, end = 0x%lx, entries = %lu\n", 180*1da177e4SLinus Torvalds start, stop, 181*1da177e4SLinus Torvalds (stop - start) / sizeof(struct unwind_table_entry)); 182*1da177e4SLinus Torvalds 183*1da177e4SLinus Torvalds unwind_table_init(&kernel_unwind_table, "kernel", KERNEL_START, 184*1da177e4SLinus Torvalds gp, 185*1da177e4SLinus Torvalds &__start___unwind[0], &__stop___unwind[0]); 186*1da177e4SLinus Torvalds #if 0 187*1da177e4SLinus Torvalds { 188*1da177e4SLinus Torvalds int i; 189*1da177e4SLinus Torvalds for (i = 0; i < 10; i++) 190*1da177e4SLinus Torvalds { 191*1da177e4SLinus Torvalds printk("region 0x%x-0x%x\n", 192*1da177e4SLinus Torvalds __start___unwind[i].region_start, 193*1da177e4SLinus Torvalds __start___unwind[i].region_end); 194*1da177e4SLinus Torvalds } 195*1da177e4SLinus Torvalds } 196*1da177e4SLinus Torvalds #endif 197*1da177e4SLinus Torvalds return 0; 198*1da177e4SLinus Torvalds } 199*1da177e4SLinus Torvalds 200*1da177e4SLinus Torvalds static void unwind_frame_regs(struct unwind_frame_info *info) 201*1da177e4SLinus Torvalds { 202*1da177e4SLinus Torvalds const struct unwind_table_entry *e; 203*1da177e4SLinus Torvalds unsigned long npc; 204*1da177e4SLinus Torvalds unsigned int insn; 205*1da177e4SLinus Torvalds long frame_size = 0; 206*1da177e4SLinus Torvalds int looking_for_rp, rpoffset = 0; 207*1da177e4SLinus Torvalds 208*1da177e4SLinus Torvalds e = find_unwind_entry(info->ip); 209*1da177e4SLinus Torvalds if (e == NULL) { 210*1da177e4SLinus Torvalds unsigned long sp; 211*1da177e4SLinus Torvalds extern char _stext[], _etext[]; 212*1da177e4SLinus Torvalds 213*1da177e4SLinus Torvalds dbg("Cannot find unwind entry for 0x%lx; forced unwinding\n", info->ip); 214*1da177e4SLinus Torvalds 215*1da177e4SLinus Torvalds #ifdef CONFIG_KALLSYMS 216*1da177e4SLinus Torvalds /* Handle some frequent special cases.... */ 217*1da177e4SLinus Torvalds { 218*1da177e4SLinus Torvalds char symname[KSYM_NAME_LEN+1]; 219*1da177e4SLinus Torvalds char *modname; 220*1da177e4SLinus Torvalds unsigned long symsize, offset; 221*1da177e4SLinus Torvalds 222*1da177e4SLinus Torvalds kallsyms_lookup(info->ip, &symsize, &offset, 223*1da177e4SLinus Torvalds &modname, symname); 224*1da177e4SLinus Torvalds 225*1da177e4SLinus Torvalds dbg("info->ip = 0x%lx, name = %s\n", info->ip, symname); 226*1da177e4SLinus Torvalds 227*1da177e4SLinus Torvalds if (strcmp(symname, "_switch_to_ret") == 0) { 228*1da177e4SLinus Torvalds info->prev_sp = info->sp - CALLEE_SAVE_FRAME_SIZE; 229*1da177e4SLinus Torvalds info->prev_ip = *(unsigned long *)(info->prev_sp - RP_OFFSET); 230*1da177e4SLinus Torvalds dbg("_switch_to_ret @ %lx - setting " 231*1da177e4SLinus Torvalds "prev_sp=%lx prev_ip=%lx\n", 232*1da177e4SLinus Torvalds info->ip, info->prev_sp, 233*1da177e4SLinus Torvalds info->prev_ip); 234*1da177e4SLinus Torvalds return; 235*1da177e4SLinus Torvalds } else if (strcmp(symname, "ret_from_kernel_thread") == 0 || 236*1da177e4SLinus Torvalds strcmp(symname, "syscall_exit") == 0) { 237*1da177e4SLinus Torvalds info->prev_ip = info->prev_sp = 0; 238*1da177e4SLinus Torvalds return; 239*1da177e4SLinus Torvalds } 240*1da177e4SLinus Torvalds } 241*1da177e4SLinus Torvalds #endif 242*1da177e4SLinus Torvalds 243*1da177e4SLinus Torvalds /* Since we are doing the unwinding blind, we don't know if 244*1da177e4SLinus Torvalds we are adjusting the stack correctly or extracting the rp 245*1da177e4SLinus Torvalds correctly. The rp is checked to see if it belongs to the 246*1da177e4SLinus Torvalds kernel text section, if not we assume we don't have a 247*1da177e4SLinus Torvalds correct stack frame and we continue to unwind the stack. 248*1da177e4SLinus Torvalds This is not quite correct, and will fail for loadable 249*1da177e4SLinus Torvalds modules. */ 250*1da177e4SLinus Torvalds sp = info->sp & ~63; 251*1da177e4SLinus Torvalds do { 252*1da177e4SLinus Torvalds unsigned long tmp; 253*1da177e4SLinus Torvalds 254*1da177e4SLinus Torvalds info->prev_sp = sp - 64; 255*1da177e4SLinus Torvalds info->prev_ip = 0; 256*1da177e4SLinus Torvalds if (get_user(tmp, (unsigned long *)(info->prev_sp - RP_OFFSET))) 257*1da177e4SLinus Torvalds break; 258*1da177e4SLinus Torvalds info->prev_ip = tmp; 259*1da177e4SLinus Torvalds sp = info->prev_sp; 260*1da177e4SLinus Torvalds } while (info->prev_ip < (unsigned long)_stext || 261*1da177e4SLinus Torvalds info->prev_ip > (unsigned long)_etext); 262*1da177e4SLinus Torvalds 263*1da177e4SLinus Torvalds info->rp = 0; 264*1da177e4SLinus Torvalds 265*1da177e4SLinus Torvalds dbg("analyzing func @ %lx with no unwind info, setting " 266*1da177e4SLinus Torvalds "prev_sp=%lx prev_ip=%lx\n", info->ip, 267*1da177e4SLinus Torvalds info->prev_sp, info->prev_ip); 268*1da177e4SLinus Torvalds } else { 269*1da177e4SLinus Torvalds dbg("e->start = 0x%x, e->end = 0x%x, Save_SP = %d, " 270*1da177e4SLinus Torvalds "Save_RP = %d, Millicode = %d size = %u\n", 271*1da177e4SLinus Torvalds e->region_start, e->region_end, e->Save_SP, e->Save_RP, 272*1da177e4SLinus Torvalds e->Millicode, e->Total_frame_size); 273*1da177e4SLinus Torvalds 274*1da177e4SLinus Torvalds looking_for_rp = e->Save_RP; 275*1da177e4SLinus Torvalds 276*1da177e4SLinus Torvalds for (npc = e->region_start; 277*1da177e4SLinus Torvalds (frame_size < (e->Total_frame_size << 3) || 278*1da177e4SLinus Torvalds looking_for_rp) && 279*1da177e4SLinus Torvalds npc < info->ip; 280*1da177e4SLinus Torvalds npc += 4) { 281*1da177e4SLinus Torvalds 282*1da177e4SLinus Torvalds insn = *(unsigned int *)npc; 283*1da177e4SLinus Torvalds 284*1da177e4SLinus Torvalds if ((insn & 0xffffc000) == 0x37de0000 || 285*1da177e4SLinus Torvalds (insn & 0xffe00000) == 0x6fc00000) { 286*1da177e4SLinus Torvalds /* ldo X(sp), sp, or stwm X,D(sp) */ 287*1da177e4SLinus Torvalds frame_size += (insn & 0x1 ? -1 << 13 : 0) | 288*1da177e4SLinus Torvalds ((insn & 0x3fff) >> 1); 289*1da177e4SLinus Torvalds dbg("analyzing func @ %lx, insn=%08x @ " 290*1da177e4SLinus Torvalds "%lx, frame_size = %ld\n", info->ip, 291*1da177e4SLinus Torvalds insn, npc, frame_size); 292*1da177e4SLinus Torvalds } else if ((insn & 0xffe00008) == 0x73c00008) { 293*1da177e4SLinus Torvalds /* std,ma X,D(sp) */ 294*1da177e4SLinus Torvalds frame_size += (insn & 0x1 ? -1 << 13 : 0) | 295*1da177e4SLinus Torvalds (((insn >> 4) & 0x3ff) << 3); 296*1da177e4SLinus Torvalds dbg("analyzing func @ %lx, insn=%08x @ " 297*1da177e4SLinus Torvalds "%lx, frame_size = %ld\n", info->ip, 298*1da177e4SLinus Torvalds insn, npc, frame_size); 299*1da177e4SLinus Torvalds } else if (insn == 0x6bc23fd9) { 300*1da177e4SLinus Torvalds /* stw rp,-20(sp) */ 301*1da177e4SLinus Torvalds rpoffset = 20; 302*1da177e4SLinus Torvalds looking_for_rp = 0; 303*1da177e4SLinus Torvalds dbg("analyzing func @ %lx, insn=stw rp," 304*1da177e4SLinus Torvalds "-20(sp) @ %lx\n", info->ip, npc); 305*1da177e4SLinus Torvalds } else if (insn == 0x0fc212c1) { 306*1da177e4SLinus Torvalds /* std rp,-16(sr0,sp) */ 307*1da177e4SLinus Torvalds rpoffset = 16; 308*1da177e4SLinus Torvalds looking_for_rp = 0; 309*1da177e4SLinus Torvalds dbg("analyzing func @ %lx, insn=std rp," 310*1da177e4SLinus Torvalds "-16(sp) @ %lx\n", info->ip, npc); 311*1da177e4SLinus Torvalds } 312*1da177e4SLinus Torvalds } 313*1da177e4SLinus Torvalds 314*1da177e4SLinus Torvalds info->prev_sp = info->sp - frame_size; 315*1da177e4SLinus Torvalds if (e->Millicode) 316*1da177e4SLinus Torvalds info->rp = info->r31; 317*1da177e4SLinus Torvalds else if (rpoffset) 318*1da177e4SLinus Torvalds info->rp = *(unsigned long *)(info->prev_sp - rpoffset); 319*1da177e4SLinus Torvalds info->prev_ip = info->rp; 320*1da177e4SLinus Torvalds info->rp = 0; 321*1da177e4SLinus Torvalds 322*1da177e4SLinus Torvalds dbg("analyzing func @ %lx, setting prev_sp=%lx " 323*1da177e4SLinus Torvalds "prev_ip=%lx npc=%lx\n", info->ip, info->prev_sp, 324*1da177e4SLinus Torvalds info->prev_ip, npc); 325*1da177e4SLinus Torvalds } 326*1da177e4SLinus Torvalds } 327*1da177e4SLinus Torvalds 328*1da177e4SLinus Torvalds void unwind_frame_init(struct unwind_frame_info *info, struct task_struct *t, 329*1da177e4SLinus Torvalds struct pt_regs *regs) 330*1da177e4SLinus Torvalds { 331*1da177e4SLinus Torvalds memset(info, 0, sizeof(struct unwind_frame_info)); 332*1da177e4SLinus Torvalds info->t = t; 333*1da177e4SLinus Torvalds info->sp = regs->gr[30]; 334*1da177e4SLinus Torvalds info->ip = regs->iaoq[0]; 335*1da177e4SLinus Torvalds info->rp = regs->gr[2]; 336*1da177e4SLinus Torvalds info->r31 = regs->gr[31]; 337*1da177e4SLinus Torvalds 338*1da177e4SLinus Torvalds dbg("(%d) Start unwind from sp=%08lx ip=%08lx\n", 339*1da177e4SLinus Torvalds t ? (int)t->pid : -1, info->sp, info->ip); 340*1da177e4SLinus Torvalds } 341*1da177e4SLinus Torvalds 342*1da177e4SLinus Torvalds void unwind_frame_init_from_blocked_task(struct unwind_frame_info *info, struct task_struct *t) 343*1da177e4SLinus Torvalds { 344*1da177e4SLinus Torvalds struct pt_regs *r = &t->thread.regs; 345*1da177e4SLinus Torvalds struct pt_regs *r2; 346*1da177e4SLinus Torvalds 347*1da177e4SLinus Torvalds r2 = (struct pt_regs *)kmalloc(sizeof(struct pt_regs), GFP_KERNEL); 348*1da177e4SLinus Torvalds if (!r2) 349*1da177e4SLinus Torvalds return; 350*1da177e4SLinus Torvalds *r2 = *r; 351*1da177e4SLinus Torvalds r2->gr[30] = r->ksp; 352*1da177e4SLinus Torvalds r2->iaoq[0] = r->kpc; 353*1da177e4SLinus Torvalds unwind_frame_init(info, t, r2); 354*1da177e4SLinus Torvalds kfree(r2); 355*1da177e4SLinus Torvalds } 356*1da177e4SLinus Torvalds 357*1da177e4SLinus Torvalds void unwind_frame_init_running(struct unwind_frame_info *info, struct pt_regs *regs) 358*1da177e4SLinus Torvalds { 359*1da177e4SLinus Torvalds unwind_frame_init(info, current, regs); 360*1da177e4SLinus Torvalds } 361*1da177e4SLinus Torvalds 362*1da177e4SLinus Torvalds int unwind_once(struct unwind_frame_info *next_frame) 363*1da177e4SLinus Torvalds { 364*1da177e4SLinus Torvalds unwind_frame_regs(next_frame); 365*1da177e4SLinus Torvalds 366*1da177e4SLinus Torvalds if (next_frame->prev_sp == 0 || 367*1da177e4SLinus Torvalds next_frame->prev_ip == 0) 368*1da177e4SLinus Torvalds return -1; 369*1da177e4SLinus Torvalds 370*1da177e4SLinus Torvalds next_frame->sp = next_frame->prev_sp; 371*1da177e4SLinus Torvalds next_frame->ip = next_frame->prev_ip; 372*1da177e4SLinus Torvalds next_frame->prev_sp = 0; 373*1da177e4SLinus Torvalds next_frame->prev_ip = 0; 374*1da177e4SLinus Torvalds 375*1da177e4SLinus Torvalds dbg("(%d) Continue unwind to sp=%08lx ip=%08lx\n", 376*1da177e4SLinus Torvalds next_frame->t ? (int)next_frame->t->pid : -1, 377*1da177e4SLinus Torvalds next_frame->sp, next_frame->ip); 378*1da177e4SLinus Torvalds 379*1da177e4SLinus Torvalds return 0; 380*1da177e4SLinus Torvalds } 381*1da177e4SLinus Torvalds 382*1da177e4SLinus Torvalds int unwind_to_user(struct unwind_frame_info *info) 383*1da177e4SLinus Torvalds { 384*1da177e4SLinus Torvalds int ret; 385*1da177e4SLinus Torvalds 386*1da177e4SLinus Torvalds do { 387*1da177e4SLinus Torvalds ret = unwind_once(info); 388*1da177e4SLinus Torvalds } while (!ret && !(info->ip & 3)); 389*1da177e4SLinus Torvalds 390*1da177e4SLinus Torvalds return ret; 391*1da177e4SLinus Torvalds } 392*1da177e4SLinus Torvalds 393*1da177e4SLinus Torvalds module_init(unwind_init); 394