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

org.apache.hadoop.hdfs.util.FoldedTreeSet Maven / Gradle / Ivy

There is a newer version: 3.4.0
Show newest version
/**
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you 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.apache.hadoop.hdfs.util;

import org.apache.hadoop.util.Time;

import java.util.Arrays;
import java.util.Collection;
import java.util.Comparator;
import java.util.ConcurrentModificationException;
import java.util.Iterator;
import java.util.NoSuchElementException;
import java.util.Objects;
import java.util.SortedSet;

/**
 * A memory efficient implementation of RBTree. Instead of having a Node for
 * each entry each node contains an array holding 64 entries.
 *
 * Based on the Apache Harmony folded TreeMap.
 *
 * @param  Entry type
 */
public class FoldedTreeSet implements SortedSet {

  private static final boolean RED = true;
  private static final boolean BLACK = false;

  private final Comparator comparator;
  private Node root;
  private int size;
  private int nodeCount;
  private int modCount;
  private Node cachedNode;

  /**
   * Internal tree node that holds a sorted array of entries.
   *
   * @param  type of the elements
   */
  private static class Node {

    private static final int NODE_SIZE = 64;

    // Tree structure
    private Node parent, left, right;
    private boolean color;
    private final E[] entries;
    private int leftIndex = 0, rightIndex = -1;
    private int size = 0;
    // List for fast ordered iteration
    private Node prev, next;

    @SuppressWarnings("unchecked")
    public Node() {
      entries = (E[]) new Object[NODE_SIZE];
    }

    public boolean isRed() {
      return color == RED;
    }

    public boolean isBlack() {
      return color == BLACK;
    }

    public Node getLeftMostNode() {
      Node node = this;
      while (node.left != null) {
        node = node.left;
      }
      return node;
    }

    public Node getRightMostNode() {
      Node node = this;
      while (node.right != null) {
        node = node.right;
      }
      return node;
    }

    public void addEntryLeft(E entry) {
      assert rightIndex < entries.length;
      assert !isFull();

      if (leftIndex == 0) {
        rightIndex++;
        // Shift entries right/up
        System.arraycopy(entries, 0, entries, 1, size);
      } else {
        leftIndex--;
      }
      size++;
      entries[leftIndex] = entry;
    }

    public void addEntryRight(E entry) {
      assert !isFull();

      if (rightIndex == NODE_SIZE - 1) {
        assert leftIndex > 0;
        // Shift entries left/down
        System.arraycopy(entries, leftIndex, entries, --leftIndex, size);
      } else {
        rightIndex++;
      }
      size++;
      entries[rightIndex] = entry;
    }

    public void addEntryAt(E entry, int index) {
      assert !isFull();

      if (leftIndex == 0 || ((rightIndex != Node.NODE_SIZE - 1)
                             && (rightIndex - index <= index - leftIndex))) {
        rightIndex++;
        System.arraycopy(entries, index,
                         entries, index + 1, rightIndex - index);
        entries[index] = entry;
      } else {
        int newLeftIndex = leftIndex - 1;
        System.arraycopy(entries, leftIndex,
                         entries, newLeftIndex, index - leftIndex);
        leftIndex = newLeftIndex;
        entries[index - 1] = entry;
      }
      size++;
    }

    public void addEntriesLeft(Node from) {
      leftIndex -= from.size;
      size += from.size;
      System.arraycopy(from.entries, from.leftIndex,
                       entries, leftIndex, from.size);
    }

    public void addEntriesRight(Node from) {
      System.arraycopy(from.entries, from.leftIndex,
                       entries, rightIndex + 1, from.size);
      size += from.size;
      rightIndex += from.size;
    }

    public E insertEntrySlideLeft(E entry, int index) {
      E pushedEntry = entries[0];
      System.arraycopy(entries, 1, entries, 0, index - 1);
      entries[index - 1] = entry;
      return pushedEntry;
    }

    public E insertEntrySlideRight(E entry, int index) {
      E movedEntry = entries[rightIndex];
      System.arraycopy(entries, index, entries, index + 1, rightIndex - index);
      entries[index] = entry;
      return movedEntry;
    }

    public E removeEntryLeft() {
      assert !isEmpty();
      E entry = entries[leftIndex];
      entries[leftIndex] = null;
      leftIndex++;
      size--;
      return entry;
    }

