feat(util/shared_mutex): add support for recursive lock at shared_mutex

We need support for recursive locks. The main user of this class is
the environment object. This commit adds a test that demonstrates that
the shared_lock of the environment object may be recursively requested.
Before this fix, the Lean was deadlocking in this example.

Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
Leonardo de Moura 2013-11-15 22:01:06 -08:00
parent 209a2d10f7
commit 9398ea5a59
4 changed files with 61 additions and 2 deletions

View file

@ -10,27 +10,42 @@
Hinnant. The proposal is also part of the Boost library which is Hinnant. The proposal is also part of the Boost library which is
licensed under http://www.boost.org/LICENSE_1_0.txt licensed under http://www.boost.org/LICENSE_1_0.txt
*/ */
#include "util/debug.h"
#include "util/shared_mutex.h" #include "util/shared_mutex.h"
namespace lean { namespace lean {
shared_mutex::shared_mutex():m_state(0) {} shared_mutex::shared_mutex():m_rw_counter(0), m_state(0) {}
shared_mutex::~shared_mutex() { shared_mutex::~shared_mutex() {
std::lock_guard<std::mutex> lock(m_mutex); std::lock_guard<std::mutex> lock(m_mutex);
} }
void shared_mutex::lock() { void shared_mutex::lock() {
std::unique_lock<std::mutex> lock(m_mutex); std::unique_lock<std::mutex> lock(m_mutex);
if (m_rw_owner == std::this_thread::get_id()) {
m_rw_counter++;
return; // already has the lock
}
while (m_state & write_entered) while (m_state & write_entered)
m_gate1.wait(lock); m_gate1.wait(lock);
m_state |= write_entered; m_state |= write_entered;
while (m_state & readers) while (m_state & readers)
m_gate2.wait(lock); m_gate2.wait(lock);
lean_assert(m_rw_counter == 0);
m_rw_owner = std::this_thread::get_id();
m_rw_counter = 1;
} }
bool shared_mutex::try_lock() { bool shared_mutex::try_lock() {
std::unique_lock<std::mutex> lock(m_mutex); std::unique_lock<std::mutex> lock(m_mutex);
if (m_rw_owner == std::this_thread::get_id()) {
m_rw_counter++;
return true; // already has the lock
}
if (m_state == 0) { if (m_state == 0) {
m_state = write_entered; m_state = write_entered;
lean_assert(m_rw_counter == 0);
m_rw_owner = std::this_thread::get_id();
m_rw_counter = 1;
return true; return true;
} }
return false; return false;
@ -38,12 +53,25 @@ bool shared_mutex::try_lock() {
void shared_mutex::unlock() { void shared_mutex::unlock() {
std::lock_guard<std::mutex> lock(m_mutex); std::lock_guard<std::mutex> lock(m_mutex);
lean_assert(m_rw_owner == std::this_thread::get_id());
lean_assert(m_rw_counter > 0);
m_rw_counter--;
if (m_rw_counter > 0)
return; // keep the lock
m_rw_owner = std::thread::id(); // reset owner
lean_assert(m_rw_counter == 0);
lean_assert(m_rw_owner != std::this_thread::get_id());
m_state = 0; m_state = 0;
m_gate1.notify_all(); m_gate1.notify_all();
} }
void shared_mutex::lock_shared() { void shared_mutex::lock_shared() {
std::unique_lock<std::mutex> lock(m_mutex); std::unique_lock<std::mutex> lock(m_mutex);
if (m_rw_owner == std::this_thread::get_id()) {
lean_assert(m_rw_counter > 0);
m_rw_counter++;
return; // already has the lock
}
while ((m_state & write_entered) || (m_state & readers) == readers) while ((m_state & write_entered) || (m_state & readers) == readers)
m_gate1.wait(lock); m_gate1.wait(lock);
unsigned num_readers = (m_state & readers) + 1; unsigned num_readers = (m_state & readers) + 1;
@ -53,6 +81,11 @@ void shared_mutex::lock_shared() {
bool shared_mutex::try_lock_shared() { bool shared_mutex::try_lock_shared() {
std::unique_lock<std::mutex> lock(m_mutex); std::unique_lock<std::mutex> lock(m_mutex);
if (m_rw_owner == std::this_thread::get_id()) {
lean_assert(m_rw_counter > 0);
m_rw_counter++;
return true; // already has the lock
}
unsigned num_readers = m_state & readers; unsigned num_readers = m_state & readers;
if (!(m_state & write_entered) && num_readers != readers) { if (!(m_state & write_entered) && num_readers != readers) {
++num_readers; ++num_readers;
@ -65,6 +98,14 @@ bool shared_mutex::try_lock_shared() {
void shared_mutex::unlock_shared() { void shared_mutex::unlock_shared() {
std::lock_guard<std::mutex> lock(m_mutex); std::lock_guard<std::mutex> lock(m_mutex);
if (m_rw_owner == std::this_thread::get_id()) {
// if we have a rw (aka unshared) lock, then
// the shared lock must be nested.
lean_assert(m_rw_counter > 1);
m_rw_counter--;
return;
}
lean_assert(m_rw_counter == 0);
unsigned num_readers = (m_state & readers) - 1; unsigned num_readers = (m_state & readers) - 1;
m_state &= ~readers; m_state &= ~readers;
m_state |= num_readers; m_state |= num_readers;

View file

@ -11,12 +11,15 @@
licensed under http://www.boost.org/LICENSE_1_0.txt licensed under http://www.boost.org/LICENSE_1_0.txt
*/ */
#include <mutex> #include <mutex>
#include <thread>
#include <condition_variable> #include <condition_variable>
#include <climits> #include <climits>
namespace lean { namespace lean {
class shared_mutex { class shared_mutex {
std::mutex m_mutex; std::mutex m_mutex;
std::thread::id m_rw_owner;
unsigned m_rw_counter;
std::condition_variable m_gate1; std::condition_variable m_gate1;
std::condition_variable m_gate2; std::condition_variable m_gate2;
unsigned m_state; unsigned m_state;

11
tests/lean/loop2.lean Normal file
View file

@ -0,0 +1,11 @@
(**
function add_paren(code)
return "(" .. "** " .. code .. " **" .. ")"
end
parse_lean_cmds(add_paren([[
local env = get_environment()
env:add_var("x", Const("Int"))
print(env:find_object("x"))
]]))
print("done")
**)

View file

@ -0,0 +1,4 @@
Set: pp::colors
Set: pp::unicode
Variable x : Int
done