Instancing with Layouts

- InstancedArraysComponent now fully uses Layouts.
  - Trivial to get the glsl type name of a given Element.
  - Also quite easy to get the attribute count without actually
    creating the attribute list.
- IndirectComponent is WIP because unpacking is hard.
  - Reorganize so all the unpacking stuff is in one loop.
This commit is contained in:
Jozufozu 2024-01-04 23:44:35 -08:00
parent c08a720954
commit 9309266435
4 changed files with 137 additions and 50 deletions

View file

@ -0,0 +1,62 @@
package com.jozufozu.flywheel.backend.compile;
import com.jozufozu.flywheel.api.layout.ElementType;
import com.jozufozu.flywheel.api.layout.FloatRepr;
import com.jozufozu.flywheel.api.layout.IntegerRepr;
import com.jozufozu.flywheel.api.layout.MatrixElementType;
import com.jozufozu.flywheel.api.layout.ScalarElementType;
import com.jozufozu.flywheel.api.layout.UnsignedIntegerRepr;
import com.jozufozu.flywheel.api.layout.ValueRepr;
import com.jozufozu.flywheel.api.layout.VectorElementType;
public class LayoutInterpreter {
public static int attributeCount(ElementType type) {
if (type instanceof ScalarElementType) {
return 1;
} else if (type instanceof VectorElementType) {
return 1;
} else if (type instanceof MatrixElementType matrix) {
return matrix.rows();
}
throw new IllegalArgumentException("Unknown type " + type);
}
public static String typeName(ElementType type) {
if (type instanceof ScalarElementType scalar) {
return scalarTypeName(scalar.repr());
} else if (type instanceof VectorElementType vector) {
return vectorTypeName(vector.repr(), vector.size());
} else if (type instanceof MatrixElementType matrix) {
return matrixTypeName(matrix);
}
throw new IllegalArgumentException("Unknown type " + type);
}
public static String matrixTypeName(MatrixElementType matrix) {
return "mat" + matrix.columns() + "x" + matrix.rows();
}
public static String vectorTypeName(ValueRepr repr, int size) {
if (repr instanceof IntegerRepr) {
return "ivec" + size;
} else if (repr instanceof UnsignedIntegerRepr) {
return "uvec" + size;
} else if (repr instanceof FloatRepr) {
return "vec" + size;
}
throw new IllegalArgumentException("Unknown repr " + repr);
}
public static String scalarTypeName(ValueRepr repr) {
if (repr instanceof IntegerRepr) {
return "int";
} else if (repr instanceof UnsignedIntegerRepr) {
return "uint";
} else if (repr instanceof FloatRepr) {
return "float";
}
throw new IllegalArgumentException("Unknown repr " + repr);
}
}

View file

