@ -156,6 +156,7 @@ class LRUCache {
Cache : : Handle * Lookup ( const Slice & key , uint32_t hash ) ;
Cache : : Handle * Lookup ( const Slice & key , uint32_t hash ) ;
void Release ( Cache : : Handle * handle ) ;
void Release ( Cache : : Handle * handle ) ;
void Erase ( const Slice & key , uint32_t hash ) ;
void Erase ( const Slice & key , uint32_t hash ) ;
size_t GetUsage ( ) const { return usage_ . load ( std : : memory_order_relaxed ) ; }
private :
private :
void LRU_Remove ( LRUHandle * e ) ;
void LRU_Remove ( LRUHandle * e ) ;
@ -172,7 +173,7 @@ class LRUCache {
// mutex_ protects the following state.
// mutex_ protects the following state.
port : : Mutex mutex_ ;
port : : Mutex mutex_ ;
size_t usage_ ;
std : : atomic_s ize_t usage_ ;
// Dummy head of LRU list.
// Dummy head of LRU list.
// lru.prev is newest entry, lru.next is oldest entry.
// lru.prev is newest entry, lru.next is oldest entry.
@ -214,7 +215,7 @@ void LRUCache::FreeEntry(LRUHandle* e) {
void LRUCache : : LRU_Remove ( LRUHandle * e ) {
void LRUCache : : LRU_Remove ( LRUHandle * e ) {
e - > next - > prev = e - > prev ;
e - > next - > prev = e - > prev ;
e - > prev - > next = e - > next ;
e - > prev - > next = e - > next ;
usage_ - = e - > charge ;
usage_ . fetch_sub ( e - > charge , std : : memory_order_relaxed ) ;
}
}
void LRUCache : : LRU_Append ( LRUHandle * e ) {
void LRUCache : : LRU_Append ( LRUHandle * e ) {
@ -223,7 +224,7 @@ void LRUCache::LRU_Append(LRUHandle* e) {
e - > prev = lru_ . prev ;
e - > prev = lru_ . prev ;
e - > prev - > next = e ;
e - > prev - > next = e ;
e - > next - > prev = e ;
e - > next - > prev = e ;
usage_ + = e - > charge ;
usage_ . fetch_add ( e - > charge , std : : memory_order_relaxed ) ;
}
}
Cache : : Handle * LRUCache : : Lookup ( const Slice & key , uint32_t hash ) {
Cache : : Handle * LRUCache : : Lookup ( const Slice & key , uint32_t hash ) {
@ -282,7 +283,7 @@ Cache::Handle* LRUCache::Insert(
// referenced by the cache first.
// referenced by the cache first.
LRUHandle * cur = lru_ . next ;
LRUHandle * cur = lru_ . next ;
for ( unsigned int scanCount = 0 ;
for ( unsigned int scanCount = 0 ;
usage_ > capacity_ & & cur ! = & lru_
GetUsage ( ) > capacity_ & & cur ! = & lru_
& & scanCount < remove_scan_count_limit_ ; scanCount + + ) {
& & scanCount < remove_scan_count_limit_ ; scanCount + + ) {
LRUHandle * next = cur - > next ;
LRUHandle * next = cur - > next ;
if ( cur - > refs < = 1 ) {
if ( cur - > refs < = 1 ) {
@ -298,7 +299,7 @@ Cache::Handle* LRUCache::Insert(
// Free the space following strict LRU policy until enough space
// Free the space following strict LRU policy until enough space
// is freed.
// is freed.
while ( usage_ > capacity_ & & lru_ . next ! = & lru_ ) {
while ( GetUsage ( ) > capacity_ & & lru_ . next ! = & lru_ ) {
LRUHandle * old = lru_ . next ;
LRUHandle * old = lru_ . next ;
LRU_Remove ( old ) ;
LRU_Remove ( old ) ;
table_ . Remove ( old - > key ( ) , old - > hash ) ;
table_ . Remove ( old - > key ( ) , old - > hash ) ;
@ -340,10 +341,10 @@ static int kRemoveScanCountLimit = 0; // default values, can be overridden
class ShardedLRUCache : public Cache {
class ShardedLRUCache : public Cache {
private :
private :
LRUCache * shard_ ;
LRUCache * shards _ ;
port : : Mutex id_mutex_ ;
port : : Mutex id_mutex_ ;
uint64_t last_id_ ;
uint64_t last_id_ ;
int numShardBits ;
int num_shard_bits_ ;
size_t capacity_ ;
size_t capacity_ ;
static inline uint32_t HashSlice ( const Slice & s ) {
static inline uint32_t HashSlice ( const Slice & s ) {
@ -352,18 +353,18 @@ class ShardedLRUCache : public Cache {
uint32_t Shard ( uint32_t hash ) {
uint32_t Shard ( uint32_t hash ) {
// Note, hash >> 32 yields hash in gcc, not the zero we expect!
// Note, hash >> 32 yields hash in gcc, not the zero we expect!
return ( numShardBits > 0 ) ? ( hash > > ( 32 - numShardBits ) ) : 0 ;
return ( num_shard_bits_ > 0 ) ? ( hash > > ( 32 - num_shard_bits_ ) ) : 0 ;
}
}
void init ( size_t capacity , int numbits , int removeScanCountLimit ) {
void init ( size_t capacity , int numbits , int removeScanCountLimit ) {
numShardBits = numbits ;
num_shard_bits_ = numbits ;
capacity_ = capacity ;
capacity_ = capacity ;
int numS hards = 1 < < numShardBits ;
int num_s hards = 1 < < num_shard_bits_ ;
shard_ = new LRUCache [ numS hards ] ;
shards _ = new LRUCache [ num_s hards ] ;
const size_t per_shard = ( capacity + ( numS hards - 1 ) ) / numS hards ;
const size_t per_shard = ( capacity + ( num_s hards - 1 ) ) / num_s hards ;
for ( int s = 0 ; s < numS hards ; s + + ) {
for ( int s = 0 ; s < num_s hards ; s + + ) {
shard_ [ s ] . SetCapacity ( per_shard ) ;
shards _ [ s ] . SetCapacity ( per_shard ) ;
shard_ [ s ] . SetRemoveScanCountLimit ( removeScanCountLimit ) ;
shards _ [ s ] . SetRemoveScanCountLimit ( removeScanCountLimit ) ;
}
}
}
}
@ -372,30 +373,30 @@ class ShardedLRUCache : public Cache {
: last_id_ ( 0 ) {
: last_id_ ( 0 ) {
init ( capacity , kNumShardBits , kRemoveScanCountLimit ) ;
init ( capacity , kNumShardBits , kRemoveScanCountLimit ) ;
}
}
ShardedLRUCache ( size_t capacity , int numShardB its ,
ShardedLRUCache ( size_t capacity , int num_shard_b its ,
int removeScanCountLimit )
int removeScanCountLimit )
: last_id_ ( 0 ) {
: last_id_ ( 0 ) {
init ( capacity , numShardB its , removeScanCountLimit ) ;
init ( capacity , num_shard_b its , removeScanCountLimit ) ;
}
}
virtual ~ ShardedLRUCache ( ) {
virtual ~ ShardedLRUCache ( ) {
delete [ ] shard_ ;
delete [ ] shards _ ;
}
}
virtual Handle * Insert ( const Slice & key , void * value , size_t charge ,
virtual Handle * Insert ( const Slice & key , void * value , size_t charge ,
void ( * deleter ) ( const Slice & key , void * value ) ) {
void ( * deleter ) ( const Slice & key , void * value ) ) {
const uint32_t hash = HashSlice ( key ) ;
const uint32_t hash = HashSlice ( key ) ;
return shard_ [ Shard ( hash ) ] . Insert ( key , hash , value , charge , deleter ) ;
return shards _ [ Shard ( hash ) ] . Insert ( key , hash , value , charge , deleter ) ;
}
}
virtual Handle * Lookup ( const Slice & key ) {
virtual Handle * Lookup ( const Slice & key ) {
const uint32_t hash = HashSlice ( key ) ;
const uint32_t hash = HashSlice ( key ) ;
return shard_ [ Shard ( hash ) ] . Lookup ( key , hash ) ;
return shards _ [ Shard ( hash ) ] . Lookup ( key , hash ) ;
}
}
virtual void Release ( Handle * handle ) {
virtual void Release ( Handle * handle ) {
LRUHandle * h = reinterpret_cast < LRUHandle * > ( handle ) ;
LRUHandle * h = reinterpret_cast < LRUHandle * > ( handle ) ;
shard_ [ Shard ( h - > hash ) ] . Release ( handle ) ;
shards _ [ Shard ( h - > hash ) ] . Release ( handle ) ;
}
}
virtual void Erase ( const Slice & key ) {
virtual void Erase ( const Slice & key ) {
const uint32_t hash = HashSlice ( key ) ;
const uint32_t hash = HashSlice ( key ) ;
shard_ [ Shard ( hash ) ] . Erase ( key , hash ) ;
shards _ [ Shard ( hash ) ] . Erase ( key , hash ) ;
}
}
virtual void * Value ( Handle * handle ) {
virtual void * Value ( Handle * handle ) {
return reinterpret_cast < LRUHandle * > ( handle ) - > value ;
return reinterpret_cast < LRUHandle * > ( handle ) - > value ;
@ -407,6 +408,16 @@ class ShardedLRUCache : public Cache {
virtual size_t GetCapacity ( ) const {
virtual size_t GetCapacity ( ) const {
return capacity_ ;
return capacity_ ;
}
}
virtual size_t GetUsage ( ) const {
// We will not lock the cache when getting the usage from shards.
// for (size_t i = 0; i < num_shard_bits_; ++i)
int num_shards = 1 < < num_shard_bits_ ;
size_t usage = 0 ;
for ( int s = 0 ; s < num_shards ; s + + ) {
usage + = shards_ [ s ] . GetUsage ( ) ;
}
return usage ;
}
} ;
} ;
} // end anonymous namespace
} // end anonymous namespace
@ -415,17 +426,17 @@ shared_ptr<Cache> NewLRUCache(size_t capacity) {
return NewLRUCache ( capacity , kNumShardBits ) ;
return NewLRUCache ( capacity , kNumShardBits ) ;
}
}
shared_ptr < Cache > NewLRUCache ( size_t capacity , int numShardB its ) {
shared_ptr < Cache > NewLRUCache ( size_t capacity , int num_shard_b its ) {
return NewLRUCache ( capacity , numShardB its , kRemoveScanCountLimit ) ;
return NewLRUCache ( capacity , num_shard_b its , kRemoveScanCountLimit ) ;
}
}
shared_ptr < Cache > NewLRUCache ( size_t capacity , int numShardB its ,
shared_ptr < Cache > NewLRUCache ( size_t capacity , int num_shard_b its ,
int removeScanCountLimit ) {
int removeScanCountLimit ) {
if ( numShardB its > = 20 ) {
if ( num_shard_b its > = 20 ) {
return nullptr ; // the cache cannot be sharded into too many fine pieces
return nullptr ; // the cache cannot be sharded into too many fine pieces
}
}
return std : : make_shared < ShardedLRUCache > ( capacity ,
return std : : make_shared < ShardedLRUCache > ( capacity ,
numShardB its ,
num_shard_b its ,
removeScanCountLimit ) ;
removeScanCountLimit ) ;
}
}