1
/* - mode: c; c-basic-offset: 2; indent-tabs-mode: nil; -*-
2
* vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
4
* Copyright (C) 2010 Brian Aker
6
* This program is free software; you can redistribute it and/or modify
7
* it under the terms of the GNU General Public License as published by
8
* the Free Software Foundation; either version 2 of the License, or
9
* (at your option) any later version.
11
* This program is distributed in the hope that it will be useful,
12
* but WITHOUT ANY WARRANTY; without even the implied warranty of
13
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14
* GNU General Public License for more details.
16
* You should have received a copy of the GNU General Public License
17
* along with this program; if not, write to the Free Software
18
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
22
#include "plugin/user_locks/module.h"
24
#include <boost/thread/locks.hpp>
28
namespace user_locks {
30
bool Locks::lock(drizzled::session_id_t id_arg, const user_locks::Key &arg, int64_t wait_for)
32
boost::unique_lock<boost::mutex> scope(mutex);
33
boost::system_time timeout= boost::get_system_time() + boost::posix_time::seconds(wait_for);
35
LockMap::iterator iter;
36
while ((iter= lock_map.find(arg)) != lock_map.end())
38
if (id_arg == (*iter).second->id)
40
// We own the lock, so we just exit.
46
bool success= release_cond.timed_wait(scope, timeout);
53
release_cond.wait(scope);
56
catch(boost::thread_interrupted const& error)
58
// Currently nothing is done here.
63
if (iter == lock_map.end())
65
create_cond.notify_all();
66
return lock_map.insert(std::make_pair(arg, new Lock(id_arg))).second;
72
// Currently we just let timeouts occur, and the caller will need to know
73
// what it is looking for/whether to go back into this.
74
void Locks::waitCreate(int64_t wait_for)
76
boost::unique_lock<boost::mutex> scope(mutex);
77
boost::system_time timeout= boost::get_system_time() + boost::posix_time::seconds(wait_for);
81
timed_out= create_cond.timed_wait(scope, timeout);
83
catch(boost::thread_interrupted const& error)
85
// Currently nothing is done here.
90
bool Locks::lock(drizzled::session_id_t id_arg, const user_locks::Keys &arg)
92
boost::unique_lock<boost::mutex> scope(mutex);
93
user_locks::Keys created;
96
for (user_locks::Keys::const_iterator iter= arg.begin(); iter != arg.end(); iter++)
98
LockMap::iterator record= lock_map.find(*iter);
100
if (record != lock_map.end()) // Found, so check ownership of the lock
102
if (id_arg != (*record).second->id)
104
// So it turns out the locks exist, and we can't grab them all
111
lock_map.insert(std::make_pair(*iter, new Lock(id_arg)));
112
created.insert(*iter);
118
for (user_locks::Keys::const_iterator iter= created.begin(); iter != created.end(); iter++)
120
lock_map.erase(*iter);
126
create_cond.notify_all();
131
bool Locks::isUsed(const user_locks::Key &arg, drizzled::session_id_t &id_arg)
133
boost::unique_lock<boost::mutex> scope(mutex);
135
LockMap::iterator iter= lock_map.find(arg);
137
if ( iter == lock_map.end())
140
id_arg= (*iter).second->id;
145
bool Locks::isFree(const user_locks::Key &arg)
147
boost::unique_lock<boost::mutex> scope(mutex);
149
LockMap::iterator iter= lock_map.find(arg);
151
return iter != lock_map.end();
154
void Locks::Copy(LockMap &lock_map_arg)
156
boost::unique_lock<boost::mutex> scope(mutex);
157
lock_map_arg= lock_map;
160
locks::return_t Locks::release(const user_locks::Key &arg, drizzled::session_id_t &id_arg, bool and_wait)
163
boost::unique_lock<boost::mutex> scope(mutex);
164
LockMap::iterator iter= lock_map.find(arg);
167
if ( iter == lock_map.end())
168
return locks::NOT_FOUND;
170
if ((*iter).second->id == id_arg)
172
elements= lock_map.erase(arg);
173
assert(elements); // If we can't find what we just found, then we are broken
177
release_cond.notify_one();
184
assert(boost::this_thread::interruption_enabled());
186
create_cond.wait(scope);
188
catch(boost::thread_interrupted const& error)
190
// Currently nothing is done here.
193
iter= lock_map.find(arg);
195
if (iter != lock_map.end())
200
return locks::SUCCESS;
204
return locks::NOT_OWNED_BY;
207
} /* namespace user_locks */