diff --git a/drape/metal/metal_base_context.mm b/drape/metal/metal_base_context.mm index 0f19cb706f..e0362647df 100644 --- a/drape/metal/metal_base_context.mm +++ b/drape/metal/metal_base_context.mm @@ -160,6 +160,7 @@ void MetalBaseContext::ApplyFramebuffer(std::string const & framebufferLabel) CHECK(m_currentCommandEncoder == nil, ("Current command encoder was not finished.")); m_currentCommandEncoder = [m_frameCommandBuffer renderCommandEncoderWithDescriptor:m_renderPassDescriptor]; + m_currentCommandEncoder.label = @(framebufferLabel.c_str()); [m_currentCommandEncoder pushDebugGroup:@(framebufferLabel.c_str())]; // Default rendering options. diff --git a/drape/metal/metal_cleaner.mm b/drape/metal/metal_cleaner.mm index e99f57913e..e0526ab174 100644 --- a/drape/metal/metal_cleaner.mm +++ b/drape/metal/metal_cleaner.mm @@ -2,6 +2,8 @@ #include "drape/metal/metal_base_context.hpp" #include "drape/metal/metal_gpu_program.hpp" +#include + namespace dp { namespace metal diff --git a/drape/metal/metal_gpu_program.hpp b/drape/metal/metal_gpu_program.hpp index 5329d316ba..f2f314b6c4 100644 --- a/drape/metal/metal_gpu_program.hpp +++ b/drape/metal/metal_gpu_program.hpp @@ -26,13 +26,17 @@ public: MetalGpuProgram(std::string const & programName, id vertexShader, id fragmentShader, int8_t vsUniformsBindingIndex, int8_t fsUniformsBindingIndex, - TexturesBindingInfo && textureBindingInfo) + TexturesBindingInfo && vertexTextureBindingInfo, + TexturesBindingInfo && fragmentTextureBindingInfo, + MTLVertexDescriptor * vertexDescriptor) : GpuProgram(programName) , m_vertexShader(vertexShader) , m_fragmentShader(fragmentShader) , m_vsUniformsBindingIndex(vsUniformsBindingIndex) , m_fsUniformsBindingIndex(fsUniformsBindingIndex) - , m_textureBindingInfo(std::move(textureBindingInfo)) + , m_vertexTextureBindingInfo(std::move(vertexTextureBindingInfo)) + , m_fragmentTextureBindingInfo(std::move(fragmentTextureBindingInfo)) + , m_vertexDescriptor(vertexDescriptor) {} void Bind() override {} @@ -44,22 +48,36 @@ public: int8_t GetVertexShaderUniformsBindingIndex() const { return m_vsUniformsBindingIndex; } int8_t GetFragmentShaderUniformsBindingIndex() const { return m_fsUniformsBindingIndex; } - // Now textures can be bound only in fragment shaders. - TextureBindingInfo const & GetTextureBindingInfo(std::string const & textureName) const + TextureBindingInfo const & GetVertexTextureBindingInfo(std::string const & textureName) const + { + return GetTextureBindingInfo(m_vertexTextureBindingInfo, textureName); + } + + TextureBindingInfo const & GetFragmentTextureBindingInfo(std::string const & textureName) const + { + return GetTextureBindingInfo(m_fragmentTextureBindingInfo, textureName); + } + + MTLVertexDescriptor * GetVertexDescriptor() const { return m_vertexDescriptor; } + +private: + TextureBindingInfo const & GetTextureBindingInfo(TexturesBindingInfo const & bindingInfo, + std::string const & textureName) const { static TextureBindingInfo kEmptyBinding; - auto const it = m_textureBindingInfo.find(textureName); - if (it == m_textureBindingInfo.cend()) + auto const it = bindingInfo.find(textureName); + if (it == bindingInfo.cend()) return kEmptyBinding; return it->second; } - -private: + id m_vertexShader; id m_fragmentShader; int8_t const m_vsUniformsBindingIndex; int8_t const m_fsUniformsBindingIndex; - TexturesBindingInfo const m_textureBindingInfo; + TexturesBindingInfo const m_vertexTextureBindingInfo; + TexturesBindingInfo const m_fragmentTextureBindingInfo; + MTLVertexDescriptor * m_vertexDescriptor; }; } // namespace metal } // namespace dp diff --git a/drape/metal/metal_states.mm b/drape/metal/metal_states.mm index 34145c89f7..f3634da99e 100644 --- a/drape/metal/metal_states.mm +++ b/drape/metal/metal_states.mm @@ -290,6 +290,7 @@ MTLRenderPipelineDescriptor * MetalStates::PipelineKey::BuildDescriptor() const ref_ptr metalProgram = m_program; desc.vertexFunction = metalProgram->GetVertexShader(); desc.fragmentFunction = metalProgram->GetFragmentShader(); + desc.vertexDescriptor = metalProgram->GetVertexDescriptor(); MTLRenderPipelineColorAttachmentDescriptor * colorAttachment = desc.colorAttachments[0]; colorAttachment.pixelFormat = m_colorFormat; desc.depthAttachmentPixelFormat = m_depthStencilFormat; diff --git a/drape/metal/render_state_metal.mm b/drape/metal/render_state_metal.mm index c3a2a06ea6..8ab6517f39 100644 --- a/drape/metal/render_state_metal.mm +++ b/drape/metal/render_state_metal.mm @@ -35,17 +35,38 @@ void ApplyTexturesForMetal(ref_ptr context, ref_ptr id encoder = metalContext->GetCommandEncoder(); for (auto const & texture : state.GetTextures()) { - auto const & bindingInfo = p->GetTextureBindingInfo(texture.first); + if (texture.second == nullptr) + continue; + ref_ptr t = texture.second->GetHardwareTexture(); - if (t != nullptr && bindingInfo.m_textureBindingIndex >= 0) + if (t == nullptr) + continue; + + dp::HWTexture::Params const & params = t->GetParams(); + + // Set texture to the vertex shader. + auto const & vsBindingInfo = p->GetVertexTextureBindingInfo(texture.first); + if (vsBindingInfo.m_textureBindingIndex >= 0) { - [encoder setFragmentTexture:t->GetTexture() atIndex:bindingInfo.m_textureBindingIndex]; - if (bindingInfo.m_samplerBindingIndex >= 0) + [encoder setVertexTexture:t->GetTexture() atIndex:vsBindingInfo.m_textureBindingIndex]; + if (vsBindingInfo.m_samplerBindingIndex >= 0) { - dp::HWTexture::Params const & params = t->GetParams(); id samplerState = metalContext->GetSamplerState(params.m_filter, params.m_wrapSMode, params.m_wrapTMode); - [encoder setFragmentSamplerState:samplerState atIndex:bindingInfo.m_samplerBindingIndex]; + [encoder setVertexSamplerState:samplerState atIndex:vsBindingInfo.m_samplerBindingIndex]; + } + } + + // Set texture to the fragment shader. + auto const & fsBindingInfo = p->GetFragmentTextureBindingInfo(texture.first); + if (fsBindingInfo.m_textureBindingIndex >= 0) + { + [encoder setFragmentTexture:t->GetTexture() atIndex:fsBindingInfo.m_textureBindingIndex]; + if (fsBindingInfo.m_samplerBindingIndex >= 0) + { + id samplerState = metalContext->GetSamplerState(params.m_filter, params.m_wrapSMode, + params.m_wrapTMode); + [encoder setFragmentSamplerState:samplerState atIndex:fsBindingInfo.m_samplerBindingIndex]; } } } diff --git a/shaders/Metal/debug_rect.metal b/shaders/Metal/debug_rect.metal index 206be760be..b795740622 100644 --- a/shaders/Metal/debug_rect.metal +++ b/shaders/Metal/debug_rect.metal @@ -4,7 +4,7 @@ using namespace metal; typedef struct { - packed_float2 a_position; + float2 a_position [[attribute(0)]]; } Vertex_T; typedef struct @@ -17,11 +17,10 @@ typedef struct float4 u_color; } Uniforms_T; -vertex Fragment_T vsDebugRect(device const Vertex_T * vertices [[buffer(0)]], - ushort vid [[vertex_id]]) +vertex Fragment_T vsDebugRect(const Vertex_T in [[stage_in]]) { Fragment_T out; - out.position = float4(vertices[vid].a_position, 0.0, 1.0); + out.position = float4(in.a_position, 0.0, 1.0); return out; } diff --git a/shaders/Metal/gui.metal b/shaders/Metal/gui.metal index 4214718c0f..72820c8e22 100644 --- a/shaders/Metal/gui.metal +++ b/shaders/Metal/gui.metal @@ -17,73 +17,67 @@ typedef struct typedef struct { - packed_float2 a_position; - packed_float2 a_normal; - packed_float2 a_texCoords; + float2 a_position [[attribute(0)]]; + float2 a_normal [[attribute(1)]]; + float2 a_texCoords [[attribute(2)]]; } RulerVertex_T; typedef struct { float4 position [[position]]; - float2 texCoords; + half4 color; } RulerFragment_T; -vertex RulerFragment_T vsRuler(device const RulerVertex_T * vertices [[buffer(0)]], - constant Uniforms_T & uniforms [[buffer(1)]], - ushort vid [[vertex_id]]) +vertex RulerFragment_T vsRuler(const RulerVertex_T in [[stage_in]], + texture2d u_colorTex [[texture(0)]], + sampler u_colorTexSampler [[sampler(0)]], + constant Uniforms_T & uniforms [[buffer(1)]]) { - RulerVertex_T const in = vertices[vid]; RulerFragment_T out; float2 p = uniforms.u_position + in.a_position + uniforms.u_length * in.a_normal; out.position = float4(p, 0.0, 1.0) * uniforms.u_projection; - out.texCoords = in.a_texCoords; + half4 color = u_colorTex.sample(u_colorTexSampler, in.a_texCoords); + color.a *= uniforms.u_opacity; + out.color = color; return out; } -fragment float4 fsRuler(const RulerFragment_T in [[stage_in]], - constant Uniforms_T & uniforms [[buffer(0)]], - texture2d u_colorTex [[texture(0)]], - sampler u_colorTexSampler [[sampler(0)]]) +fragment half4 fsRuler(const RulerFragment_T in [[stage_in]]) { - float4 color = u_colorTex.sample(u_colorTexSampler, in.texCoords); - color.a *= uniforms.u_opacity; - return color; + return in.color; } // TextStaticOutlinedGui / TextOutlinedGui typedef struct { - packed_float3 a_position; - packed_float2 a_normal; - packed_float2 a_colorTexCoord; - packed_float2 a_outlineColorTexCoord; - packed_float2 a_maskTexCoord; + float3 a_position [[attribute(0)]]; + float2 a_normal [[attribute(1)]]; + float2 a_colorTexCoord [[attribute(2)]]; + float2 a_outlineColorTexCoord [[attribute(3)]]; + float2 a_maskTexCoord [[attribute(4)]]; } TextStaticOutlinedGuiVertex_T; typedef struct { - packed_float3 a_position; - packed_float2 a_colorTexCoord; - packed_float2 a_outlineColorTexCoord; -} TextOutlinedGuiVertex0_T; - -typedef struct -{ - packed_float2 a_normal; - packed_float2 a_maskTexCoord; -} TextOutlinedGuiVertex1_T; + float3 a_position [[attribute(0)]]; + float2 a_colorTexCoord [[attribute(1)]]; + float2 a_outlineColorTexCoord [[attribute(2)]]; + float2 a_normal [[attribute(3)]]; + float2 a_maskTexCoord [[attribute(4)]]; +} TextOutlinedGuiVertex_T; typedef struct { float4 position [[position]]; - float2 colorTexCoords; + half4 glyphColor; float2 maskTexCoord; } TextOutlinedGuiFragment_T; TextOutlinedGuiFragment_T ComputeTextOutlinedGuiVertex(constant Uniforms_T & uniforms, float3 a_position, float2 a_normal, float2 a_colorTexCoord, float2 a_outlineColorTexCoord, - float2 a_maskTexCoord) + float2 a_maskTexCoord, texture2d u_colorTex, + sampler u_colorTexSampler) { constexpr float kBaseDepthShift = -10.0; @@ -95,39 +89,38 @@ TextOutlinedGuiFragment_T ComputeTextOutlinedGuiVertex(constant Uniforms_T & uni float4 pos = (float4(a_position, 1.0) + float4(0.0, 0.0, depthShift, 0.0)) * uniforms.u_modelView; float4 shiftedPos = float4(a_normal, 0.0, 0.0) + pos; out.position = shiftedPos * uniforms.u_projection; - out.colorTexCoords = mix(a_colorTexCoord, a_outlineColorTexCoord, isOutline); + out.glyphColor = u_colorTex.sample(u_colorTexSampler, + mix(a_colorTexCoord, a_outlineColorTexCoord, isOutline)); out.maskTexCoord = a_maskTexCoord; return out; } -vertex TextOutlinedGuiFragment_T vsTextStaticOutlinedGui(device const TextStaticOutlinedGuiVertex_T * vertices [[buffer(0)]], +vertex TextOutlinedGuiFragment_T vsTextStaticOutlinedGui(const TextStaticOutlinedGuiVertex_T in [[stage_in]], constant Uniforms_T & uniforms [[buffer(1)]], - ushort vid [[vertex_id]]) + texture2d u_colorTex [[texture(0)]], + sampler u_colorTexSampler [[sampler(0)]]) { - TextStaticOutlinedGuiVertex_T const in = vertices[vid]; - return ComputeTextOutlinedGuiVertex(uniforms, in.a_position, in.a_normal, in.a_colorTexCoord, in.a_outlineColorTexCoord, - in.a_maskTexCoord); + return ComputeTextOutlinedGuiVertex(uniforms, in.a_position, in.a_normal, in.a_colorTexCoord, + in.a_outlineColorTexCoord, in.a_maskTexCoord, + u_colorTex, u_colorTexSampler); } -vertex TextOutlinedGuiFragment_T vsTextOutlinedGui(device const TextOutlinedGuiVertex0_T * vertices0 [[buffer(0)]], - device const TextOutlinedGuiVertex1_T * vertices1 [[buffer(1)]], +vertex TextOutlinedGuiFragment_T vsTextOutlinedGui(const TextOutlinedGuiVertex_T in [[stage_in]], constant Uniforms_T & uniforms [[buffer(2)]], - ushort vid [[vertex_id]]) + texture2d u_colorTex [[texture(0)]], + sampler u_colorTexSampler [[sampler(0)]]) { - TextOutlinedGuiVertex0_T const in0 = vertices0[vid]; - TextOutlinedGuiVertex1_T const in1 = vertices1[vid]; - return ComputeTextOutlinedGuiVertex(uniforms, in0.a_position, in1.a_normal, in0.a_colorTexCoord, - in0.a_outlineColorTexCoord, in1.a_maskTexCoord); + return ComputeTextOutlinedGuiVertex(uniforms, in.a_position, in.a_normal, in.a_colorTexCoord, + in.a_outlineColorTexCoord, in.a_maskTexCoord, + u_colorTex, u_colorTexSampler); } -fragment float4 fsTextOutlinedGui(const TextOutlinedGuiFragment_T in [[stage_in]], - constant Uniforms_T & uniforms [[buffer(0)]], - texture2d u_colorTex [[texture(0)]], - sampler u_colorTexSampler [[sampler(0)]], - texture2d u_maskTex [[texture(1)]], - sampler u_maskTexSampler [[sampler(1)]]) +fragment half4 fsTextOutlinedGui(const TextOutlinedGuiFragment_T in [[stage_in]], + constant Uniforms_T & uniforms [[buffer(0)]], + texture2d u_maskTex [[texture(0)]], + sampler u_maskTexSampler [[sampler(0)]]) { - float4 glyphColor = u_colorTex.sample(u_colorTexSampler, in.colorTexCoords); + half4 glyphColor = in.glyphColor; float dist = u_maskTex.sample(u_maskTexSampler, in.maskTexCoord).a; float2 contrastGamma = uniforms.u_contrastGamma; float alpha = smoothstep(contrastGamma.x - contrastGamma.y, contrastGamma.x + contrastGamma.y, dist); @@ -139,8 +132,8 @@ fragment float4 fsTextOutlinedGui(const TextOutlinedGuiFragment_T in [[stage_in] typedef struct { - packed_float2 a_position; - packed_float2 a_texCoords; + float2 a_position [[attribute(0)]]; + float2 a_texCoords [[attribute(1)]]; } TexturingGuiVertex_T; typedef struct @@ -149,11 +142,9 @@ typedef struct float2 texCoords; } TexturingGuiFragment_T; -vertex TexturingGuiFragment_T vsTexturingGui(device const TexturingGuiVertex_T * vertices [[buffer(0)]], - constant Uniforms_T & uniforms [[buffer(1)]], - ushort vid [[vertex_id]]) +vertex TexturingGuiFragment_T vsTexturingGui(const TexturingGuiVertex_T in [[stage_in]], + constant Uniforms_T & uniforms [[buffer(1)]]) { - TexturingGuiVertex_T const in = vertices[vid]; TexturingGuiFragment_T out; out.position = float4(in.a_position, 0.0, 1.0) * uniforms.u_modelView * uniforms.u_projection; out.texCoords = in.a_texCoords; diff --git a/shaders/Metal/screen_quad.metal b/shaders/Metal/screen_quad.metal index 05ae891cc0..c864387d6b 100644 --- a/shaders/Metal/screen_quad.metal +++ b/shaders/Metal/screen_quad.metal @@ -4,8 +4,8 @@ using namespace metal; typedef struct { - packed_float2 a_position; - packed_float2 a_texCoords; + float2 a_position [[attribute(0)]]; + float2 a_texCoords [[attribute(1)]]; } Vertex_T; typedef struct @@ -19,10 +19,8 @@ typedef struct float u_opacity; } Uniforms_T; -vertex Fragment_T vsScreenQuad(device const Vertex_T * vertices [[buffer(0)]], - ushort vid [[vertex_id]]) +vertex Fragment_T vsScreenQuad(const Vertex_T in [[stage_in]]) { - Vertex_T const in = vertices[vid]; Fragment_T out; out.position = float4(in.a_position, 0.0, 1.0); out.texCoords = in.a_texCoords; diff --git a/shaders/Metal/system.metal b/shaders/Metal/system.metal index cdbd7ef51b..90b16c7958 100644 --- a/shaders/Metal/system.metal +++ b/shaders/Metal/system.metal @@ -4,7 +4,7 @@ using namespace metal; typedef struct { - packed_float2 a_position; + float2 a_position [[attribute(0)]]; } Vertex_T; typedef struct @@ -28,11 +28,10 @@ typedef struct float depth [[depth(any)]]; } Fragment_DepthOutput; -vertex Fragment_T vsCleaner(device const Vertex_T * vertices [[buffer(0)]], - ushort vid [[vertex_id]]) +vertex Fragment_T vsCleaner(const Vertex_T in [[stage_in]]) { Fragment_T out; - out.position = float4(vertices[vid].a_position, 0.0, 1.0); + out.position = float4(in.a_position, 0.0, 1.0); return out; } @@ -58,4 +57,3 @@ fragment Fragment_Output fsClearColorAndDepth(const Fragment_T in [[stage_in]], out.depth = 1.0; return out; } - diff --git a/shaders/metal_program_pool.hpp b/shaders/metal_program_pool.hpp index a072c03712..8c9f3802c7 100644 --- a/shaders/metal_program_pool.hpp +++ b/shaders/metal_program_pool.hpp @@ -31,8 +31,10 @@ public: drape_ptr GetSystemProgram(SystemProgram program); private: - drape_ptr Get(std::string const & programName, std::string const & vertexShaderName, - std::string const & fragmentShaderName); + drape_ptr Get(std::string const & programName, + std::string const & vertexShaderName, + std::string const & fragmentShaderName, + std::map const & layout); id GetFunction(std::string const & name); id m_device; diff --git a/shaders/metal_program_pool.mm b/shaders/metal_program_pool.mm index 7377087677..70e585ae58 100644 --- a/shaders/metal_program_pool.mm +++ b/shaders/metal_program_pool.mm @@ -5,7 +5,9 @@ #include "base/assert.hpp" +#include #include +#include namespace gpu { @@ -15,84 +17,190 @@ namespace { struct ProgramInfo { - std::string m_vertexShaderName; - std::string m_fragmentShaderName; - ProgramInfo(std::string && vertexShaderName, std::string && fragmentShaderName) + using Layout = std::map; + std::string const m_vertexShaderName; + std::string const m_fragmentShaderName; + Layout m_layout; + + // Layout is in the format { buffer0, buffer1, ..., bufferN }. + // bufferX is a pair { start attribute index, end attribute index }. + ProgramInfo(std::string && vertexShaderName, std::string && fragmentShaderName, + std::vector> const & layout) : m_vertexShaderName(std::move(vertexShaderName)) , m_fragmentShaderName(std::move(fragmentShaderName)) - {} + { + for (size_t i = 0; i < layout.size(); i++) + { + for (size_t j = layout[i].first; j <= layout[i].second; j++) + { + CHECK(m_layout.find(j) == m_layout.end(), ("Duplicate index in the layout.")); + m_layout[j] = i; + } + } + } }; std::array(SystemProgram::SystemProgramsCount)> const kMetalSystemProgramsInfo = {{ - ProgramInfo("vsCleaner", "fsClearColor"), // ClearColor - ProgramInfo("vsCleaner", "fsClearDepth"), // ClearDepth - ProgramInfo("vsCleaner", "fsClearColorAndDepth"), // ClearColorAndDepth + ProgramInfo("vsCleaner", "fsClearColor", {{0, 0}}), // ClearColor + ProgramInfo("vsCleaner", "fsClearDepth", {{0, 0}}), // ClearDepth + ProgramInfo("vsCleaner", "fsClearColorAndDepth", {{0, 0}}), // ClearColorAndDepth }}; std::array(Program::ProgramsCount)> const kMetalProgramsInfo = {{ - ProgramInfo("", ""), // ColoredSymbol - ProgramInfo("", ""), // Texturing - ProgramInfo("", ""), // MaskedTexturing - ProgramInfo("", ""), // Bookmark - ProgramInfo("", ""), // BookmarkAnim - ProgramInfo("", ""), // TextOutlined - ProgramInfo("", ""), // Text - ProgramInfo("", ""), // TextFixed - ProgramInfo("vsTextStaticOutlinedGui", "fsTextOutlinedGui"), // TextStaticOutlinedGui - ProgramInfo("vsTextOutlinedGui", "fsTextOutlinedGui"), // TextOutlinedGui - ProgramInfo("", ""), // Area - ProgramInfo("", ""), // AreaOutline - ProgramInfo("", ""), // Area3d - ProgramInfo("", ""), // Area3dOutline - ProgramInfo("", ""), // Line - ProgramInfo("", ""), // CapJoin - ProgramInfo("", ""), // TransitCircle - ProgramInfo("", ""), // DashedLine - ProgramInfo("", ""), // PathSymbol - ProgramInfo("", ""), // HatchingArea - ProgramInfo("vsTexturingGui", "fsTexturingGui"), // TexturingGui - ProgramInfo("vsRuler", "fsRuler"), // Ruler - ProgramInfo("", ""), // Accuracy - ProgramInfo("", ""), // MyPosition - ProgramInfo("", ""), // Transit - ProgramInfo("", ""), // TransitMarker - ProgramInfo("", ""), // Route - ProgramInfo("", ""), // RouteDash - ProgramInfo("", ""), // RouteArrow - ProgramInfo("", ""), // RouteMarker - ProgramInfo("", ""), // CirclePoint - ProgramInfo("vsDebugRect", "fsDebugRect"), // DebugRect - ProgramInfo("vsScreenQuad", "fsScreenQuad"), // ScreenQuad - ProgramInfo("", ""), // Arrow3d - ProgramInfo("", ""), // Arrow3dShadow - ProgramInfo("", ""), // Arrow3dOutline - ProgramInfo("", ""), // ColoredSymbolBillboard - ProgramInfo("", ""), // TexturingBillboard - ProgramInfo("", ""), // MaskedTexturingBillboard - ProgramInfo("", ""), // BookmarkBillboard - ProgramInfo("", ""), // BookmarkAnimBillboard - ProgramInfo("", ""), // TextOutlinedBillboard - ProgramInfo("", ""), // TextBillboard - ProgramInfo("", ""), // TextFixedBillboard - ProgramInfo("", ""), // Traffic - ProgramInfo("", ""), // TrafficLine - ProgramInfo("", ""), // TrafficCircle - ProgramInfo("", ""), // SmaaEdges - ProgramInfo("", ""), // SmaaBlendingWeight - ProgramInfo("", ""), // SmaaFinal + ProgramInfo("", "", {}), // ColoredSymbol + ProgramInfo("", "", {}), // Texturing + ProgramInfo("", "", {}), // MaskedTexturing + ProgramInfo("", "", {}), // Bookmark + ProgramInfo("", "", {}), // BookmarkAnim + ProgramInfo("", "", {}), // TextOutlined + ProgramInfo("", "", {}), // Text + ProgramInfo("", "", {}), // TextFixed + ProgramInfo("vsTextStaticOutlinedGui", "fsTextOutlinedGui", {{0, 4}}), // TextStaticOutlinedGui + ProgramInfo("vsTextOutlinedGui", "fsTextOutlinedGui", {{0, 2}, {3, 4}}), // TextOutlinedGui + ProgramInfo("", "", {}), // Area + ProgramInfo("", "", {}), // AreaOutline + ProgramInfo("", "", {}), // Area3d + ProgramInfo("", "", {}), // Area3dOutline + ProgramInfo("", "", {}), // Line + ProgramInfo("", "", {}), // CapJoin + ProgramInfo("", "", {}), // TransitCircle + ProgramInfo("", "", {}), // DashedLine + ProgramInfo("", "", {}), // PathSymbol + ProgramInfo("", "", {}), // HatchingArea + ProgramInfo("vsTexturingGui", "fsTexturingGui", {{0, 1}}), // TexturingGui + ProgramInfo("vsRuler", "fsRuler", {{0, 2}}), // Ruler + ProgramInfo("", "", {}), // Accuracy + ProgramInfo("", "", {}), // MyPosition + ProgramInfo("", "", {}), // Transit + ProgramInfo("", "", {}), // TransitMarker + ProgramInfo("", "", {}), // Route + ProgramInfo("", "", {}), // RouteDash + ProgramInfo("", "", {}), // RouteArrow + ProgramInfo("", "", {}), // RouteMarker + ProgramInfo("", "", {}), // CirclePoint + ProgramInfo("vsDebugRect", "fsDebugRect", {{0, 0}}), // DebugRect + ProgramInfo("vsScreenQuad", "fsScreenQuad", {{0, 1}}), // ScreenQuad + ProgramInfo("", "", {}), // Arrow3d + ProgramInfo("", "", {}), // Arrow3dShadow + ProgramInfo("", "", {}), // Arrow3dOutline + ProgramInfo("", "", {}), // ColoredSymbolBillboard + ProgramInfo("", "", {}), // TexturingBillboard + ProgramInfo("", "", {}), // MaskedTexturingBillboard + ProgramInfo("", "", {}), // BookmarkBillboard + ProgramInfo("", "", {}), // BookmarkAnimBillboard + ProgramInfo("", "", {}), // TextOutlinedBillboard + ProgramInfo("", "", {}), // TextBillboard + ProgramInfo("", "", {}), // TextFixedBillboard + ProgramInfo("", "", {}), // Traffic + ProgramInfo("", "", {}), // TrafficLine + ProgramInfo("", "", {}), // TrafficCircle + ProgramInfo("", "", {}), // SmaaEdges + ProgramInfo("", "", {}), // SmaaBlendingWeight + ProgramInfo("", "", {}), // SmaaFinal }}; + +MTLVertexFormat GetFormatByDataType(MTLDataType dataType) +{ + switch (dataType) + { + case MTLDataTypeFloat: return MTLVertexFormatFloat; + case MTLDataTypeFloat2: return MTLVertexFormatFloat2; + case MTLDataTypeFloat3: return MTLVertexFormatFloat3; + case MTLDataTypeFloat4: return MTLVertexFormatFloat4; + } + CHECK(false, ("Unsupported vertex format.")); + return MTLVertexFormatInvalid; +} + +uint32_t GetSizeByDataType(MTLDataType dataType) +{ + switch (dataType) + { + case MTLDataTypeFloat: return sizeof(float); + case MTLDataTypeFloat2: return 2 * sizeof(float); + case MTLDataTypeFloat3: return 3 * sizeof(float); + case MTLDataTypeFloat4: return 4 * sizeof(float); + } + CHECK(false, ("Unsupported vertex format.")); + return 0; +} + +void GetBindings(NSArray * arguments, int8_t & uniformsBindingIndex, + dp::metal::MetalGpuProgram::TexturesBindingInfo & textureBindingInfo) +{ + // Uniforms buffer must have the name "uniforms". + NSString * kUniformsName = @"uniforms"; + // Sampler name must be constructed as concatenation of texture name and kSamplerSuffix. + static std::string const kSamplerSuffix = "Sampler"; + + uniformsBindingIndex = dp::metal::MetalGpuProgram::kInvalidBindingIndex; + + for (MTLArgument * arg in arguments) + { + if ([arg.name compare:kUniformsName] == NSOrderedSame && arg.active && arg.type == MTLArgumentTypeBuffer) + { + uniformsBindingIndex = static_cast(arg.index); + } + else if (arg.type == MTLArgumentTypeTexture) + { + std::string const name([arg.name UTF8String]); + textureBindingInfo[name].m_textureBindingIndex = static_cast(arg.index); + } + else if (arg.type == MTLArgumentTypeSampler) + { + std::string const name([arg.name UTF8String]); + auto const pos = name.find(kSamplerSuffix); + if (pos == std::string::npos) + continue; + std::string const textureName = name.substr(0, pos); + textureBindingInfo[textureName].m_samplerBindingIndex = static_cast(arg.index); + } + } +} + +MTLVertexDescriptor * GetVertexDescriptor(id vertexShader, ProgramInfo::Layout const & layout) +{ + MTLVertexDescriptor * vertexDesc = [[MTLVertexDescriptor alloc] init]; + uint32_t offset = 0; + uint32_t lastBufferIndex = 0; + std::map sizes; + for (MTLVertexAttribute * attr in vertexShader.vertexAttributes) + { + auto const attrIndex = static_cast(attr.attributeIndex); + auto const it = layout.find(attrIndex); + CHECK(it != layout.cend(), ("Invalid layout.")); + auto const bufferIndex = it->second; + if (lastBufferIndex != bufferIndex) + { + offset = 0; + lastBufferIndex = bufferIndex; + } + MTLVertexAttributeDescriptor * attrDesc = vertexDesc.attributes[attr.attributeIndex]; + attrDesc.format = GetFormatByDataType(attr.attributeType); + auto const sz = GetSizeByDataType(attr.attributeType); + attrDesc.offset = offset; + offset += sz; + sizes[bufferIndex] += sz; + attrDesc.bufferIndex = bufferIndex; + } + + for (auto const & s : sizes) + vertexDesc.layouts[s.first].stride = s.second; + + return vertexDesc; +} } // namespace std::string DebugPrint(SystemProgram p) { switch (p) { - case SystemProgram::ClearColor: return "ClearColor"; - case SystemProgram::ClearDepth: return "ClearDepth"; - case SystemProgram::ClearColorAndDepth: return "ClearColorAndDepth"; - - case SystemProgram::SystemProgramsCount: - CHECK(false, ("Try to output SystemProgramsCount")); + case SystemProgram::ClearColor: return "ClearColor"; + case SystemProgram::ClearDepth: return "ClearDepth"; + case SystemProgram::ClearColorAndDepth: return "ClearColorAndDepth"; + + case SystemProgram::SystemProgramsCount: + CHECK(false, ("Try to output SystemProgramsCount")); } CHECK(false, ("Unknown program")); return {}; @@ -122,28 +230,32 @@ MetalProgramPool::~MetalProgramPool() drape_ptr MetalProgramPool::GetSystemProgram(SystemProgram program) { auto const & info = kMetalSystemProgramsInfo[static_cast(program)]; - return Get(DebugPrint(program), info.m_vertexShaderName, info.m_fragmentShaderName); + return Get(DebugPrint(program), info.m_vertexShaderName, info.m_fragmentShaderName, info.m_layout); } drape_ptr MetalProgramPool::Get(Program program) { auto const & info = kMetalProgramsInfo[static_cast(program)]; - return Get(DebugPrint(program), info.m_vertexShaderName, info.m_fragmentShaderName); + return Get(DebugPrint(program), info.m_vertexShaderName, info.m_fragmentShaderName, info.m_layout); } drape_ptr MetalProgramPool::Get(std::string const & programName, std::string const & vertexShaderName, - std::string const & fragmentShaderName) + std::string const & fragmentShaderName, + std::map const & layout) { CHECK(!vertexShaderName.empty(), ()); CHECK(!fragmentShaderName.empty(), ()); id vertexShader = GetFunction(vertexShaderName); id fragmentShader = GetFunction(fragmentShaderName); - + MTLVertexDescriptor * vertexDesc = GetVertexDescriptor(vertexShader, layout); + + // Reflect functions. MTLRenderPipelineDescriptor * desc = [[MTLRenderPipelineDescriptor alloc] init]; desc.colorAttachments[0].pixelFormat = MTLPixelFormatRGBA8Unorm; desc.depthAttachmentPixelFormat = MTLPixelFormatDepth32Float; + desc.vertexDescriptor = vertexDesc; desc.vertexFunction = vertexShader; desc.fragmentFunction = fragmentShader; @@ -160,48 +272,19 @@ drape_ptr MetalProgramPool::Get(std::string const & programName, CHECK(false, ("Failed to create reflection pipeline state.")); } - // Uniforms buffer must have the name "uniforms". - NSString * kUniformsName = @"uniforms"; - int8_t vsUniformsBindingIndex = dp::metal::MetalGpuProgram::kInvalidBindingIndex; - for (MTLArgument * arg in reflectionObj.vertexArguments) - { - if ([arg.name compare:kUniformsName] == NSOrderedSame && arg.active && arg.type == MTLArgumentTypeBuffer) - { - vsUniformsBindingIndex = static_cast(arg.index); - break; - } - } + dp::metal::MetalGpuProgram::TexturesBindingInfo vsTextureBindingInfo; + GetBindings(reflectionObj.vertexArguments, vsUniformsBindingIndex, vsTextureBindingInfo); int8_t fsUniformsBindingIndex = dp::metal::MetalGpuProgram::kInvalidBindingIndex; - dp::metal::MetalGpuProgram::TexturesBindingInfo textureBindingInfo; - for (MTLArgument * arg in reflectionObj.fragmentArguments) - { - if ([arg.name compare:kUniformsName] == NSOrderedSame && arg.active && arg.type == MTLArgumentTypeBuffer) - { - fsUniformsBindingIndex = static_cast(arg.index); - } - else if (arg.type == MTLArgumentTypeTexture) - { - std::string const name([arg.name UTF8String]); - textureBindingInfo[name].m_textureBindingIndex = static_cast(arg.index); - } - else if (arg.type == MTLArgumentTypeSampler) - { - std::string const name([arg.name UTF8String]); - // Sampler name must be constructed as concatenation of texture name and kSamplerSuffix. - static std::string const kSamplerSuffix = "Sampler"; - auto const pos = name.find(kSamplerSuffix); - if (pos == std::string::npos) - continue; - std::string const textureName = name.substr(0, pos); - textureBindingInfo[textureName].m_samplerBindingIndex = static_cast(arg.index); - } - } + dp::metal::MetalGpuProgram::TexturesBindingInfo fsTextureBindingInfo; + GetBindings(reflectionObj.fragmentArguments, fsUniformsBindingIndex, fsTextureBindingInfo); return make_unique_dp(programName, vertexShader, fragmentShader, vsUniformsBindingIndex, fsUniformsBindingIndex, - std::move(textureBindingInfo)); + std::move(vsTextureBindingInfo), + std::move(fsTextureBindingInfo), + vertexDesc); } id MetalProgramPool::GetFunction(std::string const & name)