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

org.glassfish.jersey.internal.util.collection.ByteBufferInputStream Maven / Gradle / Ivy

There is a newer version: 4.0.0-M1
Show newest version
/*
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
 *
 * Copyright (c) 2013-2017 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
 * https://oss.oracle.com/licenses/CDDL+GPL-1.1
 * or 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 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.
 * 

* 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. *

* * @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 buffers; /** * Content represents the buffer queue status. * * null = open, POISON_PILL = closed, Throwable = closed with error */ private final AtomicReference queueStatus = new AtomicReference(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. * *

* 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. *

* * @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. *

* 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. *

*/ 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. *

* 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. *

* * @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; } }