-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathfindphys.c
59 lines (48 loc) · 1.87 KB
/
findphys.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
#include "ptedit_header.h"
#include <x86linux/helper.h>
int main(int argc, char *argv[]) {
if (argc != 6) {
fprintf(stderr,
"usage: %s DEC_PID HEX_VIRT_START HEX_VIRT_END HEX_PHYS_ADDR "
"SHOW_PTE\n",
argv[0]);
exit(EXIT_FAILURE);
}
int pid = atoi(argv[1]);
uintptr_t virt_start = strtoul(argv[2], NULL, 16);
uintptr_t virt_end = strtoul(argv[3], NULL, 16);
loff_t phys_addr = strtol(argv[4], NULL, 16);
int show_pte = atoi(argv[5]);
log_init(NULL, 0, -1, 1, 0);
log_enable(LOG_DEBUG);
log_abort_on_error(ptedit_init());
size_t pagesize = ptedit_get_pagesize();
log_abort_if_false(!(virt_start % pagesize));
log_abort_if_false(!(virt_end % pagesize));
log_abort_if_false(!(phys_addr % pagesize));
size_t size = virt_end - virt_start;
log_abort_if_false(!(size % pagesize));
size_t nr_pages = size / pagesize + 1;
log_warning("Searching %lu page(s)...", nr_pages);
for (size_t i = 0; i < nr_pages; ++i) {
uintptr_t virt_addr = virt_start + pagesize * i;
ptedit_entry_t vm = ptedit_resolve(address_cast(virt_addr), pid);
log_abort_if_false(vm.pgd);
size_t found_pfn = ptedit_get_pfn(vm.pte);
loff_t found_phys_addr = found_pfn * pagesize;
if (phys_addr == found_phys_addr) {
log_info("FOUND [virt: 0x%lx] (pfn: 0x%lx) phys: 0x%lx", virt_addr,
found_pfn, found_phys_addr);
if (show_pte) {
log_abort_if_false(!x86_test_bit(&vm.pte, PTEDIT_PAGE_BIT_SOFTW4));
log_abort_if_false(!x86_test_bit(&vm.pte, PTEDIT_PAGE_BIT_PKEY_BIT0));
log_abort_if_false(!x86_test_bit(&vm.pte, PTEDIT_PAGE_BIT_PKEY_BIT1));
log_abort_if_false(!x86_test_bit(&vm.pte, PTEDIT_PAGE_BIT_PKEY_BIT2));
log_abort_if_false(!x86_test_bit(&vm.pte, PTEDIT_PAGE_BIT_PKEY_BIT3));
ptedit_print_entry_t(vm);
}
}
}
ptedit_cleanup();
return EXIT_SUCCESS;
}