toaruos/kernel/misc/elf.c

198 lines
6.3 KiB
C
Raw Normal View History

/* vim: tabstop=4 shiftwidth=4 noexpandtab
*
2011-04-15 05:32:19 +04:00
* ELF Static Executable Loader
*
* Part of the ToAru OS Kernel
* (C) 2011 Kevin Lange
* Released under the terms of the NCSA License, see the included
* README file for further information.
*/
#include <system.h>
#include <fs.h>
#include <elf.h>
#include <process.h>
#include <logging.h>
#define TMP_ZONE 0x20000000
/**
2011-04-15 05:32:19 +04:00
* Load and execute a static ELF binary.
*
* We make one assumption on the location the binary expects to be loaded
* at: that it be outside of the kernel memory space.
*
* Arguments are passed to the stack of the user application so that they
* can be read properly.
*
* TODO: Environment variables should be loaded somewhere.
*
* HACK: ELF verification isn't complete.
*
* @param path Path to the executable to attempt to execute.
* @param argc Number of arguments (because I'm not counting for you)
* @param argv Pointer to a string of arguments
*/
int
exec(
2011-04-15 05:32:19 +04:00
char * path, /* Path to the executable to run */
int argc, /* Argument count (ie, /bin/echo hello world = 3) */
char ** argv, /* Argument strings (including executable path) */
char ** env /* Environmen variables */
) {
2011-04-16 02:33:19 +04:00
/* Open the file */
fs_node_t * file = kopen(path,0);
if (!file) {
/* Command not found */
return 0;
2011-04-16 02:33:19 +04:00
}
/* Read in the binary contents */
for (uintptr_t x = TMP_ZONE; x < TMP_ZONE + file->length; x += 0x1000) {
alloc_frame(get_page(x, 1, current_directory), 0, 1);
}
Elf32_Header * header = (Elf32_Header *)TMP_ZONE; //(Elf32_Header *)malloc(file->length + 100);
debug_print(NOTICE, "---> Starting load.");
IRQ_RES;
2011-04-16 02:33:19 +04:00
read_fs(file, 0, file->length, (uint8_t *)header);
IRQ_OFF;
debug_print(NOTICE, "---> Finished load.");
current_process->name = malloc(strlen(path) + 1);
memcpy(current_process->name, path, strlen(path) + 1);
2013-03-19 08:52:45 +04:00
current_process->cmdline = argv;
2011-04-16 02:33:19 +04:00
/* Alright, we've read the binary, time to load the loadable sections */
/* Verify the magic */
if ( header->e_ident[0] != ELFMAG0 ||
header->e_ident[1] != ELFMAG1 ||
header->e_ident[2] != ELFMAG2 ||
header->e_ident[3] != ELFMAG3) {
/* What? This isn't an ELF... */
2014-04-06 02:23:17 +04:00
debug_print(ERROR, "Not a valid ELF executable.");
for (uintptr_t x = TMP_ZONE; x < TMP_ZONE + file->length; x += 0x1000) {
free_frame(get_page(x, 0, current_directory));
}
2011-04-16 02:33:19 +04:00
close_fs(file);
return -1;
2011-04-16 02:33:19 +04:00
}
2011-04-16 02:33:19 +04:00
/* Load the loadable segments from the binary */
2012-12-05 08:09:06 +04:00
for (uintptr_t x = 0; x < (uint32_t)header->e_shentsize * header->e_shnum; x += header->e_shentsize) {
2011-04-16 02:33:19 +04:00
/* read a section header */
Elf32_Shdr * shdr = (Elf32_Shdr *)((uintptr_t)header + (header->e_shoff + x));
if (shdr->sh_addr) {
/* If this is a loadable section, load it up. */
if (shdr->sh_addr < current_process->image.entry) {
2011-04-16 02:33:19 +04:00
/* If this is the lowest entry point, store it for memory reasons */
current_process->image.entry = shdr->sh_addr;
2011-04-16 02:33:19 +04:00
}
if (shdr->sh_addr + shdr->sh_size - current_process->image.entry > current_process->image.size) {
2011-04-16 02:33:19 +04:00
/* We also store the total size of the memory region used by the application */
current_process->image.size = shdr->sh_addr + shdr->sh_size - current_process->image.entry;
}
for (uintptr_t i = 0; i < shdr->sh_size + 0x2000; i += 0x1000) {
2011-04-16 02:33:19 +04:00
/* This doesn't care if we already allocated this page */
alloc_frame(get_page(shdr->sh_addr + i, 1, current_directory), 0, 1);
}
2011-05-01 04:09:29 +04:00
if (shdr->sh_type == SHT_NOBITS) {
2011-05-03 06:47:05 +04:00
/* This is the .bss, zero it */
2011-05-01 04:09:29 +04:00
memset((void *)(shdr->sh_addr), 0x0, shdr->sh_size);
} else {
/* Copy the section into memory */
memcpy((void *)(shdr->sh_addr), (void *)((uintptr_t)header + shdr->sh_offset), shdr->sh_size);
}
}
2011-04-16 02:33:19 +04:00
}
2011-04-16 02:33:19 +04:00
/* Store the entry point to the code segment */
uintptr_t entry = (uintptr_t)header->e_entry;
2011-04-15 05:32:19 +04:00
2011-04-16 02:33:19 +04:00
/* Free the space we used for the ELF headers and files */
for (uintptr_t x = TMP_ZONE; x < TMP_ZONE + file->length; x += 0x1000) {
free_frame(get_page(x, 0, current_directory));
}
2011-04-16 02:33:19 +04:00
close_fs(file);
2011-04-09 02:53:52 +04:00
2012-02-17 00:31:40 +04:00
for (uintptr_t stack_pointer = USER_STACK_BOTTOM; stack_pointer < USER_STACK_TOP; stack_pointer += 0x1000) {
2011-04-18 02:44:29 +04:00
alloc_frame(get_page(stack_pointer, 1, current_directory), 0, 1);
}
/* Collect arguments */
int envc = 0;
for (envc = 0; env[envc] != NULL; ++envc);
/* Format auxv */
Elf32_auxv auxv[] = {
{256, 0xDEADBEEF},
{0, 0}
};
int auxvc = 0;
for (auxvc = 0; auxv[auxvc].id != 0; ++auxvc);
uintptr_t heap = current_process->image.entry + current_process->image.size;
2011-04-18 02:44:29 +04:00
alloc_frame(get_page(heap, 1, current_directory), 0, 1);
char ** argv_ = (char **)heap;
2012-03-28 06:36:39 +04:00
heap += sizeof(char *) * (argc + 1);
char ** env_ = (char **)heap;
heap += sizeof(char *) * (envc + 1);
void * auxv_ptr = (void *)heap;
heap += sizeof(Elf32_auxv) * (auxvc);
2011-04-18 02:44:29 +04:00
for (int i = 0; i < argc; ++i) {
alloc_frame(get_page(heap, 1, current_directory), 0, 1);
argv_[i] = (char *)heap;
memcpy((void *)heap, argv[i], strlen(argv[i]) * sizeof(char) + 1);
heap += strlen(argv[i]) + 1;
}
/* Don't forget the NULL at the end of that... */
argv_[argc] = 0;
2011-04-18 02:44:29 +04:00
for (int i = 0; i < envc; ++i) {
alloc_frame(get_page(heap, 1, current_directory), 0, 1);
env_[i] = (char *)heap;
memcpy((void *)heap, env[i], strlen(env[i]) * sizeof(char) + 1);
heap += strlen(env[i]) + 1;
}
env_[envc] = 0;
memcpy(auxv_ptr, auxv, sizeof(Elf32_auxv) * (auxvc));
current_process->image.heap = heap; /* heap end */
current_process->image.heap_actual = heap + (0x1000 - heap % 0x1000);
2012-02-17 00:31:40 +04:00
current_process->image.user_stack = USER_STACK_TOP;
2011-04-18 03:28:40 +04:00
2012-02-08 12:40:44 +04:00
current_process->image.start = entry;
2011-04-16 02:33:19 +04:00
/* Go go go */
2012-02-17 00:31:40 +04:00
enter_user_jmp(entry, argc, argv_, USER_STACK_TOP);
/* We should never reach this code */
2011-04-16 02:33:19 +04:00
return -1;
}
int
system(
char * path, /* Path to the executable to run */
int argc, /* Argument count (ie, /bin/echo hello world = 3) */
char ** argv /* Argument strings (including executable path) */
) {
2013-06-07 05:29:22 +04:00
char ** argv_ = malloc(sizeof(char *) * (argc + 1));
for (int j = 0; j < argc; ++j) {
argv_[j] = malloc((strlen(argv[j]) + 1) * sizeof(char));
memcpy(argv_[j], argv[j], strlen(argv[j]) + 1);
}
argv_[argc] = 0;
char * env[] = {NULL};
set_process_environment((process_t*)current_process, clone_directory(current_directory));
current_directory = current_process->thread.page_directory;
switch_page_directory(current_directory);
2013-06-07 05:29:22 +04:00
exec(path,argc,argv_,env);
debug_print(ERROR, "Failed to execute process!");
kexit(-1);
return -1;
}