xref: /dflybsd-src/sys/platform/pc64/x86_64/db_interface.c (revision a798ebf2db231775b705e7753a58f3fd53227ce1)
1b2b3ffcdSSimon Schubert /*
2b2b3ffcdSSimon Schubert  * Copyright (c) 2008 The DragonFly Project.  All rights reserved.
3b2b3ffcdSSimon Schubert  *
4b2b3ffcdSSimon Schubert  * Redistribution and use in source and binary forms, with or without
5b2b3ffcdSSimon Schubert  * modification, are permitted provided that the following conditions
6b2b3ffcdSSimon Schubert  * are met:
7b2b3ffcdSSimon Schubert  *
8b2b3ffcdSSimon Schubert  * 1. Redistributions of source code must retain the above copyright
9b2b3ffcdSSimon Schubert  *    notice, this list of conditions and the following disclaimer.
10b2b3ffcdSSimon Schubert  * 2. Redistributions in binary form must reproduce the above copyright
11b2b3ffcdSSimon Schubert  *    notice, this list of conditions and the following disclaimer in
12b2b3ffcdSSimon Schubert  *    the documentation and/or other materials provided with the
13b2b3ffcdSSimon Schubert  *    distribution.
14b2b3ffcdSSimon Schubert  * 3. Neither the name of The DragonFly Project nor the names of its
15b2b3ffcdSSimon Schubert  *    contributors may be used to endorse or promote products derived
16b2b3ffcdSSimon Schubert  *    from this software without specific, prior written permission.
17b2b3ffcdSSimon Schubert  *
18b2b3ffcdSSimon Schubert  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19b2b3ffcdSSimon Schubert  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20b2b3ffcdSSimon Schubert  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
21b2b3ffcdSSimon Schubert  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE
22b2b3ffcdSSimon Schubert  * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
23b2b3ffcdSSimon Schubert  * INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING,
24b2b3ffcdSSimon Schubert  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
25b2b3ffcdSSimon Schubert  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
26b2b3ffcdSSimon Schubert  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
27b2b3ffcdSSimon Schubert  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
28b2b3ffcdSSimon Schubert  * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
29b2b3ffcdSSimon Schubert  * SUCH DAMAGE.
30b2b3ffcdSSimon Schubert  *
31b2b3ffcdSSimon Schubert  * --
32b2b3ffcdSSimon Schubert  *
33b2b3ffcdSSimon Schubert  * Mach Operating System
34b2b3ffcdSSimon Schubert  * Copyright (c) 1991,1990 Carnegie Mellon University
35b2b3ffcdSSimon Schubert  * All Rights Reserved.
36b2b3ffcdSSimon Schubert  *
37b2b3ffcdSSimon Schubert  * Permission to use, copy, modify and distribute this software and its
38b2b3ffcdSSimon Schubert  * documentation is hereby granted, provided that both the copyright
39b2b3ffcdSSimon Schubert  * notice and this permission notice appear in all copies of the
40b2b3ffcdSSimon Schubert  * software, derivative works or modified versions, and any portions
41b2b3ffcdSSimon Schubert  * thereof, and that both notices appear in supporting documentation.
42b2b3ffcdSSimon Schubert  *
43b2b3ffcdSSimon Schubert  * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS
44b2b3ffcdSSimon Schubert  * CONDITION.  CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
45b2b3ffcdSSimon Schubert  * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
46b2b3ffcdSSimon Schubert  *
47b2b3ffcdSSimon Schubert  * Carnegie Mellon requests users of this software to return to
48b2b3ffcdSSimon Schubert  *
49b2b3ffcdSSimon Schubert  *  Software Distribution Coordinator  or  Software.Distribution@CS.CMU.EDU
50b2b3ffcdSSimon Schubert  *  School of Computer Science
51b2b3ffcdSSimon Schubert  *  Carnegie Mellon University
52b2b3ffcdSSimon Schubert  *  Pittsburgh PA 15213-3890
53b2b3ffcdSSimon Schubert  *
54b2b3ffcdSSimon Schubert  * any improvements or extensions that they make and grant Carnegie the
55b2b3ffcdSSimon Schubert  * rights to redistribute these changes.
56b2b3ffcdSSimon Schubert  *
57b2b3ffcdSSimon Schubert  * $FreeBSD: src/sys/i386/i386/db_interface.c,v 1.48.2.1 2000/07/07 00:38:46 obrien Exp $
58b2b3ffcdSSimon Schubert  */
59b2b3ffcdSSimon Schubert 
60b2b3ffcdSSimon Schubert /*
61b2b3ffcdSSimon Schubert  * Interface to new debugger.
62b2b3ffcdSSimon Schubert  */
63b2b3ffcdSSimon Schubert #include <sys/param.h>
64b2b3ffcdSSimon Schubert #include <sys/systm.h>
65b2b3ffcdSSimon Schubert #include <sys/reboot.h>
66b2b3ffcdSSimon Schubert #include <sys/cons.h>
67b2b3ffcdSSimon Schubert #include <sys/thread.h>
68adf0eb4fSMatthew Dillon #include <sys/kerneldump.h>
69b2b3ffcdSSimon Schubert 
70b2b3ffcdSSimon Schubert #include <machine/cpu.h>
71b2b3ffcdSSimon Schubert #include <machine/smp.h>
72b2b3ffcdSSimon Schubert #include <machine/globaldata.h>
73b2b3ffcdSSimon Schubert #include <machine/md_var.h>
74b2b3ffcdSSimon Schubert 
75b2b3ffcdSSimon Schubert #include <vm/vm.h>
76b2b3ffcdSSimon Schubert #include <vm/pmap.h>
77b2b3ffcdSSimon Schubert 
78b2b3ffcdSSimon Schubert #include <ddb/ddb.h>
79b2b3ffcdSSimon Schubert 
80b2b3ffcdSSimon Schubert #include <sys/thread2.h>
81b2b3ffcdSSimon Schubert 
8209c0e0d6SSascha Wildner #include <machine/setjmp.h>
83b2b3ffcdSSimon Schubert 
844090d6ffSSascha Wildner static jmp_buf *db_nofault = NULL;
85b2b3ffcdSSimon Schubert extern jmp_buf	db_jmpbuf;
86b2b3ffcdSSimon Schubert 
87b2b3ffcdSSimon Schubert extern void	gdb_handle_exception (db_regs_t *, int, int);
88b2b3ffcdSSimon Schubert 
89b2b3ffcdSSimon Schubert int	db_active;
90b2b3ffcdSSimon Schubert db_regs_t ddb_regs;
91b2b3ffcdSSimon Schubert 
92b2b3ffcdSSimon Schubert static jmp_buf	db_global_jmpbuf;
93b2b3ffcdSSimon Schubert static int	db_global_jmpbuf_valid;
94b2b3ffcdSSimon Schubert 
95b2b3ffcdSSimon Schubert #ifdef __GNUC__
96b2b3ffcdSSimon Schubert #define	rss() ({u_short ss; __asm __volatile("mov %%ss,%0" : "=r" (ss)); ss;})
97b2b3ffcdSSimon Schubert #endif
98b2b3ffcdSSimon Schubert 
99b2b3ffcdSSimon Schubert /*
100b2b3ffcdSSimon Schubert  *  kdb_trap - field a TRACE or BPT trap
101b2b3ffcdSSimon Schubert  */
102b2b3ffcdSSimon Schubert int
kdb_trap(int type,int code,struct x86_64_saved_state * regs)103b2b3ffcdSSimon Schubert kdb_trap(int type, int code, struct x86_64_saved_state *regs)
104b2b3ffcdSSimon Schubert {
105b2b3ffcdSSimon Schubert 	volatile int ddb_mode = !(boothowto & RB_GDB);
106b2b3ffcdSSimon Schubert 
107b2b3ffcdSSimon Schubert 	/*
108b2b3ffcdSSimon Schubert 	 * XXX try to do nothing if the console is in graphics mode.
109b2b3ffcdSSimon Schubert 	 * Handle trace traps (and hardware breakpoints...) by ignoring
110b2b3ffcdSSimon Schubert 	 * them except for forgetting about them.  Return 0 for other
111b2b3ffcdSSimon Schubert 	 * traps to say that we haven't done anything.  The trap handler
112b2b3ffcdSSimon Schubert 	 * will usually panic.  We should handle breakpoint traps for
113b2b3ffcdSSimon Schubert 	 * our breakpoints by disarming our breakpoints and fixing up
114b2b3ffcdSSimon Schubert 	 * %eip.
115b2b3ffcdSSimon Schubert 	 */
116b2b3ffcdSSimon Schubert 	if (cons_unavail && ddb_mode) {
117b2b3ffcdSSimon Schubert 	    if (type == T_TRCTRAP) {
118b2b3ffcdSSimon Schubert 		regs->tf_rflags &= ~PSL_T;
119b2b3ffcdSSimon Schubert 		return (1);
120b2b3ffcdSSimon Schubert 	    }
121b2b3ffcdSSimon Schubert 	    return (0);
122b2b3ffcdSSimon Schubert 	}
123b2b3ffcdSSimon Schubert 
124b2b3ffcdSSimon Schubert 	switch (type) {
125b2b3ffcdSSimon Schubert 	    case T_BPTFLT:	/* breakpoint */
126b2b3ffcdSSimon Schubert 	    case T_TRCTRAP:	/* debug exception */
127b2b3ffcdSSimon Schubert 		break;
128b2b3ffcdSSimon Schubert 
129b2b3ffcdSSimon Schubert 	    default:
130b2b3ffcdSSimon Schubert 		/*
131b2b3ffcdSSimon Schubert 		 * XXX this is almost useless now.  In most cases,
132b2b3ffcdSSimon Schubert 		 * trap_fatal() has already printed a much more verbose
133b2b3ffcdSSimon Schubert 		 * message.  However, it is dangerous to print things in
134b2b3ffcdSSimon Schubert 		 * trap_fatal() - kprintf() might be reentered and trap.
135b2b3ffcdSSimon Schubert 		 * The debugger should be given control first.
136b2b3ffcdSSimon Schubert 		 */
137b2b3ffcdSSimon Schubert 		if (ddb_mode)
138b2b3ffcdSSimon Schubert 		    db_printf("kernel: type %d trap, code=%x\n", type, code);
139b2b3ffcdSSimon Schubert 
140b2b3ffcdSSimon Schubert 		if (db_nofault) {
141b2b3ffcdSSimon Schubert 		    jmp_buf *no_fault = db_nofault;
1424090d6ffSSascha Wildner 		    db_nofault = NULL;
143b2b3ffcdSSimon Schubert 		    longjmp(*no_fault, 1);
144b2b3ffcdSSimon Schubert 		}
145b2b3ffcdSSimon Schubert 	}
146b2b3ffcdSSimon Schubert 
147b2b3ffcdSSimon Schubert 	/*
148b2b3ffcdSSimon Schubert 	 * This handles unexpected traps in ddb commands, including calls to
149b2b3ffcdSSimon Schubert 	 * non-ddb functions.  db_nofault only applies to memory accesses by
150b2b3ffcdSSimon Schubert 	 * internal ddb commands.
151b2b3ffcdSSimon Schubert 	 */
152b2b3ffcdSSimon Schubert 	if (db_global_jmpbuf_valid)
153b2b3ffcdSSimon Schubert 	    longjmp(db_global_jmpbuf, 1);
154b2b3ffcdSSimon Schubert 
155b2b3ffcdSSimon Schubert 	/*
156b2b3ffcdSSimon Schubert 	 * XXX We really should switch to a local stack here.
157b2b3ffcdSSimon Schubert 	 */
158b2b3ffcdSSimon Schubert 	ddb_regs = *regs;
159b2b3ffcdSSimon Schubert 
160b2b3ffcdSSimon Schubert 	crit_enter();
161da23a592SMatthew Dillon 	db_printf("\nCPU%d stopping CPUs: 0x%08jx\n",
162c07315c4SMatthew Dillon 		  mycpu->gd_cpuid,
163c07315c4SMatthew Dillon 		  (uintmax_t)CPUMASK_LOWMASK(mycpu->gd_other_cpus));
164b2b3ffcdSSimon Schubert 
165b2b3ffcdSSimon Schubert 	/* We stop all CPUs except ourselves (obviously) */
166b2b3ffcdSSimon Schubert 	stop_cpus(mycpu->gd_other_cpus);
167b2b3ffcdSSimon Schubert 
168b2b3ffcdSSimon Schubert 	db_printf(" stopped\n");
169b2b3ffcdSSimon Schubert 
170b2b3ffcdSSimon Schubert 	setjmp(db_global_jmpbuf);
171b2b3ffcdSSimon Schubert 	db_global_jmpbuf_valid = TRUE;
172b2b3ffcdSSimon Schubert 	db_active++;
173b2b3ffcdSSimon Schubert 	/* vcons_set_mode(1); */
174b2b3ffcdSSimon Schubert 	if (ddb_mode) {
175b2b3ffcdSSimon Schubert 	    cndbctl(TRUE);
176b2b3ffcdSSimon Schubert 	    db_trap(type, code);
177b2b3ffcdSSimon Schubert 	    cndbctl(FALSE);
178b2b3ffcdSSimon Schubert 	} else
179b2b3ffcdSSimon Schubert 	    gdb_handle_exception(&ddb_regs, type, code);
180b2b3ffcdSSimon Schubert 	db_active--;
181b2b3ffcdSSimon Schubert 	/* vcons_set_mode(0); */
182b2b3ffcdSSimon Schubert 	db_global_jmpbuf_valid = FALSE;
183b2b3ffcdSSimon Schubert 
184*a798ebf2SImre Vadász 	if (panicstr == NULL) {
185da23a592SMatthew Dillon 		db_printf("\nCPU%d restarting CPUs: 0x%016jx\n",
186c07315c4SMatthew Dillon 			  mycpu->gd_cpuid,
187c07315c4SMatthew Dillon 			  (uintmax_t)CPUMASK_LOWMASK(stopped_cpus));
188b2b3ffcdSSimon Schubert 
189b2b3ffcdSSimon Schubert 		/* Restart all the CPUs we previously stopped */
190c07315c4SMatthew Dillon 		if (CPUMASK_CMPMASKNEQ(stopped_cpus, mycpu->gd_other_cpus)) {
191da23a592SMatthew Dillon 			db_printf("whoa, other_cpus: 0x%016jx, "
192da23a592SMatthew Dillon 				  "stopped_cpus: 0x%016jx\n",
193c07315c4SMatthew Dillon 				  (uintmax_t)CPUMASK_LOWMASK(mycpu->gd_other_cpus),
194c07315c4SMatthew Dillon 				  (uintmax_t)CPUMASK_LOWMASK(stopped_cpus));
195b2b3ffcdSSimon Schubert 			panic("stop_cpus() failed");
196b2b3ffcdSSimon Schubert 		}
197b2b3ffcdSSimon Schubert 		restart_cpus(stopped_cpus);
198*a798ebf2SImre Vadász 	}
199b2b3ffcdSSimon Schubert 
200b2b3ffcdSSimon Schubert 	db_printf(" restarted\n");
201b2b3ffcdSSimon Schubert 	crit_exit();
202b2b3ffcdSSimon Schubert 
203b2b3ffcdSSimon Schubert 	regs->tf_rip    = ddb_regs.tf_rip;
204b2b3ffcdSSimon Schubert 	regs->tf_rflags = ddb_regs.tf_rflags;
205b2b3ffcdSSimon Schubert 	regs->tf_rax    = ddb_regs.tf_rax;
206b2b3ffcdSSimon Schubert 	regs->tf_rcx    = ddb_regs.tf_rcx;
207b2b3ffcdSSimon Schubert 	regs->tf_rdx    = ddb_regs.tf_rdx;
208b2b3ffcdSSimon Schubert 	regs->tf_rbx    = ddb_regs.tf_rbx;
209b2b3ffcdSSimon Schubert 
210b2b3ffcdSSimon Schubert     	regs->tf_rsp    = ddb_regs.tf_rsp;
211b2b3ffcdSSimon Schubert     	regs->tf_ss     = ddb_regs.tf_ss & 0xffff;
212b2b3ffcdSSimon Schubert 
213b2b3ffcdSSimon Schubert 	regs->tf_rbp    = ddb_regs.tf_rbp;
214b2b3ffcdSSimon Schubert 	regs->tf_rsi    = ddb_regs.tf_rsi;
215b2b3ffcdSSimon Schubert 	regs->tf_rdi    = ddb_regs.tf_rdi;
216b2b3ffcdSSimon Schubert 
217b2b3ffcdSSimon Schubert 	regs->tf_r8     = ddb_regs.tf_r8;
218b2b3ffcdSSimon Schubert 	regs->tf_r9     = ddb_regs.tf_r9;
219b2b3ffcdSSimon Schubert 	regs->tf_r10    = ddb_regs.tf_r10;
220b2b3ffcdSSimon Schubert 	regs->tf_r11    = ddb_regs.tf_r11;
221b2b3ffcdSSimon Schubert 	regs->tf_r12    = ddb_regs.tf_r12;
222b2b3ffcdSSimon Schubert 	regs->tf_r13    = ddb_regs.tf_r13;
223b2b3ffcdSSimon Schubert 	regs->tf_r14    = ddb_regs.tf_r14;
224b2b3ffcdSSimon Schubert 	regs->tf_r15    = ddb_regs.tf_r15;
225b2b3ffcdSSimon Schubert 
226b2b3ffcdSSimon Schubert 	/* regs->tf_es     = ddb_regs.tf_es & 0xffff; */
227b2b3ffcdSSimon Schubert 	/* regs->tf_fs     = ddb_regs.tf_fs & 0xffff; */
228b2b3ffcdSSimon Schubert 	/* regs->tf_gs     = ddb_regs.tf_gs & 0xffff; */
229b2b3ffcdSSimon Schubert 	regs->tf_cs     = ddb_regs.tf_cs & 0xffff;
230b2b3ffcdSSimon Schubert 	/* regs->tf_ds     = ddb_regs.tf_ds & 0xffff; */
231b2b3ffcdSSimon Schubert 	return (1);
232b2b3ffcdSSimon Schubert }
233b2b3ffcdSSimon Schubert 
234b2b3ffcdSSimon Schubert /*
235b2b3ffcdSSimon Schubert  * Read bytes from kernel address space for debugger.
236b2b3ffcdSSimon Schubert  */
237b2b3ffcdSSimon Schubert void
db_read_bytes(vm_offset_t addr,size_t size,char * data)238b2b3ffcdSSimon Schubert db_read_bytes(vm_offset_t addr, size_t size, char *data)
239b2b3ffcdSSimon Schubert {
240b2b3ffcdSSimon Schubert 	char	*src;
241b2b3ffcdSSimon Schubert 
242b2b3ffcdSSimon Schubert 	db_nofault = &db_jmpbuf;
243b2b3ffcdSSimon Schubert 
244b2b3ffcdSSimon Schubert 	src = (char *)addr;
245b2b3ffcdSSimon Schubert 	while (size-- > 0)
246b2b3ffcdSSimon Schubert 	    *data++ = *src++;
247b2b3ffcdSSimon Schubert 
2484090d6ffSSascha Wildner 	db_nofault = NULL;
249b2b3ffcdSSimon Schubert }
250b2b3ffcdSSimon Schubert 
251b2b3ffcdSSimon Schubert /*
252b2b3ffcdSSimon Schubert  * Write bytes to kernel address space for debugger.
253b2b3ffcdSSimon Schubert  */
254b2b3ffcdSSimon Schubert void
db_write_bytes(vm_offset_t addr,size_t size,char * data)255b2b3ffcdSSimon Schubert db_write_bytes(vm_offset_t addr, size_t size, char *data)
256b2b3ffcdSSimon Schubert {
257b2b3ffcdSSimon Schubert 	char	*dst;
258b2b3ffcdSSimon Schubert #if 0
259b2b3ffcdSSimon Schubert 	vpte_t	*ptep0 = NULL;
260b2b3ffcdSSimon Schubert 	vpte_t	oldmap0 = 0;
261b2b3ffcdSSimon Schubert 	vm_offset_t	addr1;
262b2b3ffcdSSimon Schubert 	vpte_t	*ptep1 = NULL;
263b2b3ffcdSSimon Schubert 	vpte_t	oldmap1 = 0;
264b2b3ffcdSSimon Schubert #endif
265b2b3ffcdSSimon Schubert 
266b2b3ffcdSSimon Schubert 	db_nofault = &db_jmpbuf;
267b2b3ffcdSSimon Schubert #if 0
268b2b3ffcdSSimon Schubert 	if (addr > trunc_page((vm_offset_t)btext) - size &&
269b2b3ffcdSSimon Schubert 	    addr < round_page((vm_offset_t)etext)) {
270b2b3ffcdSSimon Schubert 
271b2b3ffcdSSimon Schubert 	    ptep0 = pmap_kpte(addr);
272b2b3ffcdSSimon Schubert 	    oldmap0 = *ptep0;
273a86ce0cdSMatthew Dillon 	    *ptep0 |= VPTE_RW;
274b2b3ffcdSSimon Schubert 
275b2b3ffcdSSimon Schubert 	    /* Map another page if the data crosses a page boundary. */
276b2b3ffcdSSimon Schubert 	    if ((*ptep0 & PG_PS) == 0) {
277b2b3ffcdSSimon Schubert 	    	addr1 = trunc_page(addr + size - 1);
278b2b3ffcdSSimon Schubert 	    	if (trunc_page(addr) != addr1) {
279b2b3ffcdSSimon Schubert 		    ptep1 = pmap_kpte(addr1);
280b2b3ffcdSSimon Schubert 		    oldmap1 = *ptep1;
281a86ce0cdSMatthew Dillon 		    *ptep1 |= VPTE_RW;
282b2b3ffcdSSimon Schubert 	    	}
283b2b3ffcdSSimon Schubert 	    } else {
284b2b3ffcdSSimon Schubert 		addr1 = trunc_4mpage(addr + size - 1);
285b2b3ffcdSSimon Schubert 		if (trunc_4mpage(addr) != addr1) {
286b2b3ffcdSSimon Schubert 		    ptep1 = pmap_kpte(addr1);
287b2b3ffcdSSimon Schubert 		    oldmap1 = *ptep1;
288a86ce0cdSMatthew Dillon 		    *ptep1 |= VPTE_RW;
289b2b3ffcdSSimon Schubert 		}
290b2b3ffcdSSimon Schubert 	    }
291b2b3ffcdSSimon Schubert 
292b2b3ffcdSSimon Schubert 	    cpu_invltlb();
293b2b3ffcdSSimon Schubert 	}
294b2b3ffcdSSimon Schubert #endif
295b2b3ffcdSSimon Schubert 
296b2b3ffcdSSimon Schubert 	dst = (char *)addr;
297b2b3ffcdSSimon Schubert 
298b2b3ffcdSSimon Schubert 	while (size-- > 0)
299b2b3ffcdSSimon Schubert 	    *dst++ = *data++;
300b2b3ffcdSSimon Schubert 
3014090d6ffSSascha Wildner 	db_nofault = NULL;
302b2b3ffcdSSimon Schubert 
303b2b3ffcdSSimon Schubert #if 0
304b2b3ffcdSSimon Schubert 	if (ptep0) {
305b2b3ffcdSSimon Schubert 	    *ptep0 = oldmap0;
306b2b3ffcdSSimon Schubert 
307b2b3ffcdSSimon Schubert 	    if (ptep1)
308b2b3ffcdSSimon Schubert 		*ptep1 = oldmap1;
309b2b3ffcdSSimon Schubert 
310b2b3ffcdSSimon Schubert 	    cpu_invltlb();
311b2b3ffcdSSimon Schubert 	}
312b2b3ffcdSSimon Schubert #endif
313b2b3ffcdSSimon Schubert }
314b2b3ffcdSSimon Schubert 
315b2b3ffcdSSimon Schubert /*
316b2b3ffcdSSimon Schubert  * The debugger sometimes needs to know the actual KVM address represented
317b2b3ffcdSSimon Schubert  * by the instruction pointer, stack pointer, or base pointer.  Normally
318b2b3ffcdSSimon Schubert  * the actual KVM address is simply the contents of the register.  However,
319b2b3ffcdSSimon Schubert  * if the debugger is entered from the BIOS or VM86 we need to figure out
320b2b3ffcdSSimon Schubert  * the offset from the segment register.
321b2b3ffcdSSimon Schubert  */
322b2b3ffcdSSimon Schubert db_addr_t
PC_REGS(db_regs_t * regs)323b2b3ffcdSSimon Schubert PC_REGS(db_regs_t *regs)
324b2b3ffcdSSimon Schubert {
325b2b3ffcdSSimon Schubert     return(regs->tf_rip);
326b2b3ffcdSSimon Schubert }
327b2b3ffcdSSimon Schubert 
328b2b3ffcdSSimon Schubert db_addr_t
SP_REGS(db_regs_t * regs)329b2b3ffcdSSimon Schubert SP_REGS(db_regs_t *regs)
330b2b3ffcdSSimon Schubert {
331b2b3ffcdSSimon Schubert     return(regs->tf_rsp);
332b2b3ffcdSSimon Schubert }
333b2b3ffcdSSimon Schubert 
334b2b3ffcdSSimon Schubert db_addr_t
BP_REGS(db_regs_t * regs)335b2b3ffcdSSimon Schubert BP_REGS(db_regs_t *regs)
336b2b3ffcdSSimon Schubert {
337b2b3ffcdSSimon Schubert     return(regs->tf_rbp);
338b2b3ffcdSSimon Schubert }
339b2b3ffcdSSimon Schubert 
340b2b3ffcdSSimon Schubert /*
341b2b3ffcdSSimon Schubert  * XXX
342b2b3ffcdSSimon Schubert  * Move this to machdep.c and allow it to be called if any debugger is
343b2b3ffcdSSimon Schubert  * installed.
344b2b3ffcdSSimon Schubert  */
345b2b3ffcdSSimon Schubert void
Debugger(const char * msg)346b2b3ffcdSSimon Schubert Debugger(const char *msg)
347b2b3ffcdSSimon Schubert {
348b2b3ffcdSSimon Schubert 	static volatile u_char in_Debugger;
349b2b3ffcdSSimon Schubert 
350b2b3ffcdSSimon Schubert 	/*
351b2b3ffcdSSimon Schubert 	 * XXX
352b2b3ffcdSSimon Schubert 	 * Do nothing if the console is in graphics mode.  This is
353b2b3ffcdSSimon Schubert 	 * OK if the call is for the debugger hotkey but not if the call
354b2b3ffcdSSimon Schubert 	 * is a weak form of panicing.
355b2b3ffcdSSimon Schubert 	 */
356*a798ebf2SImre Vadász 	if (cons_unavail && !(boothowto & RB_GDB)) {
357*a798ebf2SImre Vadász 		/*
358*a798ebf2SImre Vadász 		 * If we are panicing, panic() expects cpus to be stopped
359*a798ebf2SImre Vadász 		 * when Debugger() returns.
360*a798ebf2SImre Vadász 		 */
361*a798ebf2SImre Vadász 		if (panicstr != NULL)
362*a798ebf2SImre Vadász 			stop_cpus(mycpu->gd_other_cpus);
363b2b3ffcdSSimon Schubert 		return;
364*a798ebf2SImre Vadász 	}
365b2b3ffcdSSimon Schubert 
366b2b3ffcdSSimon Schubert 	if (!in_Debugger) {
367b2b3ffcdSSimon Schubert 	    in_Debugger = 1;
368b2b3ffcdSSimon Schubert 	    db_printf("Debugger(\"%s\")\n", msg);
369adf0eb4fSMatthew Dillon 
370adf0eb4fSMatthew Dillon 	    /*
371adf0eb4fSMatthew Dillon 	     * Save the pcb just in case the sysop entered the debugger
372adf0eb4fSMatthew Dillon 	     * manually and called dumpsys,
373adf0eb4fSMatthew Dillon 	     */
374adf0eb4fSMatthew Dillon 	    if (dumpthread == NULL) {
375adf0eb4fSMatthew Dillon 		    savectx(&dumppcb);
376adf0eb4fSMatthew Dillon 		    dumpthread = curthread;
377adf0eb4fSMatthew Dillon 	    }
378adf0eb4fSMatthew Dillon 
379b2b3ffcdSSimon Schubert 	    breakpoint();
380b2b3ffcdSSimon Schubert 	    in_Debugger = 0;
381adf0eb4fSMatthew Dillon 
382adf0eb4fSMatthew Dillon 	    /*
383adf0eb4fSMatthew Dillon 	     * Clear before returning from the debugger so a later panic
384adf0eb4fSMatthew Dillon 	     * saves the correct context.
385adf0eb4fSMatthew Dillon 	     */
386adf0eb4fSMatthew Dillon 	    dumpthread = NULL;
387b2b3ffcdSSimon Schubert 	}
388b2b3ffcdSSimon Schubert }
389