diff --git a/bukkit/build.gradle b/bukkit/build.gradle index 7c950a30..c7fe27b3 100644 --- a/bukkit/build.gradle +++ b/bukkit/build.gradle @@ -36,6 +36,7 @@ apply plugin: 'com.github.johnrengelman.shadow' // We only want the shadow jar produced shadowJar { dependencies { + include(dependency('com.github.luben:zstd-jni:1.1.1')) include(dependency(':core')) } archiveName = "${parent.name}-${project.name}-${parent.version}.jar" diff --git a/core/build.gradle b/core/build.gradle index 5c26d979..0cabe603 100644 --- a/core/build.gradle +++ b/core/build.gradle @@ -6,6 +6,7 @@ dependencies { compile 'com.sk89q:worldguard:6.0.0-SNAPSHOT' compile 'com.plotsquared:PlotSquared:3.4.1-SNAPSHOT' compile 'org.primesoft:BlocksHub:2.0' + compile 'com.github.luben:zstd-jni:1.1.1' compile(group: 'com.sk89q.worldedit', name: 'worldedit-core', version:'6.1.3-SNAPSHOT') { exclude(module: 'bukkit-classloader-check') } diff --git a/core/src/main/java/com/boydti/fawe/Fawe.java b/core/src/main/java/com/boydti/fawe/Fawe.java index 1dbbe6a2..e143b778 100644 --- a/core/src/main/java/com/boydti/fawe/Fawe.java +++ b/core/src/main/java/com/boydti/fawe/Fawe.java @@ -104,7 +104,6 @@ import javax.management.InstanceAlreadyExistsException; import javax.management.Notification; import javax.management.NotificationEmitter; import javax.management.NotificationListener; -import net.jpountz.util.Native; /**[ WorldEdit action] * | @@ -465,7 +464,8 @@ public class Fawe { debug("======================================="); } try { - Native.load(); + com.github.luben.zstd.util.Native.load(); + net.jpountz.util.Native.load(); try { String arch = System.getenv("PROCESSOR_ARCHITECTURE"); String wow64Arch = System.getenv("PROCESSOR_ARCHITEW6432"); diff --git a/core/src/main/java/com/boydti/fawe/command/FaweParser.java b/core/src/main/java/com/boydti/fawe/command/FaweParser.java new file mode 100644 index 00000000..96bff2cb --- /dev/null +++ b/core/src/main/java/com/boydti/fawe/command/FaweParser.java @@ -0,0 +1,61 @@ +package com.boydti.fawe.command; + +import com.boydti.fawe.util.StringMan; +import com.sk89q.worldedit.WorldEdit; +import com.sk89q.worldedit.extension.input.InputParseException; +import com.sk89q.worldedit.extension.input.ParserContext; +import com.sk89q.worldedit.internal.registry.InputParser; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; + +public abstract class FaweParser extends InputParser { + protected FaweParser(WorldEdit worldEdit) { + super(worldEdit); + } + + public List split(String input, char delim) { + List result = new ArrayList(); + int start = 0; + boolean inQuotes = false; + for (int current = 0; current < input.length(); current++) { + if (input.charAt(current) == '\"') inQuotes = !inQuotes; // toggle state + boolean atLastChar = (current == input.length() - 1); + if(atLastChar) result.add(input.substring(start)); + else if (input.charAt(current) == delim && !inQuotes) { + String toAdd = input.substring(start, current); + if (toAdd.startsWith("\"")) { + toAdd = toAdd.substring(1, toAdd.length() - 1); + } + result.add(toAdd); + start = current + 1; + } + } + return result; + } + + public T catchSuggestion(String currentInput, String nextInput, ParserContext context) throws InputParseException { + try { + return parseFromInput(nextInput, context); + } catch (SuggestInputParseException e) { + e.prepend(currentInput.substring(0, currentInput.length() - nextInput.length())); + throw e; + } + } + + public List suggestRemaining(String input, String... expected) throws InputParseException { + List remainder = split(input, ':'); + int len = remainder.size(); + if (len != expected.length - 1) { + if (len <= expected.length - 1 && len != 0) { + if (remainder.get(len - 1).endsWith(":")) { + throw new SuggestInputParseException(null, StringMan.join(expected, ":")); + } + throw new SuggestInputParseException(null, expected[0] + ":" + input + ":" + StringMan.join(Arrays.copyOfRange(expected, len + 1, 3), ":")); + } else { + throw new SuggestInputParseException(null, StringMan.join(expected, ":")); + } + } + return remainder; + } +} diff --git a/core/src/main/java/com/boydti/fawe/command/Reload.java b/core/src/main/java/com/boydti/fawe/command/Reload.java index 4bf7df55..d9942ad3 100644 --- a/core/src/main/java/com/boydti/fawe/command/Reload.java +++ b/core/src/main/java/com/boydti/fawe/command/Reload.java @@ -9,7 +9,9 @@ import com.boydti.fawe.util.HastebinUtility; import com.boydti.fawe.util.MainUtil; import java.io.File; import java.io.IOException; +import java.time.LocalDate; import java.util.Date; +import java.util.GregorianCalendar; import java.util.Map; public class Reload extends FaweCommand { @@ -31,13 +33,14 @@ public class Reload extends FaweCommand { MainUtil.sendMessage(player, "No version information available."); return false; } - MainUtil.sendMessage(player, "Version Date: " + new Date(100 + version.year, version.month, version.day).toLocaleString()); + Date date = new GregorianCalendar(2000 + version.year - 1, version.month - 1, version.day).getTime(); + MainUtil.sendMessage(player, "Version Date: " + date.toLocaleString()); MainUtil.sendMessage(player, "Version Commit: " + Integer.toHexString(version.hash)); MainUtil.sendMessage(player, "Version Build: #" + version.build); return true; } case "threads": { - Map stacks = Fawe.get().getMainThread().getAllStackTraces(); + Map stacks = Thread.getAllStackTraces(); for (Map.Entry entry : stacks.entrySet()) { Thread thread = entry.getKey(); Fawe.debug("--------------------------------------------------------------------------------------------"); diff --git a/core/src/main/java/com/boydti/fawe/object/changeset/CPUOptimizedChangeSet.java b/core/src/main/java/com/boydti/fawe/object/changeset/CPUOptimizedChangeSet.java index d4b96085..c4b9e870 100644 --- a/core/src/main/java/com/boydti/fawe/object/changeset/CPUOptimizedChangeSet.java +++ b/core/src/main/java/com/boydti/fawe/object/changeset/CPUOptimizedChangeSet.java @@ -52,7 +52,6 @@ public class CPUOptimizedChangeSet extends FaweChangeSet { } @Override - public void add(int x, int y, int z, int combinedFrom, int combinedTo) { throw new UnsupportedOperationException("Invalid mode"); } @@ -89,9 +88,9 @@ public class CPUOptimizedChangeSet extends FaweChangeSet { @Override public boolean isEmpty() { - if (changes.size() == 0) { + if (changes.isEmpty()) { flush(); - return changes.size() == 0; + return changes.isEmpty(); } else { return false; } diff --git a/core/src/main/java/com/boydti/fawe/object/changeset/FaweStreamChangeSet.java b/core/src/main/java/com/boydti/fawe/object/changeset/FaweStreamChangeSet.java index e027e7fc..402cf051 100644 --- a/core/src/main/java/com/boydti/fawe/object/changeset/FaweStreamChangeSet.java +++ b/core/src/main/java/com/boydti/fawe/object/changeset/FaweStreamChangeSet.java @@ -257,7 +257,7 @@ public abstract class FaweStreamChangeSet extends FaweChangeSet { public abstract NBTInputStream getTileCreateIS() throws IOException; public abstract NBTInputStream getTileRemoveIS() throws IOException; - public int blockSize; + private int blockSize; public int entityCreateSize; public int entityRemoveSize; public int tileCreateSize; diff --git a/core/src/main/java/com/boydti/fawe/object/clipboard/DiskOptimizedClipboard.java b/core/src/main/java/com/boydti/fawe/object/clipboard/DiskOptimizedClipboard.java index ae022383..22cf61f6 100644 --- a/core/src/main/java/com/boydti/fawe/object/clipboard/DiskOptimizedClipboard.java +++ b/core/src/main/java/com/boydti/fawe/object/clipboard/DiskOptimizedClipboard.java @@ -4,6 +4,7 @@ import com.boydti.fawe.Fawe; import com.boydti.fawe.FaweCache; import com.boydti.fawe.config.Settings; import com.boydti.fawe.jnbt.NBTStreamer; +import com.boydti.fawe.jnbt.SchematicStreamer; import com.boydti.fawe.object.IntegerTrio; import com.boydti.fawe.object.RunnableVal2; import com.boydti.fawe.object.io.BufferedRandomAccessFile; @@ -11,6 +12,7 @@ import com.boydti.fawe.util.MainUtil; import com.boydti.fawe.util.ReflectionUtils; import com.sk89q.jnbt.CompoundTag; import com.sk89q.jnbt.IntTag; +import com.sk89q.jnbt.NBTInputStream; import com.sk89q.jnbt.Tag; import com.sk89q.worldedit.BlockVector; import com.sk89q.worldedit.EditSession; @@ -21,8 +23,10 @@ import com.sk89q.worldedit.entity.Entity; import com.sk89q.worldedit.extent.Extent; import com.sk89q.worldedit.extent.clipboard.BlockArrayClipboard; import com.sk89q.worldedit.regions.CuboidRegion; +import java.io.BufferedInputStream; import java.io.Closeable; import java.io.File; +import java.io.FileInputStream; import java.io.IOException; import java.util.ArrayList; import java.util.HashMap; @@ -30,6 +34,7 @@ import java.util.HashSet; import java.util.List; import java.util.Map; import java.util.UUID; +import java.util.zip.GZIPInputStream; /** * A clipboard with disk backed storage. (lower memory + loads on crash) @@ -55,7 +60,6 @@ public class DiskOptimizedClipboard extends FaweClipboard implements Closeable { private final byte[] buffer; private final BufferedRandomAccessFile raf; - private long lastAccessed; private int last; public DiskOptimizedClipboard(int width, int height, int length, UUID uuid) { @@ -68,7 +72,6 @@ public class DiskOptimizedClipboard extends FaweClipboard implements Closeable { entities = new HashSet<>(); this.buffer = new byte[2]; this.file = file; - this.lastAccessed = System.currentTimeMillis(); this.raf = new BufferedRandomAccessFile(file, "rw", 16); raf.setLength(file.length()); long size = (raf.length() - HEADER_SIZE) >> 1; @@ -117,7 +120,6 @@ public class DiskOptimizedClipboard extends FaweClipboard implements Closeable { entities = new HashSet<>(); this.file = file; this.buffer = new byte[2]; - this.lastAccessed = System.currentTimeMillis(); this.width = width; this.height = height; this.length = length; @@ -132,6 +134,7 @@ public class DiskOptimizedClipboard extends FaweClipboard implements Closeable { } this.raf = new BufferedRandomAccessFile(file, "rw", 16); long volume = width * height * length * 2l + HEADER_SIZE; + raf.setLength(0); raf.setLength(volume); // write length etc raf.seek(2); @@ -325,7 +328,6 @@ public class DiskOptimizedClipboard extends FaweClipboard implements Closeable { int i = getIndex(x, y, z); if (i != last + 1) { raf.seek((HEADER_SIZE) + (i << 1)); - lastAccessed = System.currentTimeMillis(); } last = i; int combinedId = raf.readChar(); @@ -360,15 +362,15 @@ public class DiskOptimizedClipboard extends FaweClipboard implements Closeable { int i = x + ((ylast == y) ? ylasti : (ylasti = ((ylast = y)) * area)) + ((zlast == z) ? zlasti : (zlasti = (zlast = z) * width)); if (i != last + 1) { raf.seek((HEADER_SIZE) + (i << 1)); - lastAccessed = System.currentTimeMillis(); } last = i; final int id = block.getId(); final int data = block.getData(); int combined = (id << 4) + data; raf.writeChar(combined); - if (FaweCache.hasNBT(id)) { - setTile(x, y, z, block.getNbtData()); + CompoundTag tile = block.getNbtData(); + if (tile != null) { + setTile(x, y, z, tile); } return true; } catch (Exception e) { @@ -382,15 +384,14 @@ public class DiskOptimizedClipboard extends FaweClipboard implements Closeable { try { if (i != last + 1) { raf.seek((HEADER_SIZE) + (i << 1)); - lastAccessed = System.currentTimeMillis(); } last = i; // 00000000 00000000 // [ id ]data int id1 = raf.readCurrent(); - raf.write(id >> 4); + raf.writeUnsafe(id >> 4); int id2 = raf.readCurrent(); - raf.write(((id & 0xFF) << 4) + (id2 & 0xFF)); + raf.writeUnsafe(((id & 0xFF) << 4) + (id2 & 0xFF)); } catch (Exception e) { MainUtil.handleError(e); } @@ -400,7 +401,6 @@ public class DiskOptimizedClipboard extends FaweClipboard implements Closeable { try { if (i != last + 1) { raf.seek((HEADER_SIZE) + (i << 1)); - lastAccessed = System.currentTimeMillis(); } last = i; raf.writeChar(combined); @@ -414,31 +414,43 @@ public class DiskOptimizedClipboard extends FaweClipboard implements Closeable { try { if (i != last + 1) { raf.seek((HEADER_SIZE) + (i << 1)); - lastAccessed = System.currentTimeMillis(); } last = i; // 00000000 00000000 // [ id ]data - raf.write((raf.readCurrent() & 0xFF) + (add >> 4)); + int id = (raf.readCurrent() & 0xFF); + raf.writeUnsafe(id + (add >> 4)); raf.read1(); } catch (Exception e) { MainUtil.handleError(e); } } + public static void main(String[] args) throws IOException{ + long start = System.currentTimeMillis(); + File file = new File("C:/Users/Jesse/Desktop/OTHER/mc/plugins/WorldEdit/schematics/50mil.schematic"); + BufferedInputStream in = new BufferedInputStream(new GZIPInputStream(new BufferedInputStream(new FileInputStream(file)))); + NBTInputStream nbtin = new NBTInputStream(in); + Settings.CLIPBOARD.USE_DISK = true; + SchematicStreamer streamer = new SchematicStreamer(nbtin, UUID.randomUUID()); + streamer.getClipboard(); + System.out.println(System.currentTimeMillis() - start); + } + @Override public void setData(int i, int data) { try { if (i != last + 1) { - raf.seek((HEADER_SIZE) + (i << 1)); - lastAccessed = System.currentTimeMillis(); + raf.seek((HEADER_SIZE) + (i << 1) + 1); + } else { + raf.seek(raf.getFilePointer() + 1); } last = i; // 00000000 00000000 // [ id ]data - int id1 = raf.read1(); +// int skip = raf.read1(); int id2 = raf.readCurrent(); - raf.write((id2 & 0xF0) + data); + raf.writeUnsafe((id2 & 0xF0) + data); } catch (Exception e) { MainUtil.handleError(e); } diff --git a/core/src/main/java/com/boydti/fawe/object/collection/SparseBitSet.java b/core/src/main/java/com/boydti/fawe/object/collection/SparseBitSet.java new file mode 100644 index 00000000..fd6937cd --- /dev/null +++ b/core/src/main/java/com/boydti/fawe/object/collection/SparseBitSet.java @@ -0,0 +1,2942 @@ +package com.boydti.fawe.object.collection; + +import java.io.IOException; +import java.io.ObjectInputStream; +import java.io.ObjectOutputStream; +import java.io.Serializable; + +/** + * This class implements a set of bits that grows as needed. Each bit of the + * bit set represents a boolean value. The values of a + * SparseBitSet are indexed by nonnegative integers. + * Individual indexed values may be examined, set, cleared, or modified by + * logical operations. One SparseBitSet or logical value may be + * used to modify the contents of (another) SparseBitSet through + * logical AND, logical inclusive OR, logical exclusive + * OR, and And NOT operations over all or part of the bit sets. + *

+ * All values in a bit set initially have the value false. + *

+ * Every bit set has a current size, which is the number of bits of space + * nominally in use by the bit set from the first set bit to just after + * the last set bit. The length of the bit set effectively tells the position + * available after the last bit of the SparseBitSet. + *

+ * The maximum cardinality of a SparseBitSet is + * Integer.MAX_VALUE, which means the bits of a + * SparseBitSet are labelled + * 0 .. Integer.MAX_VALUE − 1. + * After the last set bit of a SparseBitSet, any attempt to find + * a subsequent bit (getNextSetBit()), will return an value of −1. + * If an attempt is made to use getNextClearBit(), and all the bits are + * set from the starting postion of the search to the bit labelled + * Integer.MAX_VALUE − 1, then similarly −1 + * will be returned. + *

+ * Unless otherwise noted, passing a null parameter to any of the methods in + * a SparseBitSet will result in a + * NullPointerException. + *

