Full of red flags

- Add concept of flags to TaskExecutor.
- Can raise and lower flags from any thread.
- Add TaskExecutor#syncTo
  - Behaves much like #syncPoint, but exits early as soon as it detects
    that the requested flag has been raised.
- Document all methods in TaskExecutor.
- Do not discard tasks when destroying a VisualizationManagerImpl.
- Use flags in VisualizationManagerImpl to track frame plan and tick
  plan completion.
- Use flags in BatchingEngine to track stage buffering completion and
  flush completion.
  - Synchronization is now needed in BatchedDrawTracker#draw.
- Use flags in IndirectEngine and InstancingEngine to track flush
  completion.
- Add unit tests to validate flag behavior.
- Rename OnMainThreadPlan -> SyncedPlan.
This commit is contained in:
Jozufozu 2023-11-18 16:05:49 -08:00
parent e7a6052a84
commit 8126a63216
18 changed files with 419 additions and 67 deletions

1
.gitignore vendored
View file

@ -5,6 +5,7 @@ run/
build/ build/
gradle-app.setting gradle-app.setting
out/ out/
logs/
## IntelliJ IDEA ## IntelliJ IDEA

View file

@ -13,4 +13,22 @@ public enum RenderStage {
AFTER_TRANSLUCENT_TERRAIN, AFTER_TRANSLUCENT_TERRAIN,
AFTER_PARTICLES, AFTER_PARTICLES,
AFTER_WEATHER; AFTER_WEATHER;
/**
* Is this stage the last one to be rendered in the frame?
*
* @return {@code true} if no other RenderStages will be dispatched this frame.
*/
public boolean isLast() {
return this == values()[values().length - 1];
}
/**
* Is this stage the first one to be rendered in the frame?
*
* @return {@code true} if this is the first RenderStage to be dispatched this frame.
*/
public boolean isFirst() {
return this == values()[0];
}
} }

View file

@ -0,0 +1,12 @@
package com.jozufozu.flywheel.api.task;
/**
* Marker interface for flags that can be raised and lowered in a {@link TaskExecutor}.
* <br>
* <strong>Warning:</strong> flags will only be considered equal by reference.
* This is to allow multiple instances of the same high level structures to exist at once.
* <br>
* Keep this in mind when using records as flags.
*/
public interface Flag {
}

View file

@ -4,11 +4,66 @@ import java.util.concurrent.Executor;
public interface TaskExecutor extends Executor { public interface TaskExecutor extends Executor {
/** /**
* Wait for all running tasks to finish. * Wait for <em>all</em> running tasks to finish.
* <br>
* This is useful as a nuclear option, but most of the time you should
* try to use {@link Flag flags} and {@link #syncTo(Flag) syncTo}.
*/ */
void syncPoint(); void syncPoint();
/**
* Wait for running tasks, until the given Flag is {@link #raise raised}.
* <br>
* The flag will remain raised until {@link #lower lowered} manually.
*
* @param flag The flag to wait for.
* @return {@code true} if the flag was encountered. May return false if
* this executor runs out of tasks before the flag was raised.
*/
boolean syncTo(Flag flag);
/**
* Raise a flag indicating a key point in execution.
* <br>
* If the flag was already raised, this method does nothing.
*
* @param flag The flag to raise.
*/
void raise(Flag flag);
/**
* Lower a flag that may have been previously raised.
* <br>
* If the flag was never raised, this method does nothing.
*
* @param flag The flag to lower.
*/
void lower(Flag flag);
/**
* Check if a flag is raised without waiting for it.
*
* @param flag The flag to check.
* @return {@code true} if the flag is raised.
*/
boolean isRaised(Flag flag);
/**
* Check for the number of threads this executor uses.
* <br>
* May be helpful when determining how many chunks to divide a task into.
*
* @return The number of threads this executor uses.
*/
int getThreadCount(); int getThreadCount();
void scheduleForMainThread(Runnable runnable); /**
* Schedule a task to be run on the main thread.
* <br>
* This method may be called from any thread, but the runnable will only
* be executed once somebody calls either {@link #syncPoint()} or
* {@link #syncTo(Flag)}.
* @param runnable The task to run.
*/
void scheduleForSync(Runnable runnable);
} }

View file