    public E removeEntryRight() {
      assert !isEmpty();
      E entry = entries[rightIndex];
      entries[rightIndex] = null;
      rightIndex--;
      size--;
      return entry;
    }

    public E removeEntryAt(int index) {
      assert !isEmpty();

      E entry = entries[index];
      int rightSize = rightIndex - index;
      int leftSize = index - leftIndex;
      if (rightSize <= leftSize) {
        System.arraycopy(entries, index + 1, entries, index, rightSize);
        entries[rightIndex] = null;
        rightIndex--;
      } else {
        System.arraycopy(entries, leftIndex, entries, leftIndex + 1, leftSize);
        entries[leftIndex] = null;
        leftIndex++;
      }
      size--;
      return entry;
    }

    public boolean isFull() {
      return size == NODE_SIZE;
    }

    public boolean isEmpty() {
      return size == 0;
    }

    public void clear() {
      if (leftIndex < rightIndex) {
        Arrays.fill(entries, leftIndex, rightIndex + 1, null);
      }
      size = 0;
      leftIndex = 0;
      rightIndex = -1;
      prev = null;
      next = null;
      parent = null;
      left = null;
      right = null;
      color = BLACK;
    }
  }

  private static final class TreeSetIterator implements Iterator {

    private final FoldedTreeSet tree;
    private int iteratorModCount;
    private Node node;
    private int index;
    private E lastEntry;
    private int lastIndex;
    private Node lastNode;

    private TreeSetIterator(FoldedTreeSet tree) {
      this.tree = tree;
      this.iteratorModCount = tree.modCount;
      if (!tree.isEmpty()) {
        this.node = tree.root.getLeftMostNode();
        this.index = this.node.leftIndex;
      }
    }

    @Override
    public boolean hasNext() {
      checkForModification();
      return node != null;
    }

    @Override
    public E next() {
      if (hasNext()) {
        lastEntry = node.entries[index];
        lastIndex = index;
        lastNode = node;
        if (++index > node.rightIndex) {
          node = node.next;
          if (node != null) {
            index = node.leftIndex;
          }
        }
        return lastEntry;
      } else {
        throw new NoSuchElementException("Iterator exhausted");
      }
    }

    @Override
    public void remove() {
      if (lastEntry == null) {
        throw new IllegalStateException("No current element");
      }
      checkForModification();
      if (lastNode.size == 1) {
        // Safe to remove lastNode, the iterator is on the next node
        tree.deleteNode(lastNode);
      } else if (lastNode.leftIndex == lastIndex) {
        // Safe to remove leftmost entry, the iterator is on the next index
        lastNode.removeEntryLeft();
      } else if (lastNode.rightIndex == lastIndex) {
        // Safe to remove the rightmost entry, the iterator is on the next node
        lastNode.removeEntryRight();
      } else {
        // Remove entry in the middle of the array
        assert node == lastNode;
        int oldRIndex = lastNode.rightIndex;
        lastNode.removeEntryAt(lastIndex);
        if (oldRIndex > lastNode.rightIndex) {
          // Entries moved to the left in the array so index must be reset
          index = lastIndex;
        }
      }
      lastEntry = null;
      iteratorModCount++;
      tree.modCount++;
      tree.size--;
    }

    private void checkForModification() {
      if (iteratorModCount != tree.modCount) {
        throw new ConcurrentModificationException("Tree has been modified "
                                                  + "outside of iterator");
      }
    }
  }

  /**
   * Create a new TreeSet that uses the natural ordering of objects. The element
   * type must implement Comparable.
   */
  public FoldedTreeSet() {
    this(null);
  }

  /**
   * Create a new TreeSet that orders the elements using the supplied
   * Comparator.
   *
   * @param comparator Comparator able to compare elements of type E
   */
  public FoldedTreeSet(Comparator comparator) {
    this.comparator = comparator;
  }

  private Node cachedOrNewNode(E entry) {
    Node node = (cachedNode != null) ? cachedNode : new Node();
    cachedNode = null;
    nodeCount++;
    // Since BlockIDs are always increasing for new blocks it is best to
    // add values on the left side to enable quicker inserts on the right
    node.addEntryLeft(entry);
    return node;
  }

