2016-12-07 21:49:52 +01:00
|
|
|
[vertex]
|
|
|
|
|
2018-08-24 13:42:18 +02:00
|
|
|
layout(location = 0) in highp vec4 vertex_attrib;
|
2016-12-07 21:49:52 +01:00
|
|
|
|
|
|
|
void main() {
|
|
|
|
|
|
|
|
gl_Position = vertex_attrib;
|
|
|
|
}
|
|
|
|
|
|
|
|
[fragment]
|
|
|
|
|
|
|
|
uniform highp sampler2D source_exposure; //texunit:0
|
|
|
|
|
|
|
|
#ifdef EXPOSURE_BEGIN
|
|
|
|
|
|
|
|
uniform highp ivec2 source_render_size;
|
|
|
|
uniform highp ivec2 target_size;
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef EXPOSURE_END
|
|
|
|
|
|
|
|
uniform highp sampler2D prev_exposure; //texunit:1
|
|
|
|
uniform highp float exposure_adjust;
|
|
|
|
uniform highp float min_luminance;
|
|
|
|
uniform highp float max_luminance;
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
layout(location = 0) out highp float exposure;
|
|
|
|
|
|
|
|
void main() {
|
|
|
|
|
|
|
|
#ifdef EXPOSURE_BEGIN
|
|
|
|
|
2018-08-24 13:42:18 +02:00
|
|
|
ivec2 src_pos = ivec2(gl_FragCoord.xy) * source_render_size / target_size;
|
2016-12-07 21:49:52 +01:00
|
|
|
|
|
|
|
#if 1
|
|
|
|
//more precise and expensive, but less jittery
|
2018-08-24 13:42:18 +02:00
|
|
|
ivec2 next_pos = ivec2(gl_FragCoord.xy + ivec2(1)) * source_render_size / target_size;
|
|
|
|
next_pos = max(next_pos, src_pos + ivec2(1)); //so it at least reads one pixel
|
|
|
|
highp vec3 source_color = vec3(0.0);
|
|
|
|
for (int i = src_pos.x; i < next_pos.x; i++) {
|
|
|
|
for (int j = src_pos.y; j < next_pos.y; j++) {
|
|
|
|
source_color += texelFetch(source_exposure, ivec2(i, j), 0).rgb;
|
2016-12-07 21:49:52 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-24 13:42:18 +02:00
|
|
|
source_color /= float((next_pos.x - src_pos.x) * (next_pos.y - src_pos.y));
|
2016-12-07 21:49:52 +01:00
|
|
|
#else
|
2018-08-24 13:42:18 +02:00
|
|
|
highp vec3 source_color = texelFetch(source_exposure, src_pos, 0).rgb;
|
2016-12-07 21:49:52 +01:00
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2018-08-24 13:42:18 +02:00
|
|
|
exposure = max(source_color.r, max(source_color.g, source_color.b));
|
2016-12-07 21:49:52 +01:00
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
ivec2 coord = ivec2(gl_FragCoord.xy);
|
2018-08-24 13:42:18 +02:00
|
|
|
exposure = texelFetch(source_exposure, coord * 3 + ivec2(0, 0), 0).r;
|
|
|
|
exposure += texelFetch(source_exposure, coord * 3 + ivec2(1, 0), 0).r;
|
|
|
|
exposure += texelFetch(source_exposure, coord * 3 + ivec2(2, 0), 0).r;
|
|
|
|
exposure += texelFetch(source_exposure, coord * 3 + ivec2(0, 1), 0).r;
|
|
|
|
exposure += texelFetch(source_exposure, coord * 3 + ivec2(1, 1), 0).r;
|
|
|
|
exposure += texelFetch(source_exposure, coord * 3 + ivec2(2, 1), 0).r;
|
|
|
|
exposure += texelFetch(source_exposure, coord * 3 + ivec2(0, 2), 0).r;
|
|
|
|
exposure += texelFetch(source_exposure, coord * 3 + ivec2(1, 2), 0).r;
|
|
|
|
exposure += texelFetch(source_exposure, coord * 3 + ivec2(2, 2), 0).r;
|
|
|
|
exposure *= (1.0 / 9.0);
|
2016-12-07 21:49:52 +01:00
|
|
|
|
|
|
|
#ifdef EXPOSURE_END
|
|
|
|
|
|
|
|
#ifdef EXPOSURE_FORCE_SET
|
|
|
|
//will stay as is
|
|
|
|
#else
|
2018-08-24 13:42:18 +02:00
|
|
|
highp float prev_lum = texelFetch(prev_exposure, ivec2(0, 0), 0).r; //1 pixel previous exposure
|
|
|
|
exposure = clamp(prev_lum + (exposure - prev_lum) * exposure_adjust, min_luminance, max_luminance);
|
2016-12-07 21:49:52 +01:00
|
|
|
|
|
|
|
#endif //EXPOSURE_FORCE_SET
|
|
|
|
|
|
|
|
#endif //EXPOSURE_END
|
|
|
|
|
|
|
|
#endif //EXPOSURE_BEGIN
|
|
|
|
}
|