Improved fall damage handler
- allows entities with same id in different worlds - reduce memory footprint over time
This commit is contained in:
parent
49fcf2897b
commit
37f24ae882
3 changed files with 51 additions and 14 deletions
|
@ -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 ));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
|
@ -119,6 +119,7 @@ public class WorldHandler {
|
||||||
}
|
}
|
||||||
|
|
||||||
AbstractSequencer.updateTick();
|
AbstractSequencer.updateTick();
|
||||||
|
LivingHandler.updateTick();
|
||||||
}
|
}
|
||||||
|
|
||||||
@SubscribeEvent
|
@SubscribeEvent
|
||||||
|
|
Loading…
Reference in a new issue