Remove cache eviction for model data, optimize
This commit is contained in:
parent
61b612c88a
commit
fb30820e2b
1 changed files with 29 additions and 35 deletions
|
@ -21,54 +21,43 @@ package net.minecraftforge.client.model;
|
|||
|
||||
import java.lang.ref.WeakReference;
|
||||
import java.util.Collections;
|
||||
import java.util.HashMap;
|
||||
import java.util.HashSet;
|
||||
import java.util.Map;
|
||||
import java.util.Set;
|
||||
import java.util.concurrent.ConcurrentHashMap;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
import javax.annotation.Nonnull;
|
||||
import javax.annotation.Nullable;
|
||||
|
||||
import com.google.common.base.Preconditions;
|
||||
import com.google.common.cache.CacheBuilder;
|
||||
import com.google.common.cache.CacheLoader;
|
||||
import com.google.common.cache.LoadingCache;
|
||||
|
||||
import net.minecraft.client.Minecraft;
|
||||
import net.minecraft.tileentity.TileEntity;
|
||||
import net.minecraft.util.math.BlockPos;
|
||||
import net.minecraft.util.math.ChunkPos;
|
||||
import net.minecraft.world.World;
|
||||
import net.minecraftforge.api.distmarker.Dist;
|
||||
import net.minecraftforge.client.model.data.IModelData;
|
||||
import net.minecraftforge.event.world.ChunkEvent;
|
||||
import net.minecraftforge.eventbus.api.SubscribeEvent;
|
||||
import net.minecraftforge.fml.common.Mod.EventBusSubscriber;
|
||||
import net.minecraftforge.fml.common.Mod.EventBusSubscriber.Bus;
|
||||
|
||||
@EventBusSubscriber(modid = "forge", bus = Bus.FORGE, value = Dist.CLIENT)
|
||||
public class ModelDataManager
|
||||
{
|
||||
private static WeakReference<World> currentWorld = new WeakReference<>(null);
|
||||
|
||||
private static final Map<ChunkPos, Set<BlockPos>> needModelDataRefresh = new HashMap<>();
|
||||
private static final Map<ChunkPos, Set<BlockPos>> needModelDataRefresh = new ConcurrentHashMap<>();
|
||||
|
||||
private static final LoadingCache<ChunkPos, Map<BlockPos, IModelData>> modelDataCache = CacheBuilder.newBuilder()
|
||||
.maximumSize(1000)
|
||||
.concurrencyLevel(5)
|
||||
.expireAfterAccess(30, TimeUnit.SECONDS)
|
||||
.build(new CacheLoader<ChunkPos, Map<BlockPos, IModelData>>(){
|
||||
private static final Map<ChunkPos, Map<BlockPos, IModelData>> modelDataCache = new ConcurrentHashMap<>();
|
||||
|
||||
@Override
|
||||
public Map<BlockPos, IModelData> load(@Nonnull ChunkPos key)
|
||||
{
|
||||
return new ConcurrentHashMap<>();
|
||||
}
|
||||
});
|
||||
|
||||
private static void cleanCaches(World world)
|
||||
{
|
||||
if (world != currentWorld.get())
|
||||
{
|
||||
currentWorld = new WeakReference<>(world);
|
||||
needModelDataRefresh.clear();
|
||||
modelDataCache.invalidateAll();
|
||||
modelDataCache.clear();
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -77,25 +66,20 @@ public class ModelDataManager
|
|||
World world = te.getWorld();
|
||||
Preconditions.checkNotNull(world, "Tile entity world must not be null");
|
||||
Preconditions.checkArgument(world == Minecraft.getInstance().world, "Cannot request a model data refresh for a world other than the current client world");
|
||||
synchronized (needModelDataRefresh)
|
||||
{
|
||||
cleanCaches(world);
|
||||
needModelDataRefresh.computeIfAbsent(new ChunkPos(te.getPos()), $ -> Collections.synchronizedSet(new HashSet<>()))
|
||||
.add(te.getPos());
|
||||
}
|
||||
|
||||
cleanCaches(world);
|
||||
needModelDataRefresh.computeIfAbsent(new ChunkPos(te.getPos()), $ -> Collections.synchronizedSet(new HashSet<>()))
|
||||
.add(te.getPos());
|
||||
}
|
||||
|
||||
private static void refreshModelData(World world, ChunkPos chunk)
|
||||
{
|
||||
Set<BlockPos> needUpdate;
|
||||
synchronized (needModelDataRefresh)
|
||||
{
|
||||
cleanCaches(world);
|
||||
needUpdate = needModelDataRefresh.remove(chunk);
|
||||
}
|
||||
{
|
||||
cleanCaches(world);
|
||||
Set<BlockPos> needUpdate = needModelDataRefresh.remove(chunk);
|
||||
|
||||
if (needUpdate != null)
|
||||
{
|
||||
Map<BlockPos, IModelData> data = modelDataCache.getUnchecked(chunk);
|
||||
Map<BlockPos, IModelData> data = modelDataCache.computeIfAbsent(chunk, $ -> new ConcurrentHashMap<>());
|
||||
for (BlockPos pos : needUpdate)
|
||||
{
|
||||
TileEntity toUpdate = world.getTileEntity(pos);
|
||||
|
@ -107,6 +91,16 @@ public class ModelDataManager
|
|||
}
|
||||
}
|
||||
|
||||
@SubscribeEvent
|
||||
public static void onChunkUnload(ChunkEvent.Unload event)
|
||||
{
|
||||
if (!event.getChunk().getWorldForge().isRemote()) return;
|
||||
|
||||
ChunkPos chunk = event.getChunk().getPos();
|
||||
needModelDataRefresh.remove(chunk);
|
||||
modelDataCache.remove(chunk);
|
||||
}
|
||||
|
||||
public static @Nullable IModelData getModelData(World world, BlockPos pos)
|
||||
{
|
||||
return getModelData(world, new ChunkPos(pos)).get(pos);
|
||||
|
@ -115,6 +109,6 @@ public class ModelDataManager
|
|||
public static Map<BlockPos, IModelData> getModelData(World world, ChunkPos pos)
|
||||
{
|
||||
refreshModelData(world, pos);
|
||||
return modelDataCache.getUnchecked(pos);
|
||||
return modelDataCache.getOrDefault(pos, Collections.emptyMap());
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue