#version 450 layout(push_constant) uniform Push { vec4 SourceSize; vec4 OriginalSize; vec4 OutputSize; uint FrameCount; } registers; #include "params.inc" ///////////////////////////// GPL LICENSE NOTICE ///////////////////////////// // crt-royale: A full-featured CRT shader, with cheese. // Copyright (C) 2014 TroggleMonkey // // This program is free software; you can redistribute it and/or modify it // under the terms of the GNU General Public License as published by the Free // Software Foundation; either version 2 of the License, or any later version. // // This program is distributed in the hope that it will be useful, but WITHOUT // ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or // FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for // more details. // // You should have received a copy of the GNU General Public License along with // this program; if not, write to the Free Software Foundation, Inc., 59 Temple // Place, Suite 330, Boston, MA 02111-1307 USA ///////////////////////////// SETTINGS MANAGEMENT //////////////////////////// #include "../user-settings.h" #include "derived-settings-and-constants.h" #include "bind-shader-params.h" ////////////////////////////////// INCLUDES ////////////////////////////////// #include "phosphor-mask-resizing.h" #pragma stage vertex layout(location = 0) in vec4 Position; layout(location = 1) in vec2 TexCoord; layout(location = 0) out vec2 src_tex_uv_wrap; layout(location = 1) out vec2 tile_uv_wrap; layout(location = 2) out vec2 resize_magnification_scale; layout(location = 3) out vec2 src_dxdy; layout(location = 4) out vec2 tile_size_uv; layout(location = 5) out vec2 input_tiles_per_texture; layout(location = 6) out vec2 tex_uv; void main() { gl_Position = params.MVP * Position; tex_uv = TexCoord; // First estimate the viewport size (the user will get the wrong number of // triads if it's wrong and mask_specify_num_triads is 1.0/true). const vec2 estimated_viewport_size = registers.OutputSize.xy / mask_resize_viewport_scale; // Find the final size of our resized phosphor mask tiles. We probably // estimated the viewport size and MASK_RESIZE output size differently last // pass, so do not swear they were the same. ;) const vec2 mask_resize_tile_size = get_resized_mask_tile_size( estimated_viewport_size, registers.OutputSize.xy, false); // We'll render resized tiles until filling the output FBO or meeting a // limit, so compute [wrapped] tile uv coords based on the output uv coords // and the number of tiles that will fit in the FBO. const vec2 output_tiles_this_pass = registers.OutputSize.xy / mask_resize_tile_size; const vec2 output_video_uv = tex_uv; tile_uv_wrap = output_video_uv * output_tiles_this_pass; // Get the texel size of an input tile and related values: const vec2 input_tile_size = vec2(min( mask_resize_src_lut_size.x, registers.SourceSize.x), mask_resize_tile_size.y); tile_size_uv = input_tile_size * registers.SourceSize.zw; input_tiles_per_texture = registers.SourceSize.xy / input_tile_size; // Derive [wrapped] texture uv coords from [wrapped] tile uv coords and // the tile size in uv coords, and save frac() for the fragment shader. src_tex_uv_wrap = tile_uv_wrap * tile_size_uv; resize_magnification_scale = mask_resize_tile_size / input_tile_size; src_dxdy = vec2(registers.SourceSize.z, 0.0); } #pragma stage fragment layout(location = 0) in vec2 src_tex_uv_wrap; layout(location = 1) in vec2 tile_uv_wrap; layout(location = 2) in vec2 resize_magnification_scale; layout(location = 3) in vec2 src_dxdy; layout(location = 4) in vec2 tile_size_uv; layout(location = 5) in vec2 input_tiles_per_texture; layout(location = 6) in vec2 tex_uv; layout(location = 0) out vec4 FragColor; layout(set = 0, binding = 2) uniform sampler2D Source; void main() { // The input contains one mask tile horizontally and a number vertically. // Resize the tile horizontally to its final screen size and repeat it // until drawing at least mask_resize_num_tiles, leaving it unchanged // vertically. Lanczos-resizing the phosphor mask achieves much sharper // results than mipmapping, outputting >= mask_resize_num_tiles makes for // easier tiled sampling later. #ifdef PHOSPHOR_MASK_MANUALLY_RESIZE // Discard unneeded fragments in case our profile allows real branches. const vec2 tile_uv_wrap = tile_uv_wrap; if(get_mask_sample_mode() < 0.5 && max(tile_uv_wrap.x, tile_uv_wrap.y) <= mask_resize_num_tiles) { const float src_dx = src_dxdy.x; const vec2 src_tex_uv = fract(src_tex_uv_wrap); const vec3 pixel_color = downsample_horizontal_sinc_tiled(Source, src_tex_uv, registers.SourceSize.xy, src_dxdy.x, resize_magnification_scale.x, tile_size_uv.x); // The input LUT was linear RGB, and so is our output: FragColor = vec4(pixel_color, 1.0); } else { discard; } #else discard; FragColor = vec4(1.0); #endif }