1
# Copyright (C) 2005-2010 Canonical Ltd
1
# Copyright (C) 2005, 2006, 2007, 2008, 2009 Canonical Ltd
3
3
# This program is free software; you can redistribute it and/or modify
4
4
# it under the terms of the GNU General Public License as published by
48
from bzrlib.hooks import Hooks
49
from bzrlib.i18n import gettext
48
from bzrlib.hooks import HookPoint, Hooks
51
51
class LockHooks(Hooks):
53
53
def __init__(self):
54
Hooks.__init__(self, "bzrlib.lock", "Lock.hooks")
55
self.add_hook('lock_acquired',
56
"Called with a bzrlib.lock.LockResult when a physical lock is "
58
self.add_hook('lock_released',
59
"Called with a bzrlib.lock.LockResult when a physical lock is "
61
self.add_hook('lock_broken',
62
"Called with a bzrlib.lock.LockResult when a physical lock is "
55
self.create_hook(HookPoint('lock_acquired',
56
"Called with a bzrlib.lock.LockResult when a physical lock is "
57
"acquired.", (1, 8), None))
58
self.create_hook(HookPoint('lock_released',
59
"Called with a bzrlib.lock.LockResult when a physical lock is "
60
"released.", (1, 8), None))
61
self.create_hook(HookPoint('lock_broken',
62
"Called with a bzrlib.lock.LockResult when a physical lock is "
63
"broken.", (1, 15), None))
66
66
class Lock(object):
84
84
return self.lock_url == other.lock_url and self.details == other.details
86
86
def __repr__(self):
87
return '%s(%s, %s)' % (self.__class__.__name__,
87
return '%s(%s%s)' % (self.__class__.__name__,
88
88
self.lock_url, self.details)
91
class LogicalLockResult(object):
92
"""The result of a lock_read/lock_write/lock_tree_write call on lockables.
94
:ivar unlock: A callable which will unlock the lock.
97
def __init__(self, unlock):
101
return "LogicalLockResult(%s)" % (self.unlock)
105
91
def cant_unlock_not_held(locked_object):
106
92
"""An attempt to unlock failed because the object was not locked.
198
190
if self.filename in _fcntl_WriteLock._open_locks:
200
192
raise errors.LockContention(self.filename)
201
if self.filename in _fcntl_ReadLock._open_locks:
202
if 'strict_locks' in debug.debug_flags:
204
raise errors.LockContention(self.filename)
206
trace.mutter('Write lock taken w/ an open read lock on: %s'
209
194
self._open(self.filename, 'rb+')
210
195
# reserve a slot for this lock - even if the lockf call fails,
211
# at this point unlock() will be called, because self.f is set.
196
# at thisi point unlock() will be called, because self.f is set.
212
197
# TODO: make this fully threadsafe, if we decide we care.
213
198
_fcntl_WriteLock._open_locks.add(self.filename)
235
220
def __init__(self, filename):
236
221
super(_fcntl_ReadLock, self).__init__()
237
222
self.filename = osutils.realpath(filename)
238
if self.filename in _fcntl_WriteLock._open_locks:
239
if 'strict_locks' in debug.debug_flags:
240
# We raise before calling _open so we don't need to
242
raise errors.LockContention(self.filename)
244
trace.mutter('Read lock taken w/ an open write lock on: %s'
246
223
_fcntl_ReadLock._open_locks.setdefault(self.filename, 0)
247
224
_fcntl_ReadLock._open_locks[self.filename] += 1
248
225
self._open(filename, 'rb')
441
418
DWORD, # dwFlagsAndAttributes
442
419
HANDLE # hTemplateFile
443
420
)((_function_name, ctypes.windll.kernel32))
445
422
INVALID_HANDLE_VALUE = -1
447
424
GENERIC_READ = 0x80000000
448
425
GENERIC_WRITE = 0x40000000
449
426
FILE_SHARE_READ = 1
451
428
FILE_ATTRIBUTE_NORMAL = 128
453
430
ERROR_ACCESS_DENIED = 5
454
431
ERROR_SHARING_VIOLATION = 32
526
503
# We default to using the first available lock class.
527
504
_lock_type, WriteLock, ReadLock = _lock_classes[0]
530
class _RelockDebugMixin(object):
531
"""Mixin support for -Drelock flag.
533
Add this as a base class then call self._note_lock with 'r' or 'w' when
534
acquiring a read- or write-lock. If this object was previously locked (and
535
locked the same way), and -Drelock is set, then this will trace.note a
541
def _note_lock(self, lock_type):
542
if 'relock' in debug.debug_flags and self._prev_lock == lock_type:
547
trace.note(gettext('{0!r} was {1} locked again'), self, type_name)
548
self._prev_lock = lock_type