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

org.neo4j.kernel.impl.util.collection.SimpleBitSet Maven / Gradle / Ivy

/*
 * Copyright (c) 2018-2020 "Graph Foundation,"
 * Graph Foundation, Inc. [https://graphfoundation.org]
 *
 * This file is part of ONgDB.
 *
 * ONgDB 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 3 of the License, or
 * (at your option) any later version.
 *
 * This program 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.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see .
 */
/*
 * Copyright (c) 2002-2020 "Neo4j,"
 * Neo4j Sweden AB [http://neo4j.com]
 *
 * This file is part of Neo4j.
 *
 * Neo4j 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 3 of the License, or
 * (at your option) any later version.
 *
 * This program 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.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see .
 */
package org.neo4j.kernel.impl.util.collection;

import java.util.Arrays;
import java.util.concurrent.locks.StampedLock;

import org.neo4j.collection.primitive.PrimitiveIntIterable;
import org.neo4j.collection.primitive.PrimitiveIntIterator;

/**
 * A basic bitset.
 * 

* Represented using an automatically expanding long array, with one bit per key. *

* Performance: * * put, remove, contains, size: O(1) * * clear: O(size/64) *

* Concurrency semantics: * * Concurrent writes synchronise and is thread-safe * * Concurrent reads are thread-safe and will not observe torn writes, but may become * out of date as soon as the operation returns * * Concurrent reads during write is thread-safe * * Bulk operations appear atomic to concurrent readers * * Only caveat being that the iterator is not thread-safe */ public class SimpleBitSet extends StampedLock implements PrimitiveIntIterable { private long lastCheckPointKey; private long[] data; public SimpleBitSet( int size ) { int initialCapacity = size / 64; int capacity = 1; while ( capacity < initialCapacity ) { capacity <<= 1; } long stamp = writeLock(); data = new long[capacity]; unlockWrite( stamp ); } public boolean contains( int key ) { int idx = key >>> 6; boolean result; long stamp; do { stamp = tryOptimisticRead(); result = data.length > idx && (data[idx] & ((1L << (key & 63)))) != 0; } while ( !validate( stamp ) ); return result; } public void put( int key ) { long stamp = writeLock(); int idx = key >>> 6; ensureCapacity( idx ); data[idx] = data[idx] | (1L << (key & 63)); unlockWrite( stamp ); } public void put( SimpleBitSet other ) { long stamp = writeLock(); ensureCapacity( other.data.length - 1 ); for ( int i = 0; i < data.length && i < other.data.length; i++ ) { data[i] = data[i] | other.data[i]; } unlockWrite( stamp ); } public void remove( int key ) { long stamp = writeLock(); int idx = key >>> 6; if ( data.length > idx ) { data[idx] = data[idx] & ~(1L << (key & 63)); } unlockWrite( stamp ); } public void remove( SimpleBitSet other ) { long stamp = writeLock(); for ( int i = 0; i < data.length; i++ ) { data[i] = data[i] & ~other.data[i]; } unlockWrite( stamp ); } public long checkPointAndPut( long checkPoint, int key ) { // We only need to clear the bit set if it was modified since the last check point if ( !validate( checkPoint ) || key != lastCheckPointKey ) { long stamp = writeLock(); int idx = key >>> 6; if ( idx < data.length ) { Arrays.fill( data, 0 ); } else { int len = data.length; len = findNewLength( idx, len ); data = new long[len]; } data[idx] = data[idx] | (1L << (key & 63)); lastCheckPointKey = key; checkPoint = tryConvertToOptimisticRead( stamp ); } return checkPoint; } private int findNewLength( int idx, int len ) { while ( len <= idx ) { len *= 2; } return len; } public int size() { int size = 0; for ( long s : data ) { size += Long.bitCount( s ); } return size; } private void ensureCapacity( int arrayIndex ) { data = Arrays.copyOf( data, findNewLength( arrayIndex, data.length ) ); } // // Views // @Override public PrimitiveIntIterator iterator() { return new PrimitiveIntIterator() { private int next; private final int size = data.length * 64; { // Prefetch first while ( next < size && !contains( next ) ) { next++; } } @Override public boolean hasNext() { return next < size; } @Override public int next() { int current = next; next++; while ( next < size && !contains( next ) ) { next++; } return current; } }; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy