forked from organicmaps/organicmaps
[drape][metal] Added SMAA shaders for Metal.
This commit is contained in:
parent
3cd70284fc
commit
34c7ff5640
9 changed files with 415 additions and 18 deletions
|
@ -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)
|
||||
|
|
|
@ -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<dp::GraphicsContext> context, bool activeFrame)
|
||||
|
@ -263,10 +269,13 @@ bool PostprocessRenderer::EndFrame(ref_ptr<dp::GraphicsContext> 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);
|
||||
|
|
|
@ -22,11 +22,12 @@ public:
|
|||
}
|
||||
|
||||
void SetParams(ref_ptr<gpu::ProgramManager> gpuProgramManager,
|
||||
ref_ptr<dp::Texture> texture, float opacity)
|
||||
ref_ptr<dp::Texture> 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<dp::GraphicsContext> context,
|
||||
ref_ptr<gpu::ProgramManager> mng,
|
||||
ref_ptr<dp::Texture> texture, float opacity)
|
||||
ref_ptr<dp::Texture> 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<gpu::Program>());
|
||||
Base::Render(context, program, params.GetRenderState(), mng->GetParamsSetter(),
|
||||
|
|
|
@ -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<dp::GraphicsContext> context, ref_ptr<gpu::ProgramManager> mng,
|
||||
ref_ptr<dp::Texture> texture, float opacity);
|
||||
ref_ptr<dp::Texture> texture, float opacity, bool invertV = true);
|
||||
|
||||
private:
|
||||
void Rebuild();
|
||||
|
|
|
@ -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;
|
||||
}
|
||||
|
||||
|
|
377
shaders/Metal/smaa.metal
Normal file
377
shaders/Metal/smaa.metal
Normal file
|
@ -0,0 +1,377 @@
|
|||
// Implementation of Subpixel Morphological Antialiasing (SMAA) is based on https://github.com/iryoku/smaa
|
||||
|
||||
#include <metal_stdlib>
|
||||
#include <simd/simd.h>
|
||||
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<float> 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<float> 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<float> u_colorTex, sampler u_colorTexSampler,
|
||||
texture2d<float> 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<float> u_colorTex, sampler u_colorTexSampler,
|
||||
texture2d<float> 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<float> u_colorTex, sampler u_colorTexSampler,
|
||||
texture2d<float> 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<float> u_colorTex, sampler u_colorTexSampler,
|
||||
texture2d<float> 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<float> 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<float> u_colorTex [[texture(0)]],
|
||||
sampler u_colorTexSampler [[sampler(0)]],
|
||||
texture2d<float> u_smaaArea [[texture(1)]],
|
||||
sampler u_smaaAreaSampler [[sampler(1)]],
|
||||
texture2d<float> 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<float> u_colorTex [[texture(0)]],
|
||||
sampler u_colorTexSampler [[sampler(0)]],
|
||||
texture2d<float> 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;
|
||||
}
|
|
@ -94,9 +94,9 @@ std::array<ProgramInfo, static_cast<size_t>(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)
|
||||
|
|
|
@ -199,6 +199,7 @@ struct DebugRectProgramParams
|
|||
struct ScreenQuadProgramParams
|
||||
{
|
||||
float m_opacity = 1.0f;
|
||||
float m_invertV = 1.0f;
|
||||
|
||||
BIND_PROGRAMS(ScreenQuadProgramParams, Program::ScreenQuad)
|
||||
} ALIGNMENT;
|
||||
|
|
|
@ -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 = "<group>"; };
|
||||
451E692E214A6FA200764A97 /* map.metal */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.metal; path = map.metal; sourceTree = "<group>"; };
|
||||
451E6930214A8D4200764A97 /* shaders_lib.metal */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.metal; path = shaders_lib.metal; sourceTree = "<group>"; };
|
||||
45355A42214BF358007740F8 /* smaa.metal */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.metal; path = smaa.metal; sourceTree = "<group>"; };
|
||||
4560F582213D44CE00CC736C /* screen_quad.metal */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.metal; path = screen_quad.metal; sourceTree = "<group>"; };
|
||||
4560F5AA2142AC1300CC736C /* gui.metal */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.metal; path = gui.metal; sourceTree = "<group>"; };
|
||||
4561ADF320E378CB0096BC12 /* program_manager.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = program_manager.cpp; sourceTree = "<group>"; };
|
||||
|
@ -407,6 +409,7 @@
|
|||
BBF7916F2146D8EC00D27BD8 /* system.metal */,
|
||||
BBF79177214AA53C00D27BD8 /* traffic.metal */,
|
||||
BBF79178214AA92D00D27BD8 /* transit.metal */,
|
||||
45355A42214BF358007740F8 /* smaa.metal */,
|
||||
);
|
||||
path = Metal;
|
||||
sourceTree = "<group>";
|
||||
|
@ -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 */,
|
||||
|
|
Loading…
Add table
Reference in a new issue