org.neo4j.kernel.impl.api.index.FlippableIndexProxy Maven / Gradle / Ivy
/*
* Copyright (c) "Neo4j"
* Neo4j Sweden AB [https://neo4j.com]
*
* This file is part of Neo4j.
*
* Neo4j is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
package org.neo4j.kernel.impl.api.index;
import java.io.IOException;
import java.nio.file.Path;
import java.util.Map;
import java.util.concurrent.Callable;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantReadWriteLock;
import org.neo4j.graphdb.ResourceIterator;
import org.neo4j.internal.kernel.api.InternalIndexState;
import org.neo4j.internal.kernel.api.PopulationProgress;
import org.neo4j.internal.kernel.api.exceptions.schema.IndexNotFoundKernelException;
import org.neo4j.internal.schema.IndexDescriptor;
import org.neo4j.io.pagecache.context.CursorContext;
import org.neo4j.io.pagecache.tracing.FileFlushEvent;
import org.neo4j.kernel.api.exceptions.index.ExceptionDuringFlipKernelException;
import org.neo4j.kernel.api.exceptions.index.IndexEntryConflictException;
import org.neo4j.kernel.api.exceptions.index.IndexPopulationFailedKernelException;
import org.neo4j.kernel.api.exceptions.index.IndexProxyAlreadyClosedKernelException;
import org.neo4j.kernel.api.exceptions.schema.IncompleteConstraintValidationException;
import org.neo4j.kernel.api.index.IndexUpdater;
import org.neo4j.kernel.api.index.TokenIndexReader;
import org.neo4j.kernel.api.index.ValueIndexReader;
import org.neo4j.kernel.impl.api.index.updater.DelegatingIndexUpdater;
import org.neo4j.values.storable.Value;
public class FlippableIndexProxy extends AbstractDelegatingIndexProxy {
private volatile boolean closed;
private final ReentrantReadWriteLock lock = new ReentrantReadWriteLock(true);
private volatile IndexProxyFactory flipTarget;
// This variable below is volatile because it can be changed in flip or flipTo
// and even though it may look like acquiring the read lock, when using this variable
// for various things, execution flow would go through a memory barrier of some sort.
// But it turns out that that may not be the case. F.ex. ReentrantReadWriteLock
// code uses unsafe compareAndSwap that sort of circumvents an equivalent of a volatile read.
private volatile IndexProxy delegate;
private boolean started;
public FlippableIndexProxy() {
this(null);
}
FlippableIndexProxy(IndexProxy originalDelegate) {
this.delegate = originalDelegate;
}
@Override
public IndexProxy getDelegate() {
return delegate;
}
@Override
public void start() {
lock.readLock().lock();
try {
delegate.start();
started = true;
} finally {
lock.readLock().unlock();
}
}
@Override
public IndexUpdater newUpdater(IndexUpdateMode mode, CursorContext cursorContext, boolean parallel) {
// Making use of reentrant locks to ensure that the delegate's constructor is called under lock protection
// while still retaining the lock until a call to close on the returned IndexUpdater
lock.readLock().lock();
try {
return new LockingIndexUpdater(delegate.newUpdater(mode, cursorContext, parallel));
} finally {
lock.readLock().unlock();
}
}
@Override
public void drop() {
lock.readLock().lock();
try {
closed = true;
delegate.drop();
} finally {
lock.readLock().unlock();
}
}
/**
* The {@code force()}-method is called during log rotation. At this time we do not want to wait for locks held by
* {@link LockingIndexUpdater}. Waiting on such locks would cause a serious risk of deadlocks, since very likely
* the reader we would be waiting on would be waiting on the log rotation lock held by the thread calling this
* method. The reason we would wait for a read lock while trying to acquire a read lock is if there is a third
* thread waiting on the write lock, probably an index populator wanting to
* {@linkplain #flip(Callable) flip the index into active state}.
*
* We avoid this deadlock situation by "barging" on the read lock, i.e. acquire it in an unfair way, where
* we don't care about waiting threads, only about whether the exclusive lock is held or not.
*/
@Override
public void force(FileFlushEvent flushEvent, CursorContext cursorContext) throws IOException {
barge(lock.readLock()); // see javadoc of this method (above) for rationale on why we use barge(...) here
try {
delegate.force(flushEvent, cursorContext);
} finally {
lock.readLock().unlock();
}
}
@Override
public void refresh() throws IOException {
lock.readLock().lock();
try {
delegate.refresh();
} finally {
lock.readLock().unlock();
}
}
/**
* Acquire the {@code ReadLock} in an unfair way, without waiting for queued up writers.
*
* The {@link ReentrantReadWriteLock.ReadLock#tryLock() tryLock}-method of the {@code ReadLock} implementation of
* {@code ReentrantReadWriteLock} implements a barging behaviour, where if an exclusive lock is not held,
* the shared lock will be acquired, even if there are other threads waiting for the lock. This behaviour is
* regardless of whether the lock is fair or not.
*
* This allows us to avoid deadlocks where readers would wait for writers that wait for readers in critical
* methods.
*
* The naive way to implement this method would be:
*
* if ( !lock.tryLock() ) // try to barge
* lock.lock(); // fall back to normal blocking lock call
*
* This would however not implement the appropriate barging behaviour in a scenario like the following: Say the
* exclusive lock is held, and there is a queue waiting containing first a reader and then a writer, in this case
* the {@code tryLock()} method will return false. If the writer then finishes between the naive implementation
* exiting {@code tryLock()} and before entering {@code lock()} the {@code barge(...)} method would now block in
* the exact way we don't want it to block, with a read lock held and a writer waiting.
* In order to get around this situation, the implementation of this method uses a
* {@linkplain Lock#tryLock(long, TimeUnit) timed wait} in a retry-loop in order to ensure that we make another
* attempt to barge the lock at a later point.
*
* This method is written to be compatible with the signature of {@link Lock#lock()}, which is not interruptible,
* but implemented based on the interruptible {@link Lock#tryLock(long, TimeUnit)}, so the implementation needs to
* remember being interrupted, and reset the flag before exiting, so that later invocations of interruptible
* methods detect the interruption.
*
* @param lock a {@link java.util.concurrent.locks.ReentrantReadWriteLock.ReadLock}
*/
private static void barge(ReentrantReadWriteLock.ReadLock lock) {
boolean interrupted = false;
// exponential retry back-off, no more than 1 second
for (long timeout = 10; !lock.tryLock(); timeout = Math.min(1000, timeout * 2)) {
try {
if (lock.tryLock(timeout, TimeUnit.MILLISECONDS)) {
return;
}
}
// the barge()-method is uninterruptable, but implemented based on the interruptible tryLock()-method
catch (InterruptedException e) {
Thread.interrupted(); // ensure the interrupt flag is cleared
interrupted = true; // remember to set interrupt flag before we exit
}
}
if (interrupted) {
Thread.currentThread().interrupt(); // reset the interrupt flag
}
}
@Override
public IndexDescriptor getDescriptor() {
lock.readLock().lock();
try {
return delegate.getDescriptor();
} finally {
lock.readLock().unlock();
}
}
@Override
public InternalIndexState getState() {
lock.readLock().lock();
try {
return delegate.getState();
} finally {
lock.readLock().unlock();
}
}
@Override
public void close(CursorContext cursorContext) throws IOException {
lock.readLock().lock();
try {
closed = true;
delegate.close(cursorContext);
} finally {
lock.readLock().unlock();
}
}
@Override
public ValueIndexReader newValueReader() throws IndexNotFoundKernelException {
lock.readLock().lock();
try {
return delegate.newValueReader();
} finally {
lock.readLock().unlock();
}
}
@Override
public TokenIndexReader newTokenReader() throws IndexNotFoundKernelException {
lock.readLock().lock();
try {
return delegate.newTokenReader();
} finally {
lock.readLock().unlock();
}
}
@Override
public boolean awaitStoreScanCompleted(long time, TimeUnit unit)
throws IndexPopulationFailedKernelException, InterruptedException {
IndexProxy proxy;
lock.readLock().lock();
proxy = delegate;
lock.readLock().unlock();
if (closed) {
return false;
}
boolean stillGoing = proxy.awaitStoreScanCompleted(time, unit);
if (!stillGoing) {
// The waiting has ended. However we're not done because say that the delegate typically is a populating
// proxy, when the wait is over
// the populating proxy flips into something else, and if that is a failed proxy then that failure should
// propagate out from this call.
lock.readLock().lock();
proxy = delegate;
lock.readLock().unlock();
proxy.awaitStoreScanCompleted(time, unit);
}
return stillGoing;
}
@Override
public void activate() {
// use write lock, since activate() might call flip*() which acquires a write lock itself.
lock.writeLock().lock();
try {
delegate.activate();
} finally {
lock.writeLock().unlock();
}
}
@Override
public void validate() throws IndexPopulationFailedKernelException, IncompleteConstraintValidationException {
lock.readLock().lock();
try {
delegate.validate();
} finally {
lock.readLock().unlock();
}
}
@Override
public void validateBeforeCommit(Value[] tuple, long entityId) {
lock.readLock().lock();
try {
delegate.validateBeforeCommit(tuple, entityId);
} finally {
lock.readLock().unlock();
}
}
@Override
public ResourceIterator snapshotFiles() throws IOException {
lock.readLock().lock();
try {
return delegate.snapshotFiles();
} finally {
lock.readLock().unlock();
}
}
@Override
public Map indexConfig() {
lock.readLock().lock();
try {
return delegate.indexConfig();
} finally {
lock.readLock().unlock();
}
}
@Override
public IndexPopulationFailure getPopulationFailure() throws IllegalStateException {
lock.readLock().lock();
try {
return delegate.getPopulationFailure();
} finally {
lock.readLock().unlock();
}
}
@Override
public PopulationProgress getIndexPopulationProgress() {
lock.readLock().lock();
try {
return delegate.getIndexPopulationProgress();
} finally {
lock.readLock().unlock();
}
}
@Override
public void maintenance() {
lock.readLock().lock();
try {
delegate.maintenance();
} finally {
lock.readLock().unlock();
}
}
void setFlipTarget(IndexProxyFactory flipTarget) {
lock.writeLock().lock();
try {
this.flipTarget = flipTarget;
} finally {
lock.writeLock().unlock();
}
}
void flipTo(IndexProxy targetDelegate) {
lock.writeLock().lock();
try {
this.delegate = targetDelegate;
} finally {
lock.writeLock().unlock();
}
}
public void flip(Callable actionDuringFlip)
throws IndexProxyAlreadyClosedKernelException, ExceptionDuringFlipKernelException {
lock.writeLock().lock();
try {
assertOpen();
try {
if (actionDuringFlip.call()) {
this.delegate = flipTarget.create();
if (started) {
this.delegate.start();
}
}
} catch (Exception e) {
throw new ExceptionDuringFlipKernelException(e);
}
} finally {
lock.writeLock().unlock();
}
}
@Override
public String toString() {
return getClass().getSimpleName() + " -> " + delegate + "[target:" + flipTarget + "]";
}
private void assertOpen() throws IndexProxyAlreadyClosedKernelException {
if (closed) {
throw new IndexProxyAlreadyClosedKernelException(this.getClass());
}
}
private class LockingIndexUpdater extends DelegatingIndexUpdater {
private LockingIndexUpdater(IndexUpdater delegate) {
super(delegate);
lock.readLock().lock();
}
@Override
public void close() throws IndexEntryConflictException {
try {
delegate.close();
} finally {
lock.readLock().unlock();
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy