package ru.codeinside.gws.xml.normalizer.normalize.utils;
import java.io.IOException;
import java.io.OutputStream;
/**
* A simple Unsynced ByteArrayOutputStream
*
* @author raul
*/
public class UnsyncByteArrayOutputStream extends OutputStream {
private static final int INITIAL_SIZE = 8192;
private static ThreadLocal<byte[]> bufCache = new ThreadLocal<byte[]>() {
@Override
protected synchronized byte[] initialValue() {
return new byte[INITIAL_SIZE];
}
};
private byte[] buf;
private int size = INITIAL_SIZE;
private int pos = 0;
public UnsyncByteArrayOutputStream() {
buf = (byte[]) bufCache.get();
}
public void write(byte[] arg0) {
if ((Integer.MAX_VALUE - pos) < arg0.length) {
throw new OutOfMemoryError();
}
int newPos = pos + arg0.length;
if (newPos > size) {
expandSize(newPos);
}
System.arraycopy(arg0, 0, buf, pos, arg0.length);
pos = newPos;
}
public void write(byte[] arg0, int arg1, int arg2) {
if ((Integer.MAX_VALUE - pos) < arg2) {
throw new OutOfMemoryError();
}
int newPos = pos + arg2;
if (newPos > size) {
expandSize(newPos);
}
System.arraycopy(arg0, arg1, buf, pos, arg2);
pos = newPos;
}
public void write(int arg0) {
if ((Integer.MAX_VALUE - pos) == 0) {
throw new OutOfMemoryError();
}
int newPos = pos + 1;
if (newPos > size) {
expandSize(newPos);
}
buf[pos++] = (byte) arg0;
}
public byte[] toByteArray() {
byte result[] = new byte[pos];
System.arraycopy(buf, 0, result, 0, pos);
return result;
}
public void reset() {
pos = 0;
bufCache.remove();
}
@Override
public void close() throws IOException {
bufCache.remove();
}
private void expandSize(int newPos) {
int newSize = size;
while (newPos > newSize) {
newSize = newSize << 1;
// Deal with overflow
if (newSize < 0) {
newSize = Integer.MAX_VALUE;
}
}
byte newBuf[] = new byte[newSize];
System.arraycopy(buf, 0, newBuf, 0, pos);
buf = newBuf;
size = newSize;
}
}