  private void cacheAndClear(Node node) {
    if (cachedNode == null) {
      node.clear();
      cachedNode = node;
    }
  }

  @Override
  public Comparator comparator() {
    return comparator;
  }

  @Override
  public SortedSet subSet(E fromElement, E toElement) {
    throw new UnsupportedOperationException("Not supported yet.");
  }

  @Override
  public SortedSet headSet(E toElement) {
    throw new UnsupportedOperationException("Not supported yet.");
  }

  @Override
  public SortedSet tailSet(E fromElement) {
    throw new UnsupportedOperationException("Not supported yet.");
  }

  @Override
  public E first() {
    if (!isEmpty()) {
      Node node = root.getLeftMostNode();
      return node.entries[node.leftIndex];
    }
    return null;
  }

  @Override
  public E last() {
    if (!isEmpty()) {
      Node node = root.getRightMostNode();
      return node.entries[node.rightIndex];
    }
    return null;
  }

  @Override
  public int size() {
    return size;
  }

  @Override
  public boolean isEmpty() {
    return root == null;
  }

  /**
   * Lookup and return a stored object using a user provided comparator.
   *
   * @param obj Lookup key
   * @param cmp User provided Comparator. The comparator should expect that the
   *            proved obj will always be the first method parameter and any
   *            stored object will be the second parameter.
   *
   * @return A matching stored object or null if non is found
   */
  public E get(Object obj, Comparator cmp) {
    Objects.requireNonNull(obj);

    Node node = root;
    while (node != null) {
      E[] entries = node.entries;

      int leftIndex = node.leftIndex;
      int result = compare(obj, entries[leftIndex], cmp);
      if (result < 0) {
        node = node.left;
      } else if (result == 0) {
        return entries[leftIndex];
      } else {
        int rightIndex = node.rightIndex;
        if (leftIndex != rightIndex) {
          result = compare(obj, entries[rightIndex], cmp);
        }
        if (result == 0) {
          return entries[rightIndex];
        } else if (result > 0) {
          node = node.right;
        } else {
          int low = leftIndex + 1;
          int high = rightIndex - 1;
          while (low <= high) {
            int mid = (low + high) >>> 1;
            result = compare(obj, entries[mid], cmp);
            if (result > 0) {
              low = mid + 1;
            } else if (result < 0) {
              high = mid - 1;
            } else {
              return entries[mid];
            }
          }
          return null;
        }
      }
    }
    return null;
  }

  /**
   * Lookup and return a stored object.
   *
   * @param entry Lookup entry
   *
   * @return A matching stored object or null if non is found
   */
  public E get(E entry) {
    return get(entry, comparator);
  }

  @Override
  @SuppressWarnings("unchecked")
  public boolean contains(Object obj) {
    return get((E) obj) != null;
  }

  @SuppressWarnings({"unchecked", "rawtypes"})
  private static int compare(Object lookup, Object stored, Comparator cmp) {
    return cmp != null
           ? cmp.compare(lookup, stored)
           : ((Comparable) lookup).compareTo(stored);
  }

  @Override
  public Iterator iterator() {
    return new TreeSetIterator<>(this);
  }

  @Override
  public Object[] toArray() {
    Object[] objects = new Object[size];
    if (!isEmpty()) {
      int pos = 0;
      for (Node node = root.getLeftMostNode(); node != null;
          pos += node.size, node = node.next) {
        System.arraycopy(node.entries, node.leftIndex, objects, pos, node.size);
      }
    }
    return objects;
  }

  @Override
  @SuppressWarnings("unchecked")
  public  T[] toArray(T[] a) {
    T[] r = a.length >= size ? a
            : (T[]) java.lang.reflect.Array
        .newInstance(a.getClass().getComponentType(), size);
    if (!isEmpty()) {
      Node node = root.getLeftMostNode();
      int pos = 0;
      while (node != null) {
        System.arraycopy(node.entries, node.leftIndex, r, pos, node.size);
        pos += node.size;
        node = node.next;
      }
      if (r.length > pos) {
        r[pos] = null;
      }
    } else if (a.length > 0) {
      a[0] = null;
    }
    return r;
  }

