2019-07-21 03:04:35 +03:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Implicit Subdivision Shader for Terrain Rendering
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "terrain_common.sh"
|
|
|
|
#include "fcull.sh"
|
|
|
|
|
|
|
|
BUFFER_RO(u_SubdBufferIn, uint, 8);
|
|
|
|
BUFFER_RW(u_CulledSubdBuffer, uint, 2);
|
|
|
|
BUFFER_RO(u_VertexBuffer, vec4, 6);
|
|
|
|
BUFFER_RO(u_IndexBuffer, uint, 7);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Compute LoD Shader
|
|
|
|
*
|
|
|
|
* This compute shader is responsible for updating the subdivision
|
|
|
|
* buffer and visible buffer that will be sent to the rasterizer.
|
|
|
|
*/
|
|
|
|
|
|
|
|
NUM_THREADS(COMPUTE_THREAD_COUNT, 1u, 1u)
|
|
|
|
void main()
|
|
|
|
{
|
2019-07-22 08:02:07 +03:00
|
|
|
// get threadID (each key is associated to a thread)
|
|
|
|
uint threadID = gl_GlobalInvocationID.x;
|
2019-07-21 03:04:35 +03:00
|
|
|
|
|
|
|
if (threadID >= u_AtomicCounterBuffer[2])
|
2019-07-22 08:02:07 +03:00
|
|
|
{
|
2019-07-21 03:04:35 +03:00
|
|
|
return;
|
2019-07-22 08:02:07 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// get coarse triangle associated to the key
|
|
|
|
uint primID = u_SubdBufferIn[threadID*2];
|
2019-07-21 03:04:35 +03:00
|
|
|
|
2019-07-22 08:02:07 +03:00
|
|
|
vec4 v_in[3];
|
2019-07-21 03:04:35 +03:00
|
|
|
v_in[0] = u_VertexBuffer[u_IndexBuffer[primID * 3 ]];
|
|
|
|
v_in[1] = u_VertexBuffer[u_IndexBuffer[primID * 3 + 1]];
|
|
|
|
v_in[2] = u_VertexBuffer[u_IndexBuffer[primID * 3 + 2]];
|
|
|
|
|
2019-07-22 08:02:07 +03:00
|
|
|
// compute distance-based LOD
|
|
|
|
uint key = u_SubdBufferIn[threadID*2+1];
|
|
|
|
|
|
|
|
vec4 v[3];
|
|
|
|
vec4 vp[3];
|
|
|
|
|
2019-07-21 03:04:35 +03:00
|
|
|
subd(key, v_in, v, vp);
|
2019-07-22 08:02:07 +03:00
|
|
|
|
2019-07-21 03:04:35 +03:00
|
|
|
uint targetLod; uint parentLod;
|
2019-07-22 08:02:07 +03:00
|
|
|
|
|
|
|
if (u_freeze == 0)
|
|
|
|
{
|
2019-07-21 03:04:35 +03:00
|
|
|
targetLod = uint(computeLod(v));
|
|
|
|
parentLod = uint(computeLod(vp));
|
|
|
|
}
|
2019-07-22 08:02:07 +03:00
|
|
|
else
|
|
|
|
{
|
2020-08-16 21:43:28 +03:00
|
|
|
targetLod = parentLod = findMSB_(key);
|
2019-07-21 03:04:35 +03:00
|
|
|
}
|
2019-07-22 08:02:07 +03:00
|
|
|
|
|
|
|
updateSubdBuffer(primID, key, targetLod, parentLod);
|
|
|
|
|
|
|
|
// Cull invisible nodes
|
|
|
|
mat4 mvp = u_modelViewProj;
|
|
|
|
vec4 bmin = min(min(v[0], v[1]), v[2]);
|
|
|
|
vec4 bmax = max(max(v[0], v[1]), v[2]);
|
|
|
|
|
|
|
|
// account for displacement in bound computations
|
|
|
|
bmin.z = 0;
|
|
|
|
bmax.z = u_DmapFactor;
|
|
|
|
|
|
|
|
// update CulledSubdBuffer
|
|
|
|
if (u_cull == 0
|
|
|
|
|| frustumCullingTest(mvp, bmin.xyz, bmax.xyz) )
|
|
|
|
{
|
|
|
|
// write key
|
2019-07-21 03:04:35 +03:00
|
|
|
uint idx = 0;
|
|
|
|
atomicFetchAndAdd(u_AtomicCounterBuffer[1], 2, idx);
|
2019-07-22 08:02:07 +03:00
|
|
|
u_CulledSubdBuffer[idx] = primID;
|
2019-07-21 03:04:35 +03:00
|
|
|
u_CulledSubdBuffer[idx+1] = key;
|
2019-07-22 08:02:07 +03:00
|
|
|
}
|
2019-07-21 03:04:35 +03:00
|
|
|
}
|