diff --git a/briar-android/src/org/briarproject/android/AndroidNotificationManagerImpl.java b/briar-android/src/org/briarproject/android/AndroidNotificationManagerImpl.java
index 623391cc3b356c755f20b8dd3c855a980abf619a..166c8e84fb05ba68632d9d2921fe321bdc43fbf6 100644
--- a/briar-android/src/org/briarproject/android/AndroidNotificationManagerImpl.java
+++ b/briar-android/src/org/briarproject/android/AndroidNotificationManagerImpl.java
@@ -42,6 +42,7 @@ import java.util.concurrent.Callable;
 import java.util.concurrent.ExecutionException;
 import java.util.concurrent.Executor;
 import java.util.concurrent.Future;
+import java.util.concurrent.atomic.AtomicBoolean;
 import java.util.logging.Logger;
 
 import javax.inject.Inject;
@@ -81,10 +82,10 @@ class AndroidNotificationManagerImpl implements AndroidNotificationManager,
 	private final Context appContext;
 
 	// The following must only be accessed on the main UI thread
-	private final Map<GroupId, Integer> contactCounts =
-			new HashMap<GroupId, Integer>();
-	private final Map<GroupId, Integer> forumCounts =
-			new HashMap<GroupId, Integer>();
+	private final Map<GroupId, Integer> contactCounts = new HashMap<>();
+	private final Map<GroupId, Integer> forumCounts = new HashMap<>();
+	private final AtomicBoolean used = new AtomicBoolean(false);
+
 	private int contactTotal = 0, forumTotal = 0;
 	private int nextRequestId = 0;
 	private GroupId visibleGroup = null;
@@ -106,6 +107,7 @@ class AndroidNotificationManagerImpl implements AndroidNotificationManager,
 
 	@Override
 	public void startService() throws ServiceException {
+		if (used.getAndSet(true)) throw new IllegalStateException();
 		try {
 			settings = settingsManager.getSettings(SETTINGS_NAMESPACE);
 		} catch (DbException e) {
@@ -116,6 +118,7 @@ class AndroidNotificationManagerImpl implements AndroidNotificationManager,
 	@Override
 	public void stopService() throws ServiceException {
 		Future<Void> f = androidExecutor.submit(new Callable<Void>() {
+			@Override
 			public Void call() {
 				clearPrivateMessageNotification();
 				clearForumPostNotification();
@@ -125,9 +128,7 @@ class AndroidNotificationManagerImpl implements AndroidNotificationManager,
 		});
 		try {
 			f.get();
-		} catch (InterruptedException e) {
-			throw new ServiceException(e);
-		} catch (ExecutionException e) {
+		} catch (InterruptedException | ExecutionException e) {
 			throw new ServiceException(e);
 		}
 	}
@@ -150,6 +151,7 @@ class AndroidNotificationManagerImpl implements AndroidNotificationManager,
 		nm.cancel(INTRODUCTION_SUCCESS_NOTIFICATION_ID);
 	}
 
+	@Override
 	public void eventOccurred(Event e) {
 		if (e instanceof SettingsUpdatedEvent) {
 			SettingsUpdatedEvent s = (SettingsUpdatedEvent) e;
@@ -180,6 +182,7 @@ class AndroidNotificationManagerImpl implements AndroidNotificationManager,
 
 	private void loadSettings() {
 		dbExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				try {
 					settings = settingsManager.getSettings(SETTINGS_NAMESPACE);
@@ -191,8 +194,10 @@ class AndroidNotificationManagerImpl implements AndroidNotificationManager,
 		});
 	}
 
+	@Override
 	public void showPrivateMessageNotification(final GroupId g) {
 		androidExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				Integer count = contactCounts.get(g);
 				if (count == null) contactCounts.put(g, 1);
@@ -204,8 +209,10 @@ class AndroidNotificationManagerImpl implements AndroidNotificationManager,
 		});
 	}
 
+	@Override
 	public void clearPrivateMessageNotification(final GroupId g) {
 		androidExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				Integer count = contactCounts.remove(g);
 				if (count == null) return; // Already cleared
@@ -275,8 +282,10 @@ class AndroidNotificationManagerImpl implements AndroidNotificationManager,
 		return defaults;
 	}
 
+	@Override
 	public void showForumPostNotification(final GroupId g) {
 		androidExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				Integer count = forumCounts.get(g);
 				if (count == null) forumCounts.put(g, 1);
@@ -288,8 +297,10 @@ class AndroidNotificationManagerImpl implements AndroidNotificationManager,
 		});
 	}
 
+	@Override
 	public void clearForumPostNotification(final GroupId g) {
 		androidExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				Integer count = forumCounts.remove(g);
 				if (count == null) return; // Already cleared
@@ -347,16 +358,20 @@ class AndroidNotificationManagerImpl implements AndroidNotificationManager,
 		}
 	}
 
+	@Override
 	public void blockNotification(final GroupId g) {
 		androidExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				visibleGroup = g;
 			}
 		});
 	}
 
+	@Override
 	public void unblockNotification(final GroupId g) {
 		androidExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				if (g.equals(visibleGroup)) visibleGroup = null;
 			}
