2021-05-02 01:32:09 +02:00
|
|
|
package com.jozufozu.flywheel.backend;
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-04-08 19:22:11 +02:00
|
|
|
import java.io.BufferedReader;
|
|
|
|
import java.io.FileInputStream;
|
|
|
|
import java.io.IOException;
|
|
|
|
import java.io.InputStream;
|
|
|
|
import java.io.StringReader;
|
2021-03-18 22:36:37 +01:00
|
|
|
import java.nio.Buffer;
|
|
|
|
import java.nio.ByteBuffer;
|
|
|
|
import java.nio.channels.Channels;
|
|
|
|
import java.nio.channels.FileChannel;
|
|
|
|
import java.nio.channels.ReadableByteChannel;
|
2021-04-08 19:22:11 +02:00
|
|
|
import java.util.ArrayList;
|
|
|
|
import java.util.Collection;
|
|
|
|
import java.util.HashMap;
|
|
|
|
import java.util.HashSet;
|
|
|
|
import java.util.Map;
|
|
|
|
import java.util.Set;
|
2021-03-18 22:36:37 +01:00
|
|
|
import java.util.function.Predicate;
|
|
|
|
import java.util.regex.Matcher;
|
|
|
|
import java.util.regex.Pattern;
|
|
|
|
import java.util.stream.Collectors;
|
|
|
|
import java.util.stream.Stream;
|
|
|
|
|
2021-04-08 19:22:11 +02:00
|
|
|
import org.lwjgl.system.MemoryUtil;
|
|
|
|
|
|
|
|
import com.google.common.collect.Lists;
|
2021-05-02 01:32:09 +02:00
|
|
|
import com.jozufozu.flywheel.backend.gl.attrib.IVertexAttrib;
|
|
|
|
import com.jozufozu.flywheel.backend.gl.shader.GlProgram;
|
|
|
|
import com.jozufozu.flywheel.backend.gl.shader.GlShader;
|
|
|
|
import com.jozufozu.flywheel.backend.gl.shader.IMultiProgram;
|
|
|
|
import com.jozufozu.flywheel.backend.gl.shader.ProgramSpec;
|
|
|
|
import com.jozufozu.flywheel.backend.gl.shader.ShaderConstants;
|
|
|
|
import com.jozufozu.flywheel.backend.gl.shader.ShaderType;
|
2021-04-08 19:22:11 +02:00
|
|
|
import com.mojang.blaze3d.systems.RenderSystem;
|
|
|
|
|
|
|
|
import net.minecraft.resources.IResource;
|
|
|
|
import net.minecraft.resources.IResourceManager;
|
|
|
|
import net.minecraft.util.ResourceLocation;
|
|
|
|
import net.minecraftforge.resource.IResourceType;
|
|
|
|
import net.minecraftforge.resource.VanillaResourceType;
|
|
|
|
|
2021-03-18 22:36:37 +01:00
|
|
|
public class ShaderLoader {
|
2021-03-24 23:48:15 +01:00
|
|
|
public static final String SHADER_DIR = "flywheel/shaders/";
|
|
|
|
public static final ArrayList<String> EXTENSIONS = Lists.newArrayList(".vert", ".vsh", ".frag", ".fsh", ".glsl");
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-04-09 23:48:44 +02:00
|
|
|
// #flwinclude <"valid_namespace:valid/path_to_file.glsl">
|
|
|
|
private static final Pattern includePattern = Pattern.compile("#flwinclude <\"([\\w\\d_]+:[\\w\\d_./]+)\">");
|
|
|
|
|
|
|
|
final Map<ResourceLocation, String> shaderSource = new HashMap<>();
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-04-09 23:48:44 +02:00
|
|
|
void onResourceManagerReload(IResourceManager manager, Predicate<IResourceType> predicate) {
|
2021-03-24 23:48:15 +01:00
|
|
|
if (predicate.test(VanillaResourceType.SHADERS)) {
|
|
|
|
OptifineHandler.refresh();
|
|
|
|
Backend.refresh();
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
if (Backend.gl20()) {
|
|
|
|
shaderSource.clear();
|
|
|
|
loadShaderSources(manager);
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-04-09 23:48:44 +02:00
|
|
|
Backend.programs.values().forEach(IMultiProgram::delete);
|
2021-03-24 23:48:15 +01:00
|
|
|
Backend.programs.clear();
|
2021-04-09 23:48:44 +02:00
|
|
|
Backend.registry.values().forEach(this::loadProgramFromSpec);
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
Backend.log.info("Loaded all shader programs.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-04-30 02:19:08 +02:00
|
|
|
private void loadShaderSources(IResourceManager manager) {
|
2021-03-24 23:48:15 +01:00
|
|
|
Collection<ResourceLocation> allShaders = manager.getAllResourceLocations(SHADER_DIR, s -> {
|
|
|
|
for (String ext : EXTENSIONS) {
|
|
|
|
if (s.endsWith(ext)) return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
});
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
for (ResourceLocation location : allShaders) {
|
|
|
|
try {
|
|
|
|
IResource resource = manager.getResource(location);
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
String file = readToString(resource.getInputStream());
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
ResourceLocation name = new ResourceLocation(location.getNamespace(),
|
|
|
|
location.getPath().substring(SHADER_DIR.length()));
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
shaderSource.put(name, file);
|
|
|
|
} catch (IOException e) {
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-04-09 23:48:44 +02:00
|
|
|
private <P extends GlProgram, S extends ProgramSpec<P>> void loadProgramFromSpec(S programSpec) {
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-04-13 01:00:13 +02:00
|
|
|
Backend.programs.put(programSpec, programSpec.finalizer.create(this, programSpec));
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
Backend.log.debug("Loaded program {}", programSpec.name);
|
|
|
|
}
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-04-13 01:00:13 +02:00
|
|
|
public GlProgram.Builder loadProgram(ProgramSpec<?> programSpec) {
|
|
|
|
return loadProgram(programSpec, programSpec.defines);
|
|
|
|
}
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-04-13 01:00:13 +02:00
|
|
|
public GlProgram.Builder loadProgram(ProgramSpec<?> programSpec, ShaderConstants defines) {
|
|
|
|
return loadProgram(programSpec.name, programSpec.vert, programSpec.frag, programSpec.attributes, defines);
|
|
|
|
}
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-04-13 01:00:13 +02:00
|
|
|
public GlProgram.Builder loadProgram(ResourceLocation name, ResourceLocation vert, ResourceLocation frag, Collection<IVertexAttrib> attribs, ShaderConstants defines) {
|
|
|
|
GlShader vsh = null;
|
|
|
|
GlShader fsh = null;
|
|
|
|
try {
|
|
|
|
vsh = loadShader(vert, ShaderType.VERTEX, defines);
|
|
|
|
fsh = loadShader(frag, ShaderType.FRAGMENT, defines);
|
|
|
|
|
|
|
|
return GlProgram.builder(name)
|
|
|
|
.attachShader(vsh)
|
|
|
|
.attachShader(fsh)
|
|
|
|
.addAttributes(attribs)
|
|
|
|
.link();
|
2021-03-24 23:48:15 +01:00
|
|
|
} finally {
|
2021-04-13 01:00:13 +02:00
|
|
|
if (vsh != null) vsh.delete();
|
|
|
|
if (fsh != null) fsh.delete();
|
2021-03-24 23:48:15 +01:00
|
|
|
}
|
|
|
|
}
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-04-09 23:48:44 +02:00
|
|
|
private String processIncludes(ResourceLocation baseName, String source) {
|
2021-03-24 23:48:15 +01:00
|
|
|
HashSet<ResourceLocation> seen = new HashSet<>();
|
|
|
|
seen.add(baseName);
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
return includeRecursive(source, seen).collect(Collectors.joining("\n"));
|
|
|
|
}
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-04-09 23:48:44 +02:00
|
|
|
private Stream<String> includeRecursive(String source, Set<ResourceLocation> seen) {
|
2021-03-24 23:48:15 +01:00
|
|
|
return new BufferedReader(new StringReader(source)).lines().flatMap(line -> {
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
Matcher matcher = includePattern.matcher(line);
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
if (matcher.find()) {
|
|
|
|
String includeName = matcher.group(1);
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
ResourceLocation include = new ResourceLocation(includeName);
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
if (seen.add(include)) {
|
|
|
|
String includeSource = shaderSource.get(include);
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
if (includeSource != null) {
|
|
|
|
return includeRecursive(includeSource, seen);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
return Stream.of(line);
|
|
|
|
});
|
|
|
|
}
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-04-13 01:00:13 +02:00
|
|
|
public GlShader loadShader(ResourceLocation name, ShaderType type, ShaderConstants defines) {
|
2021-03-24 23:48:15 +01:00
|
|
|
String source = shaderSource.get(name);
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
source = processIncludes(name, source);
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
if (defines != null)
|
|
|
|
source = defines.process(source);
|
2021-03-18 22:36:37 +01:00
|
|
|
|
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
return new GlShader(type, name, source);
|
|
|
|
}
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-04-09 23:48:44 +02:00
|
|
|
public String readToString(InputStream is) {
|
2021-03-24 23:48:15 +01:00
|
|
|
RenderSystem.assertThread(RenderSystem::isOnRenderThread);
|
|
|
|
ByteBuffer bytebuffer = null;
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
try {
|
|
|
|
bytebuffer = readToBuffer(is);
|
|
|
|
int i = bytebuffer.position();
|
2021-04-30 02:19:08 +02:00
|
|
|
((Buffer) bytebuffer).rewind();
|
2021-03-24 23:48:15 +01:00
|
|
|
return MemoryUtil.memASCII(bytebuffer, i);
|
|
|
|
} catch (IOException e) {
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
} finally {
|
|
|
|
if (bytebuffer != null) {
|
|
|
|
MemoryUtil.memFree(bytebuffer);
|
|
|
|
}
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
}
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
return null;
|
|
|
|
}
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-04-09 23:48:44 +02:00
|
|
|
public ByteBuffer readToBuffer(InputStream is) throws IOException {
|
2021-03-24 23:48:15 +01:00
|
|
|
ByteBuffer bytebuffer;
|
|
|
|
if (is instanceof FileInputStream) {
|
2021-04-30 02:19:08 +02:00
|
|
|
FileInputStream fileinputstream = (FileInputStream) is;
|
2021-03-24 23:48:15 +01:00
|
|
|
FileChannel filechannel = fileinputstream.getChannel();
|
2021-04-30 02:19:08 +02:00
|
|
|
bytebuffer = MemoryUtil.memAlloc((int) filechannel.size() + 1);
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-04-30 02:19:08 +02:00
|
|
|
while (filechannel.read(bytebuffer) != -1) {
|
|
|
|
}
|
2021-03-24 23:48:15 +01:00
|
|
|
} else {
|
|
|
|
bytebuffer = MemoryUtil.memAlloc(8192);
|
|
|
|
ReadableByteChannel readablebytechannel = Channels.newChannel(is);
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
while (readablebytechannel.read(bytebuffer) != -1) {
|
|
|
|
if (bytebuffer.remaining() == 0) {
|
|
|
|
bytebuffer = MemoryUtil.memRealloc(bytebuffer, bytebuffer.capacity() * 2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-03-18 22:36:37 +01:00
|
|
|
|
2021-03-24 23:48:15 +01:00
|
|
|
return bytebuffer;
|
|
|
|
}
|
2021-03-18 22:36:37 +01:00
|
|
|
}
|