mirror of
https://github.com/plexusorg/Plex-FAWE.git
synced 2025-01-09 17:27:38 +00:00
Make mca file reusable
This commit is contained in:
parent
2f3c6769c8
commit
ed7df341b4
@ -88,6 +88,10 @@ public class MCAChunk implements IChunkSet {
|
|||||||
public MCAChunk(NBTInputStream nis, int chunkX, int chunkZ, boolean readPos) throws IOException {
|
public MCAChunk(NBTInputStream nis, int chunkX, int chunkZ, boolean readPos) throws IOException {
|
||||||
this.chunkX = chunkX;
|
this.chunkX = chunkX;
|
||||||
this.chunkZ = chunkZ;
|
this.chunkZ = chunkZ;
|
||||||
|
read(nis, readPos);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void read(NBTInputStream nis, boolean readPos) throws IOException {
|
||||||
StreamDelegate root = createDelegate(nis, readPos);
|
StreamDelegate root = createDelegate(nis, readPos);
|
||||||
nis.readNamedTagLazy(root);
|
nis.readNamedTagLazy(root);
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
package com.boydti.fawe.jnbt.anvil;
|
package com.boydti.fawe.jnbt.anvil;
|
||||||
|
|
||||||
import com.boydti.fawe.Fawe;
|
import com.boydti.fawe.Fawe;
|
||||||
|
import com.boydti.fawe.beta.Trimable;
|
||||||
import com.boydti.fawe.jnbt.streamer.StreamDelegate;
|
import com.boydti.fawe.jnbt.streamer.StreamDelegate;
|
||||||
import com.boydti.fawe.object.RunnableVal;
|
import com.boydti.fawe.object.RunnableVal;
|
||||||
import com.boydti.fawe.object.RunnableVal4;
|
import com.boydti.fawe.object.RunnableVal4;
|
||||||
@ -25,6 +26,7 @@ import java.util.List;
|
|||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.concurrent.ForkJoinPool;
|
import java.util.concurrent.ForkJoinPool;
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
|
import java.util.function.Consumer;
|
||||||
import java.util.zip.Inflater;
|
import java.util.zip.Inflater;
|
||||||
import java.util.zip.InflaterInputStream;
|
import java.util.zip.InflaterInputStream;
|
||||||
|
|
||||||
@ -32,7 +34,7 @@ import java.util.zip.InflaterInputStream;
|
|||||||
* Chunk format: http://minecraft.gamepedia.com/Chunk_format#Entity_format
|
* Chunk format: http://minecraft.gamepedia.com/Chunk_format#Entity_format
|
||||||
* e.g.: `.Level.Entities.#` (Starts with a . as the root tag is unnamed)
|
* e.g.: `.Level.Entities.#` (Starts with a . as the root tag is unnamed)
|
||||||
*/
|
*/
|
||||||
public class MCAFile {
|
public class MCAFile implements Trimable {
|
||||||
|
|
||||||
private static Field fieldBuf2;
|
private static Field fieldBuf2;
|
||||||
private static Field fieldBuf3;
|
private static Field fieldBuf3;
|
||||||
@ -48,13 +50,17 @@ public class MCAFile {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private final World world;
|
private final ForkJoinPool pool;
|
||||||
private final File file;
|
private final byte[] locations;
|
||||||
|
private boolean readLocations;
|
||||||
|
|
||||||
|
private File file;
|
||||||
private RandomAccessFile raf;
|
private RandomAccessFile raf;
|
||||||
private byte[] locations;
|
|
||||||
private boolean deleted;
|
private boolean deleted;
|
||||||
private final int X, Z;
|
private int X, Z;
|
||||||
private final Int2ObjectOpenHashMap<MCAChunk> chunks = new Int2ObjectOpenHashMap<>();
|
private MCAChunk[] chunks;
|
||||||
|
private boolean[] chunkInitialized;
|
||||||
|
|
||||||
final ThreadLocal<byte[]> byteStore1 = new ThreadLocal<byte[]>() {
|
final ThreadLocal<byte[]> byteStore1 = new ThreadLocal<byte[]>() {
|
||||||
@Override
|
@Override
|
||||||
@ -75,26 +81,93 @@ public class MCAFile {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
public MCAFile(World world, File file) throws FileNotFoundException {
|
public MCAFile(ForkJoinPool pool) {
|
||||||
this.world = world;
|
this.pool = pool;
|
||||||
|
this.locations = new byte[4096];
|
||||||
|
this.chunks = new MCAChunk[32 * 32];
|
||||||
|
this.chunkInitialized = new boolean[this.chunks.length];
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean trim(boolean aggressive) {
|
||||||
|
boolean hasChunk = false;
|
||||||
|
for (int i = 0; i < chunkInitialized.length; i++) {
|
||||||
|
if (!chunkInitialized[i]) {
|
||||||
|
chunks[i] = null;
|
||||||
|
} else {
|
||||||
|
hasChunk = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
CleanableThreadLocal.clean(byteStore1);
|
||||||
|
CleanableThreadLocal.clean(byteStore2);
|
||||||
|
CleanableThreadLocal.clean(byteStore3);
|
||||||
|
return !hasChunk;
|
||||||
|
}
|
||||||
|
|
||||||
|
public MCAFile init(File file) throws FileNotFoundException {
|
||||||
|
String[] split = file.getName().split("\\.");
|
||||||
|
X = Integer.parseInt(split[1]);
|
||||||
|
Z = Integer.parseInt(split[2]);
|
||||||
|
return init(file, X, Z);
|
||||||
|
}
|
||||||
|
|
||||||
|
public MCAFile init(File file, int mcrX, int mcrZ) throws FileNotFoundException {
|
||||||
|
if (raf != null) {
|
||||||
|
synchronized (raf) {
|
||||||
|
if (raf != null) {
|
||||||
|
flush(pool);
|
||||||
|
for (int i = 0; i < 4096; i++) {
|
||||||
|
locations[i] = 0;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
raf.close();
|
||||||
|
} catch (IOException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
raf = null;
|
||||||
|
deleted = false;
|
||||||
|
Arrays.fill(chunkInitialized, false);
|
||||||
|
readLocations = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.X = mcrX;
|
||||||
|
this.Z = mcrZ;
|
||||||
|
}
|
||||||
|
|
||||||
this.file = file;
|
this.file = file;
|
||||||
if (!file.exists()) {
|
if (!file.exists()) {
|
||||||
throw new FileNotFoundException(file.getName());
|
throw new FileNotFoundException(file.getName());
|
||||||
}
|
}
|
||||||
String[] split = file.getName().split("\\.");
|
return this;
|
||||||
X = Integer.parseInt(split[1]);
|
|
||||||
Z = Integer.parseInt(split[2]);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public MCAFile(World world, int mcrX, int mcrZ) {
|
public MCAFile init(World world, int mcrX, int mcrZ) throws FileNotFoundException {
|
||||||
this(world, mcrX, mcrZ, new File(world.getStoragePath().toFile(), "r." + mcrX + "." + mcrZ + ".mca"));
|
return init(new File(world.getStoragePath().toFile(), File.separator + "regions" + File.separator + "r." + mcrX + "." + mcrZ + ".mca"));
|
||||||
}
|
}
|
||||||
|
|
||||||
public MCAFile(World world, int mcrX, int mcrZ, File file) {
|
public int getIndex(int chunkX, int chunkZ) {
|
||||||
this.world = world;
|
return ((chunkX & 31) << 2) + ((chunkZ & 31) << 7);
|
||||||
this.file = file;
|
}
|
||||||
X = mcrX;
|
|
||||||
Z = mcrZ;
|
|
||||||
|
private RandomAccessFile getRaf() throws FileNotFoundException {
|
||||||
|
if (this.raf == null) {
|
||||||
|
this.raf = new RandomAccessFile(file, "rw");
|
||||||
|
}
|
||||||
|
return this.raf;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void readHeader() throws IOException {
|
||||||
|
if (!readLocations) {
|
||||||
|
readLocations = true;
|
||||||
|
getRaf();
|
||||||
|
if (raf.length() < 8192) {
|
||||||
|
raf.setLength(8192);
|
||||||
|
} else {
|
||||||
|
raf.seek(0);
|
||||||
|
raf.readFully(locations);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void clear() {
|
public void clear() {
|
||||||
@ -106,12 +179,8 @@ public class MCAFile {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
synchronized (chunks) {
|
synchronized (chunks) {
|
||||||
chunks.clear();
|
Arrays.fill(chunkInitialized, false);
|
||||||
}
|
}
|
||||||
locations = null;
|
|
||||||
CleanableThreadLocal.clean(byteStore1);
|
|
||||||
CleanableThreadLocal.clean(byteStore2);
|
|
||||||
CleanableThreadLocal.clean(byteStore3);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -130,32 +199,6 @@ public class MCAFile {
|
|||||||
return deleted;
|
return deleted;
|
||||||
}
|
}
|
||||||
|
|
||||||
public World getWorld() {
|
|
||||||
return world;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Loads the location header from disk
|
|
||||||
*/
|
|
||||||
public void init() {
|
|
||||||
try {
|
|
||||||
if (raf == null) {
|
|
||||||
this.locations = new byte[4096];
|
|
||||||
if (file != null) {
|
|
||||||
this.raf = new RandomAccessFile(file, "rw");
|
|
||||||
if (raf.length() < 8192) {
|
|
||||||
raf.setLength(8192);
|
|
||||||
} else {
|
|
||||||
raf.seek(0);
|
|
||||||
raf.readFully(locations);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (Throwable e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public int getX() {
|
public int getX() {
|
||||||
return X;
|
return X;
|
||||||
}
|
}
|
||||||
@ -173,44 +216,46 @@ public class MCAFile {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public MCAChunk getCachedChunk(int cx, int cz) {
|
public MCAChunk getCachedChunk(int cx, int cz) {
|
||||||
int pair = MathMan.pair((short) (cx & 31), (short) (cz & 31));
|
int pair = getIndex(cx, cz);
|
||||||
synchronized (chunks) {
|
MCAChunk chunk = chunks[pair];
|
||||||
return chunks.get(pair);
|
if (chunk != null && chunkInitialized[pair]) {
|
||||||
|
return chunk;
|
||||||
}
|
}
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setChunk(MCAChunk chunk) {
|
public void setChunk(MCAChunk chunk) {
|
||||||
int cx = chunk.getX();
|
int cx = chunk.getX();
|
||||||
int cz = chunk.getZ();
|
int cz = chunk.getZ();
|
||||||
int pair = MathMan.pair((short) (cx & 31), (short) (cz & 31));
|
int pair = getIndex(cx, cz);
|
||||||
synchronized (chunks) {
|
chunks[pair] = chunk;
|
||||||
chunks.put(pair, chunk);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public MCAChunk getChunk(int cx, int cz) throws IOException {
|
public MCAChunk getChunk(int cx, int cz) throws IOException {
|
||||||
MCAChunk cached = getCachedChunk(cx, cz);
|
int pair = getIndex(cx, cz);
|
||||||
if (cached != null) {
|
MCAChunk chunk = chunks[pair];
|
||||||
return cached;
|
if (chunk == null) {
|
||||||
} else {
|
chunk = new MCAChunk();
|
||||||
return readChunk(cx, cz);
|
chunk.setPosition(cx, cz);
|
||||||
|
chunks[pair] = chunk;
|
||||||
|
} else if (chunkInitialized[pair]) {
|
||||||
|
return chunk;
|
||||||
}
|
}
|
||||||
|
readChunk(chunk, pair);
|
||||||
|
chunkInitialized[pair] = true;
|
||||||
|
return chunk;
|
||||||
}
|
}
|
||||||
|
|
||||||
public MCAChunk readChunk(int cx, int cz) throws IOException {
|
private MCAChunk readChunk(MCAChunk chunk, int i) throws IOException {
|
||||||
int i = ((cx & 31) << 2) + ((cz & 31) << 7);
|
|
||||||
int offset = (((locations[i] & 0xFF) << 16) + ((locations[i + 1] & 0xFF) << 8) + ((locations[i + 2] & 0xFF))) << 12;
|
int offset = (((locations[i] & 0xFF) << 16) + ((locations[i + 1] & 0xFF) << 8) + ((locations[i + 2] & 0xFF))) << 12;
|
||||||
int size = (locations[i + 3] & 0xFF) << 12;
|
|
||||||
if (offset == 0) {
|
if (offset == 0) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
NBTInputStream nis = getChunkIS(offset);
|
int size = (locations[i + 3] & 0xFF) << 12;
|
||||||
MCAChunk chunk = new MCAChunk(nis, cx, cz, false);
|
try (NBTInputStream nis = getChunkIS(offset)) {
|
||||||
nis.close();
|
chunk.read(nis, false);
|
||||||
int pair = MathMan.pair((short) (cx & 31), (short) (cz & 31));
|
|
||||||
synchronized (chunks) {
|
|
||||||
chunks.put(pair, chunk);
|
|
||||||
}
|
}
|
||||||
|
System.out.println("TODO multithreaded"); // TODO
|
||||||
return chunk;
|
return chunk;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -266,7 +311,7 @@ public class MCAFile {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void forEachChunk(RunnableVal<MCAChunk> onEach) {
|
public void forEachChunk(Consumer<MCAChunk> onEach) {
|
||||||
int i = 0;
|
int i = 0;
|
||||||
for (int z = 0; z < 32; z++) {
|
for (int z = 0; z < 32; z++) {
|
||||||
for (int x = 0; x < 32; x++, i += 4) {
|
for (int x = 0; x < 32; x++, i += 4) {
|
||||||
@ -274,7 +319,7 @@ public class MCAFile {
|
|||||||
int size = locations[i + 3] & 0xFF;
|
int size = locations[i + 3] & 0xFF;
|
||||||
if (size != 0) {
|
if (size != 0) {
|
||||||
try {
|
try {
|
||||||
onEach.run(getChunk(x, z));
|
onEach.accept(getChunk(x, z));
|
||||||
} catch (Throwable ignore) {
|
} catch (Throwable ignore) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -283,28 +328,16 @@ public class MCAFile {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public int getOffset(int cx, int cz) {
|
public int getOffset(int cx, int cz) {
|
||||||
int i = ((cx & 31) << 2) + ((cz & 31) << 7);
|
int i = getIndex(cx, cz);
|
||||||
int offset = (((locations[i] & 0xFF) << 16) + ((locations[i + 1] & 0xFF) << 8) + ((locations[i + 2] & 0xFF)));
|
int offset = (((locations[i] & 0xFF) << 16) + ((locations[i + 1] & 0xFF) << 8) + ((locations[i + 2] & 0xFF)));
|
||||||
return offset << 12;
|
return offset << 12;
|
||||||
}
|
}
|
||||||
|
|
||||||
public int getSize(int cx, int cz) {
|
public int getSize(int cx, int cz) {
|
||||||
int i = ((cx & 31) << 2) + ((cz & 31) << 7);
|
int i = getIndex(cx, cz);
|
||||||
return (locations[i + 3] & 0xFF) << 12;
|
return (locations[i + 3] & 0xFF) << 12;
|
||||||
}
|
}
|
||||||
|
|
||||||
public List<Integer> getChunks() {
|
|
||||||
final List<Integer> values;
|
|
||||||
synchronized (chunks) {
|
|
||||||
values = new ArrayList<>(chunks.size());
|
|
||||||
}
|
|
||||||
for (int i = 0; i < locations.length; i += 4) {
|
|
||||||
int offset = (((locations[i] & 0xFF) << 16) + ((locations[i + 1] & 0xFF) << 8) + ((locations[i + 2] & 0xFF)));
|
|
||||||
values.add(offset);
|
|
||||||
}
|
|
||||||
return values;
|
|
||||||
}
|
|
||||||
|
|
||||||
public byte[] getChunkCompressedBytes(int offset) throws IOException {
|
public byte[] getChunkCompressedBytes(int offset) throws IOException {
|
||||||
if (offset == 0) {
|
if (offset == 0) {
|
||||||
return null;
|
return null;
|
||||||
@ -363,25 +396,28 @@ public class MCAFile {
|
|||||||
/**
|
/**
|
||||||
* @param onEach chunk
|
* @param onEach chunk
|
||||||
*/
|
*/
|
||||||
public void forEachCachedChunk(RunnableVal<MCAChunk> onEach) {
|
public void forEachCachedChunk(Consumer<MCAChunk> onEach) {
|
||||||
synchronized (chunks) {
|
for (int i = 0; i < chunks.length; i++) {
|
||||||
for (Map.Entry<Integer, MCAChunk> entry : chunks.entrySet()) {
|
MCAChunk chunk = chunks[i];
|
||||||
onEach.run(entry.getValue());
|
if (chunk != null && this.chunkInitialized[i]) {
|
||||||
|
onEach.accept(chunk);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public List<MCAChunk> getCachedChunks() {
|
public List<MCAChunk> getCachedChunks() {
|
||||||
synchronized (chunks) {
|
int size = 0;
|
||||||
return new ArrayList<>(chunks.values());
|
for (int i = 0; i < chunks.length; i++) {
|
||||||
|
if (chunks[i] != null && this.chunkInitialized[i]) size++;
|
||||||
|
}
|
||||||
|
ArrayList<MCAChunk> list = new ArrayList<>(size);
|
||||||
|
for (int i = 0; i < chunks.length; i++) {
|
||||||
|
MCAChunk chunk = chunks[i];
|
||||||
|
if (chunk != null && this.chunkInitialized[i]) {
|
||||||
|
list.add(chunk);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return list;
|
||||||
public void uncache(int cx, int cz) {
|
|
||||||
int pair = MathMan.pair((short) (cx & 31), (short) (cz & 31));
|
|
||||||
synchronized (chunks) {
|
|
||||||
chunks.remove(pair);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private byte[] toBytes(MCAChunk chunk) throws Exception {
|
private byte[] toBytes(MCAChunk chunk) throws Exception {
|
||||||
@ -420,7 +456,7 @@ public class MCAFile {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void writeHeader(RandomAccessFile raf, int cx, int cz, int offsetMedium, int sizeByte, boolean writeTime) throws IOException {
|
private void writeHeader(RandomAccessFile raf, int cx, int cz, int offsetMedium, int sizeByte, boolean writeTime) throws IOException {
|
||||||
int i = ((cx & 31) << 2) + ((cz & 31) << 7);
|
int i = getIndex(cx, cz);
|
||||||
locations[i] = (byte) (offsetMedium >> 16);
|
locations[i] = (byte) (offsetMedium >> 16);
|
||||||
locations[i + 1] = (byte) (offsetMedium >> 8);
|
locations[i + 1] = (byte) (offsetMedium >> 8);
|
||||||
locations[i + 2] = (byte) (offsetMedium);
|
locations[i + 2] = (byte) (offsetMedium);
|
||||||
@ -449,7 +485,6 @@ public class MCAFile {
|
|||||||
e.printStackTrace();
|
e.printStackTrace();
|
||||||
}
|
}
|
||||||
raf = null;
|
raf = null;
|
||||||
locations = null;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -459,13 +494,15 @@ public class MCAFile {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
synchronized (chunks) {
|
synchronized (chunks) {
|
||||||
for (Int2ObjectMap.Entry<MCAChunk> entry : chunks.int2ObjectEntrySet()) {
|
for (int i = 0; i < chunks.length; i++) {
|
||||||
MCAChunk chunk = entry.getValue();
|
MCAChunk chunk = chunks[i];
|
||||||
|
if (chunk != null && this.chunkInitialized[i]) {
|
||||||
if (chunk.isModified() || chunk.isDeleted()) {
|
if (chunk.isModified() || chunk.isDeleted()) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -571,7 +608,7 @@ public class MCAFile {
|
|||||||
|
|
||||||
nextOffset += size;
|
nextOffset += size;
|
||||||
end = Math.min(start + size, end);
|
end = Math.min(start + size, end);
|
||||||
int pair = MathMan.pair((short) (cx & 31), (short) (cz & 31));
|
int pair = getIndex(cx, cz);
|
||||||
byte[] newBytes = relocate.get(pair);
|
byte[] newBytes = relocate.get(pair);
|
||||||
|
|
||||||
// newBytes is null if the chunk isn't modified or marked for moving
|
// newBytes is null if the chunk isn't modified or marked for moving
|
||||||
|
@ -17,6 +17,7 @@ import com.sk89q.worldedit.world.block.BlockState;
|
|||||||
import com.sk89q.worldedit.world.block.BlockStateHolder;
|
import com.sk89q.worldedit.world.block.BlockStateHolder;
|
||||||
|
|
||||||
import javax.annotation.Nullable;
|
import javax.annotation.Nullable;
|
||||||
|
import java.io.IOException;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
@ -67,11 +68,16 @@ public class CPUOptimizedClipboard extends LinearClipboard {
|
|||||||
public void streamBiomes(IntValueReader task) {
|
public void streamBiomes(IntValueReader task) {
|
||||||
if (!hasBiomes()) return;
|
if (!hasBiomes()) return;
|
||||||
int index = 0;
|
int index = 0;
|
||||||
|
try {
|
||||||
for (int z = 0; z < getLength(); z++) {
|
for (int z = 0; z < getLength(); z++) {
|
||||||
for (int x = 0; x < getWidth(); x++, index++) {
|
for (int x = 0; x < getWidth(); x++, index++) {
|
||||||
task.applyInt(index, biomes[index].getInternalId());
|
task.applyInt(index, biomes[index].getInternalId());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
} catch (IOException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -22,6 +22,7 @@ import com.sk89q.worldedit.world.block.BlockType;
|
|||||||
import com.sk89q.worldedit.world.block.BlockTypes;
|
import com.sk89q.worldedit.world.block.BlockTypes;
|
||||||
|
|
||||||
import javax.annotation.Nullable;
|
import javax.annotation.Nullable;
|
||||||
|
import java.io.IOException;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
@ -102,11 +103,16 @@ public class MemoryOptimizedClipboard extends LinearClipboard {
|
|||||||
public void streamBiomes(IntValueReader task) {
|
public void streamBiomes(IntValueReader task) {
|
||||||
if (!hasBiomes()) return;
|
if (!hasBiomes()) return;
|
||||||
int index = 0;
|
int index = 0;
|
||||||
|
try {
|
||||||
for (int z = 0; z < getLength(); z++) {
|
for (int z = 0; z < getLength(); z++) {
|
||||||
for (int x = 0; x < getWidth(); x++, index++) {
|
for (int x = 0; x < getWidth(); x++, index++) {
|
||||||
task.applyInt(index, biomes[index] & 0xFF);
|
task.applyInt(index, biomes[index] & 0xFF);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
} catch (IOException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -21,11 +21,6 @@ package com.sk89q.worldedit.extent.clipboard.io;
|
|||||||
|
|
||||||
import com.boydti.fawe.Fawe;
|
import com.boydti.fawe.Fawe;
|
||||||
import com.boydti.fawe.FaweCache;
|
import com.boydti.fawe.FaweCache;
|
||||||
import com.boydti.fawe.jnbt.CorruptSchematicStreamer;
|
|
||||||
import com.boydti.fawe.jnbt.SchematicStreamer;
|
|
||||||
|
|
||||||
import static com.google.common.base.Preconditions.checkNotNull;
|
|
||||||
|
|
||||||
import com.boydti.fawe.jnbt.streamer.InfoReader;
|
import com.boydti.fawe.jnbt.streamer.InfoReader;
|
||||||
import com.boydti.fawe.jnbt.streamer.IntValueReader;
|
import com.boydti.fawe.jnbt.streamer.IntValueReader;
|
||||||
import com.boydti.fawe.jnbt.streamer.StreamDelegate;
|
import com.boydti.fawe.jnbt.streamer.StreamDelegate;
|
||||||
@ -35,14 +30,9 @@ import com.boydti.fawe.object.FaweOutputStream;
|
|||||||
import com.boydti.fawe.object.clipboard.LinearClipboard;
|
import com.boydti.fawe.object.clipboard.LinearClipboard;
|
||||||
import com.boydti.fawe.object.io.FastByteArrayOutputStream;
|
import com.boydti.fawe.object.io.FastByteArrayOutputStream;
|
||||||
import com.boydti.fawe.object.io.FastByteArraysInputStream;
|
import com.boydti.fawe.object.io.FastByteArraysInputStream;
|
||||||
import com.google.common.collect.ImmutableList;
|
|
||||||
import com.sk89q.jnbt.CompoundTag;
|
import com.sk89q.jnbt.CompoundTag;
|
||||||
import com.sk89q.jnbt.IntTag;
|
|
||||||
import com.sk89q.jnbt.NBTInputStream;
|
import com.sk89q.jnbt.NBTInputStream;
|
||||||
import com.sk89q.jnbt.StringTag;
|
|
||||||
import com.sk89q.jnbt.Tag;
|
|
||||||
import com.sk89q.worldedit.entity.BaseEntity;
|
import com.sk89q.worldedit.entity.BaseEntity;
|
||||||
import com.sk89q.worldedit.extension.input.InputParseException;
|
|
||||||
import com.sk89q.worldedit.extent.clipboard.Clipboard;
|
import com.sk89q.worldedit.extent.clipboard.Clipboard;
|
||||||
import com.sk89q.worldedit.extent.clipboard.io.legacycompat.EntityNBTCompatibilityHandler;
|
import com.sk89q.worldedit.extent.clipboard.io.legacycompat.EntityNBTCompatibilityHandler;
|
||||||
import com.sk89q.worldedit.extent.clipboard.io.legacycompat.FlowerPotCompatibilityHandler;
|
import com.sk89q.worldedit.extent.clipboard.io.legacycompat.FlowerPotCompatibilityHandler;
|
||||||
@ -55,7 +45,6 @@ import com.sk89q.worldedit.math.BlockVector3;
|
|||||||
import com.sk89q.worldedit.registry.state.PropertyKey;
|
import com.sk89q.worldedit.registry.state.PropertyKey;
|
||||||
import com.sk89q.worldedit.util.Direction;
|
import com.sk89q.worldedit.util.Direction;
|
||||||
import com.sk89q.worldedit.util.Location;
|
import com.sk89q.worldedit.util.Location;
|
||||||
import com.sk89q.worldedit.world.DataFixer;
|
|
||||||
import com.sk89q.worldedit.world.biome.BiomeType;
|
import com.sk89q.worldedit.world.biome.BiomeType;
|
||||||
import com.sk89q.worldedit.world.biome.BiomeTypes;
|
import com.sk89q.worldedit.world.biome.BiomeTypes;
|
||||||
import com.sk89q.worldedit.world.block.BaseBlock;
|
import com.sk89q.worldedit.world.block.BaseBlock;
|
||||||
@ -81,6 +70,8 @@ import java.util.Map;
|
|||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
import java.util.function.Function;
|
import java.util.function.Function;
|
||||||
|
|
||||||
|
import static com.google.common.base.Preconditions.checkNotNull;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Reads schematic files based that are compatible with MCEdit and other editors.
|
* Reads schematic files based that are compatible with MCEdit and other editors.
|
||||||
*/
|
*/
|
||||||
|
Loading…
Reference in New Issue
Block a user