2015-03-13 18:52:59 +00:00
|
|
|
/* Copyright (c) 2008-2015, Avian Contributors
|
2008-11-11 15:20:49 +00:00
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software
|
|
|
|
for any purpose with or without fee is hereby granted, provided
|
|
|
|
that the above copyright notice and this permission notice appear
|
|
|
|
in all copies.
|
|
|
|
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
|
|
details. */
|
|
|
|
|
|
|
|
package java.io;
|
|
|
|
|
2013-01-22 19:10:16 +00:00
|
|
|
import java.lang.IllegalArgumentException;
|
2013-10-17 19:50:43 +00:00
|
|
|
import java.nio.ByteBuffer;
|
|
|
|
import java.nio.channels.FileChannel;
|
2013-01-22 19:10:16 +00:00
|
|
|
|
2015-07-29 02:38:59 +00:00
|
|
|
public class RandomAccessFile implements DataInput, Closeable {
|
|
|
|
|
2008-11-11 15:20:49 +00:00
|
|
|
private long peer;
|
2011-11-03 19:39:24 +00:00
|
|
|
private File file;
|
2008-11-11 15:20:49 +00:00
|
|
|
private long position = 0;
|
2011-11-04 01:30:44 +00:00
|
|
|
private long length;
|
2013-10-17 19:40:21 +00:00
|
|
|
private boolean allowWrite;
|
2011-11-03 19:39:24 +00:00
|
|
|
|
2008-11-11 15:20:49 +00:00
|
|
|
public RandomAccessFile(String name, String mode)
|
|
|
|
throws FileNotFoundException
|
|
|
|
{
|
2013-10-17 19:46:30 +00:00
|
|
|
this(new File(name), mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
public RandomAccessFile(File file, String mode)
|
|
|
|
throws FileNotFoundException
|
|
|
|
{
|
|
|
|
if (file == null) throw new NullPointerException();
|
2013-10-17 19:40:21 +00:00
|
|
|
if (mode.equals("rw")) allowWrite = true;
|
|
|
|
else if (! mode.equals("r")) throw new IllegalArgumentException();
|
2013-10-17 19:46:30 +00:00
|
|
|
this.file = file;
|
2011-11-04 01:30:44 +00:00
|
|
|
open();
|
|
|
|
}
|
2008-11-11 15:20:49 +00:00
|
|
|
|
2011-11-04 01:30:44 +00:00
|
|
|
private void open() throws FileNotFoundException {
|
2008-11-11 15:20:49 +00:00
|
|
|
long[] result = new long[2];
|
2013-10-17 19:40:21 +00:00
|
|
|
open(file.getPath(), allowWrite, result);
|
2008-11-11 15:20:49 +00:00
|
|
|
peer = result[0];
|
2011-11-04 01:30:44 +00:00
|
|
|
length = result[1];
|
2008-11-11 15:20:49 +00:00
|
|
|
}
|
|
|
|
|
2013-10-17 19:40:21 +00:00
|
|
|
private static native void open(String name, boolean allowWrite, long[] result)
|
2008-11-11 15:20:49 +00:00
|
|
|
throws FileNotFoundException;
|
|
|
|
|
2011-11-04 01:30:44 +00:00
|
|
|
private void refresh() throws IOException {
|
|
|
|
if (file.length() != length) {
|
|
|
|
close();
|
|
|
|
open();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-11-11 15:20:49 +00:00
|
|
|
public long length() throws IOException {
|
2011-11-04 01:30:44 +00:00
|
|
|
refresh();
|
|
|
|
return length;
|
2008-11-11 15:20:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public long getFilePointer() throws IOException {
|
|
|
|
return position;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void seek(long position) throws IOException {
|
2013-10-17 19:40:21 +00:00
|
|
|
if (position < 0 || (!allowWrite && position > length())) throw new IOException();
|
2008-11-11 15:20:49 +00:00
|
|
|
|
|
|
|
this.position = position;
|
|
|
|
}
|
|
|
|
|
2013-01-21 21:43:29 +00:00
|
|
|
public int skipBytes(int count) throws IOException {
|
|
|
|
if (position + count > length()) throw new IOException();
|
|
|
|
this.position = position + count;
|
|
|
|
return count;
|
|
|
|
}
|
2013-01-22 19:10:16 +00:00
|
|
|
|
|
|
|
public int read(byte b[], int off, int len) throws IOException {
|
|
|
|
if(b == null)
|
2015-07-29 02:38:59 +00:00
|
|
|
throw new IllegalArgumentException();
|
2013-01-22 19:10:16 +00:00
|
|
|
if (peer == 0)
|
2015-07-29 02:38:59 +00:00
|
|
|
throw new IOException();
|
|
|
|
if(len == 0)
|
|
|
|
return 0;
|
|
|
|
if (position + len > this.length)
|
2013-01-22 19:10:16 +00:00
|
|
|
throw new EOFException();
|
2015-07-29 02:38:59 +00:00
|
|
|
if (off < 0 || off + len > b.length)
|
2008-11-11 15:20:49 +00:00
|
|
|
throw new ArrayIndexOutOfBoundsException();
|
2013-01-22 19:10:16 +00:00
|
|
|
int bytesRead = readBytes(peer, position, b, off, len);
|
2015-07-29 02:38:59 +00:00
|
|
|
position += bytesRead;
|
|
|
|
return bytesRead;
|
2013-01-22 19:10:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public int read(byte b[]) throws IOException {
|
|
|
|
if(b == null)
|
2015-07-29 02:38:59 +00:00
|
|
|
throw new IllegalArgumentException();
|
2013-01-22 19:10:16 +00:00
|
|
|
if (peer == 0)
|
2015-07-29 02:38:59 +00:00
|
|
|
throw new IOException();
|
|
|
|
if(b.length == 0)
|
|
|
|
return 0;
|
|
|
|
if (position + b.length > this.length)
|
2013-01-22 19:10:16 +00:00
|
|
|
throw new EOFException();
|
|
|
|
int bytesRead = readBytes(peer, position, b, 0, b.length);
|
2015-07-29 02:38:59 +00:00
|
|
|
position += bytesRead;
|
|
|
|
return bytesRead;
|
2013-01-22 19:10:16 +00:00
|
|
|
}
|
2008-11-11 15:20:49 +00:00
|
|
|
|
2013-01-22 19:10:16 +00:00
|
|
|
public void readFully(byte b[], int off, int len) throws IOException {
|
2015-07-29 02:38:59 +00:00
|
|
|
if (b == null)
|
|
|
|
throw new IllegalArgumentException();
|
2013-01-22 19:10:16 +00:00
|
|
|
if (peer == 0)
|
2015-07-29 02:38:59 +00:00
|
|
|
throw new IOException();
|
|
|
|
if(len == 0)
|
|
|
|
return;
|
|
|
|
if (position + len > this.length)
|
2013-01-22 19:10:16 +00:00
|
|
|
throw new EOFException();
|
2015-07-29 02:38:59 +00:00
|
|
|
if (off < 0 || off + len > b.length)
|
2013-01-22 19:10:16 +00:00
|
|
|
throw new ArrayIndexOutOfBoundsException();
|
|
|
|
int n = 0;
|
|
|
|
do {
|
|
|
|
int count = readBytes(peer, position, b, off + n, len - n);
|
2013-01-22 19:25:37 +00:00
|
|
|
position += count;
|
|
|
|
if (count == 0)
|
2013-01-22 19:10:16 +00:00
|
|
|
throw new EOFException();
|
|
|
|
n += count;
|
|
|
|
} while (n < len);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void readFully(byte b[]) throws IOException {
|
|
|
|
readFully(b, 0, b.length);
|
2008-11-11 15:20:49 +00:00
|
|
|
}
|
|
|
|
|
2013-01-22 19:10:16 +00:00
|
|
|
private static native int readBytes(long peer, long position, byte[] buffer,
|
2008-11-11 15:20:49 +00:00
|
|
|
int offset, int length);
|
|
|
|
|
2015-07-29 02:38:59 +00:00
|
|
|
public boolean readBoolean() throws IOException {
|
|
|
|
return readByte() != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int read() throws IOException {
|
|
|
|
try {
|
|
|
|
return readByte() & 0xff;
|
|
|
|
} catch (final EOFException e) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public byte readByte() throws IOException {
|
|
|
|
final byte[] buffer = new byte[1];
|
|
|
|
readFully(buffer);
|
|
|
|
return buffer[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
public short readShort() throws IOException {
|
|
|
|
final byte[] buffer = new byte[2];
|
|
|
|
readFully(buffer);
|
|
|
|
return (short)((buffer[0] << 8) | buffer[1]);
|
|
|
|
}
|
|
|
|
|
|
|
|
public int readInt() throws IOException {
|
|
|
|
byte[] buf = new byte[4];
|
|
|
|
readFully(buf);
|
|
|
|
return ((buf[0] << 24) | (buf[1] << 16) | (buf[2] << 8) | buf[3]);
|
|
|
|
}
|
|
|
|
|
|
|
|
public float readFloat() throws IOException {
|
|
|
|
return Float.floatToIntBits(readInt());
|
|
|
|
}
|
|
|
|
|
|
|
|
public double readDouble() throws IOException {
|
|
|
|
return Double.doubleToLongBits(readLong());
|
|
|
|
}
|
|
|
|
|
|
|
|
public long readLong() throws IOException {
|
|
|
|
return ((readInt() & 0xffffffffl) << 32) | (readInt() & 0xffffffffl);
|
|
|
|
}
|
|
|
|
|
|
|
|
public char readChar() throws IOException {
|
|
|
|
return (char)readShort();
|
|
|
|
}
|
|
|
|
|
|
|
|
public int readUnsignedByte() throws IOException {
|
|
|
|
return readByte() & 0xff;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int readUnsignedShort() throws IOException {
|
|
|
|
return readShort() & 0xffff;
|
|
|
|
}
|
|
|
|
|
|
|
|
public String readUTF() throws IOException {
|
|
|
|
int length = readUnsignedShort();
|
|
|
|
byte[] bytes = new byte[length];
|
|
|
|
readFully(bytes);
|
|
|
|
return new String(bytes, "UTF-8");
|
|
|
|
}
|
|
|
|
|
|
|
|
@Deprecated
|
|
|
|
public String readLine() throws IOException {
|
|
|
|
int c = read();
|
|
|
|
if (c < 0) {
|
|
|
|
return null;
|
|
|
|
} else if (c == '\n') {
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
StringBuilder builder = new StringBuilder();
|
|
|
|
for (;;) {
|
|
|
|
builder.append((char)c);
|
|
|
|
c = read();
|
|
|
|
if (c < 0 || c == '\n') {
|
|
|
|
return builder.toString();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-17 19:46:00 +00:00
|
|
|
public void write(int b) throws IOException {
|
|
|
|
int count = writeBytes(peer, position, new byte[] { (byte)b }, 0, 1);
|
|
|
|
if (count > 0) position += count;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static native int writeBytes(long peer, long position, byte[] buffer,
|
|
|
|
int offset, int length);
|
|
|
|
|
2008-11-11 15:20:49 +00:00
|
|
|
public void close() throws IOException {
|
|
|
|
if (peer != 0) {
|
|
|
|
close(peer);
|
|
|
|
peer = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static native void close(long peer);
|
2013-10-17 19:50:43 +00:00
|
|
|
|
|
|
|
public FileChannel getChannel() {
|
|
|
|
return new FileChannel() {
|
|
|
|
public void close() {
|
|
|
|
if (peer != 0) RandomAccessFile.close(peer);
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean isOpen() {
|
|
|
|
return peer != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int read(ByteBuffer dst, long position) throws IOException {
|
|
|
|
if (!dst.hasArray()) throw new IOException("Cannot handle " + dst.getClass());
|
|
|
|
// TODO: this needs to be synchronized on the Buffer, no?
|
|
|
|
byte[] array = dst.array();
|
|
|
|
return readBytes(peer, position, array, dst.position(), dst.remaining());
|
|
|
|
}
|
|
|
|
|
|
|
|
public int read(ByteBuffer dst) throws IOException {
|
|
|
|
int count = read(dst, position);
|
|
|
|
if (count > 0) position += count;
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int write(ByteBuffer src, long position) throws IOException {
|
|
|
|
if (!src.hasArray()) throw new IOException("Cannot handle " + src.getClass());
|
|
|
|
byte[] array = src.array();
|
|
|
|
return writeBytes(peer, position, array, src.position(), src.remaining());
|
|
|
|
}
|
|
|
|
|
|
|
|
public int write(ByteBuffer src) throws IOException {
|
|
|
|
int count = write(src, position);
|
|
|
|
if (count > 0) position += count;
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
public long position() throws IOException {
|
|
|
|
return getFilePointer();
|
|
|
|
}
|
|
|
|
|
|
|
|
public FileChannel position(long position) throws IOException {
|
|
|
|
seek(position);
|
|
|
|
return this;
|
|
|
|
}
|
2013-12-03 20:07:37 +00:00
|
|
|
|
|
|
|
public long size() throws IOException {
|
|
|
|
return length();
|
|
|
|
}
|
2013-10-17 19:50:43 +00:00
|
|
|
};
|
|
|
|
}
|
2008-11-11 15:20:49 +00:00
|
|
|
}
|