2011-12-22 16:19:25 +01:00
|
|
|
/*
|
|
|
|
* \brief Lock implementation
|
|
|
|
* \author Norman Feske
|
|
|
|
* \date 2007-10-15
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
2013-01-10 21:44:47 +01:00
|
|
|
* Copyright (C) 2007-2013 Genode Labs GmbH
|
2011-12-22 16:19:25 +01: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/cancelable_lock.h>
|
|
|
|
#include <cpu/atomic.h>
|
2014-11-14 10:44:56 +01:00
|
|
|
#include <cpu/memory_barrier.h>
|
2011-12-22 16:19:25 +01:00
|
|
|
#include <base/printf.h>
|
|
|
|
|
|
|
|
/* L4/Fiasco includes */
|
|
|
|
namespace Fiasco {
|
|
|
|
#include <l4/sys/ipc.h>
|
|
|
|
}
|
|
|
|
|
|
|
|
using namespace Genode;
|
|
|
|
|
|
|
|
|
|
|
|
Cancelable_lock::Cancelable_lock(Cancelable_lock::State initial)
|
2016-01-22 22:00:54 +01:00
|
|
|
: _state(UNLOCKED), _owner(nullptr)
|
2011-12-22 16:19:25 +01:00
|
|
|
{
|
|
|
|
if (initial == LOCKED)
|
|
|
|
lock();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Cancelable_lock::lock()
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* XXX: How to notice cancel-blocking signals issued when being outside the
|
|
|
|
* 'l4_ipc_sleep' system call?
|
|
|
|
*/
|
2016-01-22 22:00:54 +01:00
|
|
|
while (!Genode::cmpxchg(&_state, UNLOCKED, LOCKED))
|
2011-12-22 16:19:25 +01:00
|
|
|
if (Fiasco::l4_ipc_sleep(Fiasco::l4_ipc_timeout(0, 0, 500, 0)) != L4_IPC_RETIMEOUT)
|
|
|
|
throw Genode::Blocking_canceled();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Cancelable_lock::unlock()
|
|
|
|
{
|
2014-11-14 10:44:56 +01:00
|
|
|
Genode::memory_barrier();
|
2016-01-22 22:00:54 +01:00
|
|
|
_state = UNLOCKED;
|
2011-12-22 16:19:25 +01:00
|
|
|
}
|