/* * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER. * * Copyright (c) 2013-2015 Oracle and/or its affiliates. All rights reserved. * * The contents of this file are subject to the terms of either the GNU * General Public License Version 2 only ("GPL") or the Common Development * and Distribution License("CDDL") (collectively, the "License"). You * may not use this file except in compliance with the License. You can * obtain a copy of the License at * http://glassfish.java.net/public/CDDL+GPL_1_1.html * or packager/legal/LICENSE.txt. See the License for the specific * language governing permissions and limitations under the License. * * When distributing the software, include this License Header Notice in each * file and include the License file at packager/legal/LICENSE.txt. * * GPL Classpath Exception: * Oracle designates this particular file as subject to the "Classpath" * exception as provided by Oracle in the GPL Version 2 section of the License * file that accompanied this code. * * Modifications: * If applicable, add the following below the License Header, with the fields * enclosed by brackets [] replaced by your own identifying information: * "Portions Copyright [year] [name of copyright owner]" * * Contributor(s): * If you wish your version of this file to be governed by only the CDDL or * only the GPL Version 2, indicate your decision by adding "[Contributor] * elects to include this software in this distribution under the [CDDL or GPL * Version 2] license." If you don't indicate a single choice of license, a * recipient has the option to distribute your version of this file under * either the CDDL, the GPL Version 2 or to extend the choice of license to * its licensees as provided above. However, if you add GPL Version 2 code * and therefore, elected the GPL Version 2 license, then the option applies * only if the new code is made subject to such option by the copyright * holder. */ package org.glassfish.jersey.internal.util.collection; import java.io.IOException; import java.nio.ByteBuffer; import java.util.concurrent.BlockingQueue; import java.util.concurrent.atomic.AtomicBoolean; import java.util.concurrent.atomic.AtomicReference; import org.glassfish.jersey.internal.LocalizationMessages; /** * An {@link java.io.InputStream} backed by a queue of {@link java.nio.ByteBuffer byte buffers} * to be read. * <p> * This input stream serves as a bridging inbound I/O component between a blocking upper I/O layer * and an underlying non-blocking I/O layer. In addition to the blocking {@code InputStream.read} operations, * this input stream provides the non-blocking {@code tryRead} counterparts. * </p> * * @author Marek Potociar (marek.potociar at oracle.com) */ public final class ByteBufferInputStream extends NonBlockingInputStream { /** * Constant buffer indicating EOF. */ private static final ByteBuffer EOF = ByteBuffer.wrap(new byte[] {}); /** * Read-side EOF flag. Does not have to be volatile, it is transient and only accessed from the reader thread. */ private boolean eof = false; /** * Currently read byte buffer. */ private ByteBuffer current; /** * Queue of byte buffers to be read. */ private final BlockingQueue<ByteBuffer> buffers; /** * Content represents the buffer queue status. * * null = open, POISON_PILL = closed, Throwable = closed with error */ private final AtomicReference<Object> queueStatus = new AtomicReference<Object>(null); /** * Closed flag. */ private final AtomicBoolean closed = new AtomicBoolean(false); /** * Create a new input stream that is backed by a a queue of {@link java.nio.ByteBuffer byte buffers} * to be read. */ public ByteBufferInputStream() { this.buffers = DataStructures.createLinkedTransferQueue(); this.current = null; } private boolean fetchChunk(final boolean block) throws InterruptedException { if (eof) { return false; } // Read until no buffers available (poll returned null) // or until a non-empty buffer or EOF is reached. do { if (closed.get()) { current = EOF; break; } current = (block) ? buffers.take() : buffers.poll(); } while (current != null && current != EOF && !current.hasRemaining()); eof = current == EOF; return !eof; } private void checkNotClosed() throws IOException { if (closed.get()) { throw new IOException(LocalizationMessages.INPUT_STREAM_CLOSED()); } } private void checkThrowable() throws IOException { final Object o = queueStatus.get(); if (o != null && o != EOF) { // should be faster than instanceof // if not null or EOF, then it must be Throwable if (queueStatus.compareAndSet(o, EOF)) { // clear throwable flag and throw exception try { throw new IOException((Throwable) o); } finally { close(); } } } } @Override public int available() throws IOException { if (eof || closed.get()) { checkThrowable(); return 0; } int available = 0; if (current != null && current.hasRemaining()) { available = current.remaining(); } for (final ByteBuffer buffer : buffers) { if (buffer == EOF) { break; } available += buffer.remaining(); } checkThrowable(); return closed.get() ? 0 : available; } @Override public int read() throws IOException { return tryRead(true); } @Override public int read(byte[] b, int off, int len) throws IOException { return tryRead(b, off, len, true); } public int tryRead() throws IOException { return tryRead(false); } @Override public int tryRead(final byte[] b) throws IOException { return tryRead(b, 0, b.length); } @Override public int tryRead(final byte[] b, final int off, final int len) throws IOException { return tryRead(b, off, len, false); } @Override public void close() throws IOException { if (closed.compareAndSet(false, true)) { closeQueue(); // we can now safely clear the queue - any blocking read waiting for a buffer // has been resumed by the EOF buffer buffers.clear(); } checkThrowable(); } /** * Put the {@code ByteBuffer} to the internal queue to be available for reading from the stream. * * <p> * If the sink is open, the method {@link BlockingQueue#put(Object) puts the buffer} into an internal * byte buffer read queue , waiting if necessary for space to become available. Then the method returns * {@code true} to indicate the buffer has been successfully queued. In case the internal read queue has been * {@link #closeQueue() closed} already, the method simply returns {@code false} without registering * the buffer in the closed queue. * </p> * * @param src the source buffer to be registered in the byte buffer read queue. * @return {@code true} if the byte buffer has been successfully put in the read queue, * {@code false} if the read queue has been closed. * @throws InterruptedException in case the put operation has been interrupted. */ public boolean put(final ByteBuffer src) throws InterruptedException { if (queueStatus.get() == null) { buffers.put(src); return true; } return false; } /** * Closes the byte buffer sink of this input stream to indicate that writing to the stream * has been finished. * <p> * If the sink has already been closed then this method returns immediately. Otherwise the * sink is marked as closed and no more data can be written to it. * </p> */ public void closeQueue() { if (queueStatus.compareAndSet(null, EOF)) { try { buffers.put(EOF); } catch (final InterruptedException e) { Thread.currentThread().interrupt(); } } } /** * Closes the byte buffer sink of this input stream to indicate that writing to the stream * has been finished due to a throwable. * * The throwable set by this method will be propagated to the reader thread when a new attempt * to read bytes is made. * <p> * If the sink has already been closed then this method only sets the throwable in the stream and * then returns immediately. Otherwise the sink is also marked as closed and no more data can be * written to it. * </p> * * @param throwable throwable that is set in the stream. It will be thrown by the stream in case * an attempt to read more data or check available bytes is made. */ public void closeQueue(final Throwable throwable) { if (queueStatus.compareAndSet(null, throwable)) { try { buffers.put(EOF); } catch (final InterruptedException e) { Thread.currentThread().interrupt(); } } } private int tryRead(final byte[] b, final int off, final int len, boolean block) throws IOException { checkThrowable(); checkNotClosed(); if (b == null) { throw new NullPointerException(); } else if (off < 0 || len < 0 || len > b.length - off) { throw new IndexOutOfBoundsException(); } else if (len == 0) { return 0; } if (eof) { return -1; } int i = 0; while (i < len) { if (current != null && current.hasRemaining()) { final int available = current.remaining(); if (available < len - i) { current.get(b, off + i, available); i += available; } else { current.get(b, off + i, len - i); return len; } } else { try { if (!fetchChunk(block) || current == null) { break; // eof or no data } } catch (final InterruptedException e) { Thread.currentThread().interrupt(); if (block) { throw new IOException(e); } } } } return i == 0 && eof ? -1 : i; } private int tryRead(boolean block) throws IOException { checkThrowable(); checkNotClosed(); if (eof) { return -1; } if (current != null && current.hasRemaining()) { return current.get() & 0xFF; } try { // try to fetch, but don't block && check if something has been fetched if (fetchChunk(block) && current != null) { return current.get() & 0xFF; } else if (block) { return -1; } } catch (final InterruptedException e) { Thread.currentThread().interrupt(); if (block) { throw new IOException(e); } } return (eof) ? -1 : NOTHING; } }