HBASE-27229 BucketCache statistics should not count evictions by hfile (#4639)

Signed-off-by: Andrew Purtell <apurtell@apache.org>
This commit is contained in:
Bryan Beaudreault 2022-07-26 16:59:57 -04:00
parent 88f957b323
commit 25136c39f9
3 changed files with 71 additions and 17 deletions

View File

@ -555,13 +555,16 @@ public class BucketCache implements BlockCache, HeapSize {
/** /**
* This method is invoked after the bucketEntry is removed from {@link BucketCache#backingMap} * This method is invoked after the bucketEntry is removed from {@link BucketCache#backingMap}
*/ */
void blockEvicted(BlockCacheKey cacheKey, BucketEntry bucketEntry, boolean decrementBlockNumber) { void blockEvicted(BlockCacheKey cacheKey, BucketEntry bucketEntry, boolean decrementBlockNumber,
boolean evictedByEvictionProcess) {
bucketEntry.markAsEvicted(); bucketEntry.markAsEvicted();
blocksByHFile.remove(cacheKey); blocksByHFile.remove(cacheKey);
if (decrementBlockNumber) { if (decrementBlockNumber) {
this.blockNumber.decrement(); this.blockNumber.decrement();
} }
cacheStats.evicted(bucketEntry.getCachedTime(), cacheKey.isPrimary()); if (evictedByEvictionProcess) {
cacheStats.evicted(bucketEntry.getCachedTime(), cacheKey.isPrimary());
}
} }
/** /**
@ -591,7 +594,7 @@ public class BucketCache implements BlockCache, HeapSize {
*/ */
@Override @Override
public boolean evictBlock(BlockCacheKey cacheKey) { public boolean evictBlock(BlockCacheKey cacheKey) {
return doEvictBlock(cacheKey, null); return doEvictBlock(cacheKey, null, false);
} }
/** /**
@ -603,7 +606,8 @@ public class BucketCache implements BlockCache, HeapSize {
* @param bucketEntry {@link BucketEntry} matched {@link BlockCacheKey} to evict. * @param bucketEntry {@link BucketEntry} matched {@link BlockCacheKey} to evict.
* @return true to indicate whether we've evicted successfully or not. * @return true to indicate whether we've evicted successfully or not.
*/ */
private boolean doEvictBlock(BlockCacheKey cacheKey, BucketEntry bucketEntry) { private boolean doEvictBlock(BlockCacheKey cacheKey, BucketEntry bucketEntry,
boolean evictedByEvictionProcess) {
if (!cacheEnabled) { if (!cacheEnabled) {
return false; return false;
} }
@ -614,14 +618,14 @@ public class BucketCache implements BlockCache, HeapSize {
final BucketEntry bucketEntryToUse = bucketEntry; final BucketEntry bucketEntryToUse = bucketEntry;
if (bucketEntryToUse == null) { if (bucketEntryToUse == null) {
if (existedInRamCache) { if (existedInRamCache && evictedByEvictionProcess) {
cacheStats.evicted(0, cacheKey.isPrimary()); cacheStats.evicted(0, cacheKey.isPrimary());
} }
return existedInRamCache; return existedInRamCache;
} else { } else {
return bucketEntryToUse.withWriteLock(offsetLock, () -> { return bucketEntryToUse.withWriteLock(offsetLock, () -> {
if (backingMap.remove(cacheKey, bucketEntryToUse)) { if (backingMap.remove(cacheKey, bucketEntryToUse)) {
blockEvicted(cacheKey, bucketEntryToUse, !existedInRamCache); blockEvicted(cacheKey, bucketEntryToUse, !existedInRamCache, evictedByEvictionProcess);
return true; return true;
} }
return false; return false;
@ -671,7 +675,7 @@ public class BucketCache implements BlockCache, HeapSize {
*/ */
boolean evictBucketEntryIfNoRpcReferenced(BlockCacheKey blockCacheKey, BucketEntry bucketEntry) { boolean evictBucketEntryIfNoRpcReferenced(BlockCacheKey blockCacheKey, BucketEntry bucketEntry) {
if (!bucketEntry.isRpcRef()) { if (!bucketEntry.isRpcRef()) {
return doEvictBlock(blockCacheKey, bucketEntry); return doEvictBlock(blockCacheKey, bucketEntry, true);
} }
return false; return false;
} }
@ -792,7 +796,7 @@ public class BucketCache implements BlockCache, HeapSize {
* blocks evicted * blocks evicted
* @param why Why we are being called * @param why Why we are being called
*/ */
private void freeSpace(final String why) { void freeSpace(final String why) {
// Ensure only one freeSpace progress at a time // Ensure only one freeSpace progress at a time
if (!freeSpaceLock.tryLock()) { if (!freeSpaceLock.tryLock()) {
return; return;
@ -986,7 +990,7 @@ public class BucketCache implements BlockCache, HeapSize {
BucketEntry previousEntry = backingMap.put(key, bucketEntry); BucketEntry previousEntry = backingMap.put(key, bucketEntry);
if (previousEntry != null && previousEntry != bucketEntry) { if (previousEntry != null && previousEntry != bucketEntry) {
previousEntry.withWriteLock(offsetLock, () -> { previousEntry.withWriteLock(offsetLock, () -> {
blockEvicted(key, previousEntry, false); blockEvicted(key, previousEntry, false, false);
return null; return null;
}); });
} }
@ -1137,7 +1141,7 @@ public class BucketCache implements BlockCache, HeapSize {
final BucketEntry bucketEntry = bucketEntries[i]; final BucketEntry bucketEntry = bucketEntries[i];
bucketEntry.withWriteLock(offsetLock, () -> { bucketEntry.withWriteLock(offsetLock, () -> {
if (backingMap.remove(key, bucketEntry)) { if (backingMap.remove(key, bucketEntry)) {
blockEvicted(key, bucketEntry, false); blockEvicted(key, bucketEntry, false, false);
} }
return null; return null;
}); });

View File

@ -263,7 +263,7 @@ public class TestBucketCache {
}; };
evictThread.start(); evictThread.start();
cache.offsetLock.waitForWaiters(lockId, 1); cache.offsetLock.waitForWaiters(lockId, 1);
cache.blockEvicted(cacheKey, cache.backingMap.remove(cacheKey), true); cache.blockEvicted(cacheKey, cache.backingMap.remove(cacheKey), true, true);
assertEquals(0, cache.getBlockCount()); assertEquals(0, cache.getBlockCount());
cacheAndWaitUntilFlushedToBucket(cache, cacheKey, cacheAndWaitUntilFlushedToBucket(cache, cacheKey,
new CacheTestUtils.ByteArrayCacheable(new byte[10])); new CacheTestUtils.ByteArrayCacheable(new byte[10]));
@ -481,6 +481,56 @@ public class TestBucketCache {
assertEquals(testValue, bucketEntry.offset()); assertEquals(testValue, bucketEntry.offset());
} }
@Test
public void testEvictionCount() throws InterruptedException {
int size = 100;
int length = HConstants.HFILEBLOCK_HEADER_SIZE + size;
ByteBuffer buf1 = ByteBuffer.allocate(size), buf2 = ByteBuffer.allocate(size);
HFileContext meta = new HFileContextBuilder().build();
ByteBuffAllocator allocator = ByteBuffAllocator.HEAP;
HFileBlock blockWithNextBlockMetadata = new HFileBlock(BlockType.DATA, size, size, -1,
ByteBuff.wrap(buf1), HFileBlock.FILL_HEADER, -1, 52, -1, meta, allocator);
HFileBlock blockWithoutNextBlockMetadata = new HFileBlock(BlockType.DATA, size, size, -1,
ByteBuff.wrap(buf2), HFileBlock.FILL_HEADER, -1, -1, -1, meta, allocator);
BlockCacheKey key = new BlockCacheKey("testEvictionCount", 0);
ByteBuffer actualBuffer = ByteBuffer.allocate(length);
ByteBuffer block1Buffer = ByteBuffer.allocate(length);
ByteBuffer block2Buffer = ByteBuffer.allocate(length);
blockWithNextBlockMetadata.serialize(block1Buffer, true);
blockWithoutNextBlockMetadata.serialize(block2Buffer, true);
// Add blockWithNextBlockMetadata, expect blockWithNextBlockMetadata back.
CacheTestUtils.getBlockAndAssertEquals(cache, key, blockWithNextBlockMetadata, actualBuffer,
block1Buffer);
waitUntilFlushedToBucket(cache, key);
assertEquals(0, cache.getStats().getEvictionCount());
// evict call should return 1, but then eviction count be 0
assertEquals(1, cache.evictBlocksByHfileName("testEvictionCount"));
assertEquals(0, cache.getStats().getEvictionCount());
// add back
CacheTestUtils.getBlockAndAssertEquals(cache, key, blockWithNextBlockMetadata, actualBuffer,
block1Buffer);
waitUntilFlushedToBucket(cache, key);
// should not increment
assertTrue(cache.evictBlock(key));
assertEquals(0, cache.getStats().getEvictionCount());
// add back
CacheTestUtils.getBlockAndAssertEquals(cache, key, blockWithNextBlockMetadata, actualBuffer,
block1Buffer);
waitUntilFlushedToBucket(cache, key);
// should finally increment eviction count
cache.freeSpace("testing");
assertEquals(1, cache.getStats().getEvictionCount());
}
@Test @Test
public void testCacheBlockNextBlockMetadataMissing() throws Exception { public void testCacheBlockNextBlockMetadataMissing() throws Exception {
int size = 100; int size = 100;

View File

@ -557,10 +557,10 @@ public class TestBucketCacheRefCnt {
} }
@Override @Override
void blockEvicted(BlockCacheKey cacheKey, BucketEntry bucketEntry, void blockEvicted(BlockCacheKey cacheKey, BucketEntry bucketEntry, boolean decrementBlockNumber,
boolean decrementBlockNumber) { boolean evictedByEvictionProcess) {
blockEvictCounter.incrementAndGet(); blockEvictCounter.incrementAndGet();
super.blockEvicted(cacheKey, bucketEntry, decrementBlockNumber); super.blockEvicted(cacheKey, bucketEntry, decrementBlockNumber, evictedByEvictionProcess);
} }
/** /**
@ -709,8 +709,8 @@ public class TestBucketCacheRefCnt {
} }
@Override @Override
void blockEvicted(BlockCacheKey cacheKey, BucketEntry bucketEntry, void blockEvicted(BlockCacheKey cacheKey, BucketEntry bucketEntry, boolean decrementBlockNumber,
boolean decrementBlockNumber) { boolean evictedByEvictionProcess) {
/** /**
* This is only invoked by {@link BucketCache.WriterThread}. {@link MyMyBucketCache2} create * This is only invoked by {@link BucketCache.WriterThread}. {@link MyMyBucketCache2} create
* only one {@link BucketCache.WriterThread}. * only one {@link BucketCache.WriterThread}.
@ -718,7 +718,7 @@ public class TestBucketCacheRefCnt {
assertTrue(Thread.currentThread() == this.writerThreads[0]); assertTrue(Thread.currentThread() == this.writerThreads[0]);
blockEvictCounter.incrementAndGet(); blockEvictCounter.incrementAndGet();
super.blockEvicted(cacheKey, bucketEntry, decrementBlockNumber); super.blockEvicted(cacheKey, bucketEntry, decrementBlockNumber, evictedByEvictionProcess);
} }
/** /**