Cocoa Port: For Metal display views, in -renderDrawable, separate buffer updates from encoding commands in an effort to try to reduce the hold time on the vended drawable as much as possible.

This commit is contained in:
rogerman 2017-09-12 00:50:57 -07:00
parent d0f16193c5
commit c94f353f2a
1 changed files with 107 additions and 96 deletions

View File

@ -1451,39 +1451,12 @@
- (void) renderToDrawable - (void) renderToDrawable
{ {
NSAutoreleasePool *renderAutoreleasePool = [[NSAutoreleasePool alloc] init];
dispatch_semaphore_wait(availableResources, DISPATCH_TIME_FOREVER);
id<CAMetalDrawable> drawable = [self nextDrawable];
if (drawable == nil)
{
puts("MacMetalDisplayView: No drawable object was available!\n");
dispatch_semaphore_signal(availableResources);
[renderAutoreleasePool release];
return;
}
id<MTLTexture> texture = [drawable texture];
if (texture == nil)
{
dispatch_semaphore_signal(availableResources);
[renderAutoreleasePool release];
return;
}
const NDSDisplayInfo &displayInfo = _cdv->GetEmuDisplayInfo(); const NDSDisplayInfo &displayInfo = _cdv->GetEmuDisplayInfo();
[[self colorAttachment0Desc] setTexture:texture];
id<MTLCommandBuffer> cb = [[sharedData commandQueue] commandBufferWithUnretainedReferences];
id<MTLRenderCommandEncoder> ce = [cb renderCommandEncoderWithDescriptor:_outputRenderPassDesc];
// Set up the view properties. // Set up the view properties.
BOOL didChangeViewProperties = NO; BOOL didChangeViewProperties = NO;
BOOL needEncodeViewport = NO;
if ([self needsViewportUpdate])
{
MTLViewport newViewport; MTLViewport newViewport;
newViewport.originX = 0.0; newViewport.originX = 0.0;
newViewport.originY = 0.0; newViewport.originY = 0.0;
@ -1491,7 +1464,10 @@
newViewport.height = _cdv->GetViewProperties().clientHeight; newViewport.height = _cdv->GetViewProperties().clientHeight;
newViewport.znear = 0.0; newViewport.znear = 0.0;
newViewport.zfar = 1.0; newViewport.zfar = 1.0;
[ce setViewport:newViewport];
if ([self needsViewportUpdate])
{
needEncodeViewport = YES;
DisplayViewShaderProperties *viewProps = (DisplayViewShaderProperties *)[_cdvPropertiesBuffer contents]; DisplayViewShaderProperties *viewProps = (DisplayViewShaderProperties *)[_cdvPropertiesBuffer contents];
viewProps->width = _cdv->GetViewProperties().clientWidth; viewProps->width = _cdv->GetViewProperties().clientWidth;
@ -1517,10 +1493,9 @@
[_cdvPropertiesBuffer didModifyRange:NSMakeRange(0, sizeof(DisplayViewShaderProperties))]; [_cdvPropertiesBuffer didModifyRange:NSMakeRange(0, sizeof(DisplayViewShaderProperties))];
} }
// Draw the NDS displays. // Set up the display properties.
if (displayInfo.pixelBytes != 0) BOOL willDrawDisplays = (displayInfo.pixelBytes != 0);
{ if (willDrawDisplays && [self needsScreenVerticesUpdate])
if ([self needsScreenVerticesUpdate])
{ {
_cdv->SetScreenVertices((float *)[_displayVtxPositionBuffer contents]); _cdv->SetScreenVertices((float *)[_displayVtxPositionBuffer contents]);
[_displayVtxPositionBuffer didModifyRange:NSMakeRange(0, sizeof(float) * (4 * 8))]; [_displayVtxPositionBuffer didModifyRange:NSMakeRange(0, sizeof(float) * (4 * 8))];
@ -1528,6 +1503,92 @@
[self setNeedsScreenVerticesUpdate:NO]; [self setNeedsScreenVerticesUpdate:NO];
} }
// Set up the HUD properties.
size_t hudLength = _cdv->GetHUDString().length();
size_t hudTouchLineLength = 0;
BOOL willDrawHUD = _cdv->GetHUDVisibility() && ([self texHUDCharMap] != nil);
if (_cdv->GetHUDShowInput())
{
hudLength += HUD_INPUT_ELEMENT_LENGTH;
switch (_cdv->GetMode())
{
case ClientDisplayMode_Main:
case ClientDisplayMode_Touch:
hudTouchLineLength = HUD_INPUT_TOUCH_LINE_ELEMENTS / 2;
break;
case ClientDisplayMode_Dual:
{
switch (_cdv->GetLayout())
{
case ClientDisplayLayout_Vertical:
case ClientDisplayLayout_Horizontal:
hudTouchLineLength = HUD_INPUT_TOUCH_LINE_ELEMENTS / 2;
break;
case ClientDisplayLayout_Hybrid_2_1:
case ClientDisplayLayout_Hybrid_16_9:
case ClientDisplayLayout_Hybrid_16_10:
hudTouchLineLength = HUD_INPUT_TOUCH_LINE_ELEMENTS;
break;
}
break;
}
}
hudLength += hudTouchLineLength;
}
willDrawHUD = willDrawHUD && (hudLength > 1);
if (willDrawHUD && _cdv->HUDNeedsUpdate())
{
_cdv->SetHUDPositionVertices((float)_cdv->GetViewProperties().clientWidth, (float)_cdv->GetViewProperties().clientHeight, (float *)[_hudVtxPositionBuffer contents]);
[_hudVtxPositionBuffer didModifyRange:NSMakeRange(0, sizeof(float) * hudLength * 8)];
_cdv->SetHUDColorVertices((uint32_t *)[_hudVtxColorBuffer contents]);
[_hudVtxColorBuffer didModifyRange:NSMakeRange(0, sizeof(uint32_t) * hudLength * 4)];
_cdv->SetHUDTextureCoordinates((float *)[_hudTexCoordBuffer contents]);
[_hudTexCoordBuffer didModifyRange:NSMakeRange(0, sizeof(float) * hudLength * 8)];
_cdv->ClearHUDNeedsUpdate();
}
// Now that everything is set up, request a layer drawable and draw everything.
dispatch_semaphore_wait(availableResources, DISPATCH_TIME_FOREVER);
id<CAMetalDrawable> drawable = [self nextDrawable];
if (drawable == nil)
{
puts("MacMetalDisplayView: No drawable object was available!\n");
dispatch_semaphore_signal(availableResources);
return;
}
id<MTLTexture> texture = [drawable texture];
if (texture == nil)
{
dispatch_semaphore_signal(availableResources);
return;
}
[[self colorAttachment0Desc] setTexture:texture];
id<MTLCommandBuffer> cb = [[sharedData commandQueue] commandBufferWithUnretainedReferences];
id<MTLRenderCommandEncoder> ce = [cb renderCommandEncoderWithDescriptor:_outputRenderPassDesc];
if (needEncodeViewport)
{
[ce setViewport:newViewport];
}
// Draw the NDS displays.
if (willDrawDisplays)
{
[ce setRenderPipelineState:[self displayOutputPipeline]]; [ce setRenderPipelineState:[self displayOutputPipeline]];
[ce setVertexBuffer:_displayVtxPositionBuffer offset:0 atIndex:0]; [ce setVertexBuffer:_displayVtxPositionBuffer offset:0 atIndex:0];
[ce setVertexBuffer:_displayTexCoordBuffer offset:0 atIndex:1]; [ce setVertexBuffer:_displayTexCoordBuffer offset:0 atIndex:1];
@ -1597,59 +1658,8 @@
} }
// Draw the HUD. // Draw the HUD.
size_t hudLength = _cdv->GetHUDString().length(); if (willDrawHUD)
size_t hudTouchLineLength = 0;
if (_cdv->GetHUDShowInput())
{ {
hudLength += HUD_INPUT_ELEMENT_LENGTH;
switch (_cdv->GetMode())
{
case ClientDisplayMode_Main:
case ClientDisplayMode_Touch:
hudTouchLineLength = HUD_INPUT_TOUCH_LINE_ELEMENTS / 2;
break;
case ClientDisplayMode_Dual:
{
switch (_cdv->GetLayout())
{
case ClientDisplayLayout_Vertical:
case ClientDisplayLayout_Horizontal:
hudTouchLineLength = HUD_INPUT_TOUCH_LINE_ELEMENTS / 2;
break;
case ClientDisplayLayout_Hybrid_2_1:
case ClientDisplayLayout_Hybrid_16_9:
case ClientDisplayLayout_Hybrid_16_10:
hudTouchLineLength = HUD_INPUT_TOUCH_LINE_ELEMENTS;
break;
}
break;
}
}
hudLength += hudTouchLineLength;
}
if ( _cdv->GetHUDVisibility() && (hudLength > 1) && ([self texHUDCharMap] != nil) )
{
if (_cdv->HUDNeedsUpdate())
{
_cdv->SetHUDPositionVertices((float)_cdv->GetViewProperties().clientWidth, (float)_cdv->GetViewProperties().clientHeight, (float *)[_hudVtxPositionBuffer contents]);
[_hudVtxPositionBuffer didModifyRange:NSMakeRange(0, sizeof(float) * hudLength * 8)];
_cdv->SetHUDColorVertices((uint32_t *)[_hudVtxColorBuffer contents]);
[_hudVtxColorBuffer didModifyRange:NSMakeRange(0, sizeof(uint32_t) * hudLength * 4)];
_cdv->SetHUDTextureCoordinates((float *)[_hudTexCoordBuffer contents]);
[_hudTexCoordBuffer didModifyRange:NSMakeRange(0, sizeof(float) * hudLength * 8)];
_cdv->ClearHUDNeedsUpdate();
}
uint8_t isScreenOverlay = 0; uint8_t isScreenOverlay = 0;
[ce setRenderPipelineState:[sharedData hudPipeline]]; [ce setRenderPipelineState:[sharedData hudPipeline]];
@ -1707,8 +1717,6 @@
}]; }];
[cb commit]; [cb commit];
[renderAutoreleasePool release];
} }
@end @end
@ -1928,9 +1936,12 @@ void MacMetalDisplayView::ProcessDisplays()
void MacMetalDisplayView::UpdateView() void MacMetalDisplayView::UpdateView()
{ {
if (this->_allowViewUpdates) if (this->_allowViewUpdates)
{
@autoreleasepool
{ {
[(DisplayViewMetalLayer *)this->GetFrontendLayer() renderToDrawable]; [(DisplayViewMetalLayer *)this->GetFrontendLayer() renderToDrawable];
} }
}
} }
#pragma mark - #pragma mark -