Skip to content

Commit 7e79669

Browse files
[gfx] Metal improvements (shader-slang#4337)
* fix binding resources for render pass * compute vertex buffer binding offset based on root layout * fix result code * cleanup * implement readTextureResource * more getNativeHandle --------- Co-authored-by: Yong He <yonghe@outlook.com>
1 parent 6909d65 commit 7e79669

6 files changed

+65
-36
lines changed

tools/gfx/metal/metal-command-encoder.cpp

+5-3
Original file line numberDiff line numberDiff line change
@@ -360,8 +360,8 @@ Result RenderCommandEncoder::prepareDraw(MTL::RenderCommandEncoder*& encoder)
360360

361361
encoder = m_commandBuffer->getMetalRenderCommandEncoder(m_renderPassDesc.get());
362362
encoder->setRenderPipelineState(pipeline->m_renderPipelineState.get());
363-
// TODO only vertex stage bindings are set for now
364-
VertexBindingContext bindingContext;
363+
364+
RenderBindingContext bindingContext;
365365
bindingContext.init(m_commandBuffer->m_device, encoder);
366366
auto program = static_cast<ShaderProgramImpl*>(m_currentPipeline->m_program.get());
367367
m_commandBuffer->m_rootObject.bindAsRoot(&bindingContext, program->m_rootObjectLayout);
@@ -406,7 +406,7 @@ Result RenderCommandEncoder::drawIndexed(
406406
SLANG_RETURN_ON_FAIL(prepareDraw(encoder));
407407
// TODO baseVertex is not supported by Metal
408408
encoder->drawIndexedPrimitives(m_primitiveType, indexCount, m_indexBufferType, m_indexBuffer, m_indexBufferOffset);
409-
return SLANG_E_NOT_IMPLEMENTED;
409+
return SLANG_OK;
410410
}
411411

412412
Result RenderCommandEncoder::drawIndirect(
@@ -483,10 +483,12 @@ Result ComputeCommandEncoder::dispatchCompute(int x, int y, int z)
483483

484484
MTL::ComputeCommandEncoder* encoder = m_commandBuffer->getMetalComputeCommandEncoder();
485485
encoder->setComputePipelineState(pipeline->m_computePipelineState.get());
486+
486487
ComputeBindingContext bindingContext;
487488
bindingContext.init(m_commandBuffer->m_device, encoder);
488489
auto program = static_cast<ShaderProgramImpl*>(m_currentPipeline->m_program.get());
489490
m_commandBuffer->m_rootObject.bindAsRoot(&bindingContext, program->m_rootObjectLayout);
491+
490492
encoder->dispatchThreadgroups(MTL::Size(x, y, z), pipeline->m_threadGroupSize);
491493

492494
return SLANG_OK;

tools/gfx/metal/metal-device.cpp

+46
Original file line numberDiff line numberDiff line change
@@ -189,7 +189,53 @@ SlangResult DeviceImpl::readTextureResource(
189189
{
190190
AUTORELEASEPOOL
191191

192+
TextureResourceImpl* textureImpl = static_cast<TextureResourceImpl*>(texture);
193+
194+
if (textureImpl->getDesc()->sampleDesc.numSamples > 1)
195+
{
192196
return SLANG_E_NOT_IMPLEMENTED;
197+
}
198+
199+
NS::SharedPtr<MTL::Texture> srcTexture = textureImpl->m_texture;
200+
201+
const ITextureResource::Desc& desc = *textureImpl->getDesc();
202+
Count width = Math::Max(desc.size.width, 1);
203+
Count height = Math::Max(desc.size.height, 1);
204+
Count depth = Math::Max(desc.size.depth, 1);
205+
FormatInfo formatInfo;
206+
gfxGetFormatInfo(desc.format, &formatInfo);
207+
Size bytesPerPixel = formatInfo.blockSizeInBytes / formatInfo.pixelsPerBlock;
208+
Size bytesPerRow = Size(width) * bytesPerPixel;
209+
Size bytesPerSlice = Size(height) * bytesPerRow;
210+
Size bufferSize = Size(depth) * bytesPerSlice;
211+
if (outRowPitch)
212+
*outRowPitch = bytesPerRow;
213+
if (outPixelSize)
214+
*outPixelSize = bytesPerPixel;
215+
216+
// create staging buffer
217+
NS::SharedPtr<MTL::Buffer> stagingBuffer = NS::TransferPtr(m_device->newBuffer(bufferSize, MTL::StorageModeShared));
218+
if (!stagingBuffer)
219+
{
220+
return SLANG_FAIL;
221+
}
222+
223+
MTL::CommandBuffer* commandBuffer = m_commandQueue->commandBuffer();
224+
MTL::BlitCommandEncoder* encoder = commandBuffer->blitCommandEncoder();
225+
encoder->copyFromTexture(
226+
srcTexture.get(), 0, 0, MTL::Origin(0, 0, 0), MTL::Size(width, height, depth),
227+
stagingBuffer.get(), 0, bytesPerRow, bytesPerSlice);
228+
encoder->endEncoding();
229+
commandBuffer->commit();
230+
commandBuffer->waitUntilCompleted();
231+
232+
List<uint8_t> blobData;
233+
blobData.setCount(bufferSize);
234+
::memcpy(blobData.getBuffer(), stagingBuffer->contents(), bufferSize);
235+
auto blob = ListBlob::moveCreate(blobData);
236+
237+
returnComPtr(outBlob, blob);
238+
return SLANG_OK;
193239
}
194240

195241
SlangResult DeviceImpl::readBufferResource(

tools/gfx/metal/metal-fence.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
// metal-gfence.cpp
1+
// metal-fence.cpp
22
#include "metal-fence.h"
33
#include "metal-device.h"
44

tools/gfx/metal/metal-helper-functions.h

+3-27
Original file line numberDiff line numberDiff line change
@@ -85,7 +85,7 @@ struct ComputeBindingContext : public BindingContext
8585
}
8686
};
8787

88-
struct VertexBindingContext : public BindingContext
88+
struct RenderBindingContext : public BindingContext
8989
{
9090
MTL::RenderCommandEncoder* encoder;
9191

@@ -99,42 +99,18 @@ struct VertexBindingContext : public BindingContext
9999
void setBuffer(MTL::Buffer* buffer, NS::UInteger index) override
100100
{
101101
encoder->setVertexBuffer(buffer, 0, index);
102-
}
103-
104-
void setTexture(MTL::Texture* texture, NS::UInteger index) override
105-
{
106-
encoder->setVertexTexture(texture, index);
107-
}
108-
109-
void setSampler(MTL::SamplerState* sampler, NS::UInteger index) override
110-
{
111-
encoder->setVertexSamplerState(sampler, index);
112-
}
113-
};
114-
115-
struct FragmentBindingContext : public BindingContext
116-
{
117-
MTL::RenderCommandEncoder* encoder;
118-
119-
Result init(DeviceImpl* device, MTL::RenderCommandEncoder* encoder)
120-
{
121-
this->device = device;
122-
this->encoder = encoder;
123-
return SLANG_OK;
124-
}
125-
126-
void setBuffer(MTL::Buffer* buffer, NS::UInteger index) override
127-
{
128102
encoder->setFragmentBuffer(buffer, 0, index);
129103
}
130104

131105
void setTexture(MTL::Texture* texture, NS::UInteger index) override
132106
{
107+
encoder->setVertexTexture(texture, index);
133108
encoder->setFragmentTexture(texture, index);
134109
}
135110

136111
void setSampler(MTL::SamplerState* sampler, NS::UInteger index) override
137112
{
113+
encoder->setVertexSamplerState(sampler, index);
138114
encoder->setFragmentSamplerState(sampler, index);
139115
}
140116
};

tools/gfx/metal/metal-pipeline-state.cpp

+4-3
Original file line numberDiff line numberDiff line change
@@ -76,9 +76,10 @@ Result PipelineStateImpl::createMetalRenderPipelineState()
7676
}
7777
}
7878

