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

com.caucho.util.RingQueue Maven / Gradle / Ivy

/*
 * Copyright (c) 1998-2018 Caucho Technology -- all rights reserved
 *
 * This file is part of Resin(R) Open Source
 *
 * Each copy or derived work must preserve the copyright notice and this
 * notice unmodified.
 *
 * Resin Open Source 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 2 of the License, or
 * (at your option) any later version.
 *
 * Resin Open Source 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, or any warranty
 * of NON-INFRINGEMENT.  See the GNU General Public License for more
 * details.
 *
 * You should have received a copy of the GNU General Public License
 * along with Resin Open Source; if not, write to the
 *
 *   Free Software Foundation, Inc.
 *   59 Temple Place, Suite 330
 *   Boston, MA 02111-1307  USA
 *
 * @author Scott Ferguson
 */

package com.caucho.util;

import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.logging.Level;
import java.util.logging.Logger;

public class RingQueue {
  private static final Logger log
    = Logger.getLogger(RingQueue.class.getName());
  
  private final T [] _ring;
  
  private final AtomicInteger _headAlloc = new AtomicInteger();
  private final AtomicInteger _head = new AtomicInteger();
  
  private final AtomicInteger _tailAlloc = new AtomicInteger();
  private final AtomicInteger _tail = new AtomicInteger();
  
  private final int _mask;
  private final int _updateSize;
  
  private final AtomicBoolean _isWait = new AtomicBoolean();
  
  public RingQueue(int capacity, RingItemFactory itemFactory)
  {
    int size = 8;
    
    while (size < capacity) {
      size *= 2;
    }
    
    _ring = (T[]) new RingItem[size];
    _mask = size - 1;
    _updateSize = size >> 2;
    
    for (int i = 0; i < _ring.length; i++) {
      _ring[i] = itemFactory.createItem(i);
    }
  }
  
  public boolean isEmpty()
  {
    return _head.get() == _tail.get();
  }
  
  public int getSize()
  {
    int head = _head.get();
    int tail = _tail.get();
    
    return (_ring.length + head - tail) & _mask;
  }
  
  public int getHead()
  {
    return _head.get();
  }
  
  public int getHeadAlloc()
  {
    return _headAlloc.get();
  }
  
  public int getTail()
  {
    return _tail.get();
  }
  
  public int getTailAlloc()
  {
    return _tailAlloc.get();
  }
  
  public final T beginOffer(boolean isWait)
  {
    // offer must allow only one thread to succeed, because 
    // completeOffer must be single threaded. A multi-threaded
    // completeOffer creates too much contention and spinning.
    
    final AtomicInteger headAllocRef = _headAlloc;
    final AtomicInteger tailRef = _tail;
    final int mask = _mask;
    int retry = 256;
    
    while (true) {
      int headAlloc = headAllocRef.get();
      int tail = tailRef.get();
          
      int nextHeadAlloc = (headAlloc + 1) & mask;
      
      if (nextHeadAlloc == tail) {
        if (! isWait) {
          return null;
        }
        else {
          waitForAvailable(headAlloc, tail);
        }
      }
      else if (headAllocRef.compareAndSet(headAlloc, nextHeadAlloc)) {
        return _ring[headAlloc];
      }
    }
  }
  
  public final void completeOffer(final T item)
  {
    item.setRingValue();
    
    completeOffer(item.getIndex());
  }
  
  private void completeOffer(final int index)
  {
    final AtomicInteger headRef = _head;
    final int mask = _mask;
    
    int nextHead = (index + 1) & mask;
    
    if (headRef.compareAndSet(index, nextHead)) {
      return;
    }
    
    final AtomicInteger headAllocRef = _headAlloc;
    final T []ring = _ring;

    // limit retry in high-contention situation, since we've acked the entry
    // int retryCount = 1024 + ((index & 0xf) << 8);
    
    while (true) {
      int head = headRef.get();
      int headAlloc = headAllocRef.get();

      if (head == headAlloc) {
        return;
      }
      
      if (ring[head].isRingValue()) {
        nextHead = (head + 1) & mask;
        
        if (headRef.compareAndSet(head, nextHead)) {
          return;
        }
      }
      
      /*
      if (((head + ring.length - index) & mask) < _updateSize) {
        // someone else acked us
        return;
      }
      */
    }
  }
 
  public final T beginPoll()
  {
    int nextTail;
    int tailAlloc;
    
    final AtomicInteger tailAllocRef = _tailAlloc;
    final AtomicInteger headRef = _head;
    final int mask = _mask;
    
    while (true) {
      tailAlloc = tailAllocRef.get();
      int head = headRef.get();
      
      if (head == tailAlloc) {
        return null;
      }
      
      nextTail = (tailAlloc + 1) & mask;
      if (tailAllocRef.compareAndSet(tailAlloc, nextTail)) {
        return _ring[tailAlloc];
      }
    }
  }
  
  public final void completePoll(final T item)
  {
    item.clearRingValue();
    
    completePoll(item.getIndex());
  }
    
  private void completePoll(final int index)
  {
    final AtomicInteger tailRef = _tail;
    final int mask = _mask;
    
    int nextTail = (index + 1) & mask;
    
    if (tailRef.compareAndSet(index, nextTail)) {
      wakeAvailable();
      return;
    }
    
    final AtomicInteger tailAllocRef = _tailAlloc;
    final T []ring = _ring;
    // int ringLength = ring.length;
    // int halfSize = _halfSize;
    
    // limit retry in high-contention situation
    // int retryCount = 1024 + ((index & 0xf) << 8);

    while (true) {
      final int tail = tailRef.get();
      final int tailAlloc = tailAllocRef.get();
      
      if (tail == tailAlloc) {
        break;
      }
      
      if (! ring[tail].isRingValue()) {
        nextTail = (tail + 1) & mask;
        
        if (tailRef.compareAndSet(tail, nextTail)) {
          break;
        }
      }

      /*
      if (((tail + ring.length - index) & mask) < _updateSize) {
        // someone else acked us
        break;
      }
      */
    }
    
    wakeAvailable();
  }
  
  private void waitForAvailable(int headAlloc, int tail)
  {
    _isWait.set(true);
    
    if (_headAlloc.get() == headAlloc && _tail.get() == tail) {
      synchronized (_isWait) {
        if (_headAlloc.get() == headAlloc
            && _tail.get() == tail
            && _isWait.get()) {
          try {
            _isWait.wait(100);
          } catch (Exception e) {
            log.log(Level.FINER, e.toString(), e);
          }
        }
      }
    }
  }
  
  private boolean isFull()
  {
    int head = _head.get();
    int tail = _tail.get();
    
    int nextHead = (head + 1) & _mask;
    
    return nextHead == tail;
  }
  
  private void wakeAvailable()
  {
    if (_isWait.compareAndSet(true, false)) {
      synchronized (_isWait) {
        _isWait.notifyAll();
      }
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy