genode/base-foc/src/core/platform_pd.cc

175 lines
4.0 KiB
C++

/*
* \brief Fiasco protection domain facility
* \author Christian Helmuth
* \author Stefan Kalkowski
* \date 2006-04-11
*/
/*
* Copyright (C) 2006-2012 Genode Labs GmbH
*
* This file is part of the Genode OS framework, which is distributed
* under the terms of the GNU General Public License version 2.
*/
/* Genode includes */
#include <base/native_types.h>
#include <util/misc_math.h>
/* core includes */
#include <util.h>
#include <platform.h>
#include <platform_pd.h>
/* Fiasco includes */
namespace Fiasco {
#include <l4/sys/utcb.h>
#include <l4/sys/factory.h>
}
using namespace Fiasco;
using namespace Genode;
static addr_t core_utcb_base() {
static addr_t base = (addr_t) l4_utcb();
return base;
}
/****************************
** Private object members **
****************************/
void Platform_pd::_create_pd(bool syscall)
{
if (!Cap_dst_policy::valid(_l4_task_cap))
_l4_task_cap = cap_alloc()->alloc();
if (syscall) {
if (!_l4_task_cap)
panic("no cap slot for pd creation available!");
l4_fpage_t utcb_area = l4_fpage(UTCB_AREA_START,
log2<unsigned>(UTCB_AREA_SIZE), 0);
l4_msgtag_t tag = l4_factory_create_task(L4_BASE_FACTORY_CAP,
_l4_task_cap, utcb_area);
if (l4_msgtag_has_error(tag))
panic("pd creation failed");
}
}
void Platform_pd::_destroy_pd()
{
l4_task_unmap(L4_BASE_TASK_CAP,
l4_obj_fpage(_l4_task_cap, 0, L4_FPAGE_RWX),
L4_FP_ALL_SPACES | L4_FP_DELETE_OBJ);
}
/***************************
** Public object members **
***************************/
int Platform_pd::bind_thread(Platform_thread *thread)
{
for (unsigned i = 0; i < THREAD_MAX; i++) {
if (_threads[i])
continue;
_threads[i] = thread;
if (thread->core_thread())
thread->_utcb = (l4_utcb_t*) (core_utcb_base() + i * L4_UTCB_OFFSET);
else
thread->_utcb =
reinterpret_cast<l4_utcb_t*>(UTCB_AREA_START + i * L4_UTCB_OFFSET);
Native_thread cap_offset = THREADS_BASE_CAP + i * THREAD_CAP_SLOT;
thread->_remote_gate_cap = Native_capability(cap_offset + THREAD_GATE_CAP,
thread->_gate_cap.local_name());
thread->_remote_pager_cap = cap_offset + THREAD_PAGER_CAP;
thread->_remote_irq_cap = cap_offset + THREAD_IRQ_CAP;
/* inform thread about binding */
thread->bind(this);
return 0;
}
PERR("thread alloc failed");
return -1;
}
void Platform_pd::unbind_thread(Platform_thread *thread)
{
/* inform thread about unbinding */
thread->unbind();
for (unsigned i = 0; i < THREAD_MAX; i++)
if (_threads[i] == thread) {
_threads[i] = (Platform_thread*) 0;
return;
}
}
int Platform_pd::assign_parent(Native_capability parent)
{
if (_parent.valid()) return -1;
_parent = parent;
return 0;
}
void Platform_pd::map_parent_cap()
{
if (!_parent_cap_mapped) {
l4_msgtag_t tag = l4_task_map(_l4_task_cap, L4_BASE_TASK_CAP,
l4_obj_fpage(_parent.dst(), 0, L4_FPAGE_RWX),
PARENT_CAP | L4_ITEM_MAP);
if (l4_msgtag_has_error(tag))
PWRN("mapping parent cap failed");
_parent_cap_mapped = true;
}
}
void Platform_pd::map_task_cap()
{
if (!_task_cap_mapped) {
l4_msgtag_t tag = l4_task_map(_l4_task_cap, L4_BASE_TASK_CAP,
l4_obj_fpage(_l4_task_cap, 0, L4_FPAGE_RWX),
TASK_CAP | L4_ITEM_MAP);
if (l4_msgtag_has_error(tag))
PWRN("mapping task cap failed");
_task_cap_mapped = true;
}
}
Platform_pd::Platform_pd(bool create, Native_task task_cap)
: _l4_task_cap(task_cap),
_badge(create ? Badge_allocator::allocator()->alloc() : 0),
_parent_cap_mapped(false),
_task_cap_mapped(false)
{
for (unsigned i = 0; i < THREAD_MAX; i++)
_threads[i] = (Platform_thread*) 0;
_create_pd(create);
}
Platform_pd::~Platform_pd()
{
for (unsigned i = 0; i < THREAD_MAX; i++) {
if (_threads[i])
_threads[i]->unbind();
}
_destroy_pd();
Badge_allocator::allocator()->free(_badge);
}