package com.jozufozu.flywheel.backend; import java.util.ArrayList; import java.util.Collection; import java.util.HashMap; import java.util.List; import java.util.Map; import javax.annotation.Nullable; import org.apache.logging.log4j.LogManager; import org.apache.logging.log4j.Logger; import org.lwjgl.opengl.GL; import org.lwjgl.opengl.GLCapabilities; import com.jozufozu.flywheel.backend.gl.versioned.GlCompat; import com.jozufozu.flywheel.backend.instancing.InstanceData; import com.jozufozu.flywheel.backend.material.MaterialSpec; import com.jozufozu.flywheel.config.FlwConfig; import com.jozufozu.flywheel.core.shader.spec.ProgramSpec; import net.minecraft.client.Minecraft; import net.minecraft.util.ResourceLocation; import net.minecraft.util.math.vector.Matrix4f; import net.minecraft.world.IWorld; import net.minecraft.world.World; public class Backend { public static final Logger log = LogManager.getLogger(Backend.class); protected static final Backend INSTANCE = new Backend(); public static Backend getInstance() { return INSTANCE; } public final Loader loader; public GLCapabilities capabilities; public GlCompat compat; private Matrix4f projectionMatrix = new Matrix4f(); private boolean instancedArrays; private boolean enabled; public boolean chunkCachingEnabled; private final List> contexts = new ArrayList<>(); private final Map> materialRegistry = new HashMap<>(); private final Map programSpecRegistry = new HashMap<>(); protected Backend() { loader = new Loader(this); OptifineHandler.init(); } /** * Get a string describing the Flywheel backend. When there are eventually multiple backends * (Meshlet, MDI, GL31 Draw Instanced are planned), this will name which one is in use. */ public String getBackendDescriptor() { if (canUseInstancing()) { return "GL33 Instanced Arrays"; } if (canUseVBOs()) { return "VBOs"; } return "Disabled"; } /** * Register a shader program. */ public ProgramSpec register(ProgramSpec spec) { ResourceLocation name = spec.name; if (programSpecRegistry.containsKey(name)) { throw new IllegalStateException("Program spec '" + name + "' already registered."); } programSpecRegistry.put(name, spec); return spec; } /** * Register a shader context. */ public > C register(C spec) { contexts.add(spec); return spec; } /** * Register an instancing material. */ public MaterialSpec register(MaterialSpec spec) { ResourceLocation name = spec.name; if (materialRegistry.containsKey(name)) { throw new IllegalStateException("Material spec '" + name + "' already registered."); } materialRegistry.put(name, spec); log.debug("registered material '" + name + "' with vertex size " + spec.getModelFormat() .getStride() + " and instance size " + spec.getInstanceType().format().getStride()); return spec; } public ProgramSpec getSpec(ResourceLocation name) { return programSpecRegistry.get(name); } public boolean available() { return canUseVBOs(); } public boolean canUseInstancing() { return enabled && instancedArrays; } public boolean canUseVBOs() { return enabled && gl20(); } public boolean gl33() { return capabilities.OpenGL33; } public boolean gl20() { return capabilities.OpenGL20; } public void refresh() { OptifineHandler.refresh(); capabilities = GL.createCapabilities(); compat = new GlCompat(capabilities); instancedArrays = compat.vertexArrayObjectsSupported() && compat.drawInstancedSupported() && compat.instancedArraysSupported(); enabled = FlwConfig.get() .enabled() && !OptifineHandler.usingShaders(); chunkCachingEnabled = FlwConfig.get() .chunkCaching(); } public boolean canUseInstancing(@Nullable World world) { return canUseInstancing() && isFlywheelWorld(world); } public Collection> allMaterials() { return materialRegistry.values(); } public Collection allPrograms() { return programSpecRegistry.values(); } public Collection> allContexts() { return contexts; } public Matrix4f getProjectionMatrix() { return projectionMatrix; } public void setProjectionMatrix(Matrix4f projectionMatrix) { this.projectionMatrix = projectionMatrix; } /** * Used to avoid calling Flywheel functions on (fake) worlds that don't specifically support it. */ public static boolean isFlywheelWorld(@Nullable IWorld world) { if (world == null) return false; if (world instanceof IFlywheelWorld && ((IFlywheelWorld) world).supportsFlywheel()) return true; return world == Minecraft.getInstance().level; } public static boolean isGameActive() { return !(Minecraft.getInstance().level == null || Minecraft.getInstance().player == null); } public static void reloadWorldRenderers() { RenderWork.enqueue(Minecraft.getInstance().levelRenderer::allChanged); } /** * INTERNAL USE ONLY */ public void _clearContexts() { SpecMetaRegistry.clear(); programSpecRegistry.clear(); contexts.forEach(IShaderContext::delete); contexts.clear(); materialRegistry.clear(); } public static void init() { } }