diff --git a/etc/examples/6-enhanced-output/config.json b/etc/examples/6-enhanced-output/config.json
index a16d67dd91b4fd9b4ecc3832161e05997292d22f..b42a3f7334446ee973c7cd64f994e9df7ae110b8 100644
--- a/etc/examples/6-enhanced-output/config.json
+++ b/etc/examples/6-enhanced-output/config.json
@@ -1,5 +1,5 @@
 {
   "output": {
-    "curveTypes": ["TOTAL", "GMM"]
+    "curveTypes": ["TOTAL", "GMM", "SOURCE"]
   }
 }
diff --git a/src/org/opensha2/calc/Calcs.java b/src/org/opensha2/calc/Calcs.java
index df2a653888caa771b53751efe1cedd3292187e28..7ddecbd88850f5e23ef8c136b2c65dd88d31b3c9 100644
--- a/src/org/opensha2/calc/Calcs.java
+++ b/src/org/opensha2/calc/Calcs.java
@@ -107,7 +107,7 @@ public class Calcs {
 
 	/**
 	 * Compute probabilistic seismic hazard, possibly using an {@link Optional}
-	 * {@link Executor}. If no {@code Executor} is supplied, the calculation
+	 * {@link Executor}. If no executor is supplied, the calculation
 	 * will run on the current thread.
 	 * 
 	 * @param model to use
diff --git a/src/org/opensha2/calc/ThreadCount.java b/src/org/opensha2/calc/ThreadCount.java
index cbae631351e8fee64532d85a9339b425e89788c0..f1dc3a94bdbb19a8170517b61c7ba984fec93207 100644
--- a/src/org/opensha2/calc/ThreadCount.java
+++ b/src/org/opensha2/calc/ThreadCount.java
@@ -2,6 +2,8 @@ package org.opensha2.calc;
 
 import java.util.concurrent.ExecutorService;
 
+import com.google.common.util.concurrent.MoreExecutors;
+
 /**
  * The number of threads with which to intialize thread pools. Values reference
  * the number of non-competing threads that could be supported on a particular
@@ -12,28 +14,57 @@ import java.util.concurrent.ExecutorService;
 public enum ThreadCount {
 
 	/**
-	 * A single thread. Use of a single thread will generally prevent an
-	 * {@link ExecutorService} from being used, and all calculations will be run
+	 * A single thread. This identifier will typically cause a program to
+	 * either use Guava's {@link MoreExecutors#directExecutor()} or skip using
+	 * an {@link ExecutorService} altogether, and all calculations will be run
 	 * on the thread from which a program was called. This is useful for
 	 * debugging.
 	 */
-	ONE,
+	ONE {
+		@Override
+		public int value() {
+			return 1;
+		}
+	},
 
 	/**
 	 * Half of {@code ALL}.
 	 */
-	HALF,
+	HALF {
+		@Override
+		public int value() {
+			return Math.max(1, CORES / 2);
+		}
+	},
 
 	/**
-	 * Two less than {@code ALL}. So as to not commandeer all available
+	 * Two less than {@code ALL}, so as to not commandeer all available
 	 * resources.
 	 */
-	N_MINUS_2,
+	N_MINUS_2 {
+		@Override
+		public int value() {
+			return Math.max(1, CORES - 2);
+		}
+	},
 
 	/**
 	 * All possible non-competing threads. The number of threads will equal the
 	 * number of available processors.
 	 */
-	ALL;
+	ALL {
+		@Override
+		public int value() {
+			return CORES;
+		}
+	};
+
+	private static final int CORES = Runtime.getRuntime().availableProcessors();
+
+	/**
+	 * The number of threads relative to the number of available processors on a
+	 * system. The value returned will never be less than one.
+	 */
+	public abstract int value();
 
 }