79-
// Offset vertex buffer indices in vertex layout.
80-
// They need to be in a range not used by other buffer bindings.
81-
m_vertexBufferOffset = 10; // TODO get from layout
79+
// Create a vertex descriptor with the vertex buffer binding indices being offset.
80+
// They need to be in a range not used by any buffers in the root object layout.
81+
// The +1 is to account for a potential constant buffer at index 0.
82+
m_vertexBufferOffset = programImpl->m_rootObjectLayout->getBufferCount() + 1;
8283
auto inputLayoutImpl = static_cast<InputLayoutImpl*>(desc.graphics.inputLayout);
8384
NS::SharedPtr<MTL::VertexDescriptor> vertexDescriptor = inputLayoutImpl->createVertexDescriptor(m_vertexBufferOffset);
8485
pd->setVertexDescriptor(vertexDescriptor.get());

tools/gfx/metal/metal-resource-views.cpp

+6-2
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,9 @@ TextureResourceViewImpl::~TextureResourceViewImpl()
1515

1616
Result TextureResourceViewImpl::getNativeHandle(InteropHandle* outHandle)
1717
{
18-
return SLANG_E_NOT_IMPLEMENTED;
18+
outHandle->api = InteropHandleAPI::Metal;
19+
outHandle->handleValue = reinterpret_cast<uintptr_t>(m_textureView.get());
20+
return SLANG_OK;
1921
}
2022

2123
BufferResourceViewImpl::~BufferResourceViewImpl()
@@ -24,7 +26,9 @@ BufferResourceViewImpl::~BufferResourceViewImpl()
2426

2527
Result BufferResourceViewImpl::getNativeHandle(InteropHandle* outHandle)
2628
{
27-
return SLANG_E_NOT_IMPLEMENTED;
29+
outHandle->api = InteropHandleAPI::Metal;
30+
outHandle->handleValue = reinterpret_cast<uintptr_t>(m_buffer->m_buffer.get());
31+
return SLANG_OK;
2832
}
2933

3034
TexelBufferResourceViewImpl::TexelBufferResourceViewImpl(DeviceImpl* device)

0 commit comments

Comments
 (0)