  /**
   * Add or replace an entry in the TreeSet.
   *
   * @param entry Entry to add or replace/update.
   *
   * @return the previous entry, or null if this set did not already contain the
   *         specified entry
   */
  public E addOrReplace(E entry) {
    return add(entry, true);
  }

  @Override
  public boolean add(E entry) {
    return add(entry, false) == null;
  }

  /**
   * Internal add method to add a entry to the set.
   *
   * @param entry   Entry to add
   * @param replace Should the entry replace an old entry which is equal to the
   *                new entry
   *
   * @return null if entry added and didn't exist or the previous value (which
   *         might not have been overwritten depending on the replace parameter)
   */
  private E add(E entry, boolean replace) {
    Objects.requireNonNull(entry);

    // Empty tree
    if (isEmpty()) {
      root = cachedOrNewNode(entry);
      size = 1;
      modCount++;
      return null;
    }

    // Compare right entry first since inserts of comperatively larger entries
    // is more likely to be inserted. BlockID is always increasing in HDFS.
    Node node = root;
    Node prevNode = null;
    int result = 0;
    while (node != null) {
      prevNode = node;
      E[] entries = node.entries;
      int rightIndex = node.rightIndex;
      result = compare(entry, entries[rightIndex], comparator);
      if (result > 0) {
        node = node.right;
      } else if (result == 0) {
        E prevEntry = entries[rightIndex];
        if (replace) {
          entries[rightIndex] = entry;
        }
        return prevEntry;
      } else {
        int leftIndex = node.leftIndex;
        if (leftIndex != rightIndex) {
          result = compare(entry, entries[leftIndex], comparator);
        }
        if (result < 0) {
          node = node.left;
        } else if (result == 0) {
          E prevEntry = entries[leftIndex];
          if (replace) {
            entries[leftIndex] = entry;
          }
          return prevEntry;
        } else {
          // Insert in this node
          int low = leftIndex + 1, high = rightIndex - 1;
          while (low <= high) {
            int mid = (low + high) >>> 1;
            result = compare(entry, entries[mid], comparator);
            if (result > 0) {
              low = mid + 1;
            } else if (result == 0) {
              E prevEntry = entries[mid];
              if (replace) {
                entries[mid] = entry;
              }
              return prevEntry;
            } else {
              high = mid - 1;
            }
          }
          addElementInNode(node, entry, low);
          return null;
        }
      }
    }

    assert prevNode != null;
    size++;
    modCount++;
    if (!prevNode.isFull()) {
      // The previous node still has space
      if (result < 0) {
        prevNode.addEntryLeft(entry);
      } else {
        prevNode.addEntryRight(entry);
      }
    } else if (result < 0) {
      // The previous node is full, add to adjencent node or a new node
      if (prevNode.prev != null && !prevNode.prev.isFull()) {
        prevNode.prev.addEntryRight(entry);
      } else {
        attachNodeLeft(prevNode, cachedOrNewNode(entry));
      }
    } else if (prevNode.next != null && !prevNode.next.isFull()) {
      prevNode.next.addEntryLeft(entry);
    } else {
      attachNodeRight(prevNode, cachedOrNewNode(entry));
    }
    return null;
  }

  /**
   * Insert an entry last in the sorted tree. The entry must be the considered
   * larger than the currently largest entry in the set when doing
   * current.compareTo(entry), if entry is not the largest entry the method will
   * fall back on the regular add method.
   *
   * @param entry entry to add
   *
   * @return True if added, false if already existed in the set
   */
  public boolean addSortedLast(E entry) {

    if (isEmpty()) {
      root = cachedOrNewNode(entry);
      size = 1;
      modCount++;
      return true;
    } else {
      Node node = root.getRightMostNode();
      if (compare(node.entries[node.rightIndex], entry, comparator) < 0) {
        size++;
        modCount++;
        if (!node.isFull()) {
          node.addEntryRight(entry);
        } else {
          attachNodeRight(node, cachedOrNewNode(entry));
        }
        return true;
      }
    }

    // Fallback on normal add if entry is unsorted
    return add(entry);
  }