diff --git a/src/org/opensha2/programs/DeaggCalc.java b/src/org/opensha2/programs/DeaggCalc.java
index 2e07692f15180a5b32f0f5d9f3ec49f43df1989e..d9a9d37df8ae57460afc871bb61d471e4606abc9 100644
--- a/src/org/opensha2/programs/DeaggCalc.java
+++ b/src/org/opensha2/programs/DeaggCalc.java
@@ -57,8 +57,9 @@ public class DeaggCalc {
 	/**
 	 * Perform a hazard deaggregation at a {@code site} for a {@code model},
 	 * {@code config}, and return period. If an {@code executor} is supplied, it
-	 * will be used to distribute hazard calculation tasks; otherwise, one will
-	 * be created.
+	 * will be used to distribute tasks; otherwise, the calculation will run on
+	 * the current thread. Be sure to shutdown any supplied executor after a
+	 * calculation completes.
 	 * 
 	 * <p><b>Note:</b> any model initialization settings in {@code config} will
 	 * be ignored as the supplied model will already have been initialized.</p>
@@ -76,12 +77,8 @@ public class DeaggCalc {
 			Site site,
 			double returnPeriod,
 			Optional<Executor> executor) {
-
-		Optional<Executor> execLocal = executor.or(Optional.of(createExecutor()));
-
 		try {
-			Hazard result = Calcs.hazard(model, config, site, execLocal);
-			if (!executor.isPresent()) ((ExecutorService) executor).shutdown();
+			Hazard result = Calcs.hazard(model, config, site, executor);
 			return Calcs.deaggregation(result, returnPeriod);
 		} catch (ExecutionException | InterruptedException e) {
 			Throwables.propagate(e);
@@ -89,8 +86,4 @@ public class DeaggCalc {
 		}
 	}
 
-	private static ExecutorService createExecutor() {
-		return newFixedThreadPool(getRuntime().availableProcessors());
-	}
-
 }
diff --git a/src/org/opensha2/programs/HazardCalc.java b/src/org/opensha2/programs/HazardCalc.java
index eb05b0344ea07dae40f62e7afce3803a8892973e..aff44addd63bb402adf03d1923b34ef203b2a758 100644
--- a/src/org/opensha2/programs/HazardCalc.java
+++ b/src/org/opensha2/programs/HazardCalc.java
@@ -1,6 +1,5 @@
 package org.opensha2.programs;
 
-import static java.lang.Runtime.getRuntime;
 import static java.nio.file.StandardOpenOption.APPEND;
 import static java.util.concurrent.Executors.newFixedThreadPool;
 import static org.opensha2.util.TextUtils.NEWLINE;
@@ -24,11 +23,11 @@ import org.opensha2.calc.Hazard;
 import org.opensha2.calc.Results;
 import org.opensha2.calc.Site;
 import org.opensha2.calc.Sites;
+import org.opensha2.calc.ThreadCount;
 import org.opensha2.eq.model.HazardModel;
 import org.opensha2.util.Logging;
 
 import com.google.common.base.Optional;
-import com.google.common.base.StandardSystemProperty;
 import com.google.common.base.Stopwatch;
 import com.google.common.base.Throwables;
 
@@ -134,7 +133,8 @@ public class HazardCalc {
 		} catch (Exception e) {
 			throw new IllegalArgumentException(
 				"'sites' [" + arg + "] must either be a 3 to 7 argument, comma-delimited string " +
-					"or specify a path to a *.csv or *.geojson file", e);
+					"or specify a path to a *.csv or *.geojson file",
+				e);
 		}
 	}
 
@@ -150,10 +150,15 @@ public class HazardCalc {
 			Iterable<Site> sites,
 			Logger log) throws IOException {
 
-		ExecutorService execSvc = createExecutor();
-		int threadCount = ((ThreadPoolExecutor) execSvc).getCorePoolSize();
-		log.info("Threads: " + threadCount);
-		Optional<Executor> executor = Optional.<Executor> of(execSvc);
+		ExecutorService execSvc = null;
+		ThreadCount threadCount = config.performance.threadCount;
+		if (threadCount != ThreadCount.ONE) {
+			execSvc = newFixedThreadPool(threadCount.value());
+			log.info("Threads: " + ((ThreadPoolExecutor) execSvc).getCorePoolSize());
+		} else {
+			log.info("Threads: Running on calling thread");
+		}
+		Optional<Executor> executor = Optional.<Executor> fromNullable(execSvc);
 
 		log.info(PROGRAM + ": calculating ...");
 		Stopwatch batchWatch = Stopwatch.createStarted();
@@ -184,13 +189,17 @@ public class HazardCalc {
 		}
 		log.info(PROGRAM + ": " + count + " complete " + totalWatch);
 
-		execSvc.shutdown();
+		if (threadCount != ThreadCount.ONE) {
+			execSvc.shutdown();
+		}
 	}
 
 	/**
 	 * Compute hazard curves at a {@code site} for a {@code model} and
 	 * {@code config}. If an {@code executor} is supplied, it will be used to
-	 * distribute tasks; otherwise, one will be created.
+	 * distribute tasks; otherwise, the calculation will run on the current
+	 * thread. Be sure to shutdown any supplied executor after a calculation
+	 * completes.
 	 * 
 	 * <p><b>Note:</b> any model initialization settings in {@code config} will
 	 * be ignored as the supplied model will already have been initialized.</p>
@@ -206,28 +215,14 @@ public class HazardCalc {
 			CalcConfig config,
 			Site site,
 			Optional<Executor> executor) {
-
-		// TODO not sure why we're mandating an executor here.
-		// legacy from refactoring?
-		Optional<Executor> execLocal = executor.or(Optional.of(createExecutor()));
-
 		try {
-			Hazard result = Calcs.hazard(model, config, site, execLocal);
-			// Shut down the locally created executor if none was supplied
-			if (!executor.isPresent()) {
-				((ExecutorService) execLocal.get()).shutdown();
-			}
-			return result;
+			return Calcs.hazard(model, config, site, executor);
 		} catch (ExecutionException | InterruptedException e) {
 			Throwables.propagate(e);
 			return null;
 		}
 	}
-
-	private static ExecutorService createExecutor() {
-		return newFixedThreadPool(getRuntime().availableProcessors());
-	}
-
+	
 	private static final String PROGRAM = HazardCalc.class.getSimpleName();
 	private static final String USAGE_COMMAND =
 		"java -cp nshmp-haz.jar org.opensha2.programs.HazardCalc model sites [config]";