Taking textures back out of context

- Contexts shouldn't be responsible for binding the api defined textures
- Move light and overlay binding to TextureBinder
- Clean up & standardize material/texture setup/teardown calls
- Revert removal of MaterialRenderState texture stuff
- Add SimpleContext + Builder
- Rename Textures -> TextureSource
- Add API methods to set basic float uniforms, more to come later
- Throw errors if a context tries to set uniforms/textures with reserved
  names.
This commit is contained in:
Jozufozu 2024-02-12 23:00:13 -08:00
parent 1028ca4633
commit 6dfd5ed563
16 changed files with 303 additions and 101 deletions

View file

@ -10,7 +10,7 @@ public interface Context {
* *
* @param material The material about to be rendered. * @param material The material about to be rendered.
* @param shader The shader to prepare. * @param shader The shader to prepare.
* @param textures Source of the textures to use. * @param textureSource Source of the textures to use.
*/ */
void prepare(Material material, Shader shader, Textures textures); void prepare(Material material, Shader shader, TextureSource textureSource);
} }

View file

@ -1,8 +1,20 @@
package com.jozufozu.flywheel.api.context; package com.jozufozu.flywheel.api.context;
import org.joml.Matrix4fc;
import com.jozufozu.flywheel.api.BackendImplemented; import com.jozufozu.flywheel.api.BackendImplemented;
@BackendImplemented @BackendImplemented
public interface Shader { public interface Shader {
void setTexture(String glslName, Texture texture); void setTexture(String glslName, Texture texture);
void setFloat(String glslName, float value);
void setVec2(String glslName, float x, float y);
void setVec3(String glslName, float x, float y, float z);
void setVec4(String glslName, float x, float y, float z, float w);
void setMat4(String glslName, Matrix4fc matrix);
} }

View file

