2021-09-09 00:48:49 +02:00
|
|
|
package com.jozufozu.flywheel.light;
|
|
|
|
|
2021-11-18 23:59:39 +01:00
|
|
|
import static org.lwjgl.opengl.GL11.GL_LINEAR;
|
|
|
|
import static org.lwjgl.opengl.GL11.GL_TEXTURE_MAG_FILTER;
|
|
|
|
import static org.lwjgl.opengl.GL11.GL_TEXTURE_MIN_FILTER;
|
|
|
|
import static org.lwjgl.opengl.GL11.GL_TEXTURE_WRAP_S;
|
|
|
|
import static org.lwjgl.opengl.GL11.GL_TEXTURE_WRAP_T;
|
|
|
|
import static org.lwjgl.opengl.GL11.GL_UNPACK_ALIGNMENT;
|
|
|
|
import static org.lwjgl.opengl.GL11.GL_UNPACK_ROW_LENGTH;
|
|
|
|
import static org.lwjgl.opengl.GL11.GL_UNPACK_SKIP_PIXELS;
|
|
|
|
import static org.lwjgl.opengl.GL11.GL_UNPACK_SKIP_ROWS;
|
|
|
|
import static org.lwjgl.opengl.GL11.GL_UNSIGNED_BYTE;
|
|
|
|
import static org.lwjgl.opengl.GL11.glPixelStorei;
|
|
|
|
import static org.lwjgl.opengl.GL11.glTexParameteri;
|
|
|
|
import static org.lwjgl.opengl.GL12.GL_TEXTURE_3D;
|
|
|
|
import static org.lwjgl.opengl.GL12.GL_TEXTURE_WRAP_R;
|
|
|
|
import static org.lwjgl.opengl.GL12.GL_UNPACK_IMAGE_HEIGHT;
|
|
|
|
import static org.lwjgl.opengl.GL12.GL_UNPACK_SKIP_IMAGES;
|
|
|
|
import static org.lwjgl.opengl.GL12.glTexImage3D;
|
|
|
|
import static org.lwjgl.opengl.GL12.glTexSubImage3D;
|
|
|
|
import static org.lwjgl.opengl.GL13.GL_TEXTURE0;
|
|
|
|
import static org.lwjgl.opengl.GL13.GL_TEXTURE4;
|
|
|
|
import static org.lwjgl.opengl.GL13.glActiveTexture;
|
|
|
|
import static org.lwjgl.opengl.GL14.GL_MIRRORED_REPEAT;
|
2021-09-09 00:48:49 +02:00
|
|
|
|
2021-12-08 07:47:05 +01:00
|
|
|
import org.lwjgl.opengl.GL30;
|
|
|
|
|
2021-09-09 00:48:49 +02:00
|
|
|
import com.jozufozu.flywheel.backend.gl.GlTexture;
|
|
|
|
import com.jozufozu.flywheel.backend.gl.GlTextureUnit;
|
|
|
|
|
2021-09-15 08:45:29 +02:00
|
|
|
import net.minecraft.world.level.LightLayer;
|
2021-09-09 00:48:49 +02:00
|
|
|
|
|
|
|
public class GPULightVolume extends LightVolume {
|
|
|
|
|
|
|
|
protected final GridAlignedBB sampleVolume = new GridAlignedBB();
|
|
|
|
private final GlTexture glTexture;
|
|
|
|
|
2021-11-09 23:59:31 +01:00
|
|
|
private final GlTextureUnit textureUnit = GlTextureUnit.T4;
|
2021-09-09 00:48:49 +02:00
|
|
|
protected boolean bufferDirty;
|
|
|
|
|
|
|
|
public GPULightVolume(ImmutableBox sampleVolume) {
|
|
|
|
super(sampleVolume);
|
|
|
|
this.sampleVolume.assign(sampleVolume);
|
|
|
|
|
|
|
|
glTexture = new GlTexture(GL_TEXTURE_3D);
|
|
|
|
|
2021-12-10 09:07:52 +01:00
|
|
|
GlTextureUnit oldState = GlTextureUnit.getActive();
|
|
|
|
|
2021-09-09 00:48:49 +02:00
|
|
|
// allocate space for the texture
|
2021-12-10 09:07:52 +01:00
|
|
|
textureUnit.makeActive();
|
2021-09-09 00:48:49 +02:00
|
|
|
glTexture.bind();
|
|
|
|
|
|
|
|
int sizeX = box.sizeX();
|
|
|
|
int sizeY = box.sizeY();
|
|
|
|
int sizeZ = box.sizeZ();
|
2021-12-08 07:47:05 +01:00
|
|
|
glTexImage3D(GL_TEXTURE_3D, 0, GL30.GL_RG8, sizeX, sizeY, sizeZ, 0, GL30.GL_RG, GL_UNSIGNED_BYTE, 0);
|
2021-09-09 00:48:49 +02:00
|
|
|
|
|
|
|
glTexture.unbind();
|
2021-12-10 09:07:52 +01:00
|
|
|
oldState.makeActive();
|
2021-09-09 00:48:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected void setBox(ImmutableBox box) {
|
|
|
|
this.box.assign(box);
|
|
|
|
this.box.nextPowerOf2Centered();
|
|
|
|
// called during super ctor
|
|
|
|
if (sampleVolume != null) this.sampleVolume.assign(box);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void bind() {
|
|
|
|
// just in case something goes wrong, or we accidentally call this before this volume is properly disposed of.
|
2021-12-10 09:07:52 +01:00
|
|
|
if (lightData == null || lightData.capacity() == 0) return;
|
2021-09-09 00:48:49 +02:00
|
|
|
|
2021-11-09 23:59:31 +01:00
|
|
|
textureUnit.makeActive();
|
2021-09-09 00:48:49 +02:00
|
|
|
glTexture.bind();
|
2021-12-10 09:07:52 +01:00
|
|
|
glTexture.setParameteri(GL_TEXTURE_MIN_FILTER, GL_LINEAR);
|
|
|
|
glTexture.setParameteri(GL_TEXTURE_MAG_FILTER, GL_LINEAR);
|
|
|
|
glTexture.setParameteri(GL_TEXTURE_WRAP_S, GL_MIRRORED_REPEAT);
|
|
|
|
glTexture.setParameteri(GL_TEXTURE_WRAP_R, GL_MIRRORED_REPEAT);
|
|
|
|
glTexture.setParameteri(GL_TEXTURE_WRAP_T, GL_MIRRORED_REPEAT);
|
2021-09-09 00:48:49 +02:00
|
|
|
|
|
|
|
uploadTexture();
|
|
|
|
}
|
|
|
|
|
|
|
|
private void uploadTexture() {
|
|
|
|
if (bufferDirty) {
|
|
|
|
glPixelStorei(GL_UNPACK_ROW_LENGTH, 0);
|
|
|
|
glPixelStorei(GL_UNPACK_SKIP_PIXELS, 0);
|
|
|
|
glPixelStorei(GL_UNPACK_SKIP_ROWS, 0);
|
|
|
|
glPixelStorei(GL_UNPACK_SKIP_IMAGES, 0);
|
|
|
|
glPixelStorei(GL_UNPACK_IMAGE_HEIGHT, 0);
|
2021-12-10 09:07:52 +01:00
|
|
|
glPixelStorei(GL_UNPACK_ALIGNMENT, 2); // we use 2 bytes per texel
|
2021-09-09 00:48:49 +02:00
|
|
|
int sizeX = box.sizeX();
|
|
|
|
int sizeY = box.sizeY();
|
|
|
|
int sizeZ = box.sizeZ();
|
|
|
|
|
2021-12-08 07:47:05 +01:00
|
|
|
glTexSubImage3D(GL_TEXTURE_3D, 0, 0, 0, 0, sizeX, sizeY, sizeZ, GL30.GL_RG, GL_UNSIGNED_BYTE, lightData);
|
2021-09-09 00:48:49 +02:00
|
|
|
|
|
|
|
glPixelStorei(GL_UNPACK_ALIGNMENT, 4); // 4 is the default
|
|
|
|
bufferDirty = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void unbind() {
|
|
|
|
glTexture.unbind();
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void delete() {
|
|
|
|
super.delete();
|
|
|
|
glTexture.delete();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void move(LightProvider world, ImmutableBox newSampleVolume) {
|
|
|
|
if (lightData == null) return;
|
|
|
|
|
|
|
|
if (box.contains(newSampleVolume)) {
|
|
|
|
if (newSampleVolume.intersects(sampleVolume)) {
|
|
|
|
GridAlignedBB newArea = newSampleVolume.intersect(sampleVolume);
|
|
|
|
sampleVolume.assign(newSampleVolume);
|
|
|
|
|
|
|
|
copyLight(world, newArea);
|
|
|
|
} else {
|
|
|
|
sampleVolume.assign(newSampleVolume);
|
|
|
|
initialize(world);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
super.move(world, newSampleVolume);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-15 08:45:29 +02:00
|
|
|
public void onLightUpdate(LightProvider world, LightLayer type, ImmutableBox changedVolume) {
|
2021-09-09 00:48:49 +02:00
|
|
|
super.onLightUpdate(world, type, changedVolume);
|
|
|
|
bufferDirty = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void onLightPacket(LightProvider world, int chunkX, int chunkZ) {
|
|
|
|
super.onLightPacket(world, chunkX, chunkZ);
|
|
|
|
bufferDirty = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Completely (re)populate this volume with block and sky lighting data.
|
|
|
|
* This is expensive and should be avoided.
|
|
|
|
*/
|
|
|
|
public void initialize(LightProvider world) {
|
|
|
|
super.initialize(world);
|
|
|
|
bufferDirty = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public ImmutableBox getVolume() {
|
|
|
|
return sampleVolume;
|
|
|
|
}
|
|
|
|
}
|