Lecture 17: More Linked Lists
Overview
- Review: Coarse- and Fine-grained Lists
- Optimistic List
- Lazy List
Announcements
- Comments on proposals posted
- see shared Google drive
- please consider/respond to comments
-
forthcoming lab/hw assignments are optional
- can be submitted by end of finals week
- lowest grades dropped
- Quizzes weeks of 4/26, 5/3, 5/10, 5/17
- Final project steps
- Proof of concept due 5/7
- Short video due 5/19
- Final submission due 5/28
Last Time
A Set
of elements:
- store a collection of distinct elements
-
add
an element
- no effect if element already there
-
remove
an element
- check if set
contains
an element
An 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);
}
Coarse-grained Locking
One lock for whole data structure
For any operation:
- Lock entire list
- Perform operation
- Unlock list
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
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
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;
}
Question
Under what conditions might optimistic synchronization be fast/slow?
Testing Optimistic Synchronization
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
So Far
All operations have been blocking
- Each method call locks a portion of the data structre
- Method calls lock out other calls
- True even for
contains()
calls
- doesn’t modify the data structure
Observation
Operations are complicated because they consist of several steps
- hard to reason about when the operation appears to take place
- coarse/fine-grained synchronization stop other threads from seeing operations “in progress”
- optimistic synchronization may encounter “in progress” operations before locking
Lazy Synchronization
-
Mark a node before physical removal
- marked nodes are logically removed, still physically present
- Only marked nodes are ever removed
Validation simplified:
- Just check if nodes are marked
- No need to traverse whole list!
Lazy Operation
- Traverse without locking
- Lock relevant nodes
- Validate list
- check nodes are
- not marked
- correct relationship
- if validation fails, go back to Step 1
- Perform operation
- for removal, mark node first
- Unlock nodes
Lazy Removal Illustrated
data:image/s3,"s3://crabby-images/9583c/9583c986fcab3a18cb220ba1025452795ed83903" alt=""
Step 1: Traverse List
data:image/s3,"s3://crabby-images/22e0f/22e0fbf82f2705536df418bd70fd806776634a73" alt=""
Step 1: Traverse List
data:image/s3,"s3://crabby-images/a75a4/a75a451f246ca51c846b46b9eb7051aa5d59b35d" alt=""
Step 2: Lock Nodes
data:image/s3,"s3://crabby-images/c28fe/c28fe21ba7f3f359e74d01bda262796999c84ec2" alt=""
Step 3: Validate pred.next == curr
?
data:image/s3,"s3://crabby-images/79651/796519a1e4e6eb4c003fb364db3865b0c16432e5" alt=""
Step 3: Validate not marked?
data:image/s3,"s3://crabby-images/79651/796519a1e4e6eb4c003fb364db3865b0c16432e5" alt=""
Step 5: Release Locks and Done!
data:image/s3,"s3://crabby-images/609b5/609b52a8b70f16b0f63f0e1ef0368467b65d0a3d" alt=""
A Node
in Code
private class Node {
T item;
int key;
Node next;
Lock lock;
volatile boolean marked;
public Node (int key) {
this.item = null;
this.key = key;
this.next = null;
this.lock = new ReentrantLock();
this.marked = false;
}
public Node (T item) {
this.item = item;
this.key = item.hashCode();
this.next = null;
this.lock = new ReentrantLock();
}
public void lock () {
lock.lock();
}
public void unlock () {
lock.unlock();
}
}
Validation, Simplified
private boolean validate (Node pred, Node curr) {
return !pred.marked && !curr.marked && pred.next == curr;
}
Improvements?
- Limited locking as in optimistic synchronization
- Simpler validation
- faster—no list traversal
- more likely to succeed?
- Logical removal easier to reason about
- linearization point at logical removal line
-
contains()
no longer acquires locks
- often most frequent operation
- now it is wait-free!
Wait-free Containment
public boolean contains (T item) {
int key = item.hashCode();
Node curr = head;
while (curr.key < key) {
curr = curr.next;
}
return curr.key == key && !curr.marked;
}
Lazy Appraisal
Advantages:
- Less locking than fine-grained
- More opportunities for parallelism than coarse-grained
- Simpler validataion than optimistic
- Wait-free
contains
method
Disadvantages:
- Validation could still fail (though maybe less likely)
- Not starvation-free
- even if locks are starvation-free
-
add
and remove
still blocking
What’s next
Can we make all of the operations wait-free?
- A concurrent list without locks?
As Before
To get correctness without locks we need atomics!
AtomicMarkableReference<T>
- Stores
- a reference to a
T
- a boolean
marked
- Atomic operations
boolean compareAndSet(T expectedRef, T newRef, boolean expectedMark, boolean newMark)
T get(boolean[] marked)
T getReference()
boolean isMarked()
Nonblocking List Idea
Similar to LazyList
- use
AtomicMarkableReference
to mark and modify references simultaneously
- modifications done by
LazyList
can be done atomically!
-
add
and remove
are lock-free
-
contains
is wait-free (hence also lock-free)