#include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <string.h> #include <console.h> #include <dprintf.h> #include <com32.h> #include <syslinux/adv.h> #include <syslinux/config.h> #include <setjmp.h> #include <linux/list.h> #include <netinet/in.h> #include <sys/cpu.h> #include <core.h> #include <fcntl.h> #include <sys/file.h> #include <fs.h> #include <ctype.h> #include <alloca.h> #include <sys/exec.h> #include <sys/module.h> #include "common.h" extern char __dynstr_start[]; extern char __dynstr_end[], __dynsym_end[]; extern char __dynsym_start[]; extern char __got_start[]; extern Elf_Dyn __dynamic_start[]; extern Elf_Word __gnu_hash_start[]; extern char __module_start[]; struct elf_module core_module = { .name = "(core)", .shallow = true, .required = LIST_HEAD_INIT((core_module.required)), .dependants = LIST_HEAD_INIT((core_module.dependants)), .list = LIST_HEAD_INIT((core_module.list)), .module_addr = (void *)0x0, .ghash_table = __gnu_hash_start, .str_table = __dynstr_start, .sym_table = __dynsym_start, .got = __got_start, .dyn_table = __dynamic_start, .syment_size = sizeof(Elf_Sym), }; /* * Initializes the module subsystem by taking the core module * (preinitialized shallow module) and placing it on top of the * modules_head_list. */ void init_module_subsystem(struct elf_module *module) { list_add(&module->list, &modules_head); } __export int start_ldlinux(int argc, char **argv) { int rv; again: rv = spawn_load(LDLINUX, argc, argv); if (rv == EEXIST) { /* * If a COM32 module calls execute() we may need to * unload all the modules loaded since ldlinux.*, * and restart initialisation. This is especially * important for config files. * * But before we do that, try our best to make sure * that spawn_load() is gonna succeed, e.g. that we * can find LDLINUX it in PATH. */ struct elf_module *ldlinux; FILE *f; f = findpath(LDLINUX); if (!f) return ENOENT; fclose(f); ldlinux = unload_modules_since(LDLINUX); /* * Finally unload LDLINUX. * * We'll reload it when we jump to 'again' which will * cause all the initialsation steps to be executed * again. */ module_unload(ldlinux); goto again; } return rv; } /* note to self: do _*NOT*_ use static key word on this function */ void load_env32(com32sys_t * regs __unused) { struct file_info *fp; int fd; char *argv[] = { LDLINUX, NULL }; char realname[FILENAME_MAX]; size_t size; static const char *search_directories[] = { "/boot/isolinux", "/isolinux", "/boot/syslinux", "/syslinux", "/", NULL }; static const char *filenames[] = { LDLINUX, NULL }; dprintf("Starting %s elf module subsystem...\n", ELF_MOD_SYS); if (strlen(CurrentDirName) && !path_add(CurrentDirName)) { printf("Couldn't allocate memory for PATH\n"); goto out; } size = (size_t)__dynstr_end - (size_t)__dynstr_start; core_module.strtable_size = size; size = (size_t)__dynsym_end - (size_t)__dynsym_start; core_module.symtable_size = size; core_module.base_addr = (Elf_Addr)__module_start; init_module_subsystem(&core_module); start_ldlinux(1, argv); /* * If we failed to load LDLINUX it could be because our * current working directory isn't the install directory. Try * a bit harder to find LDLINUX. If search_dirs() succeeds * in finding LDLINUX it will set the cwd. */ fd = opendev(&__file_dev, NULL, O_RDONLY); if (fd < 0) goto out; fp = &__file_info[fd]; if (!search_dirs(&fp->i.fd, search_directories, filenames, realname)) { char path[FILENAME_MAX]; /* * search_dirs() sets the current working directory if * it successfully opens the file. Add the directory * in which we found ldlinux.* to PATH. */ if (!core_getcwd(path, sizeof(path))) goto out; if (!path_add(path)) { printf("Couldn't allocate memory for PATH\n"); goto out; } start_ldlinux(1, argv); } out: writestr("\nFailed to load "); writestr(LDLINUX); } static const char *__cmdline; __export const char *com32_cmdline(void) { return __cmdline; } __export int create_args_and_load(char *cmdline) { char *p, **argv; int argc; int i; if (!cmdline) return -1; for (argc = 0, p = cmdline; *p; argc++) { /* Find the end of this arg */ while(*p && !isspace(*p)) p++; /* * Now skip all whitespace between arguments. */ while (*p && isspace(*p)) p++; } /* * Generate a copy of argv on the stack as this is * traditionally where process arguments go. * * argv[0] must be the command name. Remember to allocate * space for the sentinel NULL. */ argv = alloca((argc + 1) * sizeof(char *)); for (i = 0, p = cmdline; i < argc; i++) { char *start; int len = 0; start = p; /* Find the end of this arg */ while(*p && !isspace(*p)) { p++; len++; } argv[i] = malloc(len + 1); strncpy(argv[i], start, len); argv[i][len] = '\0'; /* * Now skip all whitespace between arguments. */ while (*p && isspace(*p)) p++; /* * Point __cmdline at "argv[1] ... argv[argc-1]" */ if (i == 0) __cmdline = p; } /* NUL-terminate */ argv[argc] = NULL; return spawn_load(argv[0], argc, argv); } void pm_env32_run(com32sys_t *regs) { char *cmdline; cmdline = MK_PTR(regs->es, regs->ebx.w[0]); if (create_args_and_load(cmdline) < 0) printf("Failed to run com32 module\n"); }