/* * \brief NOVA-specific signal-source client interface * \author Norman Feske * \date 2010-02-03 * * On NOVA, the signal source server does not provide a blocking * 'wait_for_signal' function because this kernel does no support * out-of-order IPC replies. Instead, we use a shared semaphore * to let the client block until a signal is present at the * server. The shared semaphore gets initialized with the first * call of 'wait_for_signal()'. */ /* * Copyright (C) 2010-2013 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__SIGNAL_SOURCE__CLIENT_H_ #define _INCLUDE__SIGNAL_SOURCE__CLIENT_H_ /* Genode includes */ #include #include /* base-internal includes */ #include /* NOVA includes */ #include #include namespace Genode { class Signal_source_client : public Rpc_client { private: /** * Capability referring to a NOVA semaphore */ Native_capability _sem; public: /** * Constructor */ Signal_source_client(Capability cap) : Rpc_client(static_cap_cast(cap)) { /* request mapping of semaphore capability selector */ Thread * myself = Thread::myself(); request_signal_sm_cap(Native_capability(myself->native_thread().ec_sel + 1), myself->native_thread().exc_pt_sel + Nova::PT_SEL_STARTUP); _sem = Native_capability(myself->native_thread().exc_pt_sel + Nova::PT_SEL_STARTUP); call(_sem); } ~Signal_source_client() { Nova::revoke(Nova::Obj_crd(_sem.local_name(), 0)); } /***************************** ** Signal source interface ** *****************************/ Signal wait_for_signal() override { using namespace Nova; mword_t imprint, count; do { /* * We set an invalid imprint (0) to detect a spurious * unblock. In this case, NOVA does not block * SEMAPHORE_DOWN nor touch our input values if the * deblocking (chained) semaphore was dequeued before we * intend to block. */ imprint = 0; count = 0; /* block on semaphore until signal context was submitted */ if (uint8_t res = si_ctrl(_sem.local_name(), SEMAPHORE_DOWN, imprint, count)) PWRN("signal reception failed - error %u", res); } while (imprint == 0); return Signal(imprint, count); } }; } #endif /* _INCLUDE__SIGNAL_SOURCE__CLIENT_H_ */