HBASE-16735 Procedure v2 - Fix yield while holding locks

This commit is contained in:
Matteo Bertozzi 2016-10-12 11:56:36 -07:00
parent a46134bffc
commit e868d9586f
3 changed files with 109 additions and 27 deletions

View File

@ -150,23 +150,17 @@ public class MasterProcedureScheduler implements ProcedureRunnableSet {
if (proc.isSuspended()) return; if (proc.isSuspended()) return;
queue.add(proc, addFront); queue.add(proc, addFront);
if (!(queue.isSuspended() || if (!queue.hasExclusiveLock() || queue.isLockOwner(proc.getProcId())) {
(queue.hasExclusiveLock() && !queue.isLockOwner(proc.getProcId())))) { // if the queue was not remove for an xlock execution
// the queue is not suspended or removed from the fairq (run-queue) // or the proc is the lock owner, put the queue back into execution
// because someone has an xlock on it. addToRunQueue(fairq, queue);
// so, if the queue is not-linked we should add it
if (queue.size() == 1 && !AvlIterableList.isLinked(queue)) {
fairq.add(queue);
}
} else if (queue.hasParentLock(proc)) { } else if (queue.hasParentLock(proc)) {
assert addFront : "expected to add a child in the front"; assert addFront : "expected to add a child in the front";
assert !queue.isSuspended() : "unexpected suspended state for the queue"; assert !queue.isSuspended() : "unexpected suspended state for the queue";
// our (proc) parent has the xlock, // our (proc) parent has the xlock,
// so the queue is not in the fairq (run-queue) // so the queue is not in the fairq (run-queue)
// add it back to let the child run (inherit the lock) // add it back to let the child run (inherit the lock)
if (!AvlIterableList.isLinked(queue)) { addToRunQueue(fairq, queue);
fairq.add(queue);
}
} }
} }
@ -348,16 +342,17 @@ public class MasterProcedureScheduler implements ProcedureRunnableSet {
} }
private <T extends Comparable<T>> void addToRunQueue(FairQueue<T> fairq, Queue<T> queue) { private <T extends Comparable<T>> void addToRunQueue(FairQueue<T> fairq, Queue<T> queue) {
if (AvlIterableList.isLinked(queue)) return; if (!AvlIterableList.isLinked(queue) &&
if (!queue.isEmpty()) { !queue.isEmpty() && !queue.isSuspended()) {
fairq.add(queue); fairq.add(queue);
} }
} }
private <T extends Comparable<T>> void removeFromRunQueue(FairQueue<T> fairq, Queue<T> queue) { private <T extends Comparable<T>> void removeFromRunQueue(FairQueue<T> fairq, Queue<T> queue) {
if (!AvlIterableList.isLinked(queue)) return; if (AvlIterableList.isLinked(queue)) {
fairq.remove(queue); fairq.remove(queue);
} }
}
// ============================================================================ // ============================================================================
// TODO: Metrics // TODO: Metrics
@ -924,6 +919,7 @@ public class MasterProcedureScheduler implements ProcedureRunnableSet {
case MERGE: case MERGE:
case ASSIGN: case ASSIGN:
case UNASSIGN: case UNASSIGN:
case REGION_EDIT:
return false; return false;
default: default:
break; break;
@ -1154,8 +1150,10 @@ public class MasterProcedureScheduler implements ProcedureRunnableSet {
// Zk lock is expensive... // Zk lock is expensive...
queue.releaseZkSharedLock(lockManager); queue.releaseZkSharedLock(lockManager);
queue.releaseSharedLock();
queue.getNamespaceQueue().releaseSharedLock(); queue.getNamespaceQueue().releaseSharedLock();
if (queue.releaseSharedLock()) {
addToRunQueue(tableRunQueue, queue);
}
schedLock.unlock(); schedLock.unlock();
} }
@ -1354,11 +1352,13 @@ public class MasterProcedureScheduler implements ProcedureRunnableSet {
public void releaseNamespaceExclusiveLock(final Procedure procedure, final String nsName) { public void releaseNamespaceExclusiveLock(final Procedure procedure, final String nsName) {
schedLock.lock(); schedLock.lock();
try { try {
TableQueue tableQueue = getTableQueue(TableName.NAMESPACE_TABLE_NAME); final TableQueue tableQueue = getTableQueue(TableName.NAMESPACE_TABLE_NAME);
tableQueue.releaseSharedLock(); final NamespaceQueue queue = getNamespaceQueue(nsName);
NamespaceQueue queue = getNamespaceQueue(nsName);
queue.releaseExclusiveLock(); queue.releaseExclusiveLock();
if (tableQueue.releaseSharedLock()) {
addToRunQueue(tableRunQueue, tableQueue);
}
} finally { } finally {
schedLock.unlock(); schedLock.unlock();
} }
@ -1503,8 +1503,8 @@ public class MasterProcedureScheduler implements ProcedureRunnableSet {
return true; return true;
} }
public synchronized void releaseSharedLock() { public synchronized boolean releaseSharedLock() {
sharedLock--; return --sharedLock == 0;
} }
protected synchronized boolean isSingleSharedLock() { protected synchronized boolean isSingleSharedLock() {

View File

@ -31,7 +31,7 @@ import org.apache.hadoop.hbase.classification.InterfaceStability;
public interface TableProcedureInterface { public interface TableProcedureInterface {
public enum TableOperationType { public enum TableOperationType {
CREATE, DELETE, DISABLE, EDIT, ENABLE, READ, CREATE, DELETE, DISABLE, EDIT, ENABLE, READ,
SPLIT, MERGE, ASSIGN, UNASSIGN, /* region operations */ REGION_EDIT, SPLIT, MERGE, ASSIGN, UNASSIGN, /* region operations */
}; };
/** /**

View File

@ -501,11 +501,11 @@ public class TestMasterProcedureScheduler {
// (this step is done by the executor/rootProc, we are simulating it) // (this step is done by the executor/rootProc, we are simulating it)
Procedure[] subProcs = new Procedure[] { Procedure[] subProcs = new Procedure[] {
new TestRegionProcedure(1, 2, tableName, new TestRegionProcedure(1, 2, tableName,
TableProcedureInterface.TableOperationType.ASSIGN, regionA), TableProcedureInterface.TableOperationType.REGION_EDIT, regionA),
new TestRegionProcedure(1, 3, tableName, new TestRegionProcedure(1, 3, tableName,
TableProcedureInterface.TableOperationType.ASSIGN, regionB), TableProcedureInterface.TableOperationType.REGION_EDIT, regionB),
new TestRegionProcedure(1, 4, tableName, new TestRegionProcedure(1, 4, tableName,
TableProcedureInterface.TableOperationType.ASSIGN, regionC), TableProcedureInterface.TableOperationType.REGION_EDIT, regionC),
}; };
// at this point the rootProc is going in a waiting state // at this point the rootProc is going in a waiting state
@ -678,6 +678,79 @@ public class TestMasterProcedureScheduler {
queue.releaseTableExclusiveLock(parentProc, tableName); queue.releaseTableExclusiveLock(parentProc, tableName);
} }
@Test
public void testYieldWithXLockHeld() throws Exception {
final TableName tableName = TableName.valueOf("testYieldWithXLockHeld");
queue.addBack(new TestTableProcedure(1, tableName,
TableProcedureInterface.TableOperationType.EDIT));
queue.addBack(new TestTableProcedure(2, tableName,
TableProcedureInterface.TableOperationType.EDIT));
// fetch from the queue and acquire xlock for the first proc
Procedure proc = queue.poll();
assertEquals(1, proc.getProcId());
assertEquals(true, queue.tryAcquireTableExclusiveLock(proc, tableName));
// nothing available, until xlock release
assertEquals(null, queue.poll(0));
// put the proc in the queue
queue.yield(proc);
// fetch from the queue, it should be the one with just added back
proc = queue.poll();
assertEquals(1, proc.getProcId());
// release the xlock
queue.releaseTableExclusiveLock(proc, tableName);
proc = queue.poll();
assertEquals(2, proc.getProcId());
}
@Test
public void testYieldWithSharedLockHeld() throws Exception {
final TableName tableName = TableName.valueOf("testYieldWithSharedLockHeld");
queue.addBack(new TestTableProcedure(1, tableName,
TableProcedureInterface.TableOperationType.READ));
queue.addBack(new TestTableProcedure(2, tableName,
TableProcedureInterface.TableOperationType.READ));
queue.addBack(new TestTableProcedure(3, tableName,
TableProcedureInterface.TableOperationType.EDIT));
// fetch and acquire the first shared-lock
Procedure proc1 = queue.poll();
assertEquals(1, proc1.getProcId());
assertEquals(true, queue.tryAcquireTableSharedLock(proc1, tableName));
// fetch and acquire the second shared-lock
Procedure proc2 = queue.poll();
assertEquals(2, proc2.getProcId());
assertEquals(true, queue.tryAcquireTableSharedLock(proc2, tableName));
// nothing available, until xlock release
assertEquals(null, queue.poll(0));
// put the procs back in the queue
queue.yield(proc2);
queue.yield(proc1);
// fetch from the queue, it should fetch the ones with just added back
proc1 = queue.poll();
assertEquals(1, proc1.getProcId());
proc2 = queue.poll();
assertEquals(2, proc2.getProcId());
// release the xlock
queue.releaseTableSharedLock(proc1, tableName);
queue.releaseTableSharedLock(proc2, tableName);
Procedure proc3 = queue.poll();
assertEquals(3, proc3.getProcId());
}
public static class TestTableProcedure extends TestProcedure public static class TestTableProcedure extends TestProcedure
implements TableProcedureInterface { implements TableProcedureInterface {
private final TableOperationType opType; private final TableOperationType opType;
@ -710,7 +783,7 @@ public class TestMasterProcedureScheduler {
@Override @Override
public void toStringClassDetails(final StringBuilder sb) { public void toStringClassDetails(final StringBuilder sb) {
sb.append(getClass().getSimpleName()); sb.append(getClass().getSimpleName());
sb.append(" (table="); sb.append("(table=");
sb.append(getTableName()); sb.append(getTableName());
sb.append(")"); sb.append(")");
} }
@ -754,7 +827,7 @@ public class TestMasterProcedureScheduler {
@Override @Override
public void toStringClassDetails(final StringBuilder sb) { public void toStringClassDetails(final StringBuilder sb) {
sb.append(getClass().getSimpleName()); sb.append(getClass().getSimpleName());
sb.append(" (region="); sb.append("(regions=");
sb.append(Arrays.toString(getRegionInfo())); sb.append(Arrays.toString(getRegionInfo()));
sb.append(")"); sb.append(")");
} }
@ -784,5 +857,14 @@ public class TestMasterProcedureScheduler {
public TableOperationType getTableOperationType() { public TableOperationType getTableOperationType() {
return opType; return opType;
} }
@Override
public void toStringClassDetails(final StringBuilder sb) {
sb.append(getClass().getSimpleName());
sb.append("(ns=");
sb.append(nsName);
sb.append(")");
}
} }
} }