mirror of https://github.com/axmolengine/axmol.git
187 lines
7.6 KiB
Plaintext
187 lines
7.6 KiB
Plaintext
#include "RenderTargetMTL.h"
|
|
#include "UtilsMTL.h"
|
|
|
|
NS_AX_BACKEND_BEGIN
|
|
|
|
static MTLLoadAction getLoadAction(const RenderPassDescriptor& params, TargetBufferFlags buffer)
|
|
{
|
|
const auto clearFlags = (TargetBufferFlags)params.flags.clear;
|
|
const auto discardStartFlags = params.flags.discardStart;
|
|
if (bitmask::any(clearFlags, buffer))
|
|
{
|
|
return MTLLoadActionClear;
|
|
}
|
|
else if (bitmask::any(discardStartFlags, buffer))
|
|
{
|
|
return MTLLoadActionDontCare;
|
|
}
|
|
return MTLLoadActionLoad;
|
|
}
|
|
|
|
static MTLStoreAction getStoreAction(const RenderPassDescriptor& params, TargetBufferFlags buffer)
|
|
{
|
|
const auto discardEndFlags = params.flags.discardEnd;
|
|
if (bitmask::any(discardEndFlags, buffer))
|
|
{
|
|
return MTLStoreActionDontCare;
|
|
}
|
|
return MTLStoreActionStore;
|
|
}
|
|
|
|
RenderTargetMTL::RenderTargetMTL(bool defaultRenderTarget) : RenderTarget(defaultRenderTarget) {}
|
|
RenderTargetMTL::~RenderTargetMTL() {}
|
|
|
|
void RenderTargetMTL::applyRenderPassAttachments(const RenderPassDescriptor& params,
|
|
MTLRenderPassDescriptor* descriptor) const
|
|
{
|
|
// const auto discardFlags = params.flags.discardEnd;
|
|
auto clearFlags = params.flags.clear;
|
|
|
|
for (size_t i = 0; i < MAX_COLOR_ATTCHMENT; i++)
|
|
{
|
|
auto attachment = getColorAttachment(static_cast<int>(i));
|
|
if (!attachment)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
const auto MRTColorFlag = getMRTColorFlag(i);
|
|
|
|
descriptor.colorAttachments[i].texture = attachment.texture;
|
|
descriptor.colorAttachments[i].level = attachment.level;
|
|
// descriptor.colorAttachments[i].slice = attachment.layer;
|
|
descriptor.colorAttachments[i].loadAction = getLoadAction(params, MRTColorFlag);
|
|
descriptor.colorAttachments[i].storeAction = getStoreAction(params, MRTColorFlag);
|
|
if (bitmask::any(clearFlags, MRTColorFlag))
|
|
descriptor.colorAttachments[i].clearColor =
|
|
MTLClearColorMake(params.clearColorValue[0], params.clearColorValue[1], params.clearColorValue[2],
|
|
params.clearColorValue[3]);
|
|
#if 0
|
|
if (multisampledColor[i]) {
|
|
// We're rendering into our temporary MSAA texture and doing an automatic resolve.
|
|
// We should not be attempting to load anything into the MSAA texture.
|
|
assert(descriptor.colorAttachments[i].loadAction != MTLLoadActionLoad);
|
|
|
|
descriptor.colorAttachments[i].texture = multisampledColor[i];
|
|
descriptor.colorAttachments[i].level = 0;
|
|
descriptor.colorAttachments[i].slice = 0;
|
|
const bool discard = any(discardFlags & getMRTColorFlag(i));
|
|
if (!discard) {
|
|
descriptor.colorAttachments[i].resolveTexture = attachment.texture;
|
|
descriptor.colorAttachments[i].resolveLevel = attachment.level;
|
|
descriptor.colorAttachments[i].resolveSlice = attachment.layer;
|
|
descriptor.colorAttachments[i].storeAction = MTLStoreActionMultisampleResolve;
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
// Sets descriptor depth and stencil params, should match RenderTargetMTL::chooseAttachmentFormat
|
|
if (bitmask::any(this->_flags, RenderTargetFlag::DEPTH_AND_STENCIL))
|
|
{
|
|
auto depthAttachment = getDepthAttachment();
|
|
if (depthAttachment)
|
|
{
|
|
descriptor.depthAttachment.texture = depthAttachment.texture;
|
|
descriptor.depthAttachment.level = depthAttachment.level;
|
|
// descriptor.depthAttachment.slice = depthAttachment.layer;
|
|
descriptor.depthAttachment.loadAction = getLoadAction(params, TargetBufferFlags::DEPTH);
|
|
descriptor.depthAttachment.storeAction = getStoreAction(params, TargetBufferFlags::DEPTH);
|
|
if (bitmask::any(clearFlags, TargetBufferFlags::DEPTH))
|
|
descriptor.depthAttachment.clearDepth = params.clearDepthValue;
|
|
}
|
|
|
|
auto stencilAttachment = getStencilAttachment();
|
|
if (stencilAttachment)
|
|
{
|
|
descriptor.stencilAttachment.texture = stencilAttachment.texture;
|
|
descriptor.stencilAttachment.level = stencilAttachment.level;
|
|
// descriptor.stencilAttachment.slice = depthAttachment.layer;
|
|
descriptor.stencilAttachment.loadAction = getLoadAction(params, TargetBufferFlags::STENCIL);
|
|
descriptor.stencilAttachment.storeAction = getStoreAction(params, TargetBufferFlags::STENCIL);
|
|
if (bitmask::any(clearFlags, TargetBufferFlags::STENCIL))
|
|
descriptor.stencilAttachment.clearStencil = params.clearStencilValue;
|
|
}
|
|
}
|
|
|
|
#if 0
|
|
if (multisampledDepth) {
|
|
// We're rendering into our temporary MSAA texture and doing an automatic resolve.
|
|
// We should not be attempting to load anything into the MSAA texture.
|
|
assert(descriptor.depthAttachment.loadAction != MTLLoadActionLoad);
|
|
|
|
descriptor.depthAttachment.texture = multisampledDepth;
|
|
descriptor.depthAttachment.level = 0;
|
|
descriptor.depthAttachment.slice = 0;
|
|
const bool discard = any(discardFlags & TargetBufferFlags::DEPTH);
|
|
if (!discard) {
|
|
descriptor.depthAttachment.resolveTexture = depthAttachment.texture;
|
|
descriptor.depthAttachment.resolveLevel = depthAttachment.level;
|
|
descriptor.depthAttachment.resolveSlice = depthAttachment.layer;
|
|
descriptor.depthAttachment.storeAction = MTLStoreActionMultisampleResolve;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
_dirty = false;
|
|
}
|
|
|
|
RenderTargetMTL::Attachment RenderTargetMTL::getColorAttachment(int index) const
|
|
{
|
|
if (isDefaultRenderTarget() && index == 0)
|
|
return {DriverMTL::getCurrentDrawable().texture, 0};
|
|
auto& rb = this->_color[index];
|
|
return RenderTargetMTL::Attachment{static_cast<bool>(rb) ? (id<MTLTexture>)(rb.texture->getHandler()) : nil,
|
|
rb.level};
|
|
}
|
|
|
|
RenderTargetMTL::Attachment RenderTargetMTL::getDepthAttachment() const
|
|
{
|
|
if (isDefaultRenderTarget())
|
|
return {UtilsMTL::getDefaultDepthStencilTexture(), 0};
|
|
auto& rb = this->_depth;
|
|
return RenderTargetMTL::Attachment{!!rb ? (id<MTLTexture>)(rb.texture->getHandler()) : nil, rb.level};
|
|
}
|
|
|
|
RenderTargetMTL::Attachment RenderTargetMTL::getStencilAttachment() const
|
|
{
|
|
if (isDefaultRenderTarget())
|
|
return RenderTargetMTL::Attachment{UtilsMTL::getDefaultDepthStencilTexture(), 0};
|
|
auto& rb = this->_stencil;
|
|
return RenderTargetMTL::Attachment{!!rb ? (id<MTLTexture>)(rb.texture->getHandler()) : nil, rb.level};
|
|
}
|
|
|
|
PixelFormat RenderTargetMTL::getColorAttachmentPixelFormat(int index) const
|
|
{
|
|
// !!!important
|
|
// the default framebuffer pixel format is: MTLPixelFormatBGRA8Unorm
|
|
if (isDefaultRenderTarget() && index == 0)
|
|
return PixelFormat::BGRA8;
|
|
auto& rb = this->_color[index];
|
|
return rb ? rb.texture->getTextureFormat() : PixelFormat::NONE;
|
|
}
|
|
|
|
PixelFormat RenderTargetMTL::getDepthAttachmentPixelFormat() const
|
|
{ // FIXME: axmol only support D24S8
|
|
if (bitmask::any(_flags, TargetBufferFlags::DEPTH_AND_STENCIL))
|
|
{
|
|
if (isDefaultRenderTarget() || !_depth)
|
|
return PixelFormat::D24S8;
|
|
return _depth.texture->getTextureFormat();
|
|
}
|
|
return PixelFormat::NONE;
|
|
}
|
|
|
|
PixelFormat RenderTargetMTL::getStencilAttachmentPixelFormat() const
|
|
{ // FIXME: axmol only support D24S8
|
|
if (bitmask::any(_flags, TargetBufferFlags::DEPTH_AND_STENCIL))
|
|
{
|
|
if (isDefaultRenderTarget() || !_stencil)
|
|
return PixelFormat::D24S8;
|
|
return _stencil.texture->getTextureFormat();
|
|
}
|
|
return PixelFormat::NONE;
|
|
}
|
|
|
|
NS_AX_BACKEND_END
|