summaryrefslogtreecommitdiffhomepage
path: root/src/loader/impl
diff options
context:
space:
mode:
authorMarvin Borner2021-07-07 19:13:11 +0200
committerMarvin Borner2021-07-07 19:13:11 +0200
commit9a2d5cbdc7972d1d5bfb1ea93f82f66d113faa17 (patch)
tree08ceb8427b64a08c89a5526dfbf8b66c78f9d12c /src/loader/impl
parent6355c3e08c9f4d3db122252abce5837c364d5b81 (diff)
Strange implementations
Diffstat (limited to 'src/loader/impl')
-rw-r--r--src/loader/impl/all.c18
-rw-r--r--src/loader/impl/mb1.c58
-rw-r--r--src/loader/impl/mb2.c100
3 files changed, 159 insertions, 17 deletions
diff --git a/src/loader/impl/all.c b/src/loader/impl/all.c
index a5c93b1..941986d 100644
--- a/src/loader/impl/all.c
+++ b/src/loader/impl/all.c
@@ -2,25 +2,31 @@
#include <impl/all.h>
#include <impl/mb1.h>
+#include <impl/mb2.h>
#include <pnc.h>
-u8 impl_detect(struct dev *dev, const char *path)
+u8 impl_detect(struct cfg_entry *cfg)
{
- if (mb1_detect(dev, path))
+ if (mb1_detect(cfg))
+ return 1;
+
+ if (mb2_detect(cfg))
return 1;
return 0;
}
-void impl_exec(struct dev *dev, const char *path)
+void impl_exec(struct cfg_entry *cfg)
{
- assert(dev->type == DEV_DISK);
+ assert(cfg->dev->type == DEV_DISK);
- switch (dev->p.disk.impl.type) {
+ switch (cfg->impl.type) {
case IMPL_MB1:
- mb1_exec(dev, path);
+ mb1_exec(cfg);
break;
case IMPL_MB2:
+ mb2_exec(cfg);
+ break;
case IMPL_NONE:
default:
panic("Invalid implementation\n");
diff --git a/src/loader/impl/mb1.c b/src/loader/impl/mb1.c
index 765433e..5541c6e 100644
--- a/src/loader/impl/mb1.c
+++ b/src/loader/impl/mb1.c
@@ -3,9 +3,11 @@
#include <elf.h>
#include <impl/mb1.h>
+#include <lib.h>
#include <pnc.h>
-#define MB1_MAGIC 0x1badb002
+// The address where data gets stored
+#define MB1_LOAD_ADDRESS 0x10000
struct multiboot_entry {
u32 magic;
@@ -28,12 +30,43 @@ static u32 mb1_checksum(struct multiboot_entry *entry)
return -(entry->magic + entry->flags);
}
+// Load data into memory and return address (not overlapping
+static u32 mb1_store(void *data, u32 size)
+{
+ static u32 offset = 0;
+ memcpy((void *)MB1_LOAD_ADDRESS, data, size);
+ offset += size;
+ return MB1_LOAD_ADDRESS + (size - offset);
+}
+
+// Load the mb1 structs into memory
+static void mb1_load(struct multiboot_entry *entry)
+{
+ (void)mb1_store;
+ (void)entry;
+}
+
+// Jump to kernel with correct info pointer in eax
+static void mb1_jump(u32 entry, u32 info)
+{
+ log("Jumping. So long and thanks for all the fish!\n");
+
+ // Move and jump!
+ __asm__ volatile("movl $" STRINGIFY(MB1_LOAD_MAGIC) ", %%eax\n\t"
+ "jmpl *%%edi\n\t"
+ :
+ : "D"(entry), "b"(info)
+ : "memory");
+
+ panic("Jumper returned\n");
+}
+
// Detect and verify mb1
-u8 mb1_detect(struct dev *dev, const char *path)
+u8 mb1_detect(struct cfg_entry *cfg)
{
u8 header[8192] = { 0 };
- s32 ret = dev->p.disk.fs.read(path, header, 0, sizeof(header), dev);
+ s32 ret = cfg->dev->p.disk.fs.read(cfg->path, header, 0, sizeof(header), cfg->dev);
if (ret < 12)
return 0;
@@ -54,19 +87,22 @@ u8 mb1_detect(struct dev *dev, const char *path)
if (checksum != entry->checksum)
return 0;
- dev->p.disk.impl.type = IMPL_MB1;
- dev->p.disk.impl.start = entry;
+ cfg->impl.type = IMPL_MB1;
+ cfg->impl.start = entry;
return 1;
}
+#include <pic.h>
+
// Execute mb1 type kernel
-void mb1_exec(struct dev *dev, const char *path)
+void mb1_exec(struct cfg_entry *cfg)
{
- u32 entry = elf_load(dev, path);
- void (*kernel)(void *);
- *(void **)(&kernel) = (void *)entry;
+ u32 entry = elf_load(cfg->dev, cfg->path);
+ mb1_load(cfg->impl.start);
- // TODO: Push mb1 stuff
- kernel(dev);
+ // This is a kind of hacky parameter stack pushing thing, just disable warning :)
+#pragma GCC diagnostic ignored "-Wpedantic"
+ jmp_kernel((void *)mb1_jump, 2, entry, MB1_LOAD_ADDRESS);
+#pragma GCC diagnostic pop
}
diff --git a/src/loader/impl/mb2.c b/src/loader/impl/mb2.c
new file mode 100644
index 0000000..f46310b
--- /dev/null
+++ b/src/loader/impl/mb2.c
@@ -0,0 +1,100 @@
+// MIT License, Copyright (c) 2021 Marvin Borner
+// Everything according to spec
+
+#include <elf.h>
+#include <impl/mb2.h>
+#include <lib.h>
+#include <pnc.h>
+
+// The address where data gets stored
+#define MB2_LOAD_ADDRESS 0x10000
+
+struct multiboot_entry {
+ u32 magic;
+ u32 flags;
+ u32 header_length;
+ u32 checksum; // Everything after that is optional
+ u8 *tags;
+};
+
+// The (really simple) multiboot checksum algorithm
+/* static u32 mb2_checksum(struct multiboot_entry *entry) */
+/* { */
+/* return -(entry->magic + entry->flags); */
+/* } */
+
+// Load data into memory and return address (not overlapping
+static u32 mb2_store(void *data, u32 size)
+{
+ static u32 offset = 0;
+ memcpy((void *)MB2_LOAD_ADDRESS, data, size);
+ offset += size;
+ return MB2_LOAD_ADDRESS + (size - offset);
+}
+
+// Load the mb2 structs into memory
+static void mb2_load(struct multiboot_entry *entry)
+{
+ (void)mb2_store;
+ (void)entry;
+}
+
+// Jump to kernel with correct info pointer in eax
+static void mb2_jump(u32 entry, u32 info)
+{
+ log("Jumping. So long and thanks for all the fish!\n");
+
+ // Move and jump!
+ __asm__ volatile("movl $" STRINGIFY(MB2_LOAD_MAGIC) ", %%eax\n\t"
+ "jmpl *%%edi\n\t"
+ :
+ : "D"(entry), "b"(info)
+ : "memory");
+
+ panic("Jumper returned\n");
+}
+
+// Detect and verify mb2
+u8 mb2_detect(struct cfg_entry *cfg)
+{
+ u8 header[8192] = { 0 };
+
+ s32 ret = cfg->dev->p.disk.fs.read(cfg->path, header, 0, sizeof(header), cfg->dev);
+ if (ret < 12)
+ return 0;
+
+ // Find start of multiboot entry by searching for magic
+ struct multiboot_entry *entry = 0;
+ for (u32 i = 0; i < sizeof(header); i++) {
+ u32 *p = (u32 *)&header[i];
+ if (*p == MB2_MAGIC) {
+ entry = (void *)p;
+ break;
+ }
+ }
+
+ if (!entry)
+ return 0;
+
+ // TODO: mb2 checksum
+ /* u32 checksum = mb2_checksum(entry); */
+ /* if (checksum != entry->checksum) */
+ /* return 0; */
+
+ cfg->impl.type = IMPL_MB2;
+ cfg->impl.start = entry;
+
+ return 1;
+}
+
+// Execute mb2 type kernel
+void mb2_exec(struct cfg_entry *cfg)
+{
+ u32 entry = elf_load(cfg->dev, cfg->path);
+ mb2_load(cfg->impl.start);
+
+ // This is a kind of hacky parameter stack pushing thing, just disable warning :)
+#pragma GCC diagnostic ignored "-Wpedantic"
+ jmp_kernel((void *)mb2_jump, 2, entry, MB2_LOAD_ADDRESS);
+#pragma GCC diagnostic pop
+}