mirror of
https://github.com/Jozufozu/Flywheel.git
synced 2025-01-28 05:44:59 +01:00
State restoration confusion
- Do not wrap BeginFrameEvent and RenderStageEvent with GL restore state - Move GL state restoration into InstancingEngine - Remove BufferUploaderMixin - Add binding enum values to GlBufferType - Remove debugNormals config - Rename some occurrences of "world" to "level"
This commit is contained in:
parent
adb0b5c2a8
commit
94e792f25e
21 changed files with 99 additions and 141 deletions
|
@ -94,7 +94,7 @@ public class Flywheel {
|
||||||
forgeEventBus.addListener(ForgeEvents::unloadWorld);
|
forgeEventBus.addListener(ForgeEvents::unloadWorld);
|
||||||
forgeEventBus.addListener(ForgeEvents::tickLight);
|
forgeEventBus.addListener(ForgeEvents::tickLight);
|
||||||
|
|
||||||
forgeEventBus.addListener(EventPriority.LOWEST, RenderWork::onRenderWorldLast);
|
forgeEventBus.addListener(EventPriority.LOWEST, RenderWork::onRenderLevelLast);
|
||||||
|
|
||||||
modEventBus.addListener(PartialModel::onModelRegistry);
|
modEventBus.addListener(PartialModel::onModelRegistry);
|
||||||
modEventBus.addListener(PartialModel::onModelBake);
|
modEventBus.addListener(PartialModel::onModelBake);
|
||||||
|
|
|
@ -1,13 +1,13 @@
|
||||||
package com.jozufozu.flywheel.api;
|
package com.jozufozu.flywheel.api;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A marker interface custom worlds can override to indicate
|
* A marker interface custom levels can override to indicate
|
||||||
* that block entities and entities inside the world should
|
* that block entities and entities inside the level should
|
||||||
* render with Flywheel.
|
* render with Flywheel.
|
||||||
*
|
*
|
||||||
* {@link net.minecraft.client.Minecraft#level Minecraft#level} is special cased and will support Flywheel by default.
|
* {@link net.minecraft.client.Minecraft#level Minecraft#level} is special cased and will support Flywheel by default.
|
||||||
*/
|
*/
|
||||||
public interface FlywheelWorld {
|
public interface FlywheelLevel {
|
||||||
default boolean supportsFlywheel() {
|
default boolean supportsFlywheel() {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
|
@ -3,7 +3,7 @@ package com.jozufozu.flywheel.backend;
|
||||||
import org.jetbrains.annotations.Nullable;
|
import org.jetbrains.annotations.Nullable;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
|
|
||||||
import com.jozufozu.flywheel.api.FlywheelWorld;
|
import com.jozufozu.flywheel.api.FlywheelLevel;
|
||||||
import com.jozufozu.flywheel.backend.gl.versioned.GlCompat;
|
import com.jozufozu.flywheel.backend.gl.versioned.GlCompat;
|
||||||
import com.jozufozu.flywheel.backend.instancing.ParallelTaskEngine;
|
import com.jozufozu.flywheel.backend.instancing.ParallelTaskEngine;
|
||||||
import com.jozufozu.flywheel.config.BackendType;
|
import com.jozufozu.flywheel.config.BackendType;
|
||||||
|
@ -61,21 +61,21 @@ public class Backend {
|
||||||
return TYPE != BackendType.OFF;
|
return TYPE != BackendType.OFF;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static boolean canUseInstancing(@Nullable Level world) {
|
public static boolean canUseInstancing(@Nullable Level level) {
|
||||||
return isOn() && isFlywheelWorld(world);
|
return isOn() && isFlywheelLevel(level);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Used to avoid calling Flywheel functions on (fake) worlds that don't specifically support it.
|
* Used to avoid calling Flywheel functions on (fake) levels that don't specifically support it.
|
||||||
*/
|
*/
|
||||||
public static boolean isFlywheelWorld(@Nullable LevelAccessor world) {
|
public static boolean isFlywheelLevel(@Nullable LevelAccessor level) {
|
||||||
if (world == null) return false;
|
if (level == null) return false;
|
||||||
|
|
||||||
if (!world.isClientSide()) return false;
|
if (!level.isClientSide()) return false;
|
||||||
|
|
||||||
if (world instanceof FlywheelWorld && ((FlywheelWorld) world).supportsFlywheel()) return true;
|
if (level instanceof FlywheelLevel && ((FlywheelLevel) level).supportsFlywheel()) return true;
|
||||||
|
|
||||||
return world == Minecraft.getInstance().level;
|
return level == Minecraft.getInstance().level;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static boolean isGameActive() {
|
public static boolean isGameActive() {
|
||||||
|
|
|
@ -80,9 +80,9 @@ public class Loader implements ResourceManagerReloadListener {
|
||||||
|
|
||||||
Backend.LOGGER.info("Compiled all programs in " + StringUtil.formatTime(compileEnd - compileStart));
|
Backend.LOGGER.info("Compiled all programs in " + StringUtil.formatTime(compileEnd - compileStart));
|
||||||
|
|
||||||
ClientLevel world = Minecraft.getInstance().level;
|
ClientLevel level = Minecraft.getInstance().level;
|
||||||
if (Backend.canUseInstancing(world)) {
|
if (Backend.canUseInstancing(level)) {
|
||||||
InstancedRenderDispatcher.resetInstanceWorld(world);
|
InstancedRenderDispatcher.resetInstanceLevel(level);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,12 +6,11 @@ import java.util.concurrent.ConcurrentLinkedQueue;
|
||||||
import net.minecraftforge.client.event.RenderLevelLastEvent;
|
import net.minecraftforge.client.event.RenderLevelLastEvent;
|
||||||
|
|
||||||
public class RenderWork {
|
public class RenderWork {
|
||||||
private static final Queue<Runnable> runs = new ConcurrentLinkedQueue<>();
|
private static final Queue<Runnable> RUNS = new ConcurrentLinkedQueue<>();
|
||||||
|
|
||||||
|
public static void onRenderLevelLast(RenderLevelLastEvent event) {
|
||||||
public static void onRenderWorldLast(RenderLevelLastEvent event) {
|
while (!RUNS.isEmpty()) {
|
||||||
while (!runs.isEmpty()) {
|
RUNS.remove()
|
||||||
runs.remove()
|
|
||||||
.run();
|
.run();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -20,6 +19,6 @@ public class RenderWork {
|
||||||
* Queue work to be executed at the end of a frame
|
* Queue work to be executed at the end of a frame
|
||||||
*/
|
*/
|
||||||
public static void enqueue(Runnable run) {
|
public static void enqueue(Runnable run) {
|
||||||
runs.add(run);
|
RUNS.add(run);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,13 +7,12 @@ import com.mojang.blaze3d.platform.GlStateManager;
|
||||||
* Tracks bound buffers/vbos because GlStateManager doesn't do that for us.
|
* Tracks bound buffers/vbos because GlStateManager doesn't do that for us.
|
||||||
*/
|
*/
|
||||||
public class GlStateTracker {
|
public class GlStateTracker {
|
||||||
|
private static final int[] BUFFERS = new int[GlBufferType.values().length];
|
||||||
private static final int[] buffers = new int[GlBufferType.values().length];
|
|
||||||
private static int vao;
|
private static int vao;
|
||||||
private static int program;
|
private static int program;
|
||||||
|
|
||||||
public static int getBuffer(GlBufferType type) {
|
public static int getBuffer(GlBufferType type) {
|
||||||
return buffers[type.ordinal()];
|
return BUFFERS[type.ordinal()];
|
||||||
}
|
}
|
||||||
|
|
||||||
public static int getVertexArray() {
|
public static int getVertexArray() {
|
||||||
|
@ -25,7 +24,7 @@ public class GlStateTracker {
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void _setBuffer(GlBufferType type, int buffer) {
|
public static void _setBuffer(GlBufferType type, int buffer) {
|
||||||
buffers[type.ordinal()] = buffer;
|
BUFFERS[type.ordinal()] = buffer;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void _setProgram(int id) {
|
public static void _setProgram(int id) {
|
||||||
|
@ -37,7 +36,7 @@ public class GlStateTracker {
|
||||||
}
|
}
|
||||||
|
|
||||||
public static State getRestoreState() {
|
public static State getRestoreState() {
|
||||||
return new State(buffers.clone(), vao, program);
|
return new State(BUFFERS.clone(), vao, program);
|
||||||
}
|
}
|
||||||
|
|
||||||
public record State(int[] buffers, int vao, int program) implements AutoCloseable {
|
public record State(int[] buffers, int vao, int program) implements AutoCloseable {
|
||||||
|
@ -53,7 +52,7 @@ public class GlStateTracker {
|
||||||
GlBufferType[] values = GlBufferType.values();
|
GlBufferType[] values = GlBufferType.values();
|
||||||
|
|
||||||
for (int i = 0; i < values.length; i++) {
|
for (int i = 0; i < values.length; i++) {
|
||||||
if (buffers[i] != GlStateTracker.buffers[i]) {
|
if (buffers[i] != GlStateTracker.BUFFERS[i]) {
|
||||||
GlStateManager._glBindBuffer(values[i].glEnum, buffers[i]);
|
GlStateManager._glBindBuffer(values[i].glEnum, buffers[i]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,25 +12,27 @@ import com.jozufozu.flywheel.backend.gl.GlStateTracker;
|
||||||
import com.mojang.blaze3d.platform.GlStateManager;
|
import com.mojang.blaze3d.platform.GlStateManager;
|
||||||
|
|
||||||
public enum GlBufferType {
|
public enum GlBufferType {
|
||||||
ARRAY_BUFFER(GL15C.GL_ARRAY_BUFFER),
|
ARRAY_BUFFER(GL15C.GL_ARRAY_BUFFER, GL15C.GL_ARRAY_BUFFER_BINDING),
|
||||||
ELEMENT_ARRAY_BUFFER(GL15C.GL_ELEMENT_ARRAY_BUFFER),
|
ELEMENT_ARRAY_BUFFER(GL15C.GL_ELEMENT_ARRAY_BUFFER, GL15C.GL_ELEMENT_ARRAY_BUFFER_BINDING),
|
||||||
PIXEL_PACK_BUFFER(GL21.GL_PIXEL_PACK_BUFFER),
|
PIXEL_PACK_BUFFER(GL21.GL_PIXEL_PACK_BUFFER, GL21.GL_PIXEL_PACK_BUFFER_BINDING),
|
||||||
PIXEL_UNPACK_BUFFER(GL21.GL_PIXEL_UNPACK_BUFFER),
|
PIXEL_UNPACK_BUFFER(GL21.GL_PIXEL_UNPACK_BUFFER, GL21.GL_PIXEL_UNPACK_BUFFER_BINDING),
|
||||||
TRANSFORM_FEEDBACK_BUFFER(GL30.GL_TRANSFORM_FEEDBACK_BUFFER),
|
TRANSFORM_FEEDBACK_BUFFER(GL30.GL_TRANSFORM_FEEDBACK_BUFFER, GL30.GL_TRANSFORM_FEEDBACK_BUFFER_BINDING),
|
||||||
UNIFORM_BUFFER(GL31.GL_UNIFORM_BUFFER),
|
UNIFORM_BUFFER(GL31.GL_UNIFORM_BUFFER, GL31.GL_UNIFORM_BUFFER_BINDING),
|
||||||
TEXTURE_BUFFER(GL31.GL_TEXTURE_BUFFER),
|
TEXTURE_BUFFER(GL31.GL_TEXTURE_BUFFER, GL31.GL_TEXTURE_BUFFER),
|
||||||
COPY_READ_BUFFER(GL31.GL_COPY_READ_BUFFER),
|
COPY_READ_BUFFER(GL31.GL_COPY_READ_BUFFER, GL31.GL_COPY_READ_BUFFER),
|
||||||
COPY_WRITE_BUFFER(GL31.GL_COPY_WRITE_BUFFER),
|
COPY_WRITE_BUFFER(GL31.GL_COPY_WRITE_BUFFER, GL31.GL_COPY_WRITE_BUFFER),
|
||||||
DRAW_INDIRECT_BUFFER(GL40.GL_DRAW_INDIRECT_BUFFER),
|
DRAW_INDIRECT_BUFFER(GL40.GL_DRAW_INDIRECT_BUFFER, GL40.GL_DRAW_INDIRECT_BUFFER_BINDING),
|
||||||
ATOMIC_COUNTER_BUFFER(GL42.GL_ATOMIC_COUNTER_BUFFER),
|
ATOMIC_COUNTER_BUFFER(GL42.GL_ATOMIC_COUNTER_BUFFER, GL42.GL_ATOMIC_COUNTER_BUFFER_BINDING),
|
||||||
DISPATCH_INDIRECT_BUFFER(GL43.GL_DISPATCH_INDIRECT_BUFFER),
|
DISPATCH_INDIRECT_BUFFER(GL43.GL_DISPATCH_INDIRECT_BUFFER, GL43.GL_DISPATCH_INDIRECT_BUFFER_BINDING),
|
||||||
SHADER_STORAGE_BUFFER(GL43.GL_SHADER_STORAGE_BUFFER),
|
SHADER_STORAGE_BUFFER(GL43.GL_SHADER_STORAGE_BUFFER, GL43.GL_SHADER_STORAGE_BUFFER_BINDING),
|
||||||
;
|
;
|
||||||
|
|
||||||
public final int glEnum;
|
public final int glEnum;
|
||||||
|
public final int glBindingEnum;
|
||||||
|
|
||||||
GlBufferType(int glEnum) {
|
GlBufferType(int glEnum, int glBindingEnum) {
|
||||||
this.glEnum = glEnum;
|
this.glEnum = glEnum;
|
||||||
|
this.glBindingEnum = glBindingEnum;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static GlBufferType fromTarget(int pTarget) {
|
public static GlBufferType fromTarget(int pTarget) {
|
||||||
|
|
|
@ -142,10 +142,10 @@ public class InstanceWorld implements AutoCloseable {
|
||||||
/**
|
/**
|
||||||
* Instantiate all the necessary instances to render the given world.
|
* Instantiate all the necessary instances to render the given world.
|
||||||
*/
|
*/
|
||||||
public void loadEntities(ClientLevel world) {
|
public void loadEntities(ClientLevel level) {
|
||||||
// Block entities are loaded while chunks are baked.
|
// Block entities are loaded while chunks are baked.
|
||||||
// Entities are loaded with the world, so when chunks are reloaded they need to be re-added.
|
// Entities are loaded with the level, so when chunks are reloaded they need to be re-added.
|
||||||
ClientLevelExtension.getAllLoadedEntities(world)
|
ClientLevelExtension.getAllLoadedEntities(level)
|
||||||
.forEach(entities::add);
|
.forEach(entities::add);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -3,6 +3,7 @@ package com.jozufozu.flywheel.backend.instancing;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
import com.jozufozu.flywheel.backend.Backend;
|
import com.jozufozu.flywheel.backend.Backend;
|
||||||
|
import com.jozufozu.flywheel.backend.gl.GlStateTracker;
|
||||||
import com.jozufozu.flywheel.backend.instancing.effect.Effect;
|
import com.jozufozu.flywheel.backend.instancing.effect.Effect;
|
||||||
import com.jozufozu.flywheel.config.FlwCommands;
|
import com.jozufozu.flywheel.config.FlwCommands;
|
||||||
import com.jozufozu.flywheel.config.FlwConfig;
|
import com.jozufozu.flywheel.config.FlwConfig;
|
||||||
|
@ -77,11 +78,11 @@ public class InstancedRenderDispatcher {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
Minecraft mc = Minecraft.getInstance();
|
Minecraft mc = Minecraft.getInstance();
|
||||||
ClientLevel world = mc.level;
|
ClientLevel level = mc.level;
|
||||||
AnimationTickHolder.tick();
|
AnimationTickHolder.tick();
|
||||||
|
|
||||||
if (Backend.isOn()) {
|
if (Backend.isOn()) {
|
||||||
instanceWorlds.get(world)
|
instanceWorlds.get(level)
|
||||||
.tick();
|
.tick();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -94,22 +95,22 @@ public class InstancedRenderDispatcher {
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void onRenderStage(RenderStageEvent event) {
|
public static void onRenderStage(RenderStageEvent event) {
|
||||||
ClientLevel world = event.getContext().level();
|
ClientLevel level = event.getContext().level();
|
||||||
if (!Backend.canUseInstancing(world)) return;
|
if (!Backend.canUseInstancing(level)) return;
|
||||||
|
|
||||||
instanceWorlds.get(world).renderStage(event.getContext(), event.getStage());
|
instanceWorlds.get(level).renderStage(event.getContext(), event.getStage());
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void onReloadRenderers(ReloadRenderersEvent event) {
|
public static void onReloadRenderers(ReloadRenderersEvent event) {
|
||||||
ClientLevel world = event.getWorld();
|
ClientLevel level = event.getLevel();
|
||||||
if (Backend.isOn() && world != null) {
|
if (Backend.isOn() && level != null) {
|
||||||
resetInstanceWorld(world);
|
resetInstanceLevel(level);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void resetInstanceWorld(ClientLevel world) {
|
public static void resetInstanceLevel(ClientLevel level) {
|
||||||
instanceWorlds.replace(world, InstanceWorld::delete)
|
instanceWorlds.replace(level, InstanceWorld::delete)
|
||||||
.loadEntities(world);
|
.loadEntities(level);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void getDebugString(List<String> debug) {
|
public static void getDebugString(List<String> debug) {
|
||||||
|
|
|
@ -47,20 +47,20 @@ public class BlockEntityInstanceManager extends InstanceManager<BlockEntity> {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
Level world = blockEntity.getLevel();
|
Level level = blockEntity.getLevel();
|
||||||
|
|
||||||
if (world == null) {
|
if (level == null) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (world.isEmptyBlock(blockEntity.getBlockPos())) {
|
if (level.isEmptyBlock(blockEntity.getBlockPos())) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Backend.isFlywheelWorld(world)) {
|
if (Backend.isFlywheelLevel(level)) {
|
||||||
BlockPos pos = blockEntity.getBlockPos();
|
BlockPos pos = blockEntity.getBlockPos();
|
||||||
|
|
||||||
BlockGetter existingChunk = world.getChunkForCollisions(pos.getX() >> 4, pos.getZ() >> 4);
|
BlockGetter existingChunk = level.getChunkForCollisions(pos.getX() >> 4, pos.getZ() >> 4);
|
||||||
|
|
||||||
return existingChunk != null;
|
return existingChunk != null;
|
||||||
}
|
}
|
||||||
|
|
|
@ -42,12 +42,12 @@ public class EntityInstanceManager extends InstanceManager<Entity> {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
Level world = entity.level;
|
Level level = entity.level;
|
||||||
|
|
||||||
if (Backend.isFlywheelWorld(world)) {
|
if (Backend.isFlywheelLevel(level)) {
|
||||||
BlockPos pos = entity.blockPosition();
|
BlockPos pos = entity.blockPosition();
|
||||||
|
|
||||||
BlockGetter existingChunk = world.getChunkForCollisions(pos.getX() >> 4, pos.getZ() >> 4);
|
BlockGetter existingChunk = level.getChunkForCollisions(pos.getX() >> 4, pos.getZ() >> 4);
|
||||||
|
|
||||||
return existingChunk != null;
|
return existingChunk != null;
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,6 +13,7 @@ import com.jozufozu.flywheel.api.instancer.InstancedPart;
|
||||||
import com.jozufozu.flywheel.api.material.Material;
|
import com.jozufozu.flywheel.api.material.Material;
|
||||||
import com.jozufozu.flywheel.api.struct.StructType;
|
import com.jozufozu.flywheel.api.struct.StructType;
|
||||||
import com.jozufozu.flywheel.api.vertex.VertexType;
|
import com.jozufozu.flywheel.api.vertex.VertexType;
|
||||||
|
import com.jozufozu.flywheel.backend.gl.GlStateTracker;
|
||||||
import com.jozufozu.flywheel.backend.gl.GlTextureUnit;
|
import com.jozufozu.flywheel.backend.gl.GlTextureUnit;
|
||||||
import com.jozufozu.flywheel.backend.instancing.Engine;
|
import com.jozufozu.flywheel.backend.instancing.Engine;
|
||||||
import com.jozufozu.flywheel.backend.instancing.InstanceManager;
|
import com.jozufozu.flywheel.backend.instancing.InstanceManager;
|
||||||
|
@ -66,8 +67,10 @@ public class InstancingEngine implements Engine {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void beginFrame(TaskEngine taskEngine, RenderContext context) {
|
public void beginFrame(TaskEngine taskEngine, RenderContext context) {
|
||||||
|
try (var restoreState = GlStateTracker.getRestoreState()) {
|
||||||
drawManager.flush();
|
drawManager.flush();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void renderStage(TaskEngine taskEngine, RenderContext context, RenderStage stage) {
|
public void renderStage(TaskEngine taskEngine, RenderContext context, RenderStage stage) {
|
||||||
|
@ -77,10 +80,12 @@ public class InstancingEngine implements Engine {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
try (var restoreState = GlStateTracker.getRestoreState()) {
|
||||||
setup();
|
setup();
|
||||||
|
|
||||||
render(drawSet);
|
render(drawSet);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
protected void setup() {
|
protected void setup() {
|
||||||
GlTextureUnit.T2.makeActive();
|
GlTextureUnit.T2.makeActive();
|
||||||
|
|
|
@ -54,23 +54,6 @@ public class FlwCommands {
|
||||||
return Command.SINGLE_SUCCESS;
|
return Command.SINGLE_SUCCESS;
|
||||||
})));
|
})));
|
||||||
|
|
||||||
commandBuilder.addValue(config.client.debugNormals, "debugNormals", (builder, value) -> booleanValueCommand(builder, value,
|
|
||||||
(source, bool) -> {
|
|
||||||
LocalPlayer player = Minecraft.getInstance().player;
|
|
||||||
if (player == null) return;
|
|
||||||
|
|
||||||
Component text = new TextComponent("Normal debug mode is currently: ").append(boolToText(bool));
|
|
||||||
player.displayClientMessage(text, false);
|
|
||||||
},
|
|
||||||
(source, bool) -> {
|
|
||||||
LocalPlayer player = Minecraft.getInstance().player;
|
|
||||||
if (player == null) return;
|
|
||||||
|
|
||||||
Component text = boolToText(bool).append(new TextComponent(" normal debug mode").withStyle(ChatFormatting.WHITE));
|
|
||||||
player.displayClientMessage(text, false);
|
|
||||||
}
|
|
||||||
));
|
|
||||||
|
|
||||||
commandBuilder.addValue(config.client.limitUpdates, "limitUpdates", (builder, value) -> booleanValueCommand(builder, value,
|
commandBuilder.addValue(config.client.limitUpdates, "limitUpdates", (builder, value) -> booleanValueCommand(builder, value,
|
||||||
(source, bool) -> {
|
(source, bool) -> {
|
||||||
LocalPlayer player = Minecraft.getInstance().player;
|
LocalPlayer player = Minecraft.getInstance().player;
|
||||||
|
@ -90,7 +73,18 @@ public class FlwCommands {
|
||||||
}
|
}
|
||||||
));
|
));
|
||||||
|
|
||||||
commandBuilder.command.then(Commands.literal("debugCrumble")
|
// TODO
|
||||||
|
commandBuilder.command.then(Commands.literal("debugNormals"))
|
||||||
|
.executes(context -> {
|
||||||
|
LocalPlayer player = Minecraft.getInstance().player;
|
||||||
|
if (player == null) return 0;
|
||||||
|
|
||||||
|
player.displayClientMessage(new TextComponent("This command is not yet implemented."), false);
|
||||||
|
|
||||||
|
return Command.SINGLE_SUCCESS;
|
||||||
|
});
|
||||||
|
|
||||||
|
commandBuilder.command.then(Commands.literal("debugCrumbling")
|
||||||
.then(Commands.argument("pos", BlockPosArgument.blockPos())
|
.then(Commands.argument("pos", BlockPosArgument.blockPos())
|
||||||
.then(Commands.argument("stage", IntegerArgumentType.integer(0, 9))
|
.then(Commands.argument("stage", IntegerArgumentType.integer(0, 9))
|
||||||
.executes(context -> {
|
.executes(context -> {
|
||||||
|
@ -106,7 +100,7 @@ public class FlwCommands {
|
||||||
|
|
||||||
executor.level.destroyBlockProgress(executor.getId(), pos, value);
|
executor.level.destroyBlockProgress(executor.getId(), pos, value);
|
||||||
|
|
||||||
return 1;
|
return Command.SINGLE_SUCCESS;
|
||||||
}))));
|
}))));
|
||||||
|
|
||||||
commandBuilder.build(event.getDispatcher());
|
commandBuilder.build(event.getDispatcher());
|
||||||
|
|
|
@ -31,10 +31,6 @@ public class FlwConfig {
|
||||||
return client.backend.get();
|
return client.backend.get();
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean debugNormals() {
|
|
||||||
return client.debugNormals.get();
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean limitUpdates() {
|
public boolean limitUpdates() {
|
||||||
return client.limitUpdates.get();
|
return client.limitUpdates.get();
|
||||||
}
|
}
|
||||||
|
@ -44,16 +40,12 @@ public class FlwConfig {
|
||||||
|
|
||||||
public static class ClientConfig {
|
public static class ClientConfig {
|
||||||
public final EnumValue<BackendType> backend;
|
public final EnumValue<BackendType> backend;
|
||||||
public final BooleanValue debugNormals;
|
|
||||||
public final BooleanValue limitUpdates;
|
public final BooleanValue limitUpdates;
|
||||||
|
|
||||||
public ClientConfig(ForgeConfigSpec.Builder builder) {
|
public ClientConfig(ForgeConfigSpec.Builder builder) {
|
||||||
backend = builder.comment("Select the backend to use.")
|
backend = builder.comment("Select the backend to use.")
|
||||||
.defineEnum("backend", BackendType.INSTANCING);
|
.defineEnum("backend", BackendType.INSTANCING);
|
||||||
|
|
||||||
debugNormals = builder.comment("Enable or disable a debug overlay that colors pixels by their normal.")
|
|
||||||
.define("debugNormals", false);
|
|
||||||
|
|
||||||
limitUpdates = builder.comment("Enable or disable instance update limiting with distance.")
|
limitUpdates = builder.comment("Enable or disable instance update limiting with distance.")
|
||||||
.define("limitUpdates", true);
|
.define("limitUpdates", true);
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,7 +11,7 @@ import java.util.function.Predicate;
|
||||||
|
|
||||||
import org.jetbrains.annotations.Nullable;
|
import org.jetbrains.annotations.Nullable;
|
||||||
|
|
||||||
import com.jozufozu.flywheel.api.FlywheelWorld;
|
import com.jozufozu.flywheel.api.FlywheelLevel;
|
||||||
|
|
||||||
import net.minecraft.core.BlockPos;
|
import net.minecraft.core.BlockPos;
|
||||||
import net.minecraft.core.Direction;
|
import net.minecraft.core.Direction;
|
||||||
|
@ -43,7 +43,7 @@ import net.minecraft.world.level.storage.WritableLevelData;
|
||||||
import net.minecraft.world.scores.Scoreboard;
|
import net.minecraft.world.scores.Scoreboard;
|
||||||
import net.minecraft.world.ticks.LevelTickAccess;
|
import net.minecraft.world.ticks.LevelTickAccess;
|
||||||
|
|
||||||
public class VirtualRenderWorld extends Level implements FlywheelWorld {
|
public class VirtualRenderWorld extends Level implements FlywheelLevel {
|
||||||
public final Map<BlockPos, BlockState> blocksAdded = new HashMap<>();
|
public final Map<BlockPos, BlockState> blocksAdded = new HashMap<>();
|
||||||
public final Map<BlockPos, BlockEntity> besAdded = new HashMap<>();
|
public final Map<BlockPos, BlockEntity> besAdded = new HashMap<>();
|
||||||
public final Set<SectionPos> spannedSections = new HashSet<>();
|
public final Set<SectionPos> spannedSections = new HashSet<>();
|
||||||
|
|
|
@ -6,14 +6,14 @@ import net.minecraft.client.multiplayer.ClientLevel;
|
||||||
import net.minecraftforge.eventbus.api.Event;
|
import net.minecraftforge.eventbus.api.Event;
|
||||||
|
|
||||||
public class ReloadRenderersEvent extends Event {
|
public class ReloadRenderersEvent extends Event {
|
||||||
private final ClientLevel world;
|
private final ClientLevel level;
|
||||||
|
|
||||||
public ReloadRenderersEvent(ClientLevel world) {
|
public ReloadRenderersEvent(ClientLevel level) {
|
||||||
this.world = world;
|
this.level = level;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Nullable
|
@Nullable
|
||||||
public ClientLevel getWorld() {
|
public ClientLevel getLevel() {
|
||||||
return world;
|
return level;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,28 +0,0 @@
|
||||||
package com.jozufozu.flywheel.mixin;
|
|
||||||
|
|
||||||
import org.jetbrains.annotations.Nullable;
|
|
||||||
import org.spongepowered.asm.mixin.Mixin;
|
|
||||||
import org.spongepowered.asm.mixin.Shadow;
|
|
||||||
import org.spongepowered.asm.mixin.injection.At;
|
|
||||||
import org.spongepowered.asm.mixin.injection.Inject;
|
|
||||||
import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
|
|
||||||
|
|
||||||
import com.jozufozu.flywheel.backend.gl.GlStateTracker;
|
|
||||||
import com.mojang.blaze3d.vertex.BufferUploader;
|
|
||||||
import com.mojang.blaze3d.vertex.VertexFormat;
|
|
||||||
|
|
||||||
@Mixin(BufferUploader.class)
|
|
||||||
public class BufferUploaderMixin {
|
|
||||||
@Shadow
|
|
||||||
@Nullable
|
|
||||||
private static VertexFormat lastFormat;
|
|
||||||
|
|
||||||
// Stop BufferUploader from clearing buffer state if nothing is bound
|
|
||||||
@Inject(method = "reset", at = @At("HEAD"))
|
|
||||||
private static void flywheel$onReset(CallbackInfo ci) {
|
|
||||||
// Trust our tracker over BufferUploader's.
|
|
||||||
if (GlStateTracker.getVertexArray() == 0) {
|
|
||||||
lastFormat = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -12,7 +12,6 @@ import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
|
||||||
|
|
||||||
import com.jozufozu.flywheel.api.RenderStage;
|
import com.jozufozu.flywheel.api.RenderStage;
|
||||||
import com.jozufozu.flywheel.backend.Backend;
|
import com.jozufozu.flywheel.backend.Backend;
|
||||||
import com.jozufozu.flywheel.backend.gl.GlStateTracker;
|
|
||||||
import com.jozufozu.flywheel.core.RenderContext;
|
import com.jozufozu.flywheel.core.RenderContext;
|
||||||
import com.jozufozu.flywheel.event.BeginFrameEvent;
|
import com.jozufozu.flywheel.event.BeginFrameEvent;
|
||||||
import com.jozufozu.flywheel.event.ReloadRenderersEvent;
|
import com.jozufozu.flywheel.event.ReloadRenderersEvent;
|
||||||
|
@ -44,10 +43,8 @@ public class LevelRendererMixin {
|
||||||
private void flywheel$beginRender(PoseStack pPoseStack, float pPartialTick, long pFinishNanoTime, boolean pRenderBlockOutline, Camera pCamera, GameRenderer pGameRenderer, LightTexture pLightTexture, Matrix4f pProjectionMatrix, CallbackInfo ci) {
|
private void flywheel$beginRender(PoseStack pPoseStack, float pPartialTick, long pFinishNanoTime, boolean pRenderBlockOutline, Camera pCamera, GameRenderer pGameRenderer, LightTexture pLightTexture, Matrix4f pProjectionMatrix, CallbackInfo ci) {
|
||||||
flywheel$renderContext = new RenderContext((LevelRenderer) (Object) this, level, pPoseStack, RenderContext.createViewProjection(pPoseStack, pProjectionMatrix), pProjectionMatrix, renderBuffers, pCamera);
|
flywheel$renderContext = new RenderContext((LevelRenderer) (Object) this, level, pPoseStack, RenderContext.createViewProjection(pPoseStack, pProjectionMatrix), pProjectionMatrix, renderBuffers, pCamera);
|
||||||
|
|
||||||
try (var restoreState = GlStateTracker.getRestoreState()) {
|
|
||||||
MinecraftForge.EVENT_BUS.post(new BeginFrameEvent(flywheel$renderContext));
|
MinecraftForge.EVENT_BUS.post(new BeginFrameEvent(flywheel$renderContext));
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@Inject(at = @At("TAIL"), method = "renderLevel")
|
@Inject(at = @At("TAIL"), method = "renderLevel")
|
||||||
private void flywheel$endRender(PoseStack pPoseStack, float pPartialTick, long pFinishNanoTime, boolean pRenderBlockOutline, Camera pCamera, GameRenderer pGameRenderer, LightTexture pLightTexture, Matrix4f pProjectionMatrix, CallbackInfo ci) {
|
private void flywheel$endRender(PoseStack pPoseStack, float pPartialTick, long pFinishNanoTime, boolean pRenderBlockOutline, Camera pCamera, GameRenderer pGameRenderer, LightTexture pLightTexture, Matrix4f pProjectionMatrix, CallbackInfo ci) {
|
||||||
|
@ -74,11 +71,9 @@ public class LevelRendererMixin {
|
||||||
@Unique
|
@Unique
|
||||||
private void flywheel$dispatch(RenderStage stage) {
|
private void flywheel$dispatch(RenderStage stage) {
|
||||||
if (flywheel$renderContext != null) {
|
if (flywheel$renderContext != null) {
|
||||||
try (var restoreState = GlStateTracker.getRestoreState()) {
|
|
||||||
MinecraftForge.EVENT_BUS.post(new RenderStageEvent(flywheel$renderContext, stage));
|
MinecraftForge.EVENT_BUS.post(new RenderStageEvent(flywheel$renderContext, stage));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@Inject(method = "renderLevel", at = @At(value = "INVOKE_STRING", target = "Lnet/minecraft/util/profiling/ProfilerFiller;popPush(Ljava/lang/String;)V", args = "ldc=sky"))
|
@Inject(method = "renderLevel", at = @At(value = "INVOKE_STRING", target = "Lnet/minecraft/util/profiling/ProfilerFiller;popPush(Ljava/lang/String;)V", args = "ldc=sky"))
|
||||||
private void flywheel$onStage$beforeSky(CallbackInfo ci) {
|
private void flywheel$onStage$beforeSky(CallbackInfo ci) {
|
||||||
|
|
|
@ -18,14 +18,14 @@ public class NetworkLightUpdateMixin {
|
||||||
@Inject(at = @At("TAIL"), method = "handleLightUpdatePacket")
|
@Inject(at = @At("TAIL"), method = "handleLightUpdatePacket")
|
||||||
private void flywheel$onLightPacket(ClientboundLightUpdatePacket packet, CallbackInfo ci) {
|
private void flywheel$onLightPacket(ClientboundLightUpdatePacket packet, CallbackInfo ci) {
|
||||||
RenderWork.enqueue(() -> {
|
RenderWork.enqueue(() -> {
|
||||||
ClientLevel world = Minecraft.getInstance().level;
|
ClientLevel level = Minecraft.getInstance().level;
|
||||||
|
|
||||||
if (world == null) return;
|
if (level == null) return;
|
||||||
|
|
||||||
int chunkX = packet.getX();
|
int chunkX = packet.getX();
|
||||||
int chunkZ = packet.getZ();
|
int chunkZ = packet.getZ();
|
||||||
|
|
||||||
LightUpdater.get(world)
|
LightUpdater.get(level)
|
||||||
.onLightPacket(chunkX, chunkZ);
|
.onLightPacket(chunkX, chunkZ);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,7 +8,6 @@
|
||||||
"BlockEntityRenderDispatcherAccessor",
|
"BlockEntityRenderDispatcherAccessor",
|
||||||
"BlockEntityTypeMixin",
|
"BlockEntityTypeMixin",
|
||||||
"BufferBuilderMixin",
|
"BufferBuilderMixin",
|
||||||
"BufferUploaderMixin",
|
|
||||||
"ChunkRebuildHooksMixin",
|
"ChunkRebuildHooksMixin",
|
||||||
"ClientLevelMixin",
|
"ClientLevelMixin",
|
||||||
"EntityTypeMixin",
|
"EntityTypeMixin",
|
||||||
|
|
Loading…
Reference in a new issue