HDFS-16016. BPServiceActor to provide new thread to handle IBR (#2998)
Contributed by Viraj Jasani
(cherry picked from commit c1bf3cb0da
)
This commit is contained in:
parent
198bc444de
commit
36a0e818ec
|
@ -36,6 +36,8 @@ import java.util.SortedSet;
|
|||
import java.util.TreeSet;
|
||||
import java.util.concurrent.BlockingQueue;
|
||||
import java.util.concurrent.CountDownLatch;
|
||||
import java.util.concurrent.ExecutorService;
|
||||
import java.util.concurrent.Executors;
|
||||
import java.util.concurrent.LinkedBlockingQueue;
|
||||
import java.util.concurrent.ThreadLocalRandom;
|
||||
import java.util.concurrent.atomic.AtomicBoolean;
|
||||
|
@ -69,6 +71,7 @@ import org.apache.hadoop.hdfs.server.protocol.VolumeFailureSummary;
|
|||
import org.apache.hadoop.io.IOUtils;
|
||||
import org.apache.hadoop.ipc.RemoteException;
|
||||
import org.apache.hadoop.net.NetUtils;
|
||||
import org.apache.hadoop.thirdparty.com.google.common.util.concurrent.ThreadFactoryBuilder;
|
||||
import org.apache.hadoop.util.Preconditions;
|
||||
import org.apache.hadoop.util.Time;
|
||||
import org.apache.hadoop.util.VersionInfo;
|
||||
|
@ -98,6 +101,8 @@ class BPServiceActor implements Runnable {
|
|||
|
||||
volatile long lastCacheReport = 0;
|
||||
private final Scheduler scheduler;
|
||||
private final Object sendIBRLock;
|
||||
private final ExecutorService ibrExecutorService;
|
||||
|
||||
Thread bpThread;
|
||||
DatanodeProtocolClientSideTranslatorPB bpNamenode;
|
||||
|
@ -153,6 +158,10 @@ class BPServiceActor implements Runnable {
|
|||
}
|
||||
commandProcessingThread = new CommandProcessingThread(this);
|
||||
commandProcessingThread.start();
|
||||
sendIBRLock = new Object();
|
||||
ibrExecutorService = Executors.newSingleThreadExecutor(
|
||||
new ThreadFactoryBuilder().setDaemon(true)
|
||||
.setNameFormat("ibr-executor-%d").build());
|
||||
}
|
||||
|
||||
public DatanodeRegistration getBpRegistration() {
|
||||
|
@ -372,8 +381,10 @@ class BPServiceActor implements Runnable {
|
|||
// we have a chance that we will miss the delHint information
|
||||
// or we will report an RBW replica after the BlockReport already reports
|
||||
// a FINALIZED one.
|
||||
synchronized (sendIBRLock) {
|
||||
ibrManager.sendIBRs(bpNamenode, bpRegistration,
|
||||
bpos.getBlockPoolId(), getRpcMetricSuffix());
|
||||
}
|
||||
|
||||
long brCreateStartTime = monotonicNow();
|
||||
Map<DatanodeStorage, BlockListAsLongs> perVolumeBlockLists =
|
||||
|
@ -604,6 +615,9 @@ class BPServiceActor implements Runnable {
|
|||
if (commandProcessingThread != null) {
|
||||
commandProcessingThread.interrupt();
|
||||
}
|
||||
if (ibrExecutorService != null && !ibrExecutorService.isShutdown()) {
|
||||
ibrExecutorService.shutdownNow();
|
||||
}
|
||||
}
|
||||
|
||||
//This must be called only by blockPoolManager
|
||||
|
@ -618,13 +632,18 @@ class BPServiceActor implements Runnable {
|
|||
} catch (InterruptedException ie) { }
|
||||
}
|
||||
|
||||
//Cleanup method to be called by current thread before exiting.
|
||||
// Cleanup method to be called by current thread before exiting.
|
||||
// Any Thread / ExecutorService started by BPServiceActor can be shutdown
|
||||
// here.
|
||||
private synchronized void cleanUp() {
|
||||
|
||||
shouldServiceRun = false;
|
||||
IOUtils.cleanupWithLogger(null, bpNamenode);
|
||||
IOUtils.cleanupWithLogger(null, lifelineSender);
|
||||
bpos.shutdownActor(this);
|
||||
if (!ibrExecutorService.isShutdown()) {
|
||||
ibrExecutorService.shutdownNow();
|
||||
}
|
||||
}
|
||||
|
||||
private void handleRollingUpgradeStatus(HeartbeatResponse resp) throws IOException {
|
||||
|
@ -710,11 +729,6 @@ class BPServiceActor implements Runnable {
|
|||
commandProcessingThread.enqueue(resp.getCommands());
|
||||
}
|
||||
}
|
||||
if (!dn.areIBRDisabledForTests() &&
|
||||
(ibrManager.sendImmediately()|| sendHeartbeat)) {
|
||||
ibrManager.sendIBRs(bpNamenode, bpRegistration,
|
||||
bpos.getBlockPoolId(), getRpcMetricSuffix());
|
||||
}
|
||||
|
||||
List<DatanodeCommand> cmds = null;
|
||||
boolean forceFullBr =
|
||||
|
@ -878,6 +892,10 @@ class BPServiceActor implements Runnable {
|
|||
initialRegistrationComplete.countDown();
|
||||
}
|
||||
|
||||
// IBR tasks to be handled separately from offerService() in order to
|
||||
// improve performance of offerService(), which can now focus only on
|
||||
// FBR and heartbeat.
|
||||
ibrExecutorService.submit(new IBRTaskHandler());
|
||||
while (shouldRun()) {
|
||||
try {
|
||||
offerService();
|
||||
|
@ -1108,6 +1126,34 @@ class BPServiceActor implements Runnable {
|
|||
}
|
||||
}
|
||||
|
||||
class IBRTaskHandler implements Runnable {
|
||||
|
||||
@Override
|
||||
public void run() {
|
||||
LOG.info("Starting IBR Task Handler.");
|
||||
while (shouldRun()) {
|
||||
try {
|
||||
final long startTime = scheduler.monotonicNow();
|
||||
final boolean sendHeartbeat = scheduler.isHeartbeatDue(startTime);
|
||||
if (!dn.areIBRDisabledForTests() &&
|
||||
(ibrManager.sendImmediately() || sendHeartbeat)) {
|
||||
synchronized (sendIBRLock) {
|
||||
ibrManager.sendIBRs(bpNamenode, bpRegistration,
|
||||
bpos.getBlockPoolId(), getRpcMetricSuffix());
|
||||
}
|
||||
}
|
||||
// There is no work to do; sleep until heartbeat timer elapses,
|
||||
// or work arrives, and then iterate again.
|
||||
ibrManager.waitTillNextIBR(scheduler.getHeartbeatWaitTime());
|
||||
} catch (Throwable t) {
|
||||
LOG.error("Exception in IBRTaskHandler.", t);
|
||||
sleepAndLogInterrupts(5000, "offering IBR service");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Utility class that wraps the timestamp computations for scheduling
|
||||
* heartbeats and block reports.
|
||||
|
|
|
@ -172,8 +172,19 @@ public class TestDatanodeReport {
|
|||
// all bad datanodes
|
||||
}
|
||||
cluster.triggerHeartbeats(); // IBR delete ack
|
||||
int retries = 0;
|
||||
while (true) {
|
||||
lb = fs.getClient().getLocatedBlocks(p.toString(), 0).get(0);
|
||||
assertEquals(0, lb.getLocations().length);
|
||||
if (0 != lb.getLocations().length) {
|
||||
retries++;
|
||||
if (retries > 7) {
|
||||
Assert.fail("getLocatedBlocks failed after 7 retries");
|
||||
}
|
||||
Thread.sleep(2000);
|
||||
} else {
|
||||
break;
|
||||
}
|
||||
}
|
||||
} finally {
|
||||
cluster.shutdown();
|
||||
}
|
||||
|
|
|
@ -25,6 +25,7 @@ import static org.mockito.Mockito.times;
|
|||
|
||||
import java.io.IOException;
|
||||
|
||||
import org.mockito.exceptions.base.MockitoAssertionError;
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
import org.apache.hadoop.conf.Configuration;
|
||||
|
@ -156,7 +157,7 @@ public class TestIncrementalBlockReports {
|
|||
|
||||
// Sleep for a very short time since IBR is generated
|
||||
// asynchronously.
|
||||
Thread.sleep(2000);
|
||||
Thread.sleep(1000);
|
||||
|
||||
// Ensure that no block report was generated immediately.
|
||||
// Deleted blocks are reported when the IBR timer elapses.
|
||||
|
@ -167,13 +168,24 @@ public class TestIncrementalBlockReports {
|
|||
|
||||
// Trigger a heartbeat, this also triggers an IBR.
|
||||
DataNodeTestUtils.triggerHeartbeat(singletonDn);
|
||||
Thread.sleep(2000);
|
||||
|
||||
// Ensure that the deleted block is reported.
|
||||
Mockito.verify(nnSpy, times(1)).blockReceivedAndDeleted(
|
||||
int retries = 0;
|
||||
while (true) {
|
||||
try {
|
||||
Mockito.verify(nnSpy, atLeastOnce()).blockReceivedAndDeleted(
|
||||
any(DatanodeRegistration.class),
|
||||
anyString(),
|
||||
any(StorageReceivedDeletedBlocks[].class));
|
||||
break;
|
||||
} catch (MockitoAssertionError e) {
|
||||
if (retries > 7) {
|
||||
throw e;
|
||||
}
|
||||
retries++;
|
||||
Thread.sleep(2000);
|
||||
}
|
||||
}
|
||||
|
||||
} finally {
|
||||
cluster.shutdown();
|
||||
|
|
Loading…
Reference in New Issue