Issue #15124: Optimize _thread.LockType deletion and acquisition when

not contested, similar to what _thread.RLock already has.
This commit is contained in:
Kristjan Valur Jonsson 2012-06-22 18:40:02 +00:00
parent 878054e97b
commit 69cf913ba1
1 changed files with 17 additions and 21 deletions

View File

@ -23,6 +23,7 @@ typedef struct {
PyObject_HEAD PyObject_HEAD
PyThread_type_lock lock_lock; PyThread_type_lock lock_lock;
PyObject *in_weakreflist; PyObject *in_weakreflist;
char locked; /* for sanity checking */
} lockobject; } lockobject;
static void static void
@ -32,9 +33,8 @@ lock_dealloc(lockobject *self)
PyObject_ClearWeakRefs((PyObject *) self); PyObject_ClearWeakRefs((PyObject *) self);
if (self->lock_lock != NULL) { if (self->lock_lock != NULL) {
/* Unlock the lock so it's safe to free it */ /* Unlock the lock so it's safe to free it */
PyThread_acquire_lock(self->lock_lock, 0); if (self->locked)
PyThread_release_lock(self->lock_lock); PyThread_release_lock(self->lock_lock);
PyThread_free_lock(self->lock_lock); PyThread_free_lock(self->lock_lock);
} }
PyObject_Del(self); PyObject_Del(self);
@ -62,9 +62,13 @@ acquire_timed(PyThread_type_lock lock, PY_TIMEOUT_T microseconds)
do { do {
/* first a simple non-blocking try without releasing the GIL */
r = PyThread_acquire_lock_timed(lock, 0, 0);
if (r == PY_LOCK_FAILURE && microseconds != 0) {
Py_BEGIN_ALLOW_THREADS Py_BEGIN_ALLOW_THREADS
r = PyThread_acquire_lock_timed(lock, microseconds, 1); r = PyThread_acquire_lock_timed(lock, microseconds, 1);
Py_END_ALLOW_THREADS Py_END_ALLOW_THREADS
}
if (r == PY_LOCK_INTR) { if (r == PY_LOCK_INTR) {
/* Run signal handlers if we were interrupted. Propagate /* Run signal handlers if we were interrupted. Propagate
@ -135,6 +139,8 @@ lock_PyThread_acquire_lock(lockobject *self, PyObject *args, PyObject *kwds)
return NULL; return NULL;
} }
if (r == PY_LOCK_ACQUIRED)
self->locked = 1;
return PyBool_FromLong(r == PY_LOCK_ACQUIRED); return PyBool_FromLong(r == PY_LOCK_ACQUIRED);
} }
@ -153,13 +159,13 @@ static PyObject *
lock_PyThread_release_lock(lockobject *self) lock_PyThread_release_lock(lockobject *self)
{ {
/* Sanity check: the lock must be locked */ /* Sanity check: the lock must be locked */
if (PyThread_acquire_lock(self->lock_lock, 0)) { if (!self->locked) {
PyThread_release_lock(self->lock_lock);
PyErr_SetString(ThreadError, "release unlocked lock"); PyErr_SetString(ThreadError, "release unlocked lock");
return NULL; return NULL;
} }
PyThread_release_lock(self->lock_lock); PyThread_release_lock(self->lock_lock);
self->locked = 0;
Py_INCREF(Py_None); Py_INCREF(Py_None);
return Py_None; return Py_None;
} }
@ -175,11 +181,7 @@ but it needn't be locked by the same thread that unlocks it.");
static PyObject * static PyObject *
lock_locked_lock(lockobject *self) lock_locked_lock(lockobject *self)
{ {
if (PyThread_acquire_lock(self->lock_lock, 0)) { return PyBool_FromLong((long)self->locked);
PyThread_release_lock(self->lock_lock);
return PyBool_FromLong(0L);
}
return PyBool_FromLong(1L);
} }
PyDoc_STRVAR(locked_doc, PyDoc_STRVAR(locked_doc,
@ -313,14 +315,7 @@ rlock_acquire(rlockobject *self, PyObject *args, PyObject *kwds)
self->rlock_count = count; self->rlock_count = count;
Py_RETURN_TRUE; Py_RETURN_TRUE;
} }
if (self->rlock_count > 0 ||
!PyThread_acquire_lock(self->rlock_lock, 0)) {
if (microseconds == 0) {
Py_RETURN_FALSE;
}
r = acquire_timed(self->rlock_lock, microseconds); r = acquire_timed(self->rlock_lock, microseconds);
}
if (r == PY_LOCK_ACQUIRED) { if (r == PY_LOCK_ACQUIRED) {
assert(self->rlock_count == 0); assert(self->rlock_count == 0);
self->rlock_owner = tid; self->rlock_owner = tid;
@ -548,6 +543,7 @@ newlockobject(void)
if (self == NULL) if (self == NULL)
return NULL; return NULL;
self->lock_lock = PyThread_allocate_lock(); self->lock_lock = PyThread_allocate_lock();
self->locked = 0;
self->in_weakreflist = NULL; self->in_weakreflist = NULL;
if (self->lock_lock == NULL) { if (self->lock_lock == NULL) {
Py_DECREF(self); Py_DECREF(self);