From e9df5fd6638a483acc8ddbddc623a4e26e087935 Mon Sep 17 00:00:00 2001 From: apio Date: Wed, 12 Oct 2022 18:04:20 +0200 Subject: [PATCH] exec: Copy pathname into kernel memory, since the user memory where it resides is going to be freed --- kernel/src/sys/exec.cpp | 18 +++++++++++++----- 1 file changed, 13 insertions(+), 5 deletions(-) diff --git a/kernel/src/sys/exec.cpp b/kernel/src/sys/exec.cpp index f7a81160..b7c1313f 100644 --- a/kernel/src/sys/exec.cpp +++ b/kernel/src/sys/exec.cpp @@ -4,6 +4,8 @@ #include "errno.h" #include "interrupts/Interrupts.h" #include "memory/MemoryManager.h" +#include "std/stdlib.h" +#include "std/string.h" #include "sys/elf/ELFLoader.h" #include "thread/Scheduler.h" @@ -31,22 +33,28 @@ void sys_exec(Context* context, const char* pathname) return; } - // FIXME: This should be done later, but since there is a very high chance that loading the executed program's ELF - // image will overwrite ours, we have to do it here. + char* kpathname = strdup( + pathname); // Since we are going to free the original task's memory, we cannot use anything coming from it. + + // FIXME: This should be done later, but since there is a very high chance that loading the executed program's + // ELF image will overwrite ours, we have to do it here. ELFLoader::release_elf_image(Scheduler::current_task()->image); // FIXME: Check the ELF image is valid before loading it into memory. This will allow us to first check, then free // the previous image, then load, which should reduce the chances of loading failing to almost zero. - ELFImage* image = ELFLoader::load_elf_from_filesystem(pathname); + ELFImage* image = ELFLoader::load_elf_from_filesystem(kpathname); if (!image) { MemoryManager::release_pages((void*)allocated_stack, TASK_PAGES_IN_STACK); Scheduler::current_task()->image = nullptr; - kwarnln("exec(): ERROR: Failed to load program. Previous program has already been freed, thus cannot " - "return to it."); + kfree(kpathname); + kerrorln("exec(): ERROR: Failed to load program. Previous program has already been freed, thus cannot " + "return to it."); return Scheduler::task_exit(context, -255); } + kfree(kpathname); + Interrupts::disable(); ASSERT(!Interrupts::are_enabled()); // This part is pretty sensitive.