public class WriterReaderPhaser extends Object
WriterReaderPhaser
provides an asymmetric means for
synchronizing the execution of wait-free "writer" critical sections against
a "reader phase flip" that needs to make sure no writer critical sections
that were active at the beginning of the flip are still active after the
flip is done. Multiple writers and multiple readers are supported.
Using a WriterReaderPhaser
for coordination, writers can continously
perform wait-free/lock-free updates to common data structures, while readers
can get hold of atomic and inactive snapshots without stalling writers.
While a WriterReaderPhaser
can be useful in multiple scenarios, a
specific and common use case is that of safely managing "double buffered"
data stream access in which writers can proceed without being blocked, while
readers gain access to stable and unchanging buffer samples.
WriterReaderPhaser
"writers" are wait free (on architectures that support
wait free atomic increment operations), "readers" block for other
"readers", and "readers" are only blocked by "writers" whose critical section
was entered before the reader's
flipPhase()
attempt.
When used to protect an actively recording data structure, the assumptions on how readers and writers act are:
writerCriticalSectionEnter()
and
writerCriticalSectionExit()
.
readerLock()
protection and only before calling flipPhase()
.writerCriticalSectionEnter()
and
writerCriticalSectionExit()
.
readerLock()
and
readerUnlock()
.flipPhase()
operations,
and only while holding the readerLock.
When the above assumptions are met, WriterReaderPhaser
guarantees
that the inactive data structures are not being modified by any writers while
being read while under readerLock() protection after a
flipPhase()
() operation.
The following progress guarantees are provided to writers and readers that adhere to the above stated assumptions:
writerCriticalSectionEnter
and writerCriticalSectionExit
)
are wait free on architectures that
support wait-free atomic increment operations (they remain lock-free [but not
wait-free] on architectures that do not support wait-free atomic increment
operations)flipPhase()
operations are guaranteed to
make forward progress, and will only be blocked by writers whose critical sections
were entered prior to the start of the reader's flipPhase operation, and have not
yet exited their critical sections.readerLock()
only blocks for other
readers that are holding the readerLock.
final WriterReaderPhaser recordingPhaser = new WriterReaderPhaser();
volatile Histogram activeHistogram;
Histogram inactiveHistogram;
...
A writer may record values the histogram:
// Wait-free recording:
long criticalValueAtEnter = recordingPhaser.writerCriticalSectionEnter();
try {
activeHistogram.recordValue(value);
} finally {
recordingPhaser.writerCriticalSectionExit(criticalValueAtEnter);
}
A reader gains access to a stable histogram of values recorded during an interval,
and reports on it:
try {
recordingPhaser.readerLock();
inactiveHistogram.reset();
// Swap active and inactive histograms:
final Histogram tempHistogram = inactiveHistogram;
inactiveHistogram = activeHistogram;
activeHistogram = tempHistogram;
recordingPhaser.flipPhase();
// At this point, inactiveHistogram content is guaranteed to be stable
logHistogram(inactiveHistogram);
} finally {
recordingPhaser.readerUnlock();
}
Constructor and Description |
---|
WriterReaderPhaser() |
Modifier and Type | Method and Description |
---|---|
void |
flipPhase()
Flip a phase in the
WriterReaderPhaser instance, flipPhase()
can only be called while holding the readerLock . |
void |
flipPhase(long yieldTimeNsec)
Flip a phase in the
WriterReaderPhaser instance, flipPhase()
can only be called while holding the readerLock . |
void |
readerLock()
Enter to a critical section containing a read operation (reentrant, mutually excludes against
readerLock calls by other threads). |
void |
readerUnlock()
Exit from a critical section containing a read operation (relinquishes mutual exclusion against other
readerLock calls). |
long |
writerCriticalSectionEnter()
Indicate entry to a critical section containing a write operation.
|
void |
writerCriticalSectionExit(long criticalValueAtEnter)
Indicate exit from a critical section containing a write operation.
|
public long writerCriticalSectionEnter()
This call is wait-free on architectures that support wait free atomic increment operations, and is lock-free on architectures that do not.
writerCriticalSectionEnter()
must be matched with a subsequent
writerCriticalSectionExit(long)
in order for CriticalSectionPhaser
synchronization to function properly.
writerCriticalSectionExit(long)
call.public void writerCriticalSectionExit(long criticalValueAtEnter)
This call is wait-free on architectures that support wait free atomic increment operations, and is lock-free on architectures that do not.
writerCriticalSectionExit(long)
must be matched with a preceding
writerCriticalSectionEnter()
call, and must be provided with the
matching writerCriticalSectionEnter()
call's return value, in
order for CriticalSectionPhaser synchronization to function properly.
criticalValueAtEnter
- the (opaque) value returned from the matching
writerCriticalSectionEnter()
call.public void readerLock()
readerLock
calls by other threads).
readerLock
DOES NOT provide synchronization
against writerCriticalSectionEnter()
calls. Use flipPhase()
to synchronize reads against writers.
public void readerUnlock()
readerLock
calls).public void flipPhase(long yieldTimeNsec)
WriterReaderPhaser
instance, flipPhase()
can only be called while holding the readerLock
.
flipPhase()
will return only after all writer critical sections (protected by
writerCriticalSectionEnter
and
writerCriticalSectionEnter
) that may have been
in flight when the flipPhase()
call were made had completed.
No actual writer critical section activity is required for flipPhase()
to
succeed.
However, flipPhase()
is lock-free with respect to calls to
writerCriticalSectionEnter()
and
writerCriticalSectionExit()
. It may spin-wait
or for active writer critical section code to complete.
yieldTimeNsec
- The amount of time (in nanoseconds) to sleep in each yield if yield loop is needed.public void flipPhase()
WriterReaderPhaser
instance, flipPhase()
can only be called while holding the readerLock
.
flipPhase()
will return only after all writer critical sections (protected by
writerCriticalSectionEnter
and
writerCriticalSectionEnter
) that may have been
in flight when the flipPhase()
call were made had completed.
No actual writer critical section activity is required for flipPhase()
to
succeed.
However, flipPhase()
is lock-free with respect to calls to
writerCriticalSectionEnter()
and
writerCriticalSectionExit()
. It may spin-wait
or for active writer critical section code to complete.
Copyright © 2019. All rights reserved.