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
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
use core::fmt;

use tracing::info;

use super::virtual_memory_mapper;

extern "C" {
    static begin: usize;
    static end: usize;
    static text_end: usize;
    static rodata_end: usize;
    static data_end: usize;
    static stack_guard_page: usize;
    static __eh_frame: usize;
}

// The virtual address of the kernel
// these are information variables, showing the memory mapping of the kernel
pub const KERNEL_BASE: usize = 0xFFFF_FFFF_8000_0000;
// memory extended start (1MB)
pub const EXTENDED_OFFSET: usize = 0x10_0000;
pub const KERNEL_LINK: usize = KERNEL_BASE + EXTENDED_OFFSET;
// 128MB (from KERNEL_BASE), and this indicates the address of the end of the kernel
// every memory used in the kernel, allocated or no, comes from the kernel memory
pub const KERNEL_MAPPED_SIZE: usize = 0x800_0000;
pub const KERNEL_END: usize = KERNEL_BASE + KERNEL_MAPPED_SIZE;

// The heap of the kernel
// this is mapped from the physical memory of the kernel
// so we are using the physical pages from the kernel space.
pub const KERNEL_HEAP_BASE: usize = KERNEL_END;
pub const KERNEL_HEAP_SIZE: usize = 0x100_0000; // 16MB

// The size of the stack for interrupt handlers
pub const INTR_STACK_SIZE: usize = PAGE_4K * 32;
pub const INTR_STACK_EMPTY_SIZE: usize = PAGE_4K;
pub const INTR_STACK_ENTRY_SIZE: usize = INTR_STACK_SIZE + INTR_STACK_EMPTY_SIZE;
pub const INTR_STACK_BASE: usize = KERNEL_HEAP_BASE + KERNEL_HEAP_SIZE;
pub const INTR_STACK_COUNT: usize = 7;
// we are going to setup a spacing at the end of the stack, so that we can detect stack overflows
pub const INTR_STACK_TOTAL_SIZE: usize = INTR_STACK_ENTRY_SIZE * INTR_STACK_COUNT;

// extra space that we can make virtual memory to when we don't care where we want to map it
// this is only in kernel space, as userspace programs should be mapped into the rest of the memory range
// that is below `KERNEL_BASE`
pub const KERNEL_EXTRA_MEMORY_BASE: usize = INTR_STACK_BASE + INTR_STACK_TOTAL_SIZE;
// to avoid overflow stuff, we don't use the last page
pub const KERNEL_LAST_POSSIBLE_ADDR: usize = 0xFFFF_FFFF_FFFF_F000;
pub const KERNEL_EXTRA_MEMORY_SIZE: usize = KERNEL_LAST_POSSIBLE_ADDR - KERNEL_EXTRA_MEMORY_BASE;

// Kernel Data specific to each process (will be mapped differently for each process)
pub const KERNEL_PROCESS_VIRTUAL_ADDRESS_START: usize =
    virtual_memory_mapper::KERNEL_PROCESS_VIRTUAL_ADDRESS_START;
pub const PROCESS_KERNEL_STACK_GUARD: usize = PAGE_4K;
// process specific kernel stack, this will be where the process is running while in the kernel
// the process can be interrupted while in the kernel, so we want to save it into a specific stack
// space so that other processes don't override it when being run
pub const PROCESS_KERNEL_STACK_BASE: usize =
    KERNEL_PROCESS_VIRTUAL_ADDRESS_START + PROCESS_KERNEL_STACK_GUARD;
pub const PROCESS_KERNEL_STACK_SIZE: usize = PAGE_4K * 64;
pub const PROCESS_KERNEL_STACK_END: usize = PROCESS_KERNEL_STACK_BASE + PROCESS_KERNEL_STACK_SIZE;

#[allow(dead_code)]
pub const KB: usize = 0x400;
pub const MB: usize = 0x100_000;
pub const GB: usize = 0x400_00000;
pub const PAGE_4K: usize = 0x1000;
pub const PAGE_2M: usize = 0x20_0000;

pub fn kernel_elf_end() -> usize {
    (unsafe { &end } as *const usize as usize)
}

