HyCodeYourTale
classpublicPriority 3

ConcurrentSizedTimeoutCache

com.hypixel.hytale.server.worldgen.util.cache.ConcurrentSizedTimeoutCache

implements Cache<K, V>

6

Methods

6

Public Methods

6

Fields

1

Constructors

Constants

floatBUCKET_LOAD_FACTOR= 0.75F
intBUCKET_MIN_CAPACITY= 16

Constructors

public
ConcurrentSizedTimeoutCache(int capacity, int concurrencyLevel, long timeout, TimeUnit timeoutUnit, Function<K, K> computeKey, Function<K, V> computeValue, BiConsumer<K, V> destroyer)

Methods

Public Methods (6)

public
void cleanup()
@Override
public
void cleanup(BiConsumer<K, V> destroyer)
public
void clear(BiConsumer<K, V> destroyer)
public
V compute(K key, Function<K, K> computeKey, Function<K, V> computeValue, BiConsumer<K, V> destroyer)
@Nonnull
public
V get(K key)
@Nonnull@Override
public
void shutdown()
@Override

Fields

Private/Package Fields (6)

privateint bucketMask
privateConcurrentSizedTimeoutCache.Bucket<K, V>[] buckets
privateCleanable cleanable
privateFunction<K, K> computeKey
privateFunction<K, V> computeValue
privateBiConsumer<K, V> destroyer

Inheritance

Parent
Current
Interface
Child

Use mouse wheel to zoom, drag to pan. Click nodes to navigate.

Related Classes

Source Code

package com.hypixel.hytale.server.worldgen.util.cache;

import com.hypixel.hytale.math.util.MathUtil;
import com.hypixel.hytale.server.core.HytaleServer;
import com.hypixel.hytale.server.worldgen.util.LogUtil;
import it.unimi.dsi.fastutil.HashCommon;
import it.unimi.dsi.fastutil.objects.Object2ObjectOpenHashMap;
import it.unimi.dsi.fastutil.objects.ObjectIterator;
import it.unimi.dsi.fastutil.objects.Object2ObjectMap.Entry;
import java.lang.invoke.MethodHandles;
import java.lang.invoke.VarHandle;
import java.lang.ref.WeakReference;
import java.lang.ref.Cleaner.Cleanable;
import java.util.ArrayDeque;
import java.util.Objects;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.StampedLock;
import java.util.function.BiConsumer;
import java.util.function.Function;
import java.util.logging.Level;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;

public class ConcurrentSizedTimeoutCache<K, V> implements Cache<K, V> {
   private static final int BUCKET_MIN_CAPACITY = 16;
   private static final float BUCKET_LOAD_FACTOR = 0.75F;
   private final int bucketMask;
   @Nonnull
   private final ConcurrentSizedTimeoutCache.Bucket<K, V>[] buckets;
   @Nonnull
   private final Function<K, K> computeKey;
   @Nonnull
   private final Function<K, V> computeValue;
   @Nonnull
   private final BiConsumer<K, V> destroyer;
   @Nonnull
   private final ScheduledFuture<?> future;
   @Nonnull
   private final Cleanable cleanable;

   public ConcurrentSizedTimeoutCache(
      int capacity,
      int concurrencyLevel,
      long timeout,
      @Nonnull TimeUnit timeoutUnit,
      @Nonnull Function<K, K> computeKey,
      @Nonnull Function<K, V> computeValue,
      @Nullable BiConsumer<K, V> destroyer
   ) {
      long timeout_ns = timeoutUnit.toNanos(timeout);
      int bucketCount = HashCommon.nextPowerOfTwo(concurrencyLevel);
      int bucketCapacity = Math.max(16, HashCommon.nextPowerOfTwo(capacity / bucketCount));
      this.bucketMask = bucketCount - 1;
      this.buckets = new ConcurrentSizedTimeoutCache.Bucket[bucketCount];

      for (int i = 0; i < bucketCount; i++) {
         this.buckets[i] = new ConcurrentSizedTimeoutCache.Bucket<>(bucketCapacity, timeout_ns);
      }

      this.computeKey = computeKey;
      this.computeValue = computeValue;
      this.destroyer = destroyer != null ? destroyer : ConcurrentSizedTimeoutCache::noopDestroy;
      this.future = HytaleServer.SCHEDULED_EXECUTOR.scheduleWithFixedDelay(new CleanupRunnable(new WeakReference<>(this)), timeout, timeout, timeoutUnit);
      this.cleanable = CleanupFutureAction.CLEANER.register(this, new CleanupFutureAction(this.future));
   }

   @Override
   public void shutdown() {
      this.cleanable.clean();

      for (ConcurrentSizedTimeoutCache.Bucket<K, V> bucket : this.buckets) {
         bucket.clear(this.destroyer);
      }
   }

   @Override
   public void cleanup() {
      for (ConcurrentSizedTimeoutCache.Bucket<K, V> bucket : this.buckets) {
         bucket.cleanup(this.destroyer);
      }
   }

   @Nonnull
   @Override
   public V get(K key) {
      if (this.future.isCancelled()) {
         throw new IllegalStateException("Cache has been shutdown!");
      } else {
         int hash = HashCommon.mix(key.hashCode());
         return this.buckets[hash & this.bucketMask].compute(key, this.computeKey, this.computeValue, this.destroyer);
      }
   }

   private static <K, V> void noopDestroy(K key, V value) {
   }

