2013-08-21 09:37:21 +00:00
|
|
|
/*
|
|
|
|
* \brief VirtualBox runtime (RT)
|
|
|
|
* \author Norman Feske
|
|
|
|
* \date 2013-08-20
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
2017-02-20 12:23:52 +00:00
|
|
|
* Copyright (C) 2013-2017 Genode Labs GmbH
|
2013-08-21 09:37:21 +00:00
|
|
|
*
|
|
|
|
* This file is distributed under the terms of the GNU General Public License
|
|
|
|
* version 2.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Genode includes */
|
base: avoid use of deprecated base/printf.h
Besides adapting the components to the use of base/log.h, the patch
cleans up a few base headers, i.e., it removes unused includes from
root/component.h, specifically base/heap.h and
ram_session/ram_session.h. Hence, components that relied on the implicit
inclusion of those headers have to manually include those headers now.
While adjusting the log messages, I repeatedly stumbled over the problem
that printing char * arguments is ambiguous. It is unclear whether to
print the argument as pointer or null-terminated string. To overcome
this problem, the patch introduces a new type 'Cstring' that allows the
caller to express that the argument should be handled as null-terminated
string. As a nice side effect, with this type in place, the optional len
argument of the 'String' class could be removed. Instead of supplying a
pair of (char const *, size_t), the constructor accepts a 'Cstring'.
This, in turn, clears the way let the 'String' constructor use the new
output mechanism to assemble a string from multiple arguments (and
thereby getting rid of snprintf within Genode in the near future).
To enforce the explicit resolution of the char * ambiguity, the 'char *'
overload of the 'print' function is marked as deleted.
Issue #1987
2016-07-13 17:07:09 +00:00
|
|
|
#include <base/log.h>
|
2016-06-08 18:15:27 +00:00
|
|
|
#include <base/allocator_avl.h>
|
2013-08-21 09:37:21 +00:00
|
|
|
|
2016-06-14 08:18:32 +00:00
|
|
|
|
2013-08-21 09:37:21 +00:00
|
|
|
/* VirtualBox includes */
|
|
|
|
#include <iprt/initterm.h>
|
|
|
|
#include <iprt/mem.h>
|
|
|
|
#include <iprt/err.h>
|
2014-09-23 11:01:47 +00:00
|
|
|
#include <iprt/assert.h>
|
2013-08-21 09:37:21 +00:00
|
|
|
#include <iprt/semaphore.h>
|
|
|
|
#include <iprt/time.h>
|
|
|
|
#include <internal/iprt.h>
|
|
|
|
|
2016-10-04 09:07:59 +00:00
|
|
|
#include "mm.h"
|
2017-01-05 12:45:37 +00:00
|
|
|
#include "vmm.h"
|
2016-10-04 09:07:59 +00:00
|
|
|
|
|
|
|
enum {
|
|
|
|
MEMORY_MAX = 64 * 1024 * 1024,
|
|
|
|
MEMORY_CACHED = 16 * 1024 * 1024,
|
|
|
|
};
|
|
|
|
|
2016-06-08 18:15:27 +00:00
|
|
|
class Avl_ds : public Genode::Avl_node<Avl_ds>
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
|
|
|
|
Genode::Ram_dataspace_capability _ds;
|
|
|
|
Genode::addr_t _virt;
|
2016-06-14 08:18:32 +00:00
|
|
|
Genode::addr_t const _size;
|
|
|
|
Genode::addr_t _used_size;
|
|
|
|
|
|
|
|
bool _inuse = true;
|
|
|
|
|
|
|
|
static Genode::addr_t _mem_allocated;
|
|
|
|
static Genode::addr_t _mem_unused;
|
|
|
|
|
|
|
|
static Genode::Avl_tree<Avl_ds> _unused_ds;
|
|
|
|
static Genode::Avl_tree<Avl_ds> _runtime_ds;
|
2016-06-08 18:15:27 +00:00
|
|
|
|
|
|
|
public:
|
|
|
|
|
2016-06-14 08:18:32 +00:00
|
|
|
static Genode::addr_t hit;
|
|
|
|
static Genode::addr_t hit_coarse;
|
|
|
|
|
|
|
|
Avl_ds(Genode::Ram_dataspace_capability ds, void * virt,
|
|
|
|
Genode::addr_t size)
|
|
|
|
:
|
|
|
|
_ds(ds), _virt(reinterpret_cast<Genode::addr_t>(virt)),
|
|
|
|
_size(size), _used_size(_size)
|
|
|
|
{
|
|
|
|
_mem_allocated += _size;
|
|
|
|
_runtime_ds.insert(this);
|
|
|
|
}
|
2016-06-08 18:15:27 +00:00
|
|
|
|
|
|
|
~Avl_ds() {
|
2016-06-14 08:18:32 +00:00
|
|
|
Assert (!_inuse);
|
|
|
|
|
|
|
|
_unused_ds.remove(this);
|
|
|
|
_mem_unused -= _size;
|
|
|
|
_mem_allocated -= _size;
|
|
|
|
|
2017-01-05 12:45:37 +00:00
|
|
|
genode_env().ram().free(_ds);
|
base: avoid use of deprecated base/printf.h
Besides adapting the components to the use of base/log.h, the patch
cleans up a few base headers, i.e., it removes unused includes from
root/component.h, specifically base/heap.h and
ram_session/ram_session.h. Hence, components that relied on the implicit
inclusion of those headers have to manually include those headers now.
While adjusting the log messages, I repeatedly stumbled over the problem
that printing char * arguments is ambiguous. It is unclear whether to
print the argument as pointer or null-terminated string. To overcome
this problem, the patch introduces a new type 'Cstring' that allows the
caller to express that the argument should be handled as null-terminated
string. As a nice side effect, with this type in place, the optional len
argument of the 'String' class could be removed. Instead of supplying a
pair of (char const *, size_t), the constructor accepts a 'Cstring'.
This, in turn, clears the way let the 'String' constructor use the new
output mechanism to assemble a string from multiple arguments (and
thereby getting rid of snprintf within Genode in the near future).
To enforce the explicit resolution of the char * ambiguity, the 'char *'
overload of the 'print' function is marked as deleted.
Issue #1987
2016-07-13 17:07:09 +00:00
|
|
|
Genode::log("free up ", _size, " ", _mem_allocated, "/",
|
|
|
|
_mem_unused, " hit=", hit, "/", hit_coarse, " avail=",
|
2017-05-07 23:33:40 +00:00
|
|
|
genode_env().ram().avail_ram());
|
2016-06-14 08:18:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void unused()
|
|
|
|
{
|
|
|
|
_inuse = false;
|
|
|
|
|
|
|
|
_runtime_ds.remove(this);
|
|
|
|
|
|
|
|
_mem_unused += _size;
|
|
|
|
_unused_ds.insert(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void used(Genode::addr_t size)
|
|
|
|
{
|
|
|
|
_inuse = true;
|
|
|
|
|
|
|
|
_unused_ds.remove(this);
|
|
|
|
_used_size = size;
|
|
|
|
_mem_unused -= _size;
|
|
|
|
_runtime_ds.insert(this);
|
2016-06-08 18:15:27 +00:00
|
|
|
}
|
|
|
|
|
2016-06-14 08:18:32 +00:00
|
|
|
bool higher(Avl_ds *e) {
|
|
|
|
return _inuse ? e->_virt > _virt : e->_size > _size; }
|
2016-06-08 18:15:27 +00:00
|
|
|
|
2016-06-14 08:18:32 +00:00
|
|
|
Avl_ds *find_virt(Genode::addr_t virt)
|
2016-06-08 18:15:27 +00:00
|
|
|
{
|
|
|
|
if (virt == _virt) return this;
|
|
|
|
Avl_ds *obj = this->child(virt > _virt);
|
2016-06-14 08:18:32 +00:00
|
|
|
return obj ? obj->find_virt(virt) : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
Avl_ds *find_coarse_match(Genode::addr_t size_min, Genode::addr_t size_max)
|
|
|
|
{
|
|
|
|
if (size_min <= _size && _size <= size_max) return this;
|
|
|
|
Avl_ds *obj = this->child(size_max > _size);
|
|
|
|
return obj ? obj->find_coarse_match(size_min, size_max) : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
Avl_ds *find_size(Genode::addr_t size, bool equal = true)
|
|
|
|
{
|
|
|
|
if (equal ? size == _size : size <= _size) return this;
|
|
|
|
Avl_ds *obj = this->child(size > _size);
|
|
|
|
return obj ? obj->find_size(size, equal) : 0;
|
2016-06-08 18:15:27 +00:00
|
|
|
}
|
|
|
|
|
2016-06-14 08:18:32 +00:00
|
|
|
static Avl_ds *find_match(Genode::addr_t size, bool coarse = false)
|
|
|
|
{
|
|
|
|
Avl_ds * head = _unused_ds.first();
|
|
|
|
if (!head)
|
|
|
|
return head;
|
|
|
|
|
|
|
|
return coarse ? head->find_coarse_match(size, size * 2)
|
|
|
|
: head->find_size(size);
|
|
|
|
}
|
|
|
|
|
|
|
|
Genode::addr_t ds_virt() const { return _virt; }
|
|
|
|
|
|
|
|
static void memory_freeup(Genode::addr_t const cb)
|
|
|
|
{
|
2016-09-15 12:40:37 +00:00
|
|
|
::size_t cbx = cb * 4;
|
2016-06-14 08:18:32 +00:00
|
|
|
while (_unused_ds.first() && cbx &&
|
|
|
|
(_mem_allocated + cb > MEMORY_MAX ||
|
|
|
|
_mem_unused + cb > MEMORY_CACHED ||
|
2017-05-07 23:33:40 +00:00
|
|
|
genode_env().ram().avail_ram().value < cb * 2
|
2016-06-14 08:18:32 +00:00
|
|
|
)
|
|
|
|
)
|
|
|
|
{
|
|
|
|
Avl_ds * ds_free = _unused_ds.first()->find_size(cbx, false);
|
|
|
|
if (!ds_free) {
|
|
|
|
cbx = cbx / 2;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-01-05 12:45:37 +00:00
|
|
|
destroy(vmm_heap(), ds_free);
|
2016-06-14 08:18:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void free_memory(void * pv, size_t cb)
|
|
|
|
{
|
|
|
|
if (cb % 0x1000)
|
|
|
|
cb = (cb & ~0xFFFUL) + 0x1000UL;
|
|
|
|
|
|
|
|
Avl_ds * ds_obj = Avl_ds::_runtime_ds.first();
|
|
|
|
if (ds_obj)
|
|
|
|
ds_obj = ds_obj->find_virt(reinterpret_cast<Genode::addr_t>(pv));
|
|
|
|
|
|
|
|
if (ds_obj && ds_obj->_used_size == cb)
|
|
|
|
ds_obj->unused();
|
|
|
|
else {
|
base: avoid use of deprecated base/printf.h
Besides adapting the components to the use of base/log.h, the patch
cleans up a few base headers, i.e., it removes unused includes from
root/component.h, specifically base/heap.h and
ram_session/ram_session.h. Hence, components that relied on the implicit
inclusion of those headers have to manually include those headers now.
While adjusting the log messages, I repeatedly stumbled over the problem
that printing char * arguments is ambiguous. It is unclear whether to
print the argument as pointer or null-terminated string. To overcome
this problem, the patch introduces a new type 'Cstring' that allows the
caller to express that the argument should be handled as null-terminated
string. As a nice side effect, with this type in place, the optional len
argument of the 'String' class could be removed. Instead of supplying a
pair of (char const *, size_t), the constructor accepts a 'Cstring'.
This, in turn, clears the way let the 'String' constructor use the new
output mechanism to assemble a string from multiple arguments (and
thereby getting rid of snprintf within Genode in the near future).
To enforce the explicit resolution of the char * ambiguity, the 'char *'
overload of the 'print' function is marked as deleted.
Issue #1987
2016-07-13 17:07:09 +00:00
|
|
|
Genode::error(__func__, " unknown memory region ", pv, "(",
|
|
|
|
Genode::Hex(ds_obj ? ds_obj->ds_virt() : 0),
|
|
|
|
")+", Genode::Hex(cb), "(",
|
|
|
|
Genode::Hex(ds_obj ? ds_obj->_size : 0), ")");
|
2016-06-14 08:18:32 +00:00
|
|
|
}
|
|
|
|
}
|
2016-06-08 18:15:27 +00:00
|
|
|
};
|
|
|
|
|
2016-06-14 08:18:32 +00:00
|
|
|
Genode::Avl_tree<Avl_ds> Avl_ds::_runtime_ds;
|
|
|
|
Genode::Avl_tree<Avl_ds> Avl_ds::_unused_ds;
|
|
|
|
static Genode::Lock lock_ds;
|
|
|
|
|
|
|
|
Genode::addr_t Avl_ds::hit = 0;
|
|
|
|
Genode::addr_t Avl_ds::hit_coarse = 0;
|
|
|
|
Genode::addr_t Avl_ds::_mem_allocated = 0;
|
|
|
|
Genode::addr_t Avl_ds::_mem_unused = 0;
|
2013-08-21 09:37:21 +00:00
|
|
|
|
2014-09-23 11:01:47 +00:00
|
|
|
static void *alloc_mem(size_t cb, const char *pszTag, bool executable = false)
|
2013-08-21 09:37:21 +00:00
|
|
|
{
|
2017-01-05 12:45:37 +00:00
|
|
|
/* using managed dataspace to have all addresses within a 1 << 31 bit range */
|
|
|
|
static Sub_rm_connection rt_memory(genode_env(), 2 * MEMORY_MAX);
|
|
|
|
|
2013-08-21 09:37:21 +00:00
|
|
|
using namespace Genode;
|
|
|
|
|
2016-06-14 08:18:32 +00:00
|
|
|
if (!cb)
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
if (cb % 0x1000)
|
|
|
|
cb = (cb & ~0xFFFUL) + 0x1000UL;
|
|
|
|
|
|
|
|
Lock::Guard guard(lock_ds);
|
|
|
|
|
|
|
|
if (Avl_ds * ds_free = Avl_ds::find_match(cb)) {
|
|
|
|
ds_free->used(cb);
|
|
|
|
|
|
|
|
Avl_ds::hit++;
|
|
|
|
return reinterpret_cast<void *>(ds_free->ds_virt());
|
|
|
|
} else
|
|
|
|
if (Avl_ds * ds_free = Avl_ds::find_match(cb, true)) {
|
|
|
|
ds_free->used(cb);
|
|
|
|
|
|
|
|
Avl_ds::hit_coarse++;
|
|
|
|
return reinterpret_cast<void *>(ds_free->ds_virt());
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check for memory freeup, give hint about required memory (cb) */
|
|
|
|
Avl_ds::memory_freeup(cb);
|
|
|
|
|
2014-09-23 11:01:47 +00:00
|
|
|
try {
|
2017-01-05 12:45:37 +00:00
|
|
|
Ram_dataspace_capability ds = genode_env().ram().alloc(cb);
|
2014-09-23 11:01:47 +00:00
|
|
|
Assert(ds.valid());
|
|
|
|
|
2016-09-15 12:40:37 +00:00
|
|
|
Genode::size_t const whole_size = 0;
|
|
|
|
Genode::off_t const offset = 0;
|
|
|
|
bool const any_addr = false;
|
|
|
|
void * any_local_addr = nullptr;
|
2013-08-21 09:37:21 +00:00
|
|
|
|
2016-10-04 09:07:59 +00:00
|
|
|
void * local_addr = rt_memory.attach(ds, whole_size, offset,
|
|
|
|
any_addr, any_local_addr,
|
|
|
|
executable);
|
2014-09-23 11:01:47 +00:00
|
|
|
|
|
|
|
Assert(local_addr);
|
|
|
|
|
2017-01-05 12:45:37 +00:00
|
|
|
new (vmm_heap()) Avl_ds(ds, local_addr, cb);
|
2016-06-08 18:15:27 +00:00
|
|
|
|
|
|
|
return local_addr;
|
2014-09-23 11:01:47 +00:00
|
|
|
} catch (...) {
|
base: avoid use of deprecated base/printf.h
Besides adapting the components to the use of base/log.h, the patch
cleans up a few base headers, i.e., it removes unused includes from
root/component.h, specifically base/heap.h and
ram_session/ram_session.h. Hence, components that relied on the implicit
inclusion of those headers have to manually include those headers now.
While adjusting the log messages, I repeatedly stumbled over the problem
that printing char * arguments is ambiguous. It is unclear whether to
print the argument as pointer or null-terminated string. To overcome
this problem, the patch introduces a new type 'Cstring' that allows the
caller to express that the argument should be handled as null-terminated
string. As a nice side effect, with this type in place, the optional len
argument of the 'String' class could be removed. Instead of supplying a
pair of (char const *, size_t), the constructor accepts a 'Cstring'.
This, in turn, clears the way let the 'String' constructor use the new
output mechanism to assemble a string from multiple arguments (and
thereby getting rid of snprintf within Genode in the near future).
To enforce the explicit resolution of the char * ambiguity, the 'char *'
overload of the 'print' function is marked as deleted.
Issue #1987
2016-07-13 17:07:09 +00:00
|
|
|
Genode::error("Could not allocate RTMem* memory of size=", cb);
|
2016-06-08 18:15:27 +00:00
|
|
|
return nullptr;
|
2014-09-23 11:01:47 +00:00
|
|
|
}
|
2013-08-21 09:37:21 +00:00
|
|
|
}
|
|
|
|
|
2016-08-16 15:18:48 +00:00
|
|
|
#ifndef RT_NO_THROW
|
|
|
|
/* not defined in vbox5, but this code is used by vbox4 and vbox5 */
|
|
|
|
#define RT_NO_THROW
|
|
|
|
#endif
|
2013-08-21 09:37:21 +00:00
|
|
|
|
2014-09-23 11:01:47 +00:00
|
|
|
/*
|
|
|
|
* 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);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-08-21 09:37:21 +00:00
|
|
|
void *RTMemPageAllocZTag(size_t cb, const char *pszTag) RT_NO_THROW
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* The RAM dataspace freshly allocated by 'RTMemExecAllocTag' is zeroed
|
|
|
|
* already.
|
|
|
|
*/
|
2016-06-14 08:18:32 +00:00
|
|
|
void * addr = alloc_mem(cb, pszTag);
|
|
|
|
if (addr)
|
|
|
|
Genode::memset(addr, 0, cb);
|
|
|
|
return addr;
|
2013-08-21 09:37:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void *RTMemPageAllocTag(size_t cb, const char *pszTag) RT_NO_THROW
|
|
|
|
{
|
2014-09-23 11:01:47 +00:00
|
|
|
return alloc_mem(cb, pszTag);
|
2013-08-21 09:37:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-09-23 11:01:47 +00:00
|
|
|
void RTMemPageFree(void *pv, size_t cb) RT_NO_THROW
|
|
|
|
{
|
2016-06-14 08:18:32 +00:00
|
|
|
Genode::Lock::Guard guard(lock_ds);
|
2016-06-08 18:15:27 +00:00
|
|
|
|
2016-06-14 08:18:32 +00:00
|
|
|
Avl_ds::free_memory(pv, cb);
|
2014-09-23 11:01:47 +00:00
|
|
|
}
|
|
|
|
|
2013-08-21 09:37:21 +00:00
|
|
|
#include <iprt/buildconfig.h>
|
|
|
|
|
|
|
|
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; }
|
|
|
|
|
|
|
|
|
2014-05-23 07:26:57 +00:00
|
|
|
extern "C" DECLHIDDEN(int) rtProcInitExePath(char *pszPath, size_t cchPath)
|
2013-08-21 09:37:21 +00:00
|
|
|
{
|
2014-05-23 07:26:57 +00:00
|
|
|
Genode::strncpy(pszPath, "/virtualbox", cchPath);
|
2013-08-21 09:37:21 +00:00
|
|
|
return 0;
|
|
|
|
}
|