2011-12-22 15:19:25 +00:00
|
|
|
/*
|
|
|
|
* \brief Pager framework
|
|
|
|
* \author Norman Feske
|
|
|
|
* \author Sebastian Sumpf
|
2012-08-01 14:16:51 +00:00
|
|
|
* \author Alexander Boettcher
|
2011-12-22 15:19:25 +00:00
|
|
|
* \date 2010-01-25
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
2015-01-05 21:09:51 +00:00
|
|
|
* Copyright (C) 2010-2015 Genode Labs GmbH
|
2011-12-22 15:19:25 +00:00
|
|
|
*
|
|
|
|
* 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/pager.h>
|
|
|
|
#include <base/sleep.h>
|
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
#include <util/construct_at.h>
|
|
|
|
|
|
|
|
#include <rm_session/rm_session.h>
|
|
|
|
|
2011-12-22 15:19:25 +00:00
|
|
|
/* NOVA includes */
|
|
|
|
#include <nova/syscalls.h>
|
2014-09-05 15:00:31 +00:00
|
|
|
#include <nova_util.h> /* map_local */
|
2011-12-22 15:19:25 +00:00
|
|
|
|
|
|
|
using namespace Genode;
|
|
|
|
using namespace Nova;
|
|
|
|
|
2012-08-01 15:04:43 +00:00
|
|
|
extern Genode::addr_t __core_pd_sel;
|
2011-12-22 15:19:25 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
static Nova::Hip * kernel_hip()
|
2012-08-23 09:02:31 +00:00
|
|
|
{
|
2014-09-05 15:00:31 +00:00
|
|
|
/**
|
|
|
|
* Initial value of esp register, saved by the crt0 startup code.
|
|
|
|
* This value contains the address of the hypervisor information page.
|
|
|
|
*/
|
|
|
|
extern addr_t __initial_sp;
|
|
|
|
return reinterpret_cast<Hip *>(__initial_sp);
|
|
|
|
}
|
2012-08-23 09:02:31 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/* pager activation threads storage and handling - one thread per CPU */
|
|
|
|
enum { PAGER_CPUS = 128, PAGER_STACK_SIZE = 2*4096 };
|
2012-08-23 09:02:31 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
static char pager_activation_mem[sizeof (Pager_activation<PAGER_STACK_SIZE>) * PAGER_CPUS];
|
|
|
|
static Pager_activation_base * pager_threads[PAGER_CPUS];
|
2012-08-23 09:02:31 +00:00
|
|
|
|
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
static unsigned which_cpu(Pager_activation_base * pager)
|
|
|
|
{
|
|
|
|
Pager_activation_base * start = reinterpret_cast<Pager_activation_base *>(&pager_activation_mem);
|
|
|
|
Pager_activation_base * end = start + PAGER_CPUS;
|
2012-08-23 09:02:31 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
if (start <= pager && pager < end) {
|
|
|
|
/* pager of one of the non boot CPUs */
|
|
|
|
unsigned cpu_id = pager - start;
|
|
|
|
return cpu_id;
|
|
|
|
}
|
2012-08-23 09:02:31 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/* pager of boot CPU */
|
|
|
|
return Affinity::Location().xpos();
|
2012-08-23 09:02:31 +00:00
|
|
|
}
|
|
|
|
|
2013-01-11 22:10:21 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
void Pager_object::_page_fault_handler(addr_t pager_obj)
|
2011-12-22 15:19:25 +00:00
|
|
|
{
|
|
|
|
Ipc_pager ipc_pager;
|
|
|
|
ipc_pager.wait_for_fault();
|
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
Thread_base * myself = Thread_base::myself();
|
|
|
|
Pager_object * obj = reinterpret_cast<Pager_object *>(pager_obj);
|
|
|
|
Utcb * utcb = reinterpret_cast<Utcb *>(myself->utcb());
|
|
|
|
Pager_activation_base * pager_thread = static_cast<Pager_activation_base *>(myself);
|
2011-12-22 15:19:25 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/* lookup fault address and decide what to do */
|
2011-12-22 15:19:25 +00:00
|
|
|
int ret = obj->pager(ipc_pager);
|
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/* don't open receive window for pager threads */
|
|
|
|
if (utcb->crd_rcv.value())
|
|
|
|
nova_die();
|
2013-02-19 10:14:21 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/* good case - found a valid region which is mappable */
|
|
|
|
if (!ret)
|
|
|
|
ipc_pager.reply_and_wait_for_fault();
|
2013-01-31 10:59:38 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
obj->_state.thread.ip = ipc_pager.fault_ip();
|
|
|
|
obj->_state.thread.sp = 0;
|
|
|
|
obj->_state.thread.trapno = PT_SEL_PAGE_FAULT;
|
2013-01-31 10:59:38 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
obj->_state.block();
|
2013-02-27 10:32:49 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
char const * client = reinterpret_cast<char const *>(obj->_badge);
|
|
|
|
/* region manager fault - to be handled */
|
|
|
|
if (ret == 1) {
|
|
|
|
PDBG("page fault, thread '%s', cpu %u, ip=%lx, fault address=0x%lx",
|
|
|
|
client, which_cpu(pager_thread), ipc_pager.fault_ip(),
|
|
|
|
ipc_pager.fault_addr());
|
2013-02-27 10:32:49 +00:00
|
|
|
|
2012-06-26 11:43:30 +00:00
|
|
|
utcb->set_msg_word(0);
|
2012-08-22 10:10:04 +00:00
|
|
|
utcb->mtd = 0;
|
2014-09-05 15:00:31 +00:00
|
|
|
|
|
|
|
/* block the faulting thread until region manager is done */
|
|
|
|
ipc_pager.reply_and_wait_for_fault(obj->sel_sm_block());
|
2011-12-22 15:19:25 +00:00
|
|
|
}
|
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/* unhandled case */
|
|
|
|
obj->_state.mark_dead();
|
|
|
|
|
|
|
|
PWRN("unresolvable page fault, thread '%s', cpu %u, ip=%lx, "
|
|
|
|
"fault address=0x%lx ret=%u", client, which_cpu(pager_thread),
|
|
|
|
ipc_pager.fault_ip(), ipc_pager.fault_addr(), ret);
|
|
|
|
|
|
|
|
Native_capability pager_cap = obj->Object_pool<Pager_object>::Entry::cap();
|
|
|
|
revoke(pager_cap.dst());
|
|
|
|
|
|
|
|
revoke(Obj_crd(obj->exc_pt_sel_client(), NUM_INITIAL_PT_LOG2));
|
|
|
|
|
|
|
|
utcb->set_msg_word(0);
|
|
|
|
utcb->mtd = 0;
|
2011-12-22 15:19:25 +00:00
|
|
|
ipc_pager.reply_and_wait_for_fault();
|
|
|
|
}
|
|
|
|
|
2013-01-11 22:10:21 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
void Pager_object::exception(uint8_t exit_id)
|
2012-08-23 09:02:31 +00:00
|
|
|
{
|
2014-09-05 15:00:31 +00:00
|
|
|
Thread_base *myself = Thread_base::myself();
|
|
|
|
Utcb * utcb = reinterpret_cast<Utcb *>(myself->utcb());
|
|
|
|
Pager_activation_base * pager_thread = static_cast<Pager_activation_base *>(myself);
|
|
|
|
|
|
|
|
if (exit_id > PT_SEL_PARENT || !pager_thread)
|
|
|
|
nova_die();
|
|
|
|
|
2013-01-16 15:07:04 +00:00
|
|
|
addr_t fault_ip = utcb->ip;
|
2013-11-04 10:05:10 +00:00
|
|
|
uint8_t res = 0xFF;
|
2015-01-06 13:20:16 +00:00
|
|
|
addr_t mtd = 0;
|
2012-08-23 09:02:31 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
if (_state.skip_requested()) {
|
|
|
|
_state.skip_reset();
|
|
|
|
|
|
|
|
utcb->set_msg_word(0);
|
|
|
|
utcb->mtd = 0;
|
|
|
|
reply(myself->stack_top());
|
|
|
|
}
|
|
|
|
|
|
|
|
/* remember exception type for cpu_session()->state() calls */
|
|
|
|
_state.thread.trapno = exit_id;
|
|
|
|
_state.thread.ip = fault_ip;
|
|
|
|
|
|
|
|
if (_exception_sigh.valid()) {
|
|
|
|
_state.submit_signal();
|
|
|
|
res = client_recall();
|
|
|
|
}
|
2013-09-26 12:31:56 +00:00
|
|
|
|
|
|
|
if (res != NOVA_OK) {
|
2014-09-05 15:00:31 +00:00
|
|
|
/* nobody handles this exception - so thread will be stopped finally */
|
|
|
|
_state.mark_dead();
|
2013-02-19 10:14:21 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
char const * client = reinterpret_cast<char const *>(_badge);
|
|
|
|
PWRN("unresolvable exception %u, thread '%s', cpu %u, ip=0x%lx, %s",
|
|
|
|
exit_id, client, which_cpu(pager_thread), fault_ip,
|
|
|
|
res == 0xFF ? "no signal handler" :
|
|
|
|
(res == NOVA_OK ? "" : "recall failed"));
|
2013-01-16 15:07:04 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
Nova::revoke(Obj_crd(exc_pt_sel_client(), NUM_INITIAL_PT_LOG2));
|
2015-01-06 13:20:16 +00:00
|
|
|
|
|
|
|
enum { TRAP_BREAKPOINT = 3 };
|
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
if (exit_id == TRAP_BREAKPOINT) {
|
2015-01-06 13:20:16 +00:00
|
|
|
utcb->ip = fault_ip - 1;
|
|
|
|
mtd = Mtd::EIP;
|
|
|
|
}
|
2012-08-23 09:02:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
utcb->set_msg_word(0);
|
2015-01-06 13:20:16 +00:00
|
|
|
utcb->mtd = mtd;
|
2012-08-23 09:02:31 +00:00
|
|
|
|
|
|
|
reply(myself->stack_top());
|
|
|
|
}
|
|
|
|
|
2013-01-11 22:10:21 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
void Pager_object::_recall_handler(addr_t pager_obj)
|
2012-08-24 08:25:24 +00:00
|
|
|
{
|
2014-09-05 15:00:31 +00:00
|
|
|
Thread_base * myself = Thread_base::myself();
|
|
|
|
Pager_object * obj = reinterpret_cast<Pager_object *>(pager_obj);
|
|
|
|
Utcb * utcb = reinterpret_cast<Utcb *>(myself->utcb());
|
2012-08-24 08:25:24 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/* save state - can be requested via cpu_session->state */
|
2012-08-24 08:25:24 +00:00
|
|
|
obj->_copy_state(utcb);
|
|
|
|
|
|
|
|
obj->_state.thread.ip = utcb->ip;
|
|
|
|
obj->_state.thread.sp = utcb->sp;
|
|
|
|
|
|
|
|
obj->_state.thread.eflags = utcb->flags;
|
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/* thread becomes blocked */
|
|
|
|
obj->_state.block();
|
2012-08-24 08:25:24 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/* deliver signal if it was requested */
|
|
|
|
if (obj->_state.to_submit())
|
|
|
|
obj->submit_exception_signal();
|
2012-08-24 08:25:24 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/* notify callers of cpu_session()->pause that the state is now valid */
|
|
|
|
if (obj->_state.notify_requested()) {
|
|
|
|
obj->_state.notify_cancel();
|
|
|
|
if (sm_ctrl(obj->sel_sm_notify(), SEMAPHORE_UP) != NOVA_OK)
|
|
|
|
PWRN("paused notification failed");
|
|
|
|
}
|
2012-08-24 08:25:24 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/* switch on/off single step */
|
2012-08-23 10:43:19 +00:00
|
|
|
bool singlestep_state = obj->_state.thread.eflags & 0x100UL;
|
2013-09-26 13:16:01 +00:00
|
|
|
if (obj->_state.singlestep() && !singlestep_state) {
|
2012-08-23 10:43:19 +00:00
|
|
|
utcb->flags = obj->_state.thread.eflags | 0x100UL;
|
|
|
|
utcb->mtd = Nova::Mtd(Mtd::EFL).value();
|
2014-09-05 15:00:31 +00:00
|
|
|
} else {
|
2013-09-26 13:16:01 +00:00
|
|
|
if (!obj->_state.singlestep() && singlestep_state) {
|
2012-12-04 08:49:01 +00:00
|
|
|
utcb->flags = obj->_state.thread.eflags & ~0x100UL;
|
|
|
|
utcb->mtd = Nova::Mtd(Mtd::EFL).value();
|
|
|
|
} else
|
|
|
|
utcb->mtd = 0;
|
2014-09-05 15:00:31 +00:00
|
|
|
}
|
2013-09-26 13:16:01 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/* block until cpu_session()->resume() respectively wake_up() call */
|
2012-08-24 08:25:24 +00:00
|
|
|
utcb->set_msg_word(0);
|
2014-09-05 15:00:31 +00:00
|
|
|
reply(myself->stack_top(), obj->sel_sm_block());
|
2012-08-24 08:25:24 +00:00
|
|
|
}
|
2011-12-22 15:19:25 +00:00
|
|
|
|
2013-01-11 22:10:21 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
void Pager_object::_startup_handler(addr_t pager_obj)
|
2011-12-22 15:19:25 +00:00
|
|
|
{
|
2014-09-05 15:00:31 +00:00
|
|
|
Thread_base *myself = Thread_base::myself();
|
|
|
|
Pager_object * obj = reinterpret_cast<Pager_object *>(pager_obj);
|
|
|
|
Utcb * utcb = reinterpret_cast<Utcb *>(myself->utcb());
|
2011-12-22 15:19:25 +00:00
|
|
|
|
2012-08-24 08:25:24 +00:00
|
|
|
utcb->ip = obj->_initial_eip;
|
|
|
|
utcb->sp = obj->_initial_esp;
|
2012-08-23 09:02:31 +00:00
|
|
|
|
2011-12-22 15:19:25 +00:00
|
|
|
utcb->mtd = Mtd::EIP | Mtd::ESP;
|
2012-06-26 11:43:30 +00:00
|
|
|
utcb->set_msg_word(0);
|
2012-08-23 09:02:31 +00:00
|
|
|
|
|
|
|
reply(myself->stack_top());
|
2011-12-22 15:19:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
void Pager_object::_invoke_handler(addr_t pager_obj)
|
2011-12-22 15:19:25 +00:00
|
|
|
{
|
2014-09-05 15:00:31 +00:00
|
|
|
Thread_base *myself = Thread_base::myself();
|
|
|
|
Pager_object * obj = reinterpret_cast<Pager_object *>(pager_obj);
|
|
|
|
Utcb * utcb = reinterpret_cast<Utcb *>(myself->utcb());
|
2011-12-22 15:19:25 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/* receive window must be closed - otherwise implementation bug */
|
|
|
|
if (utcb->crd_rcv.value())
|
|
|
|
nova_die();
|
2013-11-14 09:35:44 +00:00
|
|
|
|
2013-09-23 08:44:50 +00:00
|
|
|
addr_t const event = utcb->msg[0];
|
|
|
|
addr_t const logcount = utcb->msg[1];
|
2013-09-25 09:53:49 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/* check for translated vCPU portals */
|
|
|
|
unsigned const items_count = 1U << (Nova::NUM_INITIAL_VCPU_PT_LOG2 - 1);
|
|
|
|
|
|
|
|
if ((obj->_client_exc_vcpu != Native_thread::INVALID_INDEX) &&
|
|
|
|
(utcb->msg_items() == items_count) &&
|
|
|
|
(utcb->msg_words() == 1 && (event == 0UL || event == 1UL))) {
|
|
|
|
/* check all translated item and remap if valid */
|
|
|
|
for (unsigned i = 0; i < items_count; i++) {
|
|
|
|
Nova::Utcb::Item * item = utcb->get_item(i);
|
|
|
|
|
|
|
|
if (!item)
|
|
|
|
break;
|
|
|
|
|
|
|
|
Nova::Crd cap(item->crd);
|
|
|
|
|
|
|
|
if (cap.is_null() || item->is_del())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remap portal to dense packed region - required for vCPU running
|
|
|
|
* in separate PD (non-colocated case)
|
|
|
|
*/
|
|
|
|
Obj_crd snd(cap.base(), 0);
|
|
|
|
Obj_crd rcv(obj->_client_exc_vcpu + event * items_count + i, 0);
|
|
|
|
if (map_local(utcb, snd, rcv))
|
|
|
|
PWRN("could not remap vCPU portal 0x%x", i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if protocol is violated ignore request */
|
|
|
|
if (utcb->msg_words() != 2) {
|
|
|
|
utcb->mtd = 0;
|
|
|
|
utcb->set_msg_word(0);
|
|
|
|
reply(myself->stack_top());
|
|
|
|
}
|
|
|
|
|
2011-12-22 15:19:25 +00:00
|
|
|
utcb->mtd = 0;
|
2012-06-26 11:43:30 +00:00
|
|
|
utcb->set_msg_word(0);
|
2011-12-22 15:19:25 +00:00
|
|
|
|
2013-09-26 12:31:56 +00:00
|
|
|
/* native ec cap requested */
|
2013-09-25 09:53:49 +00:00
|
|
|
if (event == ~0UL) {
|
|
|
|
/**
|
|
|
|
* Return native EC cap with specific rights mask set.
|
|
|
|
* If the cap is mapped the kernel will demote the
|
|
|
|
* rights of the EC as specified by the rights mask.
|
|
|
|
*
|
|
|
|
* The cap is supposed to be returned to clients,
|
|
|
|
* which they have to use as argument to identify
|
|
|
|
* the thread to which they want attach portals.
|
|
|
|
*
|
|
|
|
* The demotion by the kernel during the map operation
|
|
|
|
* takes care that the EC cap itself contains
|
|
|
|
* no usable rights for the clients.
|
|
|
|
*/
|
|
|
|
bool res = utcb->append_item(Obj_crd(obj->_state.sel_client_ec, 0,
|
|
|
|
Obj_crd::RIGHT_EC_RECALL), 0);
|
2014-02-12 15:22:22 +00:00
|
|
|
/* if logcount > 0 then the pager cap should also be mapped */
|
|
|
|
if (logcount)
|
|
|
|
res = utcb->append_item(Obj_crd(obj->Object_pool<Pager_object>::Entry::cap().local_name(), 0), 1);
|
2013-09-25 09:53:49 +00:00
|
|
|
(void)res;
|
2013-09-26 12:31:56 +00:00
|
|
|
|
2013-09-25 09:53:49 +00:00
|
|
|
reply(myself->stack_top());
|
|
|
|
}
|
|
|
|
|
2013-09-26 12:31:56 +00:00
|
|
|
/* semaphore for signaling thread is requested, reuse PT_SEL_STARTUP. */
|
|
|
|
if (event == ~0UL - 1) {
|
|
|
|
/* create semaphore only once */
|
|
|
|
if (!obj->_state.has_signal_sm()) {
|
|
|
|
|
|
|
|
revoke(Obj_crd(obj->exc_pt_sel_client() + PT_SEL_STARTUP, 0));
|
|
|
|
|
|
|
|
bool res = Nova::create_sm(obj->exc_pt_sel_client() + PT_SEL_STARTUP,
|
|
|
|
__core_pd_sel, 0);
|
|
|
|
if (res != Nova::NOVA_OK)
|
|
|
|
reply(myself->stack_top());
|
|
|
|
|
|
|
|
obj->_state.mark_signal_sm();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool res = utcb->append_item(Obj_crd(obj->exc_pt_sel_client() +
|
|
|
|
PT_SEL_STARTUP, 0), 0);
|
|
|
|
(void)res;
|
|
|
|
|
|
|
|
reply(myself->stack_top());
|
|
|
|
}
|
|
|
|
|
|
|
|
/* sanity check, if event is not valid return nothing */
|
2013-09-23 08:44:50 +00:00
|
|
|
if (logcount > NUM_INITIAL_PT_LOG2 || event > 1UL << NUM_INITIAL_PT_LOG2 ||
|
|
|
|
event + (1UL << logcount) > (1UL << NUM_INITIAL_PT_LOG2))
|
|
|
|
reply(myself->stack_top());
|
2012-08-24 08:25:24 +00:00
|
|
|
|
2013-09-26 12:31:56 +00:00
|
|
|
/* valid event portal is requested, delegate it to caller */
|
2013-09-25 09:53:49 +00:00
|
|
|
bool res = utcb->append_item(Obj_crd(obj->exc_pt_sel_client() + event,
|
|
|
|
logcount), 0);
|
2013-09-23 08:44:50 +00:00
|
|
|
(void)res;
|
2011-12-22 15:19:25 +00:00
|
|
|
|
2012-08-23 09:02:31 +00:00
|
|
|
reply(myself->stack_top());
|
2011-12-22 15:19:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-09-26 13:16:01 +00:00
|
|
|
void Pager_object::wake_up()
|
|
|
|
{
|
2014-09-05 15:00:31 +00:00
|
|
|
if (!_state.blocked())
|
|
|
|
return;
|
|
|
|
|
|
|
|
_state.unblock();
|
2013-09-26 13:16:01 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
uint8_t res = sm_ctrl(sel_sm_block(), SEMAPHORE_UP);
|
|
|
|
if (res != NOVA_OK)
|
|
|
|
PWRN("canceling blocked client failed (thread sm)");
|
2013-09-26 13:16:01 +00:00
|
|
|
}
|
2011-12-22 15:19:25 +00:00
|
|
|
|
2012-08-22 10:10:04 +00:00
|
|
|
|
2013-01-11 22:10:21 +00:00
|
|
|
void Pager_object::client_cancel_blocking()
|
|
|
|
{
|
2013-09-23 07:07:33 +00:00
|
|
|
uint8_t res = sm_ctrl(exc_pt_sel_client() + SM_SEL_EC, SEMAPHORE_UP);
|
2012-08-24 08:25:24 +00:00
|
|
|
if (res != NOVA_OK)
|
2013-09-26 12:31:56 +00:00
|
|
|
PWRN("canceling blocked client failed (thread sm)");
|
|
|
|
|
|
|
|
if (!_state.has_signal_sm())
|
|
|
|
return;
|
|
|
|
|
|
|
|
res = sm_ctrl(exc_pt_sel_client() + PT_SEL_STARTUP, SEMAPHORE_UP);
|
|
|
|
if (res != NOVA_OK)
|
|
|
|
PWRN("canceling blocked client failed (signal sm)");
|
2012-08-24 08:25:24 +00:00
|
|
|
}
|
2011-12-22 15:19:25 +00:00
|
|
|
|
2012-08-22 10:10:04 +00:00
|
|
|
|
2013-01-11 22:10:21 +00:00
|
|
|
uint8_t Pager_object::client_recall()
|
|
|
|
{
|
2014-01-30 09:19:17 +00:00
|
|
|
return ec_ctrl(EC_RECALL, _state.sel_client_ec);
|
2012-08-22 10:10:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-12-04 08:49:01 +00:00
|
|
|
void Pager_object::cleanup_call()
|
|
|
|
{
|
2014-09-05 15:00:31 +00:00
|
|
|
_state.mark_dissolved();
|
|
|
|
|
2013-09-23 07:07:33 +00:00
|
|
|
/* revoke all portals handling the client. */
|
|
|
|
revoke(Obj_crd(exc_pt_sel_client(), NUM_INITIAL_PT_LOG2));
|
2012-12-04 08:49:01 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/* if we are paused or waiting for a page fault nothing is in-flight */
|
|
|
|
if (_state.blocked())
|
|
|
|
return;
|
2012-12-04 08:49:01 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
Utcb *utcb = reinterpret_cast<Utcb *>(Thread_base::myself()->utcb());
|
2012-12-04 08:49:01 +00:00
|
|
|
utcb->set_msg_word(0);
|
2014-09-05 15:00:31 +00:00
|
|
|
utcb->mtd = 0;
|
|
|
|
if (uint8_t res = call(sel_pt_cleanup()))
|
|
|
|
PERR("%8p - cleanup call to pager failed res=%d", utcb, res);
|
2012-12-04 08:49:01 +00:00
|
|
|
}
|
|
|
|
|
2013-09-26 12:31:56 +00:00
|
|
|
|
2013-02-22 14:26:43 +00:00
|
|
|
static uint8_t create_portal(addr_t pt, addr_t pd, addr_t ec, Mtd mtd,
|
2014-09-05 15:00:31 +00:00
|
|
|
addr_t eip, addr_t localname)
|
2013-02-22 14:26:43 +00:00
|
|
|
{
|
|
|
|
uint8_t res = create_pt(pt, pd, ec, mtd, eip);
|
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
if (res != NOVA_OK)
|
|
|
|
return res;
|
|
|
|
|
|
|
|
res = pt_ctrl(pt, localname);
|
2013-02-22 14:26:43 +00:00
|
|
|
if (res == NOVA_OK)
|
|
|
|
revoke(Obj_crd(pt, 0, Obj_crd::RIGHT_PT_CTRL));
|
2014-09-05 15:00:31 +00:00
|
|
|
else
|
|
|
|
revoke(Obj_crd(pt, 0));
|
2013-02-22 14:26:43 +00:00
|
|
|
|
2013-09-26 12:31:56 +00:00
|
|
|
return res;
|
2013-02-22 14:26:43 +00:00
|
|
|
}
|
2012-12-04 08:49:01 +00:00
|
|
|
|
2013-09-26 12:31:56 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/************************
|
|
|
|
** Exception handlers **
|
|
|
|
************************/
|
|
|
|
|
|
|
|
template <uint8_t EV>
|
|
|
|
void Exception_handlers::register_handler(Pager_object *obj, Mtd mtd,
|
|
|
|
void (* __attribute__((regparm(1))) func)(addr_t))
|
|
|
|
{
|
|
|
|
unsigned use_cpu = obj->location.xpos();
|
|
|
|
if (!kernel_hip()->is_cpu_enabled(use_cpu) || !pager_threads[use_cpu])
|
|
|
|
throw Rm_session::Invalid_thread();
|
|
|
|
|
|
|
|
addr_t const ec_sel = pager_threads[use_cpu]->tid().ec_sel;
|
|
|
|
|
|
|
|
/* compiler generates instance of exception entry if not specified */
|
|
|
|
addr_t entry = func ? (addr_t)func : (addr_t)(&_handler<EV>);
|
|
|
|
uint8_t res = create_portal(obj->exc_pt_sel_client() + EV,
|
|
|
|
__core_pd_sel, ec_sel, mtd, entry,
|
|
|
|
reinterpret_cast<addr_t>(obj));
|
|
|
|
if (res != Nova::NOVA_OK)
|
|
|
|
throw Rm_session::Invalid_thread();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
template <uint8_t EV>
|
|
|
|
void Exception_handlers::_handler(addr_t obj)
|
|
|
|
{
|
|
|
|
Pager_object * pager_obj = reinterpret_cast<Pager_object *>(obj);
|
|
|
|
pager_obj->exception(EV);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Exception_handlers::Exception_handlers(Pager_object *obj)
|
|
|
|
{
|
|
|
|
register_handler<0>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<1>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<2>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<3>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<4>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<5>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<6>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<7>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<8>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<9>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<10>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<11>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<12>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<13>(obj, Mtd(Mtd::EIP));
|
|
|
|
|
|
|
|
register_handler<15>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<16>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<17>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<18>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<19>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<20>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<21>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<22>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<23>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<24>(obj, Mtd(Mtd::EIP));
|
|
|
|
register_handler<25>(obj, Mtd(Mtd::EIP));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/******************
|
|
|
|
** Pager object **
|
|
|
|
******************/
|
|
|
|
|
|
|
|
|
2013-08-07 20:16:58 +00:00
|
|
|
Pager_object::Pager_object(unsigned long badge, Affinity::Location location)
|
2013-11-14 09:35:44 +00:00
|
|
|
:
|
2014-09-05 15:00:31 +00:00
|
|
|
_badge(badge),
|
|
|
|
_selectors(cap_map()->insert(2)),
|
|
|
|
_client_exc_pt_sel(cap_map()->insert(NUM_INITIAL_PT_LOG2)),
|
|
|
|
_client_exc_vcpu(Native_thread::INVALID_INDEX),
|
|
|
|
_exceptions(this),
|
|
|
|
location(location)
|
2011-12-22 15:19:25 +00:00
|
|
|
{
|
2012-08-23 09:02:31 +00:00
|
|
|
uint8_t res;
|
|
|
|
|
|
|
|
addr_t pd_sel = __core_pd_sel;
|
2013-09-26 13:16:01 +00:00
|
|
|
_state._status = 0;
|
2012-08-23 09:02:31 +00:00
|
|
|
_state.sel_client_ec = Native_thread::INVALID_INDEX;
|
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
if (Native_thread::INVALID_INDEX == _selectors ||
|
|
|
|
Native_thread::INVALID_INDEX == _client_exc_pt_sel)
|
|
|
|
throw Rm_session::Invalid_thread();
|
2013-09-23 07:07:33 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/* ypos information not supported by now */
|
|
|
|
if (location.ypos()) {
|
|
|
|
PWRN("Unsupported location %ux%u", location.xpos(), location.ypos());
|
|
|
|
throw Rm_session::Invalid_thread();
|
2012-08-23 09:02:31 +00:00
|
|
|
}
|
2011-12-22 15:19:25 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/* place Pager_object on specified CPU by selecting proper pager thread */
|
|
|
|
unsigned use_cpu = location.xpos();
|
|
|
|
if (!kernel_hip()->is_cpu_enabled(use_cpu) || !pager_threads[use_cpu])
|
|
|
|
throw Rm_session::Invalid_thread();
|
2011-12-22 15:19:25 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
addr_t ec_sel = pager_threads[use_cpu]->tid().ec_sel;
|
2012-08-23 09:02:31 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/* create portal for page-fault handler - 14 */
|
|
|
|
_exceptions.register_handler<14>(this, Mtd::QUAL | Mtd::EIP,
|
|
|
|
_page_fault_handler);
|
2011-12-22 15:19:25 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/* create portal for startup handler - 26 */
|
|
|
|
Mtd const mtd_startup(Mtd::ESP | Mtd::EIP);
|
|
|
|
_exceptions.register_handler<PT_SEL_STARTUP>(this, mtd_startup,
|
|
|
|
_startup_handler);
|
|
|
|
|
|
|
|
/* create portal for recall handler - 31 */
|
|
|
|
Mtd const mtd_recall(Mtd::ESP | Mtd::EIP | Mtd::ACDB | Mtd::EFL |
|
|
|
|
Mtd::EBSD | Mtd::FSGS);
|
|
|
|
_exceptions.register_handler<PT_SEL_RECALL>(this, mtd_recall,
|
|
|
|
_recall_handler);
|
2012-08-24 08:25:24 +00:00
|
|
|
|
2013-09-23 07:07:33 +00:00
|
|
|
/*
|
|
|
|
* Create semaphore required for Genode locking. It can be later on
|
|
|
|
* requested by the thread the same way as all exception portals.
|
|
|
|
*/
|
|
|
|
res = Nova::create_sm(exc_pt_sel_client() + SM_SEL_EC, pd_sel, 0);
|
|
|
|
if (res != Nova::NOVA_OK) {
|
2014-09-05 15:00:31 +00:00
|
|
|
throw Rm_session::Invalid_thread();
|
2013-09-23 07:07:33 +00:00
|
|
|
}
|
|
|
|
|
2013-01-11 22:10:21 +00:00
|
|
|
/* create portal for final cleanup call used during destruction */
|
2014-09-05 15:00:31 +00:00
|
|
|
res = create_portal(sel_pt_cleanup(), pd_sel, ec_sel, Mtd(0),
|
|
|
|
reinterpret_cast<addr_t>(_invoke_handler),
|
|
|
|
reinterpret_cast<addr_t>(this));
|
|
|
|
if (res != Nova::NOVA_OK) {
|
2012-08-24 08:25:24 +00:00
|
|
|
PERR("could not create pager cleanup portal, error = %u\n", res);
|
2014-09-05 15:00:31 +00:00
|
|
|
throw Rm_session::Invalid_thread();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* used to notify caller of as soon as pause succeeded */
|
|
|
|
res = Nova::create_sm(sel_sm_notify(), pd_sel, 0);
|
|
|
|
if (res != Nova::NOVA_OK) {
|
|
|
|
throw Rm_session::Invalid_thread();
|
2012-08-24 08:25:24 +00:00
|
|
|
}
|
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/* semaphore used to block paged thread during page fault or recall */
|
|
|
|
res = Nova::create_sm(sel_sm_block(), pd_sel, 0);
|
2012-08-24 08:25:24 +00:00
|
|
|
if (res != Nova::NOVA_OK) {
|
2014-09-05 15:00:31 +00:00
|
|
|
throw Rm_session::Invalid_thread();
|
2012-08-24 08:25:24 +00:00
|
|
|
}
|
2012-06-19 13:54:41 +00:00
|
|
|
}
|
2011-12-22 15:19:25 +00:00
|
|
|
|
2013-01-11 22:10:21 +00:00
|
|
|
|
2011-12-22 15:19:25 +00:00
|
|
|
Pager_object::~Pager_object()
|
|
|
|
{
|
2014-09-05 15:00:31 +00:00
|
|
|
/* sanity check that object got dissolved already - otherwise bug */
|
|
|
|
if (!_state.dissolved())
|
|
|
|
nova_die();
|
2013-01-16 15:07:04 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/* revoke portal used for the cleanup call and sm cap for blocking state */
|
|
|
|
revoke(Obj_crd(_selectors, 2));
|
|
|
|
cap_map()->remove(_selectors, 2, false);
|
2013-09-11 08:45:23 +00:00
|
|
|
cap_map()->remove(exc_pt_sel_client(), NUM_INITIAL_PT_LOG2, false);
|
2013-11-14 09:35:44 +00:00
|
|
|
|
|
|
|
if (_client_exc_vcpu == Native_thread::INVALID_INDEX)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* revoke vCPU exception portals */
|
|
|
|
revoke(Obj_crd(_client_exc_vcpu, NUM_INITIAL_VCPU_PT_LOG2));
|
|
|
|
cap_map()->remove(_client_exc_vcpu, NUM_INITIAL_VCPU_PT_LOG2, false);
|
2011-12-22 15:19:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
/**********************
|
|
|
|
** Pager activation **
|
|
|
|
**********************/
|
|
|
|
|
|
|
|
Pager_activation_base::Pager_activation_base(const char *name, size_t stack_size)
|
|
|
|
:
|
2015-03-27 13:05:55 +00:00
|
|
|
Thread_base(Cpu_session::DEFAULT_WEIGHT, name, stack_size),
|
2014-09-05 15:00:31 +00:00
|
|
|
_cap(Native_capability()), _ep(0), _cap_valid(Lock::LOCKED)
|
|
|
|
{
|
|
|
|
/* tell thread starting code on which CPU to let run the pager */
|
|
|
|
reinterpret_cast<Affinity::Location *>(stack_base())[0] = Affinity::Location(which_cpu(this), 0);
|
|
|
|
|
|
|
|
/* creates local EC */
|
|
|
|
Thread_base::start();
|
|
|
|
|
|
|
|
reinterpret_cast<Nova::Utcb *>(Thread_base::utcb())->crd_xlt = Obj_crd(0, ~0UL);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Pager_activation_base::entry() { }
|
|
|
|
|
|
|
|
|
|
|
|
/**********************
|
|
|
|
** Pager entrypoint **
|
|
|
|
**********************/
|
|
|
|
|
|
|
|
|
|
|
|
Pager_entrypoint::Pager_entrypoint(Cap_session *cap_session,
|
|
|
|
Pager_activation_base *a)
|
|
|
|
: _activation(a), _cap_session(cap_session)
|
|
|
|
{
|
|
|
|
/* sanity check space for pager threads */
|
|
|
|
if (kernel_hip()->cpu_max() > PAGER_CPUS) {
|
|
|
|
PERR("kernel supports more CPUs (%u) than Genode (%u)",
|
|
|
|
kernel_hip()->cpu_max(), PAGER_CPUS);
|
|
|
|
nova_die();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* determine boot cpu */
|
|
|
|
unsigned master_cpu = boot_cpu();
|
|
|
|
|
|
|
|
/* detect enabled CPUs and create per CPU a pager thread */
|
|
|
|
typedef Pager_activation<PAGER_STACK_SIZE> Pager;
|
|
|
|
Pager * pager_of_cpu = reinterpret_cast<Pager *>(&pager_activation_mem);
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < kernel_hip()->cpu_max(); i++, pager_of_cpu++) {
|
|
|
|
if (i == master_cpu) {
|
|
|
|
pager_threads[master_cpu] = a;
|
|
|
|
a->ep(this);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!kernel_hip()->is_cpu_enabled(i))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
pager_threads[i] = pager_of_cpu;
|
|
|
|
construct_at<Pager>(pager_threads[i]);
|
|
|
|
pager_threads[i]->ep(this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-12-22 15:19:25 +00:00
|
|
|
Pager_capability Pager_entrypoint::manage(Pager_object *obj)
|
|
|
|
{
|
2014-09-05 15:00:31 +00:00
|
|
|
/* let handle pager_object of pager thread on same CPU */
|
|
|
|
unsigned use_cpu = obj->location.xpos();
|
|
|
|
if (!kernel_hip()->is_cpu_enabled(use_cpu) || !pager_threads[use_cpu]) {
|
|
|
|
PWRN("invalid CPU parameter used in pager object");
|
|
|
|
return Pager_capability();
|
|
|
|
}
|
|
|
|
Native_capability pager_thread_cap(pager_threads[use_cpu]->tid().ec_sel);
|
|
|
|
|
2012-08-08 12:23:13 +00:00
|
|
|
/* request creation of portal bind to pager thread */
|
2012-07-30 08:56:07 +00:00
|
|
|
Native_capability cap_session =
|
|
|
|
_cap_session->alloc(pager_thread_cap, obj->handler_address());
|
2011-12-22 15:19:25 +00:00
|
|
|
|
2014-09-05 15:00:31 +00:00
|
|
|
if (NOVA_OK != pt_ctrl(cap_session.local_name(), reinterpret_cast<mword_t>(obj)))
|
|
|
|
nova_die();
|
|
|
|
|
|
|
|
/* disable the feature for security reasons now */
|
2015-01-05 21:09:51 +00:00
|
|
|
revoke(Obj_crd(cap_session.local_name(), 0, Obj_crd::RIGHT_PT_CTRL));
|
|
|
|
|
2011-12-22 15:19:25 +00:00
|
|
|
/* add server object to object pool */
|
2012-07-12 07:35:37 +00:00
|
|
|
obj->Object_pool<Pager_object>::Entry::cap(cap_session);
|
2011-12-22 15:19:25 +00:00
|
|
|
insert(obj);
|
|
|
|
|
|
|
|
/* return capability that uses the object id as badge */
|
2012-08-08 12:23:13 +00:00
|
|
|
return reinterpret_cap_cast<Pager_object>(
|
|
|
|
obj->Object_pool<Pager_object>::Entry::cap());
|
2011-12-22 15:19:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Pager_entrypoint::dissolve(Pager_object *obj)
|
|
|
|
{
|
2012-12-04 08:49:01 +00:00
|
|
|
Native_capability pager_obj = obj->Object_pool<Pager_object>::Entry::cap();
|
2012-08-08 12:23:13 +00:00
|
|
|
/* cleanup at cap session */
|
2012-12-04 08:49:01 +00:00
|
|
|
_cap_session->free(pager_obj);
|
2014-09-05 15:00:31 +00:00
|
|
|
/* revoke cap selector locally */
|
2012-12-04 08:49:01 +00:00
|
|
|
revoke(pager_obj.dst(), true);
|
2014-09-05 15:00:31 +00:00
|
|
|
/* remove object from pool */
|
2012-12-14 09:03:55 +00:00
|
|
|
remove_locked(obj);
|
2014-09-05 15:00:31 +00:00
|
|
|
/* take care that no faults are in-flight */
|
2012-12-04 08:49:01 +00:00
|
|
|
obj->cleanup_call();
|
2011-12-22 15:19:25 +00:00
|
|
|
}
|