#[allow(dead_code)]
pub fn kernel_elf_size() -> usize {
    (unsafe { &end } as *const usize as usize) - (unsafe { &begin } as *const usize as usize)
}

pub fn kernel_text_end() -> usize {
    (unsafe { &text_end } as *const usize as usize)
}

pub fn kernel_elf_rodata_end() -> usize {
    (unsafe { &rodata_end } as *const usize as usize)
}

pub fn kernel_elf_data_end() -> usize {
    (unsafe { &data_end } as *const usize as usize)
}

pub fn stack_guard_page_ptr() -> usize {
    (unsafe { &stack_guard_page } as *const usize as usize)
}

pub fn eh_frame_start() -> usize {
    (unsafe { &__eh_frame } as *const usize as usize)
}

pub fn eh_frame_end() -> usize {
    (unsafe { &rodata_end } as *const usize as usize)
}

pub trait AlignMem: Sized {
    fn align_up(self, alignment: usize) -> Self;
    fn align_down(self, alignment: usize) -> Self;
    fn is_aligned(&self, alignment: usize) -> bool;
    fn align_range(self, size: usize, alignment: usize) -> (Self, usize, usize);
}

macro_rules! impl_align_mem {
    ($t:ty) => {
        impl AlignMem for $t {
            #[inline(always)]
            fn align_up(self, alignment: usize) -> Self {
                (self + (alignment as $t) - 1) & !((alignment as $t) - 1)
            }

            #[inline(always)]
            fn align_down(self, alignment: usize) -> Self {
                self & !((alignment as $t) - 1)
            }

            #[inline(always)]
            fn is_aligned(&self, alignment: usize) -> bool {
                (self & ((alignment as $t) - 1)) == 0
            }

            #[inline(always)]
            fn align_range(self, size: usize, alignment: usize) -> (Self, usize, usize) {
                let addr_end = self + size as $t;
                let start_aligned = self.align_down(alignment);
                let end_aligned = addr_end.align_up(alignment);
                let size: usize = (end_aligned - start_aligned).try_into().unwrap();
                assert!(size > 0);
                assert!(size.is_aligned(alignment));
                let offset = (self - start_aligned).try_into().unwrap();

                (start_aligned, size, offset)
            }
        }
    };
}

impl_align_mem!(usize);
impl_align_mem!(u64);

pub fn align_up<T: AlignMem>(addr: T, alignment: usize) -> T {
    addr.align_up(alignment)
}

pub fn align_down<T: AlignMem>(addr: T, alignment: usize) -> T {
    addr.align_down(alignment)
}

pub fn is_aligned<T: AlignMem>(addr: T, alignment: usize) -> bool {
    addr.is_aligned(alignment)
}

pub fn align_range<T: AlignMem>(addr: T, size: usize, alignment: usize) -> (T, usize, usize) {
    addr.align_range(size, alignment)
}

#[inline(always)]
pub const fn virtual2physical(addr: usize) -> u64 {
    debug_assert!(addr >= KERNEL_BASE && addr <= KERNEL_BASE + KERNEL_MAPPED_SIZE);
    (addr - KERNEL_BASE) as u64
}

#[inline(always)]
pub const fn physical2virtual(addr: u64) -> usize {
    debug_assert!(addr < KERNEL_MAPPED_SIZE as u64);
    addr as usize + KERNEL_BASE
}

