151 lines
4.7 KiB
Java
151 lines
4.7 KiB
Java
package net.minecraft.commands.execution;
|
|
|
|
import com.google.common.collect.Queues;
|
|
import com.mojang.brigadier.context.ContextChain;
|
|
import com.mojang.logging.LogUtils;
|
|
import it.unimi.dsi.fastutil.objects.ObjectArrayList;
|
|
import java.util.Deque;
|
|
import java.util.List;
|
|
import net.minecraft.commands.CommandResultCallback;
|
|
import net.minecraft.commands.ExecutionCommandSource;
|
|
import net.minecraft.commands.execution.tasks.BuildContexts;
|
|
import net.minecraft.commands.execution.tasks.CallFunction;
|
|
import net.minecraft.commands.functions.InstantiatedFunction;
|
|
import net.minecraft.util.profiling.ProfilerFiller;
|
|
import org.jetbrains.annotations.Nullable;
|
|
import org.slf4j.Logger;
|
|
|
|
public class ExecutionContext<T> implements AutoCloseable {
|
|
private static final int MAX_QUEUE_DEPTH = 10000000;
|
|
private static final Logger LOGGER = LogUtils.getLogger();
|
|
private final int commandLimit;
|
|
private final int forkLimit;
|
|
private final ProfilerFiller profiler;
|
|
@Nullable
|
|
private TraceCallbacks tracer;
|
|
private int commandQuota;
|
|
private boolean queueOverflow;
|
|
private final Deque<CommandQueueEntry<T>> commandQueue = Queues.<CommandQueueEntry<T>>newArrayDeque();
|
|
private final List<CommandQueueEntry<T>> newTopCommands = new ObjectArrayList<>();
|
|
private int currentFrameDepth;
|
|
|
|
public ExecutionContext(int commandLimit, int forkLimit, ProfilerFiller profiler) {
|
|
this.commandLimit = commandLimit;
|
|
this.forkLimit = forkLimit;
|
|
this.profiler = profiler;
|
|
this.commandQuota = commandLimit;
|
|
}
|
|
|
|
private static <T extends ExecutionCommandSource<T>> Frame createTopFrame(ExecutionContext<T> executionContext, CommandResultCallback returnValueConsumer) {
|
|
if (executionContext.currentFrameDepth == 0) {
|
|
return new Frame(0, returnValueConsumer, executionContext.commandQueue::clear);
|
|
} else {
|
|
int i = executionContext.currentFrameDepth + 1;
|
|
return new Frame(i, returnValueConsumer, executionContext.frameControlForDepth(i));
|
|
}
|
|
}
|
|
|
|
public static <T extends ExecutionCommandSource<T>> void queueInitialFunctionCall(
|
|
ExecutionContext<T> executionContext, InstantiatedFunction<T> function, T source, CommandResultCallback returnValueConsumer
|
|
) {
|
|
executionContext.queueNext(
|
|
new CommandQueueEntry<>(createTopFrame(executionContext, returnValueConsumer), new CallFunction<>(function, source.callback(), false).bind(source))
|
|
);
|
|
}
|
|
|
|
public static <T extends ExecutionCommandSource<T>> void queueInitialCommandExecution(
|
|
ExecutionContext<T> executionContext, String commandInput, ContextChain<T> command, T source, CommandResultCallback returnValueConsumer
|
|
) {
|
|
executionContext.queueNext(
|
|
new CommandQueueEntry<>(createTopFrame(executionContext, returnValueConsumer), new BuildContexts.TopLevel<>(commandInput, command, source))
|
|
);
|
|
}
|
|
|
|
private void handleQueueOverflow() {
|
|
this.queueOverflow = true;
|
|
this.newTopCommands.clear();
|
|
this.commandQueue.clear();
|
|
}
|
|
|
|
public void queueNext(CommandQueueEntry<T> entry) {
|
|
if (this.newTopCommands.size() + this.commandQueue.size() > 10000000) {
|
|
this.handleQueueOverflow();
|
|
}
|
|
|
|
if (!this.queueOverflow) {
|
|
this.newTopCommands.add(entry);
|
|
}
|
|
}
|
|
|
|
public void discardAtDepthOrHigher(int depth) {
|
|
while (!this.commandQueue.isEmpty() && ((CommandQueueEntry)this.commandQueue.peek()).frame().depth() >= depth) {
|
|
this.commandQueue.removeFirst();
|
|
}
|
|
}
|
|
|
|
public Frame.FrameControl frameControlForDepth(int depth) {
|
|
return () -> this.discardAtDepthOrHigher(depth);
|
|
}
|
|
|
|
public void runCommandQueue() {
|
|
this.pushNewCommands();
|
|
|
|
while (true) {
|
|
if (this.commandQuota <= 0) {
|
|
LOGGER.info("Command execution stopped due to limit (executed {} commands)", this.commandLimit);
|
|
break;
|
|
}
|
|
|
|
CommandQueueEntry<T> commandQueueEntry = (CommandQueueEntry<T>)this.commandQueue.pollFirst();
|
|
if (commandQueueEntry == null) {
|
|
return;
|
|
}
|
|
|
|
this.currentFrameDepth = commandQueueEntry.frame().depth();
|
|
commandQueueEntry.execute(this);
|
|
if (this.queueOverflow) {
|
|
LOGGER.error("Command execution stopped due to command queue overflow (max {})", 10000000);
|
|
break;
|
|
}
|
|
|
|
this.pushNewCommands();
|
|
}
|
|
|
|
this.currentFrameDepth = 0;
|
|
}
|
|
|
|
private void pushNewCommands() {
|
|
for (int i = this.newTopCommands.size() - 1; i >= 0; i--) {
|
|
this.commandQueue.addFirst((CommandQueueEntry)this.newTopCommands.get(i));
|
|
}
|
|
|
|
this.newTopCommands.clear();
|
|
}
|
|
|
|
public void tracer(@Nullable TraceCallbacks tracer) {
|
|
this.tracer = tracer;
|
|
}
|
|
|
|
@Nullable
|
|
public TraceCallbacks tracer() {
|
|
return this.tracer;
|
|
}
|
|
|
|
public ProfilerFiller profiler() {
|
|
return this.profiler;
|
|
}
|
|
|
|
public int forkLimit() {
|
|
return this.forkLimit;
|
|
}
|
|
|
|
public void incrementCost() {
|
|
this.commandQuota--;
|
|
}
|
|
|
|
public void close() {
|
|
if (this.tracer != null) {
|
|
this.tracer.close();
|
|
}
|
|
}
|
|
}
|