aboutsummaryrefslogtreecommitdiff
path: root/kernel/multiboot.c
blob: 02a05cf5657655bcdd66c26ee8b733a7479a5a4a (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
// MIT License, Copyright (c) 2021 Marvin Borner

#include <assert.h>
#include <def.h>
#include <drivers/serial.h>
#include <mem.h>
#include <mm.h>
#include <multiboot.h>

PROTECTED static struct multiboot_info *info = NULL;
PROTECTED static u8 vbe_available = 0;
PROTECTED static char vbe[256] = { 0 };

CLEAR static void multiboot_parse_cmdline(const char *line)
{
	const char *start = line;
	for (const char *p = line; p && *p; p++) {
		if (*p == ' ')
			start = p + 1;

		if (memcmp(start, "log", 3) == 0 && !ALPHANUMERIC(start[3])) {
			serial_enable();
			start += 3;
		}
	}
}

CLEAR u32 multiboot_vbe(void)
{
	return vbe_available ? (u32)vbe : 0;
}

CLEAR void multiboot_mmap(void)
{
	assert(info->flags & MULTIBOOT_INFO_MEM_MAP);

	struct multiboot_mmap_entry *mmap = (void *)info->mmap_addr;
	u32 length = info->mmap_length;
	u32 count = length / sizeof(*mmap);

	u32 total = 0;

	for (u32 i = 0; i < count; i++) {
		/* printf("Memory region 0x%x-0x%x\n", mmap->addr, mmap->addr + mmap->len); */
		if (mmap->type == MULTIBOOT_MEMORY_AVAILABLE) {
			total += mmap->len;
			physical_set_free(memory_range_from(mmap->addr, mmap->len));
		} else if (mmap->type == MULTIBOOT_MEMORY_BADRAM) {
			printf("Defect memory at 0x%x-0x%x\n", mmap->addr, mmap->addr + mmap->len);
		} else {
			// Memory is set to 'used' by default
		}

		mmap++;
	}

	physical_set_total(total);
}

CLEAR void multiboot_init(u32 magic, u32 addr)
{
	assert(magic == MULTIBOOT_MAGIC);
	info = (void *)addr;

	if (info->flags & MULTIBOOT_INFO_CMDLINE)
		multiboot_parse_cmdline((const char *)info->cmdline);

	if (info->flags & MULTIBOOT_INFO_VBE_INFO) {
		memcpy(vbe, (void *)info->vbe_mode_info, sizeof(vbe));
		vbe_available = 1;
	}

	serial_print("Kernel was compiled at " __TIME__ " on " __DATE__ "\n");
}