Rename and clean up MapleMonsterAggroCoordinator

This commit is contained in:
P0nk
2021-09-09 21:32:52 +02:00
parent 4c803b62a8
commit e75139e5e6
3 changed files with 77 additions and 69 deletions

View File

@@ -36,57 +36,60 @@ import java.util.Map.Entry;
import java.util.concurrent.ScheduledFuture;
/**
*
* @author Ronan
*/
public class MapleMonsterAggroCoordinator {
public class MonsterAggroCoordinator {
private MonitoredReentrantLock lock = MonitoredReentrantLockFactory.createLock(MonitoredLockType.MAP_AGGRO);
private MonitoredReentrantLock idleLock = MonitoredReentrantLockFactory.createLock(MonitoredLockType.MAP_AGGRO_IDLE, true);
private final MonitoredReentrantLock idleLock = MonitoredReentrantLockFactory.createLock(MonitoredLockType.MAP_AGGRO_IDLE, true);
private long lastStopTime = Server.getInstance().getCurrentTime();
private ScheduledFuture<?> aggroMonitor = null;
private Map<MapleMonster, Map<Integer, PlayerAggroEntry>> mobAggroEntries = new HashMap<>();
private Map<MapleMonster, List<PlayerAggroEntry>> mobSortedAggros = new HashMap<>();
private Set<Integer> mapPuppetEntries = new HashSet<>();
private final Map<MapleMonster, Map<Integer, PlayerAggroEntry>> mobAggroEntries = new HashMap<>();
private final Map<MapleMonster, List<PlayerAggroEntry>> mobSortedAggros = new HashMap<>();
private final Set<Integer> mapPuppetEntries = new HashSet<>();
private class PlayerAggroEntry {
protected int cid;
protected int averageDamage = 0;
protected int currentDamageInstances = 0;
protected long accumulatedDamage = 0;
protected int expireStreak = 0;
protected int updateStreak = 0;
protected int toNextUpdate = 0;
protected int entryRank = -1;
protected PlayerAggroEntry(int cid) {
this.cid = cid;
}
}
public void stopAggroCoordinator() {
idleLock.lock();
try {
if (aggroMonitor == null) return;
if (aggroMonitor == null) {
return;
}
aggroMonitor.cancel(false);
aggroMonitor = null;
} finally {
idleLock.unlock();
}
lastStopTime = Server.getInstance().getCurrentTime();
}
public void startAggroCoordinator() {
idleLock.lock();
try {
if (aggroMonitor != null) return;
if (aggroMonitor != null) {
return;
}
aggroMonitor = TimerManager.getInstance().register(() -> {
runAggroUpdate(1);
runSortLeadingCharactersAggro();
@@ -94,17 +97,17 @@ public class MapleMonsterAggroCoordinator {
} finally {
idleLock.unlock();
}
int timeDelta = (int) Math.ceil((Server.getInstance().getCurrentTime() - lastStopTime) / YamlConfig.config.server.MOB_STATUS_AGGRO_INTERVAL);
if (timeDelta > 0) {
runAggroUpdate(timeDelta);
}
}
private static void updateEntryExpiration(PlayerAggroEntry pae) {
pae.toNextUpdate = (int) Math.ceil((120000L / YamlConfig.config.server.MOB_STATUS_AGGRO_INTERVAL) / Math.pow(2, pae.expireStreak + pae.currentDamageInstances));
}
private static void insertEntryDamage(PlayerAggroEntry pae, int damage) {
synchronized (pae) {
long totalDamage = pae.averageDamage;
@@ -116,16 +119,16 @@ public class MapleMonsterAggroCoordinator {
updateEntryExpiration(pae);
pae.currentDamageInstances += 1;
pae.averageDamage = (int)(totalDamage / pae.currentDamageInstances);
pae.averageDamage = (int) (totalDamage / pae.currentDamageInstances);
pae.accumulatedDamage = totalDamage;
}
}
private static boolean expiredAfterUpdateEntryDamage(PlayerAggroEntry pae, int deltaTime) {
synchronized (pae) {
pae.updateStreak += 1;
pae.toNextUpdate -= deltaTime;
if (pae.toNextUpdate <= 0) { // reached dmg instance expire time
pae.expireStreak += 1;
updateEntryExpiration(pae);
@@ -136,14 +139,16 @@ public class MapleMonsterAggroCoordinator {
}
pae.accumulatedDamage = pae.averageDamage * pae.currentDamageInstances;
}
return false;
}
}
public void addAggroDamage(MapleMonster mob, int cid, int damage) { // assumption: should not trigger after dispose()
if (!mob.isAlive()) return;
if (!mob.isAlive()) {
return;
}
List<PlayerAggroEntry> sortedAggro = mobSortedAggros.get(mob);
Map<Integer, PlayerAggroEntry> mobAggro = mobAggroEntries.get(mob);
if (mobAggro == null) {
@@ -153,7 +158,7 @@ public class MapleMonsterAggroCoordinator {
if (mobAggro == null) {
mobAggro = new HashMap<>();
mobAggroEntries.put(mob, mobAggro);
sortedAggro = new LinkedList<>();
mobSortedAggros.put(mob, sortedAggro);
} else {
@@ -166,15 +171,15 @@ public class MapleMonsterAggroCoordinator {
return;
}
}
PlayerAggroEntry aggroEntry = mobAggro.get(cid);
if (aggroEntry == null) {
aggroEntry = new PlayerAggroEntry(cid);
synchronized (mobAggro) {
synchronized (sortedAggro) {
PlayerAggroEntry mappedEntry = mobAggro.get(cid);
if (mappedEntry == null) {
mobAggro.put(aggroEntry.cid, aggroEntry);
sortedAggro.add(aggroEntry);
@@ -186,10 +191,10 @@ public class MapleMonsterAggroCoordinator {
} else if (damage < 1) {
return;
}
insertEntryDamage(aggroEntry, damage);
}
private void runAggroUpdate(int deltaTime) {
List<Pair<MapleMonster, Map<Integer, PlayerAggroEntry>>> aggroMobs = new LinkedList<>();
lock.lock();
@@ -200,11 +205,11 @@ public class MapleMonsterAggroCoordinator {
} finally {
lock.unlock();
}
for (Pair<MapleMonster, Map<Integer, PlayerAggroEntry>> am : aggroMobs) {
Map<Integer, PlayerAggroEntry> mobAggro = am.getRight();
List<PlayerAggroEntry> sortedAggro = mobSortedAggros.get(am.getLeft());
if (sortedAggro != null) {
List<Integer> toRemove = new LinkedList<>();
List<Integer> toRemoveIdx = new ArrayList<>(mobAggro.size());
@@ -215,8 +220,11 @@ public class MapleMonsterAggroCoordinator {
for (PlayerAggroEntry pae : mobAggro.values()) {
if (expiredAfterUpdateEntryDamage(pae, deltaTime)) {
toRemove.add(pae.cid);
if (pae.entryRank > -1) toRemoveIdx.add(pae.entryRank);
else toRemoveByFetch.add(pae.cid);
if (pae.entryRank > -1) {
toRemoveIdx.add(pae.entryRank);
} else {
toRemoveByFetch.add(pae.cid);
}
}
}
@@ -224,7 +232,7 @@ public class MapleMonsterAggroCoordinator {
for (Integer cid : toRemove) {
mobAggro.remove(cid);
}
if (mobAggro.isEmpty()) { // all aggro on this mob expired
if (!am.getLeft().isBoss()) {
am.getLeft().aggroResetAggro();
@@ -240,7 +248,7 @@ public class MapleMonsterAggroCoordinator {
sortedAggro.remove(idx);
}
}
if (!toRemoveByFetch.isEmpty()) {
for (Integer cid : toRemoveByFetch) {
for (int i = 0; i < sortedAggro.size(); i++) {
@@ -256,7 +264,7 @@ public class MapleMonsterAggroCoordinator {
}
}
}
private static void insertionSortAggroList(List<PlayerAggroEntry> paeList) {
for (int i = 1; i < paeList.size(); i++) {
PlayerAggroEntry pae = paeList.get(i);
@@ -280,23 +288,23 @@ public class MapleMonsterAggroCoordinator {
i += 1;
}
}
public boolean isLeadingCharacterAggro(MapleMonster mob, Character player) {
if (mob.isLeadingPuppetInVicinity()) {
return false;
} else if (mob.isCharacterPuppetInVicinity(player)) {
return true;
}
// by assuming the quasi-sorted nature of "mobAggroList", this method
// returns whether the player given as parameter can be elected as next aggro leader
List<PlayerAggroEntry> mobAggroList = mobSortedAggros.get(mob);
if (mobAggroList != null) {
synchronized (mobAggroList) {
mobAggroList = new ArrayList<>(mobAggroList.subList(0, Math.min(mobAggroList.size(), 5)));
}
MapleMap map = mob.getMap();
for (PlayerAggroEntry pae : mobAggroList) {
Character chr = map.getCharacterById(pae.cid);
@@ -309,10 +317,10 @@ public class MapleMonsterAggroCoordinator {
}
}
}
return false;
}
public void runSortLeadingCharactersAggro() {
List<List<PlayerAggroEntry>> aggroList;
lock.lock();
@@ -321,14 +329,14 @@ public class MapleMonsterAggroCoordinator {
} finally {
lock.unlock();
}
for (List<PlayerAggroEntry> mobAggroList : aggroList) {
synchronized (mobAggroList) {
insertionSortAggroList(mobAggroList);
}
}
}
public void removeAggroEntries(MapleMonster mob) {
lock.lock();
try {
@@ -338,28 +346,28 @@ public class MapleMonsterAggroCoordinator {
lock.unlock();
}
}
public void addPuppetAggro(Character player) {
synchronized (mapPuppetEntries) {
mapPuppetEntries.add(player.getId());
}
}
public void removePuppetAggro(Integer cid) {
synchronized (mapPuppetEntries) {
mapPuppetEntries.remove(cid);
}
}
public List<Integer> getPuppetAggroList() {
synchronized (mapPuppetEntries) {
return new ArrayList<>(mapPuppetEntries);
}
}
public void dispose() {
stopAggroCoordinator();
lock.lock();
try {
mobAggroEntries.clear();
@@ -367,14 +375,14 @@ public class MapleMonsterAggroCoordinator {
} finally {
lock.unlock();
}
disposeLocks();
}
private void disposeLocks() {
LockCollector.getInstance().registerDisposeAction(() -> emptyLocks());
}
private void emptyLocks() {
lock = lock.dispose();
}

View File

@@ -33,7 +33,7 @@ import net.server.audit.locks.MonitoredLockType;
import net.server.audit.locks.MonitoredReentrantLock;
import net.server.audit.locks.factory.MonitoredReentrantLockFactory;
import net.server.channel.Channel;
import net.server.coordinator.world.MapleMonsterAggroCoordinator;
import net.server.coordinator.world.MonsterAggroCoordinator;
import net.server.services.task.channel.MobAnimationService;
import net.server.services.task.channel.MobClearSkillService;
import net.server.services.task.channel.MobStatusService;
@@ -1417,7 +1417,7 @@ public class MapleMonster extends AbstractLoadedMapleLife {
return map;
}
public MapleMonsterAggroCoordinator getMapAggroCoordinator() {
public MonsterAggroCoordinator getMapAggroCoordinator() {
return map.getAggroCoordinator();
}
@@ -1927,7 +1927,7 @@ public class MapleMonster extends AbstractLoadedMapleLife {
}
public void aggroAddPuppet(Character player) {
MapleMonsterAggroCoordinator mmac = map.getAggroCoordinator();
MonsterAggroCoordinator mmac = map.getAggroCoordinator();
mmac.addPuppetAggro(player);
aggroUpdatePuppetController(player);
@@ -1938,7 +1938,7 @@ public class MapleMonster extends AbstractLoadedMapleLife {
}
public void aggroRemovePuppet(Character player) {
MapleMonsterAggroCoordinator mmac = map.getAggroCoordinator();
MonsterAggroCoordinator mmac = map.getAggroCoordinator();
mmac.removePuppetAggro(player.getId());
aggroUpdatePuppetController(null);
@@ -1985,7 +1985,7 @@ public class MapleMonster extends AbstractLoadedMapleLife {
}
if (newController == null || !isCharacterPuppetInVicinity(newController)) {
MapleMonsterAggroCoordinator mmac = map.getAggroCoordinator();
MonsterAggroCoordinator mmac = map.getAggroCoordinator();
List<Integer> puppetOwners = mmac.getPuppetAggroList();
List<Integer> toRemovePuppets = new LinkedList<>();
@@ -2074,7 +2074,7 @@ public class MapleMonster extends AbstractLoadedMapleLife {
*
*/
public void aggroMonsterDamage(Character attacker, int damage) {
MapleMonsterAggroCoordinator mmac = this.getMapAggroCoordinator();
MonsterAggroCoordinator mmac = this.getMapAggroCoordinator();
mmac.addAggroDamage(this, attacker.getId(), damage);
Character chrController = this.getController(); // aggro based on DPS rather than first-come-first-served, now live after suggestions thanks to MedicOP, Thora, Vcoc

View File

@@ -44,7 +44,7 @@ import net.server.audit.locks.factory.MonitoredReadLockFactory;
import net.server.audit.locks.factory.MonitoredReentrantLockFactory;
import net.server.audit.locks.factory.MonitoredWriteLockFactory;
import net.server.channel.Channel;
import net.server.coordinator.world.MapleMonsterAggroCoordinator;
import net.server.coordinator.world.MonsterAggroCoordinator;
import net.server.services.task.channel.FaceExpressionService;
import net.server.services.task.channel.MobMistService;
import net.server.services.task.channel.OverallService;
@@ -130,7 +130,7 @@ public class MapleMap {
private int fieldType;
private int fieldLimit = 0;
private int mobCapacity = -1;
private MapleMonsterAggroCoordinator aggroMonitor = null; // aggroMonitor activity in sync with itemMonitor
private MonsterAggroCoordinator aggroMonitor = null; // aggroMonitor activity in sync with itemMonitor
private ScheduledFuture<?> itemMonitor = null;
private ScheduledFuture<?> expireItemsTask = null;
private ScheduledFuture<?> mobSpawnLootTask = null;
@@ -183,7 +183,7 @@ public class MapleMap {
objectRLock = MonitoredReadLockFactory.createLock(objectLock);
objectWLock = MonitoredWriteLockFactory.createLock(objectLock);
aggroMonitor = new MapleMonsterAggroCoordinator();
aggroMonitor = new MonsterAggroCoordinator();
}
public void setEventInstance(EventInstanceManager eim) {
@@ -3052,7 +3052,7 @@ public class MapleMap {
mapArea.setBounds(vrLeft, vrTop, vrRight - vrLeft, vrBottom - vrTop);
}
public MapleMonsterAggroCoordinator getAggroCoordinator() {
public MonsterAggroCoordinator getAggroCoordinator() {
return aggroMonitor;
}