com.github.richardballard.arbeeutils.concurrent.WrappedStampedLock Maven / Gradle / Ivy
/*
* (C) Copyright 2016 Richard Ballard.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.github.richardballard.arbeeutils.concurrent;
import net.jcip.annotations.ThreadSafe;
import com.github.richardballard.arbeeutils.time.TimeTick;
import org.jetbrains.annotations.NotNull;
import java.time.Duration;
import java.util.Objects;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.StampedLock;
import java.util.function.BooleanSupplier;
import java.util.function.Function;
import java.util.function.LongSupplier;
import java.util.function.Supplier;
/**
* This class is a wrapper around an instance of {@link StampedLock} that allows the caller to use the delegate
* in a more functional manner.
*
* As with {@link StampedLock} this lock is non-reentrant, i.e. if a thread holds a lock and then attempts to acquire
* another lock then it will deadlock.
*
* For read locks it has the concept of {@code pessimistic} and {@code optimistic} read locks.
*
* Pessimistic Read Locks
*
* Pessimistic locks are only allowed when there are no outstanding write locks. Multiple pessimistic read locks are
* allowed simultaneously.
*
* Optimistic Read Locks
*
* Optimistic locks may be 'started' while there are any other kind of locks (including write locks) outstanding.
* Once the operation is finished the code tests to see if a write lock was in progress during the operation and if
* so then the optimistic lock fails. The code attempts a number of optimistic locks before failing over to an pessimistic
* lock. Because of this very loose kind of locking (and the fact that the operation may be called any number of times),
* the operation itself (that is run during the optimistic lock attempt) must not update any state - it must read only.
* See {@link #optimisticRead(Supplier)} for more information.
*/
@ThreadSafe
public class WrappedStampedLock {
// keep this number small as the typical case for doing a optimistic lock is a short operation. If the first
// lock attempt fails and the operation is short then chances are that the subsequent attempts will also fail. RMB 2016/06/05
private static final int NUM_OPTIMISTIC_ATTEMPTS = 2;
@NotNull
private final StampedLock delegate;
@NotNull
private final Function wrappedReadWriteLockFromReadWriteLockFunction;
@NotNull
private final Function wrappedLockFromLockFunction;
@NotNull
private final Supplier currentTimeTickSupplier;
protected WrappedStampedLock(@NotNull final StampedLock delegate,
@NotNull final Function wrappedReadWriteLockFromReadWriteLockFunction,
@NotNull final Function wrappedLockFromLockFunction,
@NotNull final Supplier currentTimeTickSupplier) {
assert delegate != null;
assert wrappedReadWriteLockFromReadWriteLockFunction != null;
assert wrappedLockFromLockFunction != null;
assert currentTimeTickSupplier != null;
this.delegate = delegate;
this.wrappedReadWriteLockFromReadWriteLockFunction = wrappedReadWriteLockFromReadWriteLockFunction;
this.wrappedLockFromLockFunction = wrappedLockFromLockFunction;
this.currentTimeTickSupplier = currentTimeTickSupplier;
}
@NotNull
public WrappedReadWriteLock asReadWriteLock() {
return Objects.requireNonNull(wrappedReadWriteLockFromReadWriteLockFunction.apply(delegate.asReadWriteLock()));
}
@NotNull
public WrappedLock asReadLock() {
return Objects.requireNonNull(wrappedLockFromLockFunction.apply(delegate.asReadLock()));
}
@NotNull
public WrappedLock asWriteLock() {
return Objects.requireNonNull(wrappedLockFromLockFunction.apply(delegate.asWriteLock()));
}
private T pessimisticRead(@NotNull final LongSupplier readLockStampSupplier,
@NotNull final Supplier operation) {
assert readLockStampSupplier != null;
assert operation != null;
final long stamp = readLockStampSupplier.getAsLong();
assert stamp != 0L;
try {
return operation.get();
}
finally {
delegate.unlockRead(stamp);
}
}
/**
* Runs the given {@code operation} in a pessimistic read lock. While the operation is running the only other
* locks that *may* be running are pessimistic and optimistic read locks.
*/
public T pessimisticRead(@NotNull final Supplier operation) {
assert operation != null;
return pessimisticRead(delegate::readLock,
operation);
}
/**
* Runs the given {@code operation} in a pessimistic read lock. While the operation is running the only other
* locks that *may* be running are pessimistic and optimistic read locks.
*/
public T pessimisticRead(@NotNull final Supplier operation,
@NotNull final Duration acquireTimeout) throws AcquireTimeoutException {
assert operation != null;
assert acquireTimeout != null;
return pessimisticRead(new ReadLockStampWithTimeoutSupplier(delegate,
() -> acquireTimeout),
operation);
}
private T optimisticRead(@NotNull final Supplier pessimisticReadAppliedToOperation,
@NotNull final Supplier operation) {
assert pessimisticReadAppliedToOperation != null;
assert operation != null;
T result = null; // setting this to null isn't needed *except* to keep the compiler happy - RMB 2016/06/05
boolean validReadPerformed = false;
for(int i = 0; i < NUM_OPTIMISTIC_ATTEMPTS && !validReadPerformed; i++) {
final long stamp = delegate.tryOptimisticRead();
if(stamp != 0L) { // this will be 0 if exclusively locked
result = operation.get();
if(delegate.validate(stamp)) {
validReadPerformed = true; // this will break the loop
}
}
}
if(!validReadPerformed) {
result = pessimisticReadAppliedToOperation.get();
}
return result;
}
/**
* See the class doc ({@link WrappedStampedLock}) for an overview of optimistic locks.
*
* The operation may be run at the same time as any other operation (including a write lock).
* Because of this it should only read data, and not make decisions based on the data it reads. Consider the example,
* if it were to read a long value and if the value is < 10 throw a BadValueException.
* The thread that is running the optimistic lock operation reads the first 32 bits of the long. Another thread
* writes a new value to the long (in a write lock). The read thread reads the remaining 32 bits and so reads
* a invalid number and throws BadValueException. If it were to not make the decision to throw if the value < 10
* then after reading the code would see that a write lock had occurred whilst reading and so re-attempt the optimistic
* read (thus returning the correct number).
*
* Another example - if the code in the optimistic read iterates through a list then another thread may update
* the list and so the reader gets a {@link java.util.ConcurrentModificationException}. In this case a
* pessimistic read should be used.
*/
public T optimisticRead(@NotNull final Supplier operation) {
assert operation != null;
return optimisticRead(() -> pessimisticRead(operation),
operation);
}
/**
* See the class doc ({@link WrappedStampedLock}) for an overview of optimistic locks.
*
* The operation may be run at the same time as any other operation (including a write lock).
* Because of this it should only read data, and not make decisions based on the data it reads. Consider the example,
* if it were to read a long value and if the value is < 10 throw a BadValueException.
* The thread that is running the optimistic lock operation reads the first 32 bits of the long. Another thread
* writes a new value to the long (in a write lock). The read thread reads the remaining 32 bits and so reads
* a invalid number and throws BadValueException. If it were to not make the decision to throw if the value < 10
* then after reading the code would see that a write lock had occurred whilst reading and so re-attempt the optimistic
* read (thus returning the correct number).
*
* Another example - if the code in the optimistic read iterates through a list then another thread may update
* the list and so the reader gets a {@link java.util.ConcurrentModificationException}. In this case a
* pessimistic read should be used.
*/
public T optimisticRead(@NotNull final Supplier operation,
@NotNull final Duration acquireTimeout) throws AcquireTimeoutException {
assert operation != null;
assert acquireTimeout != null;
final TimeTick startTimeTick = currentTimeTickSupplier.get();
return optimisticRead(() -> {
// figure out how much time has passed since we started and use what is left within the original duration
// as the timeout for the pessimistic lock call.
final Duration durationSinceStart = currentTimeTickSupplier.get().durationSince(startTimeTick);
return pessimisticRead(operation,
acquireTimeout.minus(durationSinceStart));
},
operation);
}
private T write(@NotNull final LongSupplier writeLockStampSupplier,
@NotNull final Supplier operation) {
assert writeLockStampSupplier != null;
assert operation != null;
final long stamp = writeLockStampSupplier.getAsLong();
assert stamp != 0L;
try {
return operation.get();
}
finally {
delegate.unlockWrite(stamp);
}
}
/**
* Performs an operation within a write lock. When this operation is running the only other locks that *may* be
* running are optimistic reads.
*/
public T write(@NotNull final Supplier operation) {
assert operation != null;
return write(delegate::writeLock,
operation);
}
/**
* Performs an operation within a write lock. When this operation is running the only other locks that *may* be
* running are optimistic reads.
*/
public T write(@NotNull final Supplier operation,
@NotNull final Duration acquireTimeout) throws AcquireTimeoutException {
assert operation != null;
assert acquireTimeout != null;
return write(new WriteLockStampWithTimeoutSupplier(delegate,
() -> acquireTimeout),
operation);
}
/**
* See {@link #write(Supplier)}
*/
public void write(@NotNull final Runnable operation) {
assert operation != null;
write(() -> {
operation.run();
//noinspection ReturnOfNull
return null;
});
}
/**
* See {@link #write(Supplier, Duration)}
*/
public void write(@NotNull final Runnable operation,
@NotNull final Duration acquireTimeout) throws AcquireTimeoutException {
assert operation != null;
assert acquireTimeout != null;
write(() -> {
operation.run();
return null;
},
acquireTimeout);
}
public enum TestFailedLockContext {
/**
* The operation will be run from within a pessimistic read lock
*/
IN_PESSIMISTIC_READ_LOCK,
/**
* The operation will not be run in any lock
*/
NO_LOCK
}
private T writeIf(@NotNull final BooleanSupplier test,
@NotNull final Supplier onTestPassedOperation,
@NotNull final Supplier onTestFailedOperation,
@NotNull final TestFailedLockContext testFailedLockContext,
@NotNull final LongSupplier readLockSupplier,
@NotNull final LongSupplier writeLockSupplier) {
assert test != null;
assert onTestPassedOperation != null;
assert onTestFailedOperation != null;
assert testFailedLockContext != null;
assert readLockSupplier != null;
assert writeLockSupplier != null;
long stamp = readLockSupplier.getAsLong();
assert stamp != 0L;
try {
while(test.getAsBoolean()) {
final long writeStamp = delegate.tryConvertToWriteLock(stamp);
if(writeStamp == 0L) { // conversion failed
delegate.unlockRead(stamp);
stamp = writeLockSupplier.getAsLong();
assert stamp != 0L;
}
else { // conversion was successful
stamp = writeStamp;
return onTestPassedOperation.get();
}
}
if(testFailedLockContext == TestFailedLockContext.IN_PESSIMISTIC_READ_LOCK) {
return onTestFailedOperation.get();
}
}
finally {
delegate.unlock(stamp); // could be read or write lock
}
assert testFailedLockContext == TestFailedLockContext.NO_LOCK : testFailedLockContext;
return onTestFailedOperation.get();
}
/**
*
* @param test this will be run in a pessimistic read lock. If it passes (returns true) then a write lock will be
* obtained and {@code onTestPassedOperation} will be run and the result returned. If it fails (returns
* false) then {@code onTestFailedOperation} will be run and the result returned.
* @param onTestPassedOperation This will be run (and result returned) if {@code test} returns true.
* @param onTestFailedOperation This will be run (and result returned) if {@code test} returns false. The lock
* context that will be used when this is called is determined by the value of
* {@code testFailedLockContext}.
* @param testFailedLockContext If the test fails then {@code onTestFailedOperation} will be called. This parameter
* determines what sort of lock to use (if any) when calling {@code testFailedLockContext}.
*/
public T writeIf(@NotNull final BooleanSupplier test,
@NotNull final Supplier onTestPassedOperation,
@NotNull final Supplier onTestFailedOperation,
@NotNull final TestFailedLockContext testFailedLockContext) {
assert test != null;
assert onTestPassedOperation != null;
assert onTestFailedOperation != null;
assert testFailedLockContext != null;
return writeIf(test,
onTestPassedOperation,
onTestFailedOperation,
testFailedLockContext,
delegate::readLock,
delegate::writeLock);
}
/**
*
* @param test this will be run in a pessimistic read lock. If it passes (returns true) then a write lock will be
* obtained and {@code onTestPassedOperation} will be run and the result returned. If it fails (returns
* false) then {@code onTestFailedOperation} will be run and the result returned.
* @param onTestPassedOperation This will be run (and result returned) if {@code test} returns true.
* @param onTestFailedOperation This will be run (and result returned) if {@code test} returns false. The lock
* context that will be used when this is called is determined by the value of
* {@code testFailedLockContext}.
* @param testFailedLockContext If the test fails then {@code onTestFailedOperation} will be called. This parameter
* determines what sort of lock to use (if any) when calling {@code testFailedLockContext}.
*/
public T writeIf(@NotNull final BooleanSupplier test,
@NotNull final Supplier onTestPassedOperation,
@NotNull final Supplier onTestFailedOperation,
@NotNull final TestFailedLockContext testFailedLockContext,
@NotNull final Duration acquireTimeout) throws AcquireTimeoutException {
assert test != null;
assert onTestPassedOperation != null;
assert onTestFailedOperation != null;
assert testFailedLockContext != null;
assert acquireTimeout != null;
final TimeTick startTimeTick = currentTimeTickSupplier.get();
final Supplier remainingTimeoutSupplier = () -> {
// figure out how much time has passed since we started and use what is left within the original duration
// as the timeout for the pessimistic lock call.
final Duration durationSinceStart = currentTimeTickSupplier.get().durationSince(startTimeTick);
return acquireTimeout.minus(durationSinceStart);
};
return writeIf(test,
onTestPassedOperation,
onTestFailedOperation,
testFailedLockContext,
new ReadLockStampWithTimeoutSupplier(delegate,
remainingTimeoutSupplier),
new WriteLockStampWithTimeoutSupplier(delegate,
remainingTimeoutSupplier));
}
public enum TestResult {
PASSED,
FAILED
}
/**
*
* @param test this will be run in a pessimistic read lock. If it passes (returns true) then a write lock will be
* obtained and {@code onTestPassedOperation} will be run and {@link TestResult#PASSED} returned.
* If it fails (returns false) then {@code onTestFailedOperation} will be run and {@link TestResult#FAILED}
* returned.
* @param onTestPassedOperation This will be run if {@code test} returns true.
* @param onTestFailedOperation This will be run if {@code test} returns false.
*/
@NotNull
public TestResult writeIf(@NotNull final BooleanSupplier test,
@NotNull final Runnable onTestPassedOperation,
@NotNull final Runnable onTestFailedOperation) {
assert test != null;
assert onTestPassedOperation != null;
assert onTestFailedOperation != null;
return writeIf(test,
() -> {
onTestPassedOperation.run();
return TestResult.PASSED;
},
() -> {
onTestFailedOperation.run();
return TestResult.FAILED;
},
TestFailedLockContext.NO_LOCK);
}
/**
*
* @param test this will be run in a pessimistic read lock. If it passes (returns true) then a write lock will be
* obtained and {@code onTestPassedOperation} will be run and {@link TestResult#PASSED} returned.
* If it fails (returns false) then {@code onTestFailedOperation} will be run and {@link TestResult#FAILED}
* returned.
* @param onTestPassedOperation This will be run if {@code test} returns true.
* @param onTestFailedOperation This will be run if {@code test} returns false.
*/
@NotNull
public TestResult writeIf(@NotNull final BooleanSupplier test,
@NotNull final Runnable onTestPassedOperation,
@NotNull final Runnable onTestFailedOperation,
@NotNull final Duration acquireTimeout) throws AcquireTimeoutException {
assert test != null;
assert onTestPassedOperation != null;
assert onTestFailedOperation != null;
assert acquireTimeout != null;
return writeIf(test,
() -> {
onTestPassedOperation.run();
return TestResult.PASSED;
},
() -> {
onTestFailedOperation.run();
return TestResult.FAILED;
},
TestFailedLockContext.NO_LOCK,
acquireTimeout);
}
@ThreadSafe
private static class ReadLockStampWithTimeoutSupplier implements LongSupplier {
@NotNull
private final StampedLock delegate;
@NotNull
private final Supplier timeoutSupplier;
public ReadLockStampWithTimeoutSupplier(@NotNull final StampedLock delegate,
@NotNull final Supplier timeoutSupplier) {
assert delegate != null;
assert timeoutSupplier != null;
this.delegate = delegate;
this.timeoutSupplier = timeoutSupplier;
}
/**
*
* @return never 0 (which would indicate a timeout)
* @throws AcquireTimeoutException if there is a timeout
*/
@Override
public long getAsLong() throws AcquireTimeoutException {
final long stamp = MoreUninterruptibles.tryReadLockUninterruptibly(delegate,
timeoutSupplier.get());
if(stamp == 0L) { // 0 here means that it timed out
throw new AcquireTimeoutException();
}
return stamp;
}
}
@ThreadSafe
private static class WriteLockStampWithTimeoutSupplier implements LongSupplier {
@NotNull
private final StampedLock delegate;
@NotNull
private final Supplier timeoutSupplier;
public WriteLockStampWithTimeoutSupplier(@NotNull final StampedLock delegate,
@NotNull final Supplier timeoutSupplier) {
assert delegate != null;
assert timeoutSupplier != null;
this.delegate = delegate;
this.timeoutSupplier = timeoutSupplier;
}
/**
*
* @return never 0 (which would indicate a timeout)
* @throws AcquireTimeoutException if there is a timeout
*/
@Override
public long getAsLong() throws AcquireTimeoutException {
final long stamp = MoreUninterruptibles.tryWriteLockUninterruptibly(delegate,
timeoutSupplier.get());
if(stamp == 0L) { // 0 here means that it timed out
throw new AcquireTimeoutException();
}
return stamp;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy