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

org.jctools.queues.IndexedQueueSizeUtil Maven / Gradle / Ivy

There is a newer version: 1.52.1
Show newest version
/*
 * 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 org.jctools.queues;

import org.jctools.util.InternalAPI;

/**
 * A note to maintainers on index assumptions: in a single threaded world it would seem intuitive to assume:
 * 
 * producerIndex >= consumerIndex
 * 
* As an invariant, but in a concurrent, long running settings all of the following need to be considered: *
    *
  • consumerIndex > producerIndex : due to counter overflow (unlikey with longs, but easy to reason) *
  • consumerIndex > producerIndex : due to consumer FastFlow like implementation discovering the * element before the counter is updated. *
  • producerIndex - consumerIndex < 0 : due to above. *
  • producerIndex - consumerIndex > Integer.MAX_VALUE : as linked buffers allow constructing queues * with more than Integer.MAX_VALUE elements. * *
*/ @InternalAPI public final class IndexedQueueSizeUtil { public static final int PLAIN_DIVISOR = 1; public static final int IGNORE_PARITY_DIVISOR = 2; public static int size(IndexedQueue iq, int divisor) { /* * It is possible for a thread to be interrupted or reschedule between the reads of the producer and * consumer indices. It is also for the indices to be updated in a `weakly` visible way. It follows that * the size value needs to be sanitized to match a valid range. */ long after = iq.lvConsumerIndex(); long size; while (true) { final long before = after; // pIndex read is "sandwiched" between 2 cIndex reads final long currentProducerIndex = iq.lvProducerIndex(); after = iq.lvConsumerIndex(); if (before == after) { size = (currentProducerIndex - after) / divisor; break; } } return sanitizedSize(iq.capacity(), size); } public static int sanitizedSize(int capacity, long size) { // Concurrent updates to cIndex and pIndex may lag behind other progress enablers (e.g. FastFlow), so we need // to check bounds [0,capacity] if (size < 0) { return 0; } if (capacity != MessagePassingQueue.UNBOUNDED_CAPACITY && size > capacity) { return capacity; } // Integer overflow is possible for the unbounded indexed queues. if (size > Integer.MAX_VALUE) { return Integer.MAX_VALUE; } return (int) size; } public static boolean isEmpty(IndexedQueue iq) { // Order matters! // Loading consumer before producer allows for producer increments after consumer index is read. // This ensures this method is conservative in it's estimate. Note that as this is an MPMC there is // nothing we can do to make this an exact method. return (iq.lvConsumerIndex() >= iq.lvProducerIndex()); } @InternalAPI public interface IndexedQueue { long lvConsumerIndex(); long lvProducerIndex(); int capacity(); } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy