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

reactor.util.concurrent.WaitStrategy Maven / Gradle / Ivy

/*
 * Copyright (c) 2011-2017 Pivotal Software Inc, All Rights Reserved.
 *
 * 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 reactor.util.concurrent;

import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.LockSupport;
import java.util.concurrent.locks.ReentrantLock;
import java.util.function.LongSupplier;


/**
 * Strategy employed to wait for specific {@link LongSupplier} values with various spinning strategies.
 */
public abstract class WaitStrategy {

    /**
     * A no-op {@link Runnable} that can be used as a placeholder spin observer  with
     * {@link #waitFor(long, LongSupplier, Runnable)}
     */
    public static final Runnable NOOP_SPIN_OBSERVER = () -> { };

    /**
     * Blocking strategy that uses a lock and condition variable for consumer waiting on a barrier.
     *
     * This strategy can be used when throughput and low-latency are not as important as CPU resource.
     * @return the wait strategy
     */
    public static WaitStrategy blocking() {
        return new Blocking();
    }

    /**
     * Busy Spin strategy that uses a busy spin loop for consumers waiting on a barrier.
     *
     * This strategy will use CPU resource to avoid syscalls which can introduce latency jitter.  It is best
     * used when threads can be bound to specific CPU cores.
     * @return the wait strategy
     */
    public static WaitStrategy busySpin() {
        return BusySpin.INSTANCE;
    }

    /**
     * Test if exception is alert
     * @param t exception checked
     * @return true if this is an alert signal
     */
    public static boolean isAlert(Throwable t){
	    return t == AlertException.INSTANCE;
    }

    /**
     * Variation of the {@link #blocking()} that attempts to elide conditional wake-ups when the lock is uncontended.
     * Shows performance improvements on microbenchmarks.  However this wait strategy should be considered experimental
     * as I have not full proved the correctness of the lock elision code.
     * @return the wait strategy
     */
    public static WaitStrategy liteBlocking() {
        return new LiteBlocking();
    }

