From 0bac36d8f63b08c231872e9c8aa21163ae3b010e Mon Sep 17 00:00:00 2001 From: Alex Cristici Date: Wed, 13 Nov 2024 17:56:29 +0200 Subject: [PATCH] Line SDF shader fix (#3010) --- include/mbgl/shaders/mtl/line.hpp | 26 ++++++++++++++------------ include/mbgl/shaders/vulkan/line.hpp | 13 +++++++------ 2 files changed, 21 insertions(+), 18 deletions(-) diff --git a/include/mbgl/shaders/mtl/line.hpp b/include/mbgl/shaders/mtl/line.hpp index eb66d8788a9..fd377413c5a 100644 --- a/include/mbgl/shaders/mtl/line.hpp +++ b/include/mbgl/shaders/mtl/line.hpp @@ -110,7 +110,7 @@ FragmentStage vertex vertexMain(thread const VertexStage vertx [[stage_in]], const float ANTIALIASING = 1.0 / DEVICE_PIXEL_RATIO / 2.0; const float2 a_extrude = float2(vertx.data.xy) - 128.0; - const float a_direction = fmod(float(vertx.data.z), 4.0) - 1.0; + const float a_direction = glMod(float(vertx.data.z), 4.0) - 1.0; const float2 pos = floor(float2(vertx.pos_normal) * 0.5); // x is 1 if it's a round cap, 0 otherwise @@ -297,7 +297,7 @@ FragmentStage vertex vertexMain(thread const VertexStage vertx [[stage_in]], const float ANTIALIASING = 1.0 / DEVICE_PIXEL_RATIO / 2.0; const float2 a_extrude = float2(vertx.data.xy) - 128.0; - const float a_direction = fmod(float(vertx.data.z), 4.0) - 1.0; + const float a_direction = glMod(float(vertx.data.z), 4.0) - 1.0; const float v_lineprogress = (floor(float(vertx.data.z) / 4.0) + vertx.data.w * 64.0) * 2.0 / MAX_LINE_DISTANCE; const float2 pos = floor(float2(vertx.pos_normal) * 0.5); @@ -501,7 +501,7 @@ FragmentStage vertex vertexMain(thread const VertexStage vertx [[stage_in]], const float LINE_DISTANCE_SCALE = 2.0; const float2 a_extrude = float2(vertx.data.xy) - 128.0; - const float a_direction = fmod(float(vertx.data.z), 4.0) - 1.0; + const float a_direction = glMod(float(vertx.data.z), 4.0) - 1.0; const float linesofar = (floor(vertx.data.z / 4.0) + vertx.data.w * 64.0) * LINE_DISTANCE_SCALE; const float2 pos = floor(float2(vertx.pos_normal) * 0.5); @@ -731,16 +731,16 @@ FragmentStage vertex vertexMain(thread const VertexStage vertx [[stage_in]], #if defined(HAS_UNIFORM_u_gapwidth) const auto exprGapWidth = (props.expressionMask & LineExpressionMask::GapWidth); - const auto gapwidth = (exprGapWidth ? expr.gapwidth.eval(paintParams.zoom) : props.gapwidth) / 2; + const auto gapwidth = (exprGapWidth ? expr.gapwidth.eval(paintParams.zoom) : props.gapwidth) / 2.0; #else - const auto gapwidth = unpack_mix_float(vertx.gapwidth, interp.gapwidth_t) / 2; + const auto gapwidth = unpack_mix_float(vertx.gapwidth, interp.gapwidth_t) / 2.0; #endif #if defined(HAS_UNIFORM_u_offset) const auto exprOffset = (props.expressionMask & LineExpressionMask::Offset); - const auto offset = (exprOffset ? expr.offset.eval(paintParams.zoom) : props.offset) * -1; + const auto offset = (exprOffset ? expr.offset.eval(paintParams.zoom) : props.offset) * -1.0; #else - const auto offset = unpack_mix_float(vertx.offset, interp.offset_t) * -1; + const auto offset = unpack_mix_float(vertx.offset, interp.offset_t) * -1.0; #endif #if defined(HAS_UNIFORM_u_width) @@ -763,8 +763,8 @@ FragmentStage vertex vertexMain(thread const VertexStage vertx [[stage_in]], const float LINE_DISTANCE_SCALE = 2.0; const float2 a_extrude = float2(vertx.data.xy) - 128.0; - const float a_direction = fmod(float(vertx.data.z), 4.0) - 1.0; - float linesofar = (floor(vertx.data.z / 4.0) + vertx.data.w * 64.0) * LINE_DISTANCE_SCALE; + const float a_direction = glMod(float(vertx.data.z), 4.0) - 1.0; + float linesofar = (floor(float(vertx.data.z) / 4.0) + float(vertx.data.w) * 64.0) * LINE_DISTANCE_SCALE; const float2 pos = floor(float2(vertx.pos_normal) * 0.5); // x is 1 if it's a round cap, 0 otherwise @@ -800,8 +800,8 @@ FragmentStage vertex vertexMain(thread const VertexStage vertx [[stage_in]], .width2 = float2(outset, inset), .normal = v_normal, .gamma_scale = half(extrude_length_without_perspective / extrude_length_with_perspective), - .tex_a = float2(linesofar * drawable.patternscale_a.x / floorwidth, (normal.y * drawable.patternscale_a.y + drawable.tex_y_a) * 2.0), - .tex_b = float2(linesofar * drawable.patternscale_b.x / floorwidth, (normal.y * drawable.patternscale_b.y + drawable.tex_y_b) * 2.0), + .tex_a = float2(linesofar * drawable.patternscale_a.x / floorwidth, v_normal.y * drawable.patternscale_a.y + drawable.tex_y_a), + .tex_b = float2(linesofar * drawable.patternscale_b.x / floorwidth, v_normal.y * drawable.patternscale_b.y + drawable.tex_y_b), #if !defined(HAS_UNIFORM_u_color) .color = unpack_mix_color(vertx.color, interp.color_t), @@ -857,6 +857,9 @@ half4 fragment fragmentMain(FragmentStage in [[stage_in]], const auto floorwidth = in.floorwidth; #endif + // Calculate the distance of the pixel from the line in pixels. + const float dist = length(in.normal) * in.width2.x; + // Calculate the antialiasing fade factor. This is either when fading in the // line in case of an offset line (`v_width2.y`) or when fading out (`v_width2.x`) const float blur2 = (blur + 1.0 / DEVICE_PIXEL_RATIO) * in.gamma_scale; @@ -864,7 +867,6 @@ half4 fragment fragmentMain(FragmentStage in [[stage_in]], const float sdfdist_a = image0.sample(image0_sampler, in.tex_a).a; const float sdfdist_b = image0.sample(image0_sampler, in.tex_b).a; const float sdfdist = mix(sdfdist_a, sdfdist_b, drawable.mix); - const float dist = length(in.normal) * in.width2.x; const float alpha = clamp(min(dist - (in.width2.y - blur2), in.width2.x - dist) / blur2, 0.0, 1.0) * smoothstep(0.5 - drawable.sdfgamma / floorwidth, 0.5 + drawable.sdfgamma / floorwidth, sdfdist); diff --git a/include/mbgl/shaders/vulkan/line.hpp b/include/mbgl/shaders/vulkan/line.hpp index e0fc93f7aee..347f212dc5f 100644 --- a/include/mbgl/shaders/vulkan/line.hpp +++ b/include/mbgl/shaders/vulkan/line.hpp @@ -887,9 +887,9 @@ void main() { #endif #ifndef HAS_UNIFORM_u_offset - const mediump float offset = unpack_mix_float(in_offset, interp.offset_t); + const mediump float offset = unpack_mix_float(in_offset, interp.offset_t) * -1.0; #else - const mediump float offset = props.offset; + const mediump float offset = props.offset * -1.0; #endif #ifndef HAS_UNIFORM_u_width @@ -919,7 +919,6 @@ void main() { // We store these in the least significant bit of in_pos_normal mediump vec2 normal = in_pos_normal - 2.0 * pos; frag_normal = vec2(normal.x, normal.y * 2.0 - 1.0); - frag_normal.y *= -1.0; // these transformations used to be applied in the JS and native code bases. // moved them into the shader for clarity and simplicity. @@ -951,8 +950,8 @@ void main() { frag_width2 = vec2(outset, inset); - frag_tex_a = vec2(linesofar * drawable.patternscale_a.x / floorwidth, (normal.y * drawable.patternscale_a.y + drawable.tex_y_a) * 2.0); - frag_tex_b = vec2(linesofar * drawable.patternscale_b.x / floorwidth, (normal.y * drawable.patternscale_b.y + drawable.tex_y_b) * 2.0); + frag_tex_a = vec2(linesofar * drawable.patternscale_a.x / floorwidth, frag_normal.y * drawable.patternscale_a.y + drawable.tex_y_a); + frag_tex_b = vec2(linesofar * drawable.patternscale_b.x / floorwidth, frag_normal.y * drawable.patternscale_b.y + drawable.tex_y_b); } )"; @@ -1050,6 +1049,9 @@ void main() { const lowp float floorwidth = frag_floorwidth; #endif + // Calculate the distance of the pixel from the line in pixels. + const float dist = length(frag_normal) * frag_width2.x; + // Calculate the antialiasing fade factor. This is either when fading in the // line in case of an offset line (`v_width2.y`) or when fading out (`v_width2.x`) const float blur2 = (blur + 1.0 / DEVICE_PIXEL_RATIO) * frag_gamma_scale; @@ -1057,7 +1059,6 @@ void main() { const float sdfdist_a = texture(image0_sampler, frag_tex_a).a; const float sdfdist_b = texture(image0_sampler, frag_tex_b).a; const float sdfdist = mix(sdfdist_a, sdfdist_b, drawable.mix); - const float dist = length(frag_normal) * frag_width2.x; const float alpha = clamp(min(dist - (frag_width2.y - blur2), frag_width2.x - dist) / blur2, 0.0, 1.0) * smoothstep(0.5 - drawable.sdfgamma / floorwidth, 0.5 + drawable.sdfgamma / floorwidth, sdfdist);