  private void addElementInNode(Node node, E entry, int index) {
    size++;
    modCount++;

    if (!node.isFull()) {
      node.addEntryAt(entry, index);
    } else {
      // Node is full, insert and push old entry
      Node prev = node.prev;
      Node next = node.next;
      if (prev == null) {
        // First check if we have space in the the next node
        if (next != null && !next.isFull()) {
          E movedEntry = node.insertEntrySlideRight(entry, index);
          next.addEntryLeft(movedEntry);
        } else {
          // Since prev is null the left child must be null
          assert node.left == null;
          E movedEntry = node.insertEntrySlideLeft(entry, index);
          Node newNode = cachedOrNewNode(movedEntry);
          attachNodeLeft(node, newNode);
        }
      } else if (!prev.isFull()) {
        // Prev has space
        E movedEntry = node.insertEntrySlideLeft(entry, index);
        prev.addEntryRight(movedEntry);
      } else if (next == null) {
        // Since next is null the right child must be null
        assert node.right == null;
        E movedEntry = node.insertEntrySlideRight(entry, index);
        Node newNode = cachedOrNewNode(movedEntry);
        attachNodeRight(node, newNode);
      } else if (!next.isFull()) {
        // Next has space
        E movedEntry = node.insertEntrySlideRight(entry, index);
        next.addEntryLeft(movedEntry);
      } else {
        // Both prev and next nodes exist and are full
        E movedEntry = node.insertEntrySlideRight(entry, index);
        Node newNode = cachedOrNewNode(movedEntry);
        if (node.right == null) {
          attachNodeRight(node, newNode);
        } else {
          // Since our right node exist,
          // the left node of our next node must be empty
          assert next.left == null;
          attachNodeLeft(next, newNode);
        }
      }
    }
  }

  private void attachNodeLeft(Node node, Node newNode) {
    newNode.parent = node;
    node.left = newNode;

    newNode.next = node;
    newNode.prev = node.prev;
    if (newNode.prev != null) {
      newNode.prev.next = newNode;
    }
    node.prev = newNode;
    balanceInsert(newNode);
  }

  private void attachNodeRight(Node node, Node newNode) {
    newNode.parent = node;
    node.right = newNode;

    newNode.prev = node;
    newNode.next = node.next;
    if (newNode.next != null) {
      newNode.next.prev = newNode;
    }
    node.next = newNode;
    balanceInsert(newNode);
  }

  /**
   * Balance the RB Tree after insert.
   *
   * @param node Added node
   */
  private void balanceInsert(Node node) {
    node.color = RED;

    while (node != root && node.parent.isRed()) {
      if (node.parent == node.parent.parent.left) {
        Node uncle = node.parent.parent.right;
        if (uncle != null && uncle.isRed()) {
          node.parent.color = BLACK;
          uncle.color = BLACK;
          node.parent.parent.color = RED;
          node = node.parent.parent;
        } else {
          if (node == node.parent.right) {
            node = node.parent;
            rotateLeft(node);
          }
          node.parent.color = BLACK;
          node.parent.parent.color = RED;
          rotateRight(node.parent.parent);
        }
      } else {
        Node uncle = node.parent.parent.left;
        if (uncle != null && uncle.isRed()) {
          node.parent.color = BLACK;
          uncle.color = BLACK;
          node.parent.parent.color = RED;
          node = node.parent.parent;
        } else {
          if (node == node.parent.left) {
            node = node.parent;
            rotateRight(node);
          }
          node.parent.color = BLACK;
          node.parent.parent.color = RED;
          rotateLeft(node.parent.parent);
        }
      }
    }
    root.color = BLACK;
  }

  private void rotateRight(Node node) {
    Node pivot = node.left;
    node.left = pivot.right;
    if (pivot.right != null) {
      pivot.right.parent = node;
    }
    pivot.parent = node.parent;
    if (node.parent == null) {
      root = pivot;
    } else if (node == node.parent.right) {
      node.parent.right = pivot;
    } else {
      node.parent.left = pivot;
    }
    pivot.right = node;
    node.parent = pivot;
  }

  private void rotateLeft(Node node) {
    Node pivot = node.right;
    node.right = pivot.left;
    if (pivot.left != null) {
      pivot.left.parent = node;
    }
    pivot.parent = node.parent;
    if (node.parent == null) {
      root = pivot;
    } else if (node == node.parent.left) {
      node.parent.left = pivot;
    } else {
      node.parent.right = pivot;
    }
    pivot.left = node;
    node.parent = pivot;
  }

