summaryrefslogtreecommitdiffhomepage
path: root/src/loader/gdt.c
blob: 61f186d1d69aa05aede0304dbbd78a078b1d09fc (plain) (blame)
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
// MIT License, Copyright (c) 2021 Marvin Borner

#include <gdt.h>

#define GDT_MAX_LIMIT 0xffff // General max access limit without extra nibble
#define GDT_EXTRA_MAX_LIMIT (0x0f) // Extra limit nibble
#define GDT_PRESENT (1 << 7) // Makes the descriptor valid
#define GDT_DESCRIPTOR (1 << 4) // Should be set for code/data segments
#define GDT_EXECUTABLE (1 << 3) // Makes the segment executable (for code segments)
#define GDT_READWRITE (1 << 1) // Makes the segment read/writeable
#define GDT_SIZE (0x40) // Enable 32-Bit
#define GDT_GRANULARITY (0x80) // Enable 4K granularity for limit

static struct gdt_desc gdt_descs[] = {
	// NULL descriptor - Offset 0
	{ 0 },

	/**
	 * These 16-Bit access segments get used by 16-Bit real mode emulation
	 *
	 * Limit + extra limit gets multiplied by 1 (no granularity/size bit)
	 */

	// Standard code segment // Offset 1
	// Access from (0) to (0xffff * 1 = 64KiB)
	{
		.limit = GDT_MAX_LIMIT,
		.base_low = 0x0000,
		.base_mid = 0x00,
		.access = GDT_DESCRIPTOR | GDT_PRESENT | GDT_READWRITE | GDT_EXECUTABLE,
		.flags = 0x00,
		.base_hi = 0x00,
	},

	// Standard data segment // Offset 2
	// Access from (0) to (0xffff * 1 = 64KiB)
	{
		.limit = GDT_MAX_LIMIT,
		.base_low = 0x0000,
		.base_mid = 0x00,
		.access = GDT_DESCRIPTOR | GDT_PRESENT | GDT_READWRITE,
		.flags = 0x00,
		.base_hi = 0x00,
	},

	/**
	 * These 32-Bit full access segments get uses by the bootloader to load
	 * ELF programs no matter whether they want to be loaded to 0-4GiB addr
	 *
	 * Limit + extra limit gets multiplied by 4096 (<< 12) with added 1s
	 */

	// Full access code segment // Offset 3
	// Access from (0) to ((0xfffff << 12) + 0xfff = 4GiB)
	{
		.limit = GDT_MAX_LIMIT,
		.base_low = 0x0000,
		.base_mid = 0x00,
		.access = GDT_DESCRIPTOR | GDT_PRESENT | GDT_READWRITE | GDT_EXECUTABLE,
		.flags = GDT_GRANULARITY | GDT_SIZE | GDT_EXTRA_MAX_LIMIT,
		.base_hi = 0x00,
	},

	// Full access data segment // Offset 4
	// Access from (0) to ((0xfffff << 12) + 0xfff = 4GiB)
	{
		.limit = GDT_MAX_LIMIT,
		.base_low = 0x0000,
		.base_mid = 0x00,
		.access = GDT_DESCRIPTOR | GDT_PRESENT | GDT_READWRITE,
		.flags = GDT_GRANULARITY | GDT_SIZE | GDT_EXTRA_MAX_LIMIT,
		.base_hi = 0x00,
	},
};

REAL struct gdtr gdt = {
	sizeof(gdt_descs) - 1,
	(u32)gdt_descs,
	0,
};