--- a/net/minecraft/server/ChunkMapDistance.java +++ b/net/minecraft/server/ChunkMapDistance.java @@ -32,11 +32,11 @@ private static final Logger LOGGER = LogManager.getLogger(); private static final int b = 33 + ChunkStatus.a(ChunkStatus.FULL) - 2; private final Long2ObjectMap> c = new Long2ObjectOpenHashMap(); - private final Long2ObjectOpenHashMap>> tickets = new Long2ObjectOpenHashMap(); + public final Long2ObjectOpenHashMap>> tickets = new Long2ObjectOpenHashMap(); // CraftBukkit - private -> public private final ChunkMapDistance.a e = new ChunkMapDistance.a(); private final ChunkMapDistance.b f = new ChunkMapDistance.b(8); private final ChunkMapDistance.c g = new ChunkMapDistance.c(33); - private final Set h = Sets.newHashSet(); + private final Set h = Sets.newHashSet(); // PAIL pendingChunkUpdates private final PlayerChunk.c i; private final Mailbox> j; private final Mailbox k; @@ -61,7 +61,7 @@ while (objectiterator.hasNext()) { Entry>> entry = (Entry) objectiterator.next(); - if (((ObjectSortedSet) entry.getValue()).removeIf((ticket) -> { + if ((entry.getValue()).removeIf((ticket) -> { // Craftbukkit - decompile error return ticket.a(this.currentTick); })) { this.e.b(entry.getLongKey(), this.a((ObjectSortedSet) entry.getValue()), false); @@ -99,10 +99,25 @@ } if (!this.h.isEmpty()) { - this.h.forEach((playerchunk) -> { + // CraftBukkit start + // Iterate pending chunk updates with protection against concurrent modification exceptions + java.util.Iterator iter = this.h.iterator(); + int expectedSize = this.h.size(); + do { + PlayerChunk playerchunk = iter.next(); + iter.remove(); + expectedSize--; + playerchunk.a(playerchunkmap); - }); - this.h.clear(); + + // Reset iterator if set was modified using add() + if (this.h.size() != expectedSize) { + expectedSize = this.h.size(); + iter = this.h.iterator(); + } + } while (iter.hasNext()); + // CraftBukkit end + return true; } else { if (!this.l.isEmpty()) { @@ -124,7 +139,7 @@ completablefuture.thenAccept((either) -> { this.m.execute(() -> { - this.k.a((Object) ChunkTaskQueueSorter.a(() -> { + this.k.a(ChunkTaskQueueSorter.a(() -> { // Craftbukkit - decompile error }, j, false)); }); }); @@ -138,7 +153,7 @@ } } - private void a(long i, Ticket ticket) { + private boolean a(long i, Ticket ticket) { // CraftBukkit - void -> boolean // PAIL addTicket ObjectSortedSet> objectsortedset = this.e(i); ObjectBidirectionalIterator> objectbidirectionaliterator = objectsortedset.iterator(); int j; @@ -149,21 +164,24 @@ j = PlayerChunkMap.GOLDEN_TICKET + 1; } + boolean ret = false; // CraftBukkit if (objectsortedset.add(ticket)) { - ; + ret = true; // CraftBukkit } if (ticket.b() < j) { this.e.b(i, ticket.b(), true); } + return ret; // CraftBukkit } - private void b(long i, Ticket ticket) { + private boolean b(long i, Ticket ticket) { // CraftBukkit - void -> boolean // PAIL removeTicket ObjectSortedSet> objectsortedset = this.e(i); + boolean removed = false; // CraftBukkit if (objectsortedset.remove(ticket)) { - ; + removed = true; // CraftBukkit } if (objectsortedset.isEmpty()) { @@ -171,16 +189,27 @@ } this.e.b(i, this.a(objectsortedset), false); + return removed; // CraftBukkit } public void a(TicketType tickettype, ChunkCoordIntPair chunkcoordintpair, int i, T t0) { - this.a(chunkcoordintpair.pair(), new Ticket<>(tickettype, i, t0, this.currentTick)); + // CraftBukkit start + this.addTicketAtLevel(tickettype, chunkcoordintpair, i, t0); + } + public boolean addTicketAtLevel(TicketType ticketType, ChunkCoordIntPair chunkPos, int level, T identifier) { + return this.a(chunkPos.pair(), new Ticket<>(ticketType, level, identifier, this.currentTick)); + // CraftBukkit end } public void b(TicketType tickettype, ChunkCoordIntPair chunkcoordintpair, int i, T t0) { - Ticket ticket = new Ticket<>(tickettype, i, t0, this.currentTick); + // CraftBukkit start + this.removeTicketAtLevel(tickettype, chunkcoordintpair, i, t0); + } + public boolean removeTicketAtLevel(TicketType ticketType, ChunkCoordIntPair chunkPos, int level, T identifier) { + Ticket ticket = new Ticket<>(ticketType, level, identifier, this.currentTick); - this.b(chunkcoordintpair.pair(), ticket); + return this.b(chunkPos.pair(), ticket); + // CraftBukkit end } public void addTicket(TicketType tickettype, ChunkCoordIntPair chunkcoordintpair, int i, T t0) { @@ -247,6 +276,21 @@ return this.f.a.containsKey(i); } + // CraftBukkit start + public void removeAllTicketsFor(TicketType ticketType, int ticketLevel, T ticketIdentifier) { + Ticket target = new Ticket<>(ticketType, ticketLevel, ticketIdentifier, this.currentTick); + + for (java.util.Iterator>> iterator = this.tickets.values().iterator(); iterator.hasNext();) { + ObjectSortedSet> tickets = iterator.next(); + tickets.remove(target); + + if (tickets.isEmpty()) { + iterator.remove(); + } + } + } + // CraftBukkit end + class a extends ChunkMap { public a() { @@ -333,7 +377,7 @@ Ticket ticket = new Ticket<>(TicketType.PLAYER, ChunkMapDistance.b, new ChunkCoordIntPair(i), ChunkMapDistance.this.currentTick); if (flag1) { - ChunkMapDistance.this.j.a((Object) ChunkTaskQueueSorter.a(() -> { + ChunkMapDistance.this.j.a(ChunkTaskQueueSorter.a(() -> { // Craftbukkit - decompile error ChunkMapDistance.this.m.execute(() -> { ChunkMapDistance.this.a(i, ticket); ChunkMapDistance.this.l.add(i); @@ -342,7 +386,7 @@ return j; })); } else { - ChunkMapDistance.this.k.a((Object) ChunkTaskQueueSorter.a(() -> { + ChunkMapDistance.this.k.a(ChunkTaskQueueSorter.a(() -> { // Craftbukkit - decompile error ChunkMapDistance.this.m.execute(() -> { ChunkMapDistance.this.b(i, ticket); });