  /**
   * Remove object using a provided comparator, and return the removed entry.
   *
   * @param obj Lookup entry
   * @param cmp User provided Comparator. The comparator should expect that the
   *            proved obj will always be the first method parameter and any
   *            stored object will be the second parameter.
   *
   * @return The removed entry or null if not found
   */
  public E removeAndGet(Object obj, Comparator cmp) {
    Objects.requireNonNull(obj);

    if (!isEmpty()) {
      Node node = root;
      while (node != null) {
        E[] entries = node.entries;
        int leftIndex = node.leftIndex;
        int result = compare(obj, entries[leftIndex], cmp);
        if (result < 0) {
          node = node.left;
        } else if (result == 0) {
          return removeElementLeft(node);
        } else {
          int rightIndex = node.rightIndex;
          if (leftIndex != rightIndex) {
            result = compare(obj, entries[rightIndex], cmp);
          }
          if (result == 0) {
            return removeElementRight(node);
          } else if (result > 0) {
            node = node.right;
          } else {
            int low = leftIndex + 1, high = rightIndex - 1;
            while (low <= high) {
              int mid = (low + high) >>> 1;
              result = compare(obj, entries[mid], cmp);
              if (result > 0) {
                low = mid + 1;
              } else if (result == 0) {
                return removeElementAt(node, mid);
              } else {
                high = mid - 1;
              }
            }
            return null;
          }
        }
      }
    }
    return null;
  }

  /**
   * Remove object and return the removed entry.
   *
   * @param obj Lookup entry
   *
   * @return The removed entry or null if not found
   */
  public E removeAndGet(Object obj) {
    return removeAndGet(obj, comparator);
  }

  /**
   * Remove object using a provided comparator.
   *
   * @param obj Lookup entry
   * @param cmp User provided Comparator. The comparator should expect that the
   *            proved obj will always be the first method parameter and any
   *            stored object will be the second parameter.
   *
   * @return True if found and removed, else false
   */
  public boolean remove(Object obj, Comparator cmp) {
    return removeAndGet(obj, cmp) != null;
  }

  @Override
  public boolean remove(Object obj) {
    return removeAndGet(obj, comparator) != null;
  }

  private E removeElementLeft(Node node) {
    modCount++;
    size--;
    E entry = node.removeEntryLeft();

    if (node.isEmpty()) {
      deleteNode(node);
    } else if (node.prev != null
               && (Node.NODE_SIZE - 1 - node.prev.rightIndex) >= node.size) {
      // Remaining entries fit in the prev node, move them and delete this node
      node.prev.addEntriesRight(node);
      deleteNode(node);
    } else if (node.next != null && node.next.leftIndex >= node.size) {
      // Remaining entries fit in the next node, move them and delete this node
      node.next.addEntriesLeft(node);
      deleteNode(node);
    } else if (node.prev != null && node.prev.size < node.leftIndex) {
      // Entries in prev node will fit in this node, move them and delete prev
      node.addEntriesLeft(node.prev);
      deleteNode(node.prev);
    }

    return entry;
  }

  private E removeElementRight(Node node) {
    modCount++;
    size--;
    E entry = node.removeEntryRight();

    if (node.isEmpty()) {
      deleteNode(node);
    } else if (node.prev != null
               && (Node.NODE_SIZE - 1 - node.prev.rightIndex) >= node.size) {
      // Remaining entries fit in the prev node, move them and delete this node
      node.prev.addEntriesRight(node);
      deleteNode(node);
    } else if (node.next != null && node.next.leftIndex >= node.size) {
      // Remaining entries fit in the next node, move them and delete this node
      node.next.addEntriesLeft(node);
      deleteNode(node);
    } else if (node.next != null
               && node.next.size < (Node.NODE_SIZE - 1 - node.rightIndex)) {
      // Entries in next node will fit in this node, move them and delete next
      node.addEntriesRight(node.next);
      deleteNode(node.next);
    }

    return entry;
  }

