/*
* \brief VirtualBox runtime (RT)
* \author Norman Feske
* \date 2013-08-20
*/
/*
* Copyright (C) 2013 Genode Labs GmbH
*
* This file is distributed under the terms of the GNU General Public License
* version 2.
*/
/* Genode includes */
#include
#include
#include
/* VirtualBox includes */
#include
#include
#include
#include
#include
#include
#include
class Avl_ds : public Genode::Avl_node
{
private:
Genode::Ram_dataspace_capability _ds;
Genode::addr_t _virt;
public:
Avl_ds(Genode::Ram_dataspace_capability ds, void * virt)
: _ds(ds), _virt(reinterpret_cast(virt))
{ }
~Avl_ds() {
Genode::env()->ram_session()->free(_ds);
}
bool higher(Avl_ds *e) { return e->_virt > _virt; }
Avl_ds *find(Genode::addr_t virt)
{
if (virt == _virt) return this;
Avl_ds *obj = this->child(virt > _virt);
return obj ? obj->find(virt) : 0;
}
};
static Genode::Avl_tree runtime_ds;
static void *alloc_mem(size_t cb, const char *pszTag, bool executable = false)
{
using namespace Genode;
try {
Ram_dataspace_capability ds = env()->ram_session()->alloc(cb);
Assert(ds.valid());
size_t const whole_size = 0;
Genode::off_t const offset = 0;
bool const any_addr = false;
void * any_local_addr = nullptr;
void * local_addr = env()->rm_session()->attach(ds, whole_size, offset,
any_addr, any_local_addr,
executable);
if (!local_addr)
PERR("%s size=0x%zx, tag=%s -> %p", __func__, cb, pszTag, local_addr);
Assert(local_addr);
runtime_ds.insert(new (env()->heap()) Avl_ds(ds, local_addr));
return local_addr;
} catch (...) {
Assert(!"Could not allocate RTMem* memory ");
return nullptr;
}
}
/*
* Called by the recompiler to allocate executable RAM
*/
void *RTMemExecAllocTag(size_t cb, const char *pszTag) RT_NO_THROW
{
return alloc_mem(cb, pszTag, true);
}
void *RTMemPageAllocZTag(size_t cb, const char *pszTag) RT_NO_THROW
{
/*
* The RAM dataspace freshly allocated by 'RTMemExecAllocTag' is zeroed
* already.
*/
return alloc_mem(cb, pszTag);
}
void *RTMemPageAllocTag(size_t cb, const char *pszTag) RT_NO_THROW
{
return alloc_mem(cb, pszTag);
}
void RTMemPageFree(void *pv, size_t cb) RT_NO_THROW
{
Avl_ds * ds_obj = runtime_ds.first();
if (ds_obj)
ds_obj = ds_obj->find(reinterpret_cast(pv));
if (ds_obj) {
runtime_ds.remove(ds_obj);
destroy(Genode::env()->heap(), ds_obj);
}
else
PERR("%s unknown memory region %p+%zx", __func__, pv, cb);
}
#include
uint32_t RTBldCfgVersionMajor(void) { return VBOX_VERSION_MAJOR; }
uint32_t RTBldCfgVersionMinor(void) { return VBOX_VERSION_MINOR; }
uint32_t RTBldCfgVersionBuild(void) { return VBOX_VERSION_BUILD; }
uint32_t RTBldCfgRevision(void) { return ~0; }
extern "C" DECLHIDDEN(int) rtProcInitExePath(char *pszPath, size_t cchPath)
{
Genode::strncpy(pszPath, "/virtualbox", cchPath);
return 0;
}