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

com.github.benmanes.caffeine.SingleConsumerQueue Maven / Gradle / Ivy

/*
 * Copyright 2014 Ben Manes. 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 com.github.benmanes.caffeine;

import static java.util.Objects.requireNonNull;

import java.io.InvalidObjectException;
import java.io.ObjectInputStream;
import java.io.Serializable;
import java.util.AbstractQueue;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;
import java.util.List;
import java.util.NoSuchElementException;
import java.util.Queue;
import java.util.concurrent.ThreadLocalRandom;
import java.util.concurrent.atomic.AtomicReference;
import java.util.function.Function;

import org.checkerframework.checker.nullness.qual.NonNull;
import org.checkerframework.checker.nullness.qual.Nullable;

/**
 * A lock-free unbounded queue based on linked nodes that supports concurrent producers and is
 * restricted to a single consumer. This queue orders elements FIFO (first-in-first-out). The
 * head of the queue is that element that has been on the queue the longest time. The
 * tail of the queue is that element that has been on the queue the shortest time. New
 * elements are inserted at the tail of the queue, and the queue retrieval operations obtain
 * elements at the head of the queue. Like most other concurrent collection implementations, this
 * class does not permit the use of {@code null} elements.
 * 

* A {@code SingleConsumerQueue} is an appropriate choice when many producer threads will share * access to a common collection and a single consumer thread drains it. This collection is useful * in scenarios such as implementing flat combining, actors, or lock amortization. *

* This implementation employs combination to transfer elements between threads that are producing * concurrently. This approach avoids contention on the queue by combining colliding operations * that have identical semantics. When a pair of producers collide, the task of performing the * combined set of operations is delegated to one of the threads and the other thread optionally * waits for its operation to be completed. This decision of whether to wait for completion is * determined by constructing either a linearizable or optimistic queue. *

* Iterators are weakly consistent, returning elements reflecting the state of the queue at * some point at or since the creation of the iterator. They do not throw {@link * java.util.ConcurrentModificationException}, and may proceed concurrently with other operations. * Elements contained in the queue since the creation of the iterator will be returned exactly once. *

* Beware that it is the responsibility of the caller to ensure that a consumer has exclusive read * access to the queue. This implementation does not include fail-fast behavior to guard * against incorrect consumer usage. *

* Beware that, unlike in most collections, the {@code size} method is NOT a * constant-time operation. Because of the asynchronous nature of these queues, determining the * current number of elements requires a traversal of the elements, and so may report inaccurate * results if this collection is modified during traversal. *

* Warning: This class is scheduled for removal in version 3.0.0. * * @author [email protected] (Ben Manes) * @param the type of elements held in this collection * @deprecated Scheduled for removal in version 3.0.0 */ @Deprecated @SuppressWarnings("NullAway") public final class SingleConsumerQueue extends SCQHeader.HeadAndTailRef implements Queue, Serializable { /* * The queue is represented as a singly-linked list with an atomic head and tail reference. It is * based on the non-intrusive multi-producer / single-consumer node queue described by * Dmitriy Vyukov [1]. * * The backoff strategy of combining operations with identical semantics is based on inverting * the elimination technique [2]. Elimination allows pairs of operations with reverse semantics, * like pushes and pops on a stack, to complete without any central coordination and therefore * substantially aids scalability. The approach of applying elimination and reversing its * semantics was explored in [3, 4]. Unlike other approaches, this implementation does not use * opcodes or a background thread. * * This implementation borrows optimizations from {@link java.util.concurrent.Exchanger} for * choosing an arena location and awaiting a match [5]. * * [1] Non-intrusive MPSC node-based queue * http://www.1024cores.net/home/lock-free-algorithms/queues/non-intrusive-mpsc-node-based-queue * [2] A Scalable Lock-free Stack Algorithm * http://citeseerx.ist.psu.edu/viewdoc/summary?doi=10.1.1.156.8728 * [3] Using Elimination to Implement Scalable and Lock-Free FIFO Queues * http://citeseerx.ist.psu.edu/viewdoc/summary?doi=10.1.1.108.6422 * [4] A Dynamic Elimination-Combining Stack Algorithm * http://www.cs.bgu.ac.il/~hendlerd/papers/DECS.pdf * [5] A Scalable Elimination-based Exchange Channel * http://citeseerx.ist.psu.edu/viewdoc/summary?doi=10.1.1.59.7396 */ /** The number of CPUs */ static final int NCPU = Runtime.getRuntime().availableProcessors(); /** The number of slots in the elimination array. */ static final int ARENA_LENGTH = ceilingPowerOfTwo((NCPU + 1) / 2); /** The mask value for indexing into the arena. */ static final int ARENA_MASK = ARENA_LENGTH - 1; /** The factory for creating an optimistic node. */ static final Function OPTIMISIC = Node::new; /** * The number of times to spin (doing nothing except polling a memory location) before giving up * while waiting to eliminate an operation. Should be zero on uniprocessors. On multiprocessors, * this value should be large enough so that two threads exchanging items as fast as possible * block only when one of them is stalled (due to GC or preemption), but not much longer, to avoid * wasting CPU resources. Seen differently, this value is a little over half the number of cycles * of an average context switch time on most systems. The value here is approximately the average * of those across a range of tested systems. */ static final int SPINS = (NCPU == 1) ? 0 : 2000; /** The offset to the thread-specific probe field. */ static final long PROBE = com.github.benmanes.caffeine.base.UnsafeAccess.objectFieldOffset( Thread.class, "threadLocalRandomProbe"); static int ceilingPowerOfTwo(int x) { // From Hacker's Delight, Chapter 3, Harry S. Warren Jr. return 1 << -Integer.numberOfLeadingZeros(x - 1); } final AtomicReference>[] arena; final Function> factory; @SuppressWarnings({"unchecked", "rawtypes"}) private SingleConsumerQueue(Function> factory) { arena = new AtomicReference[ARENA_LENGTH]; for (int i = 0; i < ARENA_LENGTH; i++) { arena[i] = new AtomicReference<>(); } Node node = new Node<>(null); this.factory = factory; lazySetTail(node); head = node; } /** * Creates a queue with an optimistic backoff strategy. A thread completes its operation * without waiting after it successfully hands off the additional element(s) to another producing * thread for batch insertion. This optimistic behavior may result in additions not appearing in * FIFO order due to the backoff strategy trying to compensate for queue contention. * * @param the type of elements held in this collection * @return a new queue where producers complete their operation immediately if combined with * another producing thread's */ public static SingleConsumerQueue optimistic() { @SuppressWarnings("unchecked") Function> factory = (Function>) OPTIMISIC; return new SingleConsumerQueue<>(factory); } /** * Creates a queue with a linearizable backoff strategy. A thread waits for a completion * signal if it successfully hands off the additional element(s) to another producing * thread for batch insertion. * * @param the type of elements held in this collection * @return a new queue where producers wait for a completion signal after combining its addition * with another producing thread's */ public static SingleConsumerQueue linearizable() { return new SingleConsumerQueue<>(LinearizableNode::new); } @Override public boolean isEmpty() { return (head == tail); } @Override public int size() { Node cursor = head; Node t = tail; int size = 0; while ((cursor != t) && (size != Integer.MAX_VALUE)) { Node next = cursor.getNextRelaxed(); if (next == null) { while ((next = cursor.next) == null) {} } cursor = next; size++; } return size; } @Override public boolean contains(Object o) { if (o == null) { return false; } for (Iterator it = iterator(); it.hasNext();) { if (o.equals(it.next())) { return true; } } return false; } @Override public E peek() { Node h = head; Node t = tail; if (h == t) { return null; } Node next = h.getNextRelaxed(); if (next == null) { while ((next = h.next) == null) {} } return next.value; } @Override public boolean offer(E e) { requireNonNull(e); Node node = factory.apply(e); append(node, node); return true; } @Override public E poll() { Node h = head; Node next = h.getNextRelaxed(); if (next == null) { if (h == tail) { return null; } else { while ((next = h.next) == null) {} } } E e = next.value; next.value = null; head = next; if (factory == OPTIMISIC) { h.next = null; // prevent nepotism } return e; } @Override public boolean add(E e) { return offer(e); } @Override public boolean addAll(Collection c) { requireNonNull(c); Node first = null; Node last = null; for (E e : c) { requireNonNull(e); if (first == null) { first = factory.apply(e); last = first; } else { Node newLast = new Node<>(e); last.lazySetNext(newLast); last = newLast; } } if (first == null) { return false; } append(first, last); return true; } /** Adds the linked list of nodes to the queue. */ void append(@NonNull Node first, @NonNull Node last) { for (;;) { Node t = tail; if (casTail(t, last)) { t.lazySetNext(first); if (factory == OPTIMISIC) { return; } for (;;) { first.complete(); if (first == last) { return; } Node next = first.getNextRelaxed(); if (next == null) { return; } else if (next.value == null) { first.next = null; // reduce nepotism } first = next; } } Node node = transferOrCombine(first, last); if (node == null) { first.await(); return; } else if (node != first) { last = node; } } } /** * Attempts to receive a linked list from a waiting producer or transfer the specified linked list * to an arriving producer. * * @param first the first node in the linked list to try to transfer * @param last the last node in the linked list to try to transfer * @return either {@code null} if the element was transferred, the first node if neither a * transfer nor receive were successful, or the received last element from a producer */ @Nullable Node transferOrCombine(@NonNull Node first, Node last) { int index = index(); AtomicReference> slot = arena[index]; for (;;) { Node found = slot.get(); if (found == null) { if (slot.compareAndSet(null, first)) { for (int spin = 0; spin < SPINS; spin++) { if (slot.get() != first) { return null; } } return slot.compareAndSet(first, null) ? first : null; } } else if (slot.compareAndSet(found, null)) { last.lazySetNext(found); last = findLast(found); for (int i = 1; i < ARENA_LENGTH; i++) { slot = arena[(i + index) & ARENA_MASK]; found = slot.get(); if ((found != null) && slot.compareAndSet(found, null)) { last.lazySetNext(found); last = findLast(found); } } return last; } } } /** Returns the arena index for the current thread. */ static int index() { int probe = com.github.benmanes.caffeine.base.UnsafeAccess.UNSAFE.getInt( Thread.currentThread(), PROBE); if (probe == 0) { ThreadLocalRandom.current(); // force initialization probe = com.github.benmanes.caffeine.base.UnsafeAccess.UNSAFE.getInt( Thread.currentThread(), PROBE); } return (probe & ARENA_MASK); } /** Returns the last node in the linked list. */ @NonNull static Node findLast(@NonNull Node node) { Node next; while ((next = node.getNextRelaxed()) != null) { node = next; } return node; } @Override public Iterator iterator() { return new Iterator() { Node prev; Node t = tail; Node cursor = head; boolean failOnRemoval = true; @Override public boolean hasNext() { return (cursor != t); } @Override public E next() { if (!hasNext()) { throw new NoSuchElementException(); } advance(); failOnRemoval = false; return cursor.value; } private void advance() { if ((prev == null) || !failOnRemoval) { prev = cursor; } cursor = awaitNext(); } @Override public void remove() { if (failOnRemoval) { throw new IllegalStateException(); } failOnRemoval = true; cursor.value = null; if (t == cursor) { prev.lazySetNext(null); if (casTail(t, prev)) { return; } } prev.lazySetNext(awaitNext()); } Node awaitNext() { if (cursor.getNextRelaxed() == null) { while (cursor.next == null) {} } return cursor.getNextRelaxed(); } }; } /* --------------- Serialization Support --------------- */ static final long serialVersionUID = 1; Object writeReplace() { return new SerializationProxy<>(this); } @SuppressWarnings("UnusedVariable") private void readObject(ObjectInputStream stream) throws InvalidObjectException { throw new InvalidObjectException("Proxy required"); } /** A proxy that is serialized instead of the queue. */ static final class SerializationProxy implements Serializable { final boolean linearizable; final List elements; SerializationProxy(SingleConsumerQueue queue) { linearizable = (queue.factory.apply(null) instanceof LinearizableNode); elements = new ArrayList<>(queue); } Object readResolve() { SingleConsumerQueue queue = linearizable ? linearizable() : optimistic(); queue.addAll(elements); return queue; } static final long serialVersionUID = 1; } static class Node { static final long NEXT_OFFSET = com.github.benmanes.caffeine.base.UnsafeAccess.objectFieldOffset(Node.class, "next"); @Nullable E value; @Nullable volatile Node next; Node(@Nullable E value) { this.value = value; } @SuppressWarnings("unchecked") @Nullable Node getNextRelaxed() { return (Node) com.github.benmanes.caffeine.base.UnsafeAccess.UNSAFE.getObject( this, NEXT_OFFSET); } void lazySetNext(@Nullable Node newNext) { com.github.benmanes.caffeine.base.UnsafeAccess.UNSAFE.putOrderedObject( this, NEXT_OFFSET, newNext); } /** A no-op notification that the element was added to the queue. */ void complete() {} /** A no-op wait until the operation has completed. */ void await() {} /** Always returns that the operation completed. */ boolean isDone() { return true; } @Override public String toString() { return getClass().getSimpleName() + "[" + value + "]"; } } static final class LinearizableNode extends Node { volatile boolean done; LinearizableNode(@Nullable E value) { super(value); } /** A notification that the element was added to the queue. */ @Override void complete() { done = true; } /** A busy wait until the operation has completed. */ @Override void await() { while (!done) {} } /** Returns whether the operation completed. */ @Override boolean isDone() { return done; } } } /** The namespace for field padding through inheritance. */ final class SCQHeader { abstract static class PadHead extends AbstractQueue { byte p000, p001, p002, p003, p004, p005, p006, p007; byte p008, p009, p010, p011, p012, p013, p014, p015; byte p016, p017, p018, p019, p020, p021, p022, p023; byte p024, p025, p026, p027, p028, p029, p030, p031; byte p032, p033, p034, p035, p036, p037, p038, p039; byte p040, p041, p042, p043, p044, p045, p046, p047; byte p048, p049, p050, p051, p052, p053, p054, p055; byte p056, p057, p058, p059, p060, p061, p062, p063; byte p064, p065, p066, p067, p068, p069, p070, p071; byte p072, p073, p074, p075, p076, p077, p078, p079; byte p080, p081, p082, p083, p084, p085, p086, p087; byte p088, p089, p090, p091, p092, p093, p094, p095; byte p096, p097, p098, p099, p100, p101, p102, p103; byte p104, p105, p106, p107, p108, p109, p110, p111; byte p112, p113, p114, p115, p116, p117, p118, p119; } /** Enforces a memory layout to avoid false sharing by padding the head node. */ abstract static class HeadRef extends PadHead { @SuppressWarnings("deprecation") SingleConsumerQueue.@Nullable Node head; } abstract static class PadHeadAndTail extends HeadRef { byte p120, p121, p122, p123, p124, p125, p126, p127; byte p128, p129, p130, p131, p132, p133, p134, p135; byte p136, p137, p138, p139, p140, p141, p142, p143; byte p144, p145, p146, p147, p148, p149, p150, p151; byte p152, p153, p154, p155, p156, p157, p158, p159; byte p160, p161, p162, p163, p164, p165, p166, p167; byte p168, p169, p170, p171, p172, p173, p174, p175; byte p176, p177, p178, p179, p180, p181, p182, p183; byte p184, p185, p186, p187, p188, p189, p190, p191; byte p192, p193, p194, p195, p196, p197, p198, p199; byte p200, p201, p202, p203, p204, p205, p206, p207; byte p208, p209, p210, p211, p212, p213, p214, p215; byte p216, p217, p218, p219, p220, p221, p222, p223; byte p224, p225, p226, p227, p228, p229, p230, p231; byte p232, p233, p234, p235, p236, p237, p238, p239; } /** Enforces a memory layout to avoid false sharing by padding the tail node. */ @SuppressWarnings("deprecation") abstract static class HeadAndTailRef extends PadHeadAndTail { static final long TAIL_OFFSET = com.github.benmanes.caffeine.base.UnsafeAccess.objectFieldOffset( HeadAndTailRef.class, "tail"); volatile SingleConsumerQueue.@Nullable Node tail; void lazySetTail(SingleConsumerQueue.Node next) { com.github.benmanes.caffeine.base.UnsafeAccess.UNSAFE.putOrderedObject( this, TAIL_OFFSET, next); } boolean casTail(SingleConsumerQueue.Node expect, SingleConsumerQueue.Node update) { return com.github.benmanes.caffeine.base.UnsafeAccess.UNSAFE.compareAndSwapObject( this, TAIL_OFFSET, expect, update); } } }