@ -52,7 +52,15 @@ public class BatchedDrawTracker {
* @param stage The RenderStage to draw. * @param stage The RenderStage to draw.
*/ */
public void draw(RenderStage stage) { public void draw(RenderStage stage) {
Set<DrawBuffer> buffers = activeBuffers.get(stage); // This may appear jank, but flag synchronization in BatchingEngine guarantees that
// the mapped-to Set will not be modified here. We don't have the same guarantee for
// activeBuffers itself, so we need to synchronize to fetch the Set.
Set<DrawBuffer> buffers;
synchronized (activeBuffers) {
buffers = activeBuffers.get(stage);
}
for (DrawBuffer buffer : buffers) { for (DrawBuffer buffer : buffers) {
_draw(buffer); _draw(buffer);
} }

View file

@ -7,8 +7,10 @@ import java.util.Map;
import java.util.concurrent.atomic.AtomicInteger; import java.util.concurrent.atomic.AtomicInteger;
import com.jozufozu.flywheel.api.event.RenderStage; import com.jozufozu.flywheel.api.event.RenderStage;
import com.jozufozu.flywheel.api.task.Flag;
import com.jozufozu.flywheel.api.task.TaskExecutor; import com.jozufozu.flywheel.api.task.TaskExecutor;
import com.jozufozu.flywheel.lib.task.SimplyComposedPlan; import com.jozufozu.flywheel.lib.task.SimplyComposedPlan;
import com.jozufozu.flywheel.lib.task.StageFlag;
import com.jozufozu.flywheel.lib.task.Synchronizer; import com.jozufozu.flywheel.lib.task.Synchronizer;
import net.minecraft.client.renderer.RenderType; import net.minecraft.client.renderer.RenderType;
@ -17,24 +19,34 @@ import net.minecraft.client.renderer.RenderType;
* All the rendering that happens within a render stage. * All the rendering that happens within a render stage.
*/ */
public class BatchedStagePlan implements SimplyComposedPlan<BatchContext> { public class BatchedStagePlan implements SimplyComposedPlan<BatchContext> {
/**
* This flag will be raised when this stage completes execution.
*/
public final Flag flag;
private final RenderStage stage; private final RenderStage stage;
private final BatchedDrawTracker tracker; private final BatchedDrawTracker tracker;
private final Map<RenderType, BufferPlan> bufferPlans = new HashMap<>(); private final Map<RenderType, BufferPlan> bufferPlans = new HashMap<>();
public BatchedStagePlan(RenderStage renderStage, BatchedDrawTracker tracker) { public BatchedStagePlan(RenderStage stage, BatchedDrawTracker tracker) {
stage = renderStage; this.flag = new StageFlag(stage);
this.stage = stage;
this.tracker = tracker; this.tracker = tracker;
} }
@Override @Override
public void execute(TaskExecutor taskExecutor, BatchContext context, Runnable onCompletion) { public void execute(TaskExecutor taskExecutor, BatchContext context, Runnable onCompletion) {
if (isEmpty()) { if (isEmpty()) {
taskExecutor.raise(flag);
onCompletion.run(); onCompletion.run();
return; return;
} }
taskExecutor.execute(() -> { taskExecutor.execute(() -> {
var sync = new Synchronizer(bufferPlans.size(), onCompletion); var sync = new Synchronizer(bufferPlans.size(), () -> {
taskExecutor.raise(flag);
onCompletion.run();
});
for (var plan : bufferPlans.values()) { for (var plan : bufferPlans.values()) {
plan.execute(taskExecutor, context, sync); plan.execute(taskExecutor, context, sync);

View file

@ -13,10 +13,12 @@ import com.jozufozu.flywheel.api.instance.InstanceType;
import com.jozufozu.flywheel.api.instance.Instancer; import com.jozufozu.flywheel.api.instance.Instancer;
import com.jozufozu.flywheel.api.model.Mesh; import com.jozufozu.flywheel.api.model.Mesh;
import com.jozufozu.flywheel.api.model.Model; import com.jozufozu.flywheel.api.model.Model;
import com.jozufozu.flywheel.api.task.Flag;
import com.jozufozu.flywheel.api.task.Plan; import com.jozufozu.flywheel.api.task.Plan;
import com.jozufozu.flywheel.api.task.TaskExecutor; import com.jozufozu.flywheel.api.task.TaskExecutor;
import com.jozufozu.flywheel.backend.engine.AbstractEngine; import com.jozufozu.flywheel.backend.engine.AbstractEngine;
import com.jozufozu.flywheel.backend.engine.InstancerKey; import com.jozufozu.flywheel.backend.engine.InstancerKey;
import com.jozufozu.flywheel.lib.task.NamedFlag;
import com.jozufozu.flywheel.lib.task.SimplyComposedPlan; import com.jozufozu.flywheel.lib.task.SimplyComposedPlan;
import com.jozufozu.flywheel.lib.task.Synchronizer; import com.jozufozu.flywheel.lib.task.Synchronizer;
import com.mojang.blaze3d.vertex.VertexFormat; import com.mojang.blaze3d.vertex.VertexFormat;
@ -31,6 +33,8 @@ public class BatchingEngine extends AbstractEngine implements SimplyComposedPlan
private final Map<RenderStage, BatchedStagePlan> stagePlans = new EnumMap<>(RenderStage.class); private final Map<RenderStage, BatchedStagePlan> stagePlans = new EnumMap<>(RenderStage.class);
private final Map<VertexFormat, BatchedMeshPool> meshPools = new HashMap<>(); private final Map<VertexFormat, BatchedMeshPool> meshPools = new HashMap<>();
private final Flag flushFlag = new NamedFlag("flushed");
public BatchingEngine(int maxOriginDistance) { public BatchingEngine(int maxOriginDistance) {
super(maxOriginDistance); super(maxOriginDistance);
} }
@ -52,6 +56,9 @@ public class BatchingEngine extends AbstractEngine implements SimplyComposedPlan
public void execute(TaskExecutor taskExecutor, RenderContext context, Runnable onCompletion) { public void execute(TaskExecutor taskExecutor, RenderContext context, Runnable onCompletion) {
flush(); flush();
// Now it's safe to read stage plans in #renderStage.
taskExecutor.raise(flushFlag);
BatchContext ctx = BatchContext.create(context, renderOrigin); BatchContext ctx = BatchContext.create(context, renderOrigin);
var sync = new Synchronizer(stagePlans.values() var sync = new Synchronizer(stagePlans.values()
@ -69,7 +76,20 @@ public class BatchingEngine extends AbstractEngine implements SimplyComposedPlan
@Override @Override
public void renderStage(TaskExecutor executor, RenderContext context, RenderStage stage) { public void renderStage(TaskExecutor executor, RenderContext context, RenderStage stage) {
executor.syncPoint(); executor.syncTo(flushFlag);
if (stage.isLast()) {
executor.lower(flushFlag);
}
var stagePlan = stagePlans.get(stage);
if (stagePlan == null) {
drawTracker.draw(stage);
return;
}
executor.syncTo(stagePlan.flag);
executor.lower(stagePlan.flag);
drawTracker.draw(stage); drawTracker.draw(stage);
} }

View file

@ -8,18 +8,22 @@ import com.jozufozu.flywheel.api.instance.Instance;
import com.jozufozu.flywheel.api.instance.InstanceType; import com.jozufozu.flywheel.api.instance.InstanceType;
import com.jozufozu.flywheel.api.instance.Instancer; import com.jozufozu.flywheel.api.instance.Instancer;
import com.jozufozu.flywheel.api.model.Model; import com.jozufozu.flywheel.api.model.Model;
import com.jozufozu.flywheel.api.task.Flag;
import com.jozufozu.flywheel.api.task.Plan; import com.jozufozu.flywheel.api.task.Plan;
import com.jozufozu.flywheel.api.task.TaskExecutor; import com.jozufozu.flywheel.api.task.TaskExecutor;
import com.jozufozu.flywheel.backend.engine.AbstractEngine; import com.jozufozu.flywheel.backend.engine.AbstractEngine;
import com.jozufozu.flywheel.gl.GlStateTracker; import com.jozufozu.flywheel.gl.GlStateTracker;
import com.jozufozu.flywheel.gl.GlTextureUnit; import com.jozufozu.flywheel.gl.GlTextureUnit;
import com.jozufozu.flywheel.lib.task.OnMainThreadPlan; import com.jozufozu.flywheel.lib.task.NamedFlag;
import com.jozufozu.flywheel.lib.task.RaisePlan;
import com.jozufozu.flywheel.lib.task.SyncedPlan;
import com.mojang.blaze3d.systems.RenderSystem; import com.mojang.blaze3d.systems.RenderSystem;
import net.minecraft.client.Minecraft; import net.minecraft.client.Minecraft;
public class IndirectEngine extends AbstractEngine { public class IndirectEngine extends AbstractEngine {
private final IndirectDrawManager drawManager = new IndirectDrawManager(); private final IndirectDrawManager drawManager = new IndirectDrawManager();
private final Flag flushFlag = new NamedFlag("flushed");
public IndirectEngine(int maxOriginDistance) { public IndirectEngine(int maxOriginDistance) {
super(maxOriginDistance); super(maxOriginDistance);
@ -32,7 +36,8 @@ public class IndirectEngine extends AbstractEngine {
@Override @Override
public Plan<RenderContext> createFramePlan() { public Plan<RenderContext> createFramePlan() {
return OnMainThreadPlan.of(this::flushDrawManager); return SyncedPlan.<RenderContext>of(this::flushDrawManager)
.then(RaisePlan.raise(flushFlag));
} }
private void flushDrawManager() { private void flushDrawManager() {
@ -43,18 +48,23 @@ public class IndirectEngine extends AbstractEngine {
@Override @Override
public void renderStage(TaskExecutor executor, RenderContext context, RenderStage stage) { public void renderStage(TaskExecutor executor, RenderContext context, RenderStage stage) {
if (!drawManager.hasStage(stage)) { if (drawManager.hasStage(stage)) {
return; executor.syncTo(flushFlag);
try (var restoreState = GlStateTracker.getRestoreState()) {
setup();
for (var list : drawManager.renderLists.values()) {
list.submit(stage);
}
}
} }
executor.syncPoint(); if (stage.isLast()) {
// Need to sync here to ensure this frame has everything executed
try (var restoreState = GlStateTracker.getRestoreState()) { // in case we didn't have any stages to draw this frame.
setup(); executor.syncTo(flushFlag);
executor.lower(flushFlag);
for (var list : drawManager.renderLists.values()) {
list.submit(stage);
}
} }
} }

View file

@ -9,6 +9,7 @@ import com.jozufozu.flywheel.api.instance.Instance;
import com.jozufozu.flywheel.api.instance.InstanceType; import com.jozufozu.flywheel.api.instance.InstanceType;
import com.jozufozu.flywheel.api.instance.Instancer; import com.jozufozu.flywheel.api.instance.Instancer;
import com.jozufozu.flywheel.api.model.Model; import com.jozufozu.flywheel.api.model.Model;
import com.jozufozu.flywheel.api.task.Flag;
import com.jozufozu.flywheel.api.task.Plan; import com.jozufozu.flywheel.api.task.Plan;
import com.jozufozu.flywheel.api.task.TaskExecutor; import com.jozufozu.flywheel.api.task.TaskExecutor;
import com.jozufozu.flywheel.backend.compile.InstancingPrograms; import com.jozufozu.flywheel.backend.compile.InstancingPrograms;
@ -17,7 +18,9 @@ import com.jozufozu.flywheel.backend.engine.UniformBuffer;
import com.jozufozu.flywheel.gl.GlStateTracker; import com.jozufozu.flywheel.gl.GlStateTracker;
import com.jozufozu.flywheel.gl.GlTextureUnit; import com.jozufozu.flywheel.gl.GlTextureUnit;
import com.jozufozu.flywheel.lib.material.MaterialIndices; import com.jozufozu.flywheel.lib.material.MaterialIndices;
import com.jozufozu.flywheel.lib.task.OnMainThreadPlan; import com.jozufozu.flywheel.lib.task.NamedFlag;
import com.jozufozu.flywheel.lib.task.RaisePlan;
import com.jozufozu.flywheel.lib.task.SyncedPlan;
import com.mojang.blaze3d.systems.RenderSystem; import com.mojang.blaze3d.systems.RenderSystem;
import net.minecraft.client.Minecraft; import net.minecraft.client.Minecraft;
@ -26,6 +29,8 @@ public class InstancingEngine extends AbstractEngine {
private final Context context; private final Context context;
private final InstancedDrawManager drawManager = new InstancedDrawManager(); private final InstancedDrawManager drawManager = new InstancedDrawManager();
private final Flag flushFlag = new NamedFlag("flushed");
public InstancingEngine(int maxOriginDistance, Context context) { public InstancingEngine(int maxOriginDistance, Context context) {
super(maxOriginDistance); super(maxOriginDistance);
this.context = context; this.context = context;
@ -38,7 +43,8 @@ public class InstancingEngine extends AbstractEngine {
@Override @Override
public Plan<RenderContext> createFramePlan() { public Plan<RenderContext> createFramePlan() {
return OnMainThreadPlan.of(this::flushDrawManager); return SyncedPlan.<RenderContext>of(this::flushDrawManager)
.then(RaisePlan.raise(flushFlag));
} }
private void flushDrawManager() { private void flushDrawManager() {
@ -51,16 +57,21 @@ public class InstancingEngine extends AbstractEngine {
public void renderStage(TaskExecutor executor, RenderContext context, RenderStage stage) { public void renderStage(TaskExecutor executor, RenderContext context, RenderStage stage) {
var drawSet = drawManager.get(stage); var drawSet = drawManager.get(stage);
if (drawSet.isEmpty()) { if (!drawSet.isEmpty()) {
return; executor.syncTo(flushFlag);
try (var state = GlStateTracker.getRestoreState()) {
setup();
render(drawSet);
}
} }
executor.syncPoint(); if (stage.isLast()) {
// Need to sync here to ensure this frame has everything executed
try (var state = GlStateTracker.getRestoreState()) { // in case we didn't have any stages to draw this frame.
setup(); executor.syncTo(flushFlag);
executor.lower(flushFlag);
render(drawSet);
} }
} }

View file

@ -1,9 +1,11 @@
package com.jozufozu.flywheel.impl.task; package com.jozufozu.flywheel.impl.task;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Collections;
import java.util.Deque; import java.util.Deque;
import java.util.List; import java.util.List;
import java.util.Queue; import java.util.Queue;
import java.util.Set;
import java.util.concurrent.ConcurrentLinkedDeque; import java.util.concurrent.ConcurrentLinkedDeque;
import java.util.concurrent.ConcurrentLinkedQueue; import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.atomic.AtomicBoolean; import java.util.concurrent.atomic.AtomicBoolean;
@ -12,10 +14,12 @@ import org.jetbrains.annotations.NotNull;
import org.slf4j.Logger; import org.slf4j.Logger;
import com.jozufozu.flywheel.Flywheel; import com.jozufozu.flywheel.Flywheel;
import com.jozufozu.flywheel.api.task.Flag;
import com.jozufozu.flywheel.api.task.TaskExecutor; import com.jozufozu.flywheel.api.task.TaskExecutor;
import com.mojang.blaze3d.systems.RenderSystem; import com.mojang.blaze3d.systems.RenderSystem;
import com.mojang.logging.LogUtils; import com.mojang.logging.LogUtils;
import it.unimi.dsi.fastutil.objects.ReferenceOpenHashSet;
import net.minecraft.util.Mth; import net.minecraft.util.Mth;
// https://github.com/CaffeineMC/sodium-fabric/blob/5d364ed5ba63f9067fcf72a078ca310bff4db3e9/src/main/java/me/jellysquid/mods/sodium/client/render/chunk/compile/ChunkBuilder.java // https://github.com/CaffeineMC/sodium-fabric/blob/5d364ed5ba63f9067fcf72a078ca310bff4db3e9/src/main/java/me/jellysquid/mods/sodium/client/render/chunk/compile/ChunkBuilder.java
@ -35,6 +39,8 @@ public class ParallelTaskExecutor implements TaskExecutor {
private final Deque<Runnable> taskQueue = new ConcurrentLinkedDeque<>(); private final Deque<Runnable> taskQueue = new ConcurrentLinkedDeque<>();
private final Queue<Runnable> mainThreadQueue = new ConcurrentLinkedQueue<>(); private final Queue<Runnable> mainThreadQueue = new ConcurrentLinkedQueue<>();
private final Set<Flag> flags = Collections.synchronizedSet(new ReferenceOpenHashSet<>());
private final ThreadGroupNotifier taskNotifier = new ThreadGroupNotifier(); private final ThreadGroupNotifier taskNotifier = new ThreadGroupNotifier();
private final WaitGroup waitGroup = new WaitGroup(); private final WaitGroup waitGroup = new WaitGroup();
@ -116,7 +122,7 @@ public class ParallelTaskExecutor implements TaskExecutor {
} }
@Override @Override
public void scheduleForMainThread(Runnable runnable) { public void scheduleForSync(Runnable runnable) {
if (!running.get()) { if (!running.get()) {
throw new IllegalStateException("Executor is stopped"); throw new IllegalStateException("Executor is stopped");
} }
@ -133,40 +139,67 @@ public class ParallelTaskExecutor implements TaskExecutor {
*/ */
@Override @Override
public void syncPoint() { public void syncPoint() {
Runnable task;
while (true) { while (true) {
if ((task = mainThreadQueue.poll()) != null) { if (syncOneTask()) {
// Prioritize main thread tasks. // Done! Nothing left to do.
processMainThreadTask(task); break;
} else if ((task = taskQueue.pollLast()) != null) {
// then work on tasks from the queue.
processTask(task);
} else {
// then wait for the other threads to finish.
boolean done = waitGroup.await(10_000);
// If we timed-out tasks may have been added to the queue, so check again.
if (done && mainThreadQueue.isEmpty()) {
// if they didn't, we're done.
break;
}
} }
} }
} }
public void discardAndAwait() { @Override
public boolean syncTo(Flag flag) {
while (true) { while (true) {
// Discard everyone else's work... if (isRaised(flag)) {
while (taskQueue.pollLast() != null) { // The flag is already raised!
waitGroup.done(); // Early return with true to indicate.
return true;
} }
// ...wait for any stragglers... if (syncOneTask()) {
if (waitGroup.await(100_000)) { // Out of tasks entirely.
break; // The flag may have been raised though so return the result of isRaised.
return isRaised(flag);
} }
} }
// ...and clear the main thread queue. }
mainThreadQueue.clear();
/**
* Attempt to process a single task.
*
* @return {@code true} if the executor has nothing left to do.
*/
private boolean syncOneTask() {
Runnable task;
if ((task = mainThreadQueue.poll()) != null) {
// Prioritize main thread tasks.
processMainThreadTask(task);
} else if ((task = taskQueue.pollLast()) != null) {
// then work on tasks from the queue.
processTask(task);
} else {
// then wait for the other threads to finish.
boolean done = waitGroup.await(10_000);
// If we timed-out tasks may have been added to the queue, so check again.
// if they didn't, we're done.
return done && mainThreadQueue.isEmpty();
}
return false;
}
@Override
public void raise(Flag flag) {
flags.add(flag);
}
@Override
public void lower(Flag flag) {
flags.remove(flag);
}
@Override
public boolean isRaised(Flag flag) {
return flags.contains(flag);
} }
private void processTask(Runnable task) { private void processTask(Runnable task) {

View file

@ -1,10 +1,17 @@
package com.jozufozu.flywheel.impl.task; package com.jozufozu.flywheel.impl.task;
import java.util.Set;
import com.jozufozu.flywheel.api.task.Flag;
import com.jozufozu.flywheel.api.task.TaskExecutor; import com.jozufozu.flywheel.api.task.TaskExecutor;
import it.unimi.dsi.fastutil.objects.ReferenceOpenHashSet;
public class SerialTaskExecutor implements TaskExecutor { public class SerialTaskExecutor implements TaskExecutor {
public static final SerialTaskExecutor INSTANCE = new SerialTaskExecutor(); public static final SerialTaskExecutor INSTANCE = new SerialTaskExecutor();
private final Set<Flag> flags = new ReferenceOpenHashSet<>();
private SerialTaskExecutor() { private SerialTaskExecutor() {
} }
@ -14,7 +21,7 @@ public class SerialTaskExecutor implements TaskExecutor {
} }
@Override @Override
public void scheduleForMainThread(Runnable runnable) { public void scheduleForSync(Runnable runnable) {
runnable.run(); runnable.run();
} }
@ -22,6 +29,26 @@ public class SerialTaskExecutor implements TaskExecutor {
public void syncPoint() { public void syncPoint() {
} }
@Override
public boolean syncTo(Flag flag) {
return isRaised(flag);
}
@Override
public void raise(Flag flag) {
flags.add(flag);
}
@Override
public void lower(Flag flag) {
flags.remove(flag);
}
@Override
public boolean isRaised(Flag flag) {
return flags.contains(flag);
}
@Override @Override
public int getThreadCount() { public int getThreadCount() {
return 1; return 1;

View file

@ -7,6 +7,7 @@ import com.jozufozu.flywheel.api.backend.BackendManager;
import com.jozufozu.flywheel.api.backend.Engine; import com.jozufozu.flywheel.api.backend.Engine;
import com.jozufozu.flywheel.api.event.RenderContext; import com.jozufozu.flywheel.api.event.RenderContext;
import com.jozufozu.flywheel.api.event.RenderStage; import com.jozufozu.flywheel.api.event.RenderStage;
import com.jozufozu.flywheel.api.task.Flag;
import com.jozufozu.flywheel.api.task.Plan; import com.jozufozu.flywheel.api.task.Plan;
import com.jozufozu.flywheel.api.task.TaskExecutor; import com.jozufozu.flywheel.api.task.TaskExecutor;
import com.jozufozu.flywheel.api.visual.DynamicVisual; import com.jozufozu.flywheel.api.visual.DynamicVisual;
@ -22,7 +23,9 @@ import com.jozufozu.flywheel.impl.visualization.manager.BlockEntityVisualManager
import com.jozufozu.flywheel.impl.visualization.manager.EffectVisualManager; import com.jozufozu.flywheel.impl.visualization.manager.EffectVisualManager;
import com.jozufozu.flywheel.impl.visualization.manager.EntityVisualManager; import com.jozufozu.flywheel.impl.visualization.manager.EntityVisualManager;
import com.jozufozu.flywheel.lib.math.MatrixUtil; import com.jozufozu.flywheel.lib.math.MatrixUtil;
import com.jozufozu.flywheel.lib.task.NamedFlag;
import com.jozufozu.flywheel.lib.task.NestedPlan; import com.jozufozu.flywheel.lib.task.NestedPlan;
import com.jozufozu.flywheel.lib.task.RaisePlan;
import com.jozufozu.flywheel.lib.task.SimplyComposedPlan; import com.jozufozu.flywheel.lib.task.SimplyComposedPlan;
import com.jozufozu.flywheel.lib.util.LevelAttached; import com.jozufozu.flywheel.lib.util.LevelAttached;
@ -49,6 +52,9 @@ public class VisualizationManagerImpl implements VisualizationManager {
private final Plan<TickContext> tickPlan; private final Plan<TickContext> tickPlan;
private final Plan<RenderContext> framePlan; private final Plan<RenderContext> framePlan;
private final Flag tickFlag = new NamedFlag("tick");
private final Flag frameFlag = new NamedFlag("frame");
private VisualizationManagerImpl(LevelAccessor level) { private VisualizationManagerImpl(LevelAccessor level) {
engine = BackendManager.getBackend() engine = BackendManager.getBackend()
.createEngine(level); .createEngine(level);
@ -62,8 +68,9 @@ public class VisualizationManagerImpl implements VisualizationManager {
tickPlan = blockEntities.createTickPlan() tickPlan = blockEntities.createTickPlan()
.and(entities.createTickPlan()) .and(entities.createTickPlan())
.and(effects.createTickPlan()) .and(effects.createTickPlan())
.then(RaisePlan.raise(tickFlag))
.simplify(); .simplify();
framePlan = new FramePlan(); framePlan = new FramePlan().then(RaisePlan.raise(frameFlag));
} }
public static boolean supportsVisualization(@Nullable LevelAccessor level) { public static boolean supportsVisualization(@Nullable LevelAccessor level) {
@ -145,7 +152,12 @@ public class VisualizationManagerImpl implements VisualizationManager {
* </p> * </p>
*/ */
public void tick(double cameraX, double cameraY, double cameraZ) { public void tick(double cameraX, double cameraY, double cameraZ) {
taskExecutor.syncPoint(); // Make sure we're done with any prior frame or tick to avoid racing.
taskExecutor.syncTo(frameFlag);
taskExecutor.lower(frameFlag);
taskExecutor.syncTo(tickFlag);
taskExecutor.lower(tickFlag);
tickPlan.execute(taskExecutor, new TickContext(cameraX, cameraY, cameraZ)); tickPlan.execute(taskExecutor, new TickContext(cameraX, cameraY, cameraZ));
} }
@ -159,7 +171,9 @@ public class VisualizationManagerImpl implements VisualizationManager {
* </p> * </p>
*/ */
public void beginFrame(RenderContext context) { public void beginFrame(RenderContext context) {
taskExecutor.syncPoint(); // Make sure we're done with the last tick.
// Note we don't lower here because many frames may happen per tick.
taskExecutor.syncTo(tickFlag);
framePlan.execute(taskExecutor, context); framePlan.execute(taskExecutor, context);
} }
@ -175,7 +189,10 @@ public class VisualizationManagerImpl implements VisualizationManager {
* Free all acquired resources and delete this manager. * Free all acquired resources and delete this manager.
*/ */
public void delete() { public void delete() {
taskExecutor.discardAndAwait(); // Just finish everything. This may include the work of others but that's okay.
taskExecutor.syncPoint();
// Now clean up.
blockEntities.invalidate(); blockEntities.invalidate();
entities.invalidate(); entities.invalidate();
effects.invalidate(); effects.invalidate();

View file

@ -0,0 +1,11 @@
package com.jozufozu.flywheel.lib.task;
import com.jozufozu.flywheel.api.task.Flag;
/**
* A flag with an arbitrary name.
*
* @param name The name of the flag, mainly for debugging purposes.
*/
public record NamedFlag(String name) implements Flag {
}

View file

@ -0,0 +1,16 @@
package com.jozufozu.flywheel.lib.task;
import com.jozufozu.flywheel.api.task.Flag;
import com.jozufozu.flywheel.api.task.TaskExecutor;
public record RaisePlan<C>(Flag flag) implements SimplyComposedPlan<C> {
public static <C> RaisePlan<C> raise(Flag flag) {
return new RaisePlan<>(flag);
}
@Override
public void execute(TaskExecutor taskExecutor, C context, Runnable onCompletion) {
taskExecutor.raise(flag);
onCompletion.run();
}
}

View file

@ -0,0 +1,12 @@
package com.jozufozu.flywheel.lib.task;
import com.jozufozu.flywheel.api.event.RenderStage;
import com.jozufozu.flywheel.api.task.Flag;
/**
* A flag that is associated with a render stage.
* <br>
* Useful for synchronizing tasks for a specific render stage.
*/
public record StageFlag(RenderStage stage) implements Flag {
}

View file

@ -3,18 +3,18 @@ package com.jozufozu.flywheel.lib.task;
import com.jozufozu.flywheel.api.task.Plan; import com.jozufozu.flywheel.api.task.Plan;
import com.jozufozu.flywheel.api.task.TaskExecutor; import com.jozufozu.flywheel.api.task.TaskExecutor;
public record OnMainThreadPlan<C>(ContextConsumer<C> task) implements SimplyComposedPlan<C> { public record SyncedPlan<C>(ContextConsumer<C> task) implements SimplyComposedPlan<C> {
public static <C> Plan<C> of(ContextConsumer<C> task) { public static <C> Plan<C> of(ContextConsumer<C> task) {
return new OnMainThreadPlan<>(task); return new SyncedPlan<>(task);
} }
public static <C> Plan<C> of(ContextRunnable<C> task) { public static <C> Plan<C> of(ContextRunnable<C> task) {
return new OnMainThreadPlan<>(task); return new SyncedPlan<>(task);
} }
@Override @Override
public void execute(TaskExecutor taskExecutor, C context, Runnable onCompletion) { public void execute(TaskExecutor taskExecutor, C context, Runnable onCompletion) {
taskExecutor.scheduleForMainThread(() -> { taskExecutor.scheduleForSync(() -> {
task.accept(context); task.accept(context);
onCompletion.run(); onCompletion.run();
}); });

View file

@ -140,7 +140,7 @@ class PlanExecutionTest {
@Test @Test
void mainThreadPlan() { void mainThreadPlan() {
var done = new AtomicBoolean(false); var done = new AtomicBoolean(false);
var plan = OnMainThreadPlan.of(() -> done.set(true)); var plan = SyncedPlan.of(() -> done.set(true));
plan.execute(EXECUTOR, Unit.INSTANCE); plan.execute(EXECUTOR, Unit.INSTANCE);
@ -151,6 +151,95 @@ class PlanExecutionTest {
Assertions.assertTrue(done.get()); Assertions.assertTrue(done.get());
} }
@Test
void flagPlan() {
var first = new NamedFlag("ready right away");
var second = new NamedFlag("ready after we sync");
var sync = new Synchronizer(2, () -> EXECUTOR.raise(second));
RaisePlan.raise(first)
.execute(EXECUTOR, Unit.INSTANCE, sync);
Assertions.assertTrue(EXECUTOR.syncTo(first), "First flag should be raised since we submitted a plan that raises it");
Assertions.assertFalse(EXECUTOR.syncTo(second), "Second flag should not be raised yet.");
sync.decrementAndEventuallyRun();
Assertions.assertTrue(EXECUTOR.syncTo(second), "Second flag should be raised since it was raised in sync.");
}
@Test
void longWaitForFlag() {
var first = new NamedFlag("ready right away");
var second = new NamedFlag("ready after 2s");
RaisePlan.raise(first)
.then(SimplePlan.of(() -> {
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
}))
.then(RaisePlan.raise(second))
.execute(EXECUTOR, Unit.INSTANCE);
Assertions.assertTrue(EXECUTOR.syncTo(first), "First flag should be raised since we submitted a plan that raises it.");
Assertions.assertFalse(EXECUTOR.isRaised(second), "Second flag should not be raised immediately.");
Assertions.assertTrue(EXECUTOR.syncTo(second), "Second flag should be raised since we were waiting for it.");
}
@Test
void syncToReturnsExpected() {
var flag = new NamedFlag("ready right away");
Assertions.assertFalse(EXECUTOR.syncTo(flag), "Flag should not be raised yet.");
EXECUTOR.raise(flag);
Assertions.assertTrue(EXECUTOR.syncTo(flag), "Flag should be raised since we raised it manually.");
EXECUTOR.lower(flag);
Assertions.assertFalse(EXECUTOR.syncTo(flag), "Flag should not be raised since we lowered it.");
}
@Test
void flagsAreReferenceEqual() {
var flagA = new NamedFlag("same");
var flagB = new NamedFlag("same");
Assertions.assertNotSame(flagA, flagB, "Flags should not be the same object.");
Assertions.assertEquals(flagA, flagB, "Flags should be equal.");
Assertions.assertFalse(EXECUTOR.isRaised(flagA), "Flag A should not be raised yet.");
Assertions.assertFalse(EXECUTOR.isRaised(flagB), "Flag B should not be raised yet.");
EXECUTOR.raise(flagA);
Assertions.assertTrue(EXECUTOR.isRaised(flagA), "Flag A should be raised since we raised it manually.");
Assertions.assertFalse(EXECUTOR.isRaised(flagB), "Flag B should not be raised yet.");
EXECUTOR.raise(flagB);
Assertions.assertTrue(EXECUTOR.isRaised(flagA), "Flag A should be raised since we raised it manually.");
Assertions.assertTrue(EXECUTOR.isRaised(flagB), "Flag B should be raised since we raised it manually.");
EXECUTOR.lower(flagA);
Assertions.assertFalse(EXECUTOR.isRaised(flagA), "Flag A should not be raised since we lowered it.");
Assertions.assertTrue(EXECUTOR.isRaised(flagB), "Flag B should be raised since we raised it manually.");
EXECUTOR.lower(flagB);
Assertions.assertFalse(EXECUTOR.isRaised(flagA), "Flag A should not be raised since we lowered it.");
Assertions.assertFalse(EXECUTOR.isRaised(flagB), "Flag B should not be raised since we lowered it.");
}
private static void assertExpectedSequence(IntArrayList sequence, int... expected) { private static void assertExpectedSequence(IntArrayList sequence, int... expected) {
Assertions.assertArrayEquals(expected, sequence.toIntArray()); Assertions.assertArrayEquals(expected, sequence.toIntArray());
} }

View file

@ -36,7 +36,7 @@ public class PlanSimplificationTest {
Assertions.assertEquals(oneSimple.simplify(), SIMPLE); Assertions.assertEquals(oneSimple.simplify(), SIMPLE);
var mainThreadNoop = new OnMainThreadPlan<>(NOOP); var mainThreadNoop = new SyncedPlan<>(NOOP);
var oneMainThread = NestedPlan.of(mainThreadNoop); var oneMainThread = NestedPlan.of(mainThreadNoop);
Assertions.assertEquals(oneMainThread.simplify(), mainThreadNoop); Assertions.assertEquals(oneMainThread.simplify(), mainThreadNoop);
@ -66,7 +66,7 @@ public class PlanSimplificationTest {
@Test @Test
void complexNesting() { void complexNesting() {
var mainThreadNoop = OnMainThreadPlan.<Unit>of(() -> { var mainThreadNoop = SyncedPlan.<Unit>of(() -> {
}); });
var nested = NestedPlan.of(mainThreadNoop, SIMPLE); var nested = NestedPlan.of(mainThreadNoop, SIMPLE);
@ -79,7 +79,7 @@ public class PlanSimplificationTest {
@Test @Test
void nestedNoSimple() { void nestedNoSimple() {
var mainThreadNoop = OnMainThreadPlan.<Unit>of(() -> { var mainThreadNoop = SyncedPlan.<Unit>of(() -> {
}); });
var barrier = new BarrierPlan<>(SIMPLE, SIMPLE); var barrier = new BarrierPlan<>(SIMPLE, SIMPLE);
var oneMainThread = NestedPlan.of(mainThreadNoop, NestedPlan.of(mainThreadNoop, barrier, barrier)); var oneMainThread = NestedPlan.of(mainThreadNoop, NestedPlan.of(mainThreadNoop, barrier, barrier));