File: ssao.glsl

package info (click to toggle)
python-moderngl-window 2.4.2-3
  • links: PTS, VCS
  • area: main
  • in suites: bookworm
  • size: 69,260 kB
  • sloc: python: 11,225; makefile: 20
file content (94 lines) | stat: -rw-r--r-- 2,872 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
#version 330

#if defined VERTEX_SHADER

uniform mat4 m_camera_inverse;
uniform mat4 m_projection_inverse;
uniform vec3 v_camera_pos;

in vec3 in_position;
in vec2 in_texcoord_0;

out vec3 view_ray;
out vec2 texcoord;

void main() {
    gl_Position = vec4(in_position, 1.0);

    // Convert in_position from clip space to view space.
    vec4 pos = m_projection_inverse * vec4(in_position, 1.0);
    // Normalize its z value.
    pos.xy /= -pos.z;
    pos.z = -1.0;
    pos.w = 1.0;
    // Convert to world space.
    pos = m_camera_inverse * pos;
    view_ray = pos.xyz - v_camera_pos;

    texcoord = in_texcoord_0;
}

#elif defined FRAGMENT_SHADER

const int n_samples = 64;

uniform vec3 f_camera_pos;
uniform mat4 mvp;
uniform vec3 samples[n_samples];
uniform float z_offset;

uniform sampler2D g_view_z;
uniform sampler2D g_norm;
uniform sampler2D noise;

in vec3 view_ray;
in vec2 texcoord;

layout(location=0) out float occlusion;

void main() {
    // Ignore background fragments.
    float f_view_z = texture(g_view_z, texcoord).x;
    if (f_view_z == 0.0) {
        discard;
    }

    // Load/compute the position and normal vectors (in world coordinates).
    vec3 f_pos = f_camera_pos + f_view_z * view_ray;
    vec3 f_norm = texture(g_norm, texcoord).xyz;

    // Compute the rotation matrix that takes us from tangent space to world space.
    // Note that the x and y axes in tangent space aren't aligned with the texture coordinates or
    // anything -- they are intentionally randomized to decorrelate our samples in nearby pixels.
    const int noise_size = 32;
    vec2 noise_pos = (1.0 / float(noise_size)) * vec2(
        float(mod(gl_FragCoord.x, noise_size)),
        float(mod(gl_FragCoord.y, noise_size))
    );
    vec3 random_vec = normalize(texture(noise, noise_pos).xyz);
    vec3 tangent_x = normalize(random_vec - f_norm * dot(random_vec, f_norm));
    vec3 tangent_y = cross(f_norm, tangent_x);
    mat3 tan_to_world = mat3(tangent_x, tangent_y, f_norm);

    // Measure occlusion.
    occlusion = 0.0;
    for (int i = 0; i < n_samples; ++i) {
        // Compute the sample position in world coordinates.
        vec3 sample_offset = tan_to_world * samples[i];
        vec4 sample_pos = vec4(f_pos + sample_offset, 1.0);

        // Convert to clip space, then scale the relevant coordinates to the range [0, 1].
        sample_pos = mvp * sample_pos;
        sample_pos.xyz /= sample_pos.w;
        sample_pos.xy = 0.5 * sample_pos.xy + 0.5;

        // Read the actual depth at the sample point.
        float actual_view_z = texture(g_view_z, sample_pos.xy).x;

        // If the actual depth is less than the depth of the sample point, the sample is occluded.
        occlusion += (actual_view_z != 0.0 && actual_view_z + z_offset < f_view_z) ? 1.0 : 0.0;
    }
    occlusion = 1.0 - (1.0 / float(n_samples)) * occlusion;
}

#endif