fork of https://github.com/oxigraph/rocksdb and https://github.com/facebook/rocksdb for nextgraph and oxigraph
				
			
			
		
			You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							174 lines
						
					
					
						
							3.4 KiB
						
					
					
				
			
		
		
	
	
							174 lines
						
					
					
						
							3.4 KiB
						
					
					
				| //  Copyright (c) 2013, Facebook, Inc.  All rights reserved.
 | |
| //  This source code is licensed under both the GPLv2 (found in the
 | |
| //  COPYING file in the root directory) and Apache 2.0 License
 | |
| //  (found in the LICENSE.Apache file in the root directory).
 | |
| //
 | |
| #pragma once
 | |
| 
 | |
| #ifndef ROCKSDB_LITE
 | |
| 
 | |
| #include <atomic>
 | |
| 
 | |
| #include "util/mutexlock.h"
 | |
| 
 | |
| namespace rocksdb {
 | |
| 
 | |
| // LRU element definition
 | |
| //
 | |
| // Any object that needs to be part of the LRU algorithm should extend this
 | |
| // class
 | |
| template <class T>
 | |
| struct LRUElement {
 | |
|   explicit LRUElement() : next_(nullptr), prev_(nullptr), refs_(0) {}
 | |
| 
 | |
|   virtual ~LRUElement() { assert(!refs_); }
 | |
| 
 | |
|   T* next_;
 | |
|   T* prev_;
 | |
|   std::atomic<size_t> refs_;
 | |
| };
 | |
| 
 | |
| // LRU implementation
 | |
| //
 | |
| // In place LRU implementation. There is no copy or allocation involved when
 | |
| // inserting or removing an element. This makes the data structure slim
 | |
| template <class T>
 | |
| class LRUList {
 | |
|  public:
 | |
|   virtual ~LRUList() {
 | |
|     MutexLock _(&lock_);
 | |
|     assert(!head_);
 | |
|     assert(!tail_);
 | |
|   }
 | |
| 
 | |
|   // Push element into the LRU at the cold end
 | |
|   inline void Push(T* const t) {
 | |
|     assert(t);
 | |
|     assert(!t->next_);
 | |
|     assert(!t->prev_);
 | |
| 
 | |
|     MutexLock _(&lock_);
 | |
| 
 | |
|     assert((!head_ && !tail_) || (head_ && tail_));
 | |
|     assert(!head_ || !head_->prev_);
 | |
|     assert(!tail_ || !tail_->next_);
 | |
| 
 | |
|     t->next_ = head_;
 | |
|     if (head_) {
 | |
|       head_->prev_ = t;
 | |
|     }
 | |
| 
 | |
|     head_ = t;
 | |
|     if (!tail_) {
 | |
|       tail_ = t;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // Unlink the element from the LRU
 | |
|   inline void Unlink(T* const t) {
 | |
|     MutexLock _(&lock_);
 | |
|     UnlinkImpl(t);
 | |
|   }
 | |
| 
 | |
|   // Evict an element from the LRU
 | |
|   inline T* Pop() {
 | |
|     MutexLock _(&lock_);
 | |
| 
 | |
|     assert(tail_ && head_);
 | |
|     assert(!tail_->next_);
 | |
|     assert(!head_->prev_);
 | |
| 
 | |
|     T* t = head_;
 | |
|     while (t && t->refs_) {
 | |
|       t = t->next_;
 | |
|     }
 | |
| 
 | |
|     if (!t) {
 | |
|       // nothing can be evicted
 | |
|       return nullptr;
 | |
|     }
 | |
| 
 | |
|     assert(!t->refs_);
 | |
| 
 | |
|     // unlike the element
 | |
|     UnlinkImpl(t);
 | |
|     return t;
 | |
|   }
 | |
| 
 | |
|   // Move the element from the front of the list to the back of the list
 | |
|   inline void Touch(T* const t) {
 | |
|     MutexLock _(&lock_);
 | |
|     UnlinkImpl(t);
 | |
|     PushBackImpl(t);
 | |
|   }
 | |
| 
 | |
|   // Check if the LRU is empty
 | |
|   inline bool IsEmpty() const {
 | |
|     MutexLock _(&lock_);
 | |
|     return !head_ && !tail_;
 | |
|   }
 | |
| 
 | |
|  private:
 | |
|   // Unlink an element from the LRU
 | |
|   void UnlinkImpl(T* const t) {
 | |
|     assert(t);
 | |
| 
 | |
|     lock_.AssertHeld();
 | |
| 
 | |
|     assert(head_ && tail_);
 | |
|     assert(t->prev_ || head_ == t);
 | |
|     assert(t->next_ || tail_ == t);
 | |
| 
 | |
|     if (t->prev_) {
 | |
|       t->prev_->next_ = t->next_;
 | |
|     }
 | |
|     if (t->next_) {
 | |
|       t->next_->prev_ = t->prev_;
 | |
|     }
 | |
| 
 | |
|     if (tail_ == t) {
 | |
|       tail_ = tail_->prev_;
 | |
|     }
 | |
|     if (head_ == t) {
 | |
|       head_ = head_->next_;
 | |
|     }
 | |
| 
 | |
|     t->next_ = t->prev_ = nullptr;
 | |
|   }
 | |
| 
 | |
|   // Insert an element at the hot end
 | |
|   inline void PushBack(T* const t) {
 | |
|     MutexLock _(&lock_);
 | |
|     PushBackImpl(t);
 | |
|   }
 | |
| 
 | |
|   inline void PushBackImpl(T* const t) {
 | |
|     assert(t);
 | |
|     assert(!t->next_);
 | |
|     assert(!t->prev_);
 | |
| 
 | |
|     lock_.AssertHeld();
 | |
| 
 | |
|     assert((!head_ && !tail_) || (head_ && tail_));
 | |
|     assert(!head_ || !head_->prev_);
 | |
|     assert(!tail_ || !tail_->next_);
 | |
| 
 | |
|     t->prev_ = tail_;
 | |
|     if (tail_) {
 | |
|       tail_->next_ = t;
 | |
|     }
 | |
| 
 | |
|     tail_ = t;
 | |
|     if (!head_) {
 | |
|       head_ = tail_;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   mutable port::Mutex lock_;  // synchronization primitive
 | |
|   T* head_ = nullptr;         // front (cold)
 | |
|   T* tail_ = nullptr;         // back (hot)
 | |
| };
 | |
| 
 | |
| }  // namespace rocksdb
 | |
| 
 | |
| #endif
 | |
| 
 |