mirror of
https://github.com/Jozufozu/Flywheel.git
synced 2025-01-15 23:55:53 +01:00
Little things
- Do not store a list of initialized instancers. - Remove AbstractInstancer#delete. Only InstancedInstancer was using it so move instancer deletion to InstancedDrawManager. - Improve CompilationHarness builder pattern and reusability. - Pass compilation keys directly to compileAndReportErrors. - Build the harness at the end of the builder chain rather than at the beginning. - Use same CompilationHarness for apply shader and scatter shader. - Remove _ prefix from packed struct fields. - Make element type's byte size 4-aligned. - Remove byteSize method from Element.
This commit is contained in:
parent
0deac32fde
commit
381288da52
17 changed files with 82 additions and 140 deletions
|
@ -24,7 +24,5 @@ public interface Layout {
|
||||||
ElementType type();
|
ElementType type();
|
||||||
|
|
||||||
int offset();
|
int offset();
|
||||||
|
|
||||||
int byteSize();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
package com.jozufozu.flywheel.backend.compile;
|
package com.jozufozu.flywheel.backend.compile;
|
||||||
|
|
||||||
|
import java.util.Collection;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
import org.jetbrains.annotations.Nullable;
|
import org.jetbrains.annotations.Nullable;
|
||||||
|
|
||||||
import com.google.common.collect.ImmutableList;
|
|
||||||
import com.jozufozu.flywheel.Flywheel;
|
import com.jozufozu.flywheel.Flywheel;
|
||||||
import com.jozufozu.flywheel.backend.compile.core.CompilerStats;
|
import com.jozufozu.flywheel.backend.compile.core.CompilerStats;
|
||||||
import com.jozufozu.flywheel.backend.compile.core.ProgramLinker;
|
import com.jozufozu.flywheel.backend.compile.core.ProgramLinker;
|
||||||
|
@ -18,12 +18,9 @@ public class CompilationHarness<K> {
|
||||||
private final SourceLoader sourceLoader;
|
private final SourceLoader sourceLoader;
|
||||||
private final ShaderCompiler shaderCompiler;
|
private final ShaderCompiler shaderCompiler;
|
||||||
private final ProgramLinker programLinker;
|
private final ProgramLinker programLinker;
|
||||||
private final ImmutableList<K> keys;
|
|
||||||
private final CompilerStats stats = new CompilerStats();
|
private final CompilerStats stats = new CompilerStats();
|
||||||
|
|
||||||
public CompilationHarness(ShaderSources sources, ImmutableList<K> keys, KeyCompiler<K> compiler) {
|
public CompilationHarness(ShaderSources sources, KeyCompiler<K> compiler) {
|
||||||
this.keys = keys;
|
|
||||||
|
|
||||||
this.compiler = compiler;
|
this.compiler = compiler;
|
||||||
sourceLoader = new SourceLoader(sources, stats);
|
sourceLoader = new SourceLoader(sources, stats);
|
||||||
shaderCompiler = new ShaderCompiler(stats);
|
shaderCompiler = new ShaderCompiler(stats);
|
||||||
|
@ -31,7 +28,7 @@ public class CompilationHarness<K> {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Nullable
|
@Nullable
|
||||||
public Map<K, GlProgram> compileAndReportErrors() {
|
public Map<K, GlProgram> compileAndReportErrors(Collection<K> keys) {
|
||||||
stats.start();
|
stats.start();
|
||||||
Map<K, GlProgram> out = new HashMap<>();
|
Map<K, GlProgram> out = new HashMap<>();
|
||||||
for (var key : keys) {
|
for (var key : keys) {
|
||||||
|
@ -59,28 +56,4 @@ public class CompilationHarness<K> {
|
||||||
public interface KeyCompiler<K> {
|
public interface KeyCompiler<K> {
|
||||||
@Nullable GlProgram compile(K key, SourceLoader loader, ShaderCompiler shaderCompiler, ProgramLinker programLinker);
|
@Nullable GlProgram compile(K key, SourceLoader loader, ShaderCompiler shaderCompiler, ProgramLinker programLinker);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class Builder<K> {
|
|
||||||
private final ShaderSources sources;
|
|
||||||
private ImmutableList<K> keys;
|
|
||||||
private KeyCompiler<K> compiler;
|
|
||||||
|
|
||||||
public Builder(ShaderSources sources) {
|
|
||||||
this.sources = sources;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Builder<K> keys(ImmutableList<K> keys) {
|
|
||||||
this.keys = keys;
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Builder<K> compiler(KeyCompiler<K> compiler) {
|
|
||||||
this.compiler = compiler;
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public CompilationHarness<K> build() {
|
|
||||||
return new CompilationHarness<>(sources, keys, compiler);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -44,15 +44,15 @@ public class Compile<K> {
|
||||||
return new ProgramLinkBuilder<>();
|
return new ProgramLinkBuilder<>();
|
||||||
}
|
}
|
||||||
|
|
||||||
public CompilationHarness.Builder<K> harness(ShaderSources sources) {
|
|
||||||
return new CompilationHarness.Builder<>(sources);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static class ProgramLinkBuilder<K> implements CompilationHarness.KeyCompiler<K> {
|
public static class ProgramLinkBuilder<K> implements CompilationHarness.KeyCompiler<K> {
|
||||||
private final Map<ShaderType, ShaderCompilerBuilder<K>> compilers = new EnumMap<>(ShaderType.class);
|
private final Map<ShaderType, ShaderCompilerBuilder<K>> compilers = new EnumMap<>(ShaderType.class);
|
||||||
private BiConsumer<K, GlProgram> onLink = (k, p) -> {
|
private BiConsumer<K, GlProgram> onLink = (k, p) -> {
|
||||||
};
|
};
|
||||||
|
|
||||||
|
public CompilationHarness<K> harness(ShaderSources sources) {
|
||||||
|
return new CompilationHarness<>(sources, this);
|
||||||
|
}
|
||||||
|
|
||||||
public ProgramLinkBuilder<K> link(ShaderCompilerBuilder<K> compilerBuilder) {
|
public ProgramLinkBuilder<K> link(ShaderCompilerBuilder<K> compilerBuilder) {
|
||||||
if (compilers.containsKey(compilerBuilder.shaderType)) {
|
if (compilers.containsKey(compilerBuilder.shaderType)) {
|
||||||
throw new IllegalArgumentException("Duplicate shader type: " + compilerBuilder.shaderType);
|
throw new IllegalArgumentException("Duplicate shader type: " + compilerBuilder.shaderType);
|
||||||
|
|
|
@ -17,7 +17,6 @@ import com.jozufozu.flywheel.backend.gl.shader.ShaderType;
|
||||||
import com.jozufozu.flywheel.backend.glsl.GlslVersion;
|
import com.jozufozu.flywheel.backend.glsl.GlslVersion;
|
||||||
import com.jozufozu.flywheel.backend.glsl.ShaderSources;
|
import com.jozufozu.flywheel.backend.glsl.ShaderSources;
|
||||||
import com.jozufozu.flywheel.backend.glsl.SourceComponent;
|
import com.jozufozu.flywheel.backend.glsl.SourceComponent;
|
||||||
import com.jozufozu.flywheel.lib.util.Unit;
|
|
||||||
|
|
||||||
import net.minecraft.resources.ResourceLocation;
|
import net.minecraft.resources.ResourceLocation;
|
||||||
|
|
||||||
|
@ -28,7 +27,7 @@ public class IndirectPrograms {
|
||||||
|
|
||||||
public static IndirectPrograms instance;
|
public static IndirectPrograms instance;
|
||||||
private static final Compile<InstanceType<?>> CULL = new Compile<>();
|
private static final Compile<InstanceType<?>> CULL = new Compile<>();
|
||||||
private static final Compile<Unit> UNIT = new Compile<>();
|
private static final Compile<ResourceLocation> UTIL = new Compile<>();
|
||||||
private final Map<PipelineProgramKey, GlProgram> pipeline;
|
private final Map<PipelineProgramKey, GlProgram> pipeline;
|
||||||
private final Map<InstanceType<?>, GlProgram> culling;
|
private final Map<InstanceType<?>, GlProgram> culling;
|
||||||
private final GlProgram apply;
|
private final GlProgram apply;
|
||||||
|
@ -43,19 +42,17 @@ public class IndirectPrograms {
|
||||||
|
|
||||||
static void reload(ShaderSources sources, ImmutableList<PipelineProgramKey> pipelineKeys, UniformComponent uniformComponent, List<SourceComponent> vertexComponents, List<SourceComponent> fragmentComponents) {
|
static void reload(ShaderSources sources, ImmutableList<PipelineProgramKey> pipelineKeys, UniformComponent uniformComponent, List<SourceComponent> vertexComponents, List<SourceComponent> fragmentComponents) {
|
||||||
_delete();
|
_delete();
|
||||||
var pipelineCompiler = PipelineCompiler.create(sources, Pipelines.INDIRECT, pipelineKeys, uniformComponent, vertexComponents, fragmentComponents);
|
var pipelineCompiler = PipelineCompiler.create(sources, Pipelines.INDIRECT, uniformComponent, vertexComponents, fragmentComponents);
|
||||||
var cullingCompiler = createCullingCompiler(uniformComponent, sources);
|
var cullingCompiler = createCullingCompiler(uniformComponent, sources);
|
||||||
var applyCompiler = createApplyCompiler(sources);
|
var applyCompiler = createUtilCompiler(sources);
|
||||||
var scatterCompiler = createScatterCompiler(sources);
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
var pipelineResult = pipelineCompiler.compileAndReportErrors();
|
var pipelineResult = pipelineCompiler.compileAndReportErrors(pipelineKeys);
|
||||||
var cullingResult = cullingCompiler.compileAndReportErrors();
|
var cullingResult = cullingCompiler.compileAndReportErrors(createCullingKeys());
|
||||||
var applyResult = applyCompiler.compileAndReportErrors();
|
var utils = applyCompiler.compileAndReportErrors(List.of(APPLY_SHADER_MAIN, SCATTER_SHADER_MAIN));
|
||||||
var scatterResult = scatterCompiler.compileAndReportErrors();
|
|
||||||
|
|
||||||
if (pipelineResult != null && cullingResult != null && applyResult != null && scatterResult != null) {
|
if (pipelineResult != null && cullingResult != null && utils != null) {
|
||||||
instance = new IndirectPrograms(pipelineResult, cullingResult, applyResult.get(Unit.INSTANCE), scatterResult.get(Unit.INSTANCE));
|
instance = new IndirectPrograms(pipelineResult, cullingResult, utils.get(APPLY_SHADER_MAIN), utils.get(SCATTER_SHADER_MAIN));
|
||||||
}
|
}
|
||||||
} catch (Throwable e) {
|
} catch (Throwable e) {
|
||||||
Flywheel.LOGGER.error("Failed to compile indirect programs", e);
|
Flywheel.LOGGER.error("Failed to compile indirect programs", e);
|
||||||
|
@ -90,37 +87,23 @@ public class IndirectPrograms {
|
||||||
}
|
}
|
||||||
|
|
||||||
private static CompilationHarness<InstanceType<?>> createCullingCompiler(UniformComponent uniformComponent, ShaderSources sources) {
|
private static CompilationHarness<InstanceType<?>> createCullingCompiler(UniformComponent uniformComponent, ShaderSources sources) {
|
||||||
return CULL.harness(sources)
|
return CULL.program()
|
||||||
.keys(createCullingKeys())
|
|
||||||
.compiler(CULL.program()
|
|
||||||
.link(CULL.shader(GlslVersion.V460, ShaderType.COMPUTE)
|
.link(CULL.shader(GlslVersion.V460, ShaderType.COMPUTE)
|
||||||
.define("_FLW_SUBGROUP_SIZE", GlCompat.SUBGROUP_SIZE)
|
.define("_FLW_SUBGROUP_SIZE", GlCompat.SUBGROUP_SIZE)
|
||||||
.withComponent(uniformComponent)
|
.withComponent(uniformComponent)
|
||||||
.withComponent(IndirectComponent::create)
|
.withComponent(IndirectComponent::create)
|
||||||
.withResource(InstanceType::cullShader)
|
.withResource(InstanceType::cullShader)
|
||||||
.withResource(CULL_SHADER_MAIN))
|
.withResource(CULL_SHADER_MAIN))
|
||||||
.then((key, program) -> program.setUniformBlockBinding("FlwUniforms", 0)))
|
.then((key, program) -> program.setUniformBlockBinding("FlwUniforms", 0))
|
||||||
.build();
|
.harness(sources);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static CompilationHarness<Unit> createApplyCompiler(ShaderSources sources) {
|
private static CompilationHarness<ResourceLocation> createUtilCompiler(ShaderSources sources) {
|
||||||
return UNIT.harness(sources)
|
return UTIL.program()
|
||||||
.keys(ImmutableList.of(Unit.INSTANCE))
|
.link(UTIL.shader(GlslVersion.V460, ShaderType.COMPUTE)
|
||||||
.compiler(UNIT.program()
|
|
||||||
.link(UNIT.shader(GlslVersion.V460, ShaderType.COMPUTE)
|
|
||||||
.define("_FLW_SUBGROUP_SIZE", GlCompat.SUBGROUP_SIZE)
|
.define("_FLW_SUBGROUP_SIZE", GlCompat.SUBGROUP_SIZE)
|
||||||
.withResource(APPLY_SHADER_MAIN)))
|
.withResource(s -> s))
|
||||||
.build();
|
.harness(sources);
|
||||||
}
|
|
||||||
|
|
||||||
private static CompilationHarness<Unit> createScatterCompiler(ShaderSources sources) {
|
|
||||||
return UNIT.harness(sources)
|
|
||||||
.keys(ImmutableList.of(Unit.INSTANCE))
|
|
||||||
.compiler(UNIT.program()
|
|
||||||
.link(UNIT.shader(GlslVersion.V460, ShaderType.COMPUTE)
|
|
||||||
.define("_FLW_SUBGROUP_SIZE", GlCompat.SUBGROUP_SIZE)
|
|
||||||
.withResource(SCATTER_SHADER_MAIN)))
|
|
||||||
.build();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public GlProgram getIndirectProgram(InstanceType<?> instanceType, Context contextShader) {
|
public GlProgram getIndirectProgram(InstanceType<?> instanceType, Context contextShader) {
|
||||||
|
|
|
@ -24,10 +24,10 @@ public class InstancingPrograms {
|
||||||
|
|
||||||
static void reload(ShaderSources sources, ImmutableList<PipelineProgramKey> pipelineKeys, UniformComponent uniformComponent, List<SourceComponent> vertexComponents, List<SourceComponent> fragmentComponents) {
|
static void reload(ShaderSources sources, ImmutableList<PipelineProgramKey> pipelineKeys, UniformComponent uniformComponent, List<SourceComponent> vertexComponents, List<SourceComponent> fragmentComponents) {
|
||||||
_delete();
|
_delete();
|
||||||
var instancingCompiler = PipelineCompiler.create(sources, Pipelines.INSTANCED_ARRAYS, pipelineKeys, uniformComponent, vertexComponents, fragmentComponents);
|
var instancingCompiler = PipelineCompiler.create(sources, Pipelines.INSTANCED_ARRAYS, uniformComponent, vertexComponents, fragmentComponents);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
var result = instancingCompiler.compileAndReportErrors();
|
var result = instancingCompiler.compileAndReportErrors(pipelineKeys);
|
||||||
|
|
||||||
if (result != null) {
|
if (result != null) {
|
||||||
instance = new InstancingPrograms(result);
|
instance = new InstancingPrograms(result);
|
||||||
|
|
|
@ -2,7 +2,6 @@ package com.jozufozu.flywheel.backend.compile;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
import com.google.common.collect.ImmutableList;
|
|
||||||
import com.jozufozu.flywheel.backend.InternalVertex;
|
import com.jozufozu.flywheel.backend.InternalVertex;
|
||||||
import com.jozufozu.flywheel.backend.compile.component.UniformComponent;
|
import com.jozufozu.flywheel.backend.compile.component.UniformComponent;
|
||||||
import com.jozufozu.flywheel.backend.gl.shader.ShaderType;
|
import com.jozufozu.flywheel.backend.gl.shader.ShaderType;
|
||||||
|
@ -12,10 +11,8 @@ import com.jozufozu.flywheel.backend.glsl.SourceComponent;
|
||||||
public class PipelineCompiler {
|
public class PipelineCompiler {
|
||||||
private static final Compile<PipelineProgramKey> PIPELINE = new Compile<>();
|
private static final Compile<PipelineProgramKey> PIPELINE = new Compile<>();
|
||||||
|
|
||||||
static CompilationHarness<PipelineProgramKey> create(ShaderSources sources, Pipeline pipeline, ImmutableList<PipelineProgramKey> pipelineKeys, UniformComponent uniformComponent, List<SourceComponent> vertexComponents, List<SourceComponent> fragmentComponents) {
|
static CompilationHarness<PipelineProgramKey> create(ShaderSources sources, Pipeline pipeline, UniformComponent uniformComponent, List<SourceComponent> vertexComponents, List<SourceComponent> fragmentComponents) {
|
||||||
return PIPELINE.harness(sources)
|
return PIPELINE.program()
|
||||||
.keys(pipelineKeys)
|
|
||||||
.compiler(PIPELINE.program()
|
|
||||||
.link(PIPELINE.shader(pipeline.glslVersion(), ShaderType.VERTEX)
|
.link(PIPELINE.shader(pipeline.glslVersion(), ShaderType.VERTEX)
|
||||||
.withComponent(uniformComponent)
|
.withComponent(uniformComponent)
|
||||||
.withResource(pipeline.vertexApiImpl())
|
.withResource(pipeline.vertexApiImpl())
|
||||||
|
@ -40,7 +37,7 @@ public class PipelineCompiler {
|
||||||
key.contextShader()
|
key.contextShader()
|
||||||
.onProgramLink(program);
|
.onProgramLink(program);
|
||||||
program.setUniformBlockBinding("FlwUniforms", 0);
|
program.setUniformBlockBinding("FlwUniforms", 0);
|
||||||
}))
|
})
|
||||||
.build();
|
.harness(sources);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -113,7 +113,7 @@ public class IndirectComponent implements SourceComponent {
|
||||||
// FIXME: I don't think we're unpacking signed byte/short values correctly
|
// FIXME: I don't think we're unpacking signed byte/short values correctly
|
||||||
// FIXME: we definitely don't consider endianness. this all assumes little endian which works on my machine.
|
// FIXME: we definitely don't consider endianness. this all assumes little endian which works on my machine.
|
||||||
var type = element.type();
|
var type = element.type();
|
||||||
var name = "_" + element.name();
|
var name = element.name();
|
||||||
|
|
||||||
if (type instanceof ScalarElementType scalar) {
|
if (type instanceof ScalarElementType scalar) {
|
||||||
return unpackScalar(name, packed, scalar);
|
return unpackScalar(name, packed, scalar);
|
||||||
|
|
|
@ -150,9 +150,6 @@ public abstract class AbstractInstancer<I extends Instance> implements Instancer
|
||||||
deleted.clear();
|
deleted.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void delete() {
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String toString() {
|
public String toString() {
|
||||||
return "AbstractInstancer[" + getInstanceCount() + ']';
|
return "AbstractInstancer[" + getInstanceCount() + ']';
|
||||||
|
|
|
@ -20,25 +20,18 @@ public abstract class InstancerStorage<N extends AbstractInstancer<?>> {
|
||||||
* <br>
|
* <br>
|
||||||
* See {@link #getInstancer} for insertion details.
|
* See {@link #getInstancer} for insertion details.
|
||||||
*/
|
*/
|
||||||
private final Map<InstancerKey<?>, N> instancers = new HashMap<>();
|
protected final Map<InstancerKey<?>, N> instancers = new HashMap<>();
|
||||||
/**
|
/**
|
||||||
* A list of instancers that have not yet been initialized.
|
* A list of instancers that have not yet been initialized.
|
||||||
* <br>
|
* <br>
|
||||||
* All new instancers land here before having resources allocated in {@link #flush}.
|
* All new instancers land here before having resources allocated in {@link #flush}.
|
||||||
* Write access to this list must be synchronized on {@link #creationLock}.
|
* Write access to this list must be synchronized on {@link #creationLock}.
|
||||||
*/
|
*/
|
||||||
private final List<UninitializedInstancer<N, ?>> uninitializedInstancers = new ArrayList<>();
|
protected final List<UninitializedInstancer<N, ?>> uninitializedInstancers = new ArrayList<>();
|
||||||
/**
|
/**
|
||||||
* Mutex for {@link #instancers} and {@link #uninitializedInstancers}.
|
* Mutex for {@link #instancers} and {@link #uninitializedInstancers}.
|
||||||
*/
|
*/
|
||||||
private final Object creationLock = new Object();
|
protected final Object creationLock = new Object();
|
||||||
|
|
||||||
/**
|
|
||||||
* A list of initialized instancers.
|
|
||||||
* <br>
|
|
||||||
* These are instancers that may need to be cleared or deleted.
|
|
||||||
*/
|
|
||||||
private final List<N> initializedInstancers = new ArrayList<>();
|
|
||||||
|
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
public <I extends Instance> Instancer<I> getInstancer(InstanceType<I> type, Model model, RenderStage stage) {
|
public <I extends Instance> Instancer<I> getInstancer(InstanceType<I> type, Model model, RenderStage stage) {
|
||||||
|
@ -69,21 +62,18 @@ public abstract class InstancerStorage<N extends AbstractInstancer<?>> {
|
||||||
public void delete() {
|
public void delete() {
|
||||||
instancers.clear();
|
instancers.clear();
|
||||||
uninitializedInstancers.clear();
|
uninitializedInstancers.clear();
|
||||||
|
|
||||||
initializedInstancers.forEach(AbstractInstancer::delete);
|
|
||||||
initializedInstancers.clear();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void flush() {
|
public void flush() {
|
||||||
for (var instancer : uninitializedInstancers) {
|
for (var instancer : uninitializedInstancers) {
|
||||||
add(instancer.key(), instancer.instancer(), instancer.model(), instancer.stage());
|
add(instancer.key(), instancer.instancer(), instancer.model(), instancer.stage());
|
||||||
initializedInstancers.add(instancer.instancer());
|
|
||||||
}
|
}
|
||||||
uninitializedInstancers.clear();
|
uninitializedInstancers.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void onRenderOriginChanged() {
|
public void onRenderOriginChanged() {
|
||||||
initializedInstancers.forEach(AbstractInstancer::clear);
|
instancers.values()
|
||||||
|
.forEach(AbstractInstancer::clear);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected abstract <I extends Instance> N create(InstanceType<I> type);
|
protected abstract <I extends Instance> N create(InstanceType<I> type);
|
||||||
|
|
|
@ -40,6 +40,9 @@ public class InstancedDrawManager extends InstancerStorage<InstancedInstancer<?>
|
||||||
}
|
}
|
||||||
|
|
||||||
public void delete() {
|
public void delete() {
|
||||||
|
instancers.values()
|
||||||
|
.forEach(InstancedInstancer::delete);
|
||||||
|
|
||||||
super.delete();
|
super.delete();
|
||||||
|
|
||||||
meshPool.delete();
|
meshPool.delete();
|
||||||
|
|
|
@ -119,7 +119,6 @@ public class InstancedInstancer<I extends Instance> extends AbstractInstancer<I>
|
||||||
vao.bindAttributes(1, startAttrib, instanceAttributes);
|
vao.bindAttributes(1, startAttrib, instanceAttributes);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void delete() {
|
public void delete() {
|
||||||
vbo.delete();
|
vbo.delete();
|
||||||
vbo = null;
|
vbo = null;
|
||||||
|
|
|
@ -14,7 +14,6 @@ import com.jozufozu.flywheel.api.layout.Layout.Element;
|
||||||
import com.jozufozu.flywheel.api.layout.LayoutBuilder;
|
import com.jozufozu.flywheel.api.layout.LayoutBuilder;
|
||||||
import com.jozufozu.flywheel.api.layout.ValueRepr;
|
import com.jozufozu.flywheel.api.layout.ValueRepr;
|
||||||
import com.jozufozu.flywheel.impl.layout.LayoutImpl.ElementImpl;
|
import com.jozufozu.flywheel.impl.layout.LayoutImpl.ElementImpl;
|
||||||
import com.jozufozu.flywheel.lib.math.MoreMath;
|
|
||||||
|
|
||||||
import it.unimi.dsi.fastutil.objects.Object2IntMap;
|
import it.unimi.dsi.fastutil.objects.Object2IntMap;
|
||||||
import it.unimi.dsi.fastutil.objects.Object2IntOpenHashMap;
|
import it.unimi.dsi.fastutil.objects.Object2IntOpenHashMap;
|
||||||
|
@ -126,9 +125,9 @@ public class LayoutBuilderImpl implements LayoutBuilder {
|
||||||
}
|
}
|
||||||
|
|
||||||
private LayoutBuilder element(String name, ElementType type) {
|
private LayoutBuilder element(String name, ElementType type) {
|
||||||
elements.add(new ElementImpl(name, type, offset, MoreMath.align4(type.byteSize())));
|
elements.add(new ElementImpl(name, type, offset));
|
||||||
|
// type.byteSize() is guaranteed to be 4-aligned.
|
||||||
offset += type.byteSize();
|
offset += type.byteSize();
|
||||||
offset = MoreMath.align4(offset);
|
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -71,6 +71,6 @@ final class LayoutImpl implements Layout {
|
||||||
return elements.equals(other.elements);
|
return elements.equals(other.elements);
|
||||||
}
|
}
|
||||||
|
|
||||||
record ElementImpl(String name, ElementType type, int offset, int byteSize) implements Element {
|
record ElementImpl(String name, ElementType type, int offset) implements Element {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,6 +4,7 @@ import org.jetbrains.annotations.Range;
|
||||||
|
|
||||||
import com.jozufozu.flywheel.api.layout.FloatRepr;
|
import com.jozufozu.flywheel.api.layout.FloatRepr;
|
||||||
import com.jozufozu.flywheel.api.layout.MatrixElementType;
|
import com.jozufozu.flywheel.api.layout.MatrixElementType;
|
||||||
|
import com.jozufozu.flywheel.lib.math.MoreMath;
|
||||||
|
|
||||||
record MatrixElementTypeImpl(FloatRepr repr, @Range(from = 2, to = 4) int rows, @Range(from = 2, to = 4) int columns,
|
record MatrixElementTypeImpl(FloatRepr repr, @Range(from = 2, to = 4) int rows, @Range(from = 2, to = 4) int columns,
|
||||||
int byteSize) implements MatrixElementType {
|
int byteSize) implements MatrixElementType {
|
||||||
|
@ -14,7 +15,7 @@ record MatrixElementTypeImpl(FloatRepr repr, @Range(from = 2, to = 4) int rows,
|
||||||
if (columns < 2 || columns > 4) {
|
if (columns < 2 || columns > 4) {
|
||||||
throw new IllegalArgumentException("Matrix element column count must be in range [2, 4]!");
|
throw new IllegalArgumentException("Matrix element column count must be in range [2, 4]!");
|
||||||
}
|
}
|
||||||
int byteSize = repr.byteSize() * rows * columns;
|
int byteSize = MoreMath.align4(repr.byteSize() * rows * columns);
|
||||||
return new MatrixElementTypeImpl(repr, rows, columns, byteSize);
|
return new MatrixElementTypeImpl(repr, rows, columns, byteSize);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,9 +2,10 @@ package com.jozufozu.flywheel.impl.layout;
|
||||||
|
|
||||||
import com.jozufozu.flywheel.api.layout.ScalarElementType;
|
import com.jozufozu.flywheel.api.layout.ScalarElementType;
|
||||||
import com.jozufozu.flywheel.api.layout.ValueRepr;
|
import com.jozufozu.flywheel.api.layout.ValueRepr;
|
||||||
|
import com.jozufozu.flywheel.lib.math.MoreMath;
|
||||||
|
|
||||||
record ScalarElementTypeImpl(ValueRepr repr, int byteSize) implements ScalarElementType {
|
record ScalarElementTypeImpl(ValueRepr repr, int byteSize) implements ScalarElementType {
|
||||||
static ScalarElementTypeImpl create(ValueRepr repr) {
|
static ScalarElementTypeImpl create(ValueRepr repr) {
|
||||||
return new ScalarElementTypeImpl(repr, repr.byteSize());
|
return new ScalarElementTypeImpl(repr, MoreMath.align4(repr.byteSize()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,6 +4,7 @@ import org.jetbrains.annotations.Range;
|
||||||
|
|
||||||
import com.jozufozu.flywheel.api.layout.ValueRepr;
|
import com.jozufozu.flywheel.api.layout.ValueRepr;
|
||||||
import com.jozufozu.flywheel.api.layout.VectorElementType;
|
import com.jozufozu.flywheel.api.layout.VectorElementType;
|
||||||
|
import com.jozufozu.flywheel.lib.math.MoreMath;
|
||||||
|
|
||||||
record VectorElementTypeImpl(ValueRepr repr, @Range(from = 2, to = 4) int size,
|
record VectorElementTypeImpl(ValueRepr repr, @Range(from = 2, to = 4) int size,
|
||||||
int byteSize) implements VectorElementType {
|
int byteSize) implements VectorElementType {
|
||||||
|
@ -13,7 +14,7 @@ record VectorElementTypeImpl(ValueRepr repr, @Range(from = 2, to = 4) int size,
|
||||||
throw new IllegalArgumentException("Vector element size must be in range [2, 4]!");
|
throw new IllegalArgumentException("Vector element size must be in range [2, 4]!");
|
||||||
}
|
}
|
||||||
|
|
||||||
int byteSize = repr.byteSize() * size;
|
int byteSize = MoreMath.align4(repr.byteSize() * size);
|
||||||
return new VectorElementTypeImpl(repr, size, byteSize);
|
return new VectorElementTypeImpl(repr, size, byteSize);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,12 +9,12 @@ public final class MoreMath {
|
||||||
*/
|
*/
|
||||||
public static final float SQRT_3_OVER_2 = (float) (Math.sqrt(3.0) / 2.0);
|
public static final float SQRT_3_OVER_2 = (float) (Math.sqrt(3.0) / 2.0);
|
||||||
|
|
||||||
public static int align16(int numToRound) {
|
public static int align16(int size) {
|
||||||
return (numToRound + 15) & ~15;
|
return (size + 15) & ~15;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static int align4(int offset1) {
|
public static int align4(int size) {
|
||||||
return (offset1 + 3) & ~3;
|
return (size + 3) & ~3;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static int ceilingDiv(int numerator, int denominator) {
|
public static int ceilingDiv(int numerator, int denominator) {
|
||||||
|
|
Loading…
Reference in a new issue