/* | 
|
 * Copyright (c) 2001, 2002, Oracle and/or its affiliates. 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.  Oracle designates this | 
|
 * particular file as subject to the "Classpath" exception as provided | 
|
 * by Oracle in the LICENSE file that accompanied this code. | 
|
 * | 
|
 * 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 Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA | 
|
 * or visit www.oracle.com if you need additional information or have any | 
|
 * questions. | 
|
*/  | 
|
package sun.nio.ch;  | 
|
import java.io.*;  | 
|
import java.nio.*;  | 
|
import java.nio.channels.*;  | 
|
import java.nio.channels.spi.*;  | 
|
/**  | 
|
* This class is defined here rather than in java.nio.channels.Channels  | 
|
* so that code can be shared with SocketAdaptor.  | 
|
*  | 
|
* @author Mike McCloskey  | 
|
* @author Mark Reinhold  | 
|
* @since 1.4  | 
|
*/  | 
|
public class ChannelInputStream  | 
|
extends InputStream  | 
|
{ | 
|
public static int read(ReadableByteChannel ch, ByteBuffer bb,  | 
|
boolean block)  | 
|
throws IOException  | 
|
    { | 
|
if (ch instanceof SelectableChannel) {  | 
|
SelectableChannel sc = (SelectableChannel)ch;  | 
|
synchronized (sc.blockingLock()) {  | 
|
boolean bm = sc.isBlocking();  | 
|
if (!bm)  | 
|
throw new IllegalBlockingModeException();  | 
|
if (bm != block)  | 
|
sc.configureBlocking(block);  | 
|
int n = ch.read(bb);  | 
|
if (bm != block)  | 
|
sc.configureBlocking(bm);  | 
|
return n;  | 
|
}  | 
|
        } else { | 
|
return ch.read(bb);  | 
|
}  | 
|
}  | 
|
protected final ReadableByteChannel ch;  | 
|
private ByteBuffer bb = null;  | 
|
    private byte[] bs = null;           // Invoker's previous array | 
|
private byte[] b1 = null;  | 
|
public ChannelInputStream(ReadableByteChannel ch) {  | 
|
this.ch = ch;  | 
|
}  | 
|
public synchronized int read() throws IOException {  | 
|
if (b1 == null)  | 
|
b1 = new byte[1];  | 
|
int n = this.read(b1);  | 
|
if (n == 1)  | 
|
return b1[0] & 0xff;  | 
|
return -1;  | 
|
}  | 
|
public synchronized int read(byte[] bs, int off, int len)  | 
|
throws IOException  | 
|
    { | 
|
if ((off < 0) || (off > bs.length) || (len < 0) ||  | 
|
((off + len) > bs.length) || ((off + len) < 0)) {  | 
|
throw new IndexOutOfBoundsException();  | 
|
} else if (len == 0)  | 
|
return 0;  | 
|
ByteBuffer bb = ((this.bs == bs)  | 
|
? this.bb  | 
|
: ByteBuffer.wrap(bs));  | 
|
bb.limit(Math.min(off + len, bb.capacity()));  | 
|
bb.position(off);  | 
|
this.bb = bb;  | 
|
this.bs = bs;  | 
|
return read(bb);  | 
|
}  | 
|
protected int read(ByteBuffer bb)  | 
|
throws IOException  | 
|
    { | 
|
return ChannelInputStream.read(ch, bb, true);  | 
|
}  | 
|
public int available() throws IOException {  | 
|
        // special case where the channel is to a file | 
|
if (ch instanceof SeekableByteChannel) {  | 
|
SeekableByteChannel sbc = (SeekableByteChannel)ch;  | 
|
long rem = Math.max(0, sbc.size() - sbc.position());  | 
|
return (rem > Integer.MAX_VALUE) ? Integer.MAX_VALUE : (int)rem;  | 
|
}  | 
|
return 0;  | 
|
}  | 
|
public void close() throws IOException {  | 
|
ch.close();  | 
|
}  | 
|
}  |