From a62bc660607bfd7adcfe05767914135507ad1db4 Mon Sep 17 00:00:00 2001 From: Jozufozu Date: Wed, 12 Jan 2022 00:19:37 -0800 Subject: [PATCH] Better memoized shader compilation and global game state - Properly separate compilation of vertex and fragment shaders - Game state is no longer per-program - Needs organization --- .../flywheel/backend/GameStateRegistry.java | 26 +++++++ .../flywheel/backend/source/SourceFile.java | 4 ++ .../core/compile/FragmentCompiler.java | 54 ++++++++++++-- .../core/compile/ProgramCompiler.java | 6 +- .../flywheel/core/compile/ProgramContext.java | 72 ++++++++++--------- .../flywheel/core/compile/VertexCompiler.java | 49 +++++++++---- .../core/shader/GameStateProvider.java | 7 +- .../core/shader/NormalDebugStateProvider.java | 6 ++ .../flywheel/core/shader/ProgramSpec.java | 50 ++----------- .../flywheel/core/shader/ProgramState.java | 17 ----- .../flywheel/core/shader/StateSnapshot.java | 12 ++++ .../flywheel/flywheel/programs/model.json | 8 +-- .../flywheel/flywheel/programs/oriented.json | 8 +-- .../flywheel/flywheel/programs/passthru.json | 8 +-- 14 files changed, 187 insertions(+), 140 deletions(-) delete mode 100644 src/main/java/com/jozufozu/flywheel/core/shader/ProgramState.java create mode 100644 src/main/java/com/jozufozu/flywheel/core/shader/StateSnapshot.java diff --git a/src/main/java/com/jozufozu/flywheel/backend/GameStateRegistry.java b/src/main/java/com/jozufozu/flywheel/backend/GameStateRegistry.java index e0aa48885..2bdd425c0 100644 --- a/src/main/java/com/jozufozu/flywheel/backend/GameStateRegistry.java +++ b/src/main/java/com/jozufozu/flywheel/backend/GameStateRegistry.java @@ -3,7 +3,9 @@ package com.jozufozu.flywheel.backend; import java.util.HashMap; import java.util.Map; +import com.jozufozu.flywheel.core.compile.ShaderConstants; import com.jozufozu.flywheel.core.shader.GameStateProvider; +import com.jozufozu.flywheel.core.shader.StateSnapshot; import net.minecraft.resources.ResourceLocation; @@ -32,4 +34,28 @@ public class GameStateRegistry { registeredStateProviders.put(context.getID(), context); } + + public static StateSnapshot takeSnapshot() { + long ctx = 0; + for (GameStateProvider state : registeredStateProviders.values()) { + if (state.isTrue()) { + ctx |= 1; + } + ctx <<= 1; + } + return new StateSnapshot(ctx); + } + + public static ShaderConstants getDefines(long ctx) { + long stateID = ctx; + ShaderConstants shaderConstants = new ShaderConstants(); + + for (GameStateProvider state : registeredStateProviders.values()) { + if ((stateID & 1) == 1) { + state.alterConstants(shaderConstants); + } + stateID >>= 1; + } + return shaderConstants; + } } diff --git a/src/main/java/com/jozufozu/flywheel/backend/source/SourceFile.java b/src/main/java/com/jozufozu/flywheel/backend/source/SourceFile.java index e69653336..6f4b04e87 100644 --- a/src/main/java/com/jozufozu/flywheel/backend/source/SourceFile.java +++ b/src/main/java/com/jozufozu/flywheel/backend/source/SourceFile.java @@ -268,4 +268,8 @@ public class SourceFile { return -1; } + @Override + public String toString() { + return name.toString(); + } } diff --git a/src/main/java/com/jozufozu/flywheel/core/compile/FragmentCompiler.java b/src/main/java/com/jozufozu/flywheel/core/compile/FragmentCompiler.java index 80d8b2229..3eaf09a42 100644 --- a/src/main/java/com/jozufozu/flywheel/core/compile/FragmentCompiler.java +++ b/src/main/java/com/jozufozu/flywheel/core/compile/FragmentCompiler.java @@ -1,12 +1,14 @@ package com.jozufozu.flywheel.core.compile; +import java.util.Objects; + import com.jozufozu.flywheel.backend.gl.shader.GlShader; import com.jozufozu.flywheel.backend.gl.shader.ShaderType; import com.jozufozu.flywheel.backend.source.FileResolution; import com.jozufozu.flywheel.backend.source.SourceFile; -import com.jozufozu.flywheel.core.shader.ProgramSpec; +import com.jozufozu.flywheel.core.shader.StateSnapshot; -public class FragmentCompiler extends Memoizer { +public class FragmentCompiler extends Memoizer { private final FileResolution header; private final Template fragment; @@ -16,9 +18,8 @@ public class FragmentCompiler extends Memoizer { } @Override - protected GlShader _create(ProgramContext key) { - ProgramSpec spec = key.spec(); - SourceFile fragmentFile = spec.getFragmentFile(); + protected GlShader _create(Context key) { + SourceFile fragmentFile = key.file; FragmentTemplateData appliedTemplate = fragment.apply(fragmentFile); StringBuilder builder = new StringBuilder(); @@ -34,11 +35,52 @@ public class FragmentCompiler extends Memoizer { builder.append(appliedTemplate.generateFooter()); - return new GlShader(spec.name, ShaderType.FRAGMENT, builder.toString()); + return new GlShader(fragmentFile.name, ShaderType.FRAGMENT, builder.toString()); } @Override protected void _destroy(GlShader value) { value.delete(); } + + public static final class Context { + private final SourceFile file; + private final StateSnapshot ctx; + private final float alphaDiscard; + + public Context(SourceFile file, StateSnapshot ctx, float alphaDiscard) { + this.file = file; + this.ctx = ctx; + this.alphaDiscard = alphaDiscard; + } + + public ShaderConstants getShaderConstants() { + ShaderConstants shaderConstants = ctx.getDefines(); + + if (alphaDiscard > 0) { + shaderConstants.define("ALPHA_DISCARD", alphaDiscard); + } + + return shaderConstants; + } + + @Override + public boolean equals(Object obj) { + if (obj == this) return true; + if (obj == null || obj.getClass() != this.getClass()) return false; + var that = (Context) obj; + return this.file == that.file && Objects.equals(this.ctx, that.ctx) && Float.floatToIntBits(this.alphaDiscard) == Float.floatToIntBits(that.alphaDiscard); + } + + @Override + public int hashCode() { + return Objects.hash(file, ctx, alphaDiscard); + } + + @Override + public String toString() { + return "Context[" + "file=" + file + ", " + "ctx=" + ctx + ", " + "alphaDiscard=" + alphaDiscard + ']'; + } + + } } diff --git a/src/main/java/com/jozufozu/flywheel/core/compile/ProgramCompiler.java b/src/main/java/com/jozufozu/flywheel/core/compile/ProgramCompiler.java index cce67bd8d..0be2aa56b 100644 --- a/src/main/java/com/jozufozu/flywheel/core/compile/ProgramCompiler.java +++ b/src/main/java/com/jozufozu/flywheel/core/compile/ProgramCompiler.java @@ -62,9 +62,9 @@ public class ProgramCompiler

extends Memoizer 0) { - shaderConstants.define("ALPHA_DISCARD", alphaDiscard); - } - - return shaderConstants; - } - - @Override - public boolean equals(Object o) { - if (this == o) return true; - if (o == null || getClass() != o.getClass()) return false; - ProgramContext that = (ProgramContext) o; - // override for instance equality on vertexType - return alphaDiscard == that.alphaDiscard && ctx == that.ctx && vertexType == that.vertexType && spec.equals(that.spec); - } - - @Override - public int hashCode() { - return Objects.hash(alphaDiscard, vertexType, spec, ctx); + return new ProgramContext(spec, getAlphaDiscard(layer), vertexType, GameStateRegistry.takeSnapshot()); } /** @@ -74,4 +46,40 @@ public record ProgramContext(float alphaDiscard, VertexType vertexType, ProgramS public static float getAlphaDiscard(@Nullable RenderLayer layer) { return layer == RenderLayer.CUTOUT ? 0.1f : 0f; } + + public final ProgramSpec spec; + public final float alphaDiscard; + public final VertexType vertexType; + public final StateSnapshot ctx; + + /** + * @param spec The program to use. + * @param alphaDiscard Alpha threshold below which pixels are discarded. + * @param vertexType The vertexType the program should be adapted for. + * @param ctx A snapshot of the game state. + */ + public ProgramContext(ProgramSpec spec, float alphaDiscard, VertexType vertexType, StateSnapshot ctx) { + this.spec = spec; + this.alphaDiscard = alphaDiscard; + this.vertexType = vertexType; + this.ctx = ctx; + } + + @Override + public boolean equals(Object o) { + if (this == o) return true; + if (o == null || getClass() != o.getClass()) return false; + var that = (ProgramContext) o; + return spec == that.spec && vertexType == that.vertexType && ctx.equals(that.ctx) && Float.floatToIntBits(alphaDiscard) == Float.floatToIntBits(that.alphaDiscard); + } + + @Override + public int hashCode() { + return Objects.hash(spec, alphaDiscard, vertexType, ctx); + } + + @Override + public String toString() { + return "ProgramContext{" + "spec=" + spec + ", alphaDiscard=" + alphaDiscard + ", vertexType=" + vertexType + ", ctx=" + ctx + '}'; + } } diff --git a/src/main/java/com/jozufozu/flywheel/core/compile/VertexCompiler.java b/src/main/java/com/jozufozu/flywheel/core/compile/VertexCompiler.java index 1e0e6c925..2de890fdb 100644 --- a/src/main/java/com/jozufozu/flywheel/core/compile/VertexCompiler.java +++ b/src/main/java/com/jozufozu/flywheel/core/compile/VertexCompiler.java @@ -1,12 +1,15 @@ package com.jozufozu.flywheel.core.compile; +import java.util.Objects; + +import com.jozufozu.flywheel.api.vertex.VertexType; import com.jozufozu.flywheel.backend.gl.shader.GlShader; import com.jozufozu.flywheel.backend.gl.shader.ShaderType; import com.jozufozu.flywheel.backend.source.FileResolution; import com.jozufozu.flywheel.backend.source.SourceFile; -import com.jozufozu.flywheel.core.shader.ProgramSpec; +import com.jozufozu.flywheel.core.shader.StateSnapshot; -public class VertexCompiler extends Memoizer { +public class VertexCompiler extends Memoizer { private final Template template; private final FileResolution header; @@ -16,12 +19,12 @@ public class VertexCompiler extends Memoizer { } @Override - protected GlShader _create(ProgramContext key) { + protected GlShader _create(Context key) { StringBuilder finalSource = new StringBuilder(); finalSource.append(CompileUtil.generateHeader(template.getVersion(), ShaderType.VERTEX)); - key.getShaderConstants().writeInto(finalSource); + key.ctx.getDefines().writeInto(finalSource); finalSource.append(""" struct Vertex { @@ -32,25 +35,47 @@ public class VertexCompiler extends Memoizer { vec3 normal; }; """); - finalSource.append(key.vertexType() - .getShaderHeader()); + finalSource.append(key.vertexType.getShaderHeader()); FileIndexImpl index = new FileIndexImpl(); header.getFile().generateFinalSource(index, finalSource); - ProgramSpec spec = key.spec(); - SourceFile vertexFile = spec.getVertexFile(); - vertexFile.generateFinalSource(index, finalSource); + key.file.generateFinalSource(index, finalSource); - VertexData appliedTemplate = template.apply(vertexFile); - finalSource.append(appliedTemplate.generateFooter(index, key.vertexType())); + VertexData appliedTemplate = template.apply(key.file); + finalSource.append(appliedTemplate.generateFooter(index, key.vertexType)); - return new GlShader(spec.name, ShaderType.VERTEX, finalSource.toString()); + return new GlShader(key.file.name, ShaderType.VERTEX, finalSource.toString()); } @Override protected void _destroy(GlShader value) { value.delete(); } + + public static class Context { + private final SourceFile file; + private final StateSnapshot ctx; + private final VertexType vertexType; + + public Context(SourceFile file, StateSnapshot ctx, VertexType vertexType) { + this.file = file; + this.ctx = ctx; + this.vertexType = vertexType; + } + + @Override + public boolean equals(Object o) { + if (this == o) return true; + if (o == null || getClass() != o.getClass()) return false; + var that = (Context) o; + return file == that.file && vertexType == that.vertexType && ctx.equals(that.ctx); + } + + @Override + public int hashCode() { + return Objects.hash(file, ctx, vertexType); + } + } } diff --git a/src/main/java/com/jozufozu/flywheel/core/shader/GameStateProvider.java b/src/main/java/com/jozufozu/flywheel/core/shader/GameStateProvider.java index fc9fe728d..9bf698ac9 100644 --- a/src/main/java/com/jozufozu/flywheel/core/shader/GameStateProvider.java +++ b/src/main/java/com/jozufozu/flywheel/core/shader/GameStateProvider.java @@ -1,15 +1,14 @@ package com.jozufozu.flywheel.core.shader; -import com.jozufozu.flywheel.backend.GameStateRegistry; -import com.mojang.serialization.Codec; +import com.jozufozu.flywheel.core.compile.ShaderConstants; import net.minecraft.resources.ResourceLocation; public interface GameStateProvider { - Codec CODEC = ResourceLocation.CODEC.xmap(GameStateRegistry::getStateProvider, GameStateProvider::getID); - ResourceLocation getID(); boolean isTrue(); + + void alterConstants(ShaderConstants constants); } diff --git a/src/main/java/com/jozufozu/flywheel/core/shader/NormalDebugStateProvider.java b/src/main/java/com/jozufozu/flywheel/core/shader/NormalDebugStateProvider.java index 61f0e50ac..b8a340448 100644 --- a/src/main/java/com/jozufozu/flywheel/core/shader/NormalDebugStateProvider.java +++ b/src/main/java/com/jozufozu/flywheel/core/shader/NormalDebugStateProvider.java @@ -2,6 +2,7 @@ package com.jozufozu.flywheel.core.shader; import com.jozufozu.flywheel.Flywheel; import com.jozufozu.flywheel.config.FlwConfig; +import com.jozufozu.flywheel.core.compile.ShaderConstants; import net.minecraft.resources.ResourceLocation; @@ -24,4 +25,9 @@ public class NormalDebugStateProvider implements GameStateProvider { public ResourceLocation getID() { return NAME; } + + @Override + public void alterConstants(ShaderConstants constants) { + constants.define("DEBUG_NORMAL"); + } } diff --git a/src/main/java/com/jozufozu/flywheel/core/shader/ProgramSpec.java b/src/main/java/com/jozufozu/flywheel/core/shader/ProgramSpec.java index bbaca28f1..fb03631f2 100644 --- a/src/main/java/com/jozufozu/flywheel/core/shader/ProgramSpec.java +++ b/src/main/java/com/jozufozu/flywheel/core/shader/ProgramSpec.java @@ -1,10 +1,5 @@ package com.jozufozu.flywheel.core.shader; -import java.util.ArrayList; -import java.util.Collections; -import java.util.List; - -import com.google.common.collect.ImmutableList; import com.jozufozu.flywheel.backend.source.FileResolution; import com.jozufozu.flywheel.backend.source.Resolver; import com.jozufozu.flywheel.backend.source.SourceFile; @@ -28,27 +23,20 @@ import net.minecraft.resources.ResourceLocation; */ public class ProgramSpec { - // TODO: Block model style inheritance? public static final Codec CODEC = RecordCodecBuilder.create(instance -> instance.group( ResourceLocation.CODEC.fieldOf("vertex") .forGetter(ProgramSpec::getSourceLoc), ResourceLocation.CODEC.fieldOf("fragment") - .forGetter(ProgramSpec::getFragmentLoc), - ProgramState.CODEC.listOf() - .optionalFieldOf("states", Collections.emptyList()) - .forGetter(ProgramSpec::getStates)) + .forGetter(ProgramSpec::getFragmentLoc)) .apply(instance, ProgramSpec::new)); public ResourceLocation name; public final FileResolution vertex; public final FileResolution fragment; - public final ImmutableList states; - - public ProgramSpec(ResourceLocation vertex, ResourceLocation fragment, List states) { + public ProgramSpec(ResourceLocation vertex, ResourceLocation fragment) { this.vertex = Resolver.INSTANCE.get(vertex); this.fragment = Resolver.INSTANCE.get(fragment); - this.states = ImmutableList.copyOf(states); } public void setName(ResourceLocation name) { @@ -73,36 +61,8 @@ public class ProgramSpec { return fragment.getFile(); } - public ImmutableList getStates() { - return states; - } - - /** - * Calculate a unique ID representing the current game state. - */ - public long getCurrentStateID() { - long ctx = 0; - for (ProgramState state : states) { - if (state.context().isTrue()) { - ctx |= 1; - } - ctx <<= 1; - } - return ctx; - } - - /** - * Given the stateID, get a list of defines to include at the top of a compiling program. - */ - public List getDefines(long stateID) { - List defines = new ArrayList<>(); - - for (ProgramState state : states) { - if ((stateID & 1) == 1) { - defines.addAll(state.defines()); - } - stateID >>= 1; - } - return defines; + @Override + public String toString() { + return name.toString(); } } diff --git a/src/main/java/com/jozufozu/flywheel/core/shader/ProgramState.java b/src/main/java/com/jozufozu/flywheel/core/shader/ProgramState.java deleted file mode 100644 index ae8d00248..000000000 --- a/src/main/java/com/jozufozu/flywheel/core/shader/ProgramState.java +++ /dev/null @@ -1,17 +0,0 @@ -package com.jozufozu.flywheel.core.shader; - -import java.util.Collections; -import java.util.List; - -import com.jozufozu.flywheel.util.CodecUtil; -import com.mojang.serialization.Codec; -import com.mojang.serialization.codecs.RecordCodecBuilder; - -public record ProgramState(GameStateProvider context, List defines) { - - public static final Codec CODEC = RecordCodecBuilder.create(state -> state.group(GameStateProvider.CODEC.fieldOf("when") - .forGetter(ProgramState::context), CodecUtil.oneOrMore(Codec.STRING) - .optionalFieldOf("define", Collections.emptyList()) - .forGetter(ProgramState::defines)) - .apply(state, ProgramState::new)); -} diff --git a/src/main/java/com/jozufozu/flywheel/core/shader/StateSnapshot.java b/src/main/java/com/jozufozu/flywheel/core/shader/StateSnapshot.java new file mode 100644 index 000000000..e11b7c0f2 --- /dev/null +++ b/src/main/java/com/jozufozu/flywheel/core/shader/StateSnapshot.java @@ -0,0 +1,12 @@ +package com.jozufozu.flywheel.core.shader; + +import com.jozufozu.flywheel.backend.GameStateRegistry; +import com.jozufozu.flywheel.core.compile.ShaderConstants; + +public record StateSnapshot(long ctx) { + // TODO: is this needed? + + public ShaderConstants getDefines() { + return GameStateRegistry.getDefines(ctx); + } +} diff --git a/src/main/resources/assets/flywheel/flywheel/programs/model.json b/src/main/resources/assets/flywheel/flywheel/programs/model.json index d6c2b6d73..dbee0ddd4 100644 --- a/src/main/resources/assets/flywheel/flywheel/programs/model.json +++ b/src/main/resources/assets/flywheel/flywheel/programs/model.json @@ -1,10 +1,4 @@ { "vertex": "flywheel:model.vert", - "fragment": "flywheel:block.frag", - "states": [ - { - "when": "flywheel:normal_debug", - "define": "DEBUG_NORMAL" - } - ] + "fragment": "flywheel:block.frag" } diff --git a/src/main/resources/assets/flywheel/flywheel/programs/oriented.json b/src/main/resources/assets/flywheel/flywheel/programs/oriented.json index 33ebc3291..f012f819b 100644 --- a/src/main/resources/assets/flywheel/flywheel/programs/oriented.json +++ b/src/main/resources/assets/flywheel/flywheel/programs/oriented.json @@ -1,10 +1,4 @@ { "vertex": "flywheel:oriented.vert", - "fragment": "flywheel:block.frag", - "states": [ - { - "when": "flywheel:normal_debug", - "define": "DEBUG_NORMAL" - } - ] + "fragment": "flywheel:block.frag" } diff --git a/src/main/resources/assets/flywheel/flywheel/programs/passthru.json b/src/main/resources/assets/flywheel/flywheel/programs/passthru.json index 5934f447c..9e180f2de 100644 --- a/src/main/resources/assets/flywheel/flywheel/programs/passthru.json +++ b/src/main/resources/assets/flywheel/flywheel/programs/passthru.json @@ -1,10 +1,4 @@ { "vertex": "flywheel:passthru.vert", - "fragment": "flywheel:block.frag", - "states": [ - { - "when": "flywheel:normal_debug", - "define": "DEBUG_NORMAL" - } - ] + "fragment": "flywheel:block.frag" }