2023-09-10 06:26:58 +00:00
|
|
|
/// A custom resolve kernel that averages color at all sample points.
|
|
|
|
#include <metal_stdlib>
|
|
|
|
using namespace metal;
|
|
|
|
|
|
|
|
// https://developer.apple.com/documentation/metal/metal_sample_code_library/improving_edge-rendering_quality_with_multisample_antialiasing_msaa?language=objc
|
2024-04-12 08:10:31 +00:00
|
|
|
kernel void colorResolveKernel(texture2d_ms<float, access::read> multisampledTexture [[texture(0)]],
|
|
|
|
texture2d<float, access::write> resolvedTexture [[texture(1)]],
|
|
|
|
uint2 gid [[thread_position_in_grid]])
|
2023-09-10 06:26:58 +00:00
|
|
|
{
|
2024-04-12 08:10:31 +00:00
|
|
|
const uint count = multisampledTexture.get_num_samples();
|
|
|
|
|
|
|
|
float4 resolved_color = 0;
|
|
|
|
|
|
|
|
for (uint i = 0; i < count; ++i)
|
|
|
|
{
|
|
|
|
resolved_color += multisampledTexture.read(gid, i);
|
|
|
|
}
|
|
|
|
|
|
|
|
resolved_color /= count;
|
|
|
|
|
|
|
|
resolvedTexture.write(resolved_color, gid);
|
2023-09-10 06:26:58 +00:00
|
|
|
}
|
|
|
|
|
2024-04-12 08:10:31 +00:00
|
|
|
kernel void depthResolveKernel(texture2d_ms<float, access::read> multisampledTexture [[texture(0)]],
|
|
|
|
texture2d<float, access::write> resolvedTexture [[texture(1)]],
|
|
|
|
uint2 gid [[thread_position_in_grid]])
|
2023-09-10 06:26:58 +00:00
|
|
|
{
|
|
|
|
const uint count = multisampledTexture.get_num_samples();
|
2024-04-12 08:10:31 +00:00
|
|
|
|
2023-09-10 06:26:58 +00:00
|
|
|
float resolved_depth = 0;
|
2024-04-12 08:10:31 +00:00
|
|
|
|
2023-09-10 06:26:58 +00:00
|
|
|
for (uint i = 0; i < count; ++i)
|
|
|
|
{
|
2024-04-12 08:10:31 +00:00
|
|
|
resolved_depth += multisampledTexture.read(gid, i).r;
|
2023-09-10 06:26:58 +00:00
|
|
|
}
|
2024-04-12 08:10:31 +00:00
|
|
|
|
2023-09-10 06:26:58 +00:00
|
|
|
resolved_depth /= count;
|
2024-04-12 08:10:31 +00:00
|
|
|
|
2023-09-10 06:26:58 +00:00
|
|
|
resolvedTexture.write(float4(resolved_depth, 0, 0, 0), gid);
|
|
|
|
}
|
2024-04-12 08:10:31 +00:00
|
|
|
|
|
|
|
struct DepthClearUBO
|
|
|
|
{
|
|
|
|
float depth;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct DepthClearOut
|
|
|
|
{
|
|
|
|
float4 pos [[position]];
|
|
|
|
};
|
|
|
|
|
|
|
|
vertex DepthClearOut depthClearVertex(constant DepthClearUBO& ubo [[buffer(0)]], uint vertexId [[vertex_id]])
|
|
|
|
{
|
|
|
|
DepthClearOut out = {};
|
|
|
|
float2 uv = float2(float((vertexId << uint(1)) & 2u), float(vertexId & 2u));
|
|
|
|
out.pos = float4((uv * float2(2.0, -2.0)) + float2(-1.0, 1.0), ubo.depth, 1.0);
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
fragment void depthClearFragment()
|
|
|
|
{
|
|
|
|
}
|