/*
* \brief Fiasco.OC-specific supplements to the IPC framework
* \author Norman Feske
* \author Stefan Kalkowski
* \date 2010-01-27
*/
/*
* Copyright (C) 2010-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.
*/
#ifndef _INCLUDE__BASE__IPC_H_
#define _INCLUDE__BASE__IPC_H_
#include
namespace Fiasco {
#include
#include
}
inline void Genode::Ipc_ostream::_marshal_capability(Genode::Native_capability const &cap)
{
bool local = cap.local();
long id = local ? (long)cap.local() : cap.local_name();
_write_to_buf(local);
_write_to_buf(id);
/* only transfer kernel-capability if it's no local capability and valid */
if (!local && id)
_snd_msg->snd_append_cap_sel(cap.dst());
}
inline void Genode::Ipc_istream::_unmarshal_capability(Genode::Native_capability &cap)
{
using namespace Fiasco;
bool local = false;
long id = 0;
/* extract capability id from message buffer, and whether it's a local cap */
_read_from_buf(local);
_read_from_buf(id);
/* if it's a local capability, the pointer is marshalled in the id */
if (local) {
cap = Capability::local_cap((Native_capability*)id);
return;
}
/* if id is zero an invalid capability was tranfered */
if (!id) {
cap = Native_capability();
return;
}
/* we received a valid, non-local capability, maybe we already own it? */
Cap_index *i = cap_map()->find(id);
bool map = false;
if (i) {
/**
* If we've a dead capability in our database, which is already
* revoked, its id might be reused.
*/
l4_msgtag_t tag = Fiasco::l4_task_cap_valid(L4_BASE_TASK_CAP, i->kcap());
if (!tag.label())
map = true;
} else {
/* insert the new capability in the map */
i = cap_map()->insert(id);
map = true;
}
/* map the received capability from the receive-buffer if necessary */
if (map)
l4_task_map(L4_BASE_TASK_CAP, L4_BASE_TASK_CAP,
l4_obj_fpage(_rcv_msg->rcv_cap_sel(), 0, L4_FPAGE_RWX),
i->kcap() | L4_ITEM_MAP | L4_MAP_ITEM_GRANT);
cap = Native_capability(i);
}
#endif /* _INCLUDE__BASE__IPC_H_ */