@ -5,7 +5,7 @@ import com.jozufozu.flywheel.api.BackendImplemented;
import net.minecraft.resources.ResourceLocation; import net.minecraft.resources.ResourceLocation;
@BackendImplemented @BackendImplemented
public interface Textures { public interface TextureSource {
/** /**
* Get a built-in texture by its resource location. * Get a built-in texture by its resource location.
* *

View file

@ -5,6 +5,7 @@ import java.util.List;
import com.jozufozu.flywheel.backend.InternalVertex; import com.jozufozu.flywheel.backend.InternalVertex;
import com.jozufozu.flywheel.backend.compile.core.CompilationHarness; import com.jozufozu.flywheel.backend.compile.core.CompilationHarness;
import com.jozufozu.flywheel.backend.compile.core.Compile; import com.jozufozu.flywheel.backend.compile.core.Compile;
import com.jozufozu.flywheel.backend.gl.shader.GlProgram;
import com.jozufozu.flywheel.backend.gl.shader.ShaderType; import com.jozufozu.flywheel.backend.gl.shader.ShaderType;
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;
@ -52,6 +53,14 @@ public class PipelineCompiler {
.then((key, program) -> { .then((key, program) -> {
program.setUniformBlockBinding("_FlwFrameUniforms", 0); program.setUniformBlockBinding("_FlwFrameUniforms", 0);
program.setUniformBlockBinding("_FlwFogUniforms", 1); program.setUniformBlockBinding("_FlwFogUniforms", 1);
program.bind();
program.setSamplerBinding("_flw_diffuseTex", 0);
program.setSamplerBinding("_flw_overlayTex", 1);
program.setSamplerBinding("_flw_lightTex", 2);
GlProgram.unbind();
}) })
.harness(pipeline.compilerMarker(), sources); .harness(pipeline.compilerMarker(), sources);
} }

View file

@ -8,9 +8,13 @@ import com.jozufozu.flywheel.api.material.DepthTest;
import com.jozufozu.flywheel.api.material.Material; import com.jozufozu.flywheel.api.material.Material;
import com.jozufozu.flywheel.api.material.Transparency; import com.jozufozu.flywheel.api.material.Transparency;
import com.jozufozu.flywheel.api.material.WriteMask; import com.jozufozu.flywheel.api.material.WriteMask;
import com.jozufozu.flywheel.backend.gl.GlTextureUnit;
import com.mojang.blaze3d.platform.GlStateManager; import com.mojang.blaze3d.platform.GlStateManager;
import com.mojang.blaze3d.systems.RenderSystem; import com.mojang.blaze3d.systems.RenderSystem;
import net.minecraft.client.Minecraft;
import net.minecraft.client.renderer.texture.AbstractTexture;
public final class MaterialRenderState { public final class MaterialRenderState {
public static final Comparator<Material> COMPARATOR = Comparator.comparing(Material::texture) public static final Comparator<Material> COMPARATOR = Comparator.comparing(Material::texture)
.thenComparing(Material::blur) .thenComparing(Material::blur)
@ -25,6 +29,7 @@ public final class MaterialRenderState {
} }
public static void setup(Material material) { public static void setup(Material material) {
setupTexture(material);
setupBackfaceCulling(material.backfaceCulling()); setupBackfaceCulling(material.backfaceCulling());
setupPolygonOffset(material.polygonOffset()); setupPolygonOffset(material.polygonOffset());
setupDepthTest(material.depthTest()); setupDepthTest(material.depthTest());
@ -32,6 +37,17 @@ public final class MaterialRenderState {
setupWriteMask(material.writeMask()); setupWriteMask(material.writeMask());
} }
private static void setupTexture(Material material) {
GlTextureUnit.T0.makeActive();
AbstractTexture texture = Minecraft.getInstance()
.getTextureManager()
.getTexture(material.texture());
texture.setFilter(material.blur(), material.mipmap());
var textureId = texture.getId();
RenderSystem.setShaderTexture(0, textureId);
RenderSystem.bindTexture(textureId);
}
private static void setupBackfaceCulling(boolean backfaceCulling) { private static void setupBackfaceCulling(boolean backfaceCulling) {
if (backfaceCulling) { if (backfaceCulling) {
RenderSystem.enableCull(); RenderSystem.enableCull();
@ -125,6 +141,7 @@ public final class MaterialRenderState {
} }
public static void reset() { public static void reset() {
resetTexture();
resetBackfaceCulling(); resetBackfaceCulling();
resetPolygonOffset(); resetPolygonOffset();
resetDepthTest(); resetDepthTest();
@ -132,6 +149,11 @@ public final class MaterialRenderState {
resetWriteMask(); resetWriteMask();
} }
private static void resetTexture() {
GlTextureUnit.T0.makeActive();
RenderSystem.setShaderTexture(0, 0);
}
private static void resetBackfaceCulling() { private static void resetBackfaceCulling() {
RenderSystem.enableCull(); RenderSystem.enableCull();
} }

View file

@ -26,7 +26,7 @@ import com.jozufozu.flywheel.api.model.Model;
import com.jozufozu.flywheel.backend.compile.IndirectPrograms; import com.jozufozu.flywheel.backend.compile.IndirectPrograms;
import com.jozufozu.flywheel.backend.engine.MaterialRenderState; import com.jozufozu.flywheel.backend.engine.MaterialRenderState;
import com.jozufozu.flywheel.backend.engine.textures.TextureBinder; import com.jozufozu.flywheel.backend.engine.textures.TextureBinder;
import com.jozufozu.flywheel.backend.engine.textures.TexturesImpl; import com.jozufozu.flywheel.backend.engine.textures.TextureSourceImpl;
import com.jozufozu.flywheel.backend.engine.uniform.Uniforms; import com.jozufozu.flywheel.backend.engine.uniform.Uniforms;
import com.jozufozu.flywheel.backend.gl.Driver; import com.jozufozu.flywheel.backend.gl.Driver;
import com.jozufozu.flywheel.backend.gl.GlCompat; import com.jozufozu.flywheel.backend.gl.GlCompat;
@ -185,7 +185,7 @@ public class IndirectCullingGroup<I extends Instance> {
hasNewDraws = true; hasNewDraws = true;
} }
public void submit(RenderStage stage, TexturesImpl textures) { public void submit(RenderStage stage, TextureSourceImpl textures) {
if (nothingToDo(stage)) { if (nothingToDo(stage)) {
return; return;
} }

View file

@ -23,7 +23,8 @@ import com.jozufozu.flywheel.backend.engine.InstancerKey;
import com.jozufozu.flywheel.backend.engine.InstancerStorage; import com.jozufozu.flywheel.backend.engine.InstancerStorage;
import com.jozufozu.flywheel.backend.engine.MaterialRenderState; import com.jozufozu.flywheel.backend.engine.MaterialRenderState;
import com.jozufozu.flywheel.backend.engine.textures.TextureBinder; import com.jozufozu.flywheel.backend.engine.textures.TextureBinder;
import com.jozufozu.flywheel.backend.engine.textures.TexturesImpl; import com.jozufozu.flywheel.backend.engine.textures.TextureSourceImpl;
import com.jozufozu.flywheel.backend.gl.GlStateTracker;
import com.jozufozu.flywheel.backend.gl.buffer.GlBuffer; import com.jozufozu.flywheel.backend.gl.buffer.GlBuffer;
import com.jozufozu.flywheel.backend.gl.buffer.GlBufferType; import com.jozufozu.flywheel.backend.gl.buffer.GlBufferType;
import com.jozufozu.flywheel.lib.context.ContextShaders; import com.jozufozu.flywheel.lib.context.ContextShaders;
@ -39,7 +40,7 @@ import net.minecraft.client.resources.model.ModelBakery;
public class IndirectDrawManager extends InstancerStorage<IndirectInstancer<?>> { public class IndirectDrawManager extends InstancerStorage<IndirectInstancer<?>> {
private final IndirectPrograms programs; private final IndirectPrograms programs;
private final StagingBuffer stagingBuffer; private final StagingBuffer stagingBuffer;
private final TexturesImpl textures = new TexturesImpl(); private final TextureSourceImpl textures = new TextureSourceImpl();
private final Map<GroupKey<?>, IndirectCullingGroup<?>> cullingGroups = new HashMap<>(); private final Map<GroupKey<?>, IndirectCullingGroup<?>> cullingGroups = new HashMap<>();
private final GlBuffer crumblingDrawBuffer = new GlBuffer(); private final GlBuffer crumblingDrawBuffer = new GlBuffer();
@ -71,9 +72,14 @@ public class IndirectDrawManager extends InstancerStorage<IndirectInstancer<?>>
} }
public void renderStage(RenderStage stage) { public void renderStage(RenderStage stage) {
TextureBinder.bindLightAndOverlay();
for (var group : cullingGroups.values()) { for (var group : cullingGroups.values()) {
group.submit(stage, textures); group.submit(stage, textures);
} }
MaterialRenderState.reset();
TextureBinder.resetLightAndOverlay();
} }
@Override @Override
@ -117,50 +123,56 @@ public class IndirectDrawManager extends InstancerStorage<IndirectInstancer<?>>
return; return;
} }
var crumblingMaterial = SimpleMaterial.builder(); try (var state = GlStateTracker.getRestoreState()) {
TextureBinder.bindLightAndOverlay();
// Scratch memory for writing draw commands. var crumblingMaterial = SimpleMaterial.builder();
var block = MemoryBlock.malloc(IndirectBuffers.DRAW_COMMAND_STRIDE);
GlBufferType.DRAW_INDIRECT_BUFFER.bind(crumblingDrawBuffer.handle()); // Scratch memory for writing draw commands.
glBindBufferRange(GL_SHADER_STORAGE_BUFFER, IndirectBuffers.DRAW_INDEX, crumblingDrawBuffer.handle(), 0, IndirectBuffers.DRAW_COMMAND_STRIDE); var block = MemoryBlock.malloc(IndirectBuffers.DRAW_COMMAND_STRIDE);
for (var instanceTypeEntry : byType.entrySet()) { GlBufferType.DRAW_INDIRECT_BUFFER.bind(crumblingDrawBuffer.handle());
var byProgress = instanceTypeEntry.getValue(); glBindBufferRange(GL_SHADER_STORAGE_BUFFER, IndirectBuffers.DRAW_INDEX, crumblingDrawBuffer.handle(), 0, IndirectBuffers.DRAW_COMMAND_STRIDE);
// Set up the crumbling program buffers. Nothing changes here between draws. for (var instanceTypeEntry : byType.entrySet()) {
var program = cullingGroups.get(instanceTypeEntry.getKey()) var byProgress = instanceTypeEntry.getValue();
.bindWithContextShader(ContextShaders.CRUMBLING);
for (var progressEntry : byProgress.int2ObjectEntrySet()) { // Set up the crumbling program buffers. Nothing changes here between draws.
for (var instanceHandlePair : progressEntry.getValue()) { var program = cullingGroups.get(instanceTypeEntry.getKey())
IndirectInstancer<?> instancer = instanceHandlePair.first(); .bindWithContextShader(ContextShaders.CRUMBLING);
int instanceIndex = instanceHandlePair.second().index;
for (IndirectDraw draw : instancer.draws()) { for (var progressEntry : byProgress.int2ObjectEntrySet()) {
for (var instanceHandlePair : progressEntry.getValue()) {
IndirectInstancer<?> instancer = instanceHandlePair.first();
int instanceIndex = instanceHandlePair.second().index;
// Transform the material to be suited for crumbling. for (IndirectDraw draw : instancer.draws()) {
CommonCrumbling.applyCrumblingProperties(crumblingMaterial, draw.material());
Contexts.CRUMBLING.get(progressEntry.getIntKey())
.prepare(crumblingMaterial, program, textures);
// Set up gl state for the draw. // Transform the material to be suited for crumbling.
MaterialRenderState.setup(crumblingMaterial); CommonCrumbling.applyCrumblingProperties(crumblingMaterial, draw.material());
var context = Contexts.CRUMBLING.get(progressEntry.getIntKey());
// Upload the draw command. context.prepare(crumblingMaterial, program, textures);
draw.writeWithOverrides(block.ptr(), instanceIndex, crumblingMaterial); MaterialRenderState.setup(crumblingMaterial);
crumblingDrawBuffer.upload(block);
// Submit! Everything is already bound by here. // Upload the draw command.
glDrawElementsIndirect(GL_TRIANGLES, GL_UNSIGNED_INT, 0); draw.writeWithOverrides(block.ptr(), instanceIndex, crumblingMaterial);
crumblingDrawBuffer.upload(block);
TextureBinder.resetTextureBindings(); // Submit! Everything is already bound by here.
glDrawElementsIndirect(GL_TRIANGLES, GL_UNSIGNED_INT, 0);
TextureBinder.resetTextureBindings();
}
} }
} }
} }
}
block.free(); MaterialRenderState.reset();
TextureBinder.resetLightAndOverlay();
block.free();
}
} }
private static Map<GroupKey<?>, Int2ObjectMap<List<Pair<IndirectInstancer<?>, InstanceHandleImpl>>>> doCrumblingSort(List<Engine.CrumblingBlock> crumblingBlocks) { private static Map<GroupKey<?>, Int2ObjectMap<List<Pair<IndirectInstancer<?>, InstanceHandleImpl>>>> doCrumblingSort(List<Engine.CrumblingBlock> crumblingBlocks) {

View file

@ -10,7 +10,6 @@ import com.jozufozu.flywheel.backend.compile.IndirectPrograms;
import com.jozufozu.flywheel.backend.engine.AbstractEngine; import com.jozufozu.flywheel.backend.engine.AbstractEngine;
import com.jozufozu.flywheel.backend.engine.AbstractInstancer; import com.jozufozu.flywheel.backend.engine.AbstractInstancer;
import com.jozufozu.flywheel.backend.engine.InstancerStorage; import com.jozufozu.flywheel.backend.engine.InstancerStorage;
import com.jozufozu.flywheel.backend.engine.MaterialRenderState;
import com.jozufozu.flywheel.backend.engine.uniform.Uniforms; import com.jozufozu.flywheel.backend.engine.uniform.Uniforms;
import com.jozufozu.flywheel.backend.gl.GlStateTracker; import com.jozufozu.flywheel.backend.gl.GlStateTracker;
import com.jozufozu.flywheel.lib.task.Flag; import com.jozufozu.flywheel.lib.task.Flag;
@ -55,8 +54,6 @@ public class IndirectEngine extends AbstractEngine {
try (var restoreState = GlStateTracker.getRestoreState()) { try (var restoreState = GlStateTracker.getRestoreState()) {
drawManager.renderStage(stage); drawManager.renderStage(stage);
MaterialRenderState.reset();
} }
} }
@ -64,11 +61,7 @@ public class IndirectEngine extends AbstractEngine {
public void renderCrumbling(TaskExecutor executor, RenderContext context, List<CrumblingBlock> crumblingBlocks) { public void renderCrumbling(TaskExecutor executor, RenderContext context, List<CrumblingBlock> crumblingBlocks) {
executor.syncUntil(flushFlag::isRaised); executor.syncUntil(flushFlag::isRaised);
try (var restoreState = GlStateTracker.getRestoreState()) { drawManager.renderCrumbling(crumblingBlocks);
drawManager.renderCrumbling(crumblingBlocks);
MaterialRenderState.reset();
}
} }
@Override @Override

View file

@ -6,7 +6,7 @@ import java.util.List;
import java.util.Map; import java.util.Map;
import com.jozufozu.flywheel.api.backend.Engine; import com.jozufozu.flywheel.api.backend.Engine;
import com.jozufozu.flywheel.api.context.Textures; import com.jozufozu.flywheel.api.context.TextureSource;
import com.jozufozu.flywheel.api.instance.Instance; import com.jozufozu.flywheel.api.instance.Instance;
import com.jozufozu.flywheel.backend.compile.InstancingPrograms; import com.jozufozu.flywheel.backend.compile.InstancingPrograms;
import com.jozufozu.flywheel.backend.engine.CommonCrumbling; import com.jozufozu.flywheel.backend.engine.CommonCrumbling;
@ -24,7 +24,7 @@ import it.unimi.dsi.fastutil.ints.Int2ObjectMap;
import net.minecraft.client.resources.model.ModelBakery; import net.minecraft.client.resources.model.ModelBakery;
public class InstancedCrumbling { public class InstancedCrumbling {
public static void render(List<Engine.CrumblingBlock> crumblingBlocks, InstancingPrograms programs, Textures textures) { public static void render(List<Engine.CrumblingBlock> crumblingBlocks, InstancingPrograms programs, TextureSource textureSource) {
// Sort draw calls into buckets, so we don't have to do as many shader binds. // Sort draw calls into buckets, so we don't have to do as many shader binds.
var byShaderState = doCrumblingSort(crumblingBlocks); var byShaderState = doCrumblingSort(crumblingBlocks);
@ -35,6 +35,8 @@ public class InstancedCrumbling {
var crumblingMaterial = SimpleMaterial.builder(); var crumblingMaterial = SimpleMaterial.builder();
try (var state = GlStateTracker.getRestoreState()) { try (var state = GlStateTracker.getRestoreState()) {
TextureBinder.bindLightAndOverlay();
for (var shaderStateEntry : byShaderState.entrySet()) { for (var shaderStateEntry : byShaderState.entrySet()) {
var byProgress = shaderStateEntry.getValue(); var byProgress = shaderStateEntry.getValue();
@ -61,8 +63,8 @@ public class InstancedCrumbling {
continue; continue;
} }
Contexts.CRUMBLING.get(progressEntry.getIntKey()) var context = Contexts.CRUMBLING.get(progressEntry.getIntKey());
.prepare(crumblingMaterial, program, textures); context.prepare(crumblingMaterial, program, textureSource);
drawCalls.forEach(Runnable::run); drawCalls.forEach(Runnable::run);
@ -71,6 +73,7 @@ public class InstancedCrumbling {
} }
MaterialRenderState.reset(); MaterialRenderState.reset();
TextureBinder.resetLightAndOverlay();
} }
} }

View file

@ -17,7 +17,7 @@ import com.jozufozu.flywheel.backend.engine.InstancerStorage;
import com.jozufozu.flywheel.backend.engine.MaterialEncoder; import com.jozufozu.flywheel.backend.engine.MaterialEncoder;
import com.jozufozu.flywheel.backend.engine.MaterialRenderState; import com.jozufozu.flywheel.backend.engine.MaterialRenderState;
import com.jozufozu.flywheel.backend.engine.textures.TextureBinder; import com.jozufozu.flywheel.backend.engine.textures.TextureBinder;
import com.jozufozu.flywheel.backend.engine.textures.TexturesImpl; import com.jozufozu.flywheel.backend.engine.textures.TextureSourceImpl;
import com.jozufozu.flywheel.backend.engine.uniform.Uniforms; import com.jozufozu.flywheel.backend.engine.uniform.Uniforms;
import com.jozufozu.flywheel.backend.gl.GlStateTracker; import com.jozufozu.flywheel.backend.gl.GlStateTracker;
import com.jozufozu.flywheel.backend.gl.shader.GlProgram; import com.jozufozu.flywheel.backend.gl.shader.GlProgram;
@ -27,7 +27,7 @@ import com.jozufozu.flywheel.lib.task.SyncedPlan;
public class InstancingEngine extends AbstractEngine { public class InstancingEngine extends AbstractEngine {
private final InstancingPrograms programs; private final InstancingPrograms programs;
private final TexturesImpl textures = new TexturesImpl(); private final TextureSourceImpl textures = new TextureSourceImpl();
private final InstancedDrawManager drawManager = new InstancedDrawManager(); private final InstancedDrawManager drawManager = new InstancedDrawManager();
private final Flag flushFlag = new NamedFlag("flushed"); private final Flag flushFlag = new NamedFlag("flushed");
@ -88,6 +88,8 @@ public class InstancingEngine extends AbstractEngine {
} }
private void render(InstancedDrawManager.DrawSet drawSet) { private void render(InstancedDrawManager.DrawSet drawSet) {
TextureBinder.bindLightAndOverlay();
for (var entry : drawSet) { for (var entry : drawSet) {
var shader = entry.getKey(); var shader = entry.getKey();
var drawCalls = entry.getValue(); var drawCalls = entry.getValue();
@ -98,16 +100,17 @@ public class InstancingEngine extends AbstractEngine {
continue; continue;
} }
var program = programs.get(shader.instanceType(), shader.context() var context = shader.context();
.contextShader()); var material = shader.material();
var program = programs.get(shader.instanceType(), context.contextShader());
program.bind(); program.bind();
Uniforms.bindForDraw(); Uniforms.bindForDraw();
uploadMaterialUniform(program, shader.material()); uploadMaterialUniform(program, material);
shader.context() context.prepare(material, program, textures);
.prepare(shader.material(), program, textures); MaterialRenderState.setup(material);
MaterialRenderState.setup(shader.material());
for (var drawCall : drawCalls) { for (var drawCall : drawCalls) {
drawCall.render(); drawCall.render();
@ -116,6 +119,7 @@ public class InstancingEngine extends AbstractEngine {
} }
MaterialRenderState.reset(); MaterialRenderState.reset();
TextureBinder.resetLightAndOverlay();
} }
public static void uploadMaterialUniform(GlProgram program, Material material) { public static void uploadMaterialUniform(GlProgram program, Material material) {

View file

@ -2,16 +2,23 @@ package com.jozufozu.flywheel.backend.engine.textures;
import static org.lwjgl.opengl.GL13.GL_TEXTURE0; import static org.lwjgl.opengl.GL13.GL_TEXTURE0;
import com.jozufozu.flywheel.backend.gl.GlTextureUnit;
import com.mojang.blaze3d.systems.RenderSystem; import com.mojang.blaze3d.systems.RenderSystem;
import it.unimi.dsi.fastutil.ints.Int2IntArrayMap; import it.unimi.dsi.fastutil.ints.Int2IntArrayMap;
import it.unimi.dsi.fastutil.ints.Int2IntMap; import it.unimi.dsi.fastutil.ints.Int2IntMap;
import net.minecraft.client.Minecraft;
public class TextureBinder { public class TextureBinder {
// TODO: some kind of cache eviction when the program changes // TODO: some kind of cache eviction when the program changes
// so we don't always reset and bind the light and overlay textures? // so we don't always reset and bind the light and overlay textures?
private static final Int2IntMap texturesToSamplerUnits = new Int2IntArrayMap(); private static final Int2IntMap texturesToSamplerUnits = new Int2IntArrayMap();
private static int nextSamplerUnit = 0; // 0 is reserved for diffuse
// 1 is overlay
// 2 is light
// 3..n are for whatever else the context needs
private static final int baseSamplerUnit = 3;
private static int nextSamplerUnit = baseSamplerUnit;
/** /**
* Binds the given texture to the next available texture unit, returning the unit it was bound to. * Binds the given texture to the next available texture unit, returning the unit it was bound to.
@ -29,7 +36,7 @@ public class TextureBinder {
} }
public static void resetTextureBindings() { public static void resetTextureBindings() {
nextSamplerUnit = 0; nextSamplerUnit = baseSamplerUnit;
for (Int2IntMap.Entry entry : texturesToSamplerUnits.int2IntEntrySet()) { for (Int2IntMap.Entry entry : texturesToSamplerUnits.int2IntEntrySet()) {
RenderSystem.activeTexture(GL_TEXTURE0 + entry.getIntValue()); RenderSystem.activeTexture(GL_TEXTURE0 + entry.getIntValue());
@ -38,4 +45,26 @@ public class TextureBinder {
texturesToSamplerUnits.clear(); texturesToSamplerUnits.clear();
} }
public static void bindLightAndOverlay() {
var gameRenderer = Minecraft.getInstance().gameRenderer;
gameRenderer.overlayTexture()
.setupOverlayColor();
gameRenderer.lightTexture()
.turnOnLightLayer();
GlTextureUnit.T1.makeActive();
RenderSystem.bindTexture(RenderSystem.getShaderTexture(1));
GlTextureUnit.T2.makeActive();
RenderSystem.bindTexture(RenderSystem.getShaderTexture(2));
}
public static void resetLightAndOverlay() {
var gameRenderer = Minecraft.getInstance().gameRenderer;
gameRenderer.overlayTexture()
.teardownOverlayColor();
gameRenderer.lightTexture()
.turnOffLightLayer();
}
} }

View file

@ -4,7 +4,7 @@ import java.util.HashMap;
import java.util.Map; import java.util.Map;
import com.jozufozu.flywheel.api.context.Texture; import com.jozufozu.flywheel.api.context.Texture;
import com.jozufozu.flywheel.api.context.Textures; import com.jozufozu.flywheel.api.context.TextureSource;
import com.jozufozu.flywheel.backend.mixin.LightTextureAccessor; import com.jozufozu.flywheel.backend.mixin.LightTextureAccessor;
import com.jozufozu.flywheel.backend.mixin.OverlayTextureAccessor; import com.jozufozu.flywheel.backend.mixin.OverlayTextureAccessor;
@ -12,13 +12,13 @@ import net.minecraft.client.Minecraft;
import net.minecraft.client.renderer.texture.AbstractTexture; import net.minecraft.client.renderer.texture.AbstractTexture;
import net.minecraft.resources.ResourceLocation; import net.minecraft.resources.ResourceLocation;
public class TexturesImpl implements Textures { public class TextureSourceImpl implements TextureSource {
private final DirectTexture lightTexture; private final DirectTexture lightTexture;
private final DirectTexture overlayTexture; private final DirectTexture overlayTexture;
private final Map<ResourceLocation, WrappedTexture> wrappers = new HashMap<>(); private final Map<ResourceLocation, WrappedTexture> wrappers = new HashMap<>();
public TexturesImpl() { public TextureSourceImpl() {
var gameRenderer = Minecraft.getInstance().gameRenderer; var gameRenderer = Minecraft.getInstance().gameRenderer;
this.lightTexture = new DirectTexture(((LightTextureAccessor) gameRenderer.lightTexture()).flywheel$texture() this.lightTexture = new DirectTexture(((LightTextureAccessor) gameRenderer.lightTexture()).flywheel$texture()

View file

@ -2,11 +2,17 @@ package com.jozufozu.flywheel.backend.gl.shader;
import static org.lwjgl.opengl.GL20.glDeleteProgram; import static org.lwjgl.opengl.GL20.glDeleteProgram;
import static org.lwjgl.opengl.GL20.glGetUniformLocation; import static org.lwjgl.opengl.GL20.glGetUniformLocation;
import static org.lwjgl.opengl.GL20.glUniform1f;
import static org.lwjgl.opengl.GL20.glUniform1i; import static org.lwjgl.opengl.GL20.glUniform1i;
import static org.lwjgl.opengl.GL20.glUniform2f;
import static org.lwjgl.opengl.GL20.glUniform3f;
import static org.lwjgl.opengl.GL20.glUniform4f;
import static org.lwjgl.opengl.GL20.glUniformMatrix4fv;
import static org.lwjgl.opengl.GL31.GL_INVALID_INDEX; import static org.lwjgl.opengl.GL31.GL_INVALID_INDEX;
import static org.lwjgl.opengl.GL31.glGetUniformBlockIndex; import static org.lwjgl.opengl.GL31.glGetUniformBlockIndex;
import static org.lwjgl.opengl.GL31.glUniformBlockBinding; import static org.lwjgl.opengl.GL31.glUniformBlockBinding;
import org.joml.Matrix4fc;
import org.slf4j.Logger; import org.slf4j.Logger;
import com.jozufozu.flywheel.api.context.Shader; import com.jozufozu.flywheel.api.context.Shader;
@ -39,6 +45,8 @@ public class GlProgram extends GlObject implements Shader {
@Override @Override
public void setTexture(String glslName, Texture texture) { public void setTexture(String glslName, Texture texture) {
throwIfReserved(glslName);
if (!(texture instanceof IdentifiedTexture identified)) { if (!(texture instanceof IdentifiedTexture identified)) {
return; return;
} }
@ -55,6 +63,71 @@ public class GlProgram extends GlObject implements Shader {
glUniform1i(uniform, binding); glUniform1i(uniform, binding);
} }
@Override
public void setFloat(String glslName, float value) {
throwIfReserved(glslName);
int uniform = getUniformLocation(glslName);
if (uniform < 0) {
return;
}
glUniform1f(uniform, value);
}
@Override
public void setVec2(String glslName, float x, float y) {
throwIfReserved(glslName);
int uniform = getUniformLocation(glslName);
if (uniform < 0) {
return;
}
glUniform2f(uniform, x, y);
}
@Override
public void setVec3(String glslName, float x, float y, float z) {
throwIfReserved(glslName);
int uniform = getUniformLocation(glslName);
if (uniform < 0) {
return;
}
glUniform3f(uniform, x, y, z);
}
@Override
public void setVec4(String glslName, float x, float y, float z, float w) {
throwIfReserved(glslName);
int uniform = getUniformLocation(glslName);
if (uniform < 0) {
return;
}
glUniform4f(uniform, x, y, z, w);
}
@Override
public void setMat4(String glslName, Matrix4fc matrix) {
throwIfReserved(glslName);
int uniform = getUniformLocation(glslName);
if (uniform < 0) {
return;
}
glUniformMatrix4fv(uniform, false, matrix.get(new float[16]));
}
/** /**
* Retrieves the index of the uniform with the given name. * Retrieves the index of the uniform with the given name.
* *
@ -101,4 +174,18 @@ public class GlProgram extends GlObject implements Shader {
protected void deleteInternal(int handle) { protected void deleteInternal(int handle) {
glDeleteProgram(handle); glDeleteProgram(handle);
} }
public static void throwIfReserved(String glslName) {
if (glslName.startsWith("flw_")) {
throw new IllegalArgumentException("Uniform names starting with flw_are reserved");
}
if (glslName.startsWith("_flw_")) {
throw new IllegalArgumentException("Uniform names starting with _flw_ are reserved for internal use");
}
if (glslName.startsWith("gl_")) {
throw new IllegalArgumentException("Uniform names cannot start with gl_");
}
}
} }

View file

@ -1,10 +1,6 @@
package com.jozufozu.flywheel.lib.context; package com.jozufozu.flywheel.lib.context;
import com.jozufozu.flywheel.api.context.Context; import com.jozufozu.flywheel.api.context.Context;
import com.jozufozu.flywheel.api.context.ContextShader;
import com.jozufozu.flywheel.api.context.Shader;
import com.jozufozu.flywheel.api.context.Textures;
import com.jozufozu.flywheel.api.material.Material;
import it.unimi.dsi.fastutil.ints.Int2ObjectArrayMap; import it.unimi.dsi.fastutil.ints.Int2ObjectArrayMap;
import it.unimi.dsi.fastutil.ints.Int2ObjectMap; import it.unimi.dsi.fastutil.ints.Int2ObjectMap;
@ -12,22 +8,8 @@ import it.unimi.dsi.fastutil.ints.Int2ObjectMaps;
import net.minecraft.client.resources.model.ModelBakery; import net.minecraft.client.resources.model.ModelBakery;
public final class Contexts { public final class Contexts {
public static final Context DEFAULT = new Context() { public static final Context DEFAULT = SimpleContext.builder(ContextShaders.DEFAULT)
@Override .build();
public ContextShader contextShader() {
return ContextShaders.DEFAULT;
}
@Override
public void prepare(Material material, Shader shader, Textures textures) {
var texture = textures.byName(material.texture());
texture.filter(material.blur(), material.mipmap());
shader.setTexture("_flw_diffuseTex", texture);
shader.setTexture("_flw_overlayTex", textures.overlay());
shader.setTexture("_flw_lightTex", textures.light());
}
};
public static final Int2ObjectMap<Context> CRUMBLING; public static final Int2ObjectMap<Context> CRUMBLING;
@ -35,7 +17,12 @@ public final class Contexts {
var map = new Int2ObjectArrayMap<Context>(); var map = new Int2ObjectArrayMap<Context>();
for (int i = 0; i < ModelBakery.BREAKING_LOCATIONS.size(); i++) { for (int i = 0; i < ModelBakery.BREAKING_LOCATIONS.size(); i++) {
map.put(i, new Crumbling(i)); var crumblingLocation = ModelBakery.BREAKING_LOCATIONS.get(i);
map.put(i, SimpleContext.builder(ContextShaders.CRUMBLING)
.preparation((material, shader, textureSource) -> {
shader.setTexture("crumblingTex", textureSource.byName(crumblingLocation));
})
.build());
} }
CRUMBLING = Int2ObjectMaps.unmodifiable(map); CRUMBLING = Int2ObjectMaps.unmodifiable(map);
@ -43,21 +30,4 @@ public final class Contexts {
private Contexts() { private Contexts() {
} }
private record Crumbling(int stage) implements Context {
@Override
public ContextShader contextShader() {
return ContextShaders.CRUMBLING;
}
@Override
public void prepare(Material material, Shader shader, Textures textures) {
var texture = textures.byName(material.texture());
texture.filter(material.blur(), material.mipmap());
shader.setTexture("_flw_diffuseTex", texture);
var crumblingTexture = textures.byName(ModelBakery.BREAKING_LOCATIONS.get(stage));
shader.setTexture("_flw_crumblingTex", crumblingTexture);
}
}
} }

View file

@ -0,0 +1,61 @@
package com.jozufozu.flywheel.lib.context;
import org.jetbrains.annotations.Nullable;
import com.jozufozu.flywheel.api.context.Context;
import com.jozufozu.flywheel.api.context.ContextShader;
import com.jozufozu.flywheel.api.context.Shader;
import com.jozufozu.flywheel.api.context.TextureSource;
import com.jozufozu.flywheel.api.material.Material;
public class SimpleContext implements Context {
private final ContextShader contextShader;
private final Preparation preparation;
public SimpleContext(ContextShader contextShader, Preparation preparation) {
this.contextShader = contextShader;
this.preparation = preparation;
}
public static Builder builder(ContextShader contextShader) {
return new Builder(contextShader);
}
@Override
public ContextShader contextShader() {
return contextShader;
}
@Override
public void prepare(Material material, Shader shader, TextureSource textureSource) {
preparation.prepare(material, shader, textureSource);
}
@FunctionalInterface
public interface Preparation {
void prepare(Material material, Shader shader, TextureSource textureSource);
}
public static class Builder {
private final ContextShader contextShader;
@Nullable
private Preparation preparation;
public Builder(ContextShader contextShader) {
this.contextShader = contextShader;
}
public Builder preparation(Preparation preparation) {
this.preparation = preparation;
return this;
}
public SimpleContext build() {
if (preparation == null) {
preparation = (material, shader, textureSource) -> {
};
}
return new SimpleContext(contextShader, preparation);
}
}
}

View file

@ -1,4 +1,4 @@
uniform sampler2D _flw_crumblingTex; uniform sampler2D crumblingTex;
in vec2 crumblingTexCoord; in vec2 crumblingTexCoord;
@ -8,7 +8,7 @@ void flw_beginFragment() {
} }
void flw_endFragment() { void flw_endFragment() {
crumblingSampleColor = texture(_flw_crumblingTex, crumblingTexCoord); crumblingSampleColor = texture(crumblingTex, crumblingTexCoord);
// Make the crumbling overlay transparent when the fragment color after the material shader is transparent. // Make the crumbling overlay transparent when the fragment color after the material shader is transparent.
flw_fragColor.rgb = crumblingSampleColor.rgb; flw_fragColor.rgb = crumblingSampleColor.rgb;