/*
* \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;
}
extern "C" int siginterrupt(int, int)
{
Genode::Thread * thread = Genode::Thread::myself();
Genode::warning(__func__, " called, caller=", thread ? thread->name() : "");
return 0;
}