@ -1,5 +1,6 @@
package com.jozufozu.flywheel.backend.compile.component; package com.jozufozu.flywheel.backend.compile.component;
import java.util.ArrayList;
import java.util.Collection; import java.util.Collection;
import java.util.Collections; import java.util.Collections;
import java.util.List; import java.util.List;
@ -7,6 +8,7 @@ import java.util.List;
import com.jozufozu.flywheel.Flywheel; import com.jozufozu.flywheel.Flywheel;
import com.jozufozu.flywheel.api.instance.InstanceType; import com.jozufozu.flywheel.api.instance.InstanceType;
import com.jozufozu.flywheel.api.layout.Layout; import com.jozufozu.flywheel.api.layout.Layout;
import com.jozufozu.flywheel.backend.compile.LayoutInterpreter;
import com.jozufozu.flywheel.backend.compile.Pipeline; import com.jozufozu.flywheel.backend.compile.Pipeline;
import com.jozufozu.flywheel.glsl.SourceComponent; import com.jozufozu.flywheel.glsl.SourceComponent;
import com.jozufozu.flywheel.glsl.generate.FnSignature; import com.jozufozu.flywheel.glsl.generate.FnSignature;
@ -58,44 +60,58 @@ public class IndirectComponent implements SourceComponent {
public String generateIndirect() { public String generateIndirect() {
var builder = new GlslBuilder(); var builder = new GlslBuilder();
layoutItems.stream() generateHeader(builder);
.map(LayoutItem::type)
.distinct()
.forEach(type -> type.declare(builder));
var instance = builder.struct(); generateInstanceStruct(builder);
instance.setName(STRUCT_NAME);
builder.blankLine();
var packed = builder.struct();
packed.setName(PACKED_STRUCT_NAME);
for (var field : layoutItems) {
packed.addField(field.type()
.packedTypeName(), field.name());
instance.addField(field.type()
.typeName(), field.name());
}
builder.blankLine(); builder.blankLine();
builder.function() generateUnpacking(builder);
.signature(FnSignature.create()
.returnType(STRUCT_NAME)
.name(UNPACK_FN_NAME)
.arg(PACKED_STRUCT_NAME, UNPACK_ARG)
.build())
.body(this::generateUnpackingBody);
builder.blankLine(); builder.blankLine();
return builder.build(); return builder.build();
} }
private void generateUnpackingBody(GlslBlock b) { private void generateHeader(GlslBuilder builder) {
var unpackedFields = layoutItems.stream() layoutItems.stream()
.map(layoutItem -> UNPACKING_VARIABLE.access(layoutItem.name()) .map(LayoutItem::type)
.transform(layoutItem.type()::unpack)) .distinct()
.toList(); .forEach(type -> type.declare(builder));
b.ret(GlslExpr.call(STRUCT_NAME, unpackedFields)); }
private void generateInstanceStruct(GlslBuilder builder) {
var instance = builder.struct();
instance.setName(STRUCT_NAME);
for (var element : layout.elements()) {
instance.addField(LayoutInterpreter.typeName(element.type()), element.name());
}
}
private void generateUnpacking(GlslBuilder builder) {
var packed = builder.struct();
packed.setName(PACKED_STRUCT_NAME);
var unpackArgs = new ArrayList<GlslExpr>();
for (LayoutItem field : layoutItems) {
GlslExpr unpack = UNPACKING_VARIABLE.access(field.name())
.transform(field.type()::unpack);
unpackArgs.add(unpack);
packed.addField(field.type()
.packedTypeName(), field.name());
}
var block = new GlslBlock();
block.ret(GlslExpr.call(STRUCT_NAME, unpackArgs));
builder.blankLine();
builder.function()
.signature(FnSignature.create()
.returnType(STRUCT_NAME)
.name(UNPACK_FN_NAME)
.arg(PACKED_STRUCT_NAME, UNPACK_ARG)
.build())
.body(block);
} }
} }

View file

