2 * Debugger stack handling
4 * Copyright 1995 Alexandre Julliard
5 * Copyright 1996 Eric Youngdale
6 * Copyright 1999 Ove Kåven
13 #include "stackframe.h"
17 * We keep this info for each frame, so that we can
18 * find local variable information correctly.
26 struct symbol_info frame;
30 static struct bt_info * frames = NULL;
49 /***********************************************************************
52 * Dump the top of the stack
54 void DEBUG_InfoStack(void)
59 addr.seg = SS_reg(&DEBUG_context);
60 addr.off = ESP_reg(&DEBUG_context);
62 fprintf(stderr,"Stack dump:\n");
63 if (IS_SELECTOR_32BIT(addr.seg))
65 DEBUG_ExamineMemory( &addr, 24, 'x' );
67 else /* 16-bit mode */
70 DEBUG_ExamineMemory( &addr, 24, 'w' );
76 static void DEBUG_ForceFrame(DBG_ADDR *stack, DBG_ADDR *code, int frameno, int bits, int noisy)
78 int theframe = nframe++;
79 frames = (struct bt_info *)DBG_realloc(frames,
80 nframe*sizeof(struct bt_info));
82 fprintf(stderr,"%s%d ", (theframe == curr_frame ? "=>" : " "),
84 frames[theframe].cs = code->seg;
85 frames[theframe].eip = code->off;
87 frames[theframe].frame = DEBUG_PrintAddressAndArgs( code, bits,
90 DEBUG_FindNearestSymbol( code, TRUE,
91 &frames[theframe].frame.sym, stack->off,
92 &frames[theframe].frame.list);
93 frames[theframe].ss = stack->seg;
94 frames[theframe].ebp = stack->off;
96 fprintf( stderr, (bits == 16) ? " (bp=%04lx)\n" : " (ebp=%08lx)\n", stack->off );
100 static BOOL DEBUG_Frame16(DBG_ADDR *addr, unsigned int *cs, int frameno, int noisy)
102 unsigned int ss = addr->seg, possible_cs = 0;
103 FRAME16 *frame = (FRAME16 *)DBG_ADDR_TO_LIN(addr);
104 int theframe = nframe;
106 if (!DBG_CHECK_READ_PTR( addr, sizeof(FRAME16) )) return FALSE;
107 if (!frame->bp) return FALSE;
109 frames = (struct bt_info *)DBG_realloc(frames,
110 nframe*sizeof(struct bt_info));
112 fprintf(stderr,"%s%d ", (theframe == curr_frame ? "=>" : " "),
114 if (frame->bp & 1) *cs = frame->cs;
116 /* not explicitly marked as far call,
117 * but check whether it could be anyway */
118 if (((frame->cs&7)==7) && (frame->cs != *cs) && !IS_SELECTOR_FREE(frame->cs)) {
120 LDT_GetEntry( SELECTOR_TO_ENTRY(frame->cs), &tcs );
121 if ( tcs.type == SEGMENT_CODE ) {
122 /* it is very uncommon to push a code segment cs as
123 * a parameter, so this should work in most cases */
124 *cs = possible_cs = frame->cs;
128 frames[theframe].cs = addr->seg = *cs;
129 frames[theframe].eip = addr->off = frame->ip;
131 frames[theframe].frame = DEBUG_PrintAddressAndArgs( addr, 16,
134 DEBUG_FindNearestSymbol( addr, TRUE,
135 &frames[theframe].frame.sym, frame->bp,
136 &frames[theframe].frame.list);
137 frames[theframe].ss = addr->seg = ss;
138 frames[theframe].ebp = addr->off = frame->bp & ~1;
140 fprintf( stderr, " (bp=%04lx", addr->off );
142 fprintf( stderr, ", far call assumed" );
144 fprintf( stderr, ")\n" );
149 static BOOL DEBUG_Frame32(DBG_ADDR *addr, unsigned int *cs, int frameno, int noisy)
151 unsigned int ss = addr->seg;
152 FRAME32 *frame = (FRAME32 *)DBG_ADDR_TO_LIN(addr);
153 int theframe = nframe;
155 if (!DBG_CHECK_READ_PTR( addr, sizeof(FRAME32) )) return FALSE;
156 if (!frame->ip) return FALSE;
158 frames = (struct bt_info *)DBG_realloc(frames,
159 nframe*sizeof(struct bt_info));
161 fprintf(stderr,"%s%d ", (theframe == curr_frame ? "=>" : " "),
163 frames[theframe].cs = addr->seg = *cs;
164 frames[theframe].eip = addr->off = frame->ip;
166 frames[theframe].frame = DEBUG_PrintAddressAndArgs( addr, 32,
169 DEBUG_FindNearestSymbol( addr, TRUE,
170 &frames[theframe].frame.sym, frame->bp,
171 &frames[theframe].frame.list);
172 if (noisy) fprintf( stderr, " (ebp=%08lx)\n", frame->bp );
173 frames[theframe].ss = addr->seg = ss;
174 frames[theframe].ebp = frame->bp;
175 if (addr->off == frame->bp) return FALSE;
176 addr->off = frame->bp;
180 static void DEBUG_DoBackTrace(int noisy)
182 DBG_ADDR addr, sw_addr;
183 unsigned int ss = SS_reg(&DEBUG_context), cs = CS_reg(&DEBUG_context);
184 int frameno = 0, is16, ok;
185 DWORD next_switch, cur_switch;
187 if (noisy) fprintf( stderr, "Backtrace:\n" );
190 if (frames) DBG_free( frames );
191 frames = (struct bt_info *) DBG_alloc( sizeof(struct bt_info) );
193 fprintf(stderr,"%s%d ",(curr_frame == 0 ? "=>" : " "), frameno);
195 if (IS_SELECTOR_SYSTEM(ss)) ss = 0;
196 if (IS_SELECTOR_SYSTEM(cs)) cs = 0;
198 /* first stack frame from registers */
199 if (IS_SELECTOR_32BIT(ss))
201 frames[0].cs = addr.seg = cs;
202 frames[0].eip = addr.off = EIP_reg(&DEBUG_context);
204 frames[0].frame = DEBUG_PrintAddress( &addr, 32, TRUE );
206 DEBUG_FindNearestSymbol( &addr, TRUE, &frames[0].frame.sym, 0,
207 &frames[0].frame.list);
208 frames[0].ss = addr.seg = ss;
209 frames[0].ebp = addr.off = EBP_reg(&DEBUG_context);
210 if (noisy) fprintf( stderr, " (ebp=%08x)\n", frames[0].ebp );
213 frames[0].cs = addr.seg = cs;
214 frames[0].eip = addr.off = IP_reg(&DEBUG_context);
216 frames[0].frame = DEBUG_PrintAddress( &addr, 16, TRUE );
218 DEBUG_FindNearestSymbol( &addr, TRUE, &frames[0].frame.sym, 0,
219 &frames[0].frame.list);
220 frames[0].ss = addr.seg = ss;
221 frames[0].ebp = addr.off = BP_reg(&DEBUG_context);
222 if (noisy) fprintf( stderr, " (bp=%04x)\n", frames[0].ebp );
226 next_switch = THREAD_Current()->cur_stack;
228 if (IsBadReadPtr((STACK32FRAME*)next_switch, sizeof(STACK32FRAME))) {
229 fprintf( stderr, "Bad stack frame %p\n", (STACK32FRAME*)next_switch );
232 cur_switch = (DWORD)((STACK32FRAME*)next_switch)->frame16;
233 sw_addr.seg = SELECTOROF(cur_switch);
234 sw_addr.off = OFFSETOF(cur_switch);
236 if (IsBadReadPtr((STACK16FRAME*)PTR_SEG_TO_LIN(next_switch), sizeof(STACK16FRAME))) {
237 fprintf( stderr, "Bad stack frame %p\n", (STACK16FRAME*)PTR_SEG_TO_LIN(next_switch) );
240 cur_switch = (DWORD)((STACK16FRAME*)PTR_SEG_TO_LIN(next_switch))->frame32;
242 sw_addr.off = cur_switch;
244 if (DEBUG_IsBadReadPtr(&sw_addr,1)) {
245 sw_addr.seg = (DWORD)-1;
246 sw_addr.off = (DWORD)-1;
249 for (ok = TRUE; ok;) {
250 if ((frames[frameno].ss == sw_addr.seg) &&
251 (frames[frameno].ebp >= sw_addr.off)) {
253 * yes, I know this is confusing, it gave me a headache too */
255 STACK32FRAME *frame = (STACK32FRAME*)next_switch;
258 if (IsBadReadPtr((STACK32FRAME*)next_switch, sizeof(STACK32FRAME))) {
259 fprintf( stderr, "Bad stack frame %p\n", (STACK32FRAME*)next_switch );
264 code.off = frame->retaddr;
268 addr.off = frame->ebp;
269 DEBUG_ForceFrame( &addr, &code, ++frameno, 32, noisy );
271 next_switch = cur_switch;
272 if (IsBadReadPtr((STACK16FRAME*)PTR_SEG_TO_LIN(next_switch), sizeof(STACK16FRAME))) {
273 fprintf( stderr, "Bad stack frame %p\n", (STACK16FRAME*)PTR_SEG_TO_LIN(next_switch) );
276 cur_switch = (DWORD)((STACK16FRAME*)PTR_SEG_TO_LIN(next_switch))->frame32;
278 sw_addr.off = cur_switch;
282 STACK16FRAME *frame = (STACK16FRAME*)PTR_SEG_TO_LIN(next_switch);
285 if (IsBadReadPtr((STACK16FRAME*)PTR_SEG_TO_LIN(next_switch), sizeof(STACK16FRAME))) {
286 fprintf( stderr, "Bad stack frame %p\n", (STACK16FRAME*)PTR_SEG_TO_LIN(next_switch) );
291 code.seg = frame->cs;
292 code.off = frame->ip;
295 addr.seg = SELECTOROF(next_switch);
296 addr.off = frame->bp;
297 DEBUG_ForceFrame( &addr, &code, ++frameno, 16, noisy );
299 next_switch = cur_switch;
300 if (IsBadReadPtr((STACK32FRAME*)next_switch, sizeof(STACK32FRAME))) {
301 fprintf( stderr, "Bad stack frame %p\n", (STACK32FRAME*)next_switch );
304 cur_switch = (DWORD)((STACK32FRAME*)next_switch)->frame16;
305 sw_addr.seg = SELECTOROF(cur_switch);
306 sw_addr.off = OFFSETOF(cur_switch);
310 if (DEBUG_IsBadReadPtr(&sw_addr,1)) {
311 sw_addr.seg = (DWORD)-1;
312 sw_addr.off = (DWORD)-1;
315 /* ordinary stack frame */
316 ok = is16 ? DEBUG_Frame16( &addr, &cs, ++frameno, noisy)
317 : DEBUG_Frame32( &addr, &cs, ++frameno, noisy);
320 if (noisy) fprintf( stderr, "\n" );
323 /***********************************************************************
326 * Display a stack back-trace.
328 void DEBUG_BackTrace(void)
330 DEBUG_DoBackTrace( TRUE );
333 /***********************************************************************
334 * DEBUG_SilentBackTrace
336 * Display a stack back-trace.
338 void DEBUG_SilentBackTrace(void)
340 DEBUG_DoBackTrace( FALSE );
344 DEBUG_SetFrame(int newframe)
348 curr_frame = newframe;
350 if( curr_frame >= nframe )
352 curr_frame = nframe - 1;
360 if( frames && frames[curr_frame].frame.list.sourcefile != NULL )
362 DEBUG_List(&frames[curr_frame].frame.list, NULL, 0);
370 DEBUG_GetCurrentFrame(struct name_hash ** name, unsigned int * eip,
374 * If we don't have a valid backtrace, then just return.
382 * If we don't know what the current function is, then we also have
383 * nothing to report here.
385 if( frames[curr_frame].frame.sym == NULL )
390 *name = frames[curr_frame].frame.sym;
391 *eip = frames[curr_frame].eip;
392 *ebp = frames[curr_frame].ebp;