Using Queues: Sleeping Instead of Spinning
This lesson discusses the use of queues to built more efficient locks.
We'll cover the following
The real problem with the previous approaches is that they leave too much to chance. The scheduler determines which thread runs next; if the scheduler makes a bad choice, a thread runs that must either spin waiting for the lock (our first approach) or yield the CPU immediately (our second approach). Either way, there is potential for waste and no prevention of starvation.
Thus, we must explicitly exert some control over which thread next gets to acquire the lock after the current holder releases it. To do this, we will need a little more OS support, as well as a queue to keep track of which threads are waiting to acquire the lock.
park()
and unpark()
For simplicity, we will use the support provided by Solaris, in terms of two calls: park()
to put a calling thread to sleep and unpark(threadID)
to wake a particular thread as designated by threadID
. These two routines can be used in tandem to build a lock that puts a caller to sleep if it tries to acquire a held lock and wakes it when the lock is free. Let’s look at the code excerpt below to understand one possible use of such primitives.
typedef struct __lock_t {int flag;int guard;queue_t *q;} lock_t;void lock_init(lock_t *m) {m->flag = 0;m->guard = 0;queue_init(m->q);}void lock(lock_t *m) {while (TestAndSet(&m->guard, 1) == 1); //acquire guard lock by spinningif (m->flag == 0) {m->flag = 1; //lock is acquiredm->guard = 0;} else {queue_add(m->q, gettid());m->guard = 0;park();}}void unlock(lock_t *m) {while (TestAndSet(&m->guard, 1) == 1); //acquire guard lock by spinningif (queue_empty(m->q))m->flag = 0; // let go of lock; no one wants itelseunpark(queue_remove(m->q)); // hold lock// (for next thread!)m->guard = 0;}
We do a couple of interesting things in this example. First, we combine the old test-and-set idea with an explicit queue of lock waiters to make a more efficient lock. Second, we use a queue to help control who gets the lock next and thus avoid starvation.
You might notice how the guard is used in this example, basically, as a spin-lock around the flag and queue manipulations, the lock is using. This approach thus doesn’t avoid spin-waiting entirely; a thread might be interrupted while acquiring or releasing the lock, and thus cause other threads to spin-wait for this one to run again. However, the time spent spinning is quite limited (just a few instructions inside the lock and unlock code, instead of the user-defined critical section), and thus this approach may be reasonable.
You might also observe that in lock()
, when a thread can not acquire the lock (it is already held), we are careful to add ourselves to a queue (by calling the gettid()
function to get the thread ID of the current thread), set guard to 0, and yield the CPU. A question for the reader: What would happen if the release of the guard lock came after the park()
, and not before? Hint: something bad.
You might further detect that the flag does not get set back to 0 when another thread gets woken up. Why is this? Well, it is not an error, but rather a necessity! When a thread is woken up, it will be as if it is returning from park()
; however, it does not hold the guard at that point in the code and thus cannot even try to set the flag to 1. Thus, we just pass the lock directly from the thread releasing the lock to the next thread acquiring it; the flag is not set to 0 in-between.
The waiting race problem
Finally, you might notice the perceived race condition in the solution, just before the call to park()
. With just the wrong timing, a thread will be about to park, assuming that it should sleep until the lock is no longer held. A switch at that time to another thread (say, a thread holding the lock) could lead to trouble, for example, if that thread then released the lock. The subsequent park by the first thread would then sleep forever (potentially), a problem sometimes called the wakeup/waiting race.
Solaris solves this problem by adding a third system call: setpark()
. By calling this routine, a thread can indicate it is about to park. If it then happens to be interrupted and another thread calls unpark before park is actually called, the subsequent park returns immediately instead of sleeping. The code modification, inside of lock()
, is quite small:
queue_add(m->q, gettid());
setpark(); // new code
m->guard = 0;
A different solution could pass the guard into the kernel. In that case, the kernel could take precautions to atomically release the lock and de- queue the running thread.
ASIDE: MORE REASON TO AVOID SPINNING: PRIORITY INVERSION
One good reason to avoid spin locks is performance: as described in the main text, if a thread is interrupted while holding a lock, other threads that use spin locks will spend a large amount of CPU time just waiting for the lock to become available. However, it turns out there is another interesting reason to avoid spin locks on some systems: correctness. The problem to be wary of is known as priority inversion, which unfortunately is an intergalactic scourge, occurring on
and Earth “OSSpinLock Is Unsafe” by J. McCall. mjtsai.com/blog/2015/12/16/osspinlock -is-unsafe. Calling OSSpinLock on a Mac is unsafe when using threads of different priorities – you might spin forever! So be careful, Mac fanatics, even your mighty system can be less than perfect… ! Mars “What Really Happened on Mars?” by Glenn E. Reeves. research.microsoft.com/ en-us/um/people/mbj/Mars Pathfinder/Authoritative Account.html. A descrip- tion of priority inversion on Mars Pathfinder. Concurrent code correctness matters, especially in space! Let’s assume there are two threads in a system. Thread 2 (
T2
) has a high scheduling priority, and Thread 1 (T1
) has a lower priority. In this example, let’s assume that the CPU scheduler will always runT2
overT1
, if indeed both are runnable;T1
only runs whenT2
is not able to do so (e.g., whenT2
is blocked on I/O).Now, the problem. Assume
T2
is blocked for some reason. SoT1
runs, grabs a spin lock, and enters a critical section.T2
now becomes unblocked (perhaps because an I/O completed), and the CPU scheduler immediately schedules it (thus deschedulingT1
).T2
now tries to acquire the lock, and because it can’t (T1
holds the lock), it just keeps spinning. Because the lock is a spin lock,T2
spins forever, and the system is hung.Just avoiding the use of spin locks, unfortunately, does not avoid the problem of inversion (alas). Imagine three threads,
T1
,T2
, andT3
, withT3
at the highest priority, andT1
the lowest. Imagine now thatT1
grabs a lock.T3
then starts, and because it is higher priority thanT1
, runs immediately (preemptingT1
).T3
tries to acquire the lock thatT1
holds, but gets stuck waiting, becauseT1
still holds it. IfT2
starts to run, it will have higher priority thanT1
, and thus it will run.T3
, which is a higher priority thanT2
, is stuck waiting forT1
, which may never run now thatT2
is running. Isn’t it sad that the mightyT3
can’t run, while lowlyT2
controls the CPU? Having high priority just ain’t what it used to be.You can address the priority inversion problem in a number of ways. In the specific case where spin locks cause the problem, you can avoid using spin locks (described more below). More generally, a higher-priority thread waiting for a lower-priority thread can temporarily boost the lower thread’s priority, thus enabling it to run and overcoming the inversion, a technique known as priority inheritance. A last solution is simplest: ensure all threads have the same priority.
Get hands-on with 1400+ tech skills courses.