+ * A SparseBitSet is not safe for multithreaded use without + * external synchronization. + * + * @author Bruce K. Haddon + * @author Arthur van Hoff + * @author Michael McCloskey + * @author Martin Buchholz + * @version 1.0, 2009-03-17 + * @since 1.6 + */ +public class SparseBitSet implements Cloneable, Serializable +{ + /* My apologies for listing all the additional authors, but concepts, code, + and even comments have been re-used in this class definition from code in + the JDK that was written and/or maintained by these people. I owe a debt, + which I acknowledge. But they are in no way responsible for what ever + misuse I have made of their work. + Bruce K. Haddon + + The representation of a SparseBitSet is packed into "words", and the words + are stored in arrays which are here referred to as "blocks." Blocks are + accessed by two levels of indirection from the master "level 1" (whole) + set array through second level arrays called "areas" (making the blocks + "level3"). A "word" is a long, consisting of 64 bits, requiring six address + bits to select a bit within a word (and this is considered in places to be + a "level4"). This choice of a long for "word" is determined purely by + performance concerns, and is built into the implementation in a deep way, + because the references to blocks are always of the form "long[]." (This + does not mean that blocks could not be changed to arrays of ints, but it + would take some extensive work.) + + The fact that there are three levels is also deeply involved in the + scanning algorithms, meaning that the accesses are always nested three + deep. Again, the change this might be a large amount of work. On the + other hand, these three levels have proven, so far, to provide adequate + speed, and an storage efficient way to deal with sparseness. + + For simplicity, the level3 blocks and the level2 areas are always "full" + size, i.e., LENGTH3 and LENGTH2 respectively, and for consistency, the + fourth level is of length LENGTH4. The level1 structure is of variable + length (as this may save scanning several thousand null pointers, and + careful consideration must be taken of this at all times, in particular, + when choosing to increase the size (see resize()). The only place where + this array is reduced in size is when a clone is made, in which case it is + created with the smallest size, and allowed to grow as entries are copied + to it. That all the arrays are kept to power-of-2 sizes is a programming + convenience (permitting shifts and masks). + + Whenever possible, a level 3 block that contains no bits (all the words are + zero, is discarded, and its reference is replaced by a null value. + Similarly, level 2 areas that contain only null pointers (to level 3 + blocks) are discarded, and their references replaced by null values. This + is the "normalized" condition, but does not have to be nor is strictly + enforced. The operations still work if the representation is partially or + totally "denormalized." In particular, the methods that deal with single + bits (setting, flipping, clearing, etc.) do not attempt to normalize the + set, in the interests of speed. However, when a set is scanned as the + resultant set of some operation, then, in most cases, the set will be + normalized--the exception being level2 areas that are not completly scanned + in a particular pass. + + The sizes of the blocks and areas has been the result of some investigation + with varying sizes, and the sizes selected appear to represent a reasonable + "sweet" spot. There is, of course, no guarantee that these are the best + for all possible situations, but, given that not having these the same + for all bit sets would be hopelessly complex (bad enough as is), these + values appear to be a fair compromise. */ + + /** + * This value controls for format of the toString() output. + * @see #toStringCompaction(int) + */ + protected transient int compactionCount; + + /** + * The compaction count default. + */ + static int compactionCountDefault = 2; // Note: this is not final! + + /** + * The storage for this SparseBitSet. The ith bit is stored in a word + * represented by a long value, and is at bit position i % 64 + * within that word (where bit position 0 refers to the least significant bit + * and 63 refers to the most significant bit). + *

+ * The words are organized into blocks, and the blocks are accessed by two + * additional levels of array indexing. + */ + protected transient long[][][] bits; + + /** + * For the current size of the bits array, this is the maximum possible + * length of the bit set, i.e., the index of the last possible bit, plus one. + * Note: this not the value returned by length(). + * @see #resize(int) + * @see #length() + */ + protected transient int bitsLength; + + //============================================================================== + // The critical parameters. These are set up so that the compiler may + // pre-compute all the values as compile-time constants. + //============================================================================== + /** + * The number of bits in a long value. + */ + protected static final int LENGTH4 = Long.SIZE; + + /** + * The number of bits in a positive integer, and the size of permitted index + * of a bit in the bit set. + */ + protected static final int INDEX_SIZE = Integer.SIZE - 1; + + /** + * The label (index) of a bit in the bit set is essentially broken into + * 4 "levels". Respectively (from the least significant end), level4, the + * address within word, the address within a level3 block, the address within + * a level2 area, and the level1 address of that area within the set. + * + * LEVEL4 is the number of bits of the level4 address (number of bits need + * to address the bits in a long) + */ + protected static final int LEVEL4 = 6; + + /** + * LEVEL3 is the number of bits of the level3 address. + */ + protected static final int LEVEL3 = 5; // Do not change! + /** + * LEVEL2 is the number of bits of the level2 address. + */ + protected static final int LEVEL2 = 5; // Do not change! + /** + * LEVEL1 is the number of bits of the level1 address. + */ + protected static final int LEVEL1 = INDEX_SIZE - LEVEL2 - LEVEL3 - LEVEL4; + + /** + * MAX_LENGTH1 is the maximum number of entries in the level1 set array. + */ + protected static final int MAX_LENGTH1 = 1 << LEVEL1; + + /** + * LENGTH2 is the number of entries in the any level2 area. + */ + protected static final int LENGTH2 = 1 << LEVEL2; + + /** + * LENGTH3 is the number of entries in the any level3 block. + */ + protected static final int LENGTH3 = 1 << LEVEL3; + + /** + * The shift to create the word index. (I.e., move it to the right end) + */ + protected static final int SHIFT3 = LEVEL4; + + /** + * MASK3 is the mask to extract the LEVEL3 address from a word index + * (after shifting by SHIFT3). + */ + protected static final int MASK3 = LENGTH3 - 1; + + /** + * SHIFT2 is the shift to bring the level2 address (from the word index) to + * the right end (i.e., after shifting by SHIFT3). + */ + protected static final int SHIFT2 = LEVEL3; + + /** + * UNIT is the greatest number of bits that can be held in one level1 entry. + * That is, bits per word by words per level3 block by blocks per level2 area. + */ + + protected static final int UNIT = LENGTH2 * LENGTH3 * LENGTH4; + + /** + * MASK2 is the mask to extract the LEVEL2 address from a word index + * (after shifting by SHIFT3 and SHIFT2). + */ + protected static final int MASK2 = LENGTH2 - 1; + + /** + * SHIFT1 is the shift to bring the level1 address (from the word index) to + * the right end (i.e., after shifting by SHIFT3). + */ + protected static final int SHIFT1 = LEVEL2 + LEVEL3; + + /** + * Holds reference to the cache of statistics values computed by the + * UpdateStrategy + * @see SparseBitSet.Cache + * @see SparseBitSet.UpdateStrategy + */ + protected transient Cache cache; + + //============================================================================= + // Stack structures used for recycling blocks + //============================================================================= + // private static final int STACK_SIZE = 1000; + + // private static final long[][] stack = new long[STACK_SIZE][LENGTH3]; + + // private static int stackIndex = 0; + + /** + * A spare level 3 block is kept for use when scanning. When a target block + * is needed, and there is not already one in the bit set, the spare is + * provided. If non-zero values are placed into this block, it is moved to the + * resulting set, and a new spare is acquired. Note: a new spare needs to + * be allocated when the set is cloned (so that the spare is not shared + * between two sets). + */ + protected transient long[] spare; + + /** An empty level 3 block is kept for use when scanning. When a source block + * is needed, and there is not already one in the corresponding bit set, the + * ZERO_BLOCK is used (as a read-only block). It is a source of zero values + * so that code does not have to test for a null level3 block. This is a + * static block shared everywhere. + */ + static final long[] ZERO_BLOCK = new long[LENGTH3]; + + /* Programming notes: + + i, j, and k are used to hold values that are actual bit indices (i.e., + the index (label) of the bit within the user's view of the bit set). + + u, v, and w, are used to hold values that refer to the indices of the + words in the set array that are used to hold the bits (with 64 bits per + word). These variable names, followed by 1, 2, or 3, refer to the component + "level" parts of the complete word index. + + word (where used) is a potential entry to or from a block, containing 64 + bits of the bit set. The prefixes a, b, result, etc., refer to the bit + sets from which these are coming or going. Without a prefix, or with the + prefix "a," the set in question is "this" set (see next paragraph). + + Operations are conceived to be in the form a.op(b), thus in the discussion + (not in the public Javadoc documentation) the two sets are referred to a + "a" and "b", where the set referred to by "this" is usually set a. + Hence, reference to set a is usually implicit, but set b will usually be + explicit. Variables beginning with these letters hold values relevant to + the corresponding set, and, in particular, these letters followed by + 1, 2, and 3 are used to refer to the corresponding (current) level1, + level3 area, and level3 block, arrays. + + The resizing of the table takes place as necessary. In this regard, it is + worth noting that the table is grown, but never shrunk (except in a new + object formed by cloning). + + Similarly, care it taken to ensure that any supplied reference to a bit + set (other than this) has an opportunity to fail for being null before + any other set (including this) has its state changed. For the most + part, this is allowed to happen "naturally," but the Strategies incorporate + an explicit check when necessary. + + There is a amount of (almost) repetitive scanning code in many of the + "singe bit" methods. The intent is that these methods for SparseBitSet be + as small and as fast as possible. + + For the scanning of complete sets, or for ranges within complete sets, + all of the scanning logic is built into one (somewhat enormous) method, + setScanner(). This contains all the considerations for matching up + corresponding level 3 blocks (if they exist), and then uses a Strategy + object to do the processing on those level3 blocks. This keeps all + the scanning and optimization logic in one place, and the Strategies are + reasonably simple (see the definition of AbstractStrategy for a discussion + of the tasks that must be defined therein). + + The test for index i (the first index in all cases) being in range is + rather perverse, but the idea was to keep the actual number of comparisons + to a minimum, hence the check is for "(i + 1) < 1". This is almost but not + quite equivalent to "i < 0", although it is for all values of i except + i=Integer.MAX_VALUE. In this latter case, (i + 1) "overflows" to + -(Integer.MAX_VALUE + 1), and thus appears to be less than 1, and thus the + check picks up the other disallowed case. Let us hope the compiler never + gets smart enough to try to do the apparent optimisation! */ + + /** + * Constructor for a new (sparse) bit set. All bits initially are effectively + * false. This is a internal constructor that collects all the + * needed actions to initialise the bit set. + *

+ * The capacity is taken to be a suggestion for a size of the bit set, + * in bits. An appropiate table size (a power of two) is then determined and + * used. The size will be grown as needed to accomodate any bits addressed + * during the use of the bit set. + * + * @param capacity a size in terms of bits + * @param compactionCount the compactionCount to be inherited (for + * internal generation) + * @exception NegativeArraySizeException if the specified initial size + * is negative + * @since 1.6 + */ + protected SparseBitSet(int capacity, int compactionCount) + throws NegativeArraySizeException + { + /* Array size is computed based on this being a capacity given in bits. */ + if (capacity < 0) // capacity can't be negative -- could only come from + throw new NegativeArraySizeException( // an erroneous user given + "(requested capacity=" + capacity + ") < 0"); // nbits value + resize(capacity - 1); // Resize takes last usable index + this.compactionCount = compactionCount; + /* Ensure there is a spare level 3 block for the use of the set scanner.*/ + constructorHelper(); + statisticsUpdate(); + } + + /** + * Constructs an empty bit set with the default initial size. + * Initially all bits are effectively false. + * + * @since 1.6 + */ + public SparseBitSet() + { + /* By requesting 1 bit, will actually get UNIT number of bits. */ + this(1, compactionCountDefault); + } + + /** + * Creates a bit set whose initial size is large enough to efficiently + * represent bits with indices in the range 0 through + * at least nbits-1. Initially all bits are effectively + * false. + *

+ * No guarantees are given for how large or small the actual object will be. + * The setting of bits above the given range is permitted (and will perhaps + * eventually cause resizing). + * + * @param nbits the initial provisional length of the SparseBitSet + * @throws java.lang.NegativeArraySizeException if the specified initial + * length is negative + * @see #SparseBitSet() + * @since 1.6 + */ + public SparseBitSet(int nbits) throws NegativeArraySizeException + { + this(nbits, compactionCountDefault); + } + + /** + * Performs a logical AND of the addressed target bit with the argument + * value. This bit set is modified so that the addressed bit has the value + * true if and only if it both initially had the value + * true and the argument value is also true. + * + * @param i a bit index + * @param value a boolean value to AND with that bit + * @exception IndexOutOfBoundsException if the specified index is negative + * or equal to Integer.MAX_VALUE + * @since 1.6 + */ + public void and(int i, boolean value) throws IndexOutOfBoundsException + { + if ((i + 1) < 1) + throw new IndexOutOfBoundsException("i=" + i); + if (!value) + clear(i); + } + + /** + * Performs a logical AND of this target bit set with the argument bit + * set within the given range of bits. Within the range, this bit set is + * modified so that each bit in it has the value true if and only + * if it both initially had the value true and the corresponding + * bit in the bit set argument also had the value true. Outside + * the range, this set is not changed. + * + * @param i index of the first bit to be included in the operation + * @param j index after the last bit to included in the operation + * @param b a SparseBitSet + * @exception IndexOutOfBoundsException if i is negative or + * equal to Integer.MAX_VALUE, or j is negative, + * or i is larger than j + * @since 1.6 + */ + + public void and(int i, int j, SparseBitSet b) throws IndexOutOfBoundsException + { + setScanner(i, j, b, andStrategy); + } + + /** + * Performs a logical AND of this target bit set with the argument bit + * set. This bit set is modified so that each bit in it has the value + * true if and only if it both initially had the value + * true and the corresponding bit in the bit set argument also + * had the value true. + * + * @param b a SparseBitSet + * @since 1.6 + */ + public void and(SparseBitSet b) + { + nullify(Math.min(bits.length, b.bits.length)); // Optimisation + setScanner(0, Math.min(bitsLength, b.bitsLength), b, andStrategy); + } + + /** + * Performs a logical AND of the two given SparseBitSets. + * The returned SparseBitSet is created so that each bit in it + * has the value true if and only if both the given sets + * initially had the corresponding bits true, otherwise + * false. + * + * @param a a SparseBitSet + * @param b another SparseBitSet + * @return a new SparseBitSet representing the AND of the two sets + * @since 1.6 + */ + public static SparseBitSet and(SparseBitSet a, SparseBitSet b) + { + final SparseBitSet result = a.clone(); + result.and(b); + return result; + } + + /** + * Performs a logical AndNOT of the addressed target bit with the + * argument value. This bit set is modified so that the addressed bit has the + * value true if and only if it both initially had the value + * true and the argument value is false. + * + * @param i a bit index + * @param value a boolean value to AndNOT with that bit + * @exception IndexOutOfBoundsException if the specified index is negative + * or equal to Integer.MAX_VALUE + * @since 1.6 + */ + public void andNot(int i, boolean value) + { + if ((i + 1) < 1) + throw new IndexOutOfBoundsException("i=" + i); + if (value) + clear(i); + } + + /** + * Performs a logical AndNOT of this target bit set with the argument + * bit set within the given range of bits. Within the range, this bit set is + * modified so that each bit in it has the value true if and only + * if it both initially had the value true and the corresponding + * bit in the bit set argument has the value false. Outside + * the range, this set is not changed. + * + * @param i index of the first bit to be included in the operation + * @param j index after the last bit to included in the operation + * @param b the SparseBitSet with which to mask this SparseBitSet + * @exception IndexOutOfBoundsException if i is negative or + * equal to Integer.MAX_VALUE, or j is negative, + * or i is larger than j + * @since 1.6 + */ + public void andNot(int i, int j, SparseBitSet b) + throws IndexOutOfBoundsException + { + setScanner(i, j, b, andNotStrategy); + } + + /** + * Performs a logical AndNOT of this target bit set with the argument + * bit set. This bit set is modified so that each bit in it has the value + * true if and only if it both initially had the value + * true and the corresponding bit in the bit set argument has + * the value false. + * + * @param b the SparseBitSet with which to mask this SparseBitSet + * @since 1.6 + */ + public void andNot(SparseBitSet b) + { + setScanner(0, Math.min(bitsLength, b.bitsLength), b, andNotStrategy); + } + + /** + * Creates a bit set from thie first SparseBitSet whose + * corresponding bits are cleared by the set bits of the second + * SparseBitSet. The resulting bit set is created so that a bit + * in it has the value true if and only if the corresponding bit + * in the SparseBitSet of the first is set, and that same + * corresponding bit is not set in the SparseBitSet of the second + * argument. + * + * @param a a SparseBitSet + * @param b another SparseBitSet + * @return a new SparseBitSet representing the AndNOT of the + * two sets + * @since 1.6 + */ + public static SparseBitSet andNot(SparseBitSet a, SparseBitSet b) + { + final SparseBitSet result = a.clone(); + result.andNot(b); + return result; + } + + /** + * Returns the number of bits set to true in this + * SparseBitSet. + * + * @return the number of bits set to true in this SparseBitSet + * @since 1.6 + */ + public int cardinality() + { + statisticsUpdate(); // Update size, cardinality and length values + return cache.cardinality; + } + + /** + * Sets the bit at the specified index to false. + * + * @param i a bit index. + * @exception IndexOutOfBoundsException if the specified index is negative + * or equal to Integer.MAX_VALUE. + * @since 1.6 + */ + public void clear(int i) + { + /* In the interests of speed, no check is made here on whether the + level3 block goes to all zero. This may be found and corrected + in some later operation. */ + if ((i + 1) < 1) + throw new IndexOutOfBoundsException("i=" + i); + if (i > bitsLength) + return; + final int w = i >> SHIFT3; + long[][] a2; + if ((a2 = bits[w >> SHIFT1]) == null) + return; + long[] a3; + if ((a3 = a2[(w >> SHIFT2) & MASK2]) == null) + return; + a3[w & MASK3] &= ~(1L << i); // Clear the indicated bit + cache.hash = 0; // Invalidate size, etc., + } + + /** + * Sets the bits from the specified i (inclusive) to the + * specified j (exclusive) to false. + * + * @param i index of the first bit to be cleared + * @param j index after the last bit to be cleared + * @exception IndexOutOfBoundsException if i is negative or + * equal to Integer.MAX_VALUE, or j is negative, + * or i is larger than j + * @since 1.6 + */ + public void clear(int i, int j) throws IndexOutOfBoundsException + { + setScanner(i, j, null, clearStrategy); + } + + /** + * Sets all of the bits in this SparseBitSet to + * false. + * + * @since 1.6 + */ + public void clear() + { + /* This simply resets to null all the entries in the set. */ + nullify(0); + } + + /** + * Cloning this SparseBitSet produces a new + * SparseBitSet that is equal() to it. The clone of the + * bit set is another bit set that has exactly the same bits set to + * true as this bit set. + *

+ * Note: the actual space allocated to the clone tries to minimise the actual + * amount of storage allocated to hold the bits, while still trying to + * keep access to the bits being a rapid as possible. Since the space + * allocated to a SparseBitSet is not normally decreased, + * replacing a bit set by its clone may be a way of both managing memory + * consumption and improving the rapidity of access. + * + * @return a clone of this SparseBitSet + * @since 1.6 + */ + @Override + public SparseBitSet clone() + { + try + { + final SparseBitSet result = (SparseBitSet) super.clone(); + /* Clear out the shallow copy of the set array (which contains just + copies of the references from this set), and then replace these + by a deep copy (created by a "copy" from the set being cloned . */ + result.bits = null; + result.resize(1); + /* Ensure the clone is not sharing a copy of a spare block with + the cloned set, nor the cache set, nor any of the visitors (which + are linked to their parent object) (Not all visitors actually use + this link to their containing object, but they are reset here just + in case of future changes). */ + result.constructorHelper(); // also creates the copyStrategy + result.setScanner(0, bitsLength, this, result.copyStrategy); + return result; + } + catch (CloneNotSupportedException ex) + { + /* This code has not been unit tested. Inspection offers hope + that is will work, but it likely never to be used. */ + throw new InternalError(ex.getMessage()); + } + } + + /** + * Compares this object against the specified object. The result is + * true if and only if the argument is not null + * and is a SparseBitSet object that has exactly the same bits + * set to true as this bit set. That is, for every nonnegative + * i indexing a bit in the set, + *

((SparseBitSet)obj).get(i) == this.get(i)
+ * must be true. + * + * @param obj the Object with which to compare + * @return true if the objects are equivalent; + * false otherwise. + * @since 1.6 + */ + @Override + public boolean equals(Object obj) + { + /* Sanity and quick checks. */ + if (!(obj instanceof SparseBitSet)) + return false; + final SparseBitSet b = (SparseBitSet) obj; + if (this == b) + return true; // Identity + + /* Do the real work. */ + setScanner(0, Math.max(bitsLength, b.bitsLength), b, equalsStrategy); + return equalsStrategy.result; + } + + /** + * Sets the bit at the specified index to the complement of its current value. + * + * @param i the index of the bit to flip + * @exception IndexOutOfBoundsException if the specified index is negative + * or equal to Integer.MAX_VALUE + * @since 1.6 + */ + public void flip(int i) + { + if ((i + 1) < 1) + throw new IndexOutOfBoundsException("i=" + i); + final int w = i >> SHIFT3; + final int w1 = w >> SHIFT1; + final int w2 = (w >> SHIFT2) & MASK2; + + if (i >= bitsLength) + resize(i); + long[][] a2; + if ((a2 = bits[w1]) == null) + a2 = bits[w1] = new long[LENGTH2][]; + long[] a3; + if ((a3 = a2[w2]) == null) + a3 = a2[w2] = new long[LENGTH3]; + a3[w & MASK3] ^= 1L << i; //Flip the designated bit + cache.hash = 0; // Invalidate size, etc., values + } + + /** + * Sets each bit from the specified i (inclusive) to the + * specified j (exclusive) to the complement of its current + * value. + * + * @param i index of the first bit to flip + * @param j index after the last bit to flip + * @exception IndexOutOfBoundsException if i is negative or is + * equal to Integer.MAX_VALUE, or j is negative, or + * i is larger than j + * @since 1.6 + */ + public void flip(int i, int j) throws IndexOutOfBoundsException + { + setScanner(i, j, null, flipStrategy); + } + + /** + * Returns the value of the bit with the specified index. The value is + * true if the bit with the index i is currently set + * in this SparseBitSet; otherwise, the result is + * false. + * + * @param i the bit index + * @return the boolean value of the bit with the specified index. + * @exception IndexOutOfBoundsException if the specified index is negative + * or equal to Integer.MAX_VALUE + * @since 1.6 + */ + public boolean get(int i) + { + if ((i + 1) < 1) + throw new IndexOutOfBoundsException("i=" + i); + final int w = i >> SHIFT3; + + long[][] a2; + long[] a3; + return i < bitsLength && (a2 = bits[w >> SHIFT1]) != null + && (a3 = a2[(w >> SHIFT2) & MASK2]) != null + && ((a3[w & MASK3] & (1L << i)) != 0); + } + + /** + * Returns a new SparseBitSet composed of bits from this + * SparseBitSet from i (inclusive) to j + * (exclusive). + * + * @param i index of the first bit to include + * @param j index after the last bit to include + * @return a new SparseBitSet from a range of this SparseBitSet + * @exception IndexOutOfBoundsException if i is negative or is + * equal to Integer.MAX_VALUE, or j is negative, or + * i is larger than j + * @since 1.6 + */ + public SparseBitSet get(int i, int j) throws IndexOutOfBoundsException + { + final SparseBitSet result = new SparseBitSet(j, compactionCount); + result.setScanner(i, j, this, result.copyStrategy); + return result; + } + + /** + * Returns a hash code value for this bit set. The hash code depends only on + * which bits have been set within this SparseBitSet. The + * algorithm used to compute it may be described as follows. + *

+ * Suppose the bits in the SparseBitSet were to be stored in an + * array of long integers called, say, bits, in such + * a manner that bit i is set in the SparseBitSet + * (for nonnegative values of i) if and only if the expression + *

+     *  ((i>>6) < bits.length) && ((bits[i>>6] & (1L << (bit & 0x3F))) != 0)
+     *  
+ * is true. Then the following definition of the hashCode method + * would be a correct implementation of the actual algorithm: + *
+     *  public int hashCode()
+     *  {
+     *      long hash = 1234L;
+     *      for( int i = bits.length; --i >= 0; )
+     *          hash ^= bits[i] * (i + 1);
+     *      return (int)((h >> 32) ^ h);
+     *  }
+ * Note that the hash code values change if the set of bits is altered. + * + * @return a hash code value for this bit set + * @since 1.6 + * @see Object#equals(Object) + * @see java.util.Hashtable + */ + @Override + public int hashCode() + { + statisticsUpdate(); + return cache.hash; + } + + /** + * Returns true if the specified SparseBitSet has any bits + * within the given range i (inclusive) to j + * (exclusive) set to true that are also set to true + * in the same range of this SparseBitSet. + * + * @param i index of the first bit to include + * @param j index after the last bit to include + * @param b the SparseBitSet with which to intersect + * @return the boolean indicating whether this SparseBitSet intersects the + * specified SparseBitSet + * @exception IndexOutOfBoundsException if i is negative or + * equal to Integer.MAX_VALUE, or j is negative, + * or i is larger than j + * @since 1.6 + */ + public boolean intersects(int i, int j, SparseBitSet b) + throws IndexOutOfBoundsException + { + setScanner(i, j, b, intersectsStrategy); + return intersectsStrategy.result; + } + + /** + * Returns true if the specified SparseBitSet has any bits set to + * true that are also set to true in this + * SparseBitSet. + * + * @param b a SparseBitSet with which to intersect + * @return boolean indicating whether this SparseBitSet intersects the + * specified SparseBitSet + * @since 1.6 + */ + public boolean intersects(SparseBitSet b) + { + setScanner(0, Math.max(bitsLength, b.bitsLength), b, intersectsStrategy); + return intersectsStrategy.result; + } + + /** + * Returns true if this SparseBitSet contains no bits that are + * set to true. + * + * @return the boolean indicating whether this SparseBitSet is empty + * @since 1.6 + */ + public boolean isEmpty() + { + statisticsUpdate(); + return cache.cardinality == 0; + } + + /** + * Returns the "logical length" of this SparseBitSet: the index + * of the highest set bit in the SparseBitSet plus one. Returns + * zero if the SparseBitSet contains no set bits. + * + * @return the logical length of this SparseBitSet + * @since 1.6 + */ + public int length() + { + statisticsUpdate(); + return cache.length; + } + + /** + * Returns the index of the first bit that is set to false that + * occurs on or after the specified starting index. + * + * @param i the index to start checking from (inclusive) + * @return the index of the next clear bit, or -1 if there is no such bit + * @exception IndexOutOfBoundsException if the specified index is negative + * @since 1.6 + */ + public int nextClearBit(int i) + { + /* The index of this method is permitted to be Integer.MAX_VALUE, as this + is needed to make this method work together with the method + nextSetBit()--as might happen if a search for the next clear bit is + started after finding a set bit labelled Integer.MAX_VALUE-1. This + case is not optimised, the code will eventually return -1 (since + the Integer.MAX_VALUEth bit does "exist," and is 0. */ + + if (i < 0) + throw new IndexOutOfBoundsException("i=" + i); + /* This is the word from which the search begins. */ + int w = i >> SHIFT3; + int w3 = w & MASK3; + int w2 = (w >> SHIFT2) & MASK2; + int w1 = w >> SHIFT1; + + long nword = ~0L << i; + final int aLength = bits.length; + + long[][] a2; + long[] a3; + /* Is the next clear bit in the same word at the nominated beginning bit + (including the nominated beginning bit itself). The first check is + whether the starting bit is within the structure at all. */ + if (w1 < aLength && (a2 = bits[w1]) != null + && (a3 = a2[w2]) != null + && ((nword = ~a3[w3] & (~0L << i))) == 0L) + { + /* So now start a search though the rest of the entries for + a null area or block, or a clear bit (a set bit in the + complemented value). */ + ++w; + w3 = w & MASK3; + w2 = (w >> SHIFT2) & MASK2; + w1 = w >> SHIFT1; + loop: for (; w1 != aLength; ++w1) + { + if ((a2 = bits[w1]) != null) + for (; w2 != LENGTH2; ++w2) + { + if ((a3 = a2[w2]) != null) + for (; w3 != LENGTH3; ++w3) + if ((nword = ~a3[w3]) != 0) + break loop; + w3 = 0; + } + w2 = w3 = 0; + } + } + final int result = (((w1 << SHIFT1) + (w2 << SHIFT2) + w3) << SHIFT3) + + Long.numberOfTrailingZeros(nword); + return (result == Integer.MAX_VALUE ? -1 : result); + } + + /** + * Returns the index of the first bit that is set to true that + * occurs on or after the specified starting index. If no such it exists then + * -1 is returned. + *

+ * To iterate over the true bits in a SparseBitSet + * sbs, use the following loop: + * + *

+     *  for( int i = sbbits.nextSetBit(0); i >= 0; i = sbbits.nextSetBit(i+1) )
+     *  {
+     *      // operate on index i here
+     *  }
+ * + * @param i the index to start checking from (inclusive) + * @return the index of the next set bit + * @exception IndexOutOfBoundsException if the specified index is negative + * @since 1.6 + */ + public int nextSetBit(int i) + { + /* The index value (i) of this method is permitted to be Integer.MAX_VALUE, + as this is needed to make the loop defined above work: just in case the + bit labelled Integer.MAX_VALUE-1 is set. This case is not optimised: + but eventually -1 will be returned, as this will be included with + any search that goes off the end of the level1 array. */ + + if (i < 0) + throw new IndexOutOfBoundsException("i=" + i); + /* This is the word from which the search begins. */ + int w = i >> SHIFT3; + int w3 = w & MASK3; + int w2 = (w >> SHIFT2) & MASK2; + int w1 = w >> SHIFT1; + + long word = 0L; + final int aLength = bits.length; + + long[][] a2; + long[] a3; + /* Is the next set bit in the same word at the nominated beginning bit + (including the nominated beginning bit itself). The first check is + whether the starting bit is within the structure at all. */ + if (w1 < aLength && ((a2 = bits[w1]) == null + || (a3 = a2[w2]) == null + || ((word = a3[w3] & (~0L << i)) == 0L))) + { + /* So now start a search though the rest of the entries for a bit. */ + ++w; + w3 = w & MASK3; + w2 = (w >> SHIFT2) & MASK2; + w1 = w >> SHIFT1; + major: for (; w1 != aLength; ++w1) + { + if ((a2 = bits[w1]) != null) + for (; w2 != LENGTH2; ++w2) + { + if ((a3 = a2[w2]) != null) + for (; w3 != LENGTH3; ++w3) + if ((word = a3[w3]) != 0) + break major; + w3 = 0; + } + w2 = w3 = 0; + } + } + return (w1 >= aLength ? -1 + : (((w1 << SHIFT1) + (w2 << SHIFT2) + w3) << SHIFT3) + + Long.numberOfTrailingZeros(word)); + } + + /** + * Performs a logical OR of the addressed target bit with the + * argument value. This bit set is modified so that the addressed bit has the + * value true if and only if it both initially had the value + * true or the argument value is true. + * + * @param i a bit index + * @param value a boolean value to OR with that bit + * @exception IndexOutOfBoundsException if the specified index is negative + * or equal to Integer.MAX_VALUE + * @since 1.6 + */ + public void or(int i, boolean value) + { + if ((i + 1) < 1) + throw new IndexOutOfBoundsException("i=" + i); + if (value) + set(i); + } + + /** + * Performs a logical OR of the addressed target bit with the + * argument value within the given range. This bit set is modified so that + * within the range a bit in it has the value true if and only if + * it either already had the value true or the corresponding bit + * in the bit set argument has the value true. Outside the range + * this set is not changed. + * + * @param i index of the first bit to be included in the operation + * @param j index after the last bit to included in the operation + * @param b the SparseBitSet with which to perform the OR + * operation with this SparseBitSet + * @exception IndexOutOfBoundsException if i is negative or + * equal to Integer.MAX_VALUE, or j is negative, + * or i is larger than j + * @since 1.6 + */ + public void or(int i, int j, SparseBitSet b) throws IndexOutOfBoundsException + { + setScanner(i, j, b, orStrategy); + } + + /** + * Performs a logical OR of this bit set with the bit set argument. + * This bit set is modified so that a bit in it has the value true + * if and only if it either already had the value true or the + * corresponding bit in the bit set argument has the value true. + * + * @param b the SparseBitSet with which to perform the OR + * operation with this SparseBitSet + * @since 1.6 + */ + public void or(SparseBitSet b) + { + setScanner(0, b.bitsLength, b, orStrategy); + } + + /** + * Performs a logical OR of the two given SparseBitSets. + * The returned SparseBitSet is created so that a bit in it has + * the value true if and only if it either had the value + * true in the set given by the first arguemetn or had the value + * true in the second argument, otherwise false. + * + * @param a a SparseBitSet + * @param b another SparseBitSet + * @return new SparseBitSet representing the OR of the two sets + * @since 1.6 + */ + public static SparseBitSet or(SparseBitSet a, SparseBitSet b) + { + final SparseBitSet result = a.clone(); + result.or(b); + return result; + } + + /** + * Sets the bit at the specified index. + * + * @param i a bit index + * @exception IndexOutOfBoundsException if the specified index is negative + * or equal to Integer.MAX_VALUE + * @since 1.6 + */ + public void set(int i) + { + if ((i + 1) < 1) + throw new IndexOutOfBoundsException("i=" + i); + final int w = i >> SHIFT3; + final int w1 = w >> SHIFT1; + final int w2 = (w >> SHIFT2) & MASK2; + + if (i >= bitsLength) + resize(i); + long[][] a2; + if ((a2 = bits[w1]) == null) + a2 = bits[w1] = new long[LENGTH2][]; + long[] a3; + if ((a3 = a2[w2]) == null) + a3 = a2[w2] = new long[LENGTH3]; + a3[w & MASK3] |= 1L << i; + cache.hash = 0; //Invalidate size, etc., scan + } + + /** + * Sets the bit at the specified index to the specified value. + * + * @param i a bit index + * @param value a boolean value to set + * @exception IndexOutOfBoundsException if the specified index is negative + * or equal to Integer.MAX_VALUE + * @since 1.6 + */ + public void set(int i, boolean value) + { + if (value) + set(i); + else + clear(i); + } + + /** + * Sets the bits from the specified i (inclusive) to the specified + * j (exclusive) to true. + * + * @param i index of the first bit to be set + * @param j index after the last bit to be se + * @exception IndexOutOfBoundsException if i is negative or is + * equal to Integer.MAX_INT, or j is negative, or + * i is larger than j. + * @since 1.6 + */ + public void set(int i, int j) throws IndexOutOfBoundsException + { + setScanner(i, j, null, setStrategy); + } + + /** + * Sets the bits from the specified i (inclusive) to the specified + * j (exclusive) to the specified value. + * + * @param i index of the first bit to be set + * @param j index after the last bit to be set + * @param value to which to set the selected bits + * @exception IndexOutOfBoundsException if i is negative or is + * equal to Integer.MAX_VALUE, or j is negative, or + * i is larger than j + * @since 1.6 + */ + public void set(int i, int j, boolean value) + { + if (value) + set(i, j); + else + clear(i, j); + } + + /** + * Returns the number of bits of space nominally in use by this + * SparseBitSet to represent bit values. The count of bits in + * the set is the (label of the last set bit) + 1 - (the label of the first + * set bit). + * + * @return the number of bits (true and false) nominally in this bit set + * at this moment + * @since 1.6 + */ + public int size() + { + statisticsUpdate(); + return cache.size; + } + + /** + * Convenience method for statistics if the individual results are not needed. + * + * @return a String detailing the statistics of the bit set + * @see #statistics(String[]) + * @since 1.6 + */ + public String statistics() + { + return statistics(null); + } + + /** + * Determine, and create a String with the bit set statistics. The statistics + * include: Size, Length, Cardinality, Total words (i.e., the total + * number of 64-bit "words"), Set array length (i.e., the number of + * references that can be held by the top level array, Level2 areas in use, + * Level3 blocks in use,, Level2 pool size, Level3 pool size, and the + * Compaction count. + *

+ * This method is intended for diagnostic use (as it is relatively expensive + * in time), but can be useful in understanding an application's use of a + * SparseBitSet. + * + * @param values an array for the individual results (if not null) + * @return a String detailing the statistics of the bit set + * @since 1.6 + */ + public String statistics(String[] values) + { + statisticsUpdate(); // Ensure statistics are up-to-date + String[] v = new String[Statistics.values().length]; + + /* Assign the statistics values to the appropriate entry. The order + of the assignments does not matter--the ordinal serves to get the + values into the matching order with the labels from the enumeration. */ + v[Statistics.Size.ordinal()] = Integer.toString(size()); + v[Statistics.Length.ordinal()] = Integer.toString(length()); + v[Statistics.Cardinality.ordinal()] = Integer.toString(cardinality()); + v[Statistics.Total_words.ordinal()] = Integer.toString(cache.count); + v[Statistics.Set_array_length.ordinal()] = Integer.toString(bits.length); + v[Statistics.Set_array_max_length.ordinal()] = + Integer.toString(MAX_LENGTH1); + v[Statistics.Level2_areas.ordinal()] = Integer.toString(cache.a2Count); + v[Statistics.Level2_area_length.ordinal()] = Integer.toString(LENGTH2); + v[Statistics.Level3_blocks.ordinal()] = Integer.toString(cache.a3Count); + v[Statistics.Level3_block_length.ordinal()] = Integer.toString(LENGTH3); + v[Statistics.Compaction_count_value.ordinal()] = + Integer.toString(compactionCount); + + /* Determine the longest label, so that the equal signs may be lined-up. */ + int longestLabel = 0; + for (Statistics s : Statistics.values()) + longestLabel = + Math.max(longestLabel, s.name().length()); + + /* Build a String that has for each statistic, the name of the statistic, + padding, and equals sign, and the value. The "Load_factor_value", + "Average_length_value", and "Average_chain_length" are printed as + floating point values. */ + final StringBuilder result = new StringBuilder(); + for (Statistics s : Statistics.values()) + { + result.append(s.name()); // The name of the statistic + for (int i = 0; i != longestLabel - s.name().length(); ++i) + result.append(' '); // Fill out the field + result.append(" = "); // Show an equals sign + result.append(v[s.ordinal()]); // and a value + result.append('\n'); + } + /* Remove the underscores. */ + for (int i = 0; i != result.length(); ++i) + if (result.charAt(i) == '_') + result.setCharAt(i, ' '); + + if (values != null) + { + final int len = Math.min(values.length, v.length); + System.arraycopy(v, 0, values, 0, len); + } + return result.toString(); + } + + /** + * Returns a string representation of this bit set. For every index for which + * this SparseBitSet contains a bit in the set state, the decimal + * representation of that index is included in the result. Such indices are + * listed in order from lowest to highest. If there is a subsequence of set + * bits longer than the value given by toStringCompaction, the subsequence + * is represented by the value for the first and the last values, with ".." + * between them. The individual bits, or the representation of sub-sequences + * are separated by ", " (a comma and a space) and surrounded by braces, + * resulting in a compact string showing (a variant of) the usual mathematical + * notation for a set of integers. + *
+ * Example (with the default value of 2 for subsequences): + *

+     *      SparseBitSet drPepper = new SparseBitSet();
+     *  
+ * Now drPepper.toString() returns "{}". + *
+ *
+     *      drPepper.set(2);
+     *  
+ * Now drPepper.toString() returns "{2}". + *
+ *
+     *      drPepper.set(3, 4);
+     *      drPepper.set(10);
+     *  
+ * Now drPepper.toString() returns "{2..4, 10}". + *
+ * This method is intended for diagnostic use (as it is relatively expensive + * in time), but can be useful in interpreting problems in an application's use + * of a SparseBitSet. + * + * @return a String representation of this SparseBitSet + * @see #toStringCompaction(int length) + * @since 1.6 + */ + @Override + public String toString() + { + final StringBuilder p = new StringBuilder(200); + p.append('{'); + int i = nextSetBit(0); + /* Loop so long as there is another bit to append to the String. */ + while (i >= 0) + { + /* Append that next bit */ + p.append(i); + /* Find the position of the next bit to show. */ + int j = nextSetBit(i + 1); + if (compactionCount > 0) + { + /* Give up if there is no next bit to show. */ + if (j < 0) + break; + /* Find the next clear bit is after the current bit, i.e., i */ + int last = nextClearBit(i); + /* Compute the position of the next clear bit after the current + subsequence of set bits. */ + last = (last < 0 ? Integer.MAX_VALUE : last); + /* If the subsequence is more than the specified bits long, then + collapse the subsequence into one entry in the String. */ + if (i + compactionCount < last) + { + p.append("..").append(last - 1); + /* Having accounted for a subsequence of bits that are all set, + recompute the label of the next bit to show. */ + j = nextSetBit(last); + } + } + /* If there is another set bit, put a comma and a space after the + last entry in the String. */ + if (j >= 0) + p.append(", "); + /* Transfer to i the index of the next set bit. */ + i = j; + } + /* Terminate the representational String, and return it. */ + p.append('}'); + return p.toString(); + } + + /** Sequences of set bits longer than this value are shown by + * {@link #toString()} as a "sub-sequence," in the form a..b. + * Setting this value to zero causes each set bit to be listed individually. + * The default default value is 2 (which means sequences of three or more + * bits set are shown as a subsequence, and all other set bits are listed + * individually). + *

+ * Note: this value will be passed to SparseBitSets that + * may be created within or as a result of the operations on this bit set, + * or, for static methods, from the value belonging to the first parameter. + * + * @param count the maximum count of a run of bits that are shown as + * individual entries in a toString() conversion. + * If 0, all bits are shown individually. + * @since 1.6 + * @see #toString() + */ + public void toStringCompaction(int count) + { + compactionCount = count; + } + + /** + * If change is true, the current value of the + * toStringCompaction() value is made the default value for all + * SparseBitSets created from this point onward in this JVM. + * + * @param change if true, change the default value + * @since 1.6 + */ + public void toStringCompaction(boolean change) + { + /* This is an assignment to a static value: the integer value assignement + is atomic, so there will not be a partial store. If multiple + invocations are made from multiple threads, there is a race + condition that cannot be resolved by synchronization. */ + if (change) + compactionCountDefault = compactionCount; + } + + /** + * Performs a logical XOR of the addressed target bit with the + * argument value. This bit set is modified so that the addressed bit has the + * value true if and only one of the following statements holds: + *

    + *
  • The addressed bit initially had the value true, and the + * value of the argument is false. + *
  • The bit initially had the value false, and the + * value of the argument is true. + *
+ * + * @param i a bit index + * @param value a boolean value to XOR with that bit + * @exception java.lang.IndexOutOfBoundsException if the specified index + * is negative + * or equal to Integer.MAX_VALUE + * @since 1.6 + */ + public void xor(int i, boolean value) + { + if ((i + 1) < 1) + throw new IndexOutOfBoundsException("i=" + i); + if (value) + flip(i); + } + + /** + * Performs a logical XOR of this bit set with the bit set argument + * within the given range. This resulting bit set is computed so that a bit + * within the range in it has the value true if and only if one + * of the following statements holds: + *
    + *
  • The bit initially had the value true, and the + * corresponding bit in the argument set has the value false. + *
  • The bit initially had the value false, and the + * corresponding bit in the argument set has the value true. + *
+ * Outside the range this set is not changed. + * + * @param i index of the first bit to be included in the operation + * @param j index after the last bit to included in the operation + * @param b the SparseBitSet with which to perform the XOR + * operation with this SparseBitSet + * @exception IndexOutOfBoundsException if i is negative or + * equal to Integer.MAX_VALUE, or j is negative, + * or i is larger than j + * @since 1.6 + */ + public void xor(int i, int j, SparseBitSet b) throws IndexOutOfBoundsException + { + setScanner(i, j, b, xorStrategy); + } + + /** + * Performs a logical XOR of this bit set with the bit set argument. + * This resulting bit set is computed so that a bit in it has the value + * true if and only if one of the following statements holds: + *
    + *
  • The bit initially had the value true, and the + * corresponding bit in the argument set has the value false. + *
  • The bit initially had the value false, and the + * corresponding bit in the argument set has the value true. + *
+ * + * @param b the SparseBitSet with which to perform the XOR + * operation with thisSparseBitSet + * @since 1.6 + */ + public void xor(SparseBitSet b) + { + setScanner(0, b.bitsLength, b, xorStrategy); + } + + /** + * Performs a logical XOR of the two given SparseBitSets. + * The resulting bit set is created so that a bit in it has the value + * true if and only if one of the following statements holds: + *
    + *
  • A bit in the first argument has the value true, and the + * corresponding bit in the second argument has the value + * false.
  • + *
  • A bit in the first argument has the value false, and the + * corresponding bit in the second argument has the value + * true.
+ * + * @param a a SparseBitSet + * @param b another SparseBitSet + * @return a new SparseBitSet representing the XOR of the two sets + * @since 1.6 + */ + public static SparseBitSet xor(SparseBitSet a, SparseBitSet b) + { + final SparseBitSet result = a.clone(); + result.xor(b); + return result; + } + + //============================================================================== + // Internal methods + //============================================================================== + /** + * Throw the exception to indicate a range error. The String + * constructed reports all the possible errors in one message. + * + * @param i lower bound for a operation + * @param j upper bound for a operation + * @exception IndexOutOfBoundsException indicating the range is not valid + * @since 1.6 + */ + protected static final void throwIndexOutOfBoundsException(int i, int j) + throws IndexOutOfBoundsException + { + String s = ""; + if (i < 0) + s += "(i=" + i + ") < 0"; + if (i == Integer.MAX_VALUE) + s += "(i=" + i + ")"; + if (j < 0) + s += (s.length() == 0 ? "" : ", ") + "(j=" + j + ") < 0"; + if (i > j) + s += (s.length() == 0 ? "" : ", ") + "(i=" + i + ") > (j=" + j + ")"; + throw new IndexOutOfBoundsException(s); + } + + /** + * Intializes all the additional objects required for correct operation. + * + * @since 1.6 + */ + protected final void constructorHelper() + { + spare = new long[LENGTH3]; + cache = new Cache(); + andStrategy = new AndStrategy(); + andNotStrategy = new AndNotStrategy(); + clearStrategy = new ClearStrategy(); + equalsStrategy = new EqualsStrategy(); + flipStrategy = new FlipStrategy(); + copyStrategy = new CopyStrategy(); + intersectsStrategy = new IntersectsStrategy(); + orStrategy = new OrStrategy(); + setStrategy = new SetStrategy(); + updateStrategy = new UpdateStrategy(); + xorStrategy = new XorStrategy(); + } + + /** + * Clear out a part of the set array with nulls, from the given start to the + * end of the array. If the given parameter is beyond the end of the bits + * array, nothing is changed. + * + * @param start word index at which to start (inclusive) + * @since 1.6 + */ + protected final void nullify(int start) + { + final int aLength = bits.length; + if (start < aLength) + { + for (int w = start; w != aLength; ++w) + bits[w] = null; + cache.hash = 0; // Invalidate size, etc., values + } + } + + /** + * Resize the bit array. Moves the entries in the the bits array of this + * SparseBitSet into an array whose size (which may be larger or smaller) is + * the given bit size (i.e., includes the bit whose index is one less + * that the given value). If the new array is smaller, the excess entries in + * the set array are discarded. If the new array is bigger, it is filled with + * nulls. + * + * @param index the desired address to be included in the set + * @since 1.6 + */ + protected final void resize(int index) + { + /* Find an array size that is a power of two that is as least as large + enough to contain the index requested. */ + final int w1 = (index >> SHIFT3) >> SHIFT1; + int newSize = Integer.highestOneBit(w1); + if (newSize == 0) + newSize = 1; + if (w1 >= newSize) + newSize <<= 1; + if (newSize > MAX_LENGTH1) + newSize = MAX_LENGTH1; + final int aLength1 = (bits != null ? bits.length : 0); + + if (newSize != aLength1) + { // only if the size needs to be changed + final long[][][] temp = new long[newSize][][]; // Get the new array + if (aLength1 != 0) + { + /* If it exists, copy old array to the new array. */ + System.arraycopy(bits, 0, temp, 0, Math.min(aLength1, newSize)); + nullify(0); // Don't leave unused pointers around. */ + } + bits = temp; // Set new array as the set array + bitsLength = // Index of last possible bit, plus one. + (newSize == MAX_LENGTH1 ? Integer.MAX_VALUE : newSize * UNIT); + } + } + + /** + * Scans over the bit set (and a second bit set if part of the operation) are + * all performed by this method. The properties and the operation executed + * are defined by a given strategy, which must be derived from the + * AbstractStrategy. The strategy defines how to operate on a + * single word, and on whole words that may or may not constitute a full + * block of words. + * + * @param i the bit (inclusive) at which to start the scan + * @param j the bit (exclusive) at which to stop the scan + * @param b a SparseBitSet, if needed, the second SparseBitSet in the + * operation + * @param op the AbstractStrategy class defining the operation to be + * executed + * @exception IndexOutOfBoundsException + * @since 1.6 + * @see AbstractStrategy + */ + protected final void setScanner(int i, int j, SparseBitSet b, + AbstractStrategy op) throws IndexOutOfBoundsException + { + /* This method has been assessed as having a McCabe cyclomatic + complexity of 47 (i.e., impossibly high). However, given that this + method incorporates all the set scanning logic for all methods + (with the exception of nextSetBit and nextClearBit, which themselves + have high cyclomatic complexities of 13), and is attempting to minimise + execution time (hence deals with processing shortcuts), it cannot be + expected to be simple. In fact, the work of lining up level3 blocks + proceeds step-wise, and each sub-section piece is reasonably + straight-forward. Nevertheless, the number of paths is high, and + caution is advised in attempting to correct anything. */ + + /* Do whatever the strategy needs to get started, and do whatever initial + checking is needed--fail here if needed before much else is done. */ + op.start(b); + if (j < i || (i + 1) < 1) + throwIndexOutOfBoundsException(i, j); + if (i == j) + return; + + /* Get the values of all the short-cut options. */ + final int properties = op.properties(); + final boolean f_op_f_eq_f = (properties & AbstractStrategy.F_OP_F_EQ_F) != 0; + final boolean f_op_x_eq_f = (properties & AbstractStrategy.F_OP_X_EQ_F) != 0; + final boolean x_op_f_eq_f = (properties & AbstractStrategy.X_OP_F_EQ_F) != 0; + final boolean x_op_f_eq_x = (properties & AbstractStrategy.X_OP_F_EQ_X) != 0; + + /* Index of the current word, and mask for the first word, + to be processed in the bit set. */ + int u = i >> SHIFT3; + final long um = ~0L << i; + + /* Index of the final word, and mask for the final word, + to be processed in the bit set. */ + final int v = (j - 1) >> SHIFT3; + final long vm = ~0L >>> -j; + + /* Set up the two bit arrays (if the second exists), and their + corresponding lengths (if any). */ + long[][][] a1 = bits; // Level1, i.e., the bit arrays + int aLength1 = bits.length; + final long[][][] b1 = (b != null ? b.bits : null); + final int bLength1 = (b1 != null ? b.bits.length : 0); + + /* Calculate the initial values of the parts of the words addresses, + as well as the location of the final block to be processed. */ + int u1 = u >> SHIFT1; + int u2 = (u >> SHIFT2) & MASK2; + int u3 = u & MASK3; + final int v1 = v >> SHIFT1; + final int v2 = (v >> SHIFT2) & MASK2; + final int v3 = v & MASK3; + final int lastA3Block = (v1 << LEVEL2) + v2; + + /* Initialize the local copies of the counts of blocks and areas; and + whether there is a partial first block. */ + int a2CountLocal = 0; + int a3CountLocal = 0; + boolean notFirstBlock = u == 0 && um == ~0L; + + /* The first level2 is cannot be judged empty if not being scanned from + the beginning. */ + boolean a2IsEmpty = u2 == 0; // Presumption + while (i < j) + { + /* Determine if there is a level2 area in both the a and the b set, + and if so, set the references to these areas. */ + long[][] a2 = null; + boolean haveA2 = u1 < aLength1 && (a2 = a1[u1]) != null; + long[][] b2 = null; + final boolean haveB2 = u1 < bLength1 + && b1 != null && (b2 = b1[u1]) != null; + /* Handling of level 2 empty areas: determined by the + properties of the strategy. It is necessary to actually visit + the first and last blocks of a scan, since not all of the block + might participate in the operation, hence making decision based + on just the references to the blocks could be wrong. */ + if ((!haveA2 && !haveB2 && f_op_f_eq_f + || !haveA2 && f_op_x_eq_f || !haveB2 && x_op_f_eq_f) + && notFirstBlock && u1 != v1) + {//nested if! + if (u1 < aLength1) + a1[u1] = null; + } + else + { + final int limit2 = (u1 != v1 ? LENGTH2 : v2 + 1); + while (u2 != limit2) + { + /* Similar logic applied here as for the level2 blocks. + The initial and final block must be examined. In other + cases, it may be possible to make a decision based on + the value of the references, as indicated by the + properties of the strategy. */ + long[] a3 = null; + final boolean haveA3 = haveA2 && (a3 = a2[u2]) != null; + long[] b3 = null; + final boolean haveB3 = haveB2 && (b3 = b2[u2]) != null; + final int a3Block = (u1 << LEVEL2) + u2; + final boolean notLastBlock = lastA3Block != a3Block; + /* Handling of level 3 empty areas: determined by the + properties of the strategy. */ + if ((!haveA3 && !haveB3 && f_op_f_eq_f + || !haveA3 && f_op_x_eq_f || !haveB3 && x_op_f_eq_f) + && notFirstBlock && notLastBlock) + { + /* Do not need level3 block, so remove it, and move on. */ + if (haveA2) + a2[u2] = null; + u3 = 0; + } + else + { + /* So what is needed is the level3 block. */ + final int base3 = a3Block << SHIFT2; + final int limit3 = (notLastBlock ? LENGTH3 : v3); + if (!haveA3) + a3 = spare; + if (!haveB3) + b3 = ZERO_BLOCK; + boolean isZero; + if (notFirstBlock && notLastBlock) + if (x_op_f_eq_x && !haveB3) + isZero = op.isZeroBlock(a3); + // b block is null, just check a block + else + isZero = op.block(base3, 0, LENGTH3, a3, b3); + // Do the operation on the whole block + else + { /* Partial block to process. */ + if (notFirstBlock) + { + /* By implication, this is the last block */ + isZero = op.block(base3, 0, limit3, a3, b3); + // Do the whole words + isZero &= op.word(base3, limit3, a3, b3, vm); + // And then the final word + } + else + { // u, v are correct if first block + if (u == v) // Scan starts and ends in one word + isZero = op.word(base3, u3, a3, b3, um & vm); + else + { // Scan starts in this a3 block + isZero = op.word(base3, u3, a3, b3, um); + // First word + isZero &= + op.block(base3, u3 + 1, limit3, a3, b3); + // Remainder of full words in block + if (limit3 != LENGTH3) + isZero &= op.word(base3, limit3, a3, b3, vm); + // If there is a partial word left + } + notFirstBlock = true; // Only one first block + } + if (isZero) + isZero = op.isZeroBlock(a3); + // If not known to have a non-zero + // value, be sure whether all zero. + } + if (isZero) // The resulting a3 block has no values + {// nested if! + /* If there is an level 2 area make the entry for this + level3 block be a null (i.e., remove any a3 block ). */ + if (haveA2) + a2[u2] = null; + } + else + { + /* If the a3 block used was the spare block, put it + into current level2 area; get a new spare block. */ + if (a3 == spare) + { + if (i >= bitsLength) //Check that the set is large + { // enough to take the new block + resize(i); // Make it large enough + a1 = bits; // Update reference and length + aLength1 = a1.length; + } + if (a2 == null) // Ensure a level 2 area + { + a1[u1] = a2 = new long[LENGTH2][]; + haveA2 = true; // Ensure know level2 not empty + } + a2[u2] = a3; // Insert the level3 block + spare = new long[LENGTH3]; // Replace the spare + } + ++a3CountLocal; // Count the level 3 block + } + a2IsEmpty &= !(haveA2 && a2[u2] != null); + } // Keep track of level 2 usage + ++u2; + u3 = 0; + } /* end while ( u2 != limit2 ) */ + /* If the loop finishes without completing the level 2, it may + be left with a reference but still be all null--this is OK. */ + if (u2 == LENGTH2 && a2IsEmpty && u1 < aLength1) + a1[u1] = null; + else + ++a2CountLocal; // Count level 2 areas + } + /* Advance the value of u based on what happened. */ + i = (u = (++u1 << SHIFT1)) << SHIFT3; + u2 = 0; // u3 = 0 + // Compute next word and bit index + if (i < 0) + i = Integer.MAX_VALUE; // Don't go over the end + } /* end while( i < j ) */ + + /* Do whatever the strategy needs in order to finish. */ + op.finish(a2CountLocal, a3CountLocal); + } + + /** + * The entirety of the bit set is examined, and the various statistics of + * the bit set (size, length, cardinality, hashCode, etc.) are computed. Level + * arrays that are empty (i.e., all zero at level 3, all null at level 2) are + * replaced by null references, ensuring a normalized representation. + * + * @since 1.6 + */ + protected final void statisticsUpdate() + { + if (cache.hash != 0) + return; + setScanner(0, bitsLength, null, updateStrategy); + } + + //============================================================================== + // Serialization/Deserialization methods + //============================================================================== + + /** + * Save the state of the SparseBitSet instance to a stream + * (i.e., serialize it). + * + * @param s the ObjectOutputStream to which to write the serialized object + * @exception java.io.IOException if an io error occurs + * @exception java.lang.InternalError if the SparseBitSet representation is + * inconsistent + * + * @serialData The default data is emitted, followed by the current + * compactionCount for the bit set, and then the + * length of the set (the position of the last bit), + * followed by the cache.count value (an int, + * the number of int->long pairs needed to describe + * the set), followed by the index (int) and word + * (long) for each int->long pair. + * The mappings need not be emitted in any particular order. This + * is followed by the hashCode for the set that can be used + * as an integrity check when the bit set is read back. + * + * @since 1.6 + */ + private void writeObject(ObjectOutputStream s) throws IOException, InternalError + { + statisticsUpdate(); // Update structure and stats if needed. + /* Write any hidden stuff. */ + s.defaultWriteObject(); + s.writeInt(compactionCount); // Needed to preserve value + s.writeInt(cache.length); // Needed to know where last bit is + + /* This is the number of index/value pairs to be written. */ + int count = cache.count; // Minimum number of words to be written + s.writeInt(count); + final long[][][] a1 = bits; + final int aLength1 = a1.length; + long[][] a2; + long[] a3; + long word; + for (int w1 = 0; w1 != aLength1; ++w1) + if ((a2 = a1[w1]) != null) + for (int w2 = 0; w2 != LENGTH2; ++w2) + if ((a3 = a2[w2]) != null) + { + final int base = (w1 << SHIFT1) + (w2 << SHIFT2); + for (int w3 = 0; w3 != LENGTH3; ++w3) + if ((word = a3[w3]) != 0) + { + s.writeInt(base + w3); + s.writeLong(word); + --count; + } + } + if (count != 0) + throw new InternalError("count of entries not consistent"); + /* As a consistency check, write the hash code of the set. */ + s.writeInt(cache.hash); + } + + /** + * serialVersionUID + */ + private static final long serialVersionUID = -6663013367427929992L; + + /** + * Reconstitute the SparseBitSet instance from a stream + * (i.e., deserialize it). + * + * @param s the ObjectInputStream to use + * @exception IOException if there is an io error + * @exception ClassNotFoundException if the stream contains an unidentified + * class + * @since 1.6 + */ + private void readObject(ObjectInputStream s) throws IOException, + ClassNotFoundException + { + /* Read in any hidden stuff that is part of the class overhead. */ + s.defaultReadObject(); + compactionCount = s.readInt(); + final int aLength = s.readInt(); + resize(aLength); // Make sure there is enough space + + /* Read in number of mappings. */ + final int count = s.readInt(); + /* Read the keys and values, them into the set array, areas, and blocks. */ + long[][] a2; + long[] a3; + for (int n = 0; n != count; ++n) + { + final int w = s.readInt(); + final int w3 = w & MASK3; + final int w2 = (w >> SHIFT2) & MASK2; + final int w1 = w >> SHIFT1; + + final long word = s.readLong(); + if ((a2 = bits[w1]) == null) + a2 = bits[w1] = new long[LENGTH2][]; + if ((a3 = a2[w2]) == null) + a3 = a2[w2] = new long[LENGTH3]; + a3[w3] = word; + } + /* Ensure all the pieces are set up for set scanning. */ + constructorHelper(); + statisticsUpdate(); + if (count != cache.count) + throw new InternalError("count of entries not consistent"); + final int hash = s.readInt(); // Get the hashcode that was stored + if (hash != cache.hash) // An error of some kind if not the same + throw new IOException("deserialized hashCode mis-match"); + } + + //============================================================================= + // Statistics enumeration + //============================================================================= + + /** + * These enumeration values are used as labels for the values in the String + * created by the statistics() method. The values of the corresponding + * statistics are ints, except for the loadFactor and + * Average_chain_length values, which are floats. + *

+ * An array of Strings may be obtained containing a + * representation of each of these values. An element of such an array, say, + * values, may be accessed, for example, by: + *

+     *      values[SparseBitSet.statistics.Buckets_available.ordinal()]
+ * + * @see #statistics(String[]) + */ + public static enum Statistics + { + /** + * The size of the bit set, as give by the size() method. + */ + Size, // 0 + /** + * The length of the bit set, as give by the length() method. + */ + Length, // 1 + /** + * The cardinality of the bit set, as give by the cardinality() method. + */ + Cardinality, // 2 + /** + * The total number of non-zero 64-bits "words" being used to hold the + * representation of the bit set. + */ + Total_words, // 3 + /** + * The length of the bit set array. + */ + Set_array_length, // 4 + /** + * The maximum permitted length of the bit set array. + */ + Set_array_max_length, // 5 + /** + * The number of level2 areas. + */ + Level2_areas, // 6 + /** + * The length of the level2 areas. + */ + Level2_area_length, // 7 + /** + * The total number of level3 blocks in use. + */ + Level3_blocks, // 8 + /** + * The length of the level3 blocks. + */ + Level3_block_length, // 9 + /** + * Is the value that determines how the toString() conversion is + * performed. + * @see #toStringCompaction(int) + */ + Compaction_count_value // 10 + } + + //============================================================================= + // A set of cached statistics values, recomputed when necessary + //============================================================================= + /** + * This class holds the values related to various statistics kept about the + * bit set. These values are not kept continuously up-to-date. Whenever the + * values become invalid, the field hash is set to zero, indicating + * that an update is required. + * + * @see #statisticsUpdate() + */ + protected class Cache + { + /** + * hash is updated by the statisticsUpdate() method. + * If the hash value is zero, it is assumed that all + * the cached values are stale, and must be updated. + */ + + protected transient int hash; + + /** + * size is updated by the statisticsUpdate() method. + * If the hash value is zero, it is assumed the all the cached + * values are stale, and must be updated. + */ + protected transient int size; + + /** + * cardinality is updated by the statisticsUpdate() method. + * If the hash value is zero, it is assumed the all the cached + * values are stale, and must be updated. + */ + protected transient int cardinality; + + /** + * length is updated by the statisticsUpdate() method. + * If the hash value is zero, it is assumed the all the cached + * values are stale, and must be updated. + */ + protected transient int length; + + /** + * count is updated by the statisticsUpdate() method. + * If the hash value is zero, it is assumed the all the cached + * values are stale, and must be updated. + */ + protected transient int count; + + /** + * a2Count is updated by the statisticsUpdate() + * method, and will only be correct immediately after a full update. The + * hash value is must be zero for all values to be updated. + */ + protected transient int a2Count; + + /** + * a3Count is updated by the statisticsUpdate() method, + * and will only be correct immediately after a full update. The + * hash value is must be zero for all values to be updated. + */ + protected transient int a3Count; + } + + //============================================================================= + // Abstract Strategy super-class for Strategies describing logical operations + //============================================================================= + /** + * This strategy class is used by the setScanner to carry out the a variety + * of operations on this set, and usually a second set. The + * setScanner() method of the main SparseBitSet class + * essentially finds matching level3 blocks, and then calls the strategy to + * do the appropriate operation on each of the elements of the block. + *

+ * The symbolic constants control optimisation paths in the + * setScanner() method of the main SparseBitSet class. + * + * @see SparseBitSet#setScanner(int i, int j, + * SparseBitSet b, AbstractStrategy op) + */ + protected abstract class AbstractStrategy + { + /** If the operation requires that when matching level2 areas or level3 + * blocks are null, that no action is required, then this property is + * required. Corresponds to the top-left entry in the logic diagram for the + * operation being 0. For all the defined actual logic operations ('and', + * 'andNot', 'or', and 'xor', this will be true, because for all these, + * "false" op "false" = "false". + */ + static final int F_OP_F_EQ_F = 0x1; + + /** If when level2 areas or level3 areas from the this set are null will + * require that area or block to remain null, irrespective of the value of + * the matching structure from the other set, then this property is required. + * Corresponds to the first row in the logic diagram being all zeros. For + * example, this is true for 'and' as well as 'andNot', and for 'clear', since + * false" & "x" = "false", and "false" &! "x" = "false". + */ + static final int F_OP_X_EQ_F = 0x2; + + /** If when level2 areas or level3 areas from the other set are null will + * require the matching area or block in this set to be set to null, + * irrespective of the current values in the matching structure from the + * this, then this property is required. Corresponds to the first column + * in the logic diagram being all zero. For example, this is true for + * 'and', since "x" & "false" = "false", as well as for 'clear'. + */ + static final int X_OP_F_EQ_F = 0x4; + + /** If when a level3 area from the other set is null will require the + * matching area or block in this set to be left as it is, then this property + * is required. Corresponds to the first column of the logic diagram being + * equal to the left hand operand column. For example, this is true for 'or', + * 'xor', and 'andNot', since for all of these "x" op "false" = "x". + */ + static final int X_OP_F_EQ_X = 0x8; + + /** + * Properties of this strategy. + * + * @return the int containing the bits representing the properties of + * this strategy + * @since 1.6 + */ + protected abstract int properties(); + + /** + * Instances of this class are to be serially reusable. To start a + * particular use, an instance is (re-)started by calling this method. It is + * passed the reference to the other bit set (usually to allow a check on + * whether it is null or not, so as to simplify the implementation of the + * block() method. + * + * @param b the "other" set, for whatever checking is needed. + * @since 1.6 + */ + protected abstract void start(SparseBitSet b); + + /** + * Deal with a scan that include a partial word within a level3 block. All + * that is required is that the result be stored (if needed) into the + * given a set block at the correct position, and that the operation only + * affect those bits selected by 1 bits in the mask. + * + * @param base the base index of the block (to be used if needed) + * @param u3 the index of the word within block + * @param a3 the level3 block from the a set. + * @param b3 the (nominal) level3 block from the b set (not null). + * @param mask for the (partial) word + * @return true if the resulting word is zero + * @since 1.6 + */ + + protected abstract boolean word(int base, int u3, long[] a3, long[] b3, long mask); + + /** + * Deals with a part of a block that consists of whole words, starting with + * the given first index, and ending with the word before the last index. + * For the words processed, the return value should indicate whether all those + * resulting words were zero, or not. + * + * @param base the base index of the block (to be used if needed) + * @param u3 the index of the first word within block to process + * @param v3 the index of the last word, which may be within block + * @param a3 the level3 block from the a set. + * @param b3 the (nominal) level3 block from the b set (not null). + * @return true if the words scanned within the level3 block were all zero + * @since 1.6 + */ + protected abstract boolean block(int base, int u3, int v3, long[] a3, long[] b3); + + /** + * This is called to finish the processing started by the strategy (if there + * needs to be anything done at all). + * + * @param a2Count possible count of level2 areas in use + * @param a3Count possible count of level3 blocks in use + * @since 1.6 + */ + protected void finish(int a2Count, int a3Count) + { + } + + /** + * Check whether a level3 block is all zero. + * + * @param a3 the block from the a set + * @return true if the values of the level3 block are all zero + * + * @since 1.6 + */ + protected final boolean isZeroBlock(long[] a3) + { + for (long word : a3) + if (word != 0L) + return false; + return true; + } + } + + //============================================================================= + // Strategies based on the Strategy super-class describing logical operations + //============================================================================= + /** + * And of two sets. Where the a set is zero, it remains zero (i.e., + * without entries or with zero words). Similarly, where the b set is + * zero, the a becomes zero (i.e., without entries). + *

+ * If level1 of the a set is longer than level1 of the bit set + * b, then the unmatched virtual "entries" of the b set (beyond + * the actual length of b) corresponding to these are all false, hence + * the result of the "and" operation will be to make all these entries in this + * set to become false--hence just remove them, and then scan only those + * entries that could match entries in the bit setb. This clearing of + * the remainder of the a set is accomplished by selecting both + * F_OP_X_EQ_F and X_OP_F_EQ_F. + * + *

+     *  and| 0 1
+     *    0| 0 0
+     *    1| 0 1 
+     */
+    protected class AndStrategy extends AbstractStrategy
+    {
+        @Override
+        //  AndStrategy
+        protected int properties()
+        {
+            return F_OP_F_EQ_F + F_OP_X_EQ_F + X_OP_F_EQ_F;
+        }
+
+        @Override
+        //  AndStrategy
+        protected void start(SparseBitSet b)
+        {
+            if (b == null)
+                throw new NullPointerException();
+            cache.hash = 0;
+        }
+
+        @Override
+        //  AndStrategy
+        protected boolean word(int base, int u3, long[] a3, long[] b3, long mask)
+        {
+            return (a3[u3] &= b3[u3] | ~mask) == 0L;
+        }
+
+        @Override
+        //  AndStrategy
+        protected boolean block(int base, int u3, int v3, long[] a3, long[] b3)
+        {
+            boolean isZero = true; //  Presumption
+            for (int w3 = u3; w3 != v3; ++w3)
+                isZero &= ((a3[w3] &= b3[w3]) == 0L);
+            return isZero;
+        }
+    }
+
+    //-----------------------------------------------------------------------------
+    /**
+     *  AndNot of two sets. Where the a set is zero, it remains zero
+     *  (i.e., without entries or with zero words). On the other hand, where the
+     *  b set is zero, the a remains unchanged.
+     *  

+ * If level1 of the a set is longer than level1 of the bit set + * b, then the unmatched virtual "entries" of the b set (beyond + * the actual length of b) corresponding to these are all false, hence + * the result of the "and" operation will be to make all these entries in this + * set to become false--hence just remove them, and then scan only those + * entries that could match entries in the bit setb. This clearing of + * the remainder of the a set is accomplished by selecting both + * F_OP_X_EQ_F and X_OP_F_EQ_F. + * + *

+     * andNot| 0 1
+     *      0| 0 0
+     *      1| 1 0 
+     */
+    protected class AndNotStrategy extends AbstractStrategy
+    {
+        @Override
+        //  AndNotStrategy
+        protected int properties()
+        {
+            return F_OP_F_EQ_F + F_OP_X_EQ_F + X_OP_F_EQ_X;
+        }
+
+        @Override
+        //  AndNotStrategy
+        protected void start(SparseBitSet b)
+        {
+            if (b == null)
+                throw new NullPointerException();
+            cache.hash = 0;
+        }
+
+        @Override
+        //  AndNotStrategy
+        protected boolean word(int base, int u3, long[] a3, long[] b3, long mask)
+        {
+            return (a3[u3] &= ~(b3[u3] & mask)) == 0L;
+        }
+
+        @Override
+        //  AndNotStrategy
+        protected boolean block(int base, int u3, int v3, long[] a3, long[] b3)
+        {
+            boolean isZero = true; //  Presumption
+            for (int w3 = u3; w3 != v3; ++w3)
+                isZero &= (a3[w3] &= ~b3[w3]) == 0L;
+            return isZero;
+        }
+    }
+
+    //-----------------------------------------------------------------------------
+    /**
+     *  Clear clears bits in the a set.
+     *
+     * 
+     * clear| 0 1
+     *     0| 0 0
+     *     1| 0 0 
+     */
+    protected class ClearStrategy extends AbstractStrategy
+    {
+        @Override
+        //  ClearStrategy
+        protected int properties()
+        {
+            return F_OP_F_EQ_F + F_OP_X_EQ_F;
+        }
+
+        @Override
+        //  ClearStrategy
+        protected void start(SparseBitSet b)
+        {
+            cache.hash = 0;
+        }
+
+        @Override
+        //  ClearStrategy
+        protected boolean word(int base, int u3, long[] a3, long[] b3, long mask)
+        {
+            return (a3[u3] &= ~mask) == 0L;
+        }
+
+        @Override
+        //  ClearStrategy
+        protected boolean block(int base, int u3, int v3, long[] a3, long[] b3)
+        {
+            if (u3 != 0 || v3 != LENGTH3) //  Optimisation
+                for (int w3 = u3; w3 != v3; ++w3)
+                    a3[w3] = 0L;
+            return true;
+        }
+    }
+
+    //-----------------------------------------------------------------------------
+    /**
+     *  Copies the needed parts of the b set to the a set.
+     *
+     * 
+     * get| 0 1
+     *   0| 0 1
+     *   1| 0 1 
+     */
+    protected class CopyStrategy extends AbstractStrategy
+    {
+        @Override
+        //  CopyStrategy
+        protected int properties()
+        {
+            return F_OP_F_EQ_F + X_OP_F_EQ_F;
+        }
+
+        @Override
+        //  CopyStrategy
+        protected void start(SparseBitSet b)
+        {
+            cache.hash = 0;
+        }
+
+        @Override
+        //  CopyStrategy
+        protected boolean word(int base, int u3, long[] a3, long[] b3, long mask)
+        {
+            return (a3[u3] = b3[u3] & mask) == 0L;
+        }
+
+        @Override
+        //  CopyStrategy
+        protected boolean block(int base, int u3, int v3, long[] a3, long[] b3)
+        {
+            boolean isZero = true;
+            for (int w3 = u3; w3 != v3; ++w3)
+                isZero &= (a3[w3] = b3[w3]) == 0L;
+            return isZero;
+        }
+    }
+
+    //-----------------------------------------------------------------------------
+    /**
+     *  Equals compares bits in the a set with those in the b set.
+     *  None of the values in either set are changed, although the a set
+     *  may have all zero level 3 blocks replaced by null references (and
+     *  similarly at level 2).
+     *
+     * 
+     * equals| 0 1
+     *      0| 0 -
+     *      1| - - 
+     */
+    protected class EqualsStrategy extends AbstractStrategy
+    {
+        boolean result; // Used to hold result of the comparison
+
+        @Override
+        //  EqualsStrategy
+        protected int properties()
+        {
+            return F_OP_F_EQ_F;
+        }
+
+        @Override
+        //  EqualsStrategy
+        protected void start(SparseBitSet b)
+        {
+            if (b == null)
+                throw new InternalError();
+            result = true;
+            /*  Equals does not change the content of the set, hence hash need
+                not be reset. */
+        }
+
+        @Override
+        //  EqualsStrategy
+        protected boolean word(int base, int u3, long[] a3, long[] b3, long mask)
+        {
+            final long word = a3[u3];
+            result &= (word & mask) == (b3[u3] & mask);
+            return word == 0L;
+        }
+
+        @Override
+        //  EqualsStrategy
+        protected boolean block(int base, int u3, int v3, long[] a3, long[] b3)
+        {
+
+            boolean isZero = true; //  Presumption
+            for (int w3 = u3; w3 != v3; ++w3)
+            {
+                final long word = a3[w3];
+                result &= word == b3[w3];
+                isZero &= word == 0L;
+            }
+            return isZero;
+        }
+    }
+
+    //-----------------------------------------------------------------------------
+    /**
+     *  Flip inverts the bits of the a set within the given range.
+     *
+     * 
+     * flip| 0 1
+     *    0| 1 1
+     *    1| 0 0 
+     */
+    protected class FlipStrategy extends AbstractStrategy
+    {
+        @Override
+        // FlipStrategy
+        protected int properties()
+        {
+            return 0;
+        }
+
+        @Override
+        // FlipStrategy
+        protected void start(SparseBitSet b)
+        {
+            cache.hash = 0;
+        }
+
+        @Override
+        // FlipStrategy
+        protected boolean word(int base, int u3, long[] a3, long[] b3, long mask)
+        {
+            return (a3[u3] ^= mask) == 0L;
+        }
+
+        @Override
+        // FlipStrategy
+        protected boolean block(int base, int u3, int v3, long[] a3, long[] b3)
+        {
+            boolean isZero = true; //  Presumption
+            for (int w3 = u3; w3 != v3; ++w3)
+                isZero &= (a3[w3] ^= ~0L) == 0L;
+            return isZero;
+        }
+    }
+
+    //-----------------------------------------------------------------------------
+    /**
+     *  Intersect has a true result if any word in the a set has a bit
+     *  in common with the b set. During the scan of the a set
+     *  blocks (and areas) that are all zero may be replaced with empty blocks
+     *  and areas (null references), but the value of the set is not changed
+     *  (which is why X_OP_F_EQ_F is not selected, since this would cause
+     *  parts of the a set to be zero-ed out).
+     *
+     * 
+     * intersect| 0 1
+     *         0| 0 0
+     *         1| 1 1 
+     */
+    protected class IntersectsStrategy extends AbstractStrategy
+    {
+        /**
+         *  The boolean result of the intersects scan Strategy is kept here.
+         */
+        protected boolean result;
+
+        @Override
+        //  IntersectsStrategy
+        protected int properties()
+        {
+            return F_OP_F_EQ_F + F_OP_X_EQ_F;
+        }
+
+        @Override
+        //  IntersectsStrategy
+        protected void start(SparseBitSet b)
+        {
+            if (b == null)
+                throw new NullPointerException();
+            result = false;
+            /*  Intersect does not change the content of the set, hence hash
+                need not be reset. */
+        }
+
+        @Override
+        //  IntersectsStrategy
+        protected boolean word(int base, int u3, long[] a3, long[] b3, long mask)
+        {
+            final long word = a3[u3];
+            result |= (word & b3[u3] & mask) != 0L;
+            return word == 0L;
+        }
+
+        @Override
+        //  IntersectsStrategy
+        protected boolean block(int base, int u3, int v3, long[] a3, long[] b3)
+        {
+            boolean isZero = true; //  Presumption
+            for (int w3 = u3; w3 != v3; ++w3)
+            {
+                final long word = a3[w3];
+                result |= (word & b3[w3]) != 0L;
+                isZero &= word == 0L;
+            }
+            return isZero;
+        }
+    }
+
+    /**
+     *  Or of two sets. Where the a set is one, it remains one. Similarly,
+     *  where the b set is one, the a becomes one. If both sets have
+     *  zeros in corresponding places, a zero results. Whole blocks or areas that
+     *  are or become zero are replaced by null arrays.
+     *  

+ * If level1 of the a set is longer than level1 of the bit set + * b, then the unmatched entries of the a set (beyond + * the actual length of b) corresponding to these remain unchanged. * + *

+     *   or| 0 1
+     *    0| 0 1
+     *    1| 1 1 
+     */
+    protected class OrStrategy extends AbstractStrategy
+    {
+        @Override
+        //  OrStrategy
+        protected int properties()
+        {
+            return F_OP_F_EQ_F + X_OP_F_EQ_X;
+        }
+
+        @Override
+        //  OrStrategy
+        protected void start(SparseBitSet b)
+        {
+            if (b == null)
+                throw new NullPointerException();
+            cache.hash = 0;
+        }
+
+        @Override
+        //  OrStrategy
+        protected boolean word(int base, int u3, long[] a3, long[] b3, long mask)
+        {
+            return (a3[u3] |= b3[u3] & mask) == 0L;
+        }
+
+        @Override
+        //  OrStrategy
+        protected boolean block(int base, int u3, int v3, long[] a3, long[] b3)
+        {
+            boolean isZero = true; //  Presumption
+            for (int w3 = u3; w3 != v3; ++w3)
+                isZero &= (a3[w3] |= b3[w3]) == 0L;
+            return isZero;
+        }
+    }
+
+    //-----------------------------------------------------------------------------
+    /**
+     *  Set creates entries everywhere within the range. Hence no empty level2
+     *  areas or level3 blocks are ignored, and no empty (all zero) blocks are
+     *  returned.
+     *
+     *  
+     * set| 0 1
+     *   0| 1 1
+     *   1| 1 1 
+     */
+    protected class SetStrategy extends AbstractStrategy
+    {
+        @Override
+        //  SetStrategy
+        protected int properties()
+        {
+            return 0;
+        }
+
+        @Override
+        //  SetStrategy
+        protected void start(SparseBitSet b)
+        {
+            cache.hash = 0;
+        }
+
+        @Override
+        //  SetStrategy
+        protected boolean word(int base, int u3, long[] a3, long[] b3, long mask)
+        {
+            a3[u3] |= mask;
+            return false;
+        }
+
+        @Override
+        //  SetStrategy
+        protected boolean block(int base, int u3, int v3, long[] a3, long[] b3)
+        {
+            for (int w3 = u3; w3 != v3; ++w3)
+                a3[w3] = ~0L;
+            return false; // set always sets bits
+        }
+    }
+
+    //-----------------------------------------------------------------------------
+    /**
+     *  Update the seven statistics that are computed for each set. These are
+     *  updated by calling statisticsUpdate, which uses this strategy.
+     *
+     *  
+     *  update| 0 1
+     *       0| 0 0
+     *       1| 1 1 
+     *
+     * @see SparseBitSet#statisticsUpdate()
+     */
+    protected class UpdateStrategy extends AbstractStrategy
+    {
+        /**
+         *  Working space for find the size and length of the bit set. Holds the
+         *  index of the first non-empty word in the set.
+         */
+        protected transient int wMin;
+
+        /**
+         *  Working space for find the size and length of the bit set. Holds copy of
+         *  the first non-empty word in the set.
+         */
+        protected transient long wordMin;
+
+        /**
+         *  Working space for find the size and length of the bit set. Holds the
+         *  index of the last non-empty word in the set.
+         */
+        protected transient int wMax;
+
+        /**
+         *  Working space for find the size and length of the bit set. Holds a copy
+         *  of the last non-empty word in the set.
+         */
+        protected transient long wordMax;
+
+        /**
+         *  Working space for find the hash value of the bit set. Holds the
+         *  current state of the computation of the hash value. This value is
+         *  ultimately transferred to the Cache object.
+         *
+         * @see SparseBitSet.Cache
+         */
+        protected transient long hash;
+
+        /**
+         *  Working space for keeping count of the number of non-zero words in the
+         *  bit set. Holds the current state of the computation of the count. This
+         *  value is ultimately transferred to the Cache object.
+         *
+         * @see SparseBitSet.Cache
+         */
+        protected transient int count;
+
+        /**
+         *  Working space for counting the number of non-zero bits in the bit set.
+         *  Holds the current state of the computation of the cardinality.This
+         *  value is ultimately transferred to the Cache object.
+         *
+         * @see SparseBitSet.Cache
+         */
+        protected transient int cardinality;
+
+        @Override
+        //  UpdateStrategy
+        protected int properties()
+        {
+            return F_OP_F_EQ_F + F_OP_X_EQ_F;
+        }
+
+        /**
+         *  This method initializes the computations by suitably resetting cache
+         *  fields or working fields.
+         *
+         * @param       b the other SparseBitSet, for checking if needed.
+         *
+         * @since       1.6
+         */
+        @Override
+        protected void start(SparseBitSet b)
+        {
+            hash = 1234L; // Magic number
+            wMin = -1; // index of first non-zero word
+            wordMin = 0L; // word at that index
+            wMax = 0; // index of last non-zero word
+            wordMax = 0L; // word at that index
+            count = 0; // count of non-zero words in whole set
+            cardinality = 0; // count of non-zero bits in the whole set
+        }
+
+        @Override
+        protected boolean word(int base, int u3, long[] a3, long[] b3, long mask)
+        {
+            final long word = a3[u3];
+            final long word1 = word & mask;
+            if (word1 != 0L)
+                compute(base + u3, word1);
+            return word == 0L;
+        }
+
+        @Override
+        //  UpdateStrategy
+        protected boolean block(int base, int u3, int v3, long[] a3, long[] b3)
+        {
+            boolean isZero = true; //  Presumption
+            for (int w3 = 0; w3 != v3; ++w3)
+            {
+                final long word = a3[w3];
+                if (word != 0)
+                {
+                    isZero = false;
+                    compute(base + w3, word);
+                }
+            }
+            return isZero;
+        }
+
+        @Override
+        //  UpdateStrategy
+        protected void finish(int a2Count, int a3Count)
+        {
+            cache.a2Count = a2Count;
+            cache.a3Count = a3Count;
+            cache.count = count;
+            cache.cardinality = cardinality;
+            cache.length = (wMax + 1) * LENGTH4 - Long.numberOfLeadingZeros(wordMax);
+            cache.size = cache.length - wMin * LENGTH4
+                    - Long.numberOfTrailingZeros(wordMin);
+            cache.hash = (int) ((hash >> Integer.SIZE) ^ hash);
+        }
+
+        /**
+         *  This method does the accumulation of the statistics. It must be called
+         *  in sequential order of the words in the set for which the statistics
+         *  are being accumulated, and only for non-null values of the second
+         *  parameter.
+         *
+         *  Two of the values (a2Count and a3Count) are not updated here,
+         *  but are done in the code near where this method is called.
+         *
+         * @param       index the word index of the word supplied
+         * @param       word the long non-zero word from the set
+         * @since       1.6
+         */
+        private void compute(final int index, final long word)
+        {
+            /*  Count the number of actual words being used. */
+            ++count;
+            /*  Contine to accumulate the hash value of the set. */
+            hash ^= word * (long) (index + 1);
+            /*  The first non-zero word contains the first actual bit of the
+                set. The location of this bit is used to compute the set size. */
+            if (wMin < 0)
+            {
+                wMin = index;
+                wordMin = word;
+            }
+            /*  The last non-zero word contains the last actual bit of the set.
+                The location of this bit is used to compute the set length. */
+            wMax = index;
+            wordMax = word;
+            /*  Count the actual bits, so as to get the cardinality of the set. */
+            cardinality += Long.bitCount(word);
+        }
+    }
+
+    //-----------------------------------------------------------------------------
+    /**
+     *  The XOR of level3 blocks is computed.
+     *
+     * 
+     * xor| 0 1
+     *   0| 0 1
+     *   1| 1 0 
+     */
+    protected class XorStrategy extends AbstractStrategy
+    {
+        @Override
+        //  XorStrategy
+        protected int properties()
+        {
+            return F_OP_F_EQ_F + X_OP_F_EQ_X;
+        }
+
+        @Override
+        //  XorStrategy
+        protected void start(SparseBitSet b)
+        {
+            if (b == null)
+                throw new NullPointerException();
+            cache.hash = 0;
+        }
+
+        @Override
+        protected boolean word(int base, int u3, long[] a3, long[] b3, long mask)
+        {
+            return (a3[u3] ^= b3[u3] & mask) == 0;
+        }
+
+        @Override
+        //  XorStrategy
+        protected boolean block(int base, int u3, int v3, long[] a3, long[] b3)
+        {
+            boolean isZero = true; //  Presumption
+            for (int w3 = u3; w3 != v3; ++w3)
+                isZero &= (a3[w3] ^= b3[w3]) == 0;
+            return isZero;
+
+        }
+    }
+
+    //-----------------------------------------------------------------------------
+    /**
+     *  Word and block and strategy.
+     */
+    protected transient AndStrategy andStrategy;
+    /**
+     *  Word and block andNot strategy.
+     */
+    protected transient AndNotStrategy andNotStrategy;
+    /**
+     *  Word and block clear strategy.
+     */
+    protected transient ClearStrategy clearStrategy;
+    /**
+     *  Word and block copy strategy.
+     */
+    protected transient CopyStrategy copyStrategy;
+    /**
+     *  Word and block equals strategy.
+     */
+    protected transient EqualsStrategy equalsStrategy;
+    /**
+     *  Word and block flip strategy.
+     */
+    protected transient FlipStrategy flipStrategy;
+    /**
+     *  Word and block intersects strategy.
+     */
+    protected transient IntersectsStrategy intersectsStrategy;
+    /**
+     *  Word and block or strategy.
+     */
+    protected transient OrStrategy orStrategy;
+    /**
+     *  Word and block set strategy.
+     */
+    protected transient SetStrategy setStrategy;
+    /**
+     *  Word and block update strategy.
+     */
+    protected transient UpdateStrategy updateStrategy;
+    /**
+     *  Word and block xor strategy.
+     */
+    protected transient XorStrategy xorStrategy;
+}
\ No newline at end of file
diff --git a/core/src/main/java/com/boydti/fawe/object/extent/DefaultTransformParser.java b/core/src/main/java/com/boydti/fawe/object/extent/DefaultTransformParser.java
index 887941bd..766d27e5 100644
--- a/core/src/main/java/com/boydti/fawe/object/extent/DefaultTransformParser.java
+++ b/core/src/main/java/com/boydti/fawe/object/extent/DefaultTransformParser.java
@@ -100,10 +100,10 @@ public class DefaultTransformParser extends InputParser {
                                 if (!rest.isEmpty()) {
                                     parent = parseFromInput(rest, context);
                                 }
-                                ExtentTraverser traverser = new ExtentTraverser(parent).find(BlockTransformExtent.class);
-                                BlockTransformExtent affine = (BlockTransformExtent) (traverser != null ? traverser.get() : null);
+                                ExtentTraverser traverser = new ExtentTraverser(parent).find(TransformExtent.class);
+                                BlockTransformExtent affine = (TransformExtent) (traverser != null ? traverser.get() : null);
                                 if (affine == null) {
-                                    parent = affine = new BlockTransformExtent(parent, context.requireWorld().getWorldData().getBlockRegistry());
+                                    parent = affine = new TransformExtent(parent, context.requireWorld().getWorldData().getBlockRegistry());
                                 }
                                 AffineTransform transform = (AffineTransform) affine.getTransform();
                                 transform = transform.rotateX(x);
diff --git a/core/src/main/java/com/boydti/fawe/object/io/BufferedRandomAccessFile.java b/core/src/main/java/com/boydti/fawe/object/io/BufferedRandomAccessFile.java
index 1ab2101e..31bd2204 100644
--- a/core/src/main/java/com/boydti/fawe/object/io/BufferedRandomAccessFile.java
+++ b/core/src/main/java/com/boydti/fawe/object/io/BufferedRandomAccessFile.java
@@ -316,7 +316,8 @@ public class BufferedRandomAccessFile extends RandomAccessFile
             if (this.curr_ == this.hi_)
                 return -1;
         }
-        byte res = this.buff_[(int) (this.curr_++ - this.lo_)];
+        byte res = this.buff_[(int) (this.curr_ - this.lo_)];
+        this.curr_++;
         return res;
     }
 
@@ -388,6 +389,13 @@ public class BufferedRandomAccessFile extends RandomAccessFile
         this.dirty_ = true;
     }
 
+    public void writeUnsafe(int b) throws IOException
+    {
+        this.buff_[(int) (this.curr_ - this.lo_)] = (byte) b;
+        this.curr_++;
+        this.dirty_ = true;
+    }
+
     @Override
     public void write(int b) throws IOException
     {
diff --git a/core/src/main/java/com/boydti/fawe/object/regions/MagicRegion.java b/core/src/main/java/com/boydti/fawe/object/regions/MagicRegion.java
new file mode 100644
index 00000000..638346b6
--- /dev/null
+++ b/core/src/main/java/com/boydti/fawe/object/regions/MagicRegion.java
@@ -0,0 +1,157 @@
+package com.boydti.fawe.object.regions;
+
+import com.boydti.fawe.util.EditSessionBuilder;
+import com.boydti.fawe.util.MathMan;
+import com.sk89q.worldedit.BlockVector;
+import com.sk89q.worldedit.EditSession;
+import com.sk89q.worldedit.Vector;
+import com.sk89q.worldedit.regions.AbstractRegion;
+import com.sk89q.worldedit.regions.RegionOperationException;
+import com.sk89q.worldedit.world.World;
+import java.util.BitSet;
+import java.util.Iterator;
+
+public class MagicRegion extends AbstractRegion {
+
+    private BitSet set = new BitSet();
+    private int minX, minY, minZ, maxX, maxY, maxZ;
+    private int offsetX, offsetY, offsetZ;
+
+    {
+        minX = minY = minZ = Integer.MAX_VALUE;
+        maxX = maxY = maxZ = Integer.MIN_VALUE;
+    }
+
+    public MagicRegion(World world) {
+        super(world);
+    }
+
+    private static int pair(int x, int y, int z) {
+        byte b1 = (byte) y;
+        byte b2 = (byte) (x);
+        byte b3 = (byte) (z);
+        int x16 = (x >> 8) & 0xF;
+        int z16 = (z >> 8) & 0xF;
+        byte b4 = MathMan.pair16(x16, z16);
+        return (b1 & 0xFF)
+             + ((b2 & 0xFF) << 8)
+             + ((b3 & 0xFF) << 16)
+             + ((b4 & 0xFF) << 24)
+             ;
+    }
+
+    public void select(int x, int y, int z) {
+        EditSession editSession = new EditSessionBuilder(getWorld())
+                .limitUnlimited()
+                .changeSetNull()
+                .fastmode(true)
+                .allowedRegionsEverywhere()
+                .checkMemory(false)
+                .autoQueue(false)
+                .build();
+
+    }
+
+    @Override
+    public Iterator iterator() {
+        return new Iterator() {
+
+            private int index = -1;
+            private BlockVector pos = new BlockVector(0, 0, 0);
+
+            @Override
+            public boolean hasNext() {
+                index = set.nextSetBit(index + 1);
+                return index != -1;
+            }
+
+            @Override
+            public BlockVector next() {
+                int b1 = index & 0xFF;
+                int b2 = (index >> 8) & 0xFF;
+                int b3 = (index >> 16) & 0xFF;
+                byte b4 = (byte) ((index >> 24) & 0xFF);
+                pos.x = offsetX + (((b2 & 0xFF) + ((MathMan.unpair16x(b4)) << 8)) << 20) >> 20;
+                pos.y = offsetY + b1;
+                pos.z = offsetZ + (((b3) + ((MathMan.unpair16y(b4)) << 8)) << 20) >> 20;
+                return pos;
+            }
+
+            @Override
+            public void remove() {
+                throw new UnsupportedOperationException();
+            }
+        };
+    }
+
+
+    public void set(int x, int y, int z) throws RegionOperationException{
+        x -= offsetX;
+        y -= offsetY;
+        z -= offsetZ;
+        set.set(pair(x, y, z), true);
+        if (x >= 2048 || x <= -2048 || z >= 2048 || z <= -2048) {
+            throw new RegionOperationException("Selection is too large!");
+        }
+        if (x > maxX) {
+            maxX = x;
+        }
+        if (x < minX) {
+            minX = x;
+        }
+        if (z > maxZ) {
+            maxZ = z;
+        }
+        if (z < minZ) {
+            minZ = z;
+        }
+        if (y > maxY) {
+            maxY = y;
+        }
+        if (y < minY) {
+            minY = y;
+        }
+    }
+
+    public boolean contains(int x, int y, int z) {
+        return set.get(pair(x - offsetX, y - offsetY, z - offsetZ));
+    }
+
+    @Override
+    public Vector getMinimumPoint() {
+        return new Vector(minX, minY, minZ);
+    }
+
+    @Override
+    public Vector getMaximumPoint() {
+        return new Vector(maxX, maxY, maxZ);
+    }
+
+    @Override
+    public void expand(Vector... changes) throws RegionOperationException {
+        throw new RegionOperationException("Selection is too large!");
+    }
+
+    @Override
+    public void contract(Vector... changes) throws RegionOperationException {
+        throw new RegionOperationException("Selection is too large!");
+    }
+
+    @Override
+    public boolean contains(Vector position) {
+        return contains((int) position.x, (int) position.y, (int) position.z);
+    }
+
+    @Override
+    public void shift(Vector change) throws RegionOperationException {
+        offsetX += change.getBlockX();
+        offsetY += change.getBlockY();
+        offsetZ += change.getBlockZ();
+        minX += change.getBlockX();
+        minY += change.getBlockY();
+        minZ += change.getBlockZ();
+        maxX += change.getBlockX();
+        maxY += change.getBlockY();
+        maxZ += change.getBlockZ();
+    }
+}
diff --git a/core/src/main/java/com/boydti/fawe/object/regions/selector/MagicRegionSelector.java b/core/src/main/java/com/boydti/fawe/object/regions/selector/MagicRegionSelector.java
new file mode 100644
index 00000000..43a9ecc1
--- /dev/null
+++ b/core/src/main/java/com/boydti/fawe/object/regions/selector/MagicRegionSelector.java
@@ -0,0 +1,104 @@
+package com.boydti.fawe.object.regions.selector;
+
+import com.boydti.fawe.object.regions.MagicRegion;
+import com.sk89q.worldedit.BlockVector;
+import com.sk89q.worldedit.IncompleteRegionException;
+import com.sk89q.worldedit.LocalSession;
+import com.sk89q.worldedit.Vector;
+import com.sk89q.worldedit.extension.platform.Actor;
+import com.sk89q.worldedit.regions.Region;
+import com.sk89q.worldedit.regions.RegionSelector;
+import com.sk89q.worldedit.regions.selector.limit.SelectorLimits;
+import com.sk89q.worldedit.world.World;
+import java.util.List;
+import javax.annotation.Nullable;
+
+public class MagicRegionSelector implements RegionSelector {
+
+    private MagicRegion region;
+
+    public MagicRegionSelector(@Nullable World world, Vector pos) {
+        this.region = new MagicRegion(world);
+    }
+
+    @Nullable
+    @Override
+    public World getWorld() {
+        return this.region.getWorld();
+    }
+
+    @Override
+    public void setWorld(@Nullable World world) {
+        this.region.setWorld(world);
+    }
+
+    @Override
+    public boolean selectPrimary(Vector position, SelectorLimits limits) {
+
+    }
+
+    @Override
+    public boolean selectSecondary(Vector position, SelectorLimits limits) {
+
+    }
+
+    @Override
+    public void explainPrimarySelection(Actor actor, LocalSession session, Vector position) {
+
+    }
+
+    @Override
+    public void explainSecondarySelection(Actor actor, LocalSession session, Vector position) {
+
+    }
+
+    @Override
+    public void explainRegionAdjust(Actor actor, LocalSession session) {
+
+    }
+
+    @Override
+    public BlockVector getPrimaryPosition() throws IncompleteRegionException {
+        return null;
+    }
+
+    @Override
+    public Region getRegion() throws IncompleteRegionException {
+        return null;
+    }
+
+    @Override
+    public Region getIncompleteRegion() {
+        return null;
+    }
+
+    @Override
+    public boolean isDefined() {
+        return false;
+    }
+
+    @Override
+    public int getArea() {
+        return 0;
+    }
+
+    @Override
+    public void learnChanges() {
+
+    }
+
+    @Override
+    public void clear() {
+
+    }
+
+    @Override
+    public String getTypeName() {
+        return null;
+    }
+
+    @Override
+    public List getInformationLines() {
+        return null;
+    }
+}
diff --git a/core/src/main/java/com/boydti/fawe/util/MainUtil.java b/core/src/main/java/com/boydti/fawe/util/MainUtil.java
index 4c75daca..5b23d5c1 100644
--- a/core/src/main/java/com/boydti/fawe/util/MainUtil.java
+++ b/core/src/main/java/com/boydti/fawe/util/MainUtil.java
@@ -3,22 +3,51 @@ package com.boydti.fawe.util;
 import com.boydti.fawe.Fawe;
 import com.boydti.fawe.config.BBC;
 import com.boydti.fawe.config.Settings;
-import com.boydti.fawe.object.*;
+import com.boydti.fawe.object.FaweInputStream;
+import com.boydti.fawe.object.FaweOutputStream;
+import com.boydti.fawe.object.FawePlayer;
+import com.boydti.fawe.object.RegionWrapper;
+import com.boydti.fawe.object.RunnableVal;
+import com.boydti.fawe.object.RunnableVal2;
 import com.boydti.fawe.object.changeset.CPUOptimizedChangeSet;
 import com.boydti.fawe.object.changeset.FaweStreamChangeSet;
 import com.boydti.fawe.object.io.AbstractDelegateOutputStream;
-import com.boydti.fawe.object.io.PGZIPOutputStream;
-import com.sk89q.jnbt.*;
+import com.github.luben.zstd.ZstdInputStream;
+import com.github.luben.zstd.ZstdOutputStream;
+import com.sk89q.jnbt.CompoundTag;
+import com.sk89q.jnbt.DoubleTag;
+import com.sk89q.jnbt.EndTag;
+import com.sk89q.jnbt.IntTag;
+import com.sk89q.jnbt.ListTag;
+import com.sk89q.jnbt.StringTag;
+import com.sk89q.jnbt.Tag;
 import com.sk89q.worldedit.entity.Entity;
 import com.sk89q.worldedit.history.changeset.ChangeSet;
 import com.sk89q.worldedit.util.Location;
-import java.io.*;
+import java.io.BufferedInputStream;
+import java.io.BufferedOutputStream;
+import java.io.File;
+import java.io.FileInputStream;
+import java.io.FileOutputStream;
+import java.io.IOException;
+import java.io.InputStream;
+import java.io.OutputStream;
+import java.io.OutputStreamWriter;
+import java.io.PrintWriter;
 import java.lang.reflect.Array;
 import java.lang.reflect.Method;
-import java.net.*;
+import java.net.HttpURLConnection;
+import java.net.MalformedURLException;
+import java.net.URISyntaxException;
+import java.net.URL;
+import java.net.URLConnection;
 import java.nio.channels.FileChannel;
 import java.nio.charset.StandardCharsets;
-import java.nio.file.*;
+import java.nio.file.FileVisitResult;
+import java.nio.file.Files;
+import java.nio.file.Path;
+import java.nio.file.Paths;
+import java.nio.file.SimpleFileVisitor;
 import java.nio.file.attribute.BasicFileAttributes;
 import java.util.ArrayList;
 import java.util.Arrays;
@@ -28,11 +57,15 @@ import java.util.Map;
 import java.util.Map.Entry;
 import java.util.UUID;
 import java.util.concurrent.atomic.AtomicLong;
-import java.util.zip.Deflater;
 import java.util.zip.GZIPInputStream;
 import java.util.zip.ZipEntry;
 import java.util.zip.ZipInputStream;
-import net.jpountz.lz4.*;
+import net.jpountz.lz4.LZ4Compressor;
+import net.jpountz.lz4.LZ4Factory;
+import net.jpountz.lz4.LZ4FastDecompressor;
+import net.jpountz.lz4.LZ4InputStream;
+import net.jpountz.lz4.LZ4OutputStream;
+import net.jpountz.lz4.LZ4Utils;
 
 public class MainUtil {
     /*
@@ -213,17 +246,15 @@ public class MainUtil {
     }
 
     public static FaweOutputStream getCompressedOS(OutputStream os, int amount, int buffer) throws IOException {
-        os.write((byte) amount);
+        os.write((byte) -amount);
         os = new BufferedOutputStream(os, buffer);
         if (amount == 0) {
             return new FaweOutputStream(os);
         }
+        os = new BufferedOutputStream(os, buffer);
         int gzipAmount = amount > 6 ? 1 : 0;
         for (int i = 0; i < gzipAmount; i++) {
-            PGZIPOutputStream gzip = new PGZIPOutputStream(os, buffer);
-            os = new BufferedOutputStream(gzip);
-            gzip.setStrategy(Deflater.DEFAULT_STRATEGY);
-            gzip.setLevel(9);
+            os = new ZstdOutputStream(os, 22);
         }
         LZ4Factory factory = LZ4Factory.fastestInstance();
         int fastAmount = 1 + ((amount - 1) % 3);
@@ -246,18 +277,22 @@ public class MainUtil {
     }
 
     public static FaweInputStream getCompressedIS(InputStream is, int buffer) throws IOException {
-        int amount = is.read();
+        int amount = (byte) is.read();
         is = new BufferedInputStream(is, buffer);
         if (amount == 0) {
             return new FaweInputStream(is);
         }
+        int amountAbs = Math.abs(amount);
         LZ4Factory factory = LZ4Factory.fastestInstance();
-        boolean gzip = amount > 6;
-        if (gzip) {
-            is = new BufferedInputStream(new GZIPInputStream(is, buffer));
+        if (amountAbs > 6) {
+            if (amount > 0) {
+                is = new BufferedInputStream(new GZIPInputStream(is, buffer));
+            } else {
+                is = new ZstdInputStream(is);
+            }
         }
-        amount = (1 + ((amount - 1) % 3)) + (amount > 3 ? 1 : 0);
-        for (int i = 0; i < amount; i++) {
+        amountAbs = (1 + ((amountAbs - 1) % 3)) + (amountAbs > 3 ? 1 : 0);
+        for (int i = 0; i < amountAbs; i++) {
             is = new LZ4InputStream(is);
         }
         return new FaweInputStream(is);
@@ -351,8 +386,8 @@ public class MainUtil {
     public static void sendCompressedMessage(FaweStreamChangeSet set, FawePlayer actor)
     {
         try {
-            int elements = set.size();
-            int compressedSize = set.getCompressedSize();
+            long elements = set.size();
+            long compressedSize = set.getCompressedSize();
             if (compressedSize == 0) {
                 return;
             }
@@ -375,10 +410,10 @@ public class MainUtil {
              *
              * This compares FAWE's usage to standard WE.
              */
-            int total = 128 * elements;
+            long total = 128 * elements;
 
-            int ratio = total / compressedSize;
-            int saved = total - compressedSize;
+            long ratio = total / compressedSize;
+            long saved = total - compressedSize;
 
             if (ratio > 3 && Thread.currentThread() != Fawe.get().getMainThread() && actor != null) {
                 BBC.COMPRESSED.send(actor, saved, ratio);
diff --git a/core/src/main/java/com/sk89q/worldedit/command/RegionCommands.java b/core/src/main/java/com/sk89q/worldedit/command/RegionCommands.java
index 8e1bf51a..b6c626c8 100644
--- a/core/src/main/java/com/sk89q/worldedit/command/RegionCommands.java
+++ b/core/src/main/java/com/sk89q/worldedit/command/RegionCommands.java
@@ -286,6 +286,24 @@ public class RegionCommands {
         BBC.VISITOR_BLOCK.send(player, affected);
     }
 
+    @Command(
+            aliases = { "/mapreplace", "/mr", "/maprep" },
+            usage = " ",
+            desc = "Replace all blocks in a selection 1:1 with the ",
+            flags = "f",
+            min = 1,
+            max = 2
+    )
+    @CommandPermissions("worldedit.region.mapreplace")
+    @Logging(REGION)
+    public void mapreplace(Player player, EditSession editSession, @Selection Region region, @Optional Mask from, Pattern to) throws WorldEditException {
+        if (from == null) {
+            from = new ExistingBlockMask(editSession);
+        }
+        int affected = editSession.replaceBlocks(region, from, Patterns.wrap(to));
+        BBC.VISITOR_BLOCK.send(player, affected);
+    }
+
     @Command(
             aliases = { "/set" },
             usage = "[pattern]",
diff --git a/core/src/main/java/com/sk89q/worldedit/extension/factory/DefaultMaskParser.java b/core/src/main/java/com/sk89q/worldedit/extension/factory/DefaultMaskParser.java
index 3d279327..d6909d67 100644
--- a/core/src/main/java/com/sk89q/worldedit/extension/factory/DefaultMaskParser.java
+++ b/core/src/main/java/com/sk89q/worldedit/extension/factory/DefaultMaskParser.java
@@ -1,5 +1,6 @@
 package com.sk89q.worldedit.extension.factory;
 
+import com.boydti.fawe.command.FaweParser;
 import com.boydti.fawe.object.mask.AdjacentMask;
 import com.boydti.fawe.object.mask.AngleMask;
 import com.boydti.fawe.object.mask.CustomMask;
@@ -33,7 +34,6 @@ import com.sk89q.worldedit.function.mask.RegionMask;
 import com.sk89q.worldedit.function.mask.SolidBlockMask;
 import com.sk89q.worldedit.internal.expression.Expression;
 import com.sk89q.worldedit.internal.expression.ExpressionException;
-import com.sk89q.worldedit.internal.registry.InputParser;
 import com.sk89q.worldedit.math.noise.RandomNoise;
 import com.sk89q.worldedit.regions.shape.WorldEditExpressionEnvironment;
 import com.sk89q.worldedit.session.request.Request;
@@ -54,7 +54,19 @@ import static com.google.common.base.Preconditions.checkNotNull;
 /**
  * Parses mask input strings.
  */
-public class DefaultMaskParser extends InputParser {
+public class DefaultMaskParser extends FaweParser {
+
+    public static final String[] EXPRESSION_MASK = new String[] { "=" };
+
+    public static final String[] BLOCK_MASK = new String[] { "" };
+
+    public static final String[] SIMPLE_MASK = new String[] {
+            "#existing", "#solid", "#dregion", "#dselection", "#dsel", "#selection", "#region", "#sel", "#xaxis", "#yaxis", "#zaxis", "#id", "#data", "#wall", "#surface",
+    };
+
+    public static final String[] MISC_PATTERNS = new String[] {
+            "hand", "pos1",
+    };
 
     public DefaultMaskParser(WorldEdit worldEdit) {
         super(worldEdit);
diff --git a/core/src/main/java/com/sk89q/worldedit/extension/factory/HashTagPatternParser.java b/core/src/main/java/com/sk89q/worldedit/extension/factory/HashTagPatternParser.java
index a7e04754..f5b2727e 100644
--- a/core/src/main/java/com/sk89q/worldedit/extension/factory/HashTagPatternParser.java
+++ b/core/src/main/java/com/sk89q/worldedit/extension/factory/HashTagPatternParser.java
@@ -1,5 +1,6 @@
 package com.sk89q.worldedit.extension.factory;
 
+import com.boydti.fawe.command.FaweParser;
 import com.boydti.fawe.command.SuggestInputParseException;
 import com.boydti.fawe.object.pattern.*;
 import com.boydti.fawe.util.MainUtil;
@@ -20,16 +21,14 @@ import com.sk89q.worldedit.function.pattern.Pattern;
 import com.sk89q.worldedit.function.pattern.RandomPattern;
 import com.sk89q.worldedit.internal.expression.Expression;
 import com.sk89q.worldedit.internal.expression.ExpressionException;
-import com.sk89q.worldedit.internal.registry.InputParser;
 import com.sk89q.worldedit.regions.shape.WorldEditExpressionEnvironment;
 import com.sk89q.worldedit.session.ClipboardHolder;
 import com.sk89q.worldedit.session.request.Request;
 import com.sk89q.worldedit.world.registry.BundledBlockData;
 import java.util.ArrayList;
-import java.util.Arrays;
 import java.util.List;
 
-public class HashTagPatternParser extends InputParser{
+public class HashTagPatternParser extends FaweParser {
 
     public static final String[] EXPRESSION_PATTERN = new String[] { "=" };
 
@@ -53,51 +52,6 @@ public class HashTagPatternParser extends InputParser{
         super(worldEdit);
     }
 
-    private List split(String input, char delim) {
-        List result = new ArrayList();
-        int start = 0;
-        boolean inQuotes = false;
-        for (int current = 0; current < input.length(); current++) {
-            if (input.charAt(current) == '\"') inQuotes = !inQuotes; // toggle state
-            boolean atLastChar = (current == input.length() - 1);
-            if(atLastChar) result.add(input.substring(start));
-            else if (input.charAt(current) == delim && !inQuotes) {
-                String toAdd = input.substring(start, current);
-                if (toAdd.startsWith("\"")) {
-                    toAdd = toAdd.substring(1, toAdd.length() - 1);
-                }
-                result.add(toAdd);
-                start = current + 1;
-            }
-        }
-        return result;
-    }
-
-    private Pattern catchSuggestion(String currentInput, String nextInput, ParserContext context) throws InputParseException{
-        try {
-            return parseFromInput(nextInput, context);
-        } catch (SuggestInputParseException e) {
-            e.prepend(currentInput.substring(0, currentInput.length() - nextInput.length()));
-            throw e;
-        }
-    }
-
-    public List handleRemainder(String input, String... expected) throws InputParseException {
-        List remainder = split(input, ':');
-        int len = remainder.size();
-        if (len != expected.length - 1) {
-            if (len <= expected.length - 1 && len != 0) {
-                if (remainder.get(len - 1).endsWith(":")) {
-                    throw new SuggestInputParseException(null, ALL_PATTERNS).prepend(expected[0] + ":" + input);
-                }
-                throw new SuggestInputParseException(null, expected[0] + ":" + input + ":" + StringMan.join(Arrays.copyOfRange(expected, len + 1, 3), ":"));
-            } else {
-                throw new SuggestInputParseException(null, StringMan.join(expected, ":"));
-            }
-        }
-        return remainder;
-    }
-
     @Override
     public Pattern parseFromInput(String input, ParserContext context) throws InputParseException {
         if (input.isEmpty()) {
@@ -172,7 +126,7 @@ public class HashTagPatternParser extends InputParser{
                             return new NoZPattern(catchSuggestion(input, rest, context));
                         }
                         case "#mask": {
-                            List split3 = handleRemainder(rest, "#mask", "", "", "");
+                            List split3 = suggestRemaining(rest, "#mask", "", "", "");
                             Pattern primary = catchSuggestion(input, split3.get(1), context);
                             Pattern secondary = catchSuggestion(input, split3.get(2), context);
                             PatternExtent extent = new PatternExtent(primary);
@@ -190,7 +144,7 @@ public class HashTagPatternParser extends InputParser{
                         }
                         case "#offset":
                             try {
-                                List split3 = handleRemainder(rest, "#offset", "", "", "", "");
+                                List split3 = suggestRemaining(rest, "#offset", "", "", "", "");
                                 int x = (int) Math.abs(Expression.compile(split3.get(0)).evaluate());
                                 int y = (int) Math.abs(Expression.compile(split3.get(1)).evaluate());
                                 int z = (int) Math.abs(Expression.compile(split3.get(2)).evaluate());
@@ -202,7 +156,7 @@ public class HashTagPatternParser extends InputParser{
                             }
                         case "#surfacespread": {
                             try {
-                                List split3 = handleRemainder(rest, "#surfacespread", "", "", "", "");
+                                List split3 = suggestRemaining(rest, "#surfacespread", "", "", "", "");
                                 int x = (int) Math.abs(Expression.compile(split3.get(0)).evaluate());
                                 int y = (int) Math.abs(Expression.compile(split3.get(1)).evaluate());
                                 int z = (int) Math.abs(Expression.compile(split3.get(2)).evaluate());
@@ -215,7 +169,7 @@ public class HashTagPatternParser extends InputParser{
                         }
                         case "#solidspread": {
                             try {
-                                List split3 = handleRemainder(rest, "#solidspread", "", "", "", "");
+                                List split3 = suggestRemaining(rest, "#solidspread", "", "", "", "");
                                 int x = (int) Math.abs(Expression.compile(split3.get(0)).evaluate());
                                 int y = (int) Math.abs(Expression.compile(split3.get(1)).evaluate());
                                 int z = (int) Math.abs(Expression.compile(split3.get(2)).evaluate());
@@ -229,7 +183,7 @@ public class HashTagPatternParser extends InputParser{
                         case "#randomoffset":
                         case "#spread": {
                             try {
-                                List split3 = handleRemainder(rest, "#spread", "", "", "", "");
+                                List split3 = suggestRemaining(rest, "#spread", "", "", "", "");
                                 int x = (int) Math.abs(Expression.compile(split3.get(0)).evaluate());
                                 int y = (int) Math.abs(Expression.compile(split3.get(1)).evaluate());
                                 int z = (int) Math.abs(Expression.compile(split3.get(2)).evaluate());
diff --git a/core/src/main/java/com/sk89q/worldedit/function/visitor/RecursiveVisitor.java b/core/src/main/java/com/sk89q/worldedit/function/visitor/RecursiveVisitor.java
index dff099f7..bf1f2aa8 100644
--- a/core/src/main/java/com/sk89q/worldedit/function/visitor/RecursiveVisitor.java
+++ b/core/src/main/java/com/sk89q/worldedit/function/visitor/RecursiveVisitor.java
@@ -58,4 +58,4 @@ public class RecursiveVisitor extends BreadthFirstSearch {
     public static Class inject() {
         return RecursiveVisitor.class;
     }
-}
+}
\ No newline at end of file
diff --git a/core/src/main/java/com/sk89q/worldedit/regions/CuboidRegion.java b/core/src/main/java/com/sk89q/worldedit/regions/CuboidRegion.java
index d7279a70..f47c7cb2 100644
--- a/core/src/main/java/com/sk89q/worldedit/regions/CuboidRegion.java
+++ b/core/src/main/java/com/sk89q/worldedit/regions/CuboidRegion.java
@@ -356,7 +356,7 @@ public class CuboidRegion extends AbstractRegion implements FlatRegion {
 
     @Override
     public Iterator iterator() {
-        if (Settings.HISTORY.COMPRESSION_LEVEL == 9) {
+        if (Settings.HISTORY.COMPRESSION_LEVEL >= 9) {
             return iterator_old();
         }
         final BlockVector v = new BlockVector(0,0,0);
diff --git a/forge110/build.gradle b/forge110/build.gradle
index 00506cf9..53de852f 100644
--- a/forge110/build.gradle
+++ b/forge110/build.gradle
@@ -57,6 +57,7 @@ processResources {
 shadowJar {
     relocate 'org.yaml.snakeyaml', 'com.boydti.fawe.yaml'
     dependencies {
+        include(dependency('com.github.luben:zstd-jni:1.1.1'))
         include(dependency(':core'))
         include(dependency('org.yaml:snakeyaml:1.16'))
     }
diff --git a/forge111/build.gradle b/forge111/build.gradle
index e2eeb0bf..ae6bfea9 100644
--- a/forge111/build.gradle
+++ b/forge111/build.gradle
@@ -57,6 +57,7 @@ processResources {
 shadowJar {
     relocate 'org.yaml.snakeyaml', 'com.boydti.fawe.yaml'
     dependencies {
+        include(dependency('com.github.luben:zstd-jni:1.1.1'))
         include(dependency(':core'))
         include(dependency('org.yaml:snakeyaml:1.16'))
     }
diff --git a/forge1710/build.gradle b/forge1710/build.gradle
index 6048ce3f..c8231b80 100644
--- a/forge1710/build.gradle
+++ b/forge1710/build.gradle
@@ -48,6 +48,7 @@ processResources {
 shadowJar {
     relocate 'org.yaml.snakeyaml', 'com.boydti.fawe.yaml'
     dependencies {
+        include(dependency('com.github.luben:zstd-jni:1.1.1'))
         include(dependency(':core'))
         include(dependency('org.yaml:snakeyaml:1.16'))
     }
diff --git a/forge189/build.gradle b/forge189/build.gradle
index 49f97575..45c79e59 100644
--- a/forge189/build.gradle
+++ b/forge189/build.gradle
@@ -57,6 +57,7 @@ processResources {
 shadowJar {
     relocate 'org.yaml.snakeyaml', 'com.boydti.fawe.yaml'
     dependencies {
+        include(dependency('com.github.luben:zstd-jni:1.1.1'))
         include(dependency(':core'))
         include(dependency('org.yaml:snakeyaml:1.16'))
     }
diff --git a/forge194/build.gradle b/forge194/build.gradle
index 6e784a01..08ca1247 100644
--- a/forge194/build.gradle
+++ b/forge194/build.gradle
@@ -56,6 +56,7 @@ processResources {
 shadowJar {
     relocate 'org.yaml.snakeyaml', 'com.boydti.fawe.yaml'
     dependencies {
+        include(dependency('com.github.luben:zstd-jni:1.1.1'))
         include(dependency(':core'))
         include(dependency('org.yaml:snakeyaml:1.16'))
     }
diff --git a/nukkit/build.gradle b/nukkit/build.gradle
index 7d9a97d2..21eaede6 100644
--- a/nukkit/build.gradle
+++ b/nukkit/build.gradle
@@ -22,6 +22,7 @@ apply plugin: 'com.github.johnrengelman.shadow'
 jar.enabled = false
 shadowJar {
     dependencies {
+        include(dependency('com.github.luben:zstd-jni:1.1.1'))
         include(dependency(name: 'worldedit-core-6.1.4-SNAPSHOT-dist'))
         include(dependency('com.google.code.gson:gson:2.2.4'))
         include(dependency('org.yaml:snakeyaml:1.16'))