@@ -365,6 +380,7 @@ class AndroidNotificationManagerImpl implements AndroidNotificationManager,
 
 	private void showNotificationForPrivateConversation(final ContactId c) {
 		androidExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				try {
 					GroupId group = messagingManager.getConversationId(c);
@@ -379,6 +395,7 @@ class AndroidNotificationManagerImpl implements AndroidNotificationManager,
 
 	private void showIntroductionSucceededNotification(final Contact c) {
 		androidExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				NotificationCompat.Builder b =
 						new NotificationCompat.Builder(appContext);
diff --git a/briar-android/src/org/briarproject/plugins/droidtooth/DroidtoothPlugin.java b/briar-android/src/org/briarproject/plugins/droidtooth/DroidtoothPlugin.java
index 40a8a041063d78cdd1403c99e99a1603b8dea026..a89216505bf439fa559489915d5eab3e8fe193e1 100644
--- a/briar-android/src/org/briarproject/plugins/droidtooth/DroidtoothPlugin.java
+++ b/briar-android/src/org/briarproject/plugins/droidtooth/DroidtoothPlugin.java
@@ -9,9 +9,9 @@ import android.content.Context;
 import android.content.Intent;
 import android.content.IntentFilter;
 
+import org.briarproject.android.api.AndroidExecutor;
 import org.briarproject.android.util.AndroidUtils;
 import org.briarproject.api.TransportId;
-import org.briarproject.android.api.AndroidExecutor;
 import org.briarproject.api.contact.ContactId;
 import org.briarproject.api.crypto.PseudoRandom;
 import org.briarproject.api.keyagreement.KeyAgreementConnection;
@@ -41,6 +41,7 @@ import java.util.concurrent.ExecutionException;
 import java.util.concurrent.Executor;
 import java.util.concurrent.ExecutorCompletionService;
 import java.util.concurrent.Future;
+import java.util.concurrent.atomic.AtomicBoolean;
 import java.util.logging.Logger;
 
 import static android.bluetooth.BluetoothAdapter.ACTION_SCAN_MODE_CHANGED;
@@ -80,6 +81,7 @@ class DroidtoothPlugin implements DuplexPlugin {
 	private final Backoff backoff;
 	private final DuplexPluginCallback callback;
 	private final int maxLatency;
+	private final AtomicBoolean used = new AtomicBoolean(false);
 
 	private volatile boolean running = false;
 	private volatile boolean wasEnabledByUs = false;
@@ -101,24 +103,30 @@ class DroidtoothPlugin implements DuplexPlugin {
 		this.maxLatency = maxLatency;
 	}
 
+	@Override
 	public TransportId getId() {
 		return ID;
 	}
 
+	@Override
 	public int getMaxLatency() {
 		return maxLatency;
 	}
 
+	@Override
 	public int getMaxIdleTime() {
 		// Bluetooth detects dead connections so we don't need keepalives
 		return Integer.MAX_VALUE;
 	}
 
+	@Override
 	public boolean start() throws IOException {
+		if (used.getAndSet(true)) throw new IllegalStateException();
 		// BluetoothAdapter.getDefaultAdapter() must be called on a thread
 		// with a message queue, so submit it to the AndroidExecutor
 		try {
 			adapter = androidExecutor.submit(new Callable<BluetoothAdapter>() {
+				@Override
 				public BluetoothAdapter call() throws Exception {
 					return BluetoothAdapter.getDefaultAdapter();
 				}
@@ -158,6 +166,7 @@ class DroidtoothPlugin implements DuplexPlugin {
 
 	private void bind() {
 		ioExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				if (!isRunning()) return;
 				String address = AndroidUtils.getBluetoothAddress(appContext,
@@ -238,6 +247,7 @@ class DroidtoothPlugin implements DuplexPlugin {
 		return new DroidtoothTransportConnection(this, s);
 	}
 
+	@Override
 	public void stop() {
 		running = false;
 		if (receiver != null) appContext.unregisterReceiver(receiver);
@@ -249,18 +259,22 @@ class DroidtoothPlugin implements DuplexPlugin {
 		}
 	}
 
+	@Override
 	public boolean isRunning() {
 		return running && adapter.isEnabled();
 	}
 
+	@Override
 	public boolean shouldPoll() {
 		return true;
 	}
 
+	@Override
 	public int getPollingInterval() {
 		return backoff.getPollingInterval();
 	}
 
+	@Override
 	public void poll(Collection<ContactId> connected) {
 		if (!isRunning()) return;
 		backoff.increment();
@@ -275,6 +289,7 @@ class DroidtoothPlugin implements DuplexPlugin {
 			final String uuid = e.getValue().get(PROP_UUID);
 			if (StringUtils.isNullOrEmpty(uuid)) continue;
 			ioExecutor.execute(new Runnable() {
+				@Override
 				public void run() {
 					if (!running) return;
 					BluetoothSocket s = connect(address, uuid);
@@ -327,6 +342,7 @@ class DroidtoothPlugin implements DuplexPlugin {
 		}
 	}
 
+	@Override
 	public DuplexTransportConnection createConnection(ContactId c) {
 		if (!isRunning()) return null;
 		TransportProperties p = callback.getRemoteProperties().get(c);
@@ -340,10 +356,12 @@ class DroidtoothPlugin implements DuplexPlugin {
 		return new DroidtoothTransportConnection(this, s);
 	}
 
+	@Override
 	public boolean supportsInvitations() {
 		return true;
 	}
 
+	@Override
 	public DuplexTransportConnection createInvitationConnection(PseudoRandom r,
 			long timeout, boolean alice) {
 		if (!isRunning()) return null;
@@ -361,9 +379,8 @@ class DroidtoothPlugin implements DuplexPlugin {
 		}
 		// Create the background tasks
 		CompletionService<BluetoothSocket> complete =
-				new ExecutorCompletionService<BluetoothSocket>(ioExecutor);
-		List<Future<BluetoothSocket>> futures =
-				new ArrayList<Future<BluetoothSocket>>();
+				new ExecutorCompletionService<>(ioExecutor);
+		List<Future<BluetoothSocket>> futures = new ArrayList<>();
 		if (alice) {
 			// Return the first connected socket
 			futures.add(complete.submit(new ListeningTask(ss)));
@@ -398,6 +415,7 @@ class DroidtoothPlugin implements DuplexPlugin {
 	private void closeSockets(final List<Future<BluetoothSocket>> futures,
 			final BluetoothSocket chosen) {
 		ioExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				for (Future<BluetoothSocket> f : futures) {
 					try {
@@ -413,9 +431,7 @@ class DroidtoothPlugin implements DuplexPlugin {
 					} catch (InterruptedException e) {
 						LOG.info("Interrupted while closing sockets");
 						return;
-					} catch (ExecutionException e) {
-						if (LOG.isLoggable(INFO)) LOG.info(e.toString());
-					} catch (IOException e) {
+					} catch (ExecutionException | IOException e) {
 						if (LOG.isLoggable(INFO)) LOG.info(e.toString());
 					}
 				}
@@ -423,14 +439,15 @@ class DroidtoothPlugin implements DuplexPlugin {
 		});
 	}
 
+	@Override
 	public boolean supportsKeyAgreement() {
 		return true;
 	}
 
-	public KeyAgreementListener createKeyAgreementListener(
-			byte[] localCommitment) {
+	@Override
+	public KeyAgreementListener createKeyAgreementListener(byte[] commitment) {
 		// No truncation necessary because COMMIT_LENGTH = 16
-		UUID uuid = UUID.nameUUIDFromBytes(localCommitment);
+		UUID uuid = UUID.nameUUIDFromBytes(commitment);
 		if (LOG.isLoggable(INFO)) LOG.info("Key agreement UUID " + uuid);
 		// Bind a server socket for receiving invitation connections
 		BluetoothServerSocket ss;
@@ -448,8 +465,9 @@ class DroidtoothPlugin implements DuplexPlugin {
 		return new BluetoothKeyAgreementListener(d, ss);
 	}
 
+	@Override
 	public DuplexTransportConnection createKeyAgreementConnection(
-			byte[] remoteCommitment, TransportDescriptor d, long timeout) {
+			byte[] commitment, TransportDescriptor d, long timeout) {
 		if (!isRunning()) return null;
 		if (!ID.equals(d.getIdentifier())) return null;
 		TransportProperties p = d.getProperties();
@@ -457,7 +475,7 @@ class DroidtoothPlugin implements DuplexPlugin {
 		String address = p.get(PROP_ADDRESS);
 		if (StringUtils.isNullOrEmpty(address)) return null;
 		// No truncation necessary because COMMIT_LENGTH = 16
-		UUID uuid = UUID.nameUUIDFromBytes(remoteCommitment);
+		UUID uuid = UUID.nameUUIDFromBytes(commitment);
 		if (LOG.isLoggable(INFO))
 			LOG.info("Connecting to key agreement UUID " + uuid);
 		BluetoothSocket s = connect(address, uuid.toString());
@@ -533,7 +551,7 @@ class DroidtoothPlugin implements DuplexPlugin {
 	private static class DiscoveryReceiver extends BroadcastReceiver {
 
 		private final CountDownLatch finished = new CountDownLatch(1);
-		private final List<String> addresses = new ArrayList<String>();
+		private final List<String> addresses = new ArrayList<>();
 
 		@Override
 		public void onReceive(Context ctx, Intent intent) {
diff --git a/briar-android/src/org/briarproject/plugins/tcp/AndroidLanTcpPlugin.java b/briar-android/src/org/briarproject/plugins/tcp/AndroidLanTcpPlugin.java
index e04e2f4fcb567eaed59f4f68d0b27eacc8b866c0..598e914b6d695d1373fba80145c392e2abe32e7f 100644
--- a/briar-android/src/org/briarproject/plugins/tcp/AndroidLanTcpPlugin.java
+++ b/briar-android/src/org/briarproject/plugins/tcp/AndroidLanTcpPlugin.java
@@ -35,6 +35,7 @@ class AndroidLanTcpPlugin extends LanTcpPlugin {
 
 	@Override
 	public boolean start() {
+		if (used.getAndSet(true)) throw new IllegalStateException();
 		running = true;
 		// Register to receive network status events
 		networkStateReceiver = new NetworkStateReceiver();
diff --git a/briar-android/src/org/briarproject/plugins/tor/TorPlugin.java b/briar-android/src/org/briarproject/plugins/tor/TorPlugin.java
index 19499171b2279cfb18510382618d787586c459c5..0f7aaf65c1c5a48584f07b9a24e5d8f7c53c47c9 100644
--- a/briar-android/src/org/briarproject/plugins/tor/TorPlugin.java
+++ b/briar-android/src/org/briarproject/plugins/tor/TorPlugin.java
@@ -51,6 +51,7 @@ import java.util.Map;
 import java.util.Scanner;
 import java.util.concurrent.CountDownLatch;
 import java.util.concurrent.Executor;
+import java.util.concurrent.atomic.AtomicBoolean;
 import java.util.logging.Logger;
 import java.util.regex.Pattern;
 import java.util.zip.ZipInputStream;
@@ -94,6 +95,7 @@ class TorPlugin implements DuplexPlugin, EventHandler, EventListener {
 	private final File torDirectory, torFile, geoIpFile, configFile;
 	private final File doneFile, cookieFile;
 	private final PowerManager.WakeLock wakeLock;
+	private final AtomicBoolean used = new AtomicBoolean(false);
 
 	private volatile boolean running = false;
 	private volatile ServerSocket socket = null;
@@ -130,19 +132,24 @@ class TorPlugin implements DuplexPlugin, EventHandler, EventListener {
 		wakeLock.setReferenceCounted(false);
 	}
 
+	@Override
 	public TransportId getId() {
 		return ID;
 	}
 
+	@Override
 	public int getMaxLatency() {
 		return maxLatency;
 	}
 
+	@Override
 	public int getMaxIdleTime() {
 		return maxIdleTime;
 	}
 
+	@Override
 	public boolean start() throws IOException {
+		if (used.getAndSet(true)) throw new IllegalStateException();
 		// Try to connect to an existing Tor process if there is one
 		boolean startProcess = false;
 		try {
@@ -210,13 +217,13 @@ class TorPlugin implements DuplexPlugin, EventHandler, EventListener {
 			// Now we should be able to connect to the new process
 			controlSocket = new Socket("127.0.0.1", CONTROL_PORT);
 		}
-		running = true;
 		// Open a control connection and authenticate using the cookie file
 		controlConnection = new TorControlConnection(controlSocket);
 		controlConnection.authenticate(read(cookieFile));
 		// Tell Tor to exit when the control connection is closed
 		controlConnection.takeOwnership();
 		controlConnection.resetConf(Collections.singletonList(OWNER));
+		running = true;
 		// Register to receive events from the Tor process
 		controlConnection.setEventHandler(this);
 		controlConnection.setEvents(Arrays.asList(EVENTS));
@@ -226,7 +233,6 @@ class TorPlugin implements DuplexPlugin, EventHandler, EventListener {
 			if (phase != null && phase.contains("PROGRESS=100")) {
 				LOG.info("Tor has already bootstrapped");
 				connectionStatus.setBootstrapped();
-				sendDevReports();
 			}
 		}
 		// Register to receive network status events
@@ -369,6 +375,7 @@ class TorPlugin implements DuplexPlugin, EventHandler, EventListener {
 
 	private void bind() {
 		ioExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				// If there's already a port number stored in config, reuse it
 				String portString = callback.getSettings().get("port");
@@ -398,6 +405,7 @@ class TorPlugin implements DuplexPlugin, EventHandler, EventListener {
 				callback.mergeSettings(s);
 				// Create a hidden service if necessary
 				ioExecutor.execute(new Runnable() {
+					@Override
 					public void run() {
 						publishHiddenService(localPort);
 					}
@@ -486,6 +494,7 @@ class TorPlugin implements DuplexPlugin, EventHandler, EventListener {
 		}
 	}
 
+	@Override
 	public void stop() throws IOException {
 		running = false;
 		tryToClose(socket);
@@ -508,18 +517,22 @@ class TorPlugin implements DuplexPlugin, EventHandler, EventListener {
 		wakeLock.release();
 	}
 
+	@Override
 	public boolean isRunning() {
 		return running && connectionStatus.isConnected();
 	}
 
+	@Override
 	public boolean shouldPoll() {
 		return true;
 	}
 
+	@Override
 	public int getPollingInterval() {
 		return backoff.getPollingInterval();
 	}
 
+	@Override
 	public void poll(Collection<ContactId> connected) {
 		if (!isRunning()) return;
 		backoff.increment();
@@ -530,6 +543,7 @@ class TorPlugin implements DuplexPlugin, EventHandler, EventListener {
 
 	private void connectAndCallBack(final ContactId c) {
 		ioExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				DuplexTransportConnection d = createConnection(c);
 				if (d != null) {
@@ -540,6 +554,7 @@ class TorPlugin implements DuplexPlugin, EventHandler, EventListener {
 		});
 	}
 
+	@Override
 	public DuplexTransportConnection createConnection(ContactId c) {
 		if (!isRunning()) return null;
 		TransportProperties p = callback.getRemoteProperties().get(c);
@@ -566,61 +581,77 @@ class TorPlugin implements DuplexPlugin, EventHandler, EventListener {
 		}
 	}
 
+	@Override
 	public boolean supportsInvitations() {
 		return false;
 	}
 
+	@Override
 	public DuplexTransportConnection createInvitationConnection(PseudoRandom r,
 			long timeout, boolean alice) {
 		throw new UnsupportedOperationException();
 	}
 
+	@Override
 	public boolean supportsKeyAgreement() {
 		return false;
 	}
 
-	public KeyAgreementListener createKeyAgreementListener(
-			byte[] commitment) {
+	@Override
+	public KeyAgreementListener createKeyAgreementListener(byte[] commitment) {
 		throw new UnsupportedOperationException();
 	}
 
+	@Override
 	public DuplexTransportConnection createKeyAgreementConnection(
 			byte[] commitment, TransportDescriptor d, long timeout) {
 		throw new UnsupportedOperationException();
 	}
 
+	@Override
 	public void circuitStatus(String status, String id, String path) {
 		if (status.equals("BUILT") &&
 				connectionStatus.getAndSetCircuitBuilt()) {
 			LOG.info("First circuit built");
 			backoff.reset();
-			if (isRunning()) callback.transportEnabled();
+			if (isRunning()) {
+				sendDevReports();
+				callback.transportEnabled();
+			}
 		}
 	}
 
+	@Override
 	public void streamStatus(String status, String id, String target) {
 	}
 
+	@Override
 	public void orConnStatus(String status, String orName) {
 		if (LOG.isLoggable(INFO)) LOG.info("OR connection " + status);
 	}
 
+	@Override
 	public void bandwidthUsed(long read, long written) {
 	}
 
+	@Override
 	public void newDescriptors(List<String> orList) {
 	}
 
+	@Override
 	public void message(String severity, String msg) {
 		if (LOG.isLoggable(INFO)) LOG.info(severity + " " + msg);
 		if (severity.equals("NOTICE") && msg.startsWith("Bootstrapped 100%")) {
 			connectionStatus.setBootstrapped();
-			sendDevReports();
 			backoff.reset();
-			if (isRunning()) callback.transportEnabled();
+			if (isRunning()) {
+				sendDevReports();
+				callback.transportEnabled();
+			}
 		}
 	}
 
+	@Override
 	public void unrecognized(String type, String msg) {
 		if (type.equals("HS_DESC") && msg.startsWith("UPLOADED"))
 			LOG.info("Descriptor uploaded");
@@ -642,6 +673,7 @@ class TorPlugin implements DuplexPlugin, EventHandler, EventListener {
 		}
 	}
 
+	@Override
 	public void eventOccurred(Event e) {
 		if (e instanceof SettingsUpdatedEvent) {
 			if (((SettingsUpdatedEvent) e).getNamespace().equals("tor")) {
@@ -653,13 +685,14 @@ class TorPlugin implements DuplexPlugin, EventHandler, EventListener {
 
 	private void updateConnectionStatus() {
 		ioExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				if (!running) return;
 
 				Object o = appContext.getSystemService(CONNECTIVITY_SERVICE);
 				ConnectivityManager cm = (ConnectivityManager) o;
 				NetworkInfo net = cm.getActiveNetworkInfo();
-				boolean online = net != null  && net.isConnected();
+				boolean online = net != null && net.isConnected();
 				boolean wifi = online && net.getType() == TYPE_WIFI;
 				String country = locationUtils.getCurrentCountry();
 				boolean blocked = TorNetworkMetadata.isTorProbablyBlocked(
diff --git a/briar-api/src/org/briarproject/api/reporting/DevReporter.java b/briar-api/src/org/briarproject/api/reporting/DevReporter.java
index f4b17820c1a46394e20ae9e9769db0adc490f17b..7c0f14aa010caa3d12135eb3199637b2171adbf8 100644
--- a/briar-api/src/org/briarproject/api/reporting/DevReporter.java
+++ b/briar-api/src/org/briarproject/api/reporting/DevReporter.java
@@ -2,7 +2,6 @@ package org.briarproject.api.reporting;
 
 import java.io.File;
 import java.io.FileNotFoundException;
-import java.io.IOException;
 
 /**
  * A task for reporting back to the developers.
diff --git a/briar-core/src/org/briarproject/lifecycle/LifecycleManagerImpl.java b/briar-core/src/org/briarproject/lifecycle/LifecycleManagerImpl.java
index b227943116e713cd5a277e7ad2699583b3149a7d..3abfa4cbc3ac732a2cff9c66bf5c2f9395580350 100644
--- a/briar-core/src/org/briarproject/lifecycle/LifecycleManagerImpl.java
+++ b/briar-core/src/org/briarproject/lifecycle/LifecycleManagerImpl.java
@@ -53,20 +53,20 @@ class LifecycleManagerImpl implements LifecycleManager {
 	@Override
 	public void registerService(Service s) {
 		if (LOG.isLoggable(INFO))
-			LOG.info("Registering service " + s.getClass().getName());
+			LOG.info("Registering service " + s.getClass().getSimpleName());
 		services.add(s);
 	}
 
 	@Override
 	public void registerClient(Client c) {
 		if (LOG.isLoggable(INFO))
-			LOG.info("Registering client " + c.getClass().getName());
+			LOG.info("Registering client " + c.getClass().getSimpleName());
 		clients.add(c);
 	}
 
 	@Override
 	public void registerForShutdown(ExecutorService e) {
-		LOG.info("Registering executor");
+		LOG.info("Registering executor " + e.getClass().getSimpleName());
 		executors.add(e);
 	}
 
@@ -94,7 +94,8 @@ class LifecycleManagerImpl implements LifecycleManager {
 					c.createLocalState(txn);
 					duration = System.currentTimeMillis() - start;
 					if (LOG.isLoggable(INFO)) {
-						LOG.info("Starting client " + c.getClass().getName()
+						LOG.info("Starting client "
+								+ c.getClass().getSimpleName()
 								+ " took " + duration + " ms");
 					}
 				}
@@ -107,7 +108,7 @@ class LifecycleManagerImpl implements LifecycleManager {
 				s.startService();
 				duration = System.currentTimeMillis() - start;
 				if (LOG.isLoggable(INFO)) {
-					LOG.info("Starting service " + s.getClass().getName()
+					LOG.info("Starting service " + s.getClass().getSimpleName()
 							+ " took " + duration + " ms");
 				}
 			}
@@ -140,13 +141,17 @@ class LifecycleManagerImpl implements LifecycleManager {
 				s.stopService();
 				long duration = System.currentTimeMillis() - start;
 				if (LOG.isLoggable(INFO)) {
-					LOG.info("Stopping service " + s.getClass().getName()
+					LOG.info("Stopping service " + s.getClass().getSimpleName()
 							+ " took " + duration + " ms");
 				}
 			}
-			for (ExecutorService e : executors) e.shutdownNow();
-			if (LOG.isLoggable(INFO))
-				LOG.info(executors.size() + " executors shut down");
+			for (ExecutorService e : executors) {
+				if (LOG.isLoggable(INFO)) {
+					LOG.info("Stopping executor "
+							+ e.getClass().getSimpleName());
+				}
+				e.shutdownNow();
+			}
 			long start = System.currentTimeMillis();
 			db.close();
 			long duration = System.currentTimeMillis() - start;
diff --git a/briar-core/src/org/briarproject/plugins/PluginManagerImpl.java b/briar-core/src/org/briarproject/plugins/PluginManagerImpl.java
index b93756b18bac6bb411d71afe1730fa3847857e71..e8b9b129d79120db7198b0dbcdad851d838d22bb 100644
--- a/briar-core/src/org/briarproject/plugins/PluginManagerImpl.java
+++ b/briar-core/src/org/briarproject/plugins/PluginManagerImpl.java
@@ -39,6 +39,7 @@ import java.util.concurrent.ConcurrentHashMap;
 import java.util.concurrent.CopyOnWriteArrayList;
 import java.util.concurrent.CountDownLatch;
 import java.util.concurrent.Executor;
+import java.util.concurrent.atomic.AtomicBoolean;
 import java.util.logging.Logger;
 
 import javax.inject.Inject;
@@ -61,6 +62,8 @@ class PluginManagerImpl implements PluginManager, Service {
 	private final Map<TransportId, Plugin> plugins;
 	private final List<SimplexPlugin> simplexPlugins;
 	private final List<DuplexPlugin> duplexPlugins;
+	private final Map<TransportId, CountDownLatch> startLatches;
+	private final AtomicBoolean used = new AtomicBoolean(false);
 
 	@Inject
 	PluginManagerImpl(@IoExecutor Executor ioExecutor, EventBus eventBus,
@@ -78,68 +81,64 @@ class PluginManagerImpl implements PluginManager, Service {
 		plugins = new ConcurrentHashMap<TransportId, Plugin>();
 		simplexPlugins = new CopyOnWriteArrayList<SimplexPlugin>();
 		duplexPlugins = new CopyOnWriteArrayList<DuplexPlugin>();
+		startLatches = new ConcurrentHashMap<TransportId, CountDownLatch>();
 	}
 
 	@Override
 	public void startService() throws ServiceException {
-		Collection<SimplexPluginFactory> simplexFactories =
-				pluginConfig.getSimplexFactories();
-		Collection<DuplexPluginFactory> duplexFactories =
-				pluginConfig.getDuplexFactories();
-		int numPlugins = simplexFactories.size() + duplexFactories.size();
-		CountDownLatch latch = new CountDownLatch(numPlugins);
-		// Instantiate and start the simplex plugins
+		if (used.getAndSet(true)) throw new IllegalStateException();
+		// Instantiate the simplex plugins and start them asynchronously
 		LOG.info("Starting simplex plugins");
-		for (SimplexPluginFactory f : simplexFactories) {
+		for (SimplexPluginFactory f : pluginConfig.getSimplexFactories()) {
 			TransportId t = f.getId();
 			SimplexPlugin s = f.createPlugin(new SimplexCallback(t));
 			if (s == null) {
 				if (LOG.isLoggable(WARNING))
 					LOG.warning("Could not create plugin for " + t);
-				latch.countDown();
 			} else {
 				plugins.put(t, s);
 				simplexPlugins.add(s);
-				ioExecutor.execute(new PluginStarter(s, latch));
+				CountDownLatch startLatch = new CountDownLatch(1);
+				startLatches.put(t, startLatch);
+				ioExecutor.execute(new PluginStarter(s, startLatch));
 			}
 		}
-		// Instantiate and start the duplex plugins
+		// Instantiate the duplex plugins and start them asynchronously
 		LOG.info("Starting duplex plugins");
-		for (DuplexPluginFactory f : duplexFactories) {
+		for (DuplexPluginFactory f : pluginConfig.getDuplexFactories()) {
 			TransportId t = f.getId();
 			DuplexPlugin d = f.createPlugin(new DuplexCallback(t));
 			if (d == null) {
 				if (LOG.isLoggable(WARNING))
 					LOG.warning("Could not create plugin for " + t);
-				latch.countDown();
 			} else {
 				plugins.put(t, d);
 				duplexPlugins.add(d);
-				ioExecutor.execute(new PluginStarter(d, latch));
+				CountDownLatch startLatch = new CountDownLatch(1);
+				startLatches.put(t, startLatch);
+				ioExecutor.execute(new PluginStarter(d, startLatch));
 			}
 		}
-		// Wait for all the plugins to start
-		try {
-			latch.await();
-		} catch (InterruptedException e) {
-			throw new ServiceException(e);
-		}
 	}
 
 	@Override
 	public void stopService() throws ServiceException {
-		CountDownLatch latch = new CountDownLatch(plugins.size());
+		CountDownLatch stopLatch = new CountDownLatch(plugins.size());
 		// Stop the simplex plugins
 		LOG.info("Stopping simplex plugins");
-		for (SimplexPlugin plugin : simplexPlugins)
-			ioExecutor.execute(new PluginStopper(plugin, latch));
+		for (SimplexPlugin s : simplexPlugins) {
+			CountDownLatch startLatch = startLatches.get(s.getId());
+			ioExecutor.execute(new PluginStopper(s, startLatch, stopLatch));
+		}
 		// Stop the duplex plugins
 		LOG.info("Stopping duplex plugins");
-		for (DuplexPlugin plugin : duplexPlugins)
-			ioExecutor.execute(new PluginStopper(plugin, latch));
+		for (DuplexPlugin d : duplexPlugins) {
+			CountDownLatch startLatch = startLatches.get(d.getId());
+			ioExecutor.execute(new PluginStopper(d, startLatch, stopLatch));
+		}
 		// Wait for all the plugins to stop
 		try {
-			latch.await();
+			stopLatch.await();
 		} catch (InterruptedException e) {
 			throw new ServiceException(e);
 		}
@@ -179,11 +178,11 @@ class PluginManagerImpl implements PluginManager, Service {
 	private class PluginStarter implements Runnable {
 
 		private final Plugin plugin;
-		private final CountDownLatch latch;
+		private final CountDownLatch startLatch;
 
-		private PluginStarter(Plugin plugin, CountDownLatch latch) {
+		private PluginStarter(Plugin plugin, CountDownLatch startLatch) {
 			this.plugin = plugin;
-			this.latch = latch;
+			this.startLatch = startLatch;
 		}
 
 		@Override
@@ -209,7 +208,7 @@ class PluginManagerImpl implements PluginManager, Service {
 						LOG.log(WARNING, e.toString(), e);
 				}
 			} finally {
-				latch.countDown();
+				startLatch.countDown();
 			}
 		}
 	}
@@ -217,16 +216,21 @@ class PluginManagerImpl implements PluginManager, Service {
 	private class PluginStopper implements Runnable {
 
 		private final Plugin plugin;
-		private final CountDownLatch latch;
+		private final CountDownLatch startLatch, stopLatch;
 
-		private PluginStopper(Plugin plugin, CountDownLatch latch) {
+		private PluginStopper(Plugin plugin, CountDownLatch startLatch,
+				CountDownLatch stopLatch) {
 			this.plugin = plugin;
-			this.latch = latch;
+			this.startLatch = startLatch;
+			this.stopLatch = stopLatch;
 		}
 
 		@Override
 		public void run() {
 			try {
+				// Wait for the plugin to finish starting
+				startLatch.await();
+				// Stop the plugin
 				long start = System.currentTimeMillis();
 				plugin.stop();
 				long duration = System.currentTimeMillis() - start;
@@ -234,10 +238,13 @@ class PluginManagerImpl implements PluginManager, Service {
 					LOG.info("Stopping plugin " + plugin.getId()
 							+ " took " + duration + " ms");
 				}
+			} catch (InterruptedException e) {
+				LOG.warning("Interrupted while waiting for plugin to start");
+				// This task runs on an executor, so don't reset the interrupt
 			} catch (IOException e) {
 				if (LOG.isLoggable(WARNING)) LOG.log(WARNING, e.toString(), e);
 			} finally {
-				latch.countDown();
+				stopLatch.countDown();
 			}
 		}
 	}
diff --git a/briar-core/src/org/briarproject/plugins/file/FilePlugin.java b/briar-core/src/org/briarproject/plugins/file/FilePlugin.java
index 3394995cfd6ff0cac1503409b0f33aacf22e0919..9819235539b04318dcef32152617f4e32f3b7254 100644
--- a/briar-core/src/org/briarproject/plugins/file/FilePlugin.java
+++ b/briar-core/src/org/briarproject/plugins/file/FilePlugin.java
@@ -14,6 +14,7 @@ import java.io.OutputStream;
 import java.util.Collection;
 import java.util.Locale;
 import java.util.concurrent.Executor;
+import java.util.concurrent.atomic.AtomicBoolean;
 import java.util.logging.Logger;
 
 import static java.util.logging.Level.WARNING;
@@ -27,6 +28,7 @@ public abstract class FilePlugin implements SimplexPlugin {
 	protected final Executor ioExecutor;
 	protected final SimplexPluginCallback callback;
 	protected final int maxLatency;
+	protected final AtomicBoolean used = new AtomicBoolean(false);
 
 	protected volatile boolean running = false;
 
@@ -42,22 +44,27 @@ public abstract class FilePlugin implements SimplexPlugin {
 		this.maxLatency = maxLatency;
 	}
 
+	@Override
 	public int getMaxLatency() {
 		return maxLatency;
 	}
 
+	@Override
 	public int getMaxIdleTime() {
 		return Integer.MAX_VALUE; // We don't need keepalives
 	}
 
+	@Override
 	public boolean isRunning() {
 		return running;
 	}
 
+	@Override
 	public TransportConnectionReader createReader(ContactId c) {
 		return null;
 	}
 
+	@Override
 	public TransportConnectionWriter createWriter(ContactId c) {
 		if (!running) return null;
 		return createWriter(createConnectionFilename());
@@ -105,6 +112,7 @@ public abstract class FilePlugin implements SimplexPlugin {
 			this.file = file;
 		}
 
+		@Override
 		public void run() {
 			if (isPossibleConnectionFilename(file.getName())) {
 				try {
diff --git a/briar-core/src/org/briarproject/plugins/tcp/LanTcpPlugin.java b/briar-core/src/org/briarproject/plugins/tcp/LanTcpPlugin.java
index 3429d06b82922618a7c037c66e4b7f15cf69a5ae..bb2a4de7faee44f5d4c5bde20949d8290a32273a 100644
--- a/briar-core/src/org/briarproject/plugins/tcp/LanTcpPlugin.java
+++ b/briar-core/src/org/briarproject/plugins/tcp/LanTcpPlugin.java
@@ -31,6 +31,7 @@ class LanTcpPlugin extends TcpPlugin {
 		super(ioExecutor, backoff, callback, maxLatency, maxIdleTime);
 	}
 
+	@Override
 	public TransportId getId() {
 		return ID;
 	}
diff --git a/briar-core/src/org/briarproject/plugins/tcp/TcpPlugin.java b/briar-core/src/org/briarproject/plugins/tcp/TcpPlugin.java
index ead2c80b89680017659d240bd2224cab5683c214..5fc8742fdc312fd2fdbb6e5d738f18be17d44b5d 100644
--- a/briar-core/src/org/briarproject/plugins/tcp/TcpPlugin.java
+++ b/briar-core/src/org/briarproject/plugins/tcp/TcpPlugin.java
@@ -24,6 +24,7 @@ import java.util.Collection;
 import java.util.Collections;
 import java.util.List;
 import java.util.concurrent.Executor;
+import java.util.concurrent.atomic.AtomicBoolean;
 import java.util.logging.Logger;
 import java.util.regex.Pattern;
 
@@ -41,6 +42,7 @@ abstract class TcpPlugin implements DuplexPlugin {
 	protected final Backoff backoff;
 	protected final DuplexPluginCallback callback;
 	protected final int maxLatency, maxIdleTime, socketTimeout;
+	protected final AtomicBoolean used = new AtomicBoolean(false);
 
 	protected volatile boolean running = false;
 	protected volatile ServerSocket socket = null;
@@ -81,15 +83,19 @@ abstract class TcpPlugin implements DuplexPlugin {
 		else socketTimeout = maxIdleTime * 2;
 	}
 
+	@Override
 	public int getMaxLatency() {
 		return maxLatency;
 	}
 
+	@Override
 	public int getMaxIdleTime() {
 		return maxIdleTime;
 	}
 
+	@Override
 	public boolean start() {
+		if (used.getAndSet(true)) throw new IllegalStateException();
 		running = true;
 		bind();
 		return true;
@@ -97,6 +103,7 @@ abstract class TcpPlugin implements DuplexPlugin {
 
 	protected void bind() {
 		ioExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				if (!running) return;
 				ServerSocket ss = null;
@@ -166,23 +173,28 @@ abstract class TcpPlugin implements DuplexPlugin {
 		}
 	}
 
+	@Override
 	public void stop() {
 		running = false;
 		tryToClose(socket);
 	}
 
+	@Override
 	public boolean isRunning() {
 		return running && socket != null && !socket.isClosed();
 	}
 
+	@Override
 	public boolean shouldPoll() {
 		return true;
 	}
 
+	@Override
 	public int getPollingInterval() {
 		return backoff.getPollingInterval();
 	}
 
+	@Override
 	public void poll(Collection<ContactId> connected) {
 		if (!isRunning()) return;
 		backoff.increment();
@@ -193,6 +205,7 @@ abstract class TcpPlugin implements DuplexPlugin {
 
 	private void connectAndCallBack(final ContactId c) {
 		ioExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				DuplexTransportConnection d = createConnection(c);
 				if (d != null) {
@@ -203,6 +216,7 @@ abstract class TcpPlugin implements DuplexPlugin {
 		});
 	}
 
+	@Override
 	public DuplexTransportConnection createConnection(ContactId c) {
 		if (!isRunning()) return null;
 		for (InetSocketAddress remote : getRemoteSocketAddresses(c)) {
@@ -250,24 +264,28 @@ abstract class TcpPlugin implements DuplexPlugin {
 		}
 	}
 
+	@Override
 	public boolean supportsInvitations() {
 		return false;
 	}
 
+	@Override
 	public DuplexTransportConnection createInvitationConnection(PseudoRandom r,
 			long timeout, boolean alice) {
 		throw new UnsupportedOperationException();
 	}
 
+	@Override
 	public boolean supportsKeyAgreement() {
 		return false;
 	}
 
-	public KeyAgreementListener createKeyAgreementListener(
-			byte[] commitment) {
+	@Override
+	public KeyAgreementListener createKeyAgreementListener(byte[] commitment) {
 		throw new UnsupportedOperationException();
 	}
 
+	@Override
 	public DuplexTransportConnection createKeyAgreementConnection(
 			byte[] commitment, TransportDescriptor d, long timeout) {
 		throw new UnsupportedOperationException();
diff --git a/briar-core/src/org/briarproject/plugins/tcp/WanTcpPlugin.java b/briar-core/src/org/briarproject/plugins/tcp/WanTcpPlugin.java
index 4274a8cbe4aa7e486ee99eea00aca5b69d7d1059..bc54da0ae4f089e42f9a7f22509c709d431fb3de 100644
--- a/briar-core/src/org/briarproject/plugins/tcp/WanTcpPlugin.java
+++ b/briar-core/src/org/briarproject/plugins/tcp/WanTcpPlugin.java
@@ -31,6 +31,7 @@ class WanTcpPlugin extends TcpPlugin {
 		this.portMapper = portMapper;
 	}
 
+	@Override
 	public TransportId getId() {
 		return ID;
 	}
diff --git a/briar-core/src/org/briarproject/sync/ValidationManagerImpl.java b/briar-core/src/org/briarproject/sync/ValidationManagerImpl.java
index a5d02b5cc497155053a5d81459ad39874ff730b1..52121c706aae350a94cb8fbddd28597127278667 100644
--- a/briar-core/src/org/briarproject/sync/ValidationManagerImpl.java
+++ b/briar-core/src/org/briarproject/sync/ValidationManagerImpl.java
@@ -26,6 +26,7 @@ import java.util.Map;
 import java.util.Queue;
 import java.util.concurrent.ConcurrentHashMap;
 import java.util.concurrent.Executor;
+import java.util.concurrent.atomic.AtomicBoolean;
 import java.util.logging.Logger;
 
 import javax.inject.Inject;
@@ -44,6 +45,7 @@ class ValidationManagerImpl implements ValidationManager, Service,
 	private final Executor cryptoExecutor;
 	private final Map<ClientId, MessageValidator> validators;
 	private final Map<ClientId, IncomingMessageHook> hooks;
+	private final AtomicBoolean used = new AtomicBoolean(false);
 
 	@Inject
 	ValidationManagerImpl(DatabaseComponent db,
@@ -58,6 +60,7 @@ class ValidationManagerImpl implements ValidationManager, Service,
 
 	@Override
 	public void startService() {
+		if (used.getAndSet(true)) throw new IllegalStateException();
 		for (ClientId c : validators.keySet()) getMessagesToValidate(c);
 	}
 
@@ -78,6 +81,7 @@ class ValidationManagerImpl implements ValidationManager, Service,
 
 	private void getMessagesToValidate(final ClientId c) {
 		dbExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				try {
 					Queue<MessageId> unvalidated = new LinkedList<MessageId>();
@@ -100,6 +104,7 @@ class ValidationManagerImpl implements ValidationManager, Service,
 	private void validateNextMessage(final Queue<MessageId> unvalidated) {
 		if (unvalidated.isEmpty()) return;
 		dbExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				try {
 					Message m = null;
@@ -141,6 +146,7 @@ class ValidationManagerImpl implements ValidationManager, Service,
 
 	private void validateMessage(final Message m, final Group g) {
 		cryptoExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				MessageValidator v = validators.get(g.getClientId());
 				if (v == null) {
@@ -156,6 +162,7 @@ class ValidationManagerImpl implements ValidationManager, Service,
 	private void storeValidationResult(final Message m, final ClientId c,
 			final Metadata meta) {
 		dbExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				try {
 					Transaction txn = db.startTransaction(false);
@@ -193,6 +200,7 @@ class ValidationManagerImpl implements ValidationManager, Service,
 
 	private void loadGroupAndValidate(final Message m) {
 		dbExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				try {
 					Group g;
diff --git a/briar-core/src/org/briarproject/transport/KeyManagerImpl.java b/briar-core/src/org/briarproject/transport/KeyManagerImpl.java
index fdcbe8f30c03506e47beacc44f218adcc095a9db..03b2c136669a0a444a0607e536a73551507555bc 100644
--- a/briar-core/src/org/briarproject/transport/KeyManagerImpl.java
+++ b/briar-core/src/org/briarproject/transport/KeyManagerImpl.java
@@ -28,6 +28,7 @@ import java.util.Map.Entry;
 import java.util.concurrent.ConcurrentHashMap;
 import java.util.concurrent.Executor;
 import java.util.concurrent.ScheduledExecutorService;
+import java.util.concurrent.atomic.AtomicBoolean;
 import java.util.logging.Logger;
 
 import javax.inject.Inject;
@@ -47,6 +48,7 @@ class KeyManagerImpl implements KeyManager, Service, EventListener {
 	private final Clock clock;
 	private final Map<ContactId, Boolean> activeContacts;
 	private final ConcurrentHashMap<TransportId, TransportKeyManager> managers;
+	private final AtomicBoolean used = new AtomicBoolean(false);
 
 	@Inject
 	KeyManagerImpl(DatabaseComponent db, CryptoComponent crypto,
@@ -66,6 +68,7 @@ class KeyManagerImpl implements KeyManager, Service, EventListener {
 
 	@Override
 	public void startService() throws ServiceException {
+		if (used.getAndSet(true)) throw new IllegalStateException();
 		Map<TransportId, Integer> transports =
 				new HashMap<TransportId, Integer>();
 		for (SimplexPluginFactory f : pluginConfig.getSimplexFactories())
diff --git a/briar-desktop/src/org/briarproject/plugins/bluetooth/BluetoothPlugin.java b/briar-desktop/src/org/briarproject/plugins/bluetooth/BluetoothPlugin.java
index 875112c5e88abe0baed8035d2a9d0da319000ec8..b8a0418c0773662fb79525bba3e523b60c832188 100644
--- a/briar-desktop/src/org/briarproject/plugins/bluetooth/BluetoothPlugin.java
+++ b/briar-desktop/src/org/briarproject/plugins/bluetooth/BluetoothPlugin.java
@@ -30,6 +30,7 @@ import java.util.concurrent.Executor;
 import java.util.concurrent.ExecutorCompletionService;
 import java.util.concurrent.Future;
 import java.util.concurrent.Semaphore;
+import java.util.concurrent.atomic.AtomicBoolean;
 import java.util.logging.Logger;
 
 import javax.bluetooth.BluetoothStateException;
@@ -62,6 +63,7 @@ class BluetoothPlugin implements DuplexPlugin {
 	private final DuplexPluginCallback callback;
 	private final int maxLatency;
 	private final Semaphore discoverySemaphore = new Semaphore(1);
+	private final AtomicBoolean used = new AtomicBoolean(false);
 
 	private volatile boolean running = false;
 	private volatile StreamConnectionNotifier socket = null;
@@ -76,20 +78,25 @@ class BluetoothPlugin implements DuplexPlugin {
 		this.maxLatency = maxLatency;
 	}
 
+	@Override
 	public TransportId getId() {
 		return ID;
 	}
 
+	@Override
 	public int getMaxLatency() {
 		return maxLatency;
 	}
 
+	@Override
 	public int getMaxIdleTime() {
 		// Bluetooth detects dead connections so we don't need keepalives
 		return Integer.MAX_VALUE;
 	}
 
+	@Override
 	public boolean start() throws IOException {
+		if (used.getAndSet(true)) throw new IllegalStateException();
 		// Initialise the Bluetooth stack
 		try {
 			localDevice = LocalDevice.getLocalDevice();
@@ -108,6 +115,7 @@ class BluetoothPlugin implements DuplexPlugin {
 
 	private void bind() {
 		ioExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				if (!running) return;
 				// Advertise the Bluetooth address to contacts
@@ -183,23 +191,28 @@ class BluetoothPlugin implements DuplexPlugin {
 		return new BluetoothTransportConnection(this, s);
 	}
 
+	@Override
 	public void stop() {
 		running = false;
 		tryToClose(socket);
 	}
 
+	@Override
 	public boolean isRunning() {
 		return running;
 	}
 
+	@Override
 	public boolean shouldPoll() {
 		return true;
 	}
 
+	@Override
 	public int getPollingInterval() {
 		return backoff.getPollingInterval();
 	}
 
+	@Override
 	public void poll(final Collection<ContactId> connected) {
 		if (!running) return;
 		backoff.increment();
@@ -214,6 +227,7 @@ class BluetoothPlugin implements DuplexPlugin {
 			final String uuid = e.getValue().get(PROP_UUID);
 			if (StringUtils.isNullOrEmpty(uuid)) continue;
 			ioExecutor.execute(new Runnable() {
+				@Override
 				public void run() {
 					if (!running) return;
 					StreamConnection s = connect(makeUrl(address, uuid));
@@ -238,6 +252,7 @@ class BluetoothPlugin implements DuplexPlugin {
 		}
 	}
 
+	@Override
 	public DuplexTransportConnection createConnection(ContactId c) {
 		if (!running) return null;
 		TransportProperties p = callback.getRemoteProperties().get(c);
@@ -252,10 +267,12 @@ class BluetoothPlugin implements DuplexPlugin {
 		return new BluetoothTransportConnection(this, s);
 	}
 
+	@Override
 	public boolean supportsInvitations() {
 		return true;
 	}
 
+	@Override
 	public DuplexTransportConnection createInvitationConnection(PseudoRandom r,
 			long timeout, boolean alice) {
 		if (!running) return null;
@@ -279,9 +296,8 @@ class BluetoothPlugin implements DuplexPlugin {
 		}
 		// Create the background tasks
 		CompletionService<StreamConnection> complete =
-				new ExecutorCompletionService<StreamConnection>(ioExecutor);
-		List<Future<StreamConnection>> futures =
-				new ArrayList<Future<StreamConnection>>();
+				new ExecutorCompletionService<>(ioExecutor);
+		List<Future<StreamConnection>> futures = new ArrayList<>();
 		if (alice) {
 			// Return the first connected socket
 			futures.add(complete.submit(new ListeningTask(ss)));
@@ -316,6 +332,7 @@ class BluetoothPlugin implements DuplexPlugin {
 	private void closeSockets(final List<Future<StreamConnection>> futures,
 			final StreamConnection chosen) {
 		ioExecutor.execute(new Runnable() {
+			@Override
 			public void run() {
 				for (Future<StreamConnection> f : futures) {
 					try {
@@ -331,9 +348,7 @@ class BluetoothPlugin implements DuplexPlugin {
 					} catch (InterruptedException e) {
 						LOG.info("Interrupted while closing sockets");
 						return;
-					} catch (ExecutionException e) {
-						if (LOG.isLoggable(INFO)) LOG.info(e.toString());
-					} catch (IOException e) {
+					} catch (ExecutionException | IOException e) {
 						if (LOG.isLoggable(INFO)) LOG.info(e.toString());
 					}
 				}
@@ -341,14 +356,15 @@ class BluetoothPlugin implements DuplexPlugin {
 		});
 	}
 
+	@Override
 	public boolean supportsKeyAgreement() {
 		return true;
 	}
 
-	public KeyAgreementListener createKeyAgreementListener(
-			byte[] localCommitment) {
+	@Override
+	public KeyAgreementListener createKeyAgreementListener(byte[] commitment) {
 		// No truncation necessary because COMMIT_LENGTH = 16
-		String uuid = UUID.nameUUIDFromBytes(localCommitment).toString();
+		String uuid = UUID.nameUUIDFromBytes(commitment).toString();
 		if (LOG.isLoggable(INFO)) LOG.info("Key agreement UUID " + uuid);
 		String url = makeUrl("localhost", uuid);
 		// Make the device discoverable if possible
@@ -371,8 +387,9 @@ class BluetoothPlugin implements DuplexPlugin {
 		return new BluetoothKeyAgreementListener(d, ss);
 	}
 
+	@Override
 	public DuplexTransportConnection createKeyAgreementConnection(
-			byte[] remoteCommitment, TransportDescriptor d, long timeout) {
+			byte[] commitment, TransportDescriptor d, long timeout) {
 		if (!isRunning()) return null;
 		if (!ID.equals(d.getIdentifier())) return null;
 		TransportProperties p = d.getProperties();
@@ -380,7 +397,7 @@ class BluetoothPlugin implements DuplexPlugin {
 		String address = p.get(PROP_ADDRESS);
 		if (StringUtils.isNullOrEmpty(address)) return null;
 		// No truncation necessary because COMMIT_LENGTH = 16
-		String uuid = UUID.nameUUIDFromBytes(remoteCommitment).toString();
+		String uuid = UUID.nameUUIDFromBytes(commitment).toString();
 		if (LOG.isLoggable(INFO))
 			LOG.info("Connecting to key agreement UUID " + uuid);
 		String url = makeUrl(address, uuid);
diff --git a/briar-desktop/src/org/briarproject/plugins/file/RemovableDrivePlugin.java b/briar-desktop/src/org/briarproject/plugins/file/RemovableDrivePlugin.java
index 116143225fde83a0762d2c77329c1002a35d1089..926f8d314800774f623b08688439bcc4dca36ca2 100644
--- a/briar-desktop/src/org/briarproject/plugins/file/RemovableDrivePlugin.java
+++ b/briar-desktop/src/org/briarproject/plugins/file/RemovableDrivePlugin.java
@@ -34,29 +34,36 @@ implements RemovableDriveMonitor.Callback {
 		this.monitor = monitor;
 	}
 
+	@Override
 	public TransportId getId() {
 		return ID;
 	}
 
+	@Override
 	public boolean start() throws IOException {
+		if (used.getAndSet(true)) throw new IllegalStateException();
 		running = true;
 		monitor.start(this);
 		return true;
 	}
 
+	@Override
 	public void stop() throws IOException {
 		running = false;
 		monitor.stop();
 	}
 
+	@Override
 	public boolean shouldPoll() {
 		return false;
 	}
 
+	@Override
 	public int getPollingInterval() {
 		throw new UnsupportedOperationException();
 	}
 
+	@Override
 	public void poll(Collection<ContactId> connected) {
 		throw new UnsupportedOperationException();
 	}
@@ -64,8 +71,7 @@ implements RemovableDriveMonitor.Callback {
 	@Override
 	protected File chooseOutputDirectory() {
 		try {
-			List<File> drives =
-					new ArrayList<File>(finder.findRemovableDrives());
+			List<File> drives = new ArrayList<>(finder.findRemovableDrives());
 			if (drives.isEmpty()) return null;
 			String[] paths = new String[drives.size()];
 			for (int i = 0; i < paths.length; i++) {
@@ -92,7 +98,7 @@ implements RemovableDriveMonitor.Callback {
 
 	@Override
 	protected Collection<File> findFilesByName(String filename) {
-		List<File> matches = new ArrayList<File>();
+		List<File> matches = new ArrayList<>();
 		try {
 			for (File drive : finder.findRemovableDrives()) {
 				File[] files = drive.listFiles();
@@ -109,6 +115,7 @@ implements RemovableDriveMonitor.Callback {
 		return Collections.unmodifiableList(matches);
 	}
 
+	@Override
 	public void driveInserted(File root) {
 		File[] files = root.listFiles();
 		if (files != null) {
@@ -116,6 +123,7 @@ implements RemovableDriveMonitor.Callback {
 		}
 	}
 
+	@Override
 	public void exceptionThrown(IOException e) {
 		if (LOG.isLoggable(WARNING)) LOG.log(WARNING, e.toString(), e);
 	}
diff --git a/briar-desktop/src/org/briarproject/plugins/modem/ModemPlugin.java b/briar-desktop/src/org/briarproject/plugins/modem/ModemPlugin.java
index 10a68aa56f909086c8c0f0ee5c7e7ba13a45556d..ce2d2d57e7c0cf5ed7e4c19c5de94b8aa6344f43 100644
--- a/briar-desktop/src/org/briarproject/plugins/modem/ModemPlugin.java
+++ b/briar-desktop/src/org/briarproject/plugins/modem/ModemPlugin.java
@@ -16,6 +16,7 @@ import java.io.IOException;
 import java.io.InputStream;
 import java.io.OutputStream;
 import java.util.Collection;
+import java.util.concurrent.atomic.AtomicBoolean;
 import java.util.logging.Logger;
 
 import static java.util.logging.Level.INFO;
@@ -32,6 +33,7 @@ class ModemPlugin implements DuplexPlugin, Modem.Callback {
 	private final SerialPortList serialPortList;
 	private final DuplexPluginCallback callback;
 	private final int maxLatency;
+	private final AtomicBoolean used = new AtomicBoolean(false);
 
 	private volatile boolean running = false;
 	private volatile Modem modem = null;
@@ -44,20 +46,25 @@ class ModemPlugin implements DuplexPlugin, Modem.Callback {
 		this.maxLatency = maxLatency;
 	}
 
+	@Override
 	public TransportId getId() {
 		return ID;
 	}
 
+	@Override
 	public int getMaxLatency() {
 		return maxLatency;
 	}
 
+	@Override
 	public int getMaxIdleTime() {
 		// FIXME: Do we need keepalives for this transport?
 		return Integer.MAX_VALUE;
 	}
 
+	@Override
 	public boolean start() {
+		if (used.getAndSet(true)) throw new IllegalStateException();
 		for (String portName : serialPortList.getPortNames()) {
 			if (LOG.isLoggable(INFO))
 				LOG.info("Trying to initialise modem on " + portName);
@@ -75,6 +82,7 @@ class ModemPlugin implements DuplexPlugin, Modem.Callback {
 		return false;
 	}
 
+	@Override
 	public void stop() {
 		running = false;
 		if (modem != null) {
@@ -86,18 +94,22 @@ class ModemPlugin implements DuplexPlugin, Modem.Callback {
 		}
 	}
 
+	@Override
 	public boolean isRunning() {
 		return running;
 	}
 
+	@Override
 	public boolean shouldPoll() {
 		return false;
 	}
 
+	@Override
 	public int getPollingInterval() {
 		throw new UnsupportedOperationException();
 	}
 
+	@Override
 	public void poll(Collection<ContactId> connected) {
 		throw new UnsupportedOperationException();
 	}
@@ -121,6 +133,7 @@ class ModemPlugin implements DuplexPlugin, Modem.Callback {
 		return false;
 	}
 
+	@Override
 	public DuplexTransportConnection createConnection(ContactId c) {
 		if (!running) return null;
 		// Get the ISO 3166 code for the caller's country
@@ -148,29 +161,34 @@ class ModemPlugin implements DuplexPlugin, Modem.Callback {
 		return new ModemTransportConnection();
 	}
 
+	@Override
 	public boolean supportsInvitations() {
 		return false;
 	}
 
+	@Override
 	public DuplexTransportConnection createInvitationConnection(PseudoRandom r,
 			long timeout, boolean alice) {
 		throw new UnsupportedOperationException();
 	}
 
+	@Override
 	public boolean supportsKeyAgreement() {
 		return false;
 	}
 
-	public KeyAgreementListener createKeyAgreementListener(
-			byte[] commitment) {
+	@Override
+	public KeyAgreementListener createKeyAgreementListener(byte[] commitment) {
 		throw new UnsupportedOperationException();
 	}
 
+	@Override
 	public DuplexTransportConnection createKeyAgreementConnection(
 			byte[] commitment, TransportDescriptor d, long timeout) {
 		throw new UnsupportedOperationException();
 	}
 
+	@Override
 	public void incomingCallConnected() {
 		LOG.info("Incoming call connected");
 		callback.incomingConnectionCreated(new ModemTransportConnection());