  private E removeElementAt(Node node, int index) {
    modCount++;
    size--;
    E entry = node.removeEntryAt(index);

    if (node.prev != null
        && (Node.NODE_SIZE - 1 - node.prev.rightIndex) >= node.size) {
      // Remaining entries fit in the prev node, move them and delete this node
      node.prev.addEntriesRight(node);
      deleteNode(node);
    } else if (node.next != null && (node.next.leftIndex) >= node.size) {
      // Remaining entries fit in the next node, move them and delete this node
      node.next.addEntriesLeft(node);
      deleteNode(node);
    } else if (node.prev != null && node.prev.size < node.leftIndex) {
      // Entries in prev node will fit in this node, move them and delete prev
      node.addEntriesLeft(node.prev);
      deleteNode(node.prev);
    } else if (node.next != null
               && node.next.size < (Node.NODE_SIZE - 1 - node.rightIndex)) {
      // Entries in next node will fit in this node, move them and delete next
      node.addEntriesRight(node.next);
      deleteNode(node.next);
    }

    return entry;
  }

  /**
   * Delete the node and ensure the tree is balanced.
   *
   * @param node node to delete
   */
  private void deleteNode(final Node node) {
    if (node.right == null) {
      if (node.left != null) {
        attachToParent(node, node.left);
      } else {
        attachNullToParent(node);
      }
    } else if (node.left == null) {
      attachToParent(node, node.right);
    } else {
      // node.left != null && node.right != null
      // node.next should replace node in tree
      // node.next != null guaranteed since node.left != null
      // node.next.left == null since node.next.prev is node
      // node.next.right may be null or non-null
      Node toMoveUp = node.next;
      if (toMoveUp.right == null) {
        attachNullToParent(toMoveUp);
      } else {
        attachToParent(toMoveUp, toMoveUp.right);
      }
      toMoveUp.left = node.left;
      if (toMoveUp.left != null) {
        toMoveUp.left.parent = toMoveUp;
      }
      toMoveUp.right = node.right;
      if (toMoveUp.right != null) {
        toMoveUp.right.parent = toMoveUp;
      }
      attachToParentNoBalance(node, toMoveUp);
      toMoveUp.color = node.color;
    }

    // Remove node from ordered list of nodes
    if (node.prev != null) {
      node.prev.next = node.next;
    }
    if (node.next != null) {
      node.next.prev = node.prev;
    }

    nodeCount--;
    cacheAndClear(node);
  }

  private void attachToParentNoBalance(Node toDelete, Node toConnect) {
    Node parent = toDelete.parent;
    toConnect.parent = parent;
    if (parent == null) {
      root = toConnect;
    } else if (toDelete == parent.left) {
      parent.left = toConnect;
    } else {
      parent.right = toConnect;
    }
  }

  private void attachToParent(Node toDelete, Node toConnect) {
    attachToParentNoBalance(toDelete, toConnect);
    if (toDelete.isBlack()) {
      balanceDelete(toConnect);
    }
  }

  private void attachNullToParent(Node toDelete) {
    Node parent = toDelete.parent;
    if (parent == null) {
      root = null;
    } else {
      if (toDelete == parent.left) {
        parent.left = null;
      } else {
        parent.right = null;
      }
      if (toDelete.isBlack()) {
        balanceDelete(parent);
      }
    }
  }

  /**
   * Balance tree after removing a node.
   *
   * @param node Node to balance after deleting another node
   */
  private void balanceDelete(Node node) {
    while (node != root && node.isBlack()) {
      if (node == node.parent.left) {
        Node sibling = node.parent.right;
        if (sibling == null) {
          node = node.parent;
          continue;
        }
        if (sibling.isRed()) {
          sibling.color = BLACK;
          node.parent.color = RED;
          rotateLeft(node.parent);
          sibling = node.parent.right;
          if (sibling == null) {
            node = node.parent;
            continue;
          }
        }
        if ((sibling.left == null || !sibling.left.isRed())
            && (sibling.right == null || !sibling.right.isRed())) {
          sibling.color = RED;
          node = node.parent;
        } else {
          if (sibling.right == null || !sibling.right.isRed()) {
            sibling.left.color = BLACK;
            sibling.color = RED;
            rotateRight(sibling);
            sibling = node.parent.right;
          }
          sibling.color = node.parent.color;
          node.parent.color = BLACK;
          sibling.right.color = BLACK;
          rotateLeft(node.parent);
          node = root;
        }
      } else {
        Node sibling = node.parent.left;
        if (sibling == null) {
          node = node.parent;
          continue;
        }
        if (sibling.isRed()) {
          sibling.color = BLACK;
          node.parent.color = RED;
          rotateRight(node.parent);
          sibling = node.parent.left;
          if (sibling == null) {
            node = node.parent;
            continue;
          }
        }
        if ((sibling.left == null || sibling.left.isBlack())
            && (sibling.right == null || sibling.right.isBlack())) {
          sibling.color = RED;
          node = node.parent;
        } else {
          if (sibling.left == null || sibling.left.isBlack()) {
            sibling.right.color = BLACK;
            sibling.color = RED;
            rotateLeft(sibling);
            sibling = node.parent.left;
          }
          sibling.color = node.parent.color;
          node.parent.color = BLACK;
          sibling.left.color = BLACK;
          rotateRight(node.parent);
          node = root;
        }
      }
    }
    node.color = BLACK;
  }

