modules/cldc-compact/src/main/java/java/io/ByteArrayInputStream.java
// -*- Mode: Java; indent-tabs-mode: t; tab-width: 4 -*-
// ---------------------------------------------------------------------------
// SquirrelJME
// Copyright (C) Stephanie Gawroriski <xer@multiphasicapps.net>
// ---------------------------------------------------------------------------
// SquirrelJME is under the Mozilla Public License Version 2.0.
// See license.mkd for licensing and copyright information.
// ---------------------------------------------------------------------------
package java.io;
import cc.squirreljme.runtime.cldc.annotation.Api;
/**
* This is a class which reads from an input byte array and returns its data.
*
* @since 2018/11/11
*/
@Api
public class ByteArrayInputStream
extends InputStream
{
/** The buffer to read from. */
@Api
protected byte[] buf;
/** The number of bytes to read. */
@Api
protected int count;
/** The current mark. */
@Api
protected int mark;
/** The position within the byte array. */
@Api
protected int pos;
/**
* Wraps the specified byte array.
*
* @param __b The array to wrap.
* @throws NullPointerException On null arguments.
* @since 2018/11/11
*/
@Api
public ByteArrayInputStream(byte[] __b)
throws NullPointerException
{
this(__b, 0, __b.length);
}
/**
* Wraps the specified byte array.
*
* @param __b The array to wrap.
* @param __o The offset.
* @param __l The length.
* @throws NullPointerException On null arguments.
* @since 2018/11/11
*/
@Api
public ByteArrayInputStream(byte[] __b, int __o, int __l)
throws NullPointerException
{
if (__b == null)
throw new NullPointerException("NARG");
this.buf = __b;
this.pos = __o;
this.mark = __o;
this.count = Math.min(__o + __l, __b.length);
}
/**
* {@inheritDoc}
* @since 2018/11/11
*/
@Override
public int available()
{
synchronized (this)
{
return Math.max(0, this.count - Math.max(0, this.pos));
}
}
/**
* {@inheritDoc}
* @since 2018/11/11
*/
@Override
public void close()
throws IOException
{
// Does nothing
}
/**
* {@inheritDoc}
* @since 2018/11/11
*/
@Override
public void mark(int __p)
{
synchronized (this)
{
this.mark = this.pos;
}
}
/**
* {@inheritDoc}
* @since 2018/11/11
*/
@Override
public boolean markSupported()
{
return true;
}
/**
* {@inheritDoc}
* @since 2018/11/11
*/
@Override
public int read()
{
synchronized (this)
{
int pos = this.pos,
count = this.count;
// EOF?
if (pos >= count)
return -1;
// Read single byte
int rv = this.buf[pos] & 0xFF;
this.pos = pos + 1;
return rv;
}
}
/**
* {@inheritDoc}
* @since 2018/12/03
*/
@Override
public int read(byte[] __b)
throws NullPointerException
{
if (__b == null)
throw new NullPointerException("NARG");
// Only read this many bytes
int len = __b.length;
synchronized (this)
{
// Determine how many bytes we can read
byte[] buf = this.buf;
int pos = this.pos,
count = this.count,
read = count - pos;
// Do not exceed this
if (len < read)
read = len;
// Copy bytes
System.arraycopy(buf, pos, __b, 0, read);
pos += read;
// Store new position
this.pos = pos;
if (read == 0)
return (pos >= count ? -1 : 0);
return read;
}
}
/**
* {@inheritDoc}
* @since 2018/11/11
*/
@Override
public int read(byte[] __b, int __o, int __l)
throws IndexOutOfBoundsException, NullPointerException
{
if (__b == null)
throw new NullPointerException("NARG");
if (__o < 0 || __l < 0 || (__o + __l) < 0 || (__o + __l) > __b.length)
throw new IndexOutOfBoundsException("IOOB");
synchronized (this)
{
// Determine how many bytes we can read
byte[] buf = this.buf;
int pos = this.pos,
count = this.count,
read = count - pos;
// Do not exceed this
if (__l < read)
read = __l;
// Copy bytes
System.arraycopy(buf, pos, __b, __o, read);
pos += read;
// Store new position
this.pos = pos;
if (read == 0)
return (pos >= count ? -1 : 0);
return read;
}
}
/**
* {@inheritDoc}
* @since 2018/11/11
*/
@Override
public void reset()
{
synchronized (this)
{
this.pos = this.mark;
}
}
/**
* {@inheritDoc}
* @since 2018/12/08
*/
@Override
public long skip(long __n)
{
// Doing nothing?
if (__n <= 0)
return 0;
synchronized (this)
{
int pos = this.pos,
count = this.count,
read = count - pos,
newpos;
// Reading less bytes than what is available
if (__n < read)
this.pos = (newpos = (int)(pos + __n));
// No more bytes to read
else
this.pos = (newpos = count);
return newpos - pos;
}
}
}