All Downloads are FREE. Search and download functionalities are using the official Maven repository.

aQute.bnd.build.WorkspaceLock Maven / Gradle / Ivy

The newest version!
package aQute.bnd.build;

import java.lang.management.ManagementFactory;
import java.lang.management.ThreadMXBean;
import java.util.Arrays;
import java.util.Objects;
import java.util.Queue;
import java.util.concurrent.Callable;
import java.util.concurrent.CancellationException;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantReadWriteLock;
import java.util.function.BooleanSupplier;
import java.util.stream.Collectors;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import aQute.bnd.exceptions.FunctionWithException;
import aQute.lib.strings.Strings;

/**
 * ReentrantReadWriteLock lock for serializing access to the Workspace.
 */
final class WorkspaceLock extends ReentrantReadWriteLock {
	private static final long	serialVersionUID	= 1L;
	private static final Logger	logger				= LoggerFactory.getLogger(WorkspaceLock.class);
	private final AtomicInteger	progress			= new AtomicInteger();
	private final Queue	readLockHolders		= new ConcurrentLinkedQueue<>();

	WorkspaceLock(boolean fair) {
		super(fair);
	}

	private String type(Lock lock) {
		if (lock == readLock()) {
			return "read";
		}
		if (lock == writeLock()) {
			return "write";
		}
		return "unknown_lock";
	}

	private void trace(String name, Lock lock) {
		if (logger.isDebugEnabled()) {
			String trace = Arrays.stream(new Exception().getStackTrace())
				.dropWhile(ste -> {
				String className = ste.getClassName();
				if (Objects.equals(className, "aQute.bnd.build.WorkspaceLock")) {
					return true;
				}
				if (Objects.equals(className, "aQute.bnd.build.Workspace")) {
					String methodName = ste.getMethodName();
					if (Objects.equals(methodName, "readLocked") || Objects.equals(methodName, "writeLocked")) {
						return true;
					}
				}
				return false;
			})
				.limit(5L)
				.map(Object::toString)
				.collect(Strings.joining("\n\tat ", "\n\tat ", "", ""));
			logger.debug("{} {} @{}[Write locks = {}, Read locks = {}]; owner {}; waiting {}{}", name, type(lock),
				Integer.toHexString(hashCode()), getWriteHoldCount(), getReadLockCount(), getOwner(),
				getQueuedThreads(), trace);
		}
	}

	private Throwable getOwnerCause() {
		final Thread owner = getOwner();
		if (owner == null) {
			return null;
		}
		final Throwable cause = new Throwable(
			owner + " owns the WorkspaceLock\n\nFull thread dump:\n" + dumpAllThreads() + "Owner stacktrace:");
		cause.setStackTrace(owner.getStackTrace());
		return cause;
	}

	private String dumpAllThreads() {
		ThreadMXBean threadMXBean = ManagementFactory.getThreadMXBean();
		return Arrays
			.stream(threadMXBean.dumpAllThreads(threadMXBean.isObjectMonitorUsageSupported(),
				threadMXBean.isSynchronizerUsageSupported()))
			.map(Object::toString)
			.collect(Collectors.joining());
	}

	private CancellationException canceled(Lock lock) {
		CancellationException e = new CancellationException(
			String.format("Canceled waiting to %s acquire %s; owner %s; waiting %s", type(lock), this, getOwner(),
				getQueuedThreads()));
		e.initCause(getOwnerCause());
		return e;
	}

	private TimeoutException timeout(Lock lock) {
		TimeoutException e = new TimeoutException(
			String.format("Timeout waiting to %s acquire %s; owner %s; waiting %s", type(lock), this, getOwner(),
				getQueuedThreads()));
		e.initCause(getOwnerCause());
		return e;
	}

	private IllegalStateException deadlock(Lock lock) {
		IllegalStateException e = new IllegalStateException(String.format(
			"Deadlock situation detected trying to %s acquire %s. The current thread already holds a read lock: %s",
			type(lock), this, Thread.currentThread()));
		return e;
	}

	 T writeReadLocked(final long timeoutInMs, final Callable underWrite,
		final FunctionWithException underRead, final BooleanSupplier canceled) throws Exception {
		Callable writeLocked = () -> {
			U writeResult = underWrite.call();
			// It is safe to let an exception propagate from here.
			// Downgrade lock by taking read lock before returning to release
			// write lock.
			trace("Downgrading", writeLock());
			readLock().lock();
			return writeResult;
		};
		U writeResult = locked(writeLock(), timeoutInMs, writeLocked, canceled);
		// If the call to locked completed normally, we hold the read
		// lock obtained in the writeLocked callable.
		try {
			Callable readLocked = () -> underRead.apply(writeResult);
			return locked(readLock(), timeoutInMs, readLocked, canceled);
		} finally {
			// Must unlock the read lock taken above when we held the write
			// lock.
			readLock().unlock();
		}
	}

	 T locked(final Lock lock, final long timeoutInMs, final Callable callable, final BooleanSupplier canceled)
		throws Exception {
		boolean interrupted = Thread.interrupted();
		final Thread currentThread = Thread.currentThread();
		final boolean readLockRequest = lock == readLock();

		trace("Enter", lock);
		try {
			int startingProgress = progress.get();
			long remaining = timeoutInMs;
			do {
				if (canceled.getAsBoolean()) {
					throw canceled(lock);
				}
				long wait = Math.min(remaining, 1_000L); // max is 1s
				long start = System.nanoTime();
				boolean locked;
				try {
					locked = lock.tryLock(wait, TimeUnit.MILLISECONDS);
				} catch (InterruptedException e) {
					interrupted = true;
					throw e;
				}
				if (locked) {
					try {
						if (readLockRequest) {
							readLockHolders.add(currentThread);
							try {
								return callable.call();
							} finally {
								readLockHolders.remove(currentThread);
							}
						}
						return callable.call();
					} finally {
						progress.incrementAndGet();
						lock.unlock();
					}
				}
				// We cannot hold read lock when requesting write lock
				if (!readLockRequest && readLockHolders.contains(currentThread)) {
					throw deadlock(lock);
				}
				int currentProgress = progress.get();
				if (startingProgress == currentProgress) {
					long elapsed = TimeUnit.NANOSECONDS.toMillis(System.nanoTime() - start);
					remaining -= elapsed;
					trace("Busy with no progress", lock);
				} else {
					startingProgress = currentProgress;
					trace("Busy but progressing", lock);
				}
			} while (remaining > 0L);
			throw timeout(lock);
		} finally {
			trace("Exit", lock);
			if (interrupted) {
				currentThread.interrupt();
			}
		}
	}
}