Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
io.nosqlbench.virtdata.userlibs.apps.valuesapp.ValuesCheckerCoordinator Maven / Gradle / Ivy
package io.nosqlbench.virtdata.userlibs.apps.valuesapp;
import io.nosqlbench.virtdata.core.bindings.DataMapper;
import io.nosqlbench.virtdata.core.bindings.VirtData;
import org.apache.logging.log4j.Logger;
import org.apache.logging.log4j.LogManager;
import java.util.List;
import java.util.concurrent.*;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.ReentrantLock;
public class ValuesCheckerCoordinator implements Callable {
private static final Logger logger =
LogManager.getLogger(ValuesCheckerCoordinator.class);
private final String specifier;
private final int threads;
private final int bufsize;
private final long end;
private final long start;
private final boolean isolated;
// private final Semaphore semaphore;
private final ReentrantLock lock;
private final Condition goTime;
private final ConcurrentLinkedDeque errors = new ConcurrentLinkedDeque<>();
private final ConcurrentLinkedQueue readyQueue = new ConcurrentLinkedQueue<>();
ExecutorService pool;
private long genTimeAccumulator = 0L;
private long cmpTimeAccumulator = 0L;
public ValuesCheckerCoordinator(
String specifier,
int threads,
int bufsize,
long start,
long end,
boolean isolated) {
this.specifier = specifier;
this.threads = threads;
this.bufsize = bufsize;
this.start = start;
this.end = end;
this.isolated = isolated;
// this.semaphore = new Semaphore(threads,false);
this.lock = new ReentrantLock();
this.goTime = lock.newCondition();
}
public void run() {
testConcurrentValues(threads, start, end, specifier);
if (this.errors.size() > 0) {
for (Throwable error : errors) {
System.out.println(error.getMessage());
}
throw new RuntimeException("Errors in verification: " + this.errors);
}
}
private void testConcurrentValues(
int threads,
long start,
long end,
String mapperSpec) {
// Generate reference values in single-threaded mode.
DataMapper mapper =
VirtData.getOptionalMapper(specifier).orElseThrow(
() -> new RuntimeException("Unable to map function for specifier: " + specifier)
);
final List reference = new CopyOnWriteArrayList<>();
// Setup concurrent generator pool
ValuesCheckerExceptionHandler valuesCheckerExceptionHandler =
new ValuesCheckerExceptionHandler(this);
IndexedThreadFactory tf =
new IndexedThreadFactory("values-checker", valuesCheckerExceptionHandler);
pool =
Executors.newFixedThreadPool(threads, tf);
logger.info("Checking [{}..{}) in chunks of {}", start, end, bufsize);
if (!isolated) {
logger.debug(
"Sharing data mapper, only expect success for " +
"explicitly thread-safe generators.");
}
for (int t = 0; t < threads; t++) {
ValuesCheckerRunnable runnable;
if (isolated) {
runnable = new ValuesCheckerRunnable(
start, end, bufsize, t, mapperSpec, null,
readyQueue, goTime, lock, reference
);
} else {
DataMapper> threadMapper = VirtData.getOptionalMapper(mapperSpec)
.orElseThrow(
() -> new RuntimeException("Unable to map function for specifier: " + specifier)
);
runnable = new ValuesCheckerRunnable(
start, end, bufsize, t,null, threadMapper,
readyQueue, goTime, lock, reference
);
}
pool.execute(runnable);
}
logger.info("starting generation loops...");
for (long intervalStart = 0; intervalStart < (end - start); intervalStart += bufsize) {
String rangeInfo = "[" + intervalStart + ".." + (intervalStart + bufsize) + ")";
long genStart = System.nanoTime();
coordinateFor("generation start " + rangeInfo);
throwInjectedExceptions();
coordinateFor("generation complete " + rangeInfo);
long genStop = System.nanoTime();
long genTime = genStop - genStart;
genTimeAccumulator += genTime;
throwInjectedExceptions();
System.out.print(".");
System.out.flush();
long cmpStart = System.nanoTime();
coordinateFor("verification start " + rangeInfo);
throwInjectedExceptions();
coordinateFor("verification complete " + rangeInfo);
long cmpEnd = System.nanoTime();
long cmpTime = cmpEnd - cmpStart;
cmpTimeAccumulator += cmpTime;
throwInjectedExceptions();
System.out.print(".");
System.out.flush();
}
System.out.println("\n");
pool.shutdown();
try {
pool.awaitTermination(60000, TimeUnit.MILLISECONDS);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
private synchronized void throwInjectedExceptions() {
if (errors.peekFirst() != null) {
int count = 0;
for (Throwable error : errors) {
System.out.print("EXCEPTION " + count++ + ": ");
System.out.println(error.getMessage());
}
throw new RuntimeException(errors.peekFirst());
}
}
synchronized void handleException(Thread t, Throwable e) {
this.errors.add(e);
if (pool!=null) {
pool.shutdownNow();
}
}
private void coordinateFor(String forWhat) {
logger.trace("coordinating " + threads + " threads for " + forWhat);
try {
long delay = 1;
while (readyQueue.size() < threads) {
// logger.debug("threads ready for " + forWhat + ": " + readyQueue.size() + ", delaying " + delay + "ms");
Thread.sleep(delay);
delay = Math.min(1024, delay * 2);
throwInjectedExceptions();
}
readyQueue.clear();
lock.lock();
goTime.signalAll();
} catch (Exception e) {
logger.error("Error while signaling threads: " + e.getMessage(), e);
throw new RuntimeException(e);
} finally {
lock.unlock();
}
}
@Override
public RunData call() throws Exception {
run();
return new RunData(
this.specifier,
this.threads,
this.start,
this.end,
this.bufsize,
this.isolated,
((double) genTimeAccumulator / 1000000.0D),
((double) cmpTimeAccumulator / 1000000.0D)
);
}
}