diff --git a/src/arch/x86/vm.c b/src/arch/x86/vm.c index 3608919..bb988ef 100644 --- a/src/arch/x86/vm.c +++ b/src/arch/x86/vm.c @@ -6,6 +6,7 @@ #include #include #include +#include #include #include @@ -131,7 +132,14 @@ int vm_arch_load_image(vm_t *v, void *data, size_t datasz) void *kernel = ((uint8_t *) v->mem) + 0x100000; memset(boot, 0, sizeof(struct boot_params)); - memmove(boot, data, sizeof(struct boot_params)); + /* According to https://www.kernel.org/doc/html/next/x86/boot.html, + * the first step in loading a Linux kernel should be to setup the boot + * parameters (struct boot_params) and initialize it to all zero. Then, + * the setup header at offset 0x01f1 of kernel image on should be loaded + * into struct boot_params. */ + memmove((void *) ((uintptr_t) boot + offsetof(struct boot_params, hdr)), + (void *) ((uintptr_t) data + offsetof(struct boot_params, hdr)), + sizeof(struct setup_header)); size_t setup_sectors = boot->hdr.setup_sects; size_t setupsz = (setup_sectors + 1) * 512;