   private static class Bucket<K, V> {
      private final int capacity;
      private final int trimThreshold;
      private final long timeout_ns;
      private final ArrayDeque<ConcurrentSizedTimeoutCache.CacheEntry<K, V>> pool;
      private final Object2ObjectOpenHashMap<K, ConcurrentSizedTimeoutCache.CacheEntry<K, V>> map;
      private final StampedLock lock = new StampedLock();

      public Bucket(int capacity, long timeout_ns) {
         this.capacity = capacity;
         this.trimThreshold = MathUtil.fastFloor((float)capacity * 0.75F);
         this.timeout_ns = timeout_ns;
         this.pool = new ArrayDeque<>(capacity);
         this.map = new Object2ObjectOpenHashMap(capacity, 0.75F);
      }

      @Nonnull
      public V compute(@Nonnull K key, @Nonnull Function<K, K> computeKey, @Nonnull Function<K, V> computeValue, @Nonnull BiConsumer<K, V> destroyer) {
         long timestamp = System.nanoTime();
         long readStamp = this.lock.readLock();

         try {
            ConcurrentSizedTimeoutCache.CacheEntry<K, V> entry = (ConcurrentSizedTimeoutCache.CacheEntry<K, V>)this.map.get(key);
            if (entry != null) {
               return entry.markAndGet(timestamp);
            }
         } finally {
            this.lock.unlockRead(readStamp);
         }

         Object var23 = computeKey.apply(key);
         V newValue = computeValue.apply(key);
         Object resultValue = newValue;
         long writeStamp = this.lock.writeLock();

         try {
            ConcurrentSizedTimeoutCache.CacheEntry<K, V> newEntry = this.pool.isEmpty() ? new ConcurrentSizedTimeoutCache.CacheEntry<>() : this.pool.poll();
            Objects.requireNonNull(newEntry, "CacheEntry pool returned null entry!");
            newEntry.key = (K)var23;
            newEntry.value = newValue;
            newEntry.timestamp = timestamp;
            ConcurrentSizedTimeoutCache.CacheEntry<K, V> currentEntry = (ConcurrentSizedTimeoutCache.CacheEntry<K, V>)this.map.putIfAbsent(var23, newEntry);
            if (currentEntry != null) {
               Objects.requireNonNull(currentEntry.value);
               resultValue = currentEntry.value;
               currentEntry.timestamp = timestamp;
               newEntry.key = null;
               newEntry.value = null;
               if (this.pool.size() < this.capacity) {
                  this.pool.offer(newEntry);
               }
            }
         } finally {
            this.lock.unlockWrite(writeStamp);
         }

         if (newValue != resultValue) {
            destroyer.accept((K)var23, newValue);
         }

         return (V)resultValue;
      }

      public void cleanup(@Nullable BiConsumer<K, V> destroyer) {
         long writeStamp = this.lock.writeLock();

         try {
            boolean needsTrim = this.map.size() >= this.trimThreshold;
            long expireTimestamp = System.nanoTime() - this.timeout_ns;
            ObjectIterator<Entry<K, ConcurrentSizedTimeoutCache.CacheEntry<K, V>>> it = this.map.object2ObjectEntrySet().fastIterator();

            while (it.hasNext()) {
               ConcurrentSizedTimeoutCache.CacheEntry<K, V> entry = (ConcurrentSizedTimeoutCache.CacheEntry<K, V>)((Entry)it.next()).getValue();
               if (entry == null) {
                  LogUtil.getLogger().at(Level.SEVERE).log("Found null entry in cache bucket during cleanup!");
                  it.remove();
               } else if (entry.timestamp >= expireTimestamp) {
                  it.remove();
                  if (destroyer != null) {
                     destroyer.accept(entry.key, entry.value);
                  }

                  entry.key = null;
                  entry.value = null;
                  if (this.pool.size() < this.capacity) {
                     this.pool.offer(entry);
                  }
               }
            }

            if (needsTrim && this.map.size() < this.capacity) {
               this.map.trim(this.capacity);
            }
         } finally {
            this.lock.unlockWrite(writeStamp);
         }
      }

      public void clear(@Nonnull BiConsumer<K, V> destroyer) {
         long writeStamp = this.lock.writeLock();

         try {
            ObjectIterator<Entry<K, ConcurrentSizedTimeoutCache.CacheEntry<K, V>>> it = this.map.object2ObjectEntrySet().fastIterator();

            while (it.hasNext()) {
               ConcurrentSizedTimeoutCache.CacheEntry<K, V> entry = (ConcurrentSizedTimeoutCache.CacheEntry<K, V>)((Entry)it.next()).getValue();
               destroyer.accept(entry.key, entry.value);
               it.remove();
            }
         } finally {
            this.lock.unlockWrite(writeStamp);
         }
      }
   }

   private static class CacheEntry<K, V> {
      private static final VarHandle TIMESTAMP;
      @Nullable
      public K key = (K)null;
      @Nullable
      public V value = (V)null;
      public long timestamp = 0L;

      private CacheEntry() {
      }

      @Nonnull
      protected V markAndGet(long timestamp) {
         Objects.requireNonNull(this.value);
         TIMESTAMP.setVolatile((ConcurrentSizedTimeoutCache.CacheEntry)this, (long)timestamp);
         return this.value;
      }

      static {
         try {
            TIMESTAMP = MethodHandles.lookup().findVarHandle(ConcurrentSizedTimeoutCache.CacheEntry.class, "timestamp", long.class);
         } catch (ReflectiveOperationException var1) {
            throw new ExceptionInInitializerError(var1);
         }
      }
   }
}