xref: /freebsd/lib/libkvm/kvm_i386.c (revision 953e4134b5b3948e32f9f0066d2f5cc8d7146c9c)
1f95a0250SRodney W. Grimes /*-
2f95a0250SRodney W. Grimes  * Copyright (c) 1989, 1992, 1993
3f95a0250SRodney W. Grimes  *	The Regents of the University of California.  All rights reserved.
4f95a0250SRodney W. Grimes  *
5f95a0250SRodney W. Grimes  * This code is derived from software developed by the Computer Systems
6f95a0250SRodney W. Grimes  * Engineering group at Lawrence Berkeley Laboratory under DARPA contract
7f95a0250SRodney W. Grimes  * BG 91-66 and contributed to Berkeley.
8f95a0250SRodney W. Grimes  *
9f95a0250SRodney W. Grimes  * Redistribution and use in source and binary forms, with or without
10f95a0250SRodney W. Grimes  * modification, are permitted provided that the following conditions
11f95a0250SRodney W. Grimes  * are met:
12f95a0250SRodney W. Grimes  * 1. Redistributions of source code must retain the above copyright
13f95a0250SRodney W. Grimes  *    notice, this list of conditions and the following disclaimer.
14f95a0250SRodney W. Grimes  * 2. Redistributions in binary form must reproduce the above copyright
15f95a0250SRodney W. Grimes  *    notice, this list of conditions and the following disclaimer in the
16f95a0250SRodney W. Grimes  *    documentation and/or other materials provided with the distribution.
17f95a0250SRodney W. Grimes  * 4. Neither the name of the University nor the names of its contributors
18f95a0250SRodney W. Grimes  *    may be used to endorse or promote products derived from this software
19f95a0250SRodney W. Grimes  *    without specific prior written permission.
20f95a0250SRodney W. Grimes  *
21f95a0250SRodney W. Grimes  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
22f95a0250SRodney W. Grimes  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
23f95a0250SRodney W. Grimes  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
24f95a0250SRodney W. Grimes  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
25f95a0250SRodney W. Grimes  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
26f95a0250SRodney W. Grimes  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
27f95a0250SRodney W. Grimes  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
28f95a0250SRodney W. Grimes  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
29f95a0250SRodney W. Grimes  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
30f95a0250SRodney W. Grimes  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
31f95a0250SRodney W. Grimes  * SUCH DAMAGE.
32f95a0250SRodney W. Grimes  */
33f95a0250SRodney W. Grimes 
34e67f5b9fSMatthew Dillon #include <sys/cdefs.h>
35e67f5b9fSMatthew Dillon __FBSDID("$FreeBSD$");
36e67f5b9fSMatthew Dillon 
37f95a0250SRodney W. Grimes #if defined(LIBC_SCCS) && !defined(lint)
38c4a7cdb3SPeter Wemm #if 0
39f95a0250SRodney W. Grimes static char sccsid[] = "@(#)kvm_hp300.c	8.1 (Berkeley) 6/4/93";
40c4a7cdb3SPeter Wemm #endif
41f95a0250SRodney W. Grimes #endif /* LIBC_SCCS and not lint */
42f95a0250SRodney W. Grimes 
43f95a0250SRodney W. Grimes /*
4421d54b07SRodney W. Grimes  * i386 machine dependent routines for kvm.  Hopefully, the forthcoming
45f95a0250SRodney W. Grimes  * vm code will one day obsolete this module.
46f95a0250SRodney W. Grimes  */
47f95a0250SRodney W. Grimes 
48f95a0250SRodney W. Grimes #include <sys/param.h>
49f95a0250SRodney W. Grimes #include <sys/user.h>
50f95a0250SRodney W. Grimes #include <sys/proc.h>
51f95a0250SRodney W. Grimes #include <sys/stat.h>
52e55a0cd8SPeter Wemm #include <sys/mman.h>
5351295a4dSJordan K. Hubbard #include <stdlib.h>
54953e4134SEd Schouten #include <string.h>
55f95a0250SRodney W. Grimes #include <unistd.h>
56f95a0250SRodney W. Grimes #include <nlist.h>
57f95a0250SRodney W. Grimes #include <kvm.h>
58f95a0250SRodney W. Grimes 
59f95a0250SRodney W. Grimes #include <vm/vm.h>
60f95a0250SRodney W. Grimes #include <vm/vm_param.h>
61f95a0250SRodney W. Grimes 
62e55a0cd8SPeter Wemm #include <machine/elf.h>
63e55a0cd8SPeter Wemm 
64f95a0250SRodney W. Grimes #include <limits.h>
65f95a0250SRodney W. Grimes 
66f95a0250SRodney W. Grimes #include "kvm_private.h"
67f95a0250SRodney W. Grimes 
68f95a0250SRodney W. Grimes #ifndef btop
6921d54b07SRodney W. Grimes #define	btop(x)		(i386_btop(x))
7021d54b07SRodney W. Grimes #define	ptob(x)		(i386_ptob(x))
71f95a0250SRodney W. Grimes #endif
72f95a0250SRodney W. Grimes 
73e55a0cd8SPeter Wemm #define	PG_FRAME_PAE	(~((uint64_t)PAGE_MASK))
74e55a0cd8SPeter Wemm #define	PDRSHIFT_PAE	21
75e55a0cd8SPeter Wemm #define	NPTEPG_PAE	(PAGE_SIZE/sizeof(uint64_t))
76e55a0cd8SPeter Wemm #define	NBPDR_PAE	(1<<PDRSHIFT_PAE)
77e55a0cd8SPeter Wemm 
78e9ca6fe4SPeter Wemm /* minidump must be the first item! */
79f95a0250SRodney W. Grimes struct vmstate {
80e9ca6fe4SPeter Wemm 	int		minidump;	/* 1 = minidump mode */
81e55a0cd8SPeter Wemm 	void		*mmapbase;
82e55a0cd8SPeter Wemm 	size_t		mmapsize;
83e55a0cd8SPeter Wemm 	void		*PTD;
84e55a0cd8SPeter Wemm 	int		pae;
85f95a0250SRodney W. Grimes };
86f95a0250SRodney W. Grimes 
87e55a0cd8SPeter Wemm /*
88e55a0cd8SPeter Wemm  * Map the ELF headers into the process' address space. We do this in two
89e55a0cd8SPeter Wemm  * steps: first the ELF header itself and using that information the whole
90e55a0cd8SPeter Wemm  * set of headers. (Taken from kvm_ia64.c)
91e55a0cd8SPeter Wemm  */
92e55a0cd8SPeter Wemm static int
93e55a0cd8SPeter Wemm _kvm_maphdrs(kvm_t *kd, size_t sz)
94e55a0cd8SPeter Wemm {
95e55a0cd8SPeter Wemm 	struct vmstate *vm = kd->vmst;
96e55a0cd8SPeter Wemm 
97e55a0cd8SPeter Wemm 	/* munmap() previous mmap(). */
98e55a0cd8SPeter Wemm 	if (vm->mmapbase != NULL) {
99e55a0cd8SPeter Wemm 		munmap(vm->mmapbase, vm->mmapsize);
100e55a0cd8SPeter Wemm 		vm->mmapbase = NULL;
101e55a0cd8SPeter Wemm 	}
102e55a0cd8SPeter Wemm 
103e55a0cd8SPeter Wemm 	vm->mmapsize = sz;
104e55a0cd8SPeter Wemm 	vm->mmapbase = mmap(NULL, sz, PROT_READ, MAP_PRIVATE, kd->pmfd, 0);
105e55a0cd8SPeter Wemm 	if (vm->mmapbase == MAP_FAILED) {
106e55a0cd8SPeter Wemm 		_kvm_err(kd, kd->program, "cannot mmap corefile");
107e55a0cd8SPeter Wemm 		return (-1);
108e55a0cd8SPeter Wemm 	}
109e55a0cd8SPeter Wemm 	return (0);
110e55a0cd8SPeter Wemm }
111e55a0cd8SPeter Wemm 
112e55a0cd8SPeter Wemm /*
113e55a0cd8SPeter Wemm  * Translate a physical memory address to a file-offset in the crash-dump.
114e55a0cd8SPeter Wemm  * (Taken from kvm_ia64.c)
115e55a0cd8SPeter Wemm  */
116e55a0cd8SPeter Wemm static size_t
117e55a0cd8SPeter Wemm _kvm_pa2off(kvm_t *kd, uint64_t pa, off_t *ofs)
118e55a0cd8SPeter Wemm {
119e55a0cd8SPeter Wemm 	Elf_Ehdr *e = kd->vmst->mmapbase;
120d7dc9f76SHidetoshi Shimokawa 	Elf_Phdr *p;
121d7dc9f76SHidetoshi Shimokawa 	int n;
122e55a0cd8SPeter Wemm 
123d7dc9f76SHidetoshi Shimokawa 	if (kd->rawdump) {
124d7dc9f76SHidetoshi Shimokawa 		*ofs = pa;
125d7dc9f76SHidetoshi Shimokawa 		return (PAGE_SIZE - ((size_t)pa & PAGE_MASK));
126d7dc9f76SHidetoshi Shimokawa 	}
127d7dc9f76SHidetoshi Shimokawa 
128d7dc9f76SHidetoshi Shimokawa 	p = (Elf_Phdr*)((char*)e + e->e_phoff);
129d7dc9f76SHidetoshi Shimokawa 	n = e->e_phnum;
130e55a0cd8SPeter Wemm 	while (n && (pa < p->p_paddr || pa >= p->p_paddr + p->p_memsz))
131e55a0cd8SPeter Wemm 		p++, n--;
132e55a0cd8SPeter Wemm 	if (n == 0)
133e55a0cd8SPeter Wemm 		return (0);
134e55a0cd8SPeter Wemm 	*ofs = (pa - p->p_paddr) + p->p_offset;
135e55a0cd8SPeter Wemm 	return (PAGE_SIZE - ((size_t)pa & PAGE_MASK));
136e55a0cd8SPeter Wemm }
137e55a0cd8SPeter Wemm 
138f95a0250SRodney W. Grimes void
1392f85bf6eSPeter Wemm _kvm_freevtop(kvm_t *kd)
1402f85bf6eSPeter Wemm {
141e55a0cd8SPeter Wemm 	struct vmstate *vm = kd->vmst;
142e55a0cd8SPeter Wemm 
143e9ca6fe4SPeter Wemm 	if (kd->vmst->minidump)
144e9ca6fe4SPeter Wemm 		return (_kvm_minidump_freevtop(kd));
145e55a0cd8SPeter Wemm 	if (vm->mmapbase != NULL)
146e55a0cd8SPeter Wemm 		munmap(vm->mmapbase, vm->mmapsize);
147e55a0cd8SPeter Wemm 	if (vm->PTD)
148e55a0cd8SPeter Wemm 		free(vm->PTD);
149e55a0cd8SPeter Wemm 	free(vm);
150e55a0cd8SPeter Wemm 	kd->vmst = NULL;
15121d54b07SRodney W. Grimes }
152f95a0250SRodney W. Grimes 
153f95a0250SRodney W. Grimes int
1542f85bf6eSPeter Wemm _kvm_initvtop(kvm_t *kd)
1552f85bf6eSPeter Wemm {
15621d54b07SRodney W. Grimes 	struct nlist nlist[2];
1572f85bf6eSPeter Wemm 	u_long pa;
158f85f3040SPeter Wemm 	u_long kernbase;
159e55a0cd8SPeter Wemm 	char		*PTD;
160e55a0cd8SPeter Wemm 	Elf_Ehdr	*ehdr;
161e55a0cd8SPeter Wemm 	size_t		hdrsz;
162e55a0cd8SPeter Wemm 	int		i;
163e9ca6fe4SPeter Wemm 	char		minihdr[8];
164e9ca6fe4SPeter Wemm 
165d7dc9f76SHidetoshi Shimokawa 	if (!kd->rawdump && pread(kd->pmfd, &minihdr, 8, 0) == 8)
166e9ca6fe4SPeter Wemm 		if (memcmp(&minihdr, "minidump", 8) == 0)
167e9ca6fe4SPeter Wemm 			return (_kvm_minidump_initvtop(kd));
168f95a0250SRodney W. Grimes 
169e55a0cd8SPeter Wemm 	kd->vmst = (struct vmstate *)_kvm_malloc(kd, sizeof(*kd->vmst));
170e55a0cd8SPeter Wemm 	if (kd->vmst == 0) {
17121d54b07SRodney W. Grimes 		_kvm_err(kd, kd->program, "cannot allocate vm");
172f95a0250SRodney W. Grimes 		return (-1);
17321d54b07SRodney W. Grimes 	}
174e55a0cd8SPeter Wemm 	kd->vmst->PTD = 0;
175e55a0cd8SPeter Wemm 
176d7dc9f76SHidetoshi Shimokawa 	if (kd->rawdump == 0) {
177e55a0cd8SPeter Wemm 		if (_kvm_maphdrs(kd, sizeof(Elf_Ehdr)) == -1)
178e55a0cd8SPeter Wemm 			return (-1);
179e55a0cd8SPeter Wemm 
180e55a0cd8SPeter Wemm 		ehdr = kd->vmst->mmapbase;
181e55a0cd8SPeter Wemm 		hdrsz = ehdr->e_phoff + ehdr->e_phentsize * ehdr->e_phnum;
182e55a0cd8SPeter Wemm 		if (_kvm_maphdrs(kd, hdrsz) == -1)
183e55a0cd8SPeter Wemm 			return (-1);
184d7dc9f76SHidetoshi Shimokawa 	}
185f95a0250SRodney W. Grimes 
186f85f3040SPeter Wemm 	nlist[0].n_name = "kernbase";
187f85f3040SPeter Wemm 	nlist[1].n_name = 0;
188f85f3040SPeter Wemm 
189f85f3040SPeter Wemm 	if (kvm_nlist(kd, nlist) != 0)
190f85f3040SPeter Wemm 		kernbase = KERNBASE;	/* for old kernels */
191f85f3040SPeter Wemm 	else
192f85f3040SPeter Wemm 		kernbase = nlist[0].n_value;
193f85f3040SPeter Wemm 
194e55a0cd8SPeter Wemm 	nlist[0].n_name = "IdlePDPT";
195e55a0cd8SPeter Wemm 	nlist[1].n_name = 0;
196e55a0cd8SPeter Wemm 
197e55a0cd8SPeter Wemm 	if (kvm_nlist(kd, nlist) == 0) {
198e55a0cd8SPeter Wemm 		uint64_t pa64;
199e55a0cd8SPeter Wemm 
200e55a0cd8SPeter Wemm 		if (kvm_read(kd, (nlist[0].n_value - kernbase), &pa,
201e55a0cd8SPeter Wemm 		    sizeof(pa)) != sizeof(pa)) {
202e55a0cd8SPeter Wemm 			_kvm_err(kd, kd->program, "cannot read IdlePDPT");
203e55a0cd8SPeter Wemm 			return (-1);
204e55a0cd8SPeter Wemm 		}
205e55a0cd8SPeter Wemm 		PTD = _kvm_malloc(kd, 4 * PAGE_SIZE);
206e55a0cd8SPeter Wemm 		for (i = 0; i < 4; i++) {
207e55a0cd8SPeter Wemm 			if (kvm_read(kd, pa + (i * sizeof(pa64)), &pa64,
208e55a0cd8SPeter Wemm 			    sizeof(pa64)) != sizeof(pa64)) {
209e55a0cd8SPeter Wemm 				_kvm_err(kd, kd->program, "Cannot read PDPT");
210e55a0cd8SPeter Wemm 				free(PTD);
211e55a0cd8SPeter Wemm 				return (-1);
212e55a0cd8SPeter Wemm 			}
213e55a0cd8SPeter Wemm 			if (kvm_read(kd, pa64 & PG_FRAME_PAE,
214e55a0cd8SPeter Wemm 			    PTD + (i * PAGE_SIZE), PAGE_SIZE) != (PAGE_SIZE)) {
215e55a0cd8SPeter Wemm 				_kvm_err(kd, kd->program, "cannot read PDPT");
216e55a0cd8SPeter Wemm 				free(PTD);
217e55a0cd8SPeter Wemm 				return (-1);
218e55a0cd8SPeter Wemm 			}
219e55a0cd8SPeter Wemm 		}
220e55a0cd8SPeter Wemm 		kd->vmst->PTD = PTD;
221e55a0cd8SPeter Wemm 		kd->vmst->pae = 1;
222e55a0cd8SPeter Wemm 	} else {
223f85f3040SPeter Wemm 		nlist[0].n_name = "IdlePTD";
22421d54b07SRodney W. Grimes 		nlist[1].n_name = 0;
225f95a0250SRodney W. Grimes 
226f95a0250SRodney W. Grimes 		if (kvm_nlist(kd, nlist) != 0) {
227f95a0250SRodney W. Grimes 			_kvm_err(kd, kd->program, "bad namelist");
228f95a0250SRodney W. Grimes 			return (-1);
229f95a0250SRodney W. Grimes 		}
230e55a0cd8SPeter Wemm 		if (kvm_read(kd, (nlist[0].n_value - kernbase), &pa,
231e55a0cd8SPeter Wemm 		    sizeof(pa)) != sizeof(pa)) {
23221d54b07SRodney W. Grimes 			_kvm_err(kd, kd->program, "cannot read IdlePTD");
233f95a0250SRodney W. Grimes 			return (-1);
234f95a0250SRodney W. Grimes 		}
2352f85bf6eSPeter Wemm 		PTD = _kvm_malloc(kd, PAGE_SIZE);
2362f85bf6eSPeter Wemm 		if (kvm_read(kd, pa, PTD, PAGE_SIZE) != PAGE_SIZE) {
23721d54b07SRodney W. Grimes 			_kvm_err(kd, kd->program, "cannot read PTD");
238f95a0250SRodney W. Grimes 			return (-1);
239f95a0250SRodney W. Grimes 		}
240e55a0cd8SPeter Wemm 		kd->vmst->PTD = PTD;
241e55a0cd8SPeter Wemm 		return (0);
242e55a0cd8SPeter Wemm 		kd->vmst->pae = 0;
243e55a0cd8SPeter Wemm 	}
244f95a0250SRodney W. Grimes 	return (0);
245f95a0250SRodney W. Grimes }
246f95a0250SRodney W. Grimes 
247f95a0250SRodney W. Grimes static int
248e55a0cd8SPeter Wemm _kvm_vatop(kvm_t *kd, u_long va, off_t *pa)
2492f85bf6eSPeter Wemm {
2502f85bf6eSPeter Wemm 	struct vmstate *vm;
2512f85bf6eSPeter Wemm 	u_long offset;
2522f85bf6eSPeter Wemm 	u_long pte_pa;
253e55a0cd8SPeter Wemm 	u_long pde_pa;
2542f85bf6eSPeter Wemm 	pd_entry_t pde;
2552f85bf6eSPeter Wemm 	pt_entry_t pte;
2562f85bf6eSPeter Wemm 	u_long pdeindex;
2572f85bf6eSPeter Wemm 	u_long pteindex;
258e55a0cd8SPeter Wemm 	size_t s;
259e55a0cd8SPeter Wemm 	u_long a;
260e55a0cd8SPeter Wemm 	off_t ofs;
261e55a0cd8SPeter Wemm 	uint32_t *PTD;
2622f85bf6eSPeter Wemm 
2632f85bf6eSPeter Wemm 	vm = kd->vmst;
264e55a0cd8SPeter Wemm 	PTD = (uint32_t *)vm->PTD;
2652f85bf6eSPeter Wemm 	offset = va & (PAGE_SIZE - 1);
2662f85bf6eSPeter Wemm 
2672f85bf6eSPeter Wemm 	/*
2682f85bf6eSPeter Wemm 	 * If we are initializing (kernel page table descriptor pointer
2692f85bf6eSPeter Wemm 	 * not yet set) then return pa == va to avoid infinite recursion.
2702f85bf6eSPeter Wemm 	 */
271e55a0cd8SPeter Wemm 	if (PTD == 0) {
272e55a0cd8SPeter Wemm 		s = _kvm_pa2off(kd, va, pa);
273e55a0cd8SPeter Wemm 		if (s == 0) {
274e55a0cd8SPeter Wemm 			_kvm_err(kd, kd->program,
275e55a0cd8SPeter Wemm 			    "_kvm_vatop: bootstrap data not in dump");
276e55a0cd8SPeter Wemm 			goto invalid;
277e55a0cd8SPeter Wemm 		} else
2782f85bf6eSPeter Wemm 			return (PAGE_SIZE - offset);
2792f85bf6eSPeter Wemm 	}
2802f85bf6eSPeter Wemm 
2812f85bf6eSPeter Wemm 	pdeindex = va >> PDRSHIFT;
282e55a0cd8SPeter Wemm 	pde = PTD[pdeindex];
283e55a0cd8SPeter Wemm 	if (((u_long)pde & PG_V) == 0) {
284e55a0cd8SPeter Wemm 		_kvm_err(kd, kd->program, "_kvm_vatop: pde not valid");
2852f85bf6eSPeter Wemm 		goto invalid;
286e55a0cd8SPeter Wemm 	}
2872f85bf6eSPeter Wemm 
2884afb0d5aSTor Egge 	if ((u_long)pde & PG_PS) {
2894afb0d5aSTor Egge 	      /*
2904afb0d5aSTor Egge 	       * No second-level page table; ptd describes one 4MB page.
2914afb0d5aSTor Egge 	       * (We assume that the kernel wouldn't set PG_PS without enabling
292e55a0cd8SPeter Wemm 	       * it cr0).
2934afb0d5aSTor Egge 	       */
2944afb0d5aSTor Egge #define	PAGE4M_MASK	(NBPDR - 1)
2954afb0d5aSTor Egge #define	PG_FRAME4M	(~PAGE4M_MASK)
296e55a0cd8SPeter Wemm 		pde_pa = ((u_long)pde & PG_FRAME4M) + (va & PAGE4M_MASK);
297e55a0cd8SPeter Wemm 		s = _kvm_pa2off(kd, pde_pa, &ofs);
298add112ffSPeter Wemm 		if (s < sizeof pde) {
299e55a0cd8SPeter Wemm 			_kvm_syserr(kd, kd->program,
300e55a0cd8SPeter Wemm 			    "_kvm_vatop: pde_pa not found");
301e55a0cd8SPeter Wemm 			goto invalid;
302e55a0cd8SPeter Wemm 		}
303e55a0cd8SPeter Wemm 		*pa = ofs;
3044afb0d5aSTor Egge 		return (NBPDR - (va & PAGE4M_MASK));
3054afb0d5aSTor Egge 	}
3064afb0d5aSTor Egge 
3072f85bf6eSPeter Wemm 	pteindex = (va >> PAGE_SHIFT) & (NPTEPG-1);
308e55a0cd8SPeter Wemm 	pte_pa = ((u_long)pde & PG_FRAME) + (pteindex * sizeof(pde));
309e55a0cd8SPeter Wemm 
310e55a0cd8SPeter Wemm 	s = _kvm_pa2off(kd, pte_pa, &ofs);
311add112ffSPeter Wemm 	if (s < sizeof pte) {
312e55a0cd8SPeter Wemm 		_kvm_err(kd, kd->program, "_kvm_vatop: pdpe_pa not found");
313e55a0cd8SPeter Wemm 		goto invalid;
314e55a0cd8SPeter Wemm 	}
3152f85bf6eSPeter Wemm 
3162f85bf6eSPeter Wemm 	/* XXX This has to be a physical address read, kvm_read is virtual */
317e55a0cd8SPeter Wemm 	if (lseek(kd->pmfd, ofs, 0) == -1) {
3182f85bf6eSPeter Wemm 		_kvm_syserr(kd, kd->program, "_kvm_vatop: lseek");
3192f85bf6eSPeter Wemm 		goto invalid;
3202f85bf6eSPeter Wemm 	}
3212f85bf6eSPeter Wemm 	if (read(kd->pmfd, &pte, sizeof pte) != sizeof pte) {
3222f85bf6eSPeter Wemm 		_kvm_syserr(kd, kd->program, "_kvm_vatop: read");
3232f85bf6eSPeter Wemm 		goto invalid;
3242f85bf6eSPeter Wemm 	}
325e55a0cd8SPeter Wemm 	if (((u_long)pte & PG_V) == 0) {
326e55a0cd8SPeter Wemm 		_kvm_err(kd, kd->program, "_kvm_kvatop: pte not valid");
3272f85bf6eSPeter Wemm 		goto invalid;
328e55a0cd8SPeter Wemm 	}
3292f85bf6eSPeter Wemm 
330e55a0cd8SPeter Wemm 	a = ((u_long)pte & PG_FRAME) + offset;
331e55a0cd8SPeter Wemm 	s =_kvm_pa2off(kd, a, pa);
332e55a0cd8SPeter Wemm 	if (s == 0) {
333e55a0cd8SPeter Wemm 		_kvm_err(kd, kd->program, "_kvm_vatop: address not in dump");
334e55a0cd8SPeter Wemm 		goto invalid;
335e55a0cd8SPeter Wemm 	} else
3362f85bf6eSPeter Wemm 		return (PAGE_SIZE - offset);
3372f85bf6eSPeter Wemm 
3382f85bf6eSPeter Wemm invalid:
339e55a0cd8SPeter Wemm 	_kvm_err(kd, 0, "invalid address (0x%lx)", va);
340e55a0cd8SPeter Wemm 	return (0);
341e55a0cd8SPeter Wemm }
342e55a0cd8SPeter Wemm 
343e55a0cd8SPeter Wemm static int
344e55a0cd8SPeter Wemm _kvm_vatop_pae(kvm_t *kd, u_long va, off_t *pa)
345e55a0cd8SPeter Wemm {
346e55a0cd8SPeter Wemm 	struct vmstate *vm;
347e55a0cd8SPeter Wemm 	uint64_t offset;
348e55a0cd8SPeter Wemm 	uint64_t pte_pa;
349e55a0cd8SPeter Wemm 	uint64_t pde_pa;
350e55a0cd8SPeter Wemm 	uint64_t pde;
351e55a0cd8SPeter Wemm 	uint64_t pte;
352e55a0cd8SPeter Wemm 	u_long pdeindex;
353e55a0cd8SPeter Wemm 	u_long pteindex;
354e55a0cd8SPeter Wemm 	size_t s;
355e55a0cd8SPeter Wemm 	uint64_t a;
356e55a0cd8SPeter Wemm 	off_t ofs;
357e55a0cd8SPeter Wemm 	uint64_t *PTD;
358e55a0cd8SPeter Wemm 
359e55a0cd8SPeter Wemm 	vm = kd->vmst;
360e55a0cd8SPeter Wemm 	PTD = (uint64_t *)vm->PTD;
361e55a0cd8SPeter Wemm 	offset = va & (PAGE_SIZE - 1);
362e55a0cd8SPeter Wemm 
363e55a0cd8SPeter Wemm 	/*
364e55a0cd8SPeter Wemm 	 * If we are initializing (kernel page table descriptor pointer
365e55a0cd8SPeter Wemm 	 * not yet set) then return pa == va to avoid infinite recursion.
366e55a0cd8SPeter Wemm 	 */
367e55a0cd8SPeter Wemm 	if (PTD == 0) {
368e55a0cd8SPeter Wemm 		s = _kvm_pa2off(kd, va, pa);
369e55a0cd8SPeter Wemm 		if (s == 0) {
370e55a0cd8SPeter Wemm 			_kvm_err(kd, kd->program,
371e55a0cd8SPeter Wemm 			    "_kvm_vatop_pae: bootstrap data not in dump");
372e55a0cd8SPeter Wemm 			goto invalid;
373e55a0cd8SPeter Wemm 		} else
374e55a0cd8SPeter Wemm 			return (PAGE_SIZE - offset);
375e55a0cd8SPeter Wemm 	}
376e55a0cd8SPeter Wemm 
377e55a0cd8SPeter Wemm 	pdeindex = va >> PDRSHIFT_PAE;
378e55a0cd8SPeter Wemm 	pde = PTD[pdeindex];
379e55a0cd8SPeter Wemm 	if (((u_long)pde & PG_V) == 0) {
380e55a0cd8SPeter Wemm 		_kvm_err(kd, kd->program, "_kvm_kvatop_pae: pde not valid");
381e55a0cd8SPeter Wemm 		goto invalid;
382e55a0cd8SPeter Wemm 	}
383e55a0cd8SPeter Wemm 
384e55a0cd8SPeter Wemm 	if ((u_long)pde & PG_PS) {
385e55a0cd8SPeter Wemm 	      /*
386e55a0cd8SPeter Wemm 	       * No second-level page table; ptd describes one 2MB page.
387e55a0cd8SPeter Wemm 	       * (We assume that the kernel wouldn't set PG_PS without enabling
388e55a0cd8SPeter Wemm 	       * it cr0).
389e55a0cd8SPeter Wemm 	       */
390e55a0cd8SPeter Wemm #define	PAGE2M_MASK	(NBPDR_PAE - 1)
391e55a0cd8SPeter Wemm #define	PG_FRAME2M	(~PAGE2M_MASK)
392e55a0cd8SPeter Wemm 		pde_pa = ((u_long)pde & PG_FRAME2M) + (va & PAGE2M_MASK);
393e55a0cd8SPeter Wemm 		s = _kvm_pa2off(kd, pde_pa, &ofs);
394add112ffSPeter Wemm 		if (s < sizeof pde) {
395e55a0cd8SPeter Wemm 			_kvm_syserr(kd, kd->program,
396e55a0cd8SPeter Wemm 			    "_kvm_vatop_pae: pde_pa not found");
397e55a0cd8SPeter Wemm 			goto invalid;
398e55a0cd8SPeter Wemm 		}
399e55a0cd8SPeter Wemm 		*pa = ofs;
400e55a0cd8SPeter Wemm 		return (NBPDR_PAE - (va & PAGE2M_MASK));
401e55a0cd8SPeter Wemm 	}
402e55a0cd8SPeter Wemm 
403e55a0cd8SPeter Wemm 	pteindex = (va >> PAGE_SHIFT) & (NPTEPG_PAE-1);
404e55a0cd8SPeter Wemm 	pte_pa = ((uint64_t)pde & PG_FRAME_PAE) + (pteindex * sizeof(pde));
405e55a0cd8SPeter Wemm 
406e55a0cd8SPeter Wemm 	s = _kvm_pa2off(kd, pte_pa, &ofs);
407add112ffSPeter Wemm 	if (s < sizeof pte) {
408e55a0cd8SPeter Wemm 		_kvm_err(kd, kd->program, "_kvm_vatop_pae: pdpe_pa not found");
409e55a0cd8SPeter Wemm 		goto invalid;
410e55a0cd8SPeter Wemm 	}
411e55a0cd8SPeter Wemm 
412e55a0cd8SPeter Wemm 	/* XXX This has to be a physical address read, kvm_read is virtual */
413e55a0cd8SPeter Wemm 	if (lseek(kd->pmfd, ofs, 0) == -1) {
414e55a0cd8SPeter Wemm 		_kvm_syserr(kd, kd->program, "_kvm_vatop_pae: lseek");
415e55a0cd8SPeter Wemm 		goto invalid;
416e55a0cd8SPeter Wemm 	}
417e55a0cd8SPeter Wemm 	if (read(kd->pmfd, &pte, sizeof pte) != sizeof pte) {
418e55a0cd8SPeter Wemm 		_kvm_syserr(kd, kd->program, "_kvm_vatop_pae: read");
419e55a0cd8SPeter Wemm 		goto invalid;
420e55a0cd8SPeter Wemm 	}
421e55a0cd8SPeter Wemm 	if (((uint64_t)pte & PG_V) == 0) {
422e55a0cd8SPeter Wemm 		_kvm_err(kd, kd->program, "_kvm_vatop_pae: pte not valid");
423e55a0cd8SPeter Wemm 		goto invalid;
424e55a0cd8SPeter Wemm 	}
425e55a0cd8SPeter Wemm 
426e55a0cd8SPeter Wemm 	a = ((uint64_t)pte & PG_FRAME_PAE) + offset;
427e55a0cd8SPeter Wemm 	s =_kvm_pa2off(kd, a, pa);
428e55a0cd8SPeter Wemm 	if (s == 0) {
429e55a0cd8SPeter Wemm 		_kvm_err(kd, kd->program,
430e55a0cd8SPeter Wemm 		    "_kvm_vatop_pae: address not in dump");
431e55a0cd8SPeter Wemm 		goto invalid;
432e55a0cd8SPeter Wemm 	} else
433e55a0cd8SPeter Wemm 		return (PAGE_SIZE - offset);
434e55a0cd8SPeter Wemm 
435e55a0cd8SPeter Wemm invalid:
436e55a0cd8SPeter Wemm 	_kvm_err(kd, 0, "invalid address (0x%lx)", va);
4372f85bf6eSPeter Wemm 	return (0);
438f95a0250SRodney W. Grimes }
439f95a0250SRodney W. Grimes 
440f95a0250SRodney W. Grimes int
441da2ef709SPaul Saab _kvm_kvatop(kvm_t *kd, u_long va, off_t *pa)
4422f85bf6eSPeter Wemm {
443e55a0cd8SPeter Wemm 
444e9ca6fe4SPeter Wemm 	if (kd->vmst->minidump)
445e9ca6fe4SPeter Wemm 		return (_kvm_minidump_kvatop(kd, va, pa));
446e55a0cd8SPeter Wemm 	if (ISALIVE(kd)) {
447e55a0cd8SPeter Wemm 		_kvm_err(kd, 0, "vatop called in live kernel!");
448e55a0cd8SPeter Wemm 		return (0);
449e55a0cd8SPeter Wemm 	}
450e55a0cd8SPeter Wemm 	if (kd->vmst->pae)
451e55a0cd8SPeter Wemm 		return (_kvm_vatop_pae(kd, va, pa));
452e55a0cd8SPeter Wemm 	else
45321d54b07SRodney W. Grimes 		return (_kvm_vatop(kd, va, pa));
454f95a0250SRodney W. Grimes }
455