mirror of
https://github.com/corda/corda.git
synced 2025-01-22 12:28:11 +00:00
6e7149061c
The main idea is to make DatagramChannel and *SocketChannel behave in a way that more closely matches the standard, e.g. allow binding sockets to addresses without necessarily listening on those addresses and accept null addresses where appropriate. It also avoids multiple redundant DNS lookups. This commit also implements CharBuffer and BindException, and adds the Readable interface.
305 lines
7.3 KiB
Java
305 lines
7.3 KiB
Java
/* Copyright (c) 2008-2013, Avian Contributors
|
|
|
|
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.nio;
|
|
|
|
public abstract class ByteBuffer
|
|
extends Buffer
|
|
implements Comparable<ByteBuffer>
|
|
{
|
|
private final boolean readOnly;
|
|
|
|
protected ByteBuffer(boolean readOnly) {
|
|
this.readOnly = readOnly;
|
|
}
|
|
|
|
public static ByteBuffer allocate(int capacity) {
|
|
return new ArrayByteBuffer(new byte[capacity], 0, capacity, false);
|
|
}
|
|
|
|
public static ByteBuffer allocateDirect(int capacity) {
|
|
return FixedArrayByteBuffer.make(capacity);
|
|
}
|
|
|
|
public static ByteBuffer wrap(byte[] array) {
|
|
return wrap(array, 0, array.length);
|
|
}
|
|
|
|
public static ByteBuffer wrap(byte[] array, int offset, int length) {
|
|
return new ArrayByteBuffer(array, offset, length, false);
|
|
}
|
|
|
|
public abstract ByteBuffer asReadOnlyBuffer();
|
|
|
|
public abstract ByteBuffer slice();
|
|
|
|
protected abstract void doPut(int offset, byte val);
|
|
|
|
public abstract ByteBuffer put(byte[] arr, int offset, int len);
|
|
|
|
protected abstract byte doGet(int offset);
|
|
|
|
public abstract ByteBuffer get(byte[] dst, int offset, int length);
|
|
|
|
public boolean hasArray() {
|
|
return false;
|
|
}
|
|
|
|
public ByteBuffer compact() {
|
|
int remaining = remaining();
|
|
|
|
if (position != 0) {
|
|
ByteBuffer b = slice();
|
|
position = 0;
|
|
put(b);
|
|
}
|
|
|
|
position = remaining;
|
|
limit(capacity());
|
|
|
|
return this;
|
|
}
|
|
|
|
public ByteBuffer put(ByteBuffer src) {
|
|
if (src.hasArray()) {
|
|
checkPut(position, src.remaining(), false);
|
|
|
|
put(src.array(), src.arrayOffset() + src.position, src.remaining());
|
|
src.position(src.position() + src.remaining());
|
|
|
|
return this;
|
|
} else {
|
|
byte[] buffer = new byte[src.remaining()];
|
|
src.get(buffer);
|
|
return put(buffer);
|
|
}
|
|
}
|
|
|
|
public int compareTo(ByteBuffer o) {
|
|
int end = (remaining() < o.remaining() ? remaining() : o.remaining());
|
|
|
|
for (int i = 0; i < end; ++i) {
|
|
int d = get(position + i) - o.get(o.position + i);
|
|
if (d != 0) {
|
|
return d;
|
|
}
|
|
}
|
|
return remaining() - o.remaining();
|
|
}
|
|
|
|
public boolean equals(Object o) {
|
|
return o instanceof ByteBuffer && compareTo((ByteBuffer) o) == 0;
|
|
}
|
|
|
|
public byte[] array() {
|
|
throw new UnsupportedOperationException();
|
|
}
|
|
|
|
public int arrayOffset() {
|
|
throw new UnsupportedOperationException();
|
|
}
|
|
|
|
public ByteBuffer put(int offset, byte val) {
|
|
checkPut(offset, 1, true);
|
|
doPut(offset, val);
|
|
return this;
|
|
}
|
|
|
|
public ByteBuffer put(byte val) {
|
|
checkPut(position, 1, false);
|
|
doPut(position, val);
|
|
++ position;
|
|
return this;
|
|
}
|
|
|
|
public ByteBuffer put(byte[] arr) {
|
|
return put(arr, 0, arr.length);
|
|
}
|
|
|
|
private void rawPutLong(int position, long val) {
|
|
doPut(position , (byte) ((val >> 56) & 0xff));
|
|
doPut(position + 1, (byte) ((val >> 48) & 0xff));
|
|
doPut(position + 2, (byte) ((val >> 40) & 0xff));
|
|
doPut(position + 3, (byte) ((val >> 32) & 0xff));
|
|
doPut(position + 4, (byte) ((val >> 24) & 0xff));
|
|
doPut(position + 5, (byte) ((val >> 16) & 0xff));
|
|
doPut(position + 6, (byte) ((val >> 8) & 0xff));
|
|
doPut(position + 7, (byte) ((val ) & 0xff));
|
|
}
|
|
|
|
private void rawPutInt(int position, int val) {
|
|
doPut(position , (byte) ((val >> 24) & 0xff));
|
|
doPut(position + 1, (byte) ((val >> 16) & 0xff));
|
|
doPut(position + 2, (byte) ((val >> 8) & 0xff));
|
|
doPut(position + 3, (byte) ((val ) & 0xff));
|
|
}
|
|
|
|
private void rawPutShort(int position, short val) {
|
|
doPut(position , (byte) ((val >> 8) & 0xff));
|
|
doPut(position + 1, (byte) ((val ) & 0xff));
|
|
}
|
|
|
|
public ByteBuffer putLong(int position, long val) {
|
|
checkPut(position, 8, true);
|
|
|
|
rawPutLong(position, val);
|
|
|
|
return this;
|
|
}
|
|
|
|
public ByteBuffer putInt(int position, int val) {
|
|
checkPut(position, 4, true);
|
|
|
|
rawPutInt(position, val);
|
|
|
|
return this;
|
|
}
|
|
|
|
public ByteBuffer putShort(int position, short val) {
|
|
checkPut(position, 2, true);
|
|
|
|
rawPutShort(position, val);
|
|
|
|
return this;
|
|
}
|
|
|
|
public ByteBuffer putLong(long val) {
|
|
checkPut(position, 8, false);
|
|
|
|
rawPutLong(position, val);
|
|
position += 8;
|
|
return this;
|
|
}
|
|
|
|
public ByteBuffer putInt(int val) {
|
|
checkPut(position, 4, false);
|
|
|
|
rawPutInt(position, val);
|
|
position += 4;
|
|
return this;
|
|
}
|
|
|
|
public ByteBuffer putShort(short val) {
|
|
checkPut(position, 2, false);
|
|
|
|
rawPutShort(position, val);
|
|
position += 2;
|
|
return this;
|
|
}
|
|
|
|
public byte get() {
|
|
checkGet(position, 1, false);
|
|
return doGet(position++);
|
|
}
|
|
|
|
public byte get(int position) {
|
|
checkGet(position, 1, true);
|
|
return doGet(position);
|
|
}
|
|
|
|
public ByteBuffer get(byte[] dst) {
|
|
return get(dst, 0, dst.length);
|
|
}
|
|
|
|
public long getLong(int position) {
|
|
checkGet(position, 8, true);
|
|
|
|
return rawGetLong(position);
|
|
}
|
|
|
|
public int getInt(int position) {
|
|
checkGet(position, 4, true);
|
|
|
|
return rawGetInt(position);
|
|
}
|
|
|
|
public short getShort(int position) {
|
|
checkGet(position, 2, true);
|
|
|
|
return rawGetShort(position);
|
|
}
|
|
|
|
private long rawGetLong(int position) {
|
|
return (((long) (doGet(position ) & 0xFF)) << 56)
|
|
| (((long) (doGet(position + 1) & 0xFF)) << 48)
|
|
| (((long) (doGet(position + 2) & 0xFF)) << 40)
|
|
| (((long) (doGet(position + 3) & 0xFF)) << 32)
|
|
| (((long) (doGet(position + 4) & 0xFF)) << 24)
|
|
| (((long) (doGet(position + 5) & 0xFF)) << 16)
|
|
| (((long) (doGet(position + 6) & 0xFF)) << 8)
|
|
| (((long) (doGet(position + 7) & 0xFF)) );
|
|
}
|
|
|
|
private int rawGetInt(int position) {
|
|
return (((int) (doGet(position ) & 0xFF)) << 24)
|
|
| (((int) (doGet(position + 1) & 0xFF)) << 16)
|
|
| (((int) (doGet(position + 2) & 0xFF)) << 8)
|
|
| (((int) (doGet(position + 3) & 0xFF)) );
|
|
}
|
|
|
|
private short rawGetShort(int position) {
|
|
return (short) (( ((int) (doGet(position ) & 0xFF)) << 8)
|
|
| (((int) (doGet(position + 1) & 0xFF)) ));
|
|
}
|
|
|
|
public long getLong() {
|
|
checkGet(position, 8, false);
|
|
|
|
long r = rawGetLong(position);
|
|
position += 8;
|
|
return r;
|
|
}
|
|
|
|
public int getInt() {
|
|
checkGet(position, 4, false);
|
|
|
|
int r = rawGetInt(position);
|
|
position += 4;
|
|
return r;
|
|
}
|
|
|
|
public short getShort() {
|
|
checkGet(position, 2, false);
|
|
|
|
short r = rawGetShort(position);
|
|
position += 2;
|
|
return r;
|
|
}
|
|
|
|
protected void checkPut(int position, int amount, boolean absolute) {
|
|
if (readOnly) {
|
|
throw new ReadOnlyBufferException();
|
|
}
|
|
|
|
if (position < 0 || position+amount > limit) {
|
|
throw absolute
|
|
? new IndexOutOfBoundsException()
|
|
: new BufferOverflowException();
|
|
}
|
|
}
|
|
|
|
protected void checkGet(int position, int amount, boolean absolute) {
|
|
if (amount > limit-position) {
|
|
throw absolute
|
|
? new IndexOutOfBoundsException()
|
|
: new BufferUnderflowException();
|
|
}
|
|
}
|
|
|
|
public ByteBuffer order(ByteOrder order) {
|
|
if (order != ByteOrder.BIG_ENDIAN) throw new UnsupportedOperationException();
|
|
return this;
|
|
}
|
|
|
|
public ByteOrder order() {
|
|
return ByteOrder.BIG_ENDIAN;
|
|
}
|
|
}
|