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.

411 lines
11 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;
10 years ago
import java.util.Collection;
import java.util.Iterator;
import java.util.ConcurrentModificationException;
import java.util.NoSuchElementException;
10 years ago
import java.util.Set;
10 years ago
public class LongHashSet implements Set<Long> {
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;
}
10 years ago
@Override
public Iterator<Long> iterator() {
return new Itr();
}
10 years ago
@Override
public int size() {
return elements;
}
10 years ago
@Override
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;
}
}
10 years ago
@Override
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>();
}
10 years ago
public long[] toPrimitiveArray() {
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;
}
10 years ago
@Override
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;
}
@Override
public <T> T[] toArray(T[] arg0) {
throw new UnsupportedOperationException();
}
public long popFirst() {
for (long value : values) {
if (value != FREE && value != REMOVED) {
remove(value);
return value;
}
}
return 0;
}
public long[] popAll() {
10 years ago
long[] ret = toPrimitiveArray();
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;
}
10 years ago
private class Itr implements Iterator<Long> {
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;
}
10 years ago
@Override
public boolean hasNext() {
return index != values.length;
}
10 years ago
@Override
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];
}
}
10 years ago
@Override
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;
}
}
}
10 years ago
@Override
public boolean add(Long value) {
return add(value.longValue());
}
@Override
public boolean addAll(Collection<? extends Long> collection) {
boolean result = false;
for (Long value : collection) result |= add(value.longValue());
return result;
}
@Override
public boolean contains(Object o) {
return o instanceof Long ? contains(((Long) o).longValue()) : false;
}
@Override
public boolean containsAll(Collection<?> collection) {
for (Object value : collection) if (!contains(value)) return false;
return true;
}
@Override
public boolean remove(Object o) {
return o instanceof Long ? remove(((Long) o).longValue()) : false;
}
@Override
public boolean removeAll(Collection<?> collection) {
boolean result = false;
for (Object value : collection) result |= remove(value);
return result;
}
@Override
public boolean retainAll(Collection<?> collection) {
boolean result = false;
Iterator<Long> iterator = iterator();
while(iterator.hasNext()) {
Long l = iterator.next();
if (!collection.contains(l)) {
iterator.remove();
result = true;
}
}
return result;
}
}