xref: /haiku/src/system/kernel/arch/m68k/arch_debug.cpp (revision 7749d0bb0c358a3279b1b9cc76d8376e900130a5)
1 /*
2  * Copyright 2003-2011, Haiku, Inc. All rights reserved.
3  * Distributed under the terms of the MIT License.
4  *
5  * Authors:
6  * 		Axel Dörfler <axeld@pinc-software.de>
7  * 		Ingo Weinhold <bonefish@cs.tu-berlin.de>
8  * 		François Revol <revol@free.fr>
9  */
10 
11 
12 #include <arch/debug.h>
13 
14 #include <arch_cpu.h>
15 #include <debug.h>
16 #include <elf.h>
17 #include <kernel.h>
18 #include <kimage.h>
19 #include <thread.h>
20 
21 
22 struct stack_frame {
23 	struct stack_frame	*previous;
24 	addr_t				return_address;
25 };
26 
27 #define NUM_PREVIOUS_LOCATIONS 32
28 
29 extern struct iframe_stack gBootFrameStack;
30 
31 
32 static bool
33 already_visited(uint32 *visited, int32 *_last, int32 *_num, uint32 framePointer)
34 {
35 	int32 last = *_last;
36 	int32 num = *_num;
37 	int32 i;
38 
39 	for (i = 0; i < num; i++) {
40 		if (visited[(NUM_PREVIOUS_LOCATIONS + last - i)
41 				% NUM_PREVIOUS_LOCATIONS] == framePointer) {
42 			return true;
43 		}
44 	}
45 
46 	*_last = last = (last + 1) % NUM_PREVIOUS_LOCATIONS;
47 	visited[last] = framePointer;
48 
49 	if (num < NUM_PREVIOUS_LOCATIONS)
50 		*_num = num + 1;
51 
52 	return false;
53 }
54 
55 
56 static inline stack_frame *
57 get_current_stack_frame()
58 {
59 	stack_frame *frame;
60 	asm volatile("move.l %%fp,%0" : "=r"(frame));
61 	return frame;
62 }
63 
64 
65 static status_t
66 get_next_frame(addr_t framePointer, addr_t *next, addr_t *ip)
67 {
68 	stack_frame frame;
69 	if (debug_memcpy(B_CURRENT_TEAM, &frame, (void*)framePointer, sizeof(frame))
70 			!= B_OK) {
71 		return B_BAD_ADDRESS;
72 	}
73 
74 	*ip = frame.return_address;
75 	*next = (addr_t)frame.previous;
76 
77 	return B_OK;
78 }
79 
80 
81 static void
82 print_stack_frame(Thread *thread, addr_t ip, addr_t framePointer,
83 	addr_t nextFramePointer)
84 {
85 	addr_t diff = nextFramePointer - framePointer;
86 
87 	// kernel space/user space switch
88 	if (diff & 0x80000000)
89 		diff = 0;
90 
91 	// lookup symbol
92 	const char *symbol, *image;
93 	addr_t baseAddress;
94 	bool exactMatch;
95 	status_t status = elf_debug_lookup_symbol_address(ip, &baseAddress, &symbol,
96 		&image, &exactMatch);
97 	if (status != B_OK && !IS_KERNEL_ADDRESS(ip) && thread) {
98 		// try to locate the image in the images loaded into user space
99 		status = image_debug_lookup_user_symbol_address(thread->team, ip,
100 			&baseAddress, &symbol, &image, &exactMatch);
101 	}
102 	if (status == B_OK) {
103 		if (symbol != NULL) {
104 			kprintf("%08lx (+%4ld) %08lx   <%s>:%s + 0x%04lx%s\n", framePointer,
105 				diff, ip, image, symbol, ip - baseAddress,
106 				(exactMatch ? "" : " (nearest)"));
107 		} else {
108 			kprintf("%08lx (+%4ld) %08lx   <%s@%p>:unknown + 0x%04lx\n",
109 				framePointer, diff, ip, image, (void *)baseAddress,
110 				ip - baseAddress);
111 		}
112 	} else
113 		kprintf("%08lx (+%4ld) %08lx\n", framePointer, diff, ip);
114 }
115 
116 
117 static int
118 stack_trace(int argc, char **argv)
119 {
120 	uint32 previousLocations[NUM_PREVIOUS_LOCATIONS];
121 	struct iframe_stack *frameStack;
122 	Thread *thread;
123 	addr_t framePointer;
124 	int32 i, num = 0, last = 0;
125 
126 	if (argc < 2) {
127 		thread = thread_get_current_thread();
128 		framePointer = (addr_t)get_current_stack_frame();
129 	} else {
130 // TODO: Add support for stack traces of other threads.
131 /*		thread_id id = strtoul(argv[1], NULL, 0);
132 		thread = Thread::GetDebug(id);
133 		if (thread == NULL) {
134 			kprintf("could not find thread %ld\n", id);
135 			return 0;
136 		}
137 
138 		// read %ebp from the thread's stack stored by a pushad
139 		ebp = thread->arch_info.current_stack.esp[2];
140 
141 		if (id != thread_get_current_thread_id()) {
142 			// switch to the page directory of the new thread to be
143 			// able to follow the stack trace into userland
144 			addr_t newPageDirectory = (addr_t)x86_next_page_directory(
145 				thread_get_current_thread(), thread);
146 
147 			if (newPageDirectory != 0) {
148 				read_cr3(oldPageDirectory);
149 				write_cr3(newPageDirectory);
150 			}
151 		}
152 */
153 kprintf("Stack traces of other threads not supported yet!\n");
154 return 0;
155 	}
156 
157 	// We don't have a thread pointer early in the boot process
158 	if (thread != NULL)
159 		frameStack = &thread->arch_info.iframes;
160 	else
161 		frameStack = &gBootFrameStack;
162 
163 	for (i = 0; i < frameStack->index; i++) {
164 		kprintf("iframe %p (end = %p)\n",
165 			frameStack->frames[i], frameStack->frames[i] + 1);
166 	}
167 
168 	if (thread != NULL) {
169 		kprintf("stack trace for thread 0x%lx \"%s\"\n", thread->id,
170 			thread->name);
171 
172 		kprintf("    kernel stack: %p to %p\n",
173 			(void *)thread->kernel_stack_base,
174 			(void *)(thread->kernel_stack_top));
175 		if (thread->user_stack_base != 0) {
176 			kprintf("      user stack: %p to %p\n",
177 				(void *)thread->user_stack_base,
178 				(void *)(thread->user_stack_base + thread->user_stack_size));
179 		}
180 	}
181 
182 	kprintf("frame            caller     <image>:function + offset\n");
183 
184 	for (;;) {
185 		// see if the frame pointer matches the iframe
186 		struct iframe *frame = NULL;
187 		for (i = 0; i < frameStack->index; i++) {
188 			if (framePointer == (addr_t)frameStack->frames[i]) {
189 				// it's an iframe
190 				frame = frameStack->frames[i];
191 				break;
192 			}
193 		}
194 
195 		if (frame) {
196 			kprintf("iframe at %p\n", frame);
197 			kprintf("   d0 0x%08lx    d1 0x%08lx    d2 0x%08lx    d3 0x%08lx\n",
198 				frame->d[0], frame->d[1], frame->d[2], frame->d[3]);
199 			kprintf("   d4 0x%08lx    d5 0x%08lx    d6 0x%08lx    d7 0x%08lx\n",
200 				frame->d[4], frame->d[5], frame->d[6], frame->d[7]);
201 			kprintf("   a0 0x%08lx    a1 0x%08lx    a2 0x%08lx    a3 0x%08lx\n",
202 				frame->a[0], frame->a[1], frame->a[2], frame->a[3]);
203 			kprintf("   a4 0x%08lx    a5 0x%08lx    a6 0x%08lx    a7 0x%08lx (sp)\n",
204 #warning M68K: a7 in iframe ??
205 				frame->a[4], frame->a[5], frame->a[6], -1L/*frame->a[7]*/);
206 
207 			/*kprintf("   pc 0x%08lx   ccr 0x%02x\n",
208 			  frame->pc, frame->ccr);*/
209 			kprintf("   pc 0x%08lx        sr 0x%04x\n",
210 				frame->cpu.pc, frame->cpu.sr);
211 #warning M68K: missing regs
212 
213 			print_stack_frame(thread, frame->cpu.pc, framePointer, frame->a[6]);
214  			framePointer = frame->a[6];
215 		} else {
216 			addr_t ip, nextFramePointer;
217 
218 			if (get_next_frame(framePointer, &nextFramePointer, &ip) != B_OK) {
219 				kprintf("%08lx -- read fault\n", framePointer);
220 				break;
221 			}
222 
223 			if (ip == 0 || framePointer == 0)
224 				break;
225 
226 			print_stack_frame(thread, ip, framePointer, nextFramePointer);
227 			framePointer = nextFramePointer;
228 		}
229 
230 		if (already_visited(previousLocations, &last, &num, framePointer)) {
231 			kprintf("circular stack frame: %p!\n", (void *)framePointer);
232 			break;
233 		}
234 		if (framePointer == 0)
235 			break;
236 	}
237 
238 /*	if (oldPageDirectory != 0) {
239 		// switch back to the previous page directory to no cause any troubles
240 		write_cr3(oldPageDirectory);
241 	}
242 */
243 
244 	return 0;
245 }
246 
247 
248 
249 // #pragma mark -
250 
251 
252 void
253 arch_debug_save_registers(struct arch_debug_registers* registers)
254 {
255 }
256 
257 
258 void
259 arch_debug_stack_trace(void)
260 {
261 }
262 
263 
264 bool
265 arch_debug_contains_call(Thread *thread, const char *symbol,
266 	addr_t start, addr_t end)
267 {
268 	return false;
269 }
270 
271 
272 void *
273 arch_debug_get_caller(void)
274 {
275 	// TODO: implement me
276 	//return __builtin_frame_address(1);
277 	struct stack_frame *frame;
278 	//frame = __builtin_frame_address(0);
279 	frame = get_current_stack_frame();
280 	return (void *)frame->previous->return_address;
281 }
282 
283 
284 int32
285 arch_debug_get_stack_trace(addr_t* returnAddresses, int32 maxCount,
286 	int32 skipIframes, int32 skipFrames, uint32 flags)
287 {
288 	struct iframe_stack *frameStack;
289 	addr_t framePointer;
290 	int32 count = 0;
291 	int32 i, num = 0, last = 0;
292 
293 	// Keep skipping normal stack frames until we've skipped the iframes we're
294 	// supposed to skip.
295 	if (skipIframes > 0)
296 		skipFrames = INT_MAX;
297 
298 	Thread* thread = thread_get_current_thread();
299 	framePointer = (addr_t)get_current_stack_frame();
300 	bool onKernelStack = true;
301 
302 	// We don't have a thread pointer early in the boot process
303 	if (thread != NULL)
304 		frameStack = &thread->arch_info.iframes;
305 	else
306 		frameStack = &gBootFrameStack;
307 
308 	while (framePointer != 0 && count < maxCount) {
309 		onKernelStack = onKernelStack && IS_KERNEL_ADDRESS(framePointer);
310 			// TODO: Correctly determine whether this is a kernel address!
311 		if (!onKernelStack && (flags & STACK_TRACE_USER) == 0)
312 			break;
313 
314 		// see if the frame pointer matches the iframe
315 		struct iframe *frame = NULL;
316 		for (i = 0; i < frameStack->index; i++) {
317 			if (framePointer == (addr_t)frameStack->frames[i]) {
318 				// it's an iframe
319 				frame = frameStack->frames[i];
320 				break;
321 			}
322 		}
323 
324 		addr_t ip;
325 		addr_t nextFrame;
326 
327 		if (frame) {
328 			ip = frame->cpu.pc;
329  			nextFrame = frame->a[6];
330 
331 			if (skipIframes > 0) {
332 				if (--skipIframes == 0)
333 					skipFrames = 0;
334 			}
335 		} else {
336 			if (get_next_frame(framePointer, &nextFrame, &ip) != B_OK)
337 				break;
338 		}
339 
340 		if (skipFrames <= 0
341 			&& ((flags & STACK_TRACE_KERNEL) != 0 || onKernelStack)) {
342 			returnAddresses[count++] = ip;
343 		} else
344 			skipFrames--;
345 
346 		framePointer = nextFrame;
347 	}
348 
349 	return count;
350 }
351 
352 
353 void*
354 arch_debug_get_interrupt_pc(bool* _isSyscall)
355 {
356 	// TODO: Implement!
357 	return NULL;
358 }
359 
360 
361 void
362 arch_debug_unset_current_thread(void)
363 {
364 	// TODO: Implement!
365 }
366 
367 
368 void
369 arch_debug_call_with_fault_handler(cpu_ent* cpu, jmp_buf jumpBuffer,
370 	void (*function)(void*), void* parameter)
371 {
372 	// TODO: Implement! Most likely in assembly.
373 	longjmp(jumpBuffer, 1);
374 }
375 
376 
377 bool
378 arch_is_debug_variable_defined(const char* variableName)
379 {
380 	// TODO: Implement!
381 	return false;
382 }
383 
384 
385 status_t
386 arch_set_debug_variable(const char* variableName, uint64 value)
387 {
388 	// TODO: Implement!
389 	return B_ENTRY_NOT_FOUND;
390 }
391 
392 
393 status_t
394 arch_get_debug_variable(const char* variableName, uint64* value)
395 {
396 	// TODO: Implement!
397 	return B_ENTRY_NOT_FOUND;
398 }
399 
400 
401 ssize_t
402 arch_debug_gdb_get_registers(char* buffer, size_t bufferSize)
403 {
404 	// TODO: Implement!
405 	return B_NOT_SUPPORTED;
406 }
407 
408 
409 status_t
410 arch_debug_init(kernel_args *args)
411 {
412 	add_debugger_command("where", &stack_trace, "Same as \"sc\"");
413 	add_debugger_command("bt", &stack_trace, "Same as \"sc\" (as in gdb)");
414 	add_debugger_command("sc", &stack_trace, "Stack crawl for current thread");
415 
416 	return B_NO_ERROR;
417 }
418 
419