|
1 | 1 | package net.caffeinemc.mods.sodium.client.render.viewport.frustum; |
2 | 2 |
|
| 3 | +import net.caffeinemc.mods.sodium.client.render.chunk.occlusion.OcclusionCuller; |
3 | 4 | import org.joml.FrustumIntersection; |
| 5 | +import org.joml.Vector4f; |
| 6 | + |
| 7 | +import java.lang.invoke.MethodHandle; |
| 8 | +import java.lang.invoke.MethodHandles; |
| 9 | +import java.lang.reflect.Field; |
4 | 10 |
|
5 | 11 | public final class SimpleFrustum implements Frustum { |
| 12 | + private float nxX, nxY, nxZ, negNxW; |
| 13 | + private float pxX, pxY, pxZ, negPxW; |
| 14 | + private float nyX, nyY, nyZ, negNyW; |
| 15 | + private float pyX, pyY, pyZ, negPyW; |
| 16 | + private float nzX, nzY, nzZ, negNzW; |
| 17 | + private float pzX, pzY, pzZ, negPzW; |
| 18 | + |
6 | 19 | private final FrustumIntersection frustum; |
7 | 20 |
|
| 21 | + private static final MethodHandle PLANES_GETTER; |
| 22 | + static { |
| 23 | + try { |
| 24 | + Field field = FrustumIntersection.class.getDeclaredField("planes"); |
| 25 | + field.setAccessible(true); |
| 26 | + PLANES_GETTER = MethodHandles.lookup().unreflectGetter(field); |
| 27 | + } catch (NoSuchFieldException | IllegalAccessException e) { |
| 28 | + throw new RuntimeException("Failed to find planes field in JOML", e); |
| 29 | + } |
| 30 | + } |
| 31 | + |
8 | 32 | public SimpleFrustum(FrustumIntersection frustumIntersection) { |
9 | 33 | this.frustum = frustumIntersection; |
| 34 | + Vector4f[] planes; |
| 35 | + try { |
| 36 | + planes = (Vector4f[]) PLANES_GETTER.invokeExact(frustumIntersection); |
| 37 | + } catch (Throwable e) { |
| 38 | + throw new RuntimeException("Failed to access planes field in FrustumIntersection", e); |
| 39 | + } |
| 40 | + |
| 41 | + nxX = planes[0].x; |
| 42 | + nxY = planes[0].y; |
| 43 | + nxZ = planes[0].z; |
| 44 | + pxX = planes[1].x; |
| 45 | + pxY = planes[1].y; |
| 46 | + pxZ = planes[1].z; |
| 47 | + nyX = planes[2].x; |
| 48 | + nyY = planes[2].y; |
| 49 | + nyZ = planes[2].z; |
| 50 | + pyX = planes[3].x; |
| 51 | + pyY = planes[3].y; |
| 52 | + pyZ = planes[3].z; |
| 53 | + nzX = planes[4].x; |
| 54 | + nzY = planes[4].y; |
| 55 | + nzZ = planes[4].z; |
| 56 | + pzX = planes[5].x; |
| 57 | + pzY = planes[5].y; |
| 58 | + pzZ = planes[5].z; |
| 59 | + |
| 60 | + final float size = OcclusionCuller.CHUNK_SECTION_SIZE; |
| 61 | + negNxW = -(planes[0].w + nxX * (nxX < 0 ? -size : size) + |
| 62 | + nxY * (nxY < 0 ? -size : size) + |
| 63 | + nxZ * (nxZ < 0 ? -size : size)); |
| 64 | + negPxW = -(planes[1].w + pxX * (pxX < 0 ? -size : size) + |
| 65 | + pxY * (pxY < 0 ? -size : size) + |
| 66 | + pxZ * (pxZ < 0 ? -size : size)); |
| 67 | + negNyW = -(planes[2].w + nyX * (nyX < 0 ? -size : size) + |
| 68 | + nyY * (nyY < 0 ? -size : size) + |
| 69 | + nyZ * (nyZ < 0 ? -size : size)); |
| 70 | + negPyW = -(planes[3].w + pyX * (pyX < 0 ? -size : size) + |
| 71 | + pyY * (pyY < 0 ? -size : size) + |
| 72 | + pyZ * (pyZ < 0 ? -size : size)); |
| 73 | + negNzW = -(planes[4].w + nzX * (nzX < 0 ? -size : size) + |
| 74 | + nzY * (nzY < 0 ? -size : size) + |
| 75 | + nzZ * (nzZ < 0 ? -size : size)); |
| 76 | + negPzW = -(planes[5].w + pzX * (pzX < 0 ? -size : size) + |
| 77 | + pzY * (pzY < 0 ? -size : size) + |
| 78 | + pzZ * (pzZ < 0 ? -size : size)); |
| 79 | + } |
| 80 | + |
| 81 | + public boolean testCubeQuick(float x, float y, float z) { |
| 82 | + // Skip far plane checks because it has been ensured by searchDistance and isWithinRenderDistance check in OcclusionCuller |
| 83 | + return nxX * x + nxY * y + nxZ * z >= negNxW && |
| 84 | + pxX * x + pxY * y + pxZ * z >= negPxW && |
| 85 | + nyX * x + nyY * y + nyZ * z >= negNyW && |
| 86 | + pyX * x + pyY * y + pyZ * z >= negPyW && |
| 87 | + nzX * x + nzY * y + nzZ * z >= negNzW; |
| 88 | + } |
| 89 | + |
| 90 | + public boolean testCubeWithExtend(float floatOriginX, float floatOriginY, float floatOriginZ, float extend) { |
| 91 | + float minX = floatOriginX - extend; |
| 92 | + float maxX = floatOriginX + extend; |
| 93 | + float minY = floatOriginY - extend; |
| 94 | + float maxY = floatOriginY + extend; |
| 95 | + float minZ = floatOriginZ - extend; |
| 96 | + float maxZ = floatOriginZ + extend; |
| 97 | + |
| 98 | + return nxX * (nxX < 0 ? minX : maxX) + nxY * (nxY < 0 ? minY : maxY) + nxZ * (nxZ < 0 ? minZ : maxZ) >= negNxW && |
| 99 | + pxX * (pxX < 0 ? minX : maxX) + pxY * (pxY < 0 ? minY : maxY) + pxZ * (pxZ < 0 ? minZ : maxZ) >= negPxW && |
| 100 | + nyX * (nyX < 0 ? minX : maxX) + nyY * (nyY < 0 ? minY : maxY) + nyZ * (nyZ < 0 ? minZ : maxZ) >= negNyW && |
| 101 | + pyX * (pyX < 0 ? minX : maxX) + pyY * (pyY < 0 ? minY : maxY) + pyZ * (pyZ < 0 ? minZ : maxZ) >= negPyW && |
| 102 | + nzX * (nzX < 0 ? minX : maxX) + nzY * (nzY < 0 ? minY : maxY) + nzZ * (nzZ < 0 ? minZ : maxZ) >= negNzW && |
| 103 | + pzX * (pzX < 0 ? minX : maxX) + pzY * (pzY < 0 ? minY : maxY) + pzZ * (pzZ < 0 ? minZ : maxZ) >= negPzW; |
10 | 104 | } |
11 | 105 |
|
12 | 106 | @Override |
|
0 commit comments