mirror of https://github.com/bsnes-emu/bsnes.git
233 lines
8.9 KiB
Objective-C
233 lines
8.9 KiB
Objective-C
#import <CoreImage/CoreImage.h>
|
|
#import "GBViewMetal.h"
|
|
#pragma clang diagnostic ignored "-Wpartial-availability"
|
|
|
|
|
|
static const vector_float2 rect[] =
|
|
{
|
|
{-1, -1},
|
|
{ 1, -1},
|
|
{-1, 1},
|
|
{ 1, 1},
|
|
};
|
|
|
|
@implementation GBViewMetal
|
|
{
|
|
id<MTLDevice> device;
|
|
id<MTLTexture> texture, previous_texture;
|
|
id<MTLBuffer> vertices;
|
|
id<MTLRenderPipelineState> pipeline_state;
|
|
id<MTLCommandQueue> command_queue;
|
|
id<MTLBuffer> frame_blending_mode_buffer;
|
|
id<MTLBuffer> output_resolution_buffer;
|
|
vector_float2 output_resolution;
|
|
}
|
|
|
|
+ (bool)isSupported
|
|
{
|
|
if (MTLCopyAllDevices) {
|
|
return [MTLCopyAllDevices() count];
|
|
}
|
|
return false;
|
|
}
|
|
|
|
- (void) allocateTextures
|
|
{
|
|
if (!device) return;
|
|
|
|
MTLTextureDescriptor *texture_descriptor = [[MTLTextureDescriptor alloc] init];
|
|
|
|
texture_descriptor.pixelFormat = MTLPixelFormatRGBA8Unorm;
|
|
|
|
texture_descriptor.width = GB_get_screen_width(self.gb);
|
|
texture_descriptor.height = GB_get_screen_height(self.gb);
|
|
|
|
texture = [device newTextureWithDescriptor:texture_descriptor];
|
|
previous_texture = [device newTextureWithDescriptor:texture_descriptor];
|
|
|
|
}
|
|
|
|
- (void)createInternalView
|
|
{
|
|
MTKView *view = [[MTKView alloc] initWithFrame:self.frame device:(device = MTLCreateSystemDefaultDevice())];
|
|
view.delegate = self;
|
|
self.internalView = view;
|
|
view.paused = true;
|
|
view.enableSetNeedsDisplay = true;
|
|
view.framebufferOnly = false;
|
|
|
|
vertices = [device newBufferWithBytes:rect
|
|
length:sizeof(rect)
|
|
options:MTLResourceStorageModeShared];
|
|
|
|
static const GB_frame_blending_mode_t default_blending_mode = GB_FRAME_BLENDING_MODE_DISABLED;
|
|
frame_blending_mode_buffer = [device newBufferWithBytes:&default_blending_mode
|
|
length:sizeof(default_blending_mode)
|
|
options:MTLResourceStorageModeShared];
|
|
|
|
output_resolution_buffer = [device newBufferWithBytes:&output_resolution
|
|
length:sizeof(output_resolution)
|
|
options:MTLResourceStorageModeShared];
|
|
|
|
output_resolution = (simd_float2){view.drawableSize.width, view.drawableSize.height};
|
|
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(loadShader) name:@"GBFilterChanged" object:nil];
|
|
[self loadShader];
|
|
}
|
|
|
|
- (void) loadShader
|
|
{
|
|
NSError *error = nil;
|
|
NSString *shader_source = [NSString stringWithContentsOfFile:[[NSBundle mainBundle] pathForResource:@"MasterShader"
|
|
ofType:@"metal"
|
|
inDirectory:@"Shaders"]
|
|
encoding:NSUTF8StringEncoding
|
|
error:nil];
|
|
|
|
NSString *shader_name = [[NSUserDefaults standardUserDefaults] objectForKey:@"GBFilter"];
|
|
NSString *scaler_source = [NSString stringWithContentsOfFile:[[NSBundle mainBundle] pathForResource:shader_name
|
|
ofType:@"fsh"
|
|
inDirectory:@"Shaders"]
|
|
encoding:NSUTF8StringEncoding
|
|
error:nil];
|
|
|
|
shader_source = [shader_source stringByReplacingOccurrencesOfString:@"{filter}"
|
|
withString:scaler_source];
|
|
|
|
MTLCompileOptions *options = [[MTLCompileOptions alloc] init];
|
|
options.fastMathEnabled = true;
|
|
id<MTLLibrary> library = [device newLibraryWithSource:shader_source
|
|
options:options
|
|
error:&error];
|
|
if (error) {
|
|
NSLog(@"Error: %@", error);
|
|
if (!library) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
id<MTLFunction> vertex_function = [library newFunctionWithName:@"vertex_shader"];
|
|
id<MTLFunction> fragment_function = [library newFunctionWithName:@"fragment_shader"];
|
|
|
|
// Set up a descriptor for creating a pipeline state object
|
|
MTLRenderPipelineDescriptor *pipeline_state_descriptor = [[MTLRenderPipelineDescriptor alloc] init];
|
|
pipeline_state_descriptor.vertexFunction = vertex_function;
|
|
pipeline_state_descriptor.fragmentFunction = fragment_function;
|
|
pipeline_state_descriptor.colorAttachments[0].pixelFormat = ((MTKView *)self.internalView).colorPixelFormat;
|
|
|
|
error = nil;
|
|
pipeline_state = [device newRenderPipelineStateWithDescriptor:pipeline_state_descriptor
|
|
error:&error];
|
|
if (error) {
|
|
NSLog(@"Failed to created pipeline state, error %@", error);
|
|
return;
|
|
}
|
|
|
|
command_queue = [device newCommandQueue];
|
|
}
|
|
|
|
- (void)mtkView:(MTKView *)view drawableSizeWillChange:(CGSize)size
|
|
{
|
|
output_resolution = (vector_float2){size.width, size.height};
|
|
dispatch_async(dispatch_get_main_queue(), ^{
|
|
[(MTKView *)self.internalView draw];
|
|
});
|
|
}
|
|
|
|
- (void)drawInMTKView:(MTKView *)view
|
|
{
|
|
if (!(view.window.occlusionState & NSWindowOcclusionStateVisible)) return;
|
|
if (!self.gb) return;
|
|
if (texture.width != GB_get_screen_width(self.gb) ||
|
|
texture.height != GB_get_screen_height(self.gb)) {
|
|
[self allocateTextures];
|
|
}
|
|
|
|
MTLRegion region = {
|
|
{0, 0, 0}, // MTLOrigin
|
|
{texture.width, texture.height, 1} // MTLSize
|
|
};
|
|
|
|
[texture replaceRegion:region
|
|
mipmapLevel:0
|
|
withBytes:[self currentBuffer]
|
|
bytesPerRow:texture.width * 4];
|
|
if ([self frameBlendingMode]) {
|
|
[previous_texture replaceRegion:region
|
|
mipmapLevel:0
|
|
withBytes:[self previousBuffer]
|
|
bytesPerRow:texture.width * 4];
|
|
}
|
|
|
|
MTLRenderPassDescriptor *render_pass_descriptor = view.currentRenderPassDescriptor;
|
|
id<MTLCommandBuffer> command_buffer = [command_queue commandBuffer];
|
|
|
|
if (render_pass_descriptor != nil) {
|
|
*(GB_frame_blending_mode_t *)[frame_blending_mode_buffer contents] = [self frameBlendingMode];
|
|
*(vector_float2 *)[output_resolution_buffer contents] = output_resolution;
|
|
|
|
id<MTLRenderCommandEncoder> render_encoder =
|
|
[command_buffer renderCommandEncoderWithDescriptor:render_pass_descriptor];
|
|
|
|
[render_encoder setViewport:(MTLViewport){0.0, 0.0,
|
|
output_resolution.x,
|
|
output_resolution.y,
|
|
-1.0, 1.0}];
|
|
|
|
[render_encoder setRenderPipelineState:pipeline_state];
|
|
|
|
[render_encoder setVertexBuffer:vertices
|
|
offset:0
|
|
atIndex:0];
|
|
|
|
[render_encoder setFragmentBuffer:frame_blending_mode_buffer
|
|
offset:0
|
|
atIndex:0];
|
|
|
|
[render_encoder setFragmentBuffer:output_resolution_buffer
|
|
offset:0
|
|
atIndex:1];
|
|
|
|
[render_encoder setFragmentTexture:texture
|
|
atIndex:0];
|
|
|
|
[render_encoder setFragmentTexture:previous_texture
|
|
atIndex:1];
|
|
|
|
[render_encoder drawPrimitives:MTLPrimitiveTypeTriangleStrip
|
|
vertexStart:0
|
|
vertexCount:4];
|
|
|
|
[render_encoder endEncoding];
|
|
|
|
[command_buffer presentDrawable:view.currentDrawable];
|
|
}
|
|
|
|
|
|
[command_buffer commit];
|
|
}
|
|
|
|
- (void)flip
|
|
{
|
|
[super flip];
|
|
dispatch_async(dispatch_get_main_queue(), ^{
|
|
[(MTKView *)self.internalView setNeedsDisplay:true];
|
|
});
|
|
}
|
|
|
|
- (NSImage *)renderToImage
|
|
{
|
|
CIImage *ciImage = [CIImage imageWithMTLTexture:[[(MTKView *)self.internalView currentDrawable] texture]
|
|
options:@{
|
|
kCIImageColorSpace: (__bridge_transfer id)CGColorSpaceCreateDeviceRGB()
|
|
}];
|
|
ciImage = [ciImage imageByApplyingTransform:CGAffineTransformTranslate(CGAffineTransformMakeScale(1, -1),
|
|
0, ciImage.extent.size.height)];
|
|
CIContext *context = [CIContext context];
|
|
CGImageRef cgImage = [context createCGImage:ciImage fromRect:ciImage.extent];
|
|
NSImage *ret = [[NSImage alloc] initWithCGImage:cgImage size:self.internalView.bounds.size];
|
|
CGImageRelease(cgImage);
|
|
return ret;
|
|
}
|
|
|
|
@end
|