/***********************************************************************************************************************
*
* Copyright (C) 2010 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.pact.runtime.resettable;
import java.io.EOFException;
import java.io.IOException;
import java.util.ArrayList;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import eu.stratosphere.nephele.services.memorymanager.ListMemorySegmentSource;
import eu.stratosphere.nephele.services.memorymanager.MemoryAllocationException;
import eu.stratosphere.nephele.services.memorymanager.MemoryManager;
import eu.stratosphere.nephele.services.memorymanager.MemorySegment;
import eu.stratosphere.nephele.template.AbstractInvokable;
import eu.stratosphere.pact.common.generic.types.TypeSerializer;
import eu.stratosphere.pact.runtime.io.RandomAccessInputView;
import eu.stratosphere.pact.runtime.io.SimpleCollectingOutputView;
import eu.stratosphere.pact.runtime.util.MathUtils;
import eu.stratosphere.pact.runtime.util.MemoryBlockIterator;
/**
* Base class for iterators that fetch a block of data into main memory and offer resettable
* access to the data in that block.
*
* @author Stephan Ewen
* @author Fabian Hueske
*/
abstract class AbstractBlockResettableIterator<T> implements MemoryBlockIterator
{
public static final Log LOG = LogFactory.getLog(AbstractBlockResettableIterator.class);
public static final int MIN_BUFFER_SIZE = 8 * 1024;
// ------------------------------------------------------------------------
protected final RandomAccessInputView readView;
protected final SimpleCollectingOutputView collectingView;
protected final TypeSerializer<T> serializer;
protected int numRecordsInBuffer;
protected int numRecordsReturned;
protected final ArrayList<MemorySegment> emptySegments;
protected final ArrayList<MemorySegment> fullSegments;
private final MemoryManager memoryManager;
protected volatile boolean closed; // volatile since it may be asynchronously set to abort after current block
// ------------------------------------------------------------------------
protected AbstractBlockResettableIterator(TypeSerializer<T> serializer, MemoryManager memoryManager,
long availableMemory, AbstractInvokable ownerTask)
throws MemoryAllocationException
{
if (availableMemory < memoryManager.getPageSize()) {
throw new IllegalArgumentException("Block Resettable iterator requires at leat one page of memory");
}
this.memoryManager = memoryManager;
this.serializer = serializer;
availableMemory = memoryManager.roundDownToPageSizeMultiple(availableMemory);
final int numPages = MathUtils.checkedDownCast(availableMemory / memoryManager.getPageSize());
if (numPages < 1) {
throw new IllegalArgumentException("The given amount of memory is smaller than one memory page.");
}
this.emptySegments = new ArrayList<MemorySegment>(numPages);
this.fullSegments = new ArrayList<MemorySegment>(numPages);
memoryManager.allocatePages(ownerTask, emptySegments, numPages);
this.collectingView = new SimpleCollectingOutputView(this.fullSegments,
new ListMemorySegmentSource(this.emptySegments), memoryManager.getPageSize());
this.readView = new RandomAccessInputView(this.fullSegments, memoryManager.getPageSize());
if (LOG.isDebugEnabled())
LOG.debug("Iterator initalized using " + numPages + " memory buffers (" + availableMemory + " bytes of memory.");
}
// --------------------------------------------------------------------------------------------
public void open()
{
if (LOG.isDebugEnabled())
LOG.debug("Block Resettable Iterator opened.");
}
/**
*
*/
public void reset()
{
if (this.closed) {
throw new IllegalStateException("Iterator was closed.");
}
this.readView.setReadPosition(0);
this.numRecordsReturned = 0;
}
/* (non-Javadoc)
* @see eu.stratosphere.pact.runtime.task.util.MemoryBlockIterator#nextBlock()
*/
@Override
public boolean nextBlock() throws IOException
{
this.numRecordsInBuffer = 0;
// add the full segments to the empty ones
for (int i = this.fullSegments.size() - 1; i >= 0; i--) {
this.emptySegments.add(this.fullSegments.remove(i));
}
// reset the views
this.collectingView.reset();
this.readView.setReadPosition(0);
return true;
}
/**
* This method closes the iterator and releases all resources. This method works both as a regular
* shutdown and as a canceling method. The method may be called multiple times and will not produce
* an error.
*/
public void close()
{
synchronized (this) {
if (this.closed) {
return;
}
this.closed = true;
}
this.numRecordsInBuffer = 0;
this.numRecordsReturned = 0;
// add the full segments to the empty ones
for (int i = this.fullSegments.size() - 1; i >= 0; i--) {
this.emptySegments.add(this.fullSegments.remove(i));
}
// release the memory segment
this.memoryManager.release(this.emptySegments);
this.emptySegments.clear();
if (LOG.isDebugEnabled())
LOG.debug("Block Resettable Iterator closed.");
}
// --------------------------------------------------------------------------------------------
protected boolean writeNextRecord(T record) throws IOException
{
try {
this.serializer.serialize(record, this.collectingView);
this.numRecordsInBuffer++;
return true;
} catch (EOFException eofex) {
return false;
}
}
protected boolean getNextRecord(T target) throws IOException
{
if (this.numRecordsReturned < this.numRecordsInBuffer) {
this.numRecordsReturned++;
this.serializer.deserialize(target, this.readView);
return true;
} else {
return false;
}
}
}