6448457: (ch) Channels.newOutputStream().write() does not write all data
Reviewed-by: iris, sherman
This commit is contained in:
parent
5d9d5e3e88
commit
d5b852b81d
@ -66,7 +66,27 @@ public final class Channels {
|
|||||||
private Channels() { } // No instantiation
|
private Channels() { } // No instantiation
|
||||||
|
|
||||||
|
|
||||||
private static int write(WritableByteChannel ch, ByteBuffer bb)
|
/**
|
||||||
|
* Write all remaining bytes in buffer to the given channel.
|
||||||
|
* If the channel is selectable then it must be configured blocking.
|
||||||
|
*/
|
||||||
|
private static void writeFullyImpl(WritableByteChannel ch, ByteBuffer bb)
|
||||||
|
throws IOException
|
||||||
|
{
|
||||||
|
while (bb.remaining() > 0) {
|
||||||
|
int n = ch.write(bb);
|
||||||
|
if (n <= 0)
|
||||||
|
throw new RuntimeException("no bytes written");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Write all remaining bytes in buffer to the given channel.
|
||||||
|
*
|
||||||
|
* @throws IllegalBlockingException
|
||||||
|
* If the channel is selectable and configured non-blocking.
|
||||||
|
*/
|
||||||
|
private static void writeFully(WritableByteChannel ch, ByteBuffer bb)
|
||||||
throws IOException
|
throws IOException
|
||||||
{
|
{
|
||||||
if (ch instanceof SelectableChannel) {
|
if (ch instanceof SelectableChannel) {
|
||||||
@ -74,14 +94,13 @@ public final class Channels {
|
|||||||
synchronized (sc.blockingLock()) {
|
synchronized (sc.blockingLock()) {
|
||||||
if (!sc.isBlocking())
|
if (!sc.isBlocking())
|
||||||
throw new IllegalBlockingModeException();
|
throw new IllegalBlockingModeException();
|
||||||
return ch.write(bb);
|
writeFullyImpl(ch, bb);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
return ch.write(bb);
|
writeFullyImpl(ch, bb);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// -- Byte streams from channels --
|
// -- Byte streams from channels --
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -148,7 +167,7 @@ public final class Channels {
|
|||||||
bb.position(off);
|
bb.position(off);
|
||||||
this.bb = bb;
|
this.bb = bb;
|
||||||
this.bs = bs;
|
this.bs = bs;
|
||||||
Channels.write(ch, bb);
|
Channels.writeFully(ch, bb);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void close() throws IOException {
|
public void close() throws IOException {
|
||||||
|
83
jdk/test/java/nio/channels/Channels/ShortWrite.java
Normal file
83
jdk/test/java/nio/channels/Channels/ShortWrite.java
Normal file
@ -0,0 +1,83 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2008 Sun Microsystems, Inc. All Rights Reserved.
|
||||||
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
|
*
|
||||||
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU General Public License version 2 only, as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
||||||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||||
|
* version 2 for more details (a copy is included in the LICENSE file that
|
||||||
|
* accompanied this code).
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License version
|
||||||
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
||||||
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
*
|
||||||
|
* Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
|
||||||
|
* CA 95054 USA or visit www.sun.com if you need additional information or
|
||||||
|
* have any questions.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* @test
|
||||||
|
* @bug 6448457
|
||||||
|
* @summary Test Channels.newOutputStream returns OutputStream that handles
|
||||||
|
* short writes from the underlying channel
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.io.OutputStream;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.nio.ByteBuffer;
|
||||||
|
import java.nio.channels.*;
|
||||||
|
import java.util.Random;
|
||||||
|
|
||||||
|
public class ShortWrite {
|
||||||
|
|
||||||
|
static Random rand = new Random();
|
||||||
|
static int bytesWritten = 0;
|
||||||
|
|
||||||
|
public static void main(String[] args) throws IOException {
|
||||||
|
|
||||||
|
WritableByteChannel wbc = new WritableByteChannel() {
|
||||||
|
public int write(ByteBuffer src) {
|
||||||
|
int rem = src.remaining();
|
||||||
|
if (rem > 0) {
|
||||||
|
// short write
|
||||||
|
int n = rand.nextInt(rem) + 1;
|
||||||
|
src.position(src.position() + n);
|
||||||
|
bytesWritten += n;
|
||||||
|
return n;
|
||||||
|
} else {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
public void close() throws IOException {
|
||||||
|
throw new RuntimeException("not implemented");
|
||||||
|
}
|
||||||
|
public boolean isOpen() {
|
||||||
|
throw new RuntimeException("not implemented");
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// wrap Channel with OutputStream
|
||||||
|
OutputStream out = Channels.newOutputStream(wbc);
|
||||||
|
|
||||||
|
|
||||||
|
// write 100, 99, 98, ... 1
|
||||||
|
// and check that the expected number of bytes is written
|
||||||
|
int expected = 0;
|
||||||
|
byte[] buf = new byte[100];
|
||||||
|
for (int i=0; i<buf.length; i++) {
|
||||||
|
int len = buf.length-i;
|
||||||
|
out.write(buf, i, len);
|
||||||
|
expected += len;
|
||||||
|
}
|
||||||
|
System.out.format("Bytes written: %d, expected: %d\n", bytesWritten,
|
||||||
|
expected);
|
||||||
|
if (bytesWritten != expected)
|
||||||
|
throw new RuntimeException("incorrect number of bytes written");
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user