@ -2,16 +2,16 @@ package com.jozufozu.flywheel.backend.compile.component;
import java.util.Collection; import java.util.Collection;
import java.util.Collections; import java.util.Collections;
import java.util.List;
import com.jozufozu.flywheel.Flywheel; import com.jozufozu.flywheel.Flywheel;
import com.jozufozu.flywheel.api.layout.Layout;
import com.jozufozu.flywheel.backend.compile.LayoutInterpreter;
import com.jozufozu.flywheel.backend.compile.Pipeline; import com.jozufozu.flywheel.backend.compile.Pipeline;
import com.jozufozu.flywheel.glsl.SourceComponent; import com.jozufozu.flywheel.glsl.SourceComponent;
import com.jozufozu.flywheel.glsl.generate.FnSignature; import com.jozufozu.flywheel.glsl.generate.FnSignature;
import com.jozufozu.flywheel.glsl.generate.GlslBlock; import com.jozufozu.flywheel.glsl.generate.GlslBlock;
import com.jozufozu.flywheel.glsl.generate.GlslBuilder; import com.jozufozu.flywheel.glsl.generate.GlslBuilder;
import com.jozufozu.flywheel.glsl.generate.GlslExpr; import com.jozufozu.flywheel.glsl.generate.GlslExpr;
import com.jozufozu.flywheel.lib.layout.LayoutItem;
import net.minecraft.resources.ResourceLocation; import net.minecraft.resources.ResourceLocation;
@ -20,12 +20,12 @@ public class InstancedArraysComponent implements SourceComponent {
private static final String STRUCT_NAME = "FlwInstance"; private static final String STRUCT_NAME = "FlwInstance";
private static final String UNPACK_FN_NAME = "_flw_unpackInstance"; private static final String UNPACK_FN_NAME = "_flw_unpackInstance";
private final List<LayoutItem> layoutItems; private final Layout layout;
private final int baseIndex; private final int baseIndex;
public InstancedArraysComponent(Pipeline.InstanceAssemblerContext ctx) { public InstancedArraysComponent(Pipeline.InstanceAssemblerContext ctx) {
this.layoutItems = ctx.instanceType() this.layout = ctx.instanceType()
.oldLayout().layoutItems; .layout();
this.baseIndex = ctx.baseAttribute(); this.baseIndex = ctx.baseAttribute();
} }
@ -43,26 +43,15 @@ public class InstancedArraysComponent implements SourceComponent {
public String source() { public String source() {
var builder = new GlslBuilder(); var builder = new GlslBuilder();
int i = baseIndex; generateVertexInput(builder);
for (var field : layoutItems) {
builder.vertexInput()
.binding(i)
.type(field.type()
.typeName())
.name(ATTRIBUTE_PREFIX + field.name());
i += field.type()
.attributeCount();
}
builder.blankLine(); builder.blankLine();
var structBuilder = builder.struct(); var structBuilder = builder.struct();
structBuilder.setName(STRUCT_NAME); structBuilder.setName(STRUCT_NAME);
for (var field : layoutItems) { for (var element : layout.elements()) {
structBuilder.addField(field.type() structBuilder.addField(LayoutInterpreter.typeName(element.type()), element.name());
.typeName(), field.name());
} }
builder.blankLine(); builder.blankLine();
@ -77,8 +66,23 @@ public class InstancedArraysComponent implements SourceComponent {
return builder.build(); return builder.build();
} }
private void generateVertexInput(GlslBuilder builder) {
int i = baseIndex;
for (var element : layout.elements()) {
var type = element.type();
builder.vertexInput()
.binding(i)
.type(LayoutInterpreter.typeName(type))
.name(ATTRIBUTE_PREFIX + element.name());
i += LayoutInterpreter.attributeCount(type);
}
}
private void generateUnpackingBody(GlslBlock b) { private void generateUnpackingBody(GlslBlock b) {
var fields = layoutItems.stream() var fields = layout.elements()
.stream()
.map(it -> new GlslExpr.Variable(ATTRIBUTE_PREFIX + it.name())) .map(it -> new GlslExpr.Variable(ATTRIBUTE_PREFIX + it.name()))
.toList(); .toList();
b.ret(GlslExpr.call(STRUCT_NAME, fields)); b.ret(GlslExpr.call(STRUCT_NAME, fields));

View file

@ -5,7 +5,7 @@ import java.util.function.Consumer;
import com.jozufozu.flywheel.lib.util.StringUtil; import com.jozufozu.flywheel.lib.util.StringUtil;
public class GlslFn implements GlslBuilder.Declaration { public class GlslFn implements GlslBuilder.Declaration {
private final GlslBlock body = new GlslBlock(); private GlslBlock body = new GlslBlock();
private FnSignature signature; private FnSignature signature;
public GlslFn signature(FnSignature signature) { public GlslFn signature(FnSignature signature) {
@ -18,6 +18,11 @@ public class GlslFn implements GlslBuilder.Declaration {
return this; return this;
} }
public GlslFn body(GlslBlock block) {
body = block;
return this;
}
public String prettyPrint() { public String prettyPrint() {
return """ return """
%s { %s {