pub fn display_kernel_map() {
    info!("Kernel map:");
    let nothing = KERNEL_BASE..KERNEL_LINK;
    let kernel_elf_end = align_up(kernel_elf_end(), PAGE_4K);
    let kernel_elf = KERNEL_LINK..kernel_elf_end;
    let kernel_elf_text = KERNEL_LINK..kernel_text_end();
    let kernel_elf_rodata = kernel_text_end()..kernel_elf_rodata_end();
    let kernel_elf_data = kernel_elf_rodata_end()..kernel_elf_data_end();
    let kernel_elf_bss = kernel_elf_data_end()..kernel_elf_end;
    let kernel_physical_allocator_low = kernel_elf_end..KERNEL_END;
    let kernel_heap = KERNEL_HEAP_BASE..KERNEL_HEAP_BASE + KERNEL_HEAP_SIZE;
    let interrupt_stack = INTR_STACK_BASE..INTR_STACK_BASE + INTR_STACK_TOTAL_SIZE;
    let kernel_extra_memory =
        KERNEL_EXTRA_MEMORY_BASE..KERNEL_EXTRA_MEMORY_BASE + KERNEL_EXTRA_MEMORY_SIZE;

    info!(
        "  range={:016x}..{:016x}, len={:4}  nothing",
        nothing.start,
        nothing.end,
        MemSize(nothing.len())
    );
    info!(
        "  range={:016x}..{:016x}, len={:4}  kernel elf",
        kernel_elf.start,
        kernel_elf.end,
        MemSize(kernel_elf.len())
    );
    // inner map for the elf
    info!(
        "    range={:016x}..{:016x}, len={:4}  kernel elf text",
        kernel_elf_text.start,
        kernel_elf_text.end,
        MemSize(kernel_elf_text.len())
    );
    info!(
        "    range={:016x}..{:016x}, len={:4}  kernel elf rodata",
        kernel_elf_rodata.start,
        kernel_elf_rodata.end,
        MemSize(kernel_elf_rodata.len())
    );
    info!(
        "    range={:016x}..{:016x}, len={:4}  kernel elf data",
        kernel_elf_data.start,
        kernel_elf_data.end,
        MemSize(kernel_elf_data.len())
    );
    info!(
        "    range={:016x}..{:016x}, len={:4}  kernel elf bss",
        kernel_elf_bss.start,
        kernel_elf_bss.end,
        MemSize(kernel_elf_bss.len())
    );
    info!(
        "  range={:016x}..{:016x}, len={:4}  kernel physical allocator low",
        kernel_physical_allocator_low.start,
        kernel_physical_allocator_low.end,
        MemSize(kernel_physical_allocator_low.len())
    );
    info!(
        "  range={:016x}..{:016x}, len={:4}  kernel heap",
        kernel_heap.start,
        kernel_heap.end,
        MemSize(kernel_heap.len())
    );
    info!(
        "  range={:016x}..{:016x}, len={:4}  interrupt stack",
        interrupt_stack.start,
        interrupt_stack.end,
        MemSize(interrupt_stack.len())
    );
    info!(
        "  range={:016x}..{:016x}, len={:4}  kernel extra (virtual space)",
        kernel_extra_memory.start,
        kernel_extra_memory.end,
        MemSize(kernel_extra_memory.len())
    );

    // number of bytes approx used from physical memory
    info!(
        "whole kernel physical size (startup/low): {}",
        MemSize(KERNEL_END - KERNEL_BASE)
    );
    // total addressable virtual kernel memory
    info!(
        "whole kernel size: {}",
        MemSize(usize::MAX - KERNEL_BASE + 1)
    );
}

#[repr(transparent)]
pub struct MemSize<T>(pub T);

impl<T> fmt::Display for MemSize<T>
where
    T: TryInto<u64> + Copy,
    <T as TryInto<u64>>::Error: fmt::Debug,
{
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        // find the best unit
        let mut size = self.0.try_into().unwrap();
        let mut remaining = 0;
        let mut unit = "B";
        if size >= 1024 {
            remaining = size % 1024;
            size /= 1024;
            unit = "KB";
        }
        if size >= 1024 {
            remaining = size % 1024;
            size /= 1024;
            unit = "MB";
        }
        if size >= 1024 {
            remaining = size % 1024;
            size /= 1024;
            unit = "GB";
        }
        if size >= 1024 {
            remaining = size % 1024;
            size /= 1024;
            unit = "TB";
        }
        if size >= 1024 {
            remaining = size % 1024;
            size /= 1024;
            unit = "PB";
        }

        size.fmt(f).and_then(|_| {
            let remaining = remaining * 100 / 1024;
            write!(f, ".{remaining:02}")?;
            write!(f, "{unit}")
        })
    }
}

impl<T> fmt::Debug for MemSize<T>
where
    T: TryInto<u64> + Copy,
    <T as TryInto<u64>>::Error: fmt::Debug,
{
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        fmt::Display::fmt(self, f)
    }
}