From 34c7ff56401aa9a57cbf0cec8a74613f80555395 Mon Sep 17 00:00:00 2001 From: "r.kuznetsov" Date: Fri, 14 Sep 2018 19:34:43 +0300 Subject: [PATCH] [drape][metal] Added SMAA shaders for Metal. --- drape_frontend/frontend_renderer.cpp | 3 +- drape_frontend/postprocess_renderer.cpp | 25 +- drape_frontend/screen_quad_renderer.cpp | 7 +- drape_frontend/screen_quad_renderer.hpp | 4 +- shaders/Metal/screen_quad.metal | 6 +- shaders/Metal/smaa.metal | 377 ++++++++++++++++++ shaders/metal_program_pool.mm | 6 +- shaders/program_params.hpp | 1 + .../shaders/shaders.xcodeproj/project.pbxproj | 4 + 9 files changed, 415 insertions(+), 18 deletions(-) create mode 100644 shaders/Metal/smaa.metal diff --git a/drape_frontend/frontend_renderer.cpp b/drape_frontend/frontend_renderer.cpp index df52ffec03..9bae524d5d 100755 --- a/drape_frontend/frontend_renderer.cpp +++ b/drape_frontend/frontend_renderer.cpp @@ -1456,7 +1456,8 @@ void FrontendRenderer::RenderTransitBackground() m_transitBackground->SetTextureRect(region.GetTexRect()); CHECK(m_context != nullptr, ()); - m_transitBackground->RenderTexture(m_context, make_ref(m_gpuProgramManager), region.GetTexture(), 1.0f); + m_transitBackground->RenderTexture(m_context, make_ref(m_gpuProgramManager), + region.GetTexture(), 1.0f /* opacity */, false /* invertV */); } void FrontendRenderer::RenderRouteLayer(ScreenBase const & modelView) diff --git a/drape_frontend/postprocess_renderer.cpp b/drape_frontend/postprocess_renderer.cpp index 35f2df7ec2..8a07d3765f 100644 --- a/drape_frontend/postprocess_renderer.cpp +++ b/drape_frontend/postprocess_renderer.cpp @@ -224,13 +224,19 @@ bool PostprocessRenderer::CanRenderAntialiasing() const return false; } - if (m_staticTextures == nullptr) + if (m_staticTextures == nullptr || + m_staticTextures->m_smaaSearchTexture == nullptr || + m_staticTextures->m_smaaAreaTexture == nullptr) + { return false; + } - return m_staticTextures->m_smaaSearchTexture != nullptr && - m_staticTextures->m_smaaAreaTexture != nullptr && - m_staticTextures->m_smaaAreaTexture->GetID() != 0 && - m_staticTextures->m_smaaSearchTexture->GetID() != 0; + if (m_apiVersion == dp::ApiVersion::OpenGLES2 || m_apiVersion == dp::ApiVersion::OpenGLES3) + { + return m_staticTextures->m_smaaAreaTexture->GetID() != 0 && + m_staticTextures->m_smaaSearchTexture->GetID() != 0; + } + return true; } bool PostprocessRenderer::BeginFrame(ref_ptr context, bool activeFrame) @@ -263,10 +269,13 @@ bool PostprocessRenderer::EndFrame(ref_ptr context, if (m_frameStarted && CanRenderAntialiasing()) { ASSERT(m_staticTextures->m_smaaAreaTexture != nullptr, ()); - ASSERT_GREATER(m_staticTextures->m_smaaAreaTexture->GetID(), 0, ()); - ASSERT(m_staticTextures->m_smaaSearchTexture != nullptr, ()); - ASSERT_GREATER(m_staticTextures->m_smaaSearchTexture->GetID(), 0, ()); + + if (m_apiVersion == dp::ApiVersion::OpenGLES2 || m_apiVersion == dp::ApiVersion::OpenGLES3) + { + ASSERT_GREATER(m_staticTextures->m_smaaAreaTexture->GetID(), 0, ()); + ASSERT_GREATER(m_staticTextures->m_smaaSearchTexture->GetID(), 0, ()); + } context->SetClearColor(dp::Color::Transparent()); context->SetStencilTestEnabled(true); diff --git a/drape_frontend/screen_quad_renderer.cpp b/drape_frontend/screen_quad_renderer.cpp index e41f7edd04..3e72d6ea5a 100644 --- a/drape_frontend/screen_quad_renderer.cpp +++ b/drape_frontend/screen_quad_renderer.cpp @@ -22,11 +22,12 @@ public: } void SetParams(ref_ptr gpuProgramManager, - ref_ptr texture, float opacity) + ref_ptr texture, float opacity, bool invertV) { UNUSED_VALUE(gpuProgramManager); m_state.SetTexture("u_colorTex", std::move(texture)); m_params.m_opacity = opacity; + m_params.m_invertV = invertV ? 1.0f : 0.0f; } dp::RenderState const & GetRenderState() const { return m_state; } @@ -58,10 +59,10 @@ void ScreenQuadRenderer::Rebuild() void ScreenQuadRenderer::RenderTexture(ref_ptr context, ref_ptr mng, - ref_ptr texture, float opacity) + ref_ptr texture, float opacity, bool invertV) { TextureRenderParams params; - params.SetParams(mng, std::move(texture), opacity); + params.SetParams(mng, std::move(texture), opacity, invertV); auto program = mng->GetProgram(params.GetRenderState().GetProgram()); Base::Render(context, program, params.GetRenderState(), mng->GetParamsSetter(), diff --git a/drape_frontend/screen_quad_renderer.hpp b/drape_frontend/screen_quad_renderer.hpp index f5fef58ecb..e46f97e047 100644 --- a/drape_frontend/screen_quad_renderer.hpp +++ b/drape_frontend/screen_quad_renderer.hpp @@ -20,8 +20,10 @@ public: void SetTextureRect(m2::RectF const & rect); m2::RectF const & GetTextureRect() const { return m_textureRect; } + // The parameter invertV is necessary, since there are some APIs (e.g. Metal) there render target + // coordinates system is inverted by V axis. void RenderTexture(ref_ptr context, ref_ptr mng, - ref_ptr texture, float opacity); + ref_ptr texture, float opacity, bool invertV = true); private: void Rebuild(); diff --git a/shaders/Metal/screen_quad.metal b/shaders/Metal/screen_quad.metal index 0e5695024f..bf0e6366aa 100644 --- a/shaders/Metal/screen_quad.metal +++ b/shaders/Metal/screen_quad.metal @@ -17,13 +17,15 @@ typedef struct typedef struct { float u_opacity; + float u_invertV; } Uniforms_T; -vertex Fragment_T vsScreenQuad(const Vertex_T in [[stage_in]]) +vertex Fragment_T vsScreenQuad(const Vertex_T in [[stage_in]], + constant Uniforms_T & uniforms [[buffer(1)]]) { Fragment_T out; out.position = float4(in.a_position, 0.0, 1.0); - out.texCoords = float2(in.a_texCoords.x, 1.0 - in.a_texCoords.y); + out.texCoords = mix(in.a_texCoords, float2(in.a_texCoords.x, 1.0 - in.a_texCoords.y), uniforms.u_invertV); return out; } diff --git a/shaders/Metal/smaa.metal b/shaders/Metal/smaa.metal new file mode 100644 index 0000000000..da9a7abb8a --- /dev/null +++ b/shaders/Metal/smaa.metal @@ -0,0 +1,377 @@ +// Implementation of Subpixel Morphological Antialiasing (SMAA) is based on https://github.com/iryoku/smaa + +#include +#include +using namespace metal; + +typedef struct +{ + float4 u_framebufferMetrics; +} Uniforms_T; + +typedef struct +{ + float2 a_pos [[attribute(0)]]; + float2 a_tcoord [[attribute(1)]]; +} SmaaVertex_T; + +// SmaaEdges + +typedef struct +{ + float4 position [[position]]; + float4 offset0; + float4 offset1; + float4 offset2; + float2 coords; +} SmaaEdgesFragment_T; + +vertex SmaaEdgesFragment_T vsSmaaEdges(const SmaaVertex_T in [[stage_in]], + constant Uniforms_T & uniforms [[buffer(1)]]) +{ + SmaaEdgesFragment_T out; + + out.position = float4(in.a_pos, 0.0, 1.0); + float2 tcoord = float2(in.a_tcoord.x, 1.0 - in.a_tcoord.y); + out.coords = tcoord; + float4 coords1 = uniforms.u_framebufferMetrics.xyxy; + float4 coords2 = tcoord.xyxy; + out.offset0 = coords1 * float4(-1.0, 0.0, 0.0, -1.0) + coords2; + out.offset1 = coords1 * float4( 1.0, 0.0, 0.0, 1.0) + coords2; + out.offset2 = coords1 * float4(-2.0, 0.0, 0.0, -2.0) + coords2; + + return out; +} + +// SMAA_THRESHOLD specifies the threshold or sensitivity to edges. +// Lowering this value you will be able to detect more edges at the expense of +// performance. +// Range: [0, 0.5] +// 0.1 is a reasonable value, and allows to catch most visible edges. +// 0.05 is a rather overkill value, that allows to catch 'em all. +#define SMAA_THRESHOLD 0.05 +constant float2 kThreshold = float2(SMAA_THRESHOLD, SMAA_THRESHOLD); + +// If there is an neighbor edge that has SMAA_LOCAL_CONTRAST_FACTOR times +// bigger contrast than current edge, current edge will be discarded. +// This allows to eliminate spurious crossing edges, and is based on the fact +// that, if there is too much contrast in a direction, that will hide +// perceptually contrast in the other neighbors. +#define SMAA_LOCAL_CONTRAST_ADAPTATION_FACTOR 2.0 + +// Standard relative luminance weights. +// https://en.wikipedia.org/wiki/Relative_luminance +constant float3 kWeights = float3(0.2126, 0.7152, 0.0722); + +fragment float4 fsSmaaEdges(const SmaaEdgesFragment_T in [[stage_in]], + texture2d u_colorTex [[texture(0)]], + sampler u_colorTexSampler [[sampler(0)]]) +{ + // Calculate lumas. + float L = dot(u_colorTex.sample(u_colorTexSampler, in.coords).rgb, kWeights); + float Lleft = dot(u_colorTex.sample(u_colorTexSampler, in.offset0.xy).rgb, kWeights); + float Ltop = dot(u_colorTex.sample(u_colorTexSampler, in.offset0.zw).rgb, kWeights); + + // We do the usual threshold. + float4 delta; + delta.xy = abs(L - float2(Lleft, Ltop)); + float2 edges = step(kThreshold, delta.xy); + if (dot(edges, float2(1.0, 1.0)) == 0.0) + discard_fragment(); + + // Calculate right and bottom deltas. + float Lright = dot(u_colorTex.sample(u_colorTexSampler, in.offset1.xy).rgb, kWeights); + float Lbottom = dot(u_colorTex.sample(u_colorTexSampler, in.offset1.zw).rgb, kWeights); + delta.zw = abs(L - float2(Lright, Lbottom)); + + // Calculate the maximum delta in the direct neighborhood. + float2 maxDelta = max(delta.xy, delta.zw); + + // Calculate left-left and top-top deltas. + float Lleftleft = dot(u_colorTex.sample(u_colorTexSampler, in.offset2.xy).rgb, kWeights); + float Ltoptop = dot(u_colorTex.sample(u_colorTexSampler, in.offset2.zw).rgb, kWeights); + delta.zw = abs(float2(Lleft, Ltop) - float2(Lleftleft, Ltoptop)); + + // Calculate the final maximum delta. + maxDelta = max(maxDelta.xy, delta.zw); + float finalDelta = max(maxDelta.x, maxDelta.y); + + // Local contrast adaptation. + edges *= step(finalDelta, SMAA_LOCAL_CONTRAST_ADAPTATION_FACTOR * delta.xy); + + return float4(edges, 0.0, 1.0); +} + +// SmaaBlendingWeight + +typedef struct +{ + float4 position [[position]]; + float4 offset0; + float4 offset1; + float4 offset2; + float4 coords; +} SmaaBlendingWeightFragment_T; + +// SMAA_MAX_SEARCH_STEPS specifies the maximum steps performed in the +// horizontal/vertical pattern searches, at each side of the pixel. +#define SMAA_MAX_SEARCH_STEPS 8.0 +constant float4 kMaxSearchSteps = float4(-2.0 * SMAA_MAX_SEARCH_STEPS, 2.0 * SMAA_MAX_SEARCH_STEPS, + -2.0 * SMAA_MAX_SEARCH_STEPS, 2.0 * SMAA_MAX_SEARCH_STEPS); + +vertex SmaaBlendingWeightFragment_T vsSmaaBlendingWeight(const SmaaVertex_T in [[stage_in]], + constant Uniforms_T & uniforms [[buffer(1)]]) +{ + SmaaBlendingWeightFragment_T out; + + out.position = float4(in.a_pos, 0.0, 1.0); + float2 tcoord = float2(in.a_tcoord.x, 1.0 - in.a_tcoord.y); + out.coords = float4(tcoord, tcoord * uniforms.u_framebufferMetrics.zw); + // We will use these offsets for the searches. + float4 coords1 = uniforms.u_framebufferMetrics.xyxy; + float4 coords2 = tcoord.xyxy; + out.offset0 = coords1 * float4(-0.25, -0.125, 1.25, -0.125) + coords2; + out.offset1 = coords1 * float4(-0.125, -0.25, -0.125, 1.25) + coords2; + // And these for the searches, they indicate the ends of the loops. + out.offset2 = uniforms.u_framebufferMetrics.xxyy * kMaxSearchSteps + float4(out.offset0.xz, out.offset1.yw); + + return out; +} + +#define SMAA_SEARCHTEX_SIZE float2(66.0, 33.0) +#define SMAA_SEARCHTEX_PACKED_SIZE float2(64.0, 16.0) +#define SMAA_AREATEX_MAX_DISTANCE 16.0 +#define SMAA_AREATEX_PIXEL_SIZE (float2(1.0 / 256.0, 1.0 / 1024.0)) + +constant float2 kAreaTexMaxDistance = float2(SMAA_AREATEX_MAX_DISTANCE, SMAA_AREATEX_MAX_DISTANCE); +constant float kActivationThreshold = 0.8281; + +float SMAASearchLength(texture2d u_smaaSearch, sampler u_smaaSearchSampler, + float2 e, float offset) +{ + // The texture is flipped vertically, with left and right cases taking half + // of the space horizontally. + float2 scale = SMAA_SEARCHTEX_SIZE * float2(0.5, -1.0); + float2 bias = SMAA_SEARCHTEX_SIZE * float2(offset, 1.0); + + // Scale and bias to access texel centers. + scale += float2(-1.0, 1.0); + bias += float2( 0.5, -0.5); + + // Convert from pixel coordinates to texcoords. + // (We use SMAA_SEARCHTEX_PACKED_SIZE because the texture is cropped). + scale *= 1.0 / SMAA_SEARCHTEX_PACKED_SIZE; + bias *= 1.0 / SMAA_SEARCHTEX_PACKED_SIZE; + + // Lookup the search texture. + return u_smaaSearch.sample(u_smaaSearchSampler, scale * e + bias, level(0)).a; +} + +float SMAASearchXLeft(texture2d u_colorTex, sampler u_colorTexSampler, + texture2d u_smaaSearch, sampler u_smaaSearchSampler, + float2 texcoord, float end, float4 framebufferMetrics) +{ + float2 e = float2(0.0, 1.0); + while (texcoord.x > end && e.g > kActivationThreshold && e.r == 0.0) + { + e = u_colorTex.sample(u_colorTexSampler, texcoord, level(0)).rg; + texcoord = float2(-2.0, 0.0) * framebufferMetrics.xy + texcoord; + } + float offset = 3.25 - (255.0 / 127.0) * SMAASearchLength(u_smaaSearch, u_smaaSearchSampler, e, 0.0); + return framebufferMetrics.x * offset + texcoord.x; +} + +float SMAASearchXRight(texture2d u_colorTex, sampler u_colorTexSampler, + texture2d u_smaaSearch, sampler u_smaaSearchSampler, + float2 texcoord, float end, float4 framebufferMetrics) +{ + float2 e = float2(0.0, 1.0); + while (texcoord.x < end && e.g > kActivationThreshold && e.r == 0.0) + { + e = u_colorTex.sample(u_colorTexSampler, texcoord, level(0)).rg; + texcoord = float2(2.0, 0.0) * framebufferMetrics.xy + texcoord; + } + float offset = 3.25 - (255.0 / 127.0) * SMAASearchLength(u_smaaSearch, u_smaaSearchSampler, e, 0.5); + return -framebufferMetrics.x * offset + texcoord.x; +} + +float SMAASearchYUp(texture2d u_colorTex, sampler u_colorTexSampler, + texture2d u_smaaSearch, sampler u_smaaSearchSampler, + float2 texcoord, float end, float4 framebufferMetrics) +{ + float2 e = float2(1.0, 0.0); + while (texcoord.y > end && e.r > kActivationThreshold && e.g == 0.0) + { + e = u_colorTex.sample(u_colorTexSampler, texcoord, level(0)).rg; + texcoord = float2(0.0, -2.0) * framebufferMetrics.xy + texcoord; + } + float offset = 3.25 - (255.0 / 127.0) * SMAASearchLength(u_smaaSearch, u_smaaSearchSampler, e.gr, 0.0); + return framebufferMetrics.y * offset + texcoord.y; +} + +float SMAASearchYDown(texture2d u_colorTex, sampler u_colorTexSampler, + texture2d u_smaaSearch, sampler u_smaaSearchSampler, + float2 texcoord, float end, float4 framebufferMetrics) +{ + float2 e = float2(1.0, 0.0); + while (texcoord.y < end && e.r > kActivationThreshold && e.g == 0.0) + { + e = u_colorTex.sample(u_colorTexSampler, texcoord, level(0)).rg; + texcoord = float2(0.0, 2.0) * framebufferMetrics.xy + texcoord; + } + float offset = 3.25 - (255.0 / 127.0) * SMAASearchLength(u_smaaSearch, u_smaaSearchSampler, e.gr, 0.5); + return -framebufferMetrics.y * offset + texcoord.y; +} + +// Here, we have the distance and both crossing edges. So, what are the areas +// at each side of current edge? +float2 SMAAArea(texture2d u_smaaArea, sampler u_smaaAreaSampler, float2 dist, float e1, float e2) +{ + // Rounding prevents precision errors of bilinear filtering. + float2 texcoord = kAreaTexMaxDistance * round(4.0 * float2(e1, e2)) + dist; + // We do a scale and bias for mapping to texel space. + texcoord = SMAA_AREATEX_PIXEL_SIZE * (texcoord + 0.5); + return u_smaaArea.sample(u_smaaAreaSampler, texcoord, level(0)).rg; +} + +fragment float4 fsSmaaBlendingWeight(const SmaaBlendingWeightFragment_T in [[stage_in]], + constant Uniforms_T & uniforms [[buffer(0)]], + texture2d u_colorTex [[texture(0)]], + sampler u_colorTexSampler [[sampler(0)]], + texture2d u_smaaArea [[texture(1)]], + sampler u_smaaAreaSampler [[sampler(1)]], + texture2d u_smaaSearch [[texture(2)]], + sampler u_smaaSearchSampler [[sampler(2)]]) +{ + float4 weights = float4(0.0, 0.0, 0.0, 0.0); + float2 e = u_colorTex.sample(u_colorTexSampler, in.coords.xy).rg; + + if (e.g > 0.0) // Edge at north + { + float2 d; + + // Find the distance to the left. + float3 coords; + coords.x = SMAASearchXLeft(u_colorTex, u_colorTexSampler, u_smaaSearch, u_smaaSearchSampler, + in.offset0.xy, in.offset2.x, uniforms.u_framebufferMetrics); + coords.y = in.offset1.y; + d.x = coords.x; + + // Now fetch the left crossing edges, two at a time using bilinear + // filtering. Sampling at -0.25 enables to discern what value each edge has. + float e1 = u_colorTex.sample(u_colorTexSampler, coords.xy, level(0)).r; + + // Find the distance to the right. + coords.z = SMAASearchXRight(u_colorTex, u_colorTexSampler, u_smaaSearch, u_smaaSearchSampler, + in.offset0.zw, in.offset2.y, uniforms.u_framebufferMetrics); + d.y = coords.z; + + // We want the distances to be in pixel units (doing this here allow to + // better interleave arithmetic and memory accesses). + d = abs(round(uniforms.u_framebufferMetrics.zz * d - in.coords.zz)); + + // SMAAArea below needs a sqrt, as the areas texture is compressed + // quadratically. + float2 sqrtD = sqrt(d); + + // Fetch the right crossing edges. + float e2 = u_colorTex.sample(u_colorTexSampler, coords.zy, level(0), int2(1, 0)).r; + + // Here we know how this pattern looks like, now it is time for getting + // the actual area. + weights.rg = SMAAArea(u_smaaArea, u_smaaAreaSampler, sqrtD, e1, e2); + } + + if (e.r > 0.0) // Edge at west + { + float2 d; + + // Find the distance to the top. + float3 coords; + coords.y = SMAASearchYUp(u_colorTex, u_colorTexSampler, u_smaaSearch, u_smaaSearchSampler, + in.offset1.xy, in.offset2.z, uniforms.u_framebufferMetrics); + coords.x = in.offset0.x; + d.x = coords.y; + + // Fetch the top crossing edges. + float e1 = u_colorTex.sample(u_colorTexSampler, coords.xy, level(0)).g; + + // Find the distance to the bottom. + coords.z = SMAASearchYDown(u_colorTex, u_colorTexSampler, u_smaaSearch, u_smaaSearchSampler, + in.offset1.zw, in.offset2.w, uniforms.u_framebufferMetrics); + d.y = coords.z; + + // We want the distances to be in pixel units. + d = abs(round(uniforms.u_framebufferMetrics.ww * d - in.coords.ww)); + + // SMAAArea below needs a sqrt, as the areas texture is compressed + // quadratically. + float2 sqrtD = sqrt(d); + + // Fetch the bottom crossing edges. + float e2 = u_colorTex.sample(u_colorTexSampler, coords.xz, level(0), int2(0, 1)).g; + + // Get the area for this direction. + weights.ba = SMAAArea(u_smaaArea, u_smaaAreaSampler, sqrtD, e1, e2); + } + + return weights; +} + +// SmaaFinal + +typedef struct +{ + float4 position [[position]]; + float4 offset; + float2 texCoords; +} SmaaFinalFragment_T; + +vertex SmaaFinalFragment_T vsSmaaFinal(const SmaaVertex_T in [[stage_in]], + constant Uniforms_T & uniforms [[buffer(1)]]) +{ + SmaaFinalFragment_T out; + float2 tcoord = float2(in.a_tcoord.x, 1.0 - in.a_tcoord.y); + out.position = float4(in.a_pos, 0.0, 1.0); + out.offset = uniforms.u_framebufferMetrics.xyxy * float4(1.0, 0.0, 0.0, 1.0) + tcoord.xyxy; + out.texCoords = tcoord; + + return out; +} + +fragment float4 fsSmaaFinal(const SmaaFinalFragment_T in [[stage_in]], + constant Uniforms_T & uniforms [[buffer(0)]], + texture2d u_colorTex [[texture(0)]], + sampler u_colorTexSampler [[sampler(0)]], + texture2d u_blendingWeightTex [[texture(1)]], + sampler u_blendingWeightTexSampler [[sampler(1)]]) +{ + // Fetch the blending weights for current pixel. + float4 a; + a.x = u_blendingWeightTex.sample(u_blendingWeightTexSampler, in.offset.xy).a; // Right + a.y = u_blendingWeightTex.sample(u_blendingWeightTexSampler, in.offset.zw).g; // Top + a.wz = u_blendingWeightTex.sample(u_blendingWeightTexSampler, in.texCoords).xz; // Bottom / Left + + // Is there any blending weight with a value greater than 0.0? + if (dot(a, float4(1.0, 1.0, 1.0, 1.0)) < 1e-5) + return u_colorTex.sample(u_colorTexSampler, in.texCoords); + + // Calculate the blending offsets. + float4 blendingOffset = float4(0.0, a.y, 0.0, a.w); + float2 blendingWeight = a.yw; + if (max(a.x, a.z) > max(a.y, a.w)) + { + blendingOffset = float4(a.x, 0.0, a.z, 0.0); + blendingWeight = a.xz; + } + blendingWeight /= dot(blendingWeight, float2(1.0, 1.0)); + + // Calculate the texture coordinates. + float4 bc = blendingOffset * float4(uniforms.u_framebufferMetrics.xy, -uniforms.u_framebufferMetrics.xy); + bc += in.texCoords.xyxy; + + // We exploit bilinear filtering to mix current pixel with the chosen neighbor. + float4 color = blendingWeight.x * u_colorTex.sample(u_colorTexSampler, bc.xy, level(0)); + color += blendingWeight.y * u_colorTex.sample(u_colorTexSampler, bc.zw, level(0)); + return color; +} diff --git a/shaders/metal_program_pool.mm b/shaders/metal_program_pool.mm index 894e24d1a9..4f0f92768c 100644 --- a/shaders/metal_program_pool.mm +++ b/shaders/metal_program_pool.mm @@ -94,9 +94,9 @@ std::array(Program::ProgramsCount)> const kMeta ProgramInfo("vsTraffic", "fsTraffic", {{0, 2}}), // Traffic ProgramInfo("vsTrafficLine", "fsTrafficLine", {{0, 1}}), // TrafficLine ProgramInfo("vsTrafficCircle", "fsTrafficCircle", {{0, 2}}), // TrafficCircle - ProgramInfo("", "", {}), // SmaaEdges - ProgramInfo("", "", {}), // SmaaBlendingWeight - ProgramInfo("", "", {}), // SmaaFinal + ProgramInfo("vsSmaaEdges", "fsSmaaEdges", {{0, 1}}), // SmaaEdges + ProgramInfo("vsSmaaBlendingWeight", "fsSmaaBlendingWeight", {{0, 1}}), // SmaaBlendingWeight + ProgramInfo("vsSmaaFinal", "fsSmaaFinal", {{0, 1}}), // SmaaFinal }}; MTLVertexFormat GetFormatByDataType(MTLDataType dataType) diff --git a/shaders/program_params.hpp b/shaders/program_params.hpp index 10abadd2c7..81b9ca2dbe 100644 --- a/shaders/program_params.hpp +++ b/shaders/program_params.hpp @@ -199,6 +199,7 @@ struct DebugRectProgramParams struct ScreenQuadProgramParams { float m_opacity = 1.0f; + float m_invertV = 1.0f; BIND_PROGRAMS(ScreenQuadProgramParams, Program::ScreenQuad) } ALIGNMENT; diff --git a/xcode/shaders/shaders.xcodeproj/project.pbxproj b/xcode/shaders/shaders.xcodeproj/project.pbxproj index 52174520ab..de28ab2ed6 100644 --- a/xcode/shaders/shaders.xcodeproj/project.pbxproj +++ b/xcode/shaders/shaders.xcodeproj/project.pbxproj @@ -10,6 +10,7 @@ 451E692D214A53C000764A97 /* route.metal in Sources */ = {isa = PBXBuildFile; fileRef = 451E692C214A53C000764A97 /* route.metal */; }; 451E692F214A6FA200764A97 /* map.metal in Sources */ = {isa = PBXBuildFile; fileRef = 451E692E214A6FA200764A97 /* map.metal */; }; 451E6931214A8D4200764A97 /* shaders_lib.metal in Sources */ = {isa = PBXBuildFile; fileRef = 451E6930214A8D4200764A97 /* shaders_lib.metal */; }; + 45355A43214BF358007740F8 /* smaa.metal in Sources */ = {isa = PBXBuildFile; fileRef = 45355A42214BF358007740F8 /* smaa.metal */; }; 4560F58A213D57D600CC736C /* debug_rect.metal in Sources */ = {isa = PBXBuildFile; fileRef = 45789EDC21342BDE009955CC /* debug_rect.metal */; }; 4560F58B213D57D600CC736C /* screen_quad.metal in Sources */ = {isa = PBXBuildFile; fileRef = 4560F582213D44CE00CC736C /* screen_quad.metal */; }; 4560F5AB2142AC1300CC736C /* gui.metal in Sources */ = {isa = PBXBuildFile; fileRef = 4560F5AA2142AC1300CC736C /* gui.metal */; }; @@ -70,6 +71,7 @@ 451E692C214A53C000764A97 /* route.metal */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.metal; path = route.metal; sourceTree = ""; }; 451E692E214A6FA200764A97 /* map.metal */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.metal; path = map.metal; sourceTree = ""; }; 451E6930214A8D4200764A97 /* shaders_lib.metal */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.metal; path = shaders_lib.metal; sourceTree = ""; }; + 45355A42214BF358007740F8 /* smaa.metal */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.metal; path = smaa.metal; sourceTree = ""; }; 4560F582213D44CE00CC736C /* screen_quad.metal */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.metal; path = screen_quad.metal; sourceTree = ""; }; 4560F5AA2142AC1300CC736C /* gui.metal */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.metal; path = gui.metal; sourceTree = ""; }; 4561ADF320E378CB0096BC12 /* program_manager.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = program_manager.cpp; sourceTree = ""; }; @@ -407,6 +409,7 @@ BBF7916F2146D8EC00D27BD8 /* system.metal */, BBF79177214AA53C00D27BD8 /* traffic.metal */, BBF79178214AA92D00D27BD8 /* transit.metal */, + 45355A42214BF358007740F8 /* smaa.metal */, ); path = Metal; sourceTree = ""; @@ -565,6 +568,7 @@ buildActionMask = 2147483647; files = ( 451E692F214A6FA200764A97 /* map.metal in Sources */, + 45355A43214BF358007740F8 /* smaa.metal in Sources */, 4560F5AB2142AC1300CC736C /* gui.metal in Sources */, 451E692D214A53C000764A97 /* route.metal in Sources */, BBF791702146D8EC00D27BD8 /* system.metal in Sources */,