Skip to content

Commit ad00ce6

Browse files
committed
vendor RWLock
1 parent 5b99264 commit ad00ce6

File tree

2 files changed

+260
-0
lines changed

2 files changed

+260
-0
lines changed

src/ecdsa/_rwlock.py

Lines changed: 85 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,85 @@
1+
# Copyright Mateusz Kobos, (c) 2011
2+
# https://code.activestate.com/recipes/577803-reader-writer-lock-with-priority-for-writers/
3+
# released under the MIT licence
4+
5+
import threading
6+
7+
8+
__author__ = "Mateusz Kobos"
9+
10+
11+
class RWLock:
12+
"""
13+
Read-Write locking primitive
14+
15+
Synchronization object used in a solution of so-called second
16+
readers-writers problem. In this problem, many readers can simultaneously
17+
access a share, and a writer has an exclusive access to this share.
18+
Additionally, the following constraints should be met:
19+
1) no reader should be kept waiting if the share is currently opened for
20+
reading unless a writer is also waiting for the share,
21+
2) no writer should be kept waiting for the share longer than absolutely
22+
necessary.
23+
24+
The implementation is based on [1, secs. 4.2.2, 4.2.6, 4.2.7]
25+
with a modification -- adding an additional lock (C{self.__readers_queue})
26+
-- in accordance with [2].
27+
28+
Sources:
29+
[1] A.B. Downey: "The little book of semaphores", Version 2.1.5, 2008
30+
[2] P.J. Courtois, F. Heymans, D.L. Parnas:
31+
"Concurrent Control with 'Readers' and 'Writers'",
32+
Communications of the ACM, 1971 (via [3])
33+
[3] http://en.wikipedia.org/wiki/Readers-writers_problem
34+
"""
35+
36+
def __init__(self):
37+
"""
38+
A lock giving an even higher priority to the writer in certain
39+
cases (see [2] for a discussion).
40+
"""
41+
self.__read_switch = _LightSwitch()
42+
self.__write_switch = _LightSwitch()
43+
self.__no_readers = threading.Lock()
44+
self.__no_writers = threading.Lock()
45+
self.__readers_queue = threading.Lock()
46+
47+
def reader_acquire(self):
48+
self.__readers_queue.acquire()
49+
self.__no_readers.acquire()
50+
self.__read_switch.acquire(self.__no_writers)
51+
self.__no_readers.release()
52+
self.__readers_queue.release()
53+
54+
def reader_release(self):
55+
self.__read_switch.release(self.__no_writers)
56+
57+
def writer_acquire(self):
58+
self.__write_switch.acquire(self.__no_readers)
59+
self.__no_writers.acquire()
60+
61+
def writer_release(self):
62+
self.__no_writers.release()
63+
self.__write_switch.release(self.__no_readers)
64+
65+
66+
class _LightSwitch:
67+
"""An auxiliary "light switch"-like object. The first thread turns on the
68+
"switch", the last one turns it off (see [1, sec. 4.2.2] for details)."""
69+
def __init__(self):
70+
self.__counter = 0
71+
self.__mutex = threading.Lock()
72+
73+
def acquire(self, lock):
74+
self.__mutex.acquire()
75+
self.__counter += 1
76+
if self.__counter == 1:
77+
lock.acquire()
78+
self.__mutex.release()
79+
80+
def release(self, lock):
81+
self.__mutex.acquire()
82+
self.__counter -= 1
83+
if self.__counter == 0:
84+
lock.release()
85+
self.__mutex.release()

src/ecdsa/test_rw_lock.py

