com.emc.mongoose.base.item.io.DelayedTransferConvertBuffer Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of mongoose-base Show documentation
Show all versions of mongoose-base Show documentation
Mongoose is a high-load storage performance testing tool
package com.emc.mongoose.base.item.io;
import static com.github.akurilov.commons.lang.Exceptions.throwUnchecked;
import static java.lang.System.nanoTime;
import com.emc.mongoose.base.item.Item;
import com.emc.mongoose.base.item.TransferConvertBuffer;
import com.emc.mongoose.base.item.op.Operation;
import com.emc.mongoose.base.logging.Loggers;
import com.github.akurilov.commons.io.Input;
import java.io.EOFException;
import java.io.IOException;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
import java.util.ListIterator;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.LockSupport;
import java.util.concurrent.locks.ReentrantLock;
/** Created by kurila on 16.01.17. */
public final class DelayedTransferConvertBuffer>
implements TransferConvertBuffer {
private final List ioResultsBuff;
private final int ioResultsBuffLimit;
private final List markBuffer;
private final long delayMicroseconds;
private final Lock lock = new ReentrantLock();
private volatile int markLimit = 0;
private volatile int ioResultsBuffSize = 0;
private volatile boolean poisonedFlag = false;
public DelayedTransferConvertBuffer(final int limit, final long delay, final TimeUnit timeUnit) {
this.ioResultsBuff = new LinkedList<>();
this.ioResultsBuffLimit = limit;
this.markBuffer = new LinkedList<>();
this.delayMicroseconds = timeUnit.toMicros(delay);
}
/**
* Block until the free space in the buff is available
*
* @param ioResult
* @return always true
* @throws IOException
*/
@Override
public final boolean put(final O ioResult) {
if (poisonedFlag) {
throwUnchecked(new EOFException(this.toString() + ": has been poisoned before"));
}
if (ioResult == null) {
Loggers.MSG.debug("{}: poisoned", this);
return poisonedFlag = true;
}
int ioResultsBuffSize;
while (true) {
if (lock.tryLock()) {
try {
ioResultsBuffSize = this.ioResultsBuffSize;
if (ioResultsBuffSize < ioResultsBuffLimit) {
ioResultsBuff.add(ioResult);
this.ioResultsBuffSize = ioResultsBuffSize + 1;
return true;
}
} finally {
lock.unlock();
}
}
LockSupport.parkNanos(1);
}
}
/**
* Block until all the items from the given range are consumed
*
* @param ioResults
* @param from
* @param to
* @return
* @throws IOException
*/
@Override
public final int put(final List ioResults, final int from, final int to) {
if (poisonedFlag) {
throwUnchecked(new EOFException(this + ": has been poisoned before"));
}
int n;
O ioResult;
for (int i = from; i < to; i++) {
if (lock.tryLock()) {
try {
n = Math.min(to - i, ioResultsBuffLimit - ioResultsBuffSize);
if (n > 0) {
for (int j = 0; j < n; j++) {
ioResult = ioResults.get(i + j);
if (ioResult == null) {
Loggers.MSG.debug("{}: poisoned", this);
poisonedFlag = true;
return to - i - j;
}
ioResultsBuff.add(ioResult);
}
i += n;
// avoid blocking, there's a chance to exit the outer loop
continue;
}
} finally {
lock.unlock();
}
}
LockSupport.parkNanos(1);
}
return to - from;
}
/**
* Block until all the given items are consumed
*
* @param ioResults
* @return
* @throws IOException
*/
@Override
public final int put(final List ioResults) {
return put(ioResults, 0, ioResults.size());
}
/** Don't use this method, it will cause the assertion error */
@Override
public final Input getInput() {
throw new AssertionError();
}
@Override
public final I get() {
I item = null;
if (lock.tryLock()) {
try {
final var ioResultsBuffSize = this.ioResultsBuffSize;
if (ioResultsBuffSize == 0 && poisonedFlag) {
throwUnchecked(new EOFException());
}
O nextIoResult;
long nextFinishTime, currTime;
final var ioResultsIter = ioResultsBuff.listIterator();
while (ioResultsIter.hasNext()) {
nextIoResult = ioResultsIter.next();
final var markLimit = this.markLimit;
if (delayMicroseconds > 0) {
nextFinishTime = nextIoResult.respTimeDone();
currTime = Operation.START_OFFSET_MICROS + nanoTime() / 1000;
if (currTime - nextFinishTime > delayMicroseconds) {
item = nextIoResult.item();
if (markLimit > 0 && markLimit > markBuffer.size()) {
markBuffer.add(nextIoResult);
}
ioResultsIter.remove();
this.ioResultsBuffSize = ioResultsBuffSize + 1;
break;
}
} else {
item = nextIoResult.item();
if (markBuffer.size() < markLimit) {
markBuffer.add(nextIoResult);
}
ioResultsIter.remove();
this.ioResultsBuffSize = ioResultsBuffSize + 1;
break;
}
}
} finally {
lock.unlock();
}
}
return item;
}
@Override
public final int get(final List buffer, final int limit) {
int n = 0;
if (lock.tryLock()) {
try {
final var ioResultsBuffSize = this.ioResultsBuffSize;
if (ioResultsBuffSize == 0 && poisonedFlag) {
throwUnchecked(new EOFException());
}
O nextIoResult;
long nextFinishTime, currTime;
final var ioResultsIter = ioResultsBuff.listIterator();
final var markLimit = this.markLimit;
if (delayMicroseconds > 0) {
while (ioResultsIter.hasNext() && n < limit) {
nextIoResult = ioResultsIter.next();
nextFinishTime = nextIoResult.respTimeDone();
currTime = Operation.START_OFFSET_MICROS + nanoTime() / 1000;
if (currTime - nextFinishTime > delayMicroseconds) {
buffer.add(nextIoResult.item());
if (markLimit > 0 && markLimit > markBuffer.size()) {
markBuffer.add(nextIoResult);
}
ioResultsIter.remove();
this.ioResultsBuffSize = ioResultsBuffSize + 1;
n++;
}
}
} else {
while (ioResultsIter.hasNext() && n < limit) {
nextIoResult = ioResultsIter.next();
buffer.add(nextIoResult.item());
if (markLimit > 0 && markLimit > markBuffer.size()) {
markBuffer.add(nextIoResult);
}
ioResultsIter.remove();
this.ioResultsBuffSize = ioResultsBuffSize + 1;
n++;
}
}
} finally {
lock.unlock();
}
}
return n;
}
@Override
public final long skip(final long count) {
long n = 0;
if (lock.tryLock()) {
try {
final Iterator ioResultsIter = ioResultsBuff.iterator();
while (n < count && ioResultsIter.hasNext()) {
ioResultsIter.remove();
n++;
}
} finally {
lock.unlock();
}
}
return n;
}
@Override
public final void reset() {
throw new AssertionError("Unable to reset this input");
}
@Override
public final void close() {
lock.lock();
try {
poisonedFlag = true;
ioResultsBuff.clear();
ioResultsBuffSize = 0;
} finally {
lock.unlock();
}
}
@Override
public final String toString() {
if (delayMicroseconds > 0) {
return "PreviousItemsWithDelay" + (delayMicroseconds / 1_000_000) + "s";
} else {
return "PreviousItems";
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy