All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.activemq.transport.nio.NIOBufferedInputStream Maven / Gradle / Ivy

There is a newer version: 6.1.2
Show newest version
/**
 *
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You under the Apache License, Version 2.0
 * (the "License"); you may not use this file except in compliance with
 * the License.  You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.apache.activemq.transport.nio;

import java.io.IOException;
import java.io.InputStream;
import java.nio.ByteBuffer;
import java.nio.channels.Channel;
import java.nio.channels.ClosedChannelException;
import java.nio.channels.ReadableByteChannel;
import java.nio.channels.SelectionKey;
import java.nio.channels.Selector;
import java.nio.channels.SocketChannel;

/**
 * Implementation of InputStream using Java NIO channel,direct buffer and
 * Selector
 */
public class NIOBufferedInputStream extends InputStream {

    private final static int BUFFER_SIZE = 8192;

    private SocketChannel sc = null;

    private ByteBuffer bb = null;

    private Selector rs = null;

    public NIOBufferedInputStream(ReadableByteChannel channel, int size)
            throws ClosedChannelException, IOException {

        if (size <= 0) {
            throw new IllegalArgumentException("Buffer size <= 0");
        }

        this.bb = ByteBuffer.allocateDirect(size);
        this.sc = (SocketChannel) channel;

        this.sc.configureBlocking(false);

        this.rs = Selector.open();

        sc.register(rs, SelectionKey.OP_READ);

        bb.position(0);
        bb.limit(0);
    }

    public NIOBufferedInputStream(ReadableByteChannel channel)
            throws ClosedChannelException, IOException {
        this(channel, BUFFER_SIZE);
    }

    public int available() throws IOException {
        if (!rs.isOpen())
            throw new IOException("Input Stream Closed");

        return bb.remaining();
    }

    public void close() throws IOException {
        if (rs.isOpen()) {
            rs.close();

            if (sc.isOpen()) {
                sc.socket().shutdownInput();
                sc.socket().close();
            }

            bb = null;
            sc = null;
        }
    }

    public int read() throws IOException {
        if (!rs.isOpen())
            throw new IOException("Input Stream Closed");

        if (!bb.hasRemaining()) {
            try {
                fill(1);
            } catch (ClosedChannelException e) {
                close();
                return -1;
            }
        }

        return (bb.get() & 0xFF);
    }

    public int read(byte[] b, int off, int len) throws IOException {
        int bytesCopied = -1;

        if (!rs.isOpen())
            throw new IOException("Input Stream Closed");

        while (bytesCopied == -1) {
            if (bb.hasRemaining()) {
                bytesCopied = (len < bb.remaining() ? len : bb.remaining());
                bb.get(b, off, bytesCopied);
            } else {
                try {
                    fill(1);
                } catch (ClosedChannelException e) {
                    close();
                    return -1;
                }
            }
        }

        return bytesCopied;
    }

    public long skip(long n) throws IOException {
        long skiped = 0;

        if (!rs.isOpen())
            throw new IOException("Input Stream Closed");

        while (n > 0) {
            if (n <= bb.remaining()) {
                skiped += n;
                bb.position(bb.position() + (int) n);
                n = 0;
            } else {
                skiped += bb.remaining();
                n -= bb.remaining();

                bb.position(bb.limit());

                try {
                    fill((int) n);
                } catch (ClosedChannelException e) {
                    close();
                    return skiped;
                }
            }
        }

        return skiped;
    }

    private void fill(int n) throws IOException, ClosedChannelException {
        int bytesRead = -1;

        if ((n <= 0) || (n <= bb.remaining()))
            return;

        bb.compact();

        n = (bb.remaining() < n ? bb.remaining() : n);

        for (;;) {
            bytesRead = sc.read(bb);

            if (bytesRead == -1)
                throw new ClosedChannelException();

            n -= bytesRead;

            if (n <= 0)
                break;

            rs.select(0);
            rs.selectedKeys().clear();
        }

        bb.flip();
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy