Improved fall damage handler

- allows entities with same id in different worlds
- reduce memory footprint over time
This commit is contained in:
Unknown 2019-04-14 22:02:22 +02:00
parent 49fcf2897b
commit 37f24ae882
3 changed files with 51 additions and 14 deletions

View file

@ -83,6 +83,10 @@ public class LocalProfiler {
} }
public static void stop() { public static void stop() {
stop(0L);
}
public static void stop(final long tolerance_us) {
if (stack.isEmpty()) { if (stack.isEmpty()) {
return; return;
} }
@ -90,7 +94,7 @@ public class LocalProfiler {
final StackElement stackElement = stack.pop(); final StackElement stackElement = stack.pop();
final long end = System.nanoTime(); final long end = System.nanoTime();
final long timeElapsed = end - stackElement.start; final long timeElapsed = end - stackElement.start;
if (!stack.isEmpty()) { if (!stack.isEmpty()) {
final StackElement nextStackElement = stack.peek(); final StackElement nextStackElement = stack.peek();
nextStackElement.internal += timeElapsed; nextStackElement.internal += timeElapsed;
@ -99,12 +103,16 @@ public class LocalProfiler {
// convert to microseconds // convert to microseconds
final long self = (timeElapsed - stackElement.internal) / 1000; final long self = (timeElapsed - stackElement.internal) / 1000;
final long total = timeElapsed / 1000; final long total = timeElapsed / 1000;
if (total == self) {
WarpDrive.logger.info(String.format("Profiling %s: %.3f ms", // only log if it was too slow
stackElement.name, (self / 1000.0F) )); if (self > tolerance_us) {
} else { if (total == self) {
WarpDrive.logger.info(String.format("Profiling %s: %.3f ms, total; %.3f ms", WarpDrive.logger.info(String.format("Profiling %s: %.3f ms",
stackElement.name, (self / 1000.0F), total / 1000.0F )); stackElement.name, (self / 1000.0F) ));
} else {
WarpDrive.logger.info(String.format("Profiling %s: %.3f ms, total; %.3f ms",
stackElement.name, (self / 1000.0F), total / 1000.0F ));
}
} }
} }
} }

View file

@ -2,6 +2,7 @@ package cr0s.warpdrive.event;
import cr0s.warpdrive.BreathingManager; import cr0s.warpdrive.BreathingManager;
import cr0s.warpdrive.Commons; import cr0s.warpdrive.Commons;
import cr0s.warpdrive.LocalProfiler;
import cr0s.warpdrive.WarpDrive; import cr0s.warpdrive.WarpDrive;
import cr0s.warpdrive.block.forcefield.BlockForceField; import cr0s.warpdrive.block.forcefield.BlockForceField;
import cr0s.warpdrive.data.CelestialObjectManager; import cr0s.warpdrive.data.CelestialObjectManager;
@ -13,9 +14,11 @@ import cr0s.warpdrive.data.Vector3;
import cr0s.warpdrive.data.VectorI; import cr0s.warpdrive.data.VectorI;
import java.util.HashMap; import java.util.HashMap;
import java.util.Iterator;
import java.util.UUID; import java.util.UUID;
import net.minecraft.block.Block; import net.minecraft.block.Block;
import net.minecraft.entity.Entity;
import net.minecraft.entity.EntityLivingBase; import net.minecraft.entity.EntityLivingBase;
import net.minecraft.entity.player.EntityPlayer; import net.minecraft.entity.player.EntityPlayer;
import net.minecraft.entity.player.EntityPlayerMP; import net.minecraft.entity.player.EntityPlayerMP;
@ -28,6 +31,7 @@ import net.minecraft.util.text.TextComponentTranslation;
import net.minecraft.world.World; import net.minecraft.world.World;
import net.minecraft.world.WorldServer; import net.minecraft.world.WorldServer;
import net.minecraftforge.common.DimensionManager;
import net.minecraftforge.event.entity.living.EnderTeleportEvent; import net.minecraftforge.event.entity.living.EnderTeleportEvent;
import net.minecraftforge.event.entity.living.LivingEvent.LivingUpdateEvent; import net.minecraftforge.event.entity.living.LivingEvent.LivingUpdateEvent;
import net.minecraftforge.event.entity.living.LivingFallEvent; import net.minecraftforge.event.entity.living.LivingFallEvent;
@ -35,8 +39,8 @@ import net.minecraftforge.fml.common.eventhandler.SubscribeEvent;
public class LivingHandler { public class LivingHandler {
private final HashMap<UUID, Integer> player_cloakTicks; private static final HashMap<UUID, Integer> player_cloakTicks = new HashMap<>();
private final HashMap<Integer, Double> entity_yMotion; private static final HashMap<Long, Double> entity_yMotion = new HashMap<>();
private static final int CLOAK_CHECK_TIMEOUT_TICKS = 100; private static final int CLOAK_CHECK_TIMEOUT_TICKS = 100;
@ -45,9 +49,32 @@ public class LivingHandler {
private static final int BORDER_BYPASS_PULL_BACK_BLOCKS = 16; private static final int BORDER_BYPASS_PULL_BACK_BLOCKS = 16;
private static final int BORDER_BYPASS_DAMAGES_PER_TICK = 9000; private static final int BORDER_BYPASS_DAMAGES_PER_TICK = 9000;
public LivingHandler() { private static final int PURGE_PERIOD_TICKS = 6000; // 5 mn
player_cloakTicks = new HashMap<>();
entity_yMotion = new HashMap<>(); private static int tickUpdate = PURGE_PERIOD_TICKS;
public static void updateTick() {
tickUpdate--;
if (tickUpdate < 0) {
tickUpdate = PURGE_PERIOD_TICKS;
LocalProfiler.start("LivingHandler cleanup");
final Iterator<Long> iterator = entity_yMotion.keySet().iterator();
while (iterator.hasNext()) {
final Long key = iterator.next();
final int dimensionId = (int) (key >> 32);
final WorldServer worldServer = DimensionManager.getWorld(dimensionId);
if (worldServer == null) {
iterator.remove();
continue;
}
final Entity entity = worldServer.getEntityByID((int) (key & 0xFFFFFFFFL));
if (entity == null) {
iterator.remove();
// continue;
}
}
LocalProfiler.stop(1000);
}
} }
@SubscribeEvent @SubscribeEvent
@ -62,8 +89,9 @@ public class LivingHandler {
final int z = MathHelper.floor(entityLivingBase.posZ); final int z = MathHelper.floor(entityLivingBase.posZ);
// *** save motion for fall damage computation // *** save motion for fall damage computation
// note: dead entities don't tick. Checking health level is too taxing. Hence, cleanup is done indirectly.
if (!entityLivingBase.onGround) { if (!entityLivingBase.onGround) {
entity_yMotion.put(entityLivingBase.getEntityId(), entityLivingBase.motionY); entity_yMotion.put((long) entityLivingBase.dimension << 32 | entityLivingBase.getEntityId(), entityLivingBase.motionY);
} }
// *** world border handling // *** world border handling
@ -228,7 +256,7 @@ public class LivingHandler {
// cancel in case of very low speed // cancel in case of very low speed
final EntityLivingBase entityLivingBase = event.getEntityLiving(); final EntityLivingBase entityLivingBase = event.getEntityLiving();
Double motionY = entity_yMotion.get(entityLivingBase.getEntityId()); Double motionY = entity_yMotion.get((long) entityLivingBase.dimension << 32 | entityLivingBase.getEntityId());
if (motionY == null) { if (motionY == null) {
motionY = entityLivingBase.motionY; motionY = entityLivingBase.motionY;
} }

View file

@ -119,6 +119,7 @@ public class WorldHandler {
} }
AbstractSequencer.updateTick(); AbstractSequencer.updateTick();
LivingHandler.updateTick();
} }
@SubscribeEvent @SubscribeEvent