Commit 35d0b14d by Per Bothner Committed by Per Bothner

ByteBuffer.java (endian): Make non-private so other java.nio classes can inherit it.


	* java/nio/ByteBuffer.java (endian):  Make non-private so other
	java.nio classes can inherit it.
	(<init>):  Don't bother clearing array_offset.
	* java/nio/ByteBuffer.java (allocate):  Re-implement using wrap.
	* java/nio/ByteBuffer.java (get(byte[],int,int)):  Check underflow.
	Remove redundant test.

	* java/nio/ByteBufferImpl.java (asCharBuffer, asShortBuffer,
	asIntBuffer, asLongBuffer, asFloatBuffer, asDoubleBuffer):
	Use new XxxViewBufferImpl constructors.
	* java/nio/MappedByteBufferImpl.java:  Likewise.
	* java/nio/DirectByteBufferImpl.java:  Likewise.

	* java/nio/ByteBufferImpl.java:  Remove one constructor.
	Inline super in remaining constructor.
	* java/nio/ByteBuffer.java:  Remove unused constructor.

	* java/nio/ByteBufferImpl.java (shiftDown):  New optimized method.

	* java/nio/ByteBufferImpl.java (get, put):  Add array_offset.
	* java/nio/DirectByteBufferImpl.java (owner):  New field.
	(offset):  Remove unused field.
	(<init>):  Modify one and add another constructor.  Change callers.
	(allocateDirect):  Removed - not used.
	(getImpl, putImpl):  Make static and pass address explicitly,
	to make them useful for MappedByteBufferImpl.
	(get, put):  Check for underflow.  Modify for new getImpl.
	(getImpl):  New native method where target is array.
	(get(byte[],int,int)):  Use the above.
	(adjustAddress):  New static native method.
	(slice, duplicate, asReadOnly):  New implementations.
	* java/nio/natDirectByteBufferImpl.cc (getImpl, putImpl, shiftDown,
	adjustAddress):  New or updated native methods.

From-SVN: r77919
parent 5693912a
...@@ -19,6 +19,40 @@ ...@@ -19,6 +19,40 @@
* java/nio/IntViewBufferImpl.java: Likewise. * java/nio/IntViewBufferImpl.java: Likewise.
* java/nio/LongViewBufferImpl.java: Likewise. * java/nio/LongViewBufferImpl.java: Likewise.
* java/nio/ShortViewBufferImpl.java: Likewise. * java/nio/ShortViewBufferImpl.java: Likewise.
* java/nio/ByteBuffer.java (endian): Make non-private so other
java.nio classes can inherit it.
(<init>): Don't bother clearing array_offset.
* java/nio/ByteBuffer.java (allocate): Re-implement using wrap.
* java/nio/ByteBuffer.java (get(byte[],int,int)): Check underflow.
Remove redundant test.
* java/nio/ByteBufferImpl.java (asCharBuffer, asShortBuffer,
asIntBuffer, asLongBuffer, asFloatBuffer, asDoubleBuffer):
Use new XxxViewBufferImpl constructors.
* java/nio/MappedByteBufferImpl.java: Likewise.
* java/nio/DirectByteBufferImpl.java: Likewise.
* java/nio/ByteBufferImpl.java: Remove one constructor.
Inline super in remaining constructor.
* java/nio/ByteBuffer.java: Remove unused constructor.
* java/nio/ByteBufferImpl.java (shiftDown): New optimized method.
* java/nio/ByteBufferImpl.java (get, put): Add array_offset.
* java/nio/DirectByteBufferImpl.java (owner): New field.
(offset): Remove unused field.
(<init>): Modify one and add another constructor. Change callers.
(allocateDirect): Removed - not used.
(getImpl, putImpl): Make static and pass address explicitly,
to make them useful for MappedByteBufferImpl.
(get, put): Check for underflow. Modify for new getImpl.
(getImpl): New native method where target is array.
(get(byte[],int,int)): Use the above.
(adjustAddress): New static native method.
(slice, duplicate, asReadOnly): New implementations.
* java/nio/natDirectByteBufferImpl.cc (getImpl, putImpl, shiftDown,
adjustAddress): New or updated native methods.
2004-02-15 Ito Kazumitsu <kaz@maczuka.gcd.org> 2004-02-15 Ito Kazumitsu <kaz@maczuka.gcd.org>
......
...@@ -44,7 +44,7 @@ package java.nio; ...@@ -44,7 +44,7 @@ package java.nio;
public abstract class ByteBuffer extends Buffer public abstract class ByteBuffer extends Buffer
implements Comparable implements Comparable
{ {
private ByteOrder endian = ByteOrder.BIG_ENDIAN; ByteOrder endian = ByteOrder.BIG_ENDIAN;
int array_offset; int array_offset;
byte[] backing_buffer; byte[] backing_buffer;
...@@ -52,16 +52,8 @@ public abstract class ByteBuffer extends Buffer ...@@ -52,16 +52,8 @@ public abstract class ByteBuffer extends Buffer
ByteBuffer (int capacity, int limit, int position, int mark) ByteBuffer (int capacity, int limit, int position, int mark)
{ {
super (capacity, limit, position, mark); super (capacity, limit, position, mark);
array_offset = 0;
} }
ByteBuffer (byte[] buffer, int offset, int capacity, int limit, int position, int mark)
{
super (capacity, limit, position, mark);
this.backing_buffer = buffer;
this.array_offset = offset;
}
/** /**
* Allocates a new direct byte buffer. * Allocates a new direct byte buffer.
*/ */
...@@ -75,7 +67,7 @@ public abstract class ByteBuffer extends Buffer ...@@ -75,7 +67,7 @@ public abstract class ByteBuffer extends Buffer
*/ */
public static ByteBuffer allocate (int capacity) public static ByteBuffer allocate (int capacity)
{ {
return new ByteBufferImpl (capacity); return wrap(new byte[capacity], 0, capacity);
} }
/** /**
...@@ -87,6 +79,14 @@ public abstract class ByteBuffer extends Buffer ...@@ -87,6 +79,14 @@ public abstract class ByteBuffer extends Buffer
*/ */
final public static ByteBuffer wrap (byte[] array, int offset, int length) final public static ByteBuffer wrap (byte[] array, int offset, int length)
{ {
// FIXME: In GCJ and other implementations where arrays may not
// move we might consider, at least when offset==0:
// return new DirectByteBufferImpl(array,
// address_of_data(array) + offset,
// length, length, 0, false);
// This may be more efficient, mainly because we can then use the
// same logic for all ByteBuffers.
return new ByteBufferImpl (array, 0, array.length, offset + length, offset, -1, false); return new ByteBufferImpl (array, 0, array.length, offset + length, offset, -1, false);
} }
...@@ -116,11 +116,10 @@ public abstract class ByteBuffer extends Buffer ...@@ -116,11 +116,10 @@ public abstract class ByteBuffer extends Buffer
*/ */
public ByteBuffer get (byte[] dst, int offset, int length) public ByteBuffer get (byte[] dst, int offset, int length)
{ {
if ((offset < 0) if (offset < 0 || length < 0 || offset + length > dst.length)
|| (offset > dst.length)
|| (length < 0)
|| (length > (dst.length - offset)))
throw new IndexOutOfBoundsException (); throw new IndexOutOfBoundsException ();
if (length > remaining())
throw new BufferUnderflowException();
for (int i = offset; i < offset + length; i++) for (int i = offset; i < offset + length; i++)
{ {
......
...@@ -45,45 +45,42 @@ final class ByteBufferImpl extends ByteBuffer ...@@ -45,45 +45,42 @@ final class ByteBufferImpl extends ByteBuffer
{ {
private boolean readOnly; private boolean readOnly;
ByteBufferImpl (int capacity)
{
this (new byte [capacity], 0, capacity, capacity, 0, -1, false);
}
ByteBufferImpl (byte[] buffer, int offset, int capacity, int limit, int position, int mark, boolean readOnly) ByteBufferImpl (byte[] buffer, int offset, int capacity, int limit, int position, int mark, boolean readOnly)
{ {
super (buffer, offset, capacity, limit, position, mark); super (capacity, limit, position, mark);
this.backing_buffer = buffer;
this.array_offset = offset;
this.readOnly = readOnly; this.readOnly = readOnly;
} }
public CharBuffer asCharBuffer () public CharBuffer asCharBuffer ()
{ {
return new CharViewBufferImpl (this, position (), remaining(), remaining (), 0, -1, isReadOnly (), order()); return new CharViewBufferImpl (this, remaining() >> 1);
} }
public ShortBuffer asShortBuffer () public ShortBuffer asShortBuffer ()
{ {
return new ShortViewBufferImpl (this, position (), remaining(), remaining (), 0, -1, isReadOnly (), order()); return new ShortViewBufferImpl (this, remaining() >> 1);
} }
public IntBuffer asIntBuffer () public IntBuffer asIntBuffer ()
{ {
return new IntViewBufferImpl (this, position (), remaining(), remaining (), 0, -1, isReadOnly (), order()); return new IntViewBufferImpl (this, remaining() >> 2);
} }
public LongBuffer asLongBuffer () public LongBuffer asLongBuffer ()
{ {
return new LongViewBufferImpl (this, position (), remaining(), remaining (), 0, -1, isReadOnly (), order()); return new LongViewBufferImpl (this, remaining() >> 3);
} }
public FloatBuffer asFloatBuffer () public FloatBuffer asFloatBuffer ()
{ {
return new FloatViewBufferImpl (this, position (), remaining(), remaining (), 0, -1, isReadOnly (), order()); return new FloatViewBufferImpl (this, remaining() >> 2);
} }
public DoubleBuffer asDoubleBuffer () public DoubleBuffer asDoubleBuffer ()
{ {
return new DoubleViewBufferImpl (this, position (), remaining(), remaining (), 0, -1, isReadOnly (), order()); return new DoubleViewBufferImpl (this, remaining() >> 3);
} }
public boolean isReadOnly () public boolean isReadOnly ()
...@@ -106,6 +103,13 @@ final class ByteBufferImpl extends ByteBuffer ...@@ -106,6 +103,13 @@ final class ByteBufferImpl extends ByteBuffer
return new ByteBufferImpl (backing_buffer, array_offset, capacity (), limit (), position (), mark, true); return new ByteBufferImpl (backing_buffer, array_offset, capacity (), limit (), position (), mark, true);
} }
void shiftDown (int dst_offset, int src_offset, int count)
{
System.arraycopy(backing_buffer, array_offset + src_offset,
backing_buffer, array_offset + dst_offset,
count);
}
public ByteBuffer compact () public ByteBuffer compact ()
{ {
int pos = position(); int pos = position();
...@@ -129,7 +133,7 @@ final class ByteBufferImpl extends ByteBuffer ...@@ -129,7 +133,7 @@ final class ByteBufferImpl extends ByteBuffer
*/ */
final public byte get () final public byte get ()
{ {
byte result = backing_buffer [position ()]; byte result = backing_buffer [position () + array_offset];
position (position () + 1); position (position () + 1);
return result; return result;
} }
...@@ -144,9 +148,10 @@ final class ByteBufferImpl extends ByteBuffer ...@@ -144,9 +148,10 @@ final class ByteBufferImpl extends ByteBuffer
{ {
if (readOnly) if (readOnly)
throw new ReadOnlyBufferException (); throw new ReadOnlyBufferException ();
backing_buffer [position ()] = value; int pos = position();
position (position () + 1); backing_buffer [pos + array_offset] = value;
position (pos + 1);
return this; return this;
} }
...@@ -159,7 +164,7 @@ final class ByteBufferImpl extends ByteBuffer ...@@ -159,7 +164,7 @@ final class ByteBufferImpl extends ByteBuffer
*/ */
final public byte get (int index) final public byte get (int index)
{ {
return backing_buffer [index]; return backing_buffer [index + array_offset];
} }
/** /**
...@@ -175,7 +180,7 @@ final class ByteBufferImpl extends ByteBuffer ...@@ -175,7 +180,7 @@ final class ByteBufferImpl extends ByteBuffer
if (readOnly) if (readOnly)
throw new ReadOnlyBufferException (); throw new ReadOnlyBufferException ();
backing_buffer [index] = value; backing_buffer [index + array_offset] = value;
return this; return this;
} }
......
...@@ -52,23 +52,25 @@ class DirectByteBufferImpl extends ByteBuffer ...@@ -52,23 +52,25 @@ class DirectByteBufferImpl extends ByteBuffer
} }
} }
/** Used by MappedByteBufferImpl to prevent premature GC. */
protected Object owner;
RawData address; RawData address;
private int offset;
private boolean readOnly; private boolean readOnly;
public DirectByteBufferImpl (RawData address, long len) public DirectByteBufferImpl (RawData address, long len)
{ {
this (address, 0, (int) len, (int) len, 0, -1, false); this (null, address, (int) len, (int) len, 0, false);
} }
public DirectByteBufferImpl (RawData address, int offset, int capacity, public DirectByteBufferImpl (Object owner, RawData address,
int limit, int position, int mark, int capacity, int limit,
boolean readOnly) int position, boolean readOnly)
{ {
super (capacity, limit, position, mark); super (capacity, limit, position, -1);
this.address = address; this.address = address;
this.offset = offset;
this.readOnly = readOnly; this.readOnly = readOnly;
this.owner = owner;
} }
private static native RawData allocateImpl (int capacity); private static native RawData allocateImpl (int capacity);
...@@ -79,41 +81,58 @@ class DirectByteBufferImpl extends ByteBuffer ...@@ -79,41 +81,58 @@ class DirectByteBufferImpl extends ByteBuffer
freeImpl (address); freeImpl (address);
} }
public static ByteBuffer allocateDirect (int capacity) static native byte getImpl (RawData address, int index);
{ static native void putImpl (RawData address, int index, byte value);
RawData address = allocateImpl (capacity);
if (address == null)
throw new InternalError ("Not enough memory to create direct buffer");
return new DirectByteBufferImpl (address, 0, capacity, capacity, 0, -1, false);
}
private native byte getImpl (int index);
private native void putImpl (int index, byte value);
public byte get () public byte get ()
{ {
byte result = getImpl (position () + offset); int pos = position();
position (position () + 1); if (pos >= limit())
throw new BufferUnderflowException();
byte result = getImpl (address, pos);
position (pos + 1);
return result; return result;
} }
public byte get (int index) public byte get (int index)
{ {
return getImpl (index); if (index >= limit())
throw new BufferUnderflowException();
return getImpl (address, index);
}
static native void getImpl (RawData address, int index,
byte[] dst, int offset, int length);
public ByteBuffer get (byte[] dst, int offset, int length)
{
if (offset < 0 || length < 0 || offset + length > dst.length)
throw new IndexOutOfBoundsException ();
if (length > remaining())
throw new BufferUnderflowException();
int index = position();
getImpl(address, index, dst, offset, length);
position(index+length);
return this;
} }
public ByteBuffer put (byte value) public ByteBuffer put (byte value)
{ {
putImpl (position (), value); int pos = position();
position (position () + 1); if (pos >= limit())
throw new BufferUnderflowException();
putImpl (address, pos, value);
position (pos + 1);
return this; return this;
} }
public ByteBuffer put (int index, byte value) public ByteBuffer put (int index, byte value)
{ {
putImpl (index, value); if (index >= limit())
throw new BufferUnderflowException();
putImpl (address, index, value);
return this; return this;
} }
...@@ -132,21 +151,42 @@ class DirectByteBufferImpl extends ByteBuffer ...@@ -132,21 +151,42 @@ class DirectByteBufferImpl extends ByteBuffer
return this; return this;
} }
public ByteBuffer duplicate () public static native RawData adjustAddress(RawData address, int offset);
public ByteBuffer slice ()
{ {
return new DirectByteBufferImpl ( int rem = remaining();
address, offset, capacity (), limit (), position (), -1, isReadOnly ()); return new DirectByteBufferImpl (owner,
adjustAddress(address, position()),
rem, rem, 0, isReadOnly ());
} }
public ByteBuffer slice () private ByteBuffer duplicate (boolean readOnly)
{
int pos = position();
reset();
int mark = position();
position(pos);
DirectByteBufferImpl result
= new DirectByteBufferImpl (owner, address, capacity (), limit (),
pos, readOnly);
if (mark != pos)
{
result.position(mark);
result.mark();
result.position(pos);
}
return result;
}
public ByteBuffer duplicate ()
{ {
return new DirectByteBufferImpl (address, position () + offset, remaining (), remaining (), 0, -1, isReadOnly ()); return duplicate(isReadOnly());
} }
public ByteBuffer asReadOnlyBuffer () public ByteBuffer asReadOnlyBuffer ()
{ {
return new DirectByteBufferImpl ( return duplicate(true);
address, offset, capacity (), limit (), position (), -1, true);
} }
public boolean isReadOnly () public boolean isReadOnly ()
...@@ -161,34 +201,34 @@ class DirectByteBufferImpl extends ByteBuffer ...@@ -161,34 +201,34 @@ class DirectByteBufferImpl extends ByteBuffer
public CharBuffer asCharBuffer () public CharBuffer asCharBuffer ()
{ {
return new CharViewBufferImpl (this, position (), remaining (), remaining (), 0, -1, isReadOnly (), order()); return new CharViewBufferImpl (this, remaining() >> 1);
}
public DoubleBuffer asDoubleBuffer ()
{
return new DoubleViewBufferImpl (this, position (), remaining (), remaining (), 0, -1, isReadOnly (), order());
} }
public FloatBuffer asFloatBuffer () public ShortBuffer asShortBuffer ()
{ {
return new FloatViewBufferImpl (this, position (), remaining (), remaining (), 0, -1, isReadOnly (), order()); return new ShortViewBufferImpl (this, remaining() >> 1);
} }
public IntBuffer asIntBuffer () public IntBuffer asIntBuffer ()
{ {
return new IntViewBufferImpl (this, position (), remaining (), remaining (), 0, -1, isReadOnly (), order()); return new IntViewBufferImpl (this, remaining() >> 2);
} }
public LongBuffer asLongBuffer () public LongBuffer asLongBuffer ()
{ {
return new LongViewBufferImpl (this, position (), remaining (), remaining (), 0, -1, isReadOnly (), order()); return new LongViewBufferImpl (this, remaining() >> 3);
} }
public ShortBuffer asShortBuffer () public FloatBuffer asFloatBuffer ()
{ {
return new ShortViewBufferImpl (this, position (), remaining (), remaining (), 0, -1, isReadOnly (), order()); return new FloatViewBufferImpl (this, remaining() >> 2);
} }
public DoubleBuffer asDoubleBuffer ()
{
return new DoubleViewBufferImpl (this, remaining() >> 3);
}
final public char getChar () final public char getChar ()
{ {
return ByteBufferHelper.getChar(this, order()); return ByteBufferHelper.getChar(this, order());
......
...@@ -144,32 +144,32 @@ public class MappedByteBufferImpl extends MappedByteBuffer ...@@ -144,32 +144,32 @@ public class MappedByteBufferImpl extends MappedByteBuffer
public CharBuffer asCharBuffer () public CharBuffer asCharBuffer ()
{ {
return new CharViewBufferImpl (this, position (), remaining(), remaining (), 0, -1, isReadOnly (), order()); return new CharViewBufferImpl (this, remaining() >> 1);
} }
public ShortBuffer asShortBuffer () public ShortBuffer asShortBuffer ()
{ {
return new ShortViewBufferImpl (this, position (), remaining(), remaining (), 0, -1, isReadOnly (), order()); return new ShortViewBufferImpl (this, remaining() >> 1);
} }
public IntBuffer asIntBuffer () public IntBuffer asIntBuffer ()
{ {
return new IntViewBufferImpl (this, position (), remaining(), remaining (), 0, -1, isReadOnly (), order()); return new IntViewBufferImpl (this, remaining() >> 2);
} }
public LongBuffer asLongBuffer () public LongBuffer asLongBuffer ()
{ {
return new LongViewBufferImpl (this, position (), remaining(), remaining (), 0, -1, isReadOnly (), order()); return new LongViewBufferImpl (this, remaining() >> 3);
} }
public FloatBuffer asFloatBuffer () public FloatBuffer asFloatBuffer ()
{ {
return new FloatViewBufferImpl (this, position (), remaining(), remaining (), 0, -1, isReadOnly (), order()); return new FloatViewBufferImpl (this, remaining() >> 2);
} }
public DoubleBuffer asDoubleBuffer () public DoubleBuffer asDoubleBuffer ()
{ {
return new DoubleViewBufferImpl (this, position (), remaining(), remaining (), 0, -1, isReadOnly (), order()); return new DoubleViewBufferImpl (this, remaining() >> 3);
} }
final public char getChar () final public char getChar ()
......
// natDirectByteBufferImpl.cc // natDirectByteBufferImpl.cc
/* Copyright (C) 2003 Free Software Foundation /* Copyright (C) 2003, 2004 Free Software Foundation
This file is part of libgcj. This file is part of libgcj.
...@@ -18,7 +18,10 @@ details. */ ...@@ -18,7 +18,10 @@ details. */
#include <gnu/gcj/RawData.h> #include <gnu/gcj/RawData.h>
#include <java/nio/DirectByteBufferImpl.h> #include <java/nio/DirectByteBufferImpl.h>
gnu::gcj::RawData* using gnu::gcj::RawData;
using java::nio::DirectByteBufferImpl;
RawData*
java::nio::DirectByteBufferImpl::allocateImpl (jint capacity) java::nio::DirectByteBufferImpl::allocateImpl (jint capacity)
{ {
return reinterpret_cast<gnu::gcj::RawData*> (::malloc (capacity)); return reinterpret_cast<gnu::gcj::RawData*> (::malloc (capacity));
...@@ -31,24 +34,40 @@ java::nio::DirectByteBufferImpl::freeImpl (gnu::gcj::RawData* address) ...@@ -31,24 +34,40 @@ java::nio::DirectByteBufferImpl::freeImpl (gnu::gcj::RawData* address)
} }
jbyte jbyte
java::nio::DirectByteBufferImpl::getImpl (jint index) DirectByteBufferImpl::getImpl (RawData* address, jint index)
{ {
jbyte* pointer = reinterpret_cast<jbyte*> (address) + offset + index; jbyte* pointer = reinterpret_cast<jbyte*> (address) + index;
return *pointer; return *pointer;
} }
void void
java::nio::DirectByteBufferImpl::putImpl (jint index, jbyte value) DirectByteBufferImpl::getImpl (RawData* address, jint index,
jbyteArray dst, jint offset, jint length)
{
jbyte* src = reinterpret_cast<jbyte*> (address) + index;
memcpy (elements (dst) + offset, src, length);
}
void
java::nio::DirectByteBufferImpl::putImpl (gnu::gcj::RawData* address,
jint index, jbyte value)
{ {
jbyte* pointer = reinterpret_cast<jbyte*> (address) + offset + index; jbyte* pointer = reinterpret_cast<jbyte*> (address) + index;
*pointer = value; *pointer = value;
} }
RawData*
java::nio::DirectByteBufferImpl::adjustAddress (RawData* address, jint offset)
{
jbyte* start = reinterpret_cast<jbyte*> (address) + offset;
return reinterpret_cast<RawData*>(start);
}
void void
java::nio::DirectByteBufferImpl::shiftDown java::nio::DirectByteBufferImpl::shiftDown
(jint dst_offset, jint src_offset, jint count) (jint dst_offset, jint src_offset, jint count)
{ {
jbyte* dst = reinterpret_cast<jbyte*> (address) + offset + dst_offset; jbyte* dst = reinterpret_cast<jbyte*> (address) + array_offset + dst_offset;
jbyte* src = reinterpret_cast<jbyte*> (address) + offset + src_offset; jbyte* src = reinterpret_cast<jbyte*> (address) + array_offset + src_offset;
::memmove(dst, src, count); ::memmove(dst, src, count);
} }
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment