/* * \brief VirtualBox runtime (RT) * \author Norman Feske * \date 2013-08-20 */ /* * Copyright (C) 2013-2017 Genode Labs GmbH * * This file is distributed under the terms of the GNU General Public License * version 2. */ /* Genode includes */ #include #include #include /* libc includes */ #include #include #include /* statfs */ #include #include #include #include /* open */ /* Genode specific Vbox include */ #include "vmm.h" /* libc memory allocator */ #include /* VirtualBox includes */ #include #include static const bool verbose = false; /* * We need an initial buffer currently, since the libc issues during * initialization malloc (dup) calls to the one defined below. At this * point we have not any env pointer. * Additionally static constructors are executed currently before the libc * is done with initialization and so we also have no Env pointer here. */ static char buffer[3 * 1024]; static unsigned buffer_len = 0; static bool initial_memory(void * ptr) { return buffer <= ptr && ptr < buffer + sizeof(buffer); } /* * We cannot use the libc's version of malloc because it does not satisfies * the alignment constraints asserted by 'Runtime/r3/alloc.cpp'. */ static Libc::Mem_alloc_impl * memory() { try { genode_env(); } catch (...) { return nullptr; } static Libc::Mem_alloc_impl mem( genode_env().rm(), genode_env().ram()); return &mem; } extern "C" void *malloc(::size_t size) { if (memory()) return memory()->alloc(size, Genode::log2(RTMEM_ALIGNMENT)); void * ret = buffer + buffer_len; buffer_len += (size + RTMEM_ALIGNMENT - 1) & ~(0ULL + RTMEM_ALIGNMENT - 1); if (buffer_len <= sizeof(buffer)) return ret; struct Not_enough_initial_memory : Genode::Exception { }; throw Not_enough_initial_memory(); } extern "C" void *calloc(::size_t nmemb, ::size_t size) { void *ret = malloc(nmemb*size); if (ret) Genode::memset(ret, 0, nmemb*size); return ret; } extern "C" void free(void *ptr) { if (!initial_memory(ptr)) memory()->free(ptr); } extern "C" void *realloc(void *ptr, ::size_t size) { if (!ptr) return malloc(size); if (!size) { free(ptr); return 0; } unsigned long old_size = size; if (!initial_memory(ptr)) { /* determine size of old block content (without header) */ old_size = memory()->size_at(ptr); /* do not reallocate if new size is less than the current size */ if (size <= old_size) return ptr; } /* allocate new block */ void *new_addr = malloc(size); /* copy content from old block into new block */ if (new_addr) Genode::memcpy(new_addr, ptr, Genode::min(old_size, (unsigned long)size)); /* free old block */ free(ptr); return new_addr; } extern "C" char *getenv(const char *name) { /* * Logging to the pseudo file '/log' is done via the libc plugin provided * by 'logging.cc'. */ if (Genode::strcmp(name, "VBOX_LOG_DEST") == 0 || Genode::strcmp(name, "VBOX_RELEASE_LOG_DEST") == 0) return (char *)"file=log"; if (Genode::strcmp(name, "VBOX_LOG") == 0 || Genode::strcmp(name, "VBOX_RELEASE_LOG") == 0) return (char *)"+rem_disas.e.l.f" "+rem_printf.e.l.f" // "+rem_run.e.l.f" // "+pgm.e.l.f" // "+pdm" // "+cpum.e.l.f" // "+dev_pcnet.e.l.f" // "+dev_pic.e.l.f" // "+dev_apic.e.l.f" // "+usb_mouse.e.l.f" // "+main.e.l3.f" // "+shared_folders.e.l.f" // "+drv_host_serial.e.l.f" // "+dev_audio.e.l.f" // "+dev_hda.e" // "+drv_host_audio.e.l.f" // "+drv_audio.e.l.f" // "+dev_vmm_backdoor.e.l.f" // "+hgcm.e.l.f" // "+dev_vmm.e.l.f" ; /* * see https://www.virtualbox.org/wiki/VBoxLogging for useful tips to * enable useful debugging of Guest additions in Windows/Linux together * with +dev_vmm_backdoor, hgcm and dev_vmm */ if (Genode::strcmp(name, "VBOX_LOG_FLAGS") == 0 || Genode::strcmp(name, "VBOX_RELEASE_LOG_FLAGS") == 0) return (char *)"thread"; //" tsc"; return 0; } extern "C" int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact) { /* * Break infinite loop at 'VBox/Runtime/r3/init.cpp' :451 */ if (oldact) oldact->sa_flags = SA_SIGINFO; return 0; } /* Some dummy implementation for LibC functions */ extern "C" pid_t getpid(void) { if (verbose) Genode::log(__func__, " called - rip ", __builtin_return_address(0)); return 1345; } extern "C" int sigprocmask(int how, const sigset_t *set, sigset_t *oldset) { if (verbose) Genode::log(__func__, " called - rip ", __builtin_return_address(0)); return -1; } extern "C" int _sigaction(int, const struct sigaction *, struct sigaction *) { if (verbose) Genode::log(__func__, " called - rip ", __builtin_return_address(0)); return -1; } extern "C" int futimes(int fd, const struct timeval tv[2]) { Genode::log("%s called - rip %p", __func__, __builtin_return_address(0)); return 0; } extern "C" int lutimes(const char *filename, const struct timeval tv[2]) { Genode::log(__func__, ": called - file '", filename, "' - rip ", __builtin_return_address(0)); return 0; } extern "C" int _sigprocmask() { if (verbose) Genode::log("%s called - rip %p", __func__, __builtin_return_address(0)); return 0; } /** * Used by Shared Folders Guest additions */ extern "C" int statfs(const char *path, struct statfs *buf) { if (!buf) { errno = EFAULT; return -1; } int fd = open(path, 0); if (fd < 0) return fd; struct statvfs result; int res = fstatvfs(fd, &result); close(fd); if (res) return res; Genode::memset(buf, 0, sizeof(*buf)); buf->f_bavail = result.f_bavail; buf->f_bfree = result.f_bfree; buf->f_blocks = result.f_blocks; buf->f_ffree = result.f_ffree; buf->f_files = result.f_files; buf->f_bsize = result.f_bsize; bool show_warning = !buf->f_bsize || !buf->f_blocks || !buf->f_bavail; if (!buf->f_bsize) buf->f_bsize = 4096; if (!buf->f_blocks) buf->f_blocks = 128 * 1024; if (!buf->f_bavail) buf->f_bavail = buf->f_blocks; if (show_warning) Genode::warning("statfs provides bogus values for '", path, "' (probably a shared folder)"); return res; } extern "C" long pathconf(char const *path, int name) { if (name == _PC_NAME_MAX) return 255; Genode::error("pathconf does not support config option ", name); errno = EINVAL; return -1; }