You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

333 lines
9.0 KiB

/*
Based on CompactHashSet Copyright 2011 Ontopia Project
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.bukkit.craftbukkit.util;
import java.util.Iterator;
import java.util.ConcurrentModificationException;
import java.util.NoSuchElementException;
public class LongHashSet {
private final static int INITIAL_SIZE = 3;
private final static double LOAD_FACTOR = 0.75;
private final static long FREE = 0;
private final static long REMOVED = Long.MIN_VALUE;
private int freeEntries;
private int elements;
private long[] values;
private int modCount;
private org.spigotmc.FlatMap<Boolean> flat = new org.spigotmc.FlatMap<Boolean>(); // Spigot
public LongHashSet() {
this(INITIAL_SIZE);
}
public LongHashSet(int size) {
values = new long[(size==0 ? 1 : size)];
elements = 0;
freeEntries = values.length;
modCount = 0;
}
public Iterator iterator() {
return new Itr();
}
public int size() {
return elements;
}
public boolean isEmpty() {
return elements == 0;
}
public boolean contains(int msw, int lsw) {
// Spigot start
if ( elements == 0 )
{
return false;
}
if ( flat.contains( msw, lsw ) )
{
return true;
}
// Spigot end
return contains(LongHash.toLong(msw, lsw));
}
public boolean contains(long value) {
// Spigot start
if ( elements == 0 )
{
return false;
}
if ( flat.contains( value ) )
{
return true;
}
// Spigot end
int hash = hash(value);
int index = (hash & 0x7FFFFFFF) % values.length;
int offset = 1;
// search for the object (continue while !null and !this object)
while(values[index] != FREE && !(hash(values[index]) == hash && values[index] == value)) {
index = ((index + offset) & 0x7FFFFFFF) % values.length;
offset = offset * 2 + 1;
if (offset == -1) {
offset = 2;
}
}
return values[index] != FREE;
}
public boolean add(int msw, int lsw) {
return add(LongHash.toLong(msw, lsw));
}
public boolean add(long value) {
flat.put( value, Boolean.TRUE ); // Spigot
int hash = hash(value);
int index = (hash & 0x7FFFFFFF) % values.length;
int offset = 1;
int deletedix = -1;
// search for the object (continue while !null and !this object)
while(values[index] != FREE && !(hash(values[index]) == hash && values[index] == value)) {
// if there's a deleted object here we can put this object here,
// provided it's not in here somewhere else already
if (values[index] == REMOVED) {
deletedix = index;
}
index = ((index + offset) & 0x7FFFFFFF) % values.length;
offset = offset * 2 + 1;
if (offset == -1) {
offset = 2;
}
}
if (values[index] == FREE) {
if (deletedix != -1) { // reusing a deleted cell
index = deletedix;
} else {
freeEntries--;
}
modCount++;
elements++;
values[index] = value;
if (1 - (freeEntries / (double) values.length) > LOAD_FACTOR) {
rehash();
}
return true;
} else {
return false;
}
}
public void remove(int msw, int lsw) {
// Spigot start
flat.remove(msw, lsw);
remove0(LongHash.toLong(msw, lsw));
}
public boolean remove(long value) {
flat.remove(value);
return remove0(value);
}
private boolean remove0(long value) {
// Spigot end
int hash = hash(value);
int index = (hash & 0x7FFFFFFF) % values.length;
int offset = 1;
// search for the object (continue while !null and !this object)
while(values[index] != FREE && !(hash(values[index]) == hash && values[index] == value)) {
index = ((index + offset) & 0x7FFFFFFF) % values.length;
offset = offset * 2 + 1;
if (offset == -1) {
offset = 2;
}
}
if (values[index] != FREE) {
values[index] = REMOVED;
modCount++;
elements--;
return true;
} else {
return false;
}
}
public void clear() {
elements = 0;
for (int ix = 0; ix < values.length; ix++) {
values[ix] = FREE;
}
freeEntries = values.length;
modCount++;
flat = new org.spigotmc.FlatMap<Boolean>();
}
public long[] toArray() {
long[] result = new long[elements];
long[] values = Java15Compat.Arrays_copyOf(this.values, this.values.length);
int pos = 0;
for (long value : values) {
if (value != FREE && value != REMOVED) {
result[pos++] = value;
}
}
return result;
}
public long popFirst() {
for (long value : values) {
if (value != FREE && value != REMOVED) {
remove(value);
return value;
}
}
return 0;
}
public long[] popAll() {
long[] ret = toArray();
clear();
return ret;
}
// This method copied from Murmur3, written by Austin Appleby released under Public Domain
private int hash(long value) {
value ^= value >>> 33;
value *= 0xff51afd7ed558ccdL;
value ^= value >>> 33;
value *= 0xc4ceb9fe1a85ec53L;
value ^= value >>> 33;
return (int) value;
}
private void rehash() {
int gargagecells = values.length - (elements + freeEntries);
if (gargagecells / (double) values.length > 0.05) {
rehash(values.length);
} else {
rehash(values.length * 2 + 1);
}
}
private void rehash(int newCapacity) {
long[] newValues = new long[newCapacity];
for (long value : values) {
if (value == FREE || value == REMOVED) {
continue;
}
int hash = hash(value);
int index = (hash & 0x7FFFFFFF) % newCapacity;
int offset = 1;
// search for the object
while (newValues[index] != FREE) {
index = ((index + offset) & 0x7FFFFFFF) % newCapacity;
offset = offset * 2 + 1;
if (offset == -1) {
offset = 2;
}
}
newValues[index] = value;
}
values = newValues;
freeEntries = values.length - elements;
}
private class Itr implements Iterator {
private int index;
private int lastReturned = -1;
private int expectedModCount;
public Itr() {
for (index = 0; index < values.length && (values[index] == FREE || values[index] == REMOVED); index++) {
// This is just to drive the index forward to the first valid entry
}
expectedModCount = modCount;
}
public boolean hasNext() {
return index != values.length;
}
public Long next() {
if (modCount != expectedModCount) {
throw new ConcurrentModificationException();
}
int length = values.length;
if (index >= length) {
lastReturned = -2;
throw new NoSuchElementException();
}
lastReturned = index;
for (index += 1; index < length && (values[index] == FREE || values[index] == REMOVED); index++) {
// This is just to drive the index forward to the next valid entry
}
if (values[lastReturned] == FREE) {
return FREE;
} else {
return values[lastReturned];
}
}
public void remove() {
if (modCount != expectedModCount) {
throw new ConcurrentModificationException();
}
if (lastReturned == -1 || lastReturned == -2) {
throw new IllegalStateException();
}
if (values[lastReturned] != FREE && values[lastReturned] != REMOVED) {
values[lastReturned] = REMOVED;
elements--;
modCount++;
expectedModCount = modCount;
}
}
}
}