/*
* \brief Mapping of Genode's capability names to capabilities selectors.
* \author Alexander Boettcher
* \date 2013-08-26
*
* This header is public to allow user-level VMMs to manually allocate windows
* of consecutive selectors (for virtualization event portals) in the
* component's capability space.
*/
/*
* Copyright (C) 2013-2017 Genode Labs GmbH
*
* This file is part of the Genode OS framework, which is distributed
* under the terms of the GNU Affero General Public License version 3.
*/
#ifndef _INCLUDE__NOVA__CAP_MAP_H_
#define _INCLUDE__NOVA__CAP_MAP_H_
/* Genode includes */
#include
#include
#include
#include
namespace Genode {
class Cap_range : public Avl_node {
private:
Mutex _mutex { };
addr_t _base = 0;
addr_t _last = 0;
enum {
HEADER = sizeof(_base) + sizeof(_mutex) + sizeof(_last),
CAP_RANGE_SIZE = 4096,
WORDS = (CAP_RANGE_SIZE - HEADER - sizeof(Avl_node)) / sizeof(addr_t),
};
uint16_t _cap_array[WORDS * sizeof(addr_t) / 2];
bool _match(addr_t id) {
return _base <= id && id < _base + elements(); };
public:
Cap_range(addr_t base) : _base(base)
{
static_assert(sizeof(*this) == CAP_RANGE_SIZE,
"Cap_range misconfigured");
for (unsigned i = 0; i < elements(); i++)
_cap_array[i] = 0;
}
addr_t base() const { return _base; }
unsigned elements() const { return sizeof(_cap_array) / sizeof(_cap_array[0]); }
Cap_range *find_by_id(addr_t);
void inc(unsigned id);
void dec(unsigned id, bool revoke = true, unsigned num_log2 = 0);
addr_t alloc(size_t const num_log2);
/************************
** Avl node interface **
************************/
bool higher(Cap_range *n) { return n->_base > _base; }
};
class Cap_index
{
private:
Cap_range * _range;
addr_t _local_name;
public:
Cap_index(Cap_range *range, addr_t local_name)
: _range(range), _local_name(local_name) {}
bool valid() const { return _range; }
inline void inc()
{
if (_range)
_range->inc(_local_name - _range->base());
}
inline void dec()
{
if (_range)
_range->dec(_local_name - _range->base());
}
};
class Capability_map : private Noncopyable
{
private:
Avl_tree _tree { };
public:
Cap_index find(addr_t local_sel);
void insert(Cap_range &range) { _tree.insert(&range); }
addr_t insert(size_t num_log_2 = 0, addr_t cap = ~0UL);
void remove(addr_t sel, uint8_t num_log_2 = 0, bool revoke = true);
};
/**
* Get the global Capability_map of the process.
*/
Capability_map &cap_map();
}
#endif /* _INCLUDE__NOVA__CAP_MAP_H_ */