HBASE-14793 Allow limiting size of block into L1 block cache.
This commit is contained in:
parent
f8ee447c55
commit
604c9b2cca
|
@ -227,6 +227,9 @@ public interface MetricsRegionServerSource extends BaseSource {
|
|||
String BLOCK_CACHE_EXPRESS_HIT_PERCENT = "blockCacheExpressHitPercent";
|
||||
String BLOCK_CACHE_EXPRESS_HIT_PERCENT_DESC =
|
||||
"The percent of the time that requests with the cache turned on hit the cache.";
|
||||
String BLOCK_CACHE_FAILED_INSERTION_COUNT = "blockCacheFailedInsertionCount";
|
||||
String BLOCK_CACHE_FAILED_INSERTION_COUNT_DESC = "Number of times that a block cache " +
|
||||
"insertion failed. Usually due to size restrictions.";
|
||||
String RS_START_TIME_NAME = "regionServerStartTime";
|
||||
String ZOOKEEPER_QUORUM_NAME = "zookeeperQuorum";
|
||||
String SERVER_NAME_NAME = "serverName";
|
||||
|
|
|
@ -233,6 +233,11 @@ public interface MetricsRegionServerWrapper {
|
|||
*/
|
||||
double getBlockCacheHitCachingPercent();
|
||||
|
||||
/**
|
||||
* Number of cache insertions that failed.
|
||||
*/
|
||||
long getBlockCacheFailedInsertions();
|
||||
|
||||
/**
|
||||
* Force a re-computation of the metrics.
|
||||
*/
|
||||
|
|
|
@ -35,6 +35,7 @@ import org.apache.hadoop.metrics2.lib.MutableCounterLong;
|
|||
public class MetricsRegionServerSourceImpl
|
||||
extends BaseSourceImpl implements MetricsRegionServerSource {
|
||||
|
||||
|
||||
final MetricsRegionServerWrapper rsWrap;
|
||||
private final MetricHistogram putHisto;
|
||||
private final MetricHistogram deleteHisto;
|
||||
|
@ -250,6 +251,8 @@ public class MetricsRegionServerSourceImpl
|
|||
rsWrap.getBlockCacheHitPercent())
|
||||
.addGauge(Interns.info(BLOCK_CACHE_EXPRESS_HIT_PERCENT,
|
||||
BLOCK_CACHE_EXPRESS_HIT_PERCENT_DESC), rsWrap.getBlockCacheHitCachingPercent())
|
||||
.addCounter(Interns.info(BLOCK_CACHE_FAILED_INSERTION_COUNT,
|
||||
BLOCK_CACHE_FAILED_INSERTION_COUNT_DESC),rsWrap.getBlockCacheFailedInsertions())
|
||||
.addCounter(Interns.info(UPDATES_BLOCKED_TIME, UPDATES_BLOCKED_DESC),
|
||||
rsWrap.getUpdatesBlockedTime())
|
||||
.addCounter(Interns.info(FLUSHED_CELLS, FLUSHED_CELLS_DESC),
|
||||
|
|
|
@ -74,6 +74,9 @@ public class CacheStats {
|
|||
/** The total number of blocks for primary replica that have been evicted */
|
||||
private final AtomicLong primaryEvictedBlockCount = new AtomicLong(0);
|
||||
|
||||
/** The total number of blocks that were not inserted. */
|
||||
private final AtomicLong failedInserts = new AtomicLong(0);
|
||||
|
||||
/** The number of metrics periods to include in window */
|
||||
private final int numPeriodsInWindow;
|
||||
/** Hit counts for each period in window */
|
||||
|
@ -154,6 +157,10 @@ public class CacheStats {
|
|||
}
|
||||
}
|
||||
|
||||
public long failInsert() {
|
||||
return failedInserts.incrementAndGet();
|
||||
}
|
||||
|
||||
public long getRequestCount() {
|
||||
return getHitCount() + getMissCount();
|
||||
}
|
||||
|
@ -218,6 +225,10 @@ public class CacheStats {
|
|||
return ((float)getEvictedCount()/(float)getEvictionCount());
|
||||
}
|
||||
|
||||
public long getFailedInserts() {
|
||||
return failedInserts.get();
|
||||
}
|
||||
|
||||
public void rollMetricsPeriod() {
|
||||
hitCounts[windowIndex] = getHitCount() - lastHitCount;
|
||||
lastHitCount = getHitCount();
|
||||
|
|
|
@ -142,12 +142,15 @@ public class LruBlockCache implements ResizableBlockCache, HeapSize {
|
|||
|
||||
/** Statistics thread */
|
||||
static final int statThreadPeriod = 60 * 5;
|
||||
private static final String LRU_MAX_BLOCK_SIZE = "hbase.lru.max.block.size";
|
||||
private static final long DEFAULT_MAX_BLOCK_SIZE = 16L * 1024L * 1024L;
|
||||
|
||||
/** Concurrent map (the cache) */
|
||||
private final Map<BlockCacheKey,LruCachedBlock> map;
|
||||
|
||||
/** Eviction lock (locked when eviction in process) */
|
||||
private final ReentrantLock evictionLock = new ReentrantLock(true);
|
||||
private final long maxBlockSize;
|
||||
|
||||
/** Volatile boolean to track if we are in an eviction process or not */
|
||||
private volatile boolean evictionInProgress = false;
|
||||
|
@ -225,7 +228,8 @@ public class LruBlockCache implements ResizableBlockCache, HeapSize {
|
|||
DEFAULT_SINGLE_FACTOR,
|
||||
DEFAULT_MULTI_FACTOR,
|
||||
DEFAULT_MEMORY_FACTOR,
|
||||
false
|
||||
false,
|
||||
DEFAULT_MAX_BLOCK_SIZE
|
||||
);
|
||||
}
|
||||
|
||||
|
@ -239,7 +243,8 @@ public class LruBlockCache implements ResizableBlockCache, HeapSize {
|
|||
conf.getFloat(LRU_SINGLE_PERCENTAGE_CONFIG_NAME, DEFAULT_SINGLE_FACTOR),
|
||||
conf.getFloat(LRU_MULTI_PERCENTAGE_CONFIG_NAME, DEFAULT_MULTI_FACTOR),
|
||||
conf.getFloat(LRU_MEMORY_PERCENTAGE_CONFIG_NAME, DEFAULT_MEMORY_FACTOR),
|
||||
conf.getBoolean(LRU_IN_MEMORY_FORCE_MODE_CONFIG_NAME, DEFAULT_IN_MEMORY_FORCE_MODE)
|
||||
conf.getBoolean(LRU_IN_MEMORY_FORCE_MODE_CONFIG_NAME, DEFAULT_IN_MEMORY_FORCE_MODE),
|
||||
conf.getLong(LRU_MAX_BLOCK_SIZE, DEFAULT_MAX_BLOCK_SIZE)
|
||||
);
|
||||
}
|
||||
|
||||
|
@ -264,7 +269,8 @@ public class LruBlockCache implements ResizableBlockCache, HeapSize {
|
|||
public LruBlockCache(long maxSize, long blockSize, boolean evictionThread,
|
||||
int mapInitialSize, float mapLoadFactor, int mapConcurrencyLevel,
|
||||
float minFactor, float acceptableFactor, float singleFactor,
|
||||
float multiFactor, float memoryFactor, boolean forceInMemory) {
|
||||
float multiFactor, float memoryFactor, boolean forceInMemory, long maxBlockSize) {
|
||||
this.maxBlockSize = maxBlockSize;
|
||||
if(singleFactor + multiFactor + memoryFactor != 1 ||
|
||||
singleFactor < 0 || multiFactor < 0 || memoryFactor < 0) {
|
||||
throw new IllegalArgumentException("Single, multi, and memory factors " +
|
||||
|
@ -326,6 +332,21 @@ public class LruBlockCache implements ResizableBlockCache, HeapSize {
|
|||
@Override
|
||||
public void cacheBlock(BlockCacheKey cacheKey, Cacheable buf, boolean inMemory,
|
||||
final boolean cacheDataInL1) {
|
||||
|
||||
if (buf.heapSize() > maxBlockSize) {
|
||||
// If there are a lot of blocks that are too
|
||||
// big this can make the logs way too noisy.
|
||||
// So we log 2%
|
||||
if (stats.failInsert() % 50 == 0) {
|
||||
LOG.warn("Trying to cache too large a block "
|
||||
+ cacheKey.getHfileName() + " @ "
|
||||
+ cacheKey.getOffset()
|
||||
+ " is " + buf.heapSize()
|
||||
+ " which is larger than " + maxBlockSize);
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
LruCachedBlock cb = map.get(cacheKey);
|
||||
if (cb != null) {
|
||||
// compare the contents, if they are not equal, we are in big trouble
|
||||
|
@ -883,8 +904,8 @@ public class LruBlockCache implements ResizableBlockCache, HeapSize {
|
|||
}
|
||||
|
||||
public final static long CACHE_FIXED_OVERHEAD = ClassSize.align(
|
||||
(3 * Bytes.SIZEOF_LONG) + (9 * ClassSize.REFERENCE) +
|
||||
(5 * Bytes.SIZEOF_FLOAT) + Bytes.SIZEOF_BOOLEAN
|
||||
(3 * Bytes.SIZEOF_LONG) + (10 * ClassSize.REFERENCE) +
|
||||
(5 * Bytes.SIZEOF_FLOAT) + (2 * Bytes.SIZEOF_BOOLEAN)
|
||||
+ ClassSize.OBJECT);
|
||||
|
||||
@Override
|
||||
|
|
|
@ -150,7 +150,6 @@ public class BucketCache implements BlockCache, HeapSize {
|
|||
private final AtomicLong heapSize = new AtomicLong(0);
|
||||
/** Current number of cached elements */
|
||||
private final AtomicLong blockNumber = new AtomicLong(0);
|
||||
private final AtomicLong failedBlockAdditions = new AtomicLong(0);
|
||||
|
||||
/** Cache access count (sequential ID) */
|
||||
private final AtomicLong accessCount = new AtomicLong(0);
|
||||
|
@ -374,7 +373,7 @@ public class BucketCache implements BlockCache, HeapSize {
|
|||
}
|
||||
if (!successfulAddition) {
|
||||
ramCache.remove(cacheKey);
|
||||
failedBlockAdditions.incrementAndGet();
|
||||
cacheStats.failInsert();
|
||||
} else {
|
||||
this.blockNumber.incrementAndGet();
|
||||
this.heapSize.addAndGet(cachedItem.heapSize());
|
||||
|
@ -514,7 +513,7 @@ public class BucketCache implements BlockCache, HeapSize {
|
|||
long usedSize = bucketAllocator.getUsedSize();
|
||||
long freeSize = totalSize - usedSize;
|
||||
long cacheSize = getRealCacheSize();
|
||||
LOG.info("failedBlockAdditions=" + getFailedBlockAdditions() + ", " +
|
||||
LOG.info("failedBlockAdditions=" + cacheStats.getFailedInserts() + ", " +
|
||||
"totalSize=" + StringUtils.byteDesc(totalSize) + ", " +
|
||||
"freeSize=" + StringUtils.byteDesc(freeSize) + ", " +
|
||||
"usedSize=" + StringUtils.byteDesc(usedSize) +", " +
|
||||
|
@ -535,10 +534,6 @@ public class BucketCache implements BlockCache, HeapSize {
|
|||
cacheStats.reset();
|
||||
}
|
||||
|
||||
public long getFailedBlockAdditions() {
|
||||
return this.failedBlockAdditions.get();
|
||||
}
|
||||
|
||||
public long getRealCacheSize() {
|
||||
return this.realCacheSize.get();
|
||||
}
|
||||
|
|
|
@ -310,6 +310,11 @@ class MetricsRegionServerWrapperImpl
|
|||
return (ratio * 100);
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getBlockCacheFailedInsertions() {
|
||||
return this.cacheStats.getFailedInserts();
|
||||
}
|
||||
|
||||
@Override public void forceRecompute() {
|
||||
this.runnable.run();
|
||||
}
|
||||
|
|
|
@ -19,6 +19,8 @@
|
|||
package org.apache.hadoop.hbase.io.hfile;
|
||||
|
||||
import static org.junit.Assert.assertEquals;
|
||||
import static org.junit.Assert.assertNotNull;
|
||||
import static org.junit.Assert.assertNull;
|
||||
import static org.junit.Assert.assertTrue;
|
||||
|
||||
import java.nio.ByteBuffer;
|
||||
|
@ -265,7 +267,8 @@ public class TestLruBlockCache {
|
|||
0.33f, // single
|
||||
0.33f, // multi
|
||||
0.34f, // memory
|
||||
false);
|
||||
false,
|
||||
16 * 1024 * 1024);
|
||||
|
||||
CachedItem [] singleBlocks = generateFixedBlocks(5, blockSize, "single");
|
||||
CachedItem [] multiBlocks = generateFixedBlocks(5, blockSize, "multi");
|
||||
|
@ -385,7 +388,8 @@ public class TestLruBlockCache {
|
|||
0.2f, // single
|
||||
0.3f, // multi
|
||||
0.5f, // memory
|
||||
true);
|
||||
true,
|
||||
16 * 1024 * 1024);
|
||||
|
||||
CachedItem [] singleBlocks = generateFixedBlocks(10, blockSize, "single");
|
||||
CachedItem [] multiBlocks = generateFixedBlocks(10, blockSize, "multi");
|
||||
|
@ -490,7 +494,8 @@ public class TestLruBlockCache {
|
|||
0.33f, // single
|
||||
0.33f, // multi
|
||||
0.34f, // memory
|
||||
false);
|
||||
false,
|
||||
16 * 1024 * 1024);
|
||||
|
||||
CachedItem [] singleBlocks = generateFixedBlocks(20, blockSize, "single");
|
||||
CachedItem [] multiBlocks = generateFixedBlocks(5, blockSize, "multi");
|
||||
|
@ -538,6 +543,43 @@ public class TestLruBlockCache {
|
|||
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testMaxBlockSize() throws Exception {
|
||||
long maxSize = 100000;
|
||||
long blockSize = calculateBlockSize(maxSize, 10);
|
||||
|
||||
LruBlockCache cache = new LruBlockCache(maxSize, blockSize, false,
|
||||
(int)Math.ceil(1.2*maxSize/blockSize),
|
||||
LruBlockCache.DEFAULT_LOAD_FACTOR,
|
||||
LruBlockCache.DEFAULT_CONCURRENCY_LEVEL,
|
||||
0.66f, // min
|
||||
0.99f, // acceptable
|
||||
0.33f, // single
|
||||
0.33f, // multi
|
||||
0.34f, // memory
|
||||
false,
|
||||
1024);
|
||||
CachedItem [] tooLong = generateFixedBlocks(10, 1024+5, "long");
|
||||
CachedItem [] small = generateFixedBlocks(15, 600, "small");
|
||||
|
||||
|
||||
for (CachedItem i:tooLong) {
|
||||
cache.cacheBlock(i.cacheKey, i);
|
||||
}
|
||||
for (CachedItem i:small) {
|
||||
cache.cacheBlock(i.cacheKey, i);
|
||||
}
|
||||
assertEquals(15,cache.getBlockCount());
|
||||
for (CachedItem i:small) {
|
||||
assertNotNull(cache.getBlock(i.cacheKey, true, false, false));
|
||||
}
|
||||
for (CachedItem i:tooLong) {
|
||||
assertNull(cache.getBlock(i.cacheKey, true, false, false));
|
||||
}
|
||||
|
||||
assertEquals(10, cache.getStats().getFailedInserts());
|
||||
}
|
||||
|
||||
// test setMaxSize
|
||||
@Test
|
||||
public void testResizeBlockCache() throws Exception {
|
||||
|
@ -554,7 +596,8 @@ public class TestLruBlockCache {
|
|||
0.33f, // single
|
||||
0.33f, // multi
|
||||
0.34f, // memory
|
||||
false);
|
||||
false,
|
||||
16 * 1024 * 1024);
|
||||
|
||||
CachedItem [] singleBlocks = generateFixedBlocks(10, blockSize, "single");
|
||||
CachedItem [] multiBlocks = generateFixedBlocks(10, blockSize, "multi");
|
||||
|
|
|
@ -210,6 +210,10 @@ public class MetricsRegionServerWrapperStub implements MetricsRegionServerWrappe
|
|||
return 97;
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getBlockCacheFailedInsertions() {
|
||||
return 36;
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getUpdatesBlockedTime() {
|
||||
|
|
|
@ -86,6 +86,7 @@ public class TestMetricsRegionServer {
|
|||
HELPER.assertCounter("blockCacheEvictionCount", 418, serverSource);
|
||||
HELPER.assertGauge("blockCacheCountHitPercent", 98, serverSource);
|
||||
HELPER.assertGauge("blockCacheExpressHitPercent", 97, serverSource);
|
||||
HELPER.assertCounter("blockCacheFailedInsertionCount", 36, serverSource);
|
||||
HELPER.assertCounter("updatesBlockedTime", 419, serverSource);
|
||||
}
|
||||
|
||||
|
|
Loading…
Reference in New Issue