Lines changed: 175 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,175 @@
1+
# Copyright Mateusz Kobos, (c) 2011
2+
# https://code.activestate.com/recipes/577803-reader-writer-lock-with-priority-for-writers/
3+
# released under the MIT licence
4+
5+
import unittest
6+
import threading
7+
import time
8+
import copy
9+
from ._rwlock import RWLock
10+
11+
12+
class Writer(threading.Thread):
13+
def __init__(self, buffer_, rw_lock, init_sleep_time, sleep_time, to_write):
14+
"""
15+
@param buffer_: common buffer_ shared by the readers and writers
16+
@type buffer_: list
17+
@type rw_lock: L{RWLock}
18+
@param init_sleep_time: sleep time before doing any action
19+
@type init_sleep_time: C{float}
20+
@param sleep_time: sleep time while in critical section
21+
@type sleep_time: C{float}
22+
@param to_write: data that will be appended to the buffer
23+
"""
24+
threading.Thread.__init__(self)
25+
self.__buffer = buffer_
26+
self.__rw_lock = rw_lock
27+
self.__init_sleep_time = init_sleep_time
28+
self.__sleep_time = sleep_time
29+
self.__to_write = to_write
30+
self.entry_time = None
31+
"""Time of entry to the critical section"""
32+
self.exit_time = None
33+
"""Time of exit from the critical section"""
34+
35+
def run(self):
36+
time.sleep(self.__init_sleep_time)
37+
self.__rw_lock.writer_acquire()
38+
self.entry_time = time.time()
39+
time.sleep(self.__sleep_time)
40+
self.__buffer.append(self.__to_write)
41+
self.exit_time = time.time()
42+
self.__rw_lock.writer_release()
43+
44+
45+
class Reader(threading.Thread):
46+
def __init__(self, buffer_, rw_lock, init_sleep_time, sleep_time):
47+
"""
48+
@param buffer_: common buffer shared by the readers and writers
49+
@type buffer_: list
50+
@type rw_lock: L{RWLock}
51+
@param init_sleep_time: sleep time before doing any action
52+
@type init_sleep_time: C{float}
53+
@param sleep_time: sleep time while in critical section
54+
@type sleep_time: C{float}
55+
"""
56+
threading.Thread.__init__(self)
57+
self.__buffer = buffer_
58+
self.__rw_lock = rw_lock
59+
self.__init_sleep_time = init_sleep_time
60+
self.__sleep_time = sleep_time
61+
self.buffer_read = None
62+
"""a copy of a the buffer read while in critical section"""
63+
self.entry_time = None
64+
"""Time of entry to the critical section"""
65+
self.exit_time = None
66+
"""Time of exit from the critical section"""
67+
68+
def run(self):
69+
time.sleep(self.__init_sleep_time)
70+
self.__rw_lock.reader_acquire()
71+
self.entry_time = time.time()
72+
time.sleep(self.__sleep_time)
73+
self.buffer_read = copy.deepcopy(self.__buffer)
74+
self.exit_time = time.time()
75+
self.__rw_lock.reader_release()
76+
77+
78+
class RWLockTestCase(unittest.TestCase):
79+
def test_readers_nonexclusive_access(self):
80+
(buffer_, rw_lock, threads) = self.__init_variables()
81+
82+
threads.append(Reader(buffer_, rw_lock, 0, 0))
83+
threads.append(Writer(buffer_, rw_lock, 0.2, 0.4, 1))
84+
threads.append(Reader(buffer_, rw_lock, 0.3, 0.3))
85+
threads.append(Reader(buffer_, rw_lock, 0.5, 0))
86+
87+
self.__start_and_join_threads(threads)
88+
89+
## The third reader should enter after the second one but it should
90+
## exit before the second one exits
91+
## (i.e. the readers should be in the critical section
92+
## at the same time)
93+
94+
self.assertEqual([], threads[0].buffer_read)
95+
self.assertEqual([1], threads[2].buffer_read)
96+
self.assertEqual([1], threads[3].buffer_read)
97+
self.assert_(threads[1].exit_time <= threads[2].entry_time)
98+
self.assert_(threads[2].entry_time <= threads[3].entry_time)
99+
self.assert_(threads[3].exit_time < threads[2].exit_time)
100+
101+
def test_writers_exclusive_access(self):
102+
(buffer_, rw_lock, threads) = self.__init_variables()
103+
104+
threads.append(Writer(buffer_, rw_lock, 0, 0.4, 1))
105+
threads.append(Writer(buffer_, rw_lock, 0.1, 0, 2))
106+
threads.append(Reader(buffer_, rw_lock, 0.2, 0))
107+
108+
self.__start_and_join_threads(threads)
109+
110+
## The second writer should wait for the first one to exit
111+
112+
self.assertEqual([1, 2], threads[2].buffer_read)
113+
self.assert_(threads[0].exit_time <= threads[1].entry_time)
114+
self.assert_(threads[1].exit_time <= threads[2].exit_time)
115+
116+
def test_writer_priority(self):
117+
(buffer_, rw_lock, threads) = self.__init_variables()
118+
119+
threads.append(Writer(buffer_, rw_lock, 0, 0, 1))
120+
threads.append(Reader(buffer_, rw_lock, 0.1, 0.4))
121+
threads.append(Writer(buffer_, rw_lock, 0.2, 0, 2))
122+
threads.append(Reader(buffer_, rw_lock, 0.3, 0))
123+
threads.append(Reader(buffer_, rw_lock, 0.3, 0))
124+
125+
self.__start_and_join_threads(threads)
126+
127+
## The second writer should go before the second and the third reader
128+
129+
self.assertEqual([1], threads[1].buffer_read)
130+
self.assertEqual([1, 2], threads[3].buffer_read)
131+
self.assertEqual([1, 2], threads[4].buffer_read)
132+
self.assert_(threads[0].exit_time < threads[1].entry_time)
133+
self.assert_(threads[1].exit_time <= threads[2].entry_time)
134+
self.assert_(threads[2].exit_time <= threads[3].entry_time)
135+
self.assert_(threads[2].exit_time <= threads[4].entry_time)
136+
137+
def test_many_writers_priority(self):
138+
(buffer_, rw_lock, threads) = self.__init_variables()
139+
140+
threads.append(Writer(buffer_, rw_lock, 0, 0, 1))
141+
threads.append(Reader(buffer_, rw_lock, 0.1, 0.6))
142+
threads.append(Writer(buffer_, rw_lock, 0.2, 0.1, 2))
143+
threads.append(Reader(buffer_, rw_lock, 0.3, 0))
144+
threads.append(Reader(buffer_, rw_lock, 0.4, 0))
145+
threads.append(Writer(buffer_, rw_lock, 0.5, 0.1, 3))
146+
147+
self.__start_and_join_threads(threads)
148+
149+
## The two last writers should go first -- after the first reader and
150+
## before the second and the third reader
151+
152+
self.assertEqual([1], threads[1].buffer_read)
153+
self.assertEqual([1, 2, 3], threads[3].buffer_read)
154+
self.assertEqual([1, 2, 3], threads[4].buffer_read)
155+
self.assert_(threads[0].exit_time < threads[1].entry_time)
156+
self.assert_(threads[1].exit_time <= threads[2].entry_time)
157+
self.assert_(threads[1].exit_time <= threads[5].entry_time)
158+
self.assert_(threads[2].exit_time <= threads[3].entry_time)
159+
self.assert_(threads[2].exit_time <= threads[4].entry_time)
160+
self.assert_(threads[5].exit_time <= threads[3].entry_time)
161+
self.assert_(threads[5].exit_time <= threads[4].entry_time)
162+
163+
@staticmethod
164+
def __init_variables():
165+
buffer_ = []
166+
rw_lock = RWLock()
167+
threads = []
168+
return (buffer_, rw_lock, threads)
169+
170+
@staticmethod
171+
def __start_and_join_threads(threads):
172+
for t in threads:
173+
t.start()
174+
for t in threads:
175+
t.join()

0 commit comments

Comments
 (0)