From a471fdbafd9efd40a63a8ac1b7f4845e4e269002 Mon Sep 17 00:00:00 2001
From: Jozufozu
Date: Sat, 9 Jul 2022 11:32:28 -0400
Subject: [PATCH] Structless instances
- Instance shaders define their own input variables
- Input locations are transformed during compilation based on vertex type
---
.../flywheel/core/compile/VertexCompiler.java | 95 ++++++-------------
.../flywheel/core/source/ShaderField.java | 43 +++++++++
.../flywheel/core/source/SourceFile.java | 76 +++++++++++----
.../core/source/parse/ShaderFunction.java | 2 +-
.../flywheel/core/source/span/Span.java | 9 +-
.../core/structs/InstanceShaders.java | 2 +-
.../flywheel/flywheel/instance/oriented.vert | 22 ++---
.../flywheel/instance/transformed.vert | 20 ++--
8 files changed, 159 insertions(+), 110 deletions(-)
create mode 100644 src/main/java/com/jozufozu/flywheel/core/source/ShaderField.java
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 66a9352c1..1c6c2c7d5 100644
--- a/src/main/java/com/jozufozu/flywheel/core/compile/VertexCompiler.java
+++ b/src/main/java/com/jozufozu/flywheel/core/compile/VertexCompiler.java
@@ -1,5 +1,7 @@
package com.jozufozu.flywheel.core.compile;
+import java.util.ArrayList;
+
import com.google.common.collect.ImmutableList;
import com.jozufozu.flywheel.api.vertex.VertexType;
import com.jozufozu.flywheel.backend.gl.GLSLVersion;
@@ -8,9 +10,12 @@ import com.jozufozu.flywheel.backend.gl.shader.ShaderType;
import com.jozufozu.flywheel.core.shader.StateSnapshot;
import com.jozufozu.flywheel.core.source.FileIndex;
import com.jozufozu.flywheel.core.source.FileResolution;
+import com.jozufozu.flywheel.core.source.ShaderField;
import com.jozufozu.flywheel.core.source.SourceFile;
import com.jozufozu.flywheel.core.source.parse.ShaderStruct;
import com.jozufozu.flywheel.core.source.parse.StructField;
+import com.jozufozu.flywheel.core.source.span.Span;
+import com.jozufozu.flywheel.util.Pair;
/**
* Handles compilation and deletion of vertex shaders.
@@ -43,8 +48,21 @@ public class VertexCompiler extends Memoizer {
// INSTANCE
+ int attributeBaseIndex = key.vertexType.getLayout()
+ .getAttributeCount();
+
var instanceShader = key.instanceShader;
- instanceShader.generateFinalSource(index, finalSource);
+ var replacements = new ArrayList>();
+ for (ShaderField field : instanceShader.fields.values()) {
+ if (field.decoration != ShaderField.Decoration.IN) {
+ continue;
+ }
+
+ int location = Integer.parseInt(field.location.get());
+ int newLocation = location + attributeBaseIndex;
+ replacements.add(Pair.of(field.location, Integer.toString(newLocation)));
+ }
+ instanceShader.generateFinalSource(index, finalSource, replacements);
// MATERIAL
@@ -58,11 +76,17 @@ public class VertexCompiler extends Memoizer {
// MAIN
- var instanceStruct = instanceShader.findFunction("flw_instanceVertex")
- .flatMap(f -> f.getParameterType(0)
- .findStruct())
- .orElseThrow();
- finalSource.append(generateFooter(key.vertexType, instanceStruct));
+ finalSource.append("""
+ void main() {
+ flw_layoutVertex();
+
+ flw_instanceVertex();
+
+ flw_materialVertex();
+
+ flw_contextVertex();
+ }
+ """);
try {
return new GlShader(finalSource.toString(), ShaderType.VERTEX, ImmutableList.of(layoutShader.name, instanceShader.name, materialShader.name, contextShaderSource.name), shaderConstants);
@@ -71,65 +95,6 @@ public class VertexCompiler extends Memoizer {
}
}
- protected String generateFooter(VertexType vertexType, ShaderStruct instance) {
- ImmutableList fields = instance.getFields();
-
- int attributeBinding = vertexType.getLayout()
- .getAttributeCount();
-
- StringBuilder footer = new StringBuilder();
-
- for (StructField field : fields) {
- footer.append("layout(location = ")
- .append(attributeBinding)
- .append(") in")
- .append(' ')
- .append(field.type)
- .append(' ')
- .append("_flw_a_i_")
- .append(field.name)
- .append(";\n");
- attributeBinding += CompileUtil.getAttributeCount(field.type);
- }
- footer.append('\n');
-
- footer.append(String.format("""
- void main() {
- flw_layoutVertex();
-
- %s instance;
- %s
- flw_instanceVertex(instance);
-
- flw_materialVertex();
-
- flw_contextVertex();
- }
- """,
- instance.name,
- assignFields(instance, "instance.", "_flw_a_i_")
- ));
-
- return footer.toString();
- }
-
- protected static StringBuilder assignFields(ShaderStruct struct, String prefix1, String prefix2) {
- ImmutableList fields = struct.getFields();
-
- StringBuilder builder = new StringBuilder();
-
- for (StructField field : fields) {
- builder.append(prefix1)
- .append(field.name)
- .append(" = ")
- .append(prefix2)
- .append(field.name)
- .append(";\n");
- }
-
- return builder;
- }
-
@Override
protected void _destroy(GlShader value) {
value.delete();
diff --git a/src/main/java/com/jozufozu/flywheel/core/source/ShaderField.java b/src/main/java/com/jozufozu/flywheel/core/source/ShaderField.java
new file mode 100644
index 000000000..9c61684ac
--- /dev/null
+++ b/src/main/java/com/jozufozu/flywheel/core/source/ShaderField.java
@@ -0,0 +1,43 @@
+package com.jozufozu.flywheel.core.source;
+
+import java.util.regex.Pattern;
+
+import org.jetbrains.annotations.Nullable;
+
+import com.jozufozu.flywheel.core.source.parse.AbstractShaderElement;
+import com.jozufozu.flywheel.core.source.span.Span;
+
+public class ShaderField extends AbstractShaderElement {
+ public static final Pattern PATTERN = Pattern.compile("layout\\s+\\(location\\s+=\\s+(\\d+)\\)\\s+(in|out)\\s+([\\w\\d]+)\\s+" + "([\\w\\d]+)");
+
+ public final Span location;
+ public final @Nullable Decoration decoration;
+ public final Span type;
+ public final Span name;
+
+ public ShaderField(Span self, Span location, Span inOut, Span type, Span name) {
+ super(self);
+
+ this.location = location;
+ this.decoration = Decoration.fromSpan(inOut);
+ this.type = type;
+ this.name = name;
+ }
+
+ public enum Decoration {
+ IN,
+ OUT,
+ FLAT,
+ ;
+
+ @Nullable
+ public static Decoration fromSpan(Span span) {
+ return switch (span.toString()) {
+ case "in" -> IN;
+ case "out" -> OUT;
+ case "flat" -> FLAT;
+ default -> null;
+ };
+ }
+ }
+}
diff --git a/src/main/java/com/jozufozu/flywheel/core/source/SourceFile.java b/src/main/java/com/jozufozu/flywheel/core/source/SourceFile.java
index 7332ae7cf..72b05f946 100644
--- a/src/main/java/com/jozufozu/flywheel/core/source/SourceFile.java
+++ b/src/main/java/com/jozufozu/flywheel/core/source/SourceFile.java
@@ -1,6 +1,8 @@
package com.jozufozu.flywheel.core.source;
import java.util.ArrayList;
+import java.util.Collections;
+import java.util.Comparator;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
@@ -18,6 +20,7 @@ import com.jozufozu.flywheel.core.source.parse.ShaderStruct;
import com.jozufozu.flywheel.core.source.span.ErrorSpan;
import com.jozufozu.flywheel.core.source.span.Span;
import com.jozufozu.flywheel.core.source.span.StringSpan;
+import com.jozufozu.flywheel.util.Pair;
import net.minecraft.resources.ResourceLocation;
@@ -34,10 +37,6 @@ public class SourceFile {
public final ShaderSources parent;
public final String source;
- /**
- * Sections of the source that must be trimmed for compilation. Currently, it only contains the spans of imports.
- */
- public final String elided;
public final SourceLines lines;
@@ -55,6 +54,8 @@ public class SourceFile {
* Includes ordered as defined in the source.
*/
public final ImmutableList imports;
+ public final ImmutableMap fields;
+ private final List elisions;
public SourceFile(ErrorReporter errorReporter, ShaderSources parent, ResourceLocation name, String source) {
this.parent = parent;
@@ -63,13 +64,12 @@ public class SourceFile {
this.lines = new SourceLines(source);
- List elisions = new ArrayList<>();
+ this.elisions = new ArrayList<>();
- this.imports = parseImports(errorReporter, elisions);
+ this.imports = parseImports(errorReporter);
this.functions = parseFunctions();
this.structs = parseStructs();
-
- this.elided = elideSource(source, elisions).toString();
+ this.fields = parseFields();
}
public Span getLineSpan(int line) {
@@ -136,11 +136,15 @@ public class SourceFile {
}
public void generateFinalSource(FileIndex env, StringBuilder source) {
+ generateFinalSource(env, source, Collections.emptyList());
+ }
+
+ public void generateFinalSource(FileIndex env, StringBuilder source, List> replacements) {
for (Import include : imports) {
SourceFile file = include.getFile();
if (file != null && !env.exists(file)) {
- file.generateFinalSource(env, source);
+ file.generateFinalSource(env, source, replacements);
}
}
@@ -151,7 +155,13 @@ public class SourceFile {
.append(" // ")
.append(name)
.append('\n');
- source.append(elided);
+
+ var replacementsAndElisions = new ArrayList<>(replacements);
+ for (Span elision : elisions) {
+ replacementsAndElisions.add(Pair.of(elision, ""));
+ }
+
+ source.append(this.replace(replacementsAndElisions));
source.append('\n');
}
@@ -159,18 +169,27 @@ public class SourceFile {
return "Source for shader '" + name + "':\n" + lines.printLinesWithNumbers();
}
- private static CharSequence elideSource(String source, List elisions) {
+ private CharSequence replace(List> replacements) {
StringBuilder out = new StringBuilder();
int lastEnd = 0;
- for (Span elision : elisions) {
- out.append(source, lastEnd, elision.getStartPos());
+ replacements.sort(Comparator.comparing(Pair::first));
- lastEnd = elision.getEndPos();
+ for (var replacement : replacements) {
+ var loc = replacement.first();
+
+ if (loc.getSourceFile() != this) {
+ continue;
+ }
+
+ out.append(this.source, lastEnd, loc.getStartPos());
+ out.append(replacement.second());
+
+ lastEnd = loc.getEndPos();
}
- out.append(source, lastEnd, source.length());
+ out.append(this.source, lastEnd, this.source.length());
return out;
}
@@ -179,7 +198,7 @@ public class SourceFile {
* Scan the source for function definitions and "parse" them into objects that contain properties of the function.
*/
private ImmutableMap parseFunctions() {
- Matcher matcher = ShaderFunction.functionDeclaration.matcher(source);
+ Matcher matcher = ShaderFunction.PATTERN.matcher(source);
Map functions = new HashMap<>();
@@ -229,12 +248,31 @@ public class SourceFile {
return structs.build();
}
+ /**
+ * Scan the source for function definitions and "parse" them into objects that contain properties of the function.
+ */
+ private ImmutableMap parseFields() {
+ Matcher matcher = ShaderField.PATTERN.matcher(source);
+
+ ImmutableMap.Builder fields = ImmutableMap.builder();
+ while (matcher.find()) {
+ Span self = Span.fromMatcher(this, matcher);
+ Span location = Span.fromMatcher(this, matcher, 1);
+ Span decoration = Span.fromMatcher(this, matcher, 2);
+ Span type = Span.fromMatcher(this, matcher, 3);
+ Span name = Span.fromMatcher(this, matcher, 4);
+
+ fields.put(location.get(), new ShaderField(self, location, decoration, type, name));
+ }
+
+ return fields.build();
+ }
+
/**
* Scan the source for {@code #use "..."} directives.
* Records the contents of the directive into an {@link Import} object, and marks the directive for elision.
- * @param elisions
*/
- private ImmutableList parseImports(ErrorReporter errorReporter, List elisions) {
+ private ImmutableList parseImports(ErrorReporter errorReporter) {
Matcher uses = Import.PATTERN.matcher(source);
Set importedFiles = new HashSet<>();
@@ -253,7 +291,7 @@ public class SourceFile {
}
}
- elisions.add(use); // we have to trim that later
+ this.elisions.add(use); // we have to trim that later
}
return ImmutableList.copyOf(imports);
diff --git a/src/main/java/com/jozufozu/flywheel/core/source/parse/ShaderFunction.java b/src/main/java/com/jozufozu/flywheel/core/source/parse/ShaderFunction.java
index 134d0214f..38dda848c 100644
--- a/src/main/java/com/jozufozu/flywheel/core/source/parse/ShaderFunction.java
+++ b/src/main/java/com/jozufozu/flywheel/core/source/parse/ShaderFunction.java
@@ -10,7 +10,7 @@ import com.jozufozu.flywheel.core.source.span.Span;
public class ShaderFunction extends AbstractShaderElement {
// https://regexr.com/60n3d
- public static final Pattern functionDeclaration = Pattern.compile("(\\w+)\\s+(\\w+)\\s*\\(([\\w,\\s]*)\\)\\s*\\{");
+ public static final Pattern PATTERN = Pattern.compile("(\\w+)\\s+(\\w+)\\s*\\(([\\w,\\s]*)\\)\\s*\\{");
public static final Pattern argument = Pattern.compile("(?:(inout|in|out) )?(\\w+)\\s+(\\w+)");
public static final Pattern assignment = Pattern.compile("(\\w+)\\s*=");
diff --git a/src/main/java/com/jozufozu/flywheel/core/source/span/Span.java b/src/main/java/com/jozufozu/flywheel/core/source/span/Span.java
index 81b8e848a..7491f0cf4 100644
--- a/src/main/java/com/jozufozu/flywheel/core/source/span/Span.java
+++ b/src/main/java/com/jozufozu/flywheel/core/source/span/Span.java
@@ -3,6 +3,8 @@ package com.jozufozu.flywheel.core.source.span;
import java.util.Optional;
import java.util.regex.Matcher;
+import org.jetbrains.annotations.NotNull;
+
import com.jozufozu.flywheel.core.source.SourceFile;
import com.jozufozu.flywheel.core.source.parse.ShaderFunction;
import com.jozufozu.flywheel.core.source.parse.ShaderStruct;
@@ -14,7 +16,7 @@ import com.jozufozu.flywheel.core.source.parse.ShaderStruct;
* Spans are used for pretty-printing errors.
*
*/
-public abstract class Span implements CharSequence {
+public abstract class Span implements CharSequence, Comparable {
protected final SourceFile in;
protected final CharPos start;
@@ -138,4 +140,9 @@ public abstract class Span implements CharSequence {
}
return in.findFunction(this);
}
+
+ @Override
+ public int compareTo(@NotNull Span o) {
+ return Integer.compareUnsigned(getStartPos(), o.getStartPos());
+ }
}
diff --git a/src/main/java/com/jozufozu/flywheel/core/structs/InstanceShaders.java b/src/main/java/com/jozufozu/flywheel/core/structs/InstanceShaders.java
index 0b253cfeb..f86faaee2 100644
--- a/src/main/java/com/jozufozu/flywheel/core/structs/InstanceShaders.java
+++ b/src/main/java/com/jozufozu/flywheel/core/structs/InstanceShaders.java
@@ -12,7 +12,7 @@ import com.jozufozu.flywheel.util.ResourceUtil;
import net.minecraft.resources.ResourceLocation;
public class InstanceShaders {
- public static final BiConsumer CHECK = SourceChecks.checkFunctionParameterTypeExists("flw_instanceVertex", 1, 0);
+ public static final BiConsumer CHECK = SourceChecks.checkFunctionArity("flw_instanceVertex", 0);
public static final FileResolution TRANSFORMED = create(ResourceUtil.subPath(Names.TRANSFORMED, ".vert"));
public static final FileResolution ORIENTED = create(ResourceUtil.subPath(Names.ORIENTED, ".vert"));
diff --git a/src/main/resources/assets/flywheel/flywheel/instance/oriented.vert b/src/main/resources/assets/flywheel/flywheel/instance/oriented.vert
index c9e5b5cff..dc49c75ca 100644
--- a/src/main/resources/assets/flywheel/flywheel/instance/oriented.vert
+++ b/src/main/resources/assets/flywheel/flywheel/instance/oriented.vert
@@ -2,17 +2,15 @@
#use "flywheel:util/light.glsl"
#use "flywheel:util/quaternion.glsl"
-struct Oriented {
- vec2 light;
- vec4 color;
- vec3 pos;
- vec3 pivot;
- vec4 rotation;
-};
+layout (location = 0) in vec2 oriented_light;
+layout (location = 1) in vec4 oriented_color;
+layout (location = 2) in vec3 oriented_pos;
+layout (location = 3) in vec3 oriented_pivot;
+layout (location = 4) in vec4 oriented_rotation;
-void flw_instanceVertex(Oriented oriented) {
- flw_vertexPos = vec4(rotateVertexByQuat(flw_vertexPos.xyz - oriented.pivot, oriented.rotation) + oriented.pivot + oriented.pos, 1.0);
- flw_vertexNormal = rotateVertexByQuat(flw_vertexNormal, oriented.rotation);
- flw_vertexColor = oriented.color;
- flw_vertexLight = shiftLight(oriented.light);
+void flw_instanceVertex() {
+ flw_vertexPos = vec4(rotateVertexByQuat(flw_vertexPos.xyz - oriented_pivot, oriented_rotation) + oriented_pivot + oriented_pos, 1.0);
+ flw_vertexNormal = rotateVertexByQuat(flw_vertexNormal, oriented_rotation);
+ flw_vertexColor = oriented_color;
+ flw_vertexLight = shiftLight(oriented_light);
}
diff --git a/src/main/resources/assets/flywheel/flywheel/instance/transformed.vert b/src/main/resources/assets/flywheel/flywheel/instance/transformed.vert
index cea24a54a..d5fe6fc62 100644
--- a/src/main/resources/assets/flywheel/flywheel/instance/transformed.vert
+++ b/src/main/resources/assets/flywheel/flywheel/instance/transformed.vert
@@ -1,16 +1,14 @@
#use "flywheel:api/vertex.glsl"
#use "flywheel:util/light.glsl"
-struct Instance {
- vec2 light;
- vec4 color;
- mat4 transform;
- mat3 normalMat;
-};
+layout (location = 0) in vec2 transformed_light;
+layout (location = 1) in vec4 transformed_color;
+layout (location = 2) in mat4 transformed_pose;
+layout (location = 6) in mat3 transformed_normal;
-void flw_instanceVertex(Instance instance) {
- flw_vertexPos = instance.transform * flw_vertexPos;
- flw_vertexNormal = instance.normalMat * flw_vertexNormal;
- flw_vertexColor = instance.color;
- flw_vertexLight = shiftLight(instance.light);
+void flw_instanceVertex() {
+ flw_vertexPos = transformed_pose * flw_vertexPos;
+ flw_vertexNormal = transformed_normal * flw_vertexNormal;
+ flw_vertexColor = transformed_color;
+ flw_vertexLight = shiftLight(transformed_light);
}