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

eu.stratosphere.nephele.taskmanager.runtime.RuntimeOutputChannelBroker Maven / Gradle / Ivy

/***********************************************************************************************************************
 * Copyright (C) 2010-2013 by the Stratosphere project (http://stratosphere.eu)
 *
 * Licensed 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 eu.stratosphere.nephele.taskmanager.runtime;

import java.io.IOException;

import eu.stratosphere.nephele.event.task.AbstractEvent;
import eu.stratosphere.nephele.event.task.AbstractTaskEvent;
import eu.stratosphere.nephele.io.channels.Buffer;
import eu.stratosphere.nephele.io.channels.bytebuffered.AbstractByteBufferedOutputChannel;
import eu.stratosphere.nephele.io.channels.bytebuffered.ByteBufferedChannelCloseEvent;
import eu.stratosphere.nephele.io.channels.bytebuffered.ByteBufferedOutputChannelBroker;
import eu.stratosphere.nephele.taskmanager.bytebuffered.AbstractOutputChannelForwarder;
import eu.stratosphere.nephele.taskmanager.bytebuffered.OutputChannelForwardingChain;
import eu.stratosphere.nephele.taskmanager.bytebuffered.ReceiverNotFoundEvent;
import eu.stratosphere.nephele.taskmanager.transferenvelope.TransferEnvelope;

final class RuntimeOutputChannelBroker extends AbstractOutputChannelForwarder implements
		ByteBufferedOutputChannelBroker {

	/**
	 * The byte buffered output channel this context belongs to.
	 */
	private final AbstractByteBufferedOutputChannel byteBufferedOutputChannel;

	/**
	 * The buffer provider this channel broker to obtain buffers from.
	 */
	private final RuntimeOutputGateContext outputGateContext;

	/**
	 * The forwarding chain along which the created transfer envelopes will be pushed.
	 */
	private OutputChannelForwardingChain forwardingChain;

	/**
	 * Points to the {@link TransferEnvelope} object that will be passed to the framework upon
	 * the next releaseWriteBuffers call.
	 */
	private TransferEnvelope outgoingTransferEnvelope = null;

	/**
	 * Stores whether the receiver has acknowledged the close request from this channel.
	 */
	private boolean closeAcknowledgmentReceived = false;

	/**
	 * Stores the last sequence number of the transfer envelope for which the receiver could not be found.
	 */
	private int lastSequenceNumberWithReceiverNotFound = -1;

	/**
	 * The sequence number for the next {@link TransferEnvelope} to be created.
	 */
	private int sequenceNumber = 0;

	RuntimeOutputChannelBroker(final RuntimeOutputGateContext outputGateContext,
			final AbstractByteBufferedOutputChannel byteBufferedOutputChannel,
			final AbstractOutputChannelForwarder next) {

		super(next);

		if (next == null) {
			throw new IllegalArgumentException("Argument next must not be null");
		}

		this.outputGateContext = outputGateContext;
		this.byteBufferedOutputChannel = byteBufferedOutputChannel;
		this.byteBufferedOutputChannel.setByteBufferedOutputChannelBroker(this);
	}

	public void setForwardingChain(final OutputChannelForwardingChain forwardingChain) {
		this.forwardingChain = forwardingChain;
	}


	@Override
	public boolean hasDataLeft() throws IOException, InterruptedException {

		if (this.closeAcknowledgmentReceived) {
			return getNext().hasDataLeft();
		}

		if ((this.lastSequenceNumberWithReceiverNotFound + 1) == this.sequenceNumber) {
			return getNext().hasDataLeft();
		}

		return true;
	}


	@Override
	public void processEvent(final AbstractEvent event) {

		if (event instanceof ByteBufferedChannelCloseEvent) {
			this.closeAcknowledgmentReceived = true;
		} else if (event instanceof ReceiverNotFoundEvent) {
			this.lastSequenceNumberWithReceiverNotFound = ((ReceiverNotFoundEvent) event).getSequenceNumber();
		} else if (event instanceof AbstractTaskEvent) {
			throw new IllegalStateException("Received synchronous task event " + event);
		}

		getNext().processEvent(event);
	}


	@Override
	public Buffer requestEmptyWriteBuffer() throws InterruptedException, IOException {

		if (this.outgoingTransferEnvelope == null) {
			this.outgoingTransferEnvelope = createNewOutgoingTransferEnvelope();
		}

		final int uncompressedBufferSize = calculateBufferSize();

		return this.outputGateContext.requestEmptyBufferBlocking(uncompressedBufferSize);
	}

	/**
	 * Creates a new {@link TransferEnvelope} object. The method assigns
	 * and increases the sequence number. Moreover, it will look up the list of receivers for this transfer envelope.
	 * This method will block until the lookup is completed.
	 * 
	 * @return a new {@link TransferEnvelope} object containing the correct sequence number and receiver list
	 */
	private TransferEnvelope createNewOutgoingTransferEnvelope() {

		final TransferEnvelope transferEnvelope = new TransferEnvelope(this.sequenceNumber++,
			this.byteBufferedOutputChannel.getJobID(),
			this.byteBufferedOutputChannel.getID());

		return transferEnvelope;
	}

	/**
	 * Calculates the recommended size of the next buffer to be
	 * handed to the attached channel object in bytes.
	 * 
	 * @return the recommended size of the next buffer in bytes
	 */
	private int calculateBufferSize() {

		// TODO: Include latency considerations
		return this.outputGateContext.getMaximumBufferSize();
	}


	@Override
	public void releaseWriteBuffer(final Buffer buffer) throws IOException, InterruptedException {

		// Check for events
		this.forwardingChain.processQueuedEvents();

		if (this.outgoingTransferEnvelope == null) {
			throw new IllegalStateException("Cannot find transfer envelope for channel with ID "
				+ this.byteBufferedOutputChannel.getID());
		}

		// Consistency check
		if (this.outgoingTransferEnvelope.getBuffer() != null) {
			throw new IllegalStateException("Channel " + this.byteBufferedOutputChannel.getID()
				+ " has already a buffer attached");
		}
		buffer.flip();
		this.outgoingTransferEnvelope.setBuffer(buffer);

		this.forwardingChain.pushEnvelope(this.outgoingTransferEnvelope);
		this.outgoingTransferEnvelope = null;
	}


	@Override
	public boolean hasDataLeftToTransmit() throws IOException, InterruptedException {

		// Check for events
		this.forwardingChain.processQueuedEvents();

		return this.forwardingChain.anyForwarderHasDataLeft();
	}


	@Override
	public void transferEventToInputChannel(final AbstractEvent event) throws IOException, InterruptedException {

		if (this.outgoingTransferEnvelope != null) {
			this.outgoingTransferEnvelope.addEvent(event);
		} else {

			final TransferEnvelope ephemeralTransferEnvelope = createNewOutgoingTransferEnvelope();
			ephemeralTransferEnvelope.addEvent(event);

			this.forwardingChain.pushEnvelope(ephemeralTransferEnvelope);
		}
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy