/* * \brief Fiasco protection domain facility * \author Christian Helmuth * \author Stefan Kalkowski * \date 2006-04-11 */ /* * Copyright (C) 2006-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. */ /* Genode includes */ #include #include /* core includes */ #include #include #include #include /* Fiasco includes */ namespace Fiasco { #include #include } using namespace Fiasco; using namespace Genode; static addr_t core_utcb_base() { static addr_t base = (addr_t) l4_utcb(); return base; } /*************************** ** Public object members ** ***************************/ bool Platform_pd::bind_thread(Platform_thread *thread) { /* * Fiasco.OC limits the UTCB area for roottask to 16K. Therefore, the * number of threads is limited to 16K / L4_UTCB_OFFSET. * (see kernel/fiasco/src/kern/kernel_thread-std.cpp:94) */ unsigned const thread_max = thread->core_thread() ? 16*1024/L4_UTCB_OFFSET : THREAD_MAX; for (unsigned i = 0; i < thread_max; i++) { if (_threads[i]) continue; _threads[i] = thread; if (thread->core_thread()) thread->_utcb = (addr_t) (core_utcb_base() + i * L4_UTCB_OFFSET); else thread->_utcb = reinterpret_cast(utcb_area_start() + i * L4_UTCB_OFFSET); Fiasco::l4_cap_idx_t cap_offset = THREAD_AREA_BASE + i * THREAD_AREA_SLOT; thread->_gate.remote = cap_offset + THREAD_GATE_CAP; thread->_pager.remote = cap_offset + THREAD_PAGER_CAP; thread->_irq.remote = cap_offset + THREAD_IRQ_CAP; /* if it's no core-thread we have to map parent and pager gate cap */ if (!thread->core_thread()) { _task.map(_task.local.data()->kcap()); _debug.map(_task.local.data()->kcap()); } /* inform thread about binding */ thread->bind(this); return true; } error("thread alloc failed"); return false; } 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; } } void Platform_pd::assign_parent(Native_capability parent) { if (_parent.remote == Fiasco::L4_INVALID_CAP && parent.valid()) { _parent.local = parent; _parent.remote = PARENT_CAP; _parent.map(_task.local.data()->kcap()); } } void Platform_pd::flush(addr_t, size_t size, Core_local_addr core_local) { unmap_local(core_local.value, size >> get_page_size_log2()); } static Core_cap_index * debug_cap() { unsigned long id = platform_specific()->cap_id_alloc()->alloc(); static Cap_index * idx = cap_map()->insert(id, DEBUG_CAP); return reinterpret_cast(idx); } Platform_pd::Platform_pd(Core_cap_index* i) : _task(Native_capability(i), TASK_CAP) { for (unsigned i = 0; i < THREAD_MAX; i++) _threads[i] = (Platform_thread*) 0; } Platform_pd::Platform_pd(Allocator *, char const *) : _task(true, TASK_CAP), _debug(debug_cap(), DEBUG_CAP) { for (unsigned i = 0; i < THREAD_MAX; i++) _threads[i] = (Platform_thread*) 0; l4_fpage_t utcb_area = l4_fpage(utcb_area_start(), log2(UTCB_AREA_SIZE), 0); l4_msgtag_t tag = l4_factory_create_task(L4_BASE_FACTORY_CAP, _task.local.data()->kcap(), utcb_area); if (l4_msgtag_has_error(tag)) error("pd creation failed"); } Platform_pd::~Platform_pd() { for (unsigned i = 0; i < THREAD_MAX; i++) { if (_threads[i]) _threads[i]->unbind(); } }