/*
Реализация среды исполнения языка программирования
Объектно-ориентированный продвинутый векторный транслятор
Copyright © 2021, 2024 Малик Разработчик
Это свободная программа: вы можете перераспространять ее и/или изменять
ее на условиях Меньшей Стандартной общественной лицензии GNU в том виде,
в каком она была опубликована Фондом свободного программного обеспечения;
либо версии 3 лицензии, либо (по вашему выбору) любой более поздней версии.
Эта программа распространяется в надежде, что она будет полезной,
но БЕЗО ВСЯКИХ ГАРАНТИЙ; даже без неявной гарантии ТОВАРНОГО ВИДА
или ПРИГОДНОСТИ ДЛЯ ОПРЕДЕЛЕННЫХ ЦЕЛЕЙ. Подробнее см. в Меньшей Стандартной
общественной лицензии GNU.
Вы должны были получить копию Меньшей Стандартной общественной лицензии GNU
вместе с этой программой. Если это не так, см.
<https://www.gnu.org/licenses/>.
*/
package avt.io;
import avt.io.extension.*;
public class ByteArrayOutputStream(Object, Closeable, Extension, MarkExtension, ByteWriter, TruncateExtension, LimitedSizeExtension, SeekExtension, Extendable)
{
protected int fldMarked;
protected int fldLength;
protected int fldPosition;
protected byte[] fldArray;
public () {
fldArray = new byte[0x7fi];
}
public (byte[] array) {
int slength = fldLength = fldPosition = array == null ? 0 : array.length;
Array.copy(array, 0, fldArray = new byte[slength >= Int.MAX_VALUE - 0x7fi ? Int.MAX_VALUE : slength + 0x7fi], 0, slength);
}
public (byte[] array, int length) {
int alength = array == null ? 0 : array.length;
int slength = fldLength = fldPosition = length < 0 ? 0 : length < alength ? length : alength;
Array.copy(array, 0, fldArray = new byte[slength >= Int.MAX_VALUE - 0x7fi ? Int.MAX_VALUE : slength + 0x7fi], 0, slength);
}
public (byte[] array, int length, int position) {
int alength = array == null ? 0 : array.length;
int slength = fldLength = length < 0 ? 0 : length < alength ? length : alength;
fldPosition = position < 0 ? 0 : position < slength ? position : slength;
Array.copy(array, 0, fldArray = new byte[slength >= Int.MAX_VALUE - 0x7fi ? Int.MAX_VALUE : slength + 0x7fi], 0, slength);
}
public String toString() { return new String(fldArray, 0, fldLength); }
public void close() { }
public void reset() { fldPosition = fldMarked; }
public void mark(int transferLimit) { fldMarked = fldPosition; }
public void write(int byteData) {
int curLength = fldLength;
int curPosition = fldPosition;
int newPosition = curPosition + 1;
byte[] array = newPosition >= 0 && newPosition <= curLength ? fldArray : madeLarger(curLength, newPosition);
array[curPosition] = (byte) byteData;
fldPosition = newPosition;
}
public void write(byte[] src) { write(src, 0, src == null ? 0 : src.length); }
public void write(byte[] src, int offset, int length) {
if(src == null)
{
throw new NullPointerException(String.format(avt.lang.package.getResourceString("null-pointer.argument"), new Object[] { "src" }));
}
Array.checkBounds(src, offset, length);
if(length > 0)
{
int curLength = fldLength;
int curPosition = fldPosition;
int newPosition = curPosition + length;
byte[] array = newPosition >= 0 && newPosition <= curLength ? fldArray : madeLarger(curLength, newPosition);
Array.copy(src, offset, array, curPosition, length);
fldPosition = newPosition;
}
}
public void flush() { }
public void truncate() {
int length = fldLength = fldPosition;
if(fldMarked > length) fldMarked = length;
}
public long available() { return (long) (fldLength - fldPosition); }
public long position() { return (long) fldPosition; }
public long size() { return (long) fldLength; }
public long seek(long offset, int from) {
long position;
long length = (long) fldLength;
switch(from)
{
case BEGIN:
position = offset;
break;
case END:
position = offset + length;
break;
case CURRENT:
position = offset + (long) fldPosition;
break;
default:
throw new IllegalArgumentException(String.format(avt.lang.package.getResourceString("illegal-argument"), new Object[] { "from" }));
}
if(position < 0L) position = 0L;
if(position > length) position = length;
return (long) (fldPosition = (int) position);
}
public Extension[] getExtensions() { return new Extension[] { this }; }
public byte[] toByteArray() {
int length = fldLength;
byte[] result = new byte[length];
Array.copy(fldArray, 0, result, 0, length);
return result;
}
protected byte[] madeLarger(int curLength, int newLength) {
if(newLength < 0)
{
throw new BufferTooLargeError(avt.lang.package.getResourceString("!error.buffer-too-large"));
}
byte[] array = fldArray;
if(array == null)
{
fldArray = array = new byte[newLength >= Int.MAX_VALUE - 0x7fi ? Int.MAX_VALUE : newLength + 0x7fi];
} else
{
int curCapacity = array.length;
if(newLength > curCapacity)
{
int newCapacity = curCapacity << 1 | 1;
if(newCapacity < 0) newCapacity = Int.MAX_VALUE;
if(newCapacity < newLength) newCapacity = newLength >= Int.MAX_VALUE - 0x7fi ? Int.MAX_VALUE : newLength + 0x7fi;
Array.copy(array, 0, fldArray = array = new byte[newCapacity], 0, curLength);
}
}
fldLength = newLength;
return array;
}
}