  @Override
  public boolean containsAll(Collection c) {
    for (Object entry : c) {
      if (!contains(entry)) {
        return false;
      }
    }
    return true;
  }

  @Override
  public boolean addAll(Collection c) {
    boolean modified = false;
    for (E entry : c) {
      if (add(entry)) {
        modified = true;
      }
    }
    return modified;
  }

  @Override
  public boolean retainAll(Collection c) {
    boolean modified = false;
    Iterator it = iterator();
    while (it.hasNext()) {
      if (!c.contains(it.next())) {
        it.remove();
        modified = true;
      }
    }
    return modified;
  }

  @Override
  public boolean removeAll(Collection c) {
    boolean modified = false;
    for (Object entry : c) {
      if (remove(entry)) {
        modified = true;
      }
    }
    return modified;
  }

  @Override
  public void clear() {
    modCount++;
    if (!isEmpty()) {
      size = 0;
      nodeCount = 0;
      cacheAndClear(root);
      root = null;
    }
  }

  /**
   * Returns the current size divided by the capacity of the tree. A value
   * between 0.0 and 1.0, where 1.0 means that every allocated node in the tree
   * is completely full.
   *
   * An empty set will return 1.0
   *
   * @return the fill ratio of the tree
   */
  public double fillRatio() {
    if (nodeCount > 1) {
      // Count the last node as completely full since it can't be compacted
      return (size + (Node.NODE_SIZE - root.getRightMostNode().size))
             / (double) (nodeCount * Node.NODE_SIZE);
    }
    return 1.0;
  }

  /**
   * Compact all the entries to use the fewest number of nodes in the tree.
   *
   * Having a compact tree minimize memory usage, but can cause inserts to get
   * slower due to new nodes needs to be allocated as there is no space in any
   * of the existing nodes anymore for entries added in the middle of the set.
   *
   * Useful to do to reduce memory consumption and if the tree is know to not
   * change after compaction or mainly added to at either extreme.
   *
   * @param timeout Maximum time to spend compacting the tree set in
   *                milliseconds.
   *
   * @return true if compaction completed, false if aborted
   */
  public boolean compact(long timeout) {

    if (!isEmpty()) {
      long start = Time.monotonicNow();
      Node node = root.getLeftMostNode();
      while (node != null) {
        if (node.prev != null && !node.prev.isFull()) {
          Node prev = node.prev;
          int count = Math.min(Node.NODE_SIZE - prev.size, node.size);
          System.arraycopy(node.entries, node.leftIndex,
                           prev.entries, prev.rightIndex + 1, count);
          node.leftIndex += count;
          node.size -= count;
          prev.rightIndex += count;
          prev.size += count;
        }
        if (node.isEmpty()) {
          Node temp = node.next;
          deleteNode(node);
          node = temp;
          continue;
        } else if (!node.isFull()) {
          if (node.leftIndex != 0) {
            System.arraycopy(node.entries, node.leftIndex,
                             node.entries, 0, node.size);
            Arrays.fill(node.entries, node.size, node.rightIndex + 1, null);
            node.leftIndex = 0;
            node.rightIndex = node.size - 1;
          }
        }
        node = node.next;

        if (Time.monotonicNow() - start > timeout) {
          return false;
        }
      }
    }

    return true;
  }
}