mirror of
https://github.com/Minestom/Minestom.git
synced 2025-01-07 08:58:20 +01:00
Renamed BatchThread.java
This commit is contained in:
parent
defdbea29b
commit
0a837d2714
@ -1,7 +1,7 @@
|
||||
package net.minestom.server.entity.acquirable;
|
||||
|
||||
import net.minestom.server.entity.Entity;
|
||||
import net.minestom.server.thread.BatchThread;
|
||||
import net.minestom.server.thread.TickThread;
|
||||
import net.minestom.server.thread.ThreadProvider;
|
||||
import net.minestom.server.utils.consumer.EntityConsumer;
|
||||
import org.jetbrains.annotations.ApiStatus;
|
||||
@ -61,7 +61,7 @@ public class AcquirableEntity {
|
||||
*/
|
||||
public void acquire(@NotNull EntityConsumer consumer) {
|
||||
final Thread currentThread = Thread.currentThread();
|
||||
final BatchThread elementThread = getHandler().getBatchThread();
|
||||
final TickThread elementThread = getHandler().getBatchThread();
|
||||
Acquisition.acquire(currentThread, elementThread, () -> consumer.accept(unwrap()));
|
||||
}
|
||||
|
||||
@ -75,7 +75,7 @@ public class AcquirableEntity {
|
||||
*/
|
||||
public boolean tryAcquire(@NotNull EntityConsumer consumer) {
|
||||
final Thread currentThread = Thread.currentThread();
|
||||
final BatchThread elementThread = getHandler().getBatchThread();
|
||||
final TickThread elementThread = getHandler().getBatchThread();
|
||||
if (Objects.equals(currentThread, elementThread)) {
|
||||
consumer.accept(unwrap());
|
||||
return true;
|
||||
@ -91,7 +91,7 @@ public class AcquirableEntity {
|
||||
*/
|
||||
public @Nullable Entity tryAcquire() {
|
||||
final Thread currentThread = Thread.currentThread();
|
||||
final BatchThread elementThread = getHandler().getBatchThread();
|
||||
final TickThread elementThread = getHandler().getBatchThread();
|
||||
if (Objects.equals(currentThread, elementThread)) {
|
||||
return unwrap();
|
||||
}
|
||||
@ -143,7 +143,7 @@ public class AcquirableEntity {
|
||||
this.chunkEntry = chunkEntry;
|
||||
}
|
||||
|
||||
public BatchThread getBatchThread() {
|
||||
public TickThread getBatchThread() {
|
||||
return chunkEntry != null ? chunkEntry.getThread() : null;
|
||||
}
|
||||
}
|
||||
|
@ -1,7 +1,7 @@
|
||||
package net.minestom.server.entity.acquirable;
|
||||
|
||||
import net.minestom.server.entity.Entity;
|
||||
import net.minestom.server.thread.BatchThread;
|
||||
import net.minestom.server.thread.TickThread;
|
||||
import org.jetbrains.annotations.NotNull;
|
||||
import org.jetbrains.annotations.Nullable;
|
||||
|
||||
@ -32,15 +32,15 @@ public final class Acquisition {
|
||||
public static void acquireForEach(@NotNull Collection<AcquirableEntity> collection,
|
||||
@NotNull Consumer<Entity> consumer) {
|
||||
final Thread currentThread = Thread.currentThread();
|
||||
Map<BatchThread, List<Entity>> threadCacheMap = retrieveOptionalThreadMap(collection, currentThread, consumer);
|
||||
Map<TickThread, List<Entity>> threadCacheMap = retrieveOptionalThreadMap(collection, currentThread, consumer);
|
||||
|
||||
// Acquire all the threads one by one
|
||||
{
|
||||
for (Map.Entry<BatchThread, List<Entity>> entry : threadCacheMap.entrySet()) {
|
||||
final BatchThread batchThread = entry.getKey();
|
||||
for (Map.Entry<TickThread, List<Entity>> entry : threadCacheMap.entrySet()) {
|
||||
final TickThread tickThread = entry.getKey();
|
||||
final List<Entity> entities = entry.getValue();
|
||||
|
||||
acquire(currentThread, batchThread, () -> {
|
||||
acquire(currentThread, tickThread, () -> {
|
||||
for (Entity entity : entities) {
|
||||
consumer.accept(entity);
|
||||
}
|
||||
@ -76,7 +76,7 @@ public final class Acquisition {
|
||||
/**
|
||||
* Ensures that {@code callback} is safely executed inside the batch thread.
|
||||
*/
|
||||
protected static void acquire(@NotNull Thread currentThread, @Nullable BatchThread elementThread,
|
||||
protected static void acquire(@NotNull Thread currentThread, @Nullable TickThread elementThread,
|
||||
@NotNull Runnable callback) {
|
||||
if (Objects.equals(currentThread, elementThread)) {
|
||||
callback.run();
|
||||
@ -87,14 +87,14 @@ public final class Acquisition {
|
||||
}
|
||||
}
|
||||
|
||||
protected static ReentrantLock acquireEnter(Thread currentThread, BatchThread elementThread) {
|
||||
protected static ReentrantLock acquireEnter(Thread currentThread, TickThread elementThread) {
|
||||
// Monitoring
|
||||
long time = System.nanoTime();
|
||||
|
||||
ReentrantLock currentLock;
|
||||
{
|
||||
final BatchThread current = currentThread instanceof BatchThread ?
|
||||
(BatchThread) currentThread : null;
|
||||
final TickThread current = currentThread instanceof TickThread ?
|
||||
(TickThread) currentThread : null;
|
||||
currentLock = current != null && current.getLock().isHeldByCurrentThread() ?
|
||||
current.getLock() : null;
|
||||
}
|
||||
@ -118,7 +118,7 @@ public final class Acquisition {
|
||||
return !acquired ? lock : null;
|
||||
}
|
||||
|
||||
protected static ReentrantLock acquireEnter(BatchThread elementThread) {
|
||||
protected static ReentrantLock acquireEnter(TickThread elementThread) {
|
||||
return acquireEnter(Thread.currentThread(), elementThread);
|
||||
}
|
||||
|
||||
@ -145,23 +145,23 @@ public final class Acquisition {
|
||||
* @param consumer the consumer to execute when an element is already in the current thread
|
||||
* @return a new Thread to acquirable elements map
|
||||
*/
|
||||
protected static Map<BatchThread, List<Entity>> retrieveOptionalThreadMap(@NotNull Collection<AcquirableEntity> collection,
|
||||
protected static Map<TickThread, List<Entity>> retrieveOptionalThreadMap(@NotNull Collection<AcquirableEntity> collection,
|
||||
@NotNull Thread currentThread,
|
||||
@NotNull Consumer<? super Entity> consumer) {
|
||||
// Separate a collection of acquirable elements into a map of thread->elements
|
||||
// Useful to reduce the number of acquisition
|
||||
|
||||
Map<BatchThread, List<Entity>> threadCacheMap = new HashMap<>();
|
||||
Map<TickThread, List<Entity>> threadCacheMap = new HashMap<>();
|
||||
for (AcquirableEntity element : collection) {
|
||||
final Entity value = element.unwrap();
|
||||
|
||||
final BatchThread elementThread = element.getHandler().getBatchThread();
|
||||
final TickThread elementThread = element.getHandler().getBatchThread();
|
||||
if (currentThread == elementThread) {
|
||||
// The element is managed in the current thread, consumer can be immediately called
|
||||
consumer.accept(value);
|
||||
} else {
|
||||
// The element is manager in a different thread, cache it
|
||||
List<Entity> threadCacheList = threadCacheMap.computeIfAbsent(elementThread, batchThread -> new ArrayList<>());
|
||||
List<Entity> threadCacheList = threadCacheMap.computeIfAbsent(elementThread, tickThread -> new ArrayList<>());
|
||||
threadCacheList.add(value);
|
||||
}
|
||||
}
|
||||
@ -169,15 +169,15 @@ public final class Acquisition {
|
||||
return threadCacheMap;
|
||||
}
|
||||
|
||||
protected static Map<BatchThread, List<Entity>> retrieveThreadMap(@NotNull Collection<AcquirableEntity> collection) {
|
||||
protected static Map<TickThread, List<Entity>> retrieveThreadMap(@NotNull Collection<AcquirableEntity> collection) {
|
||||
// Separate a collection of acquirable elements into a map of thread->elements
|
||||
// Useful to reduce the number of acquisition
|
||||
Map<BatchThread, List<Entity>> threadCacheMap = new HashMap<>();
|
||||
Map<TickThread, List<Entity>> threadCacheMap = new HashMap<>();
|
||||
for (AcquirableEntity acquirableEntity : collection) {
|
||||
final Entity entity = acquirableEntity.unwrap();
|
||||
final BatchThread elementThread = acquirableEntity.getHandler().getBatchThread();
|
||||
final TickThread elementThread = acquirableEntity.getHandler().getBatchThread();
|
||||
|
||||
List<Entity> threadCacheList = threadCacheMap.computeIfAbsent(elementThread, batchThread -> new ArrayList<>());
|
||||
List<Entity> threadCacheList = threadCacheMap.computeIfAbsent(elementThread, tickThread -> new ArrayList<>());
|
||||
threadCacheList.add(entity);
|
||||
}
|
||||
|
||||
|
@ -24,9 +24,9 @@ import java.util.stream.Collectors;
|
||||
*/
|
||||
public abstract class ThreadProvider {
|
||||
|
||||
private final List<BatchThread> threads;
|
||||
private final List<TickThread> threads;
|
||||
|
||||
private final Map<BatchThread, Set<ChunkEntry>> threadChunkMap = new HashMap<>();
|
||||
private final Map<TickThread, Set<ChunkEntry>> threadChunkMap = new HashMap<>();
|
||||
private final Map<Chunk, ChunkEntry> chunkEntryMap = new HashMap<>();
|
||||
// Iterated over to refresh the thread used to update entities & chunks
|
||||
private final ArrayDeque<Chunk> chunks = new ArrayDeque<>();
|
||||
@ -40,11 +40,11 @@ public abstract class ThreadProvider {
|
||||
this.threads = new ArrayList<>(threadCount);
|
||||
|
||||
for (int i = 0; i < threadCount; i++) {
|
||||
final BatchThread.BatchRunnable batchRunnable = new BatchThread.BatchRunnable();
|
||||
final BatchThread batchThread = new BatchThread(batchRunnable, i);
|
||||
this.threads.add(batchThread);
|
||||
final TickThread.BatchRunnable batchRunnable = new TickThread.BatchRunnable();
|
||||
final TickThread tickThread = new TickThread(batchRunnable, i);
|
||||
this.threads.add(tickThread);
|
||||
|
||||
batchThread.start();
|
||||
tickThread.start();
|
||||
}
|
||||
}
|
||||
|
||||
@ -126,17 +126,17 @@ public abstract class ThreadProvider {
|
||||
return;
|
||||
chunks.remove(chunkEntry);
|
||||
|
||||
setChunkThread(chunk, batchThread -> {
|
||||
chunkEntry.thread = batchThread;
|
||||
setChunkThread(chunk, tickThread -> {
|
||||
chunkEntry.thread = tickThread;
|
||||
return chunkEntry;
|
||||
});
|
||||
}
|
||||
|
||||
protected @NotNull ChunkEntry setChunkThread(@NotNull Chunk chunk,
|
||||
@NotNull Function<@NotNull BatchThread, @NotNull ChunkEntry> chunkEntrySupplier) {
|
||||
@NotNull Function<@NotNull TickThread, @NotNull ChunkEntry> chunkEntrySupplier) {
|
||||
final int threadId = getThreadId(chunk);
|
||||
BatchThread thread = threads.get(threadId);
|
||||
var chunks = threadChunkMap.computeIfAbsent(thread, batchThread -> ConcurrentHashMap.newKeySet());
|
||||
TickThread thread = threads.get(threadId);
|
||||
var chunks = threadChunkMap.computeIfAbsent(thread, tickThread -> ConcurrentHashMap.newKeySet());
|
||||
|
||||
ChunkEntry chunkEntry = chunkEntrySupplier.apply(thread);
|
||||
chunks.add(chunkEntry);
|
||||
@ -146,7 +146,7 @@ public abstract class ThreadProvider {
|
||||
protected void removeChunk(Chunk chunk) {
|
||||
ChunkEntry chunkEntry = chunkEntryMap.get(chunk);
|
||||
if (chunkEntry != null) {
|
||||
BatchThread thread = chunkEntry.thread;
|
||||
TickThread thread = chunkEntry.thread;
|
||||
var chunks = threadChunkMap.get(thread);
|
||||
if (chunks != null) {
|
||||
chunks.remove(chunkEntry);
|
||||
@ -167,7 +167,7 @@ public abstract class ThreadProvider {
|
||||
*/
|
||||
public synchronized @NotNull CountDownLatch update(long time) {
|
||||
CountDownLatch countDownLatch = new CountDownLatch(threads.size());
|
||||
for (BatchThread thread : threads) {
|
||||
for (TickThread thread : threads) {
|
||||
final var chunkEntries = threadChunkMap.get(thread);
|
||||
if (chunkEntries == null || chunkEntries.isEmpty()) {
|
||||
// The thread never had any task
|
||||
@ -176,7 +176,7 @@ public abstract class ThreadProvider {
|
||||
}
|
||||
|
||||
// Execute tick
|
||||
thread.getMainRunnable().startTick(countDownLatch, () -> {
|
||||
thread.runnable.startTick(countDownLatch, () -> {
|
||||
final var entitiesList = chunkEntries.stream().map(chunkEntry -> chunkEntry.entities).collect(Collectors.toList());
|
||||
final var entities = entitiesList.stream()
|
||||
.flatMap(Collection::stream)
|
||||
@ -270,10 +270,10 @@ public abstract class ThreadProvider {
|
||||
}
|
||||
|
||||
public void shutdown() {
|
||||
this.threads.forEach(BatchThread::shutdown);
|
||||
this.threads.forEach(TickThread::shutdown);
|
||||
}
|
||||
|
||||
public @NotNull List<@NotNull BatchThread> getThreads() {
|
||||
public @NotNull List<@NotNull TickThread> getThreads() {
|
||||
return threads;
|
||||
}
|
||||
|
||||
@ -326,16 +326,16 @@ public abstract class ThreadProvider {
|
||||
}
|
||||
|
||||
public static class ChunkEntry {
|
||||
private volatile BatchThread thread;
|
||||
private volatile TickThread thread;
|
||||
private final Chunk chunk;
|
||||
private final List<Entity> entities = new ArrayList<>();
|
||||
|
||||
private ChunkEntry(BatchThread thread, Chunk chunk) {
|
||||
private ChunkEntry(TickThread thread, Chunk chunk) {
|
||||
this.thread = thread;
|
||||
this.chunk = chunk;
|
||||
}
|
||||
|
||||
public @NotNull BatchThread getThread() {
|
||||
public @NotNull TickThread getThread() {
|
||||
return thread;
|
||||
}
|
||||
|
||||
|
@ -11,35 +11,44 @@ import java.util.concurrent.atomic.AtomicReference;
|
||||
import java.util.concurrent.locks.LockSupport;
|
||||
import java.util.concurrent.locks.ReentrantLock;
|
||||
|
||||
public class BatchThread extends Thread {
|
||||
/**
|
||||
* Thread responsible for ticking {@link net.minestom.server.instance.Chunk chunks} and {@link net.minestom.server.entity.Entity entities}.
|
||||
* <p>
|
||||
* Created in {@link ThreadProvider}, and awaken every tick with a task to execute.
|
||||
*/
|
||||
public class TickThread extends Thread {
|
||||
|
||||
private final BatchRunnable runnable;
|
||||
protected final BatchRunnable runnable;
|
||||
private final ReentrantLock lock = new ReentrantLock();
|
||||
|
||||
public BatchThread(@NotNull BatchRunnable runnable, int number) {
|
||||
public TickThread(@NotNull BatchRunnable runnable, int number) {
|
||||
super(runnable, MinecraftServer.THREAD_NAME_TICK + "-" + number);
|
||||
this.runnable = runnable;
|
||||
|
||||
this.runnable.setLinkedThread(this);
|
||||
}
|
||||
|
||||
public @NotNull BatchRunnable getMainRunnable() {
|
||||
return runnable;
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the lock used to ensure the safety of entity acquisition.
|
||||
*
|
||||
* @return the thread lock
|
||||
*/
|
||||
public @NotNull ReentrantLock getLock() {
|
||||
return lock;
|
||||
}
|
||||
|
||||
/**
|
||||
* Shutdowns the thread. Cannot be undone.
|
||||
*/
|
||||
public void shutdown() {
|
||||
this.runnable.stop = true;
|
||||
LockSupport.unpark(this);
|
||||
}
|
||||
|
||||
public static class BatchRunnable implements Runnable {
|
||||
protected static class BatchRunnable implements Runnable {
|
||||
|
||||
private volatile boolean stop;
|
||||
private BatchThread batchThread;
|
||||
private TickThread tickThread;
|
||||
|
||||
private volatile boolean inTick;
|
||||
private final AtomicReference<CountDownLatch> countDownLatch = new AtomicReference<>();
|
||||
@ -48,9 +57,9 @@ public class BatchThread extends Thread {
|
||||
|
||||
@Override
|
||||
public void run() {
|
||||
Check.notNull(batchThread, "The linked BatchThread cannot be null!");
|
||||
Check.notNull(tickThread, "The linked BatchThread cannot be null!");
|
||||
while (!stop) {
|
||||
LockSupport.park(batchThread);
|
||||
LockSupport.park(tickThread);
|
||||
if (stop)
|
||||
break;
|
||||
CountDownLatch localCountDownLatch = this.countDownLatch.get();
|
||||
@ -79,15 +88,15 @@ public class BatchThread extends Thread {
|
||||
public synchronized void startTick(@NotNull CountDownLatch countDownLatch, @NotNull Runnable runnable) {
|
||||
this.countDownLatch.set(countDownLatch);
|
||||
this.queue.add(runnable);
|
||||
LockSupport.unpark(batchThread);
|
||||
LockSupport.unpark(tickThread);
|
||||
}
|
||||
|
||||
public boolean isInTick() {
|
||||
return inTick;
|
||||
}
|
||||
|
||||
private void setLinkedThread(BatchThread batchThread) {
|
||||
this.batchThread = batchThread;
|
||||
private void setLinkedThread(TickThread tickThread) {
|
||||
this.tickThread = tickThread;
|
||||
}
|
||||
}
|
||||
|
@ -1,8 +1,9 @@
|
||||
package net.minestom.server.utils.consumer;
|
||||
|
||||
import net.minestom.server.entity.Entity;
|
||||
import org.jetbrains.annotations.NotNull;
|
||||
|
||||
@FunctionalInterface
|
||||
public interface EntityConsumer {
|
||||
void accept(Entity entity);
|
||||
void accept(@NotNull Entity entity);
|
||||
}
|
||||
|
@ -1,7 +1,8 @@
|
||||
package net.minestom.server.utils.consumer;
|
||||
|
||||
import net.minestom.server.entity.Player;
|
||||
import org.jetbrains.annotations.NotNull;
|
||||
|
||||
public interface PlayerConsumer extends EntityConsumer {
|
||||
void accept(Player player);
|
||||
void accept(@NotNull Player player);
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user