Lecture 31: Optimistic Linked Lists
COSC 273: Parallel and Distributed Computing
Spring 2023
Annoucements
- Leaderboard submission results tomorrow
- Next leaderboard submission is Friday? Monday?
- Take-home quiz: released Wednesday (Gradescope), due Friday
Today
Concurrent Linked Lists, Three Ways:
- Coarse locking
- Fine-grained locking
- Optimistic locking
A Generic Task
Store, access, & modify collection of distinct elements:
The Set
ADT:
-
add
an element
- no effect if element already there
-
remove
an element
- check if set
contains
an element
Java SimpleSet
Interface
public interface SimpleSet<T> {
// Add an element to the SimpleSet. Returns true if the element
// was not already in the set.
boolean add(T x);
// Remove an element from the SimpleSet. Returns true if the
// element was previously in the set.
boolean remove(T x);
// Test if a given element is contained in the set.
boolean contains(T x);
}
Linked List SimpleSet
s
- Each
Node
stores:
- reference to the stored object
- reference to the next
Node
- a numerical key associated with the object
- The list stores
- reference to
head
node
- a
tail
node
-
head
and tail
have min and max key values
- nodes have strictly increasing keys
Why Keys?
Question. Why is it helpful to store keys in increasing order?
Our Goals
- Correctness, safety, liveness
- deadlock-freedom
- starvation-freedom?
- nonblocking??
- linearizability???
- Performance
Synchronization Philosophies
- Coarse-Grained (
CoarseList.java
)
- lock whole data structure for every operation
- Fine-Grained (
FineList.java
)
- only lock what is needed to avoid disaster
- Optimistic (
OptimisticList.java
)
- don’t lock anything to read, only lock to modify
- Lazy (
LazyList.java
)
- use “logical” removal, only lock occasionally
- Nonblocking (
NonblockingList.java
)
Coarse-grained Locking
One lock for whole data structure
For any operation:
- Lock entire list
- Perform operation
- Unlock list
See CoarseList.java
Coarse-grained Insertion
data:image/s3,"s3://crabby-images/c5b68/c5b680388c0f1e084c874c7670d883672224ea1e" alt=""
Step 1: Acquire Lock
data:image/s3,"s3://crabby-images/fb44b/fb44bb8418a85fcb8a815e41bc895c402b513108" alt=""
Step 2: Iterate to Find Location
data:image/s3,"s3://crabby-images/1e540/1e540e3a913385967e88864b33490a0f60626027" alt=""
Step 2: Iterate to Find Location
data:image/s3,"s3://crabby-images/71984/71984b2932fc412927b2d7447b09d659d9eedd88" alt=""
Step 2: Iterate to Find Location
data:image/s3,"s3://crabby-images/28966/28966fe751c7aaa28d92c5d6f15bb6cc032c7cac" alt=""
Step 3: Insert Item
data:image/s3,"s3://crabby-images/77c5d/77c5d554251b61c685e48d4ab582e797af3f728f" alt=""
Step 4: Unlock List
data:image/s3,"s3://crabby-images/c7ea5/c7ea55dfba3b5624c342f473db04e65c2bd581be" alt=""
Coarse-grained Appraisal
Advantages:
- Easy to reason about
- Easy to implement
Disadvantages:
- No parallelism
- All operations are blocking
Fine-grained Locking
One lock per node
For any operation:
- Lock head and its next
- Hand-over-hand locking while searching
- always hold at least one lock
- Perform operation
- Release locks
See FineList.java
A Fine-grained Insertion
data:image/s3,"s3://crabby-images/323ea/323ead0748bc8298f880624f1ee78365e17ed048" alt=""
Step 1: Lock Initial Nodes
data:image/s3,"s3://crabby-images/e74a3/e74a390ff8f12faa01e628156ce800c84d5405f8" alt=""
Step 2: Hand-over-hand Locking
data:image/s3,"s3://crabby-images/e74a3/e74a390ff8f12faa01e628156ce800c84d5405f8" alt=""
Step 2: Hand-over-hand Locking
data:image/s3,"s3://crabby-images/9325e/9325eff1f007c3cd2896e8144ea2c5dd0f802711" alt=""
Step 2: Hand-over-hand Locking
data:image/s3,"s3://crabby-images/5cdac/5cdac69d89263748027beb2b841edeba8a6810c6" alt=""
Step 4: Unlock Nodes
data:image/s3,"s3://crabby-images/c8ad9/c8ad9dbfee64baf31809cd5b25b3d8ddba4dda96" alt=""
An Advantage: Parallel Access
data:image/s3,"s3://crabby-images/08e29/08e29b401be991f3625fe05fbc55df1293711a33" alt=""
An Advantage: Parallel Access
data:image/s3,"s3://crabby-images/cb969/cb96986316b409362669871d6d87077f9feacaae" alt=""
An Advantage: Parallel Access
data:image/s3,"s3://crabby-images/1b79d/1b79d8183cb27dae4b7e52a5ec16c0052335fc76" alt=""
An Advantage: Parallel Access
data:image/s3,"s3://crabby-images/39206/39206c8400124d73c2850a03d079cf1904c2b14d" alt=""
An Advantage: Parallel Access
data:image/s3,"s3://crabby-images/ea5ed/ea5ed7b632c9a63475ca4f086a224a093c3ffddf" alt=""
An Advantage: Parallel Access
data:image/s3,"s3://crabby-images/e4b8e/e4b8e7ba434fdab7167a28060c7266b5bbd54c3c" alt=""
An Advantage: Parallel Access
data:image/s3,"s3://crabby-images/4432c/4432cb26064a56fd34711e647f4d80f710225d11" alt=""
An Advantage: Parallel Access
data:image/s3,"s3://crabby-images/8b4a6/8b4a6a7c100a3ac8f827b6d04148938db35d03e8" alt=""
An Advantage: Parallel Access
data:image/s3,"s3://crabby-images/77d58/77d5853bcbc9414ca35e76b74886f69d07f75f8b" alt=""
An Advantage: Parallel Access
data:image/s3,"s3://crabby-images/65c0a/65c0a83839fd450374a3bff0044db0048953a462" alt=""
An Advantage: Parallel Access
data:image/s3,"s3://crabby-images/16f01/16f015fb512f62dad66d379c0d06d131430652c2" alt=""
An Advantage: Parallel Access
data:image/s3,"s3://crabby-images/2447b/2447bfe73db416f38737e4cd3fde29bc479e3775" alt=""
Fine-grained Appraisal
Advantages:
- Parallel access
- Reasonably simple implementation
Disadvantages:
- More locking overhead
- can be much slower than coarse-grained
- All operations are blocking
Optimistic Synchronization
Fine-grained wastes resources locking
- Nodes are locked when traversed
- Locked even if not modified!
A better procedure?
- Traverse without locking
- Lock relevant nodes
- Perform operation
- Unlock nodes
A Better Way?
data:image/s3,"s3://crabby-images/afdd5/afdd524864d2ba2f05ff9a07fc4665ba44dea84d" alt=""
A Better Way?
data:image/s3,"s3://crabby-images/9b523/9b52346439ff1fd824d172af3c08ca07ab0e5cb3" alt=""
A Better Way?
data:image/s3,"s3://crabby-images/519d6/519d67f5772cd8e0fe477551a4acf2174dfe8d23" alt=""
A Better Way?
data:image/s3,"s3://crabby-images/ea26f/ea26fda9d82d7ef998f37f192ea368639ca95170" alt=""
A Better Way?
data:image/s3,"s3://crabby-images/986c6/986c6ece19f61bb7c4ac2d7721fa4e815be7b6d1" alt=""
A Better Way?
data:image/s3,"s3://crabby-images/b1d3e/b1d3ef817f164d185899931f047b356a74bc8a8d" alt=""
An Issue!
Between traversing and locking
- Another thread modifies the list
- Now locked nodes aren’t the right nodes!
An Issue, Illustrated
data:image/s3,"s3://crabby-images/ea26f/ea26fda9d82d7ef998f37f192ea368639ca95170" alt=""
An Issue, Illustrated
data:image/s3,"s3://crabby-images/75cf0/75cf092a4ee5e3a7a1a38eded4a21030fa1e2380" alt=""
An Issue, Illustrated
data:image/s3,"s3://crabby-images/58894/588945ceba8b4639612e0a7f05506b77d8e1e3de" alt=""
An Issue, Illustrated
data:image/s3,"s3://crabby-images/61176/61176add845115cc6fbeec27a72a8611f95bbc41" alt=""
An Issue, Illustrated
data:image/s3,"s3://crabby-images/f107b/f107baa34cc9a9ce0bdb0ff514fd875972a3ee73" alt=""
An Issue, Illustrated
data:image/s3,"s3://crabby-images/c1931/c19315cefc73a38c2e1cbcd3d1aff186d7ad54aa" alt=""
How can we Address this Issue?
Optimistic Synchronization, Validated
- Traverse without locking
- Lock relevant nodes
-
Validate list
- if validation fails, go back to Step 1
- Perform operation
- Unlock nodes
Seet OptimisticList.java
How do we Validate?
After locking, ensure that:
-
pred
is reachable from head
-
curr
is pred
’s successor
If these conditions aren’t met:
Optimistic Insertion
data:image/s3,"s3://crabby-images/afdd5/afdd524864d2ba2f05ff9a07fc4665ba44dea84d" alt=""
Step 1: Traverse the List
data:image/s3,"s3://crabby-images/9b523/9b52346439ff1fd824d172af3c08ca07ab0e5cb3" alt=""
Step 1: Traverse the List
data:image/s3,"s3://crabby-images/519d6/519d67f5772cd8e0fe477551a4acf2174dfe8d23" alt=""
Step 1: Traverse the List
data:image/s3,"s3://crabby-images/ea26f/ea26fda9d82d7ef998f37f192ea368639ca95170" alt=""
Step 2: Acquire Locks
data:image/s3,"s3://crabby-images/986c6/986c6ece19f61bb7c4ac2d7721fa4e815be7b6d1" alt=""
Step 3: Validate List - Traverse
data:image/s3,"s3://crabby-images/a1084/a1084d5509ffa7ba43a521e58e2d5159874a75f2" alt=""
Step 3: Validate List - pred
Reachable?
data:image/s3,"s3://crabby-images/f7970/f7970d3c3a5950261daea4fd658569b623ec6f01" alt=""
Step 3: Validate List - Is curr
next?
data:image/s3,"s3://crabby-images/2a6f1/2a6f1ed01166440e55633e29955b13b4934f1de7" alt=""
Step 5: Release Locks
data:image/s3,"s3://crabby-images/312c3/312c379944c3f2d4126c825b47780e193cf83b88" alt=""
Implementing Validation
private boolean validate (Node pred, Node curr) {
Node node = head;
while (node.key <= pred.key) {
if (node == pred) {
return pred.next == curr;
}
node = node.next;
}
return false;
}
Optimistic Appraisal
Advantages:
- Less locking than fine-grained
- More opportunities for parallelism than coarse-grained
Disadvantages:
- Validation could fail
- Not starvation-free
- even if locks are starvation-free
Time v. Threads, 8 Elements
data:image/s3,"s3://crabby-images/0846f/0846fb1957080d0130f75278e360e8aafbf2d92d" alt=""
Time v. Threads, 8,192 Elements
data:image/s3,"s3://crabby-images/9db30/9db3098bf061e3a63daef5c1444f8072e2f1a6da" alt=""
Coarse Time v. Threads
data:image/s3,"s3://crabby-images/a1d2a/a1d2af06a04f5176ab4819c7661c14c29da958c1" alt=""
Fine Time v. Threads
data:image/s3,"s3://crabby-images/182d2/182d2860915a7a389ec581cf095acdc5faa003e5" alt=""
Optimistic Time v. Threads
data:image/s3,"s3://crabby-images/38d03/38d03389cc7b51f554a906562d7bb9692017caea" alt=""
Next Time
Another Way: Lazy Synchronization
- don’t modify the list unless you really have to