-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpt.c
116 lines (89 loc) · 2.95 KB
/
pt.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
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
//
// Created by Roy Noyman on 28/10/2021.
//
#include "os.h"
int VPN_PARTS_NUMBER = 5;
int MAX_MOVED_BITS = 36;
void map_ppn() {
}
int current_pte_addr(uint64_t vpn, int i) {
int curr_pte_part = vpn >> (MAX_MOVED_BITS - 9 * i);
curr_pte_part = curr_pte_part & 0x1ff;
return curr_pte_part;
}
int check_valid_bit(uint64_t pt_entry) {
return pt_entry & 1;
}
u_int64_t find_current_pte(uint64_t *page_table, uint64_t vpn, int vpn_part) {
int pte_i = current_pte_addr(vpn, vpn_part);
uint64_t curr_pte = page_table[pte_i];
return curr_pte;
}
void store_new_ppn(uint64_t *page_table, uint64_t vpn) {
int i = 0;
int pte_i;
for (i = 0; i < VPN_PARTS_NUMBER - 1; i++) {
uint64_t curr_pte = find_current_pte(page_table, vpn, i);
if (check_valid_bit(curr_pte) == 0) {
pte_i = current_pte_addr(vpn, i);
page_table[pte_i] = (alloc_page_frame() << 12) + 1;
}
page_table = phys_to_virt(curr_pte - 1);
}
}
void page_table_update(uint64_t pt, uint64_t vpn, uint64_t ppn) {
uint64_t phys_pt = pt << 12;
uint64_t *page_table = phys_to_virt(phys_pt);
uint64_t is_mapped = page_table_query(pt, vpn);
uint64_t correct_pte_addr;
if (ppn == NO_MAPPING) {
if (is_mapped == NO_MAPPING) {
return;
} else { //remove last part of vpn in pt
correct_pte_addr = vpn & 0x1ff;
page_table[correct_pte_addr] = 0;
}
} else { //needs to map
uint64_t ppn_to_store = (ppn << 12) + 1;
if (is_mapped == NO_MAPPING) { // MAP ppn on the right place.
store_new_ppn(page_table, vpn);
}
correct_pte_addr = vpn & 0x1ff;
page_table[correct_pte_addr] = ppn_to_store;
}
}
uint64_t page_table_query_rec(uint64_t *page_table, uint64_t vpn, int vpn_part) {
uint64_t curr_pte = find_current_pte(page_table, vpn, vpn_part);
if (check_valid_bit(curr_pte) == 0) {
return NO_MAPPING;
}
if (vpn_part == VPN_PARTS_NUMBER - 1) {
return curr_pte >> 12;
}
page_table = phys_to_virt(curr_pte - 1);
return page_table_query_rec(page_table, vpn, vpn_part + 1);
}
uint64_t page_table_query(uint64_t pt, uint64_t vpn) {
uint64_t phys_pt = pt << 12;
uint64_t *page_table = phys_to_virt(phys_pt);
uint64_t ppn = page_table_query_rec(page_table, vpn, 0);
return ppn;
}
//uint64_t page_table_query(uint64_t pt, uint64_t vpn) {
// uint64_t phys_pt = pt << 12;
// uint64_t *page_table = phys_to_virt(phys_pt);
// int vpn_part = 0;
// uint64_t curr_pte;
// int pte_i;
// for (int i = 0; i < VPN_PARTS_NUMBER - 1; i++) {
// pte_i = current_pte_addr(vpn, i);
// curr_pte = page_table[pte_i];
// if (check_valid_bit(curr_pte) == 0) {
// return NO_MAPPING;
// }
// if (i == VPN_PARTS_NUMBER - 1) {
// return curr_pte >> 12;
// }
// page_table = phys_to_virt(curr_pte - 1);
// }
//}