    /**
     * Parking strategy that initially spins, then uses a Thread.yield(), and eventually sleep
     * (LockSupport.parkNanos(1)) for the minimum number of nanos the OS and JVM will allow while the
     * consumers are waiting on a barrier.
     * 

* This strategy is a good compromise between performance and CPU resource. Latency spikes can occur after quiet * periods. * @return the wait strategy */ public static WaitStrategy parking() { return Parking.INSTANCE; } /** * Parking strategy that initially spins, then uses a Thread.yield(), and eventually * sleep (LockSupport.parkNanos(1)) for the minimum number of nanos the * OS and JVM will allow while the consumers are waiting on a barrier. *

* This strategy is a good compromise between performance and CPU resource. Latency * spikes can occur after quiet periods. * * @param retries the spin cycle count before parking * @return the wait strategy */ public static WaitStrategy parking(int retries) { return new Parking(retries); } /** *

Phased wait strategy for waiting consumers on a barrier.

*

*

This strategy can be used when throughput and low-latency are not as important as CPU resource. Spins, then * yields, then waits using the configured fallback WaitStrategy.

* * @param spinTimeout the spin timeout * @param yieldTimeout the yield timeout * @param units the time unit * @param delegate the target wait strategy to fallback on * @return the wait strategy */ public static WaitStrategy phasedOff(long spinTimeout, long yieldTimeout, TimeUnit units, WaitStrategy delegate) { return new PhasedOff(spinTimeout, yieldTimeout, units, delegate); } /** * Block with wait/notifyAll semantics * @param spinTimeout the spin timeout * @param yieldTimeout the yield timeout * @param units the time unit * @return the wait strategy */ public static WaitStrategy phasedOffLiteLock(long spinTimeout, long yieldTimeout, TimeUnit units) { return phasedOff(spinTimeout, yieldTimeout, units, liteBlocking()); } /** * Block with wait/notifyAll semantics * @param spinTimeout the spin timeout * @param yieldTimeout the yield timeout * @param units the time unit * @return the wait strategy */ public static WaitStrategy phasedOffLock(long spinTimeout, long yieldTimeout, TimeUnit units) { return phasedOff(spinTimeout, yieldTimeout, units, blocking()); } /** * Block by parking in a loop * @param spinTimeout the spin timeout * @param yieldTimeout the yield timeout * @param units the time unit * @return the wait strategy */ public static WaitStrategy phasedOffSleep(long spinTimeout, long yieldTimeout, TimeUnit units) { return phasedOff(spinTimeout, yieldTimeout, units, parking(0)); } /** * Yielding strategy that uses a Thread.sleep(1) for consumers waiting on a * barrier * after an initially spinning. * * This strategy will incur up a latency of 1ms and save a maximum CPU resources. * @return the wait strategy */ public static WaitStrategy sleeping() { return Sleeping.INSTANCE; } /** * Yielding strategy that uses a Thread.yield() for consumers waiting on a * barrier * after an initially spinning. * * This strategy is a good compromise between performance and CPU resource without incurring significant latency spikes. * @return the wait strategy */ public static WaitStrategy yielding() { return Yielding.INSTANCE; } /** * Implementations should signal the waiting consumers that the cursor has advanced. */ public void signalAllWhenBlocking() { } /** * Wait for the given sequence to be available. It is possible for this method to return a value * less than the sequence number supplied depending on the implementation of the WaitStrategy. A common * use for this is to signal a timeout. Any EventProcessor that is using a WaitStrategy to get notifications * about message becoming available should remember to handle this case. * * @param sequence to be waited on. * @param cursor the main sequence from ringbuffer. Wait/notify strategies will * need this as is notified upon update. * @param spinObserver Spin observer * @return the sequence that is available which may be greater than the requested sequence. * @throws InterruptedException if the thread is interrupted. */ public abstract long waitFor(long sequence, LongSupplier cursor, Runnable spinObserver) throws InterruptedException; /** * Throw an Alert signal exception (singleton) that can be checked against {@link #isAlert(Throwable)} */ public static void alert(){ throw AlertException.INSTANCE; } /** * Used to alert consumers waiting with a {@link WaitStrategy} for status changes. *

* It does not fill in a stack trace for performance reasons. */ @SuppressWarnings("serial") static final class AlertException extends RuntimeException { /** Pre-allocated exception to avoid garbage generation */ public static final AlertException INSTANCE = new AlertException(); /** * Private constructor so only a single instance any. */ private AlertException() { } /** * Overridden so the stack trace is not filled in for this exception for performance reasons. * * @return this instance. */ @Override public Throwable fillInStackTrace() { return this; } } final static class Blocking extends WaitStrategy { private final Lock lock = new ReentrantLock(); private final Condition processorNotifyCondition = lock.newCondition(); @Override public void signalAllWhenBlocking() { lock.lock(); try { processorNotifyCondition.signalAll(); } finally { lock.unlock(); } } @SuppressWarnings("UnusedAssignment") //for availableSequence @Override public long waitFor(long sequence, LongSupplier cursorSequence, Runnable barrier) throws InterruptedException { long availableSequence; if ((availableSequence = cursorSequence.getAsLong()) < sequence) { lock.lock(); try { while ((availableSequence = cursorSequence.getAsLong()) < sequence) { barrier.run(); processorNotifyCondition.await(); } } finally { lock.unlock(); } } while ((availableSequence = cursorSequence.getAsLong()) < sequence) { barrier.run(); } return availableSequence; } } final static class BusySpin extends WaitStrategy { static final BusySpin INSTANCE = new BusySpin(); @Override public long waitFor(final long sequence, LongSupplier cursor, final Runnable barrier) throws InterruptedException { long availableSequence; while ((availableSequence = cursor.getAsLong()) < sequence) { barrier.run(); } return availableSequence; } } final static class Sleeping extends WaitStrategy { static final Sleeping INSTANCE = new Sleeping(); @Override public long waitFor(final long sequence, LongSupplier cursor, final Runnable barrier) throws InterruptedException { long availableSequence; while ((availableSequence = cursor.getAsLong()) < sequence) { barrier.run(); Thread.sleep(1); } return availableSequence; } } final static class LiteBlocking extends WaitStrategy { private final Lock lock = new ReentrantLock(); private final Condition processorNotifyCondition = lock.newCondition(); private final AtomicBoolean signalNeeded = new AtomicBoolean(false); @Override public void signalAllWhenBlocking() { if (signalNeeded.getAndSet(false)) { lock.lock(); try { processorNotifyCondition.signalAll(); } finally { lock.unlock(); } } } @SuppressWarnings("UnusedAssignment") //for availableSequence @Override public long waitFor(long sequence, LongSupplier cursorSequence, Runnable barrier) throws InterruptedException { long availableSequence; if ((availableSequence = cursorSequence.getAsLong()) < sequence) { lock.lock(); try { do { signalNeeded.getAndSet(true); if ((availableSequence = cursorSequence.getAsLong()) >= sequence) { break; } barrier.run(); processorNotifyCondition.await(); } while ((availableSequence = cursorSequence.getAsLong()) < sequence); } finally { lock.unlock(); } } while ((availableSequence = cursorSequence.getAsLong()) < sequence) { barrier.run(); } return availableSequence; } } final static class PhasedOff extends WaitStrategy { private final long spinTimeoutNanos; private final long yieldTimeoutNanos; private final WaitStrategy fallbackStrategy; PhasedOff(long spinTimeout, long yieldTimeout, TimeUnit units, WaitStrategy fallbackStrategy) { this.spinTimeoutNanos = units.toNanos(spinTimeout); this.yieldTimeoutNanos = spinTimeoutNanos + units.toNanos(yieldTimeout); this.fallbackStrategy = fallbackStrategy; } @Override public void signalAllWhenBlocking() { fallbackStrategy.signalAllWhenBlocking(); } @Override public long waitFor(long sequence, LongSupplier cursor, Runnable barrier) throws InterruptedException { long availableSequence; long startTime = 0; int counter = SPIN_TRIES; do { if ((availableSequence = cursor.getAsLong()) >= sequence) { return availableSequence; } if (0 == --counter) { if (0 == startTime) { startTime = System.nanoTime(); } else { long timeDelta = System.nanoTime() - startTime; if (timeDelta > yieldTimeoutNanos) { return fallbackStrategy.waitFor(sequence, cursor, barrier); } else if (timeDelta > spinTimeoutNanos) { Thread.yield(); } } counter = SPIN_TRIES; } barrier.run(); } while (true); } private static final int SPIN_TRIES = 10000; } final static class Parking extends WaitStrategy { static final Parking INSTANCE = new Parking(); private final int retries; Parking() { this(DEFAULT_RETRIES); } Parking(int retries) { this.retries = retries; } @Override public long waitFor(final long sequence, LongSupplier cursor, final Runnable barrier) throws InterruptedException { long availableSequence; int counter = retries; while ((availableSequence = cursor.getAsLong()) < sequence) { counter = applyWaitMethod(barrier, counter); } return availableSequence; } private int applyWaitMethod(final Runnable barrier, int counter) throws WaitStrategy.AlertException { barrier.run(); if (counter > 100) { --counter; } else if (counter > 0) { --counter; Thread.yield(); } else { LockSupport.parkNanos(1L); } return counter; } private static final int DEFAULT_RETRIES = 200; } final static class Yielding extends WaitStrategy { static final Yielding INSTANCE = new Yielding(); @Override public long waitFor(final long sequence, LongSupplier cursor, final Runnable barrier) throws InterruptedException { long availableSequence; int counter = SPIN_TRIES; while ((availableSequence = cursor.getAsLong()) < sequence) { counter = applyWaitMethod(barrier, counter); } return availableSequence; } private int applyWaitMethod(final Runnable barrier, int counter) throws WaitStrategy.AlertException { barrier.run(); if (0 == counter) { Thread.yield(); } else { --counter; } return counter; } private static final int SPIN_TRIES = 100; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy