mirror of
https://github.com/wheremyfoodat/Panda3DS.git
synced 2025-04-06 22:25:41 +12:00
More iOS progress
This commit is contained in:
parent
1948bea209
commit
9bc50a4b9c
6 changed files with 67 additions and 26 deletions
|
@ -1,3 +1,4 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
void iosCreateEmulator();
|
void iosCreateEmulator();
|
||||||
|
void iosRunFrame(void* drawable, void* drawableTexture);
|
|
@ -86,6 +86,10 @@ class Renderer {
|
||||||
// Called to notify the core to use OpenGL ES and not desktop GL
|
// Called to notify the core to use OpenGL ES and not desktop GL
|
||||||
virtual void setupGLES() {}
|
virtual void setupGLES() {}
|
||||||
|
|
||||||
|
// Only relevant for Metal renderer on iOS
|
||||||
|
// Passes a SwiftUI MTKView Drawable & its texture to the renderer
|
||||||
|
virtual void setMTKDrawable(void* drawable, void* drawableTexture) {};
|
||||||
|
|
||||||
// This function is called on every draw call before parsing vertex data.
|
// This function is called on every draw call before parsing vertex data.
|
||||||
// It is responsible for things like looking up which vertex/fragment shaders to use, recompiling them if they don't exist, choosing between
|
// It is responsible for things like looking up which vertex/fragment shaders to use, recompiling them if they don't exist, choosing between
|
||||||
// ubershaders and shadergen, and so on.
|
// ubershaders and shadergen, and so on.
|
||||||
|
|
|
@ -42,11 +42,16 @@ class RendererMTL final : public Renderer {
|
||||||
virtual void initGraphicsContext([[maybe_unused]] GL::Context* context) override {}
|
virtual void initGraphicsContext([[maybe_unused]] GL::Context* context) override {}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
private:
|
virtual void setMTKDrawable(void* drawable, void* drawableTexture) override;
|
||||||
CA::MetalLayer* metalLayer;
|
|
||||||
|
|
||||||
MTL::Device* device;
|
private:
|
||||||
MTL::CommandQueue* commandQueue;
|
CA::MetalLayer* metalLayer = nullptr;
|
||||||
|
|
||||||
|
CA::MetalDrawable* metalDrawable = nullptr;
|
||||||
|
MTL::Texture* drawableTexture = nullptr;
|
||||||
|
|
||||||
|
MTL::Device* device = nullptr;
|
||||||
|
MTL::CommandQueue* commandQueue = nullptr;
|
||||||
|
|
||||||
Metal::CommandEncoder commandEncoder;
|
Metal::CommandEncoder commandEncoder;
|
||||||
|
|
||||||
|
|
|
@ -30,7 +30,6 @@ PICA::ColorFmt ToColorFormat(u32 format) {
|
||||||
}
|
}
|
||||||
|
|
||||||
MTL::Library* loadLibrary(MTL::Device* device, const cmrc::file& shaderSource) {
|
MTL::Library* loadLibrary(MTL::Device* device, const cmrc::file& shaderSource) {
|
||||||
// MTL::CompileOptions* compileOptions = MTL::CompileOptions::alloc()->init();
|
|
||||||
NS::Error* error = nullptr;
|
NS::Error* error = nullptr;
|
||||||
MTL::Library* library = device->newLibrary(Metal::createDispatchData(shaderSource.begin(), shaderSource.size()), &error);
|
MTL::Library* library = device->newLibrary(Metal::createDispatchData(shaderSource.begin(), shaderSource.size()), &error);
|
||||||
// MTL::Library* library = device->newLibrary(NS::String::string(source.c_str(), NS::ASCIIStringEncoding), compileOptions, &error);
|
// MTL::Library* library = device->newLibrary(NS::String::string(source.c_str(), NS::ASCIIStringEncoding), compileOptions, &error);
|
||||||
|
@ -56,16 +55,38 @@ void RendererMTL::reset() {
|
||||||
colorRenderTargetCache.reset();
|
colorRenderTargetCache.reset();
|
||||||
}
|
}
|
||||||
|
|
||||||
void RendererMTL::display() {
|
void RendererMTL::setMTKDrawable(void* drawable, void* tex) {
|
||||||
#ifdef PANDA3DS_IOS
|
this->metalDrawable = (CA::MetalDrawable*)drawable;
|
||||||
return;
|
this->drawableTexture = (MTL::Texture*)tex;
|
||||||
#endif
|
}
|
||||||
|
|
||||||
|
void RendererMTL::display() {
|
||||||
|
static int frameCount = 0;
|
||||||
|
frameCount++;
|
||||||
|
|
||||||
|
auto manager = MTL::CaptureManager::sharedCaptureManager();
|
||||||
|
auto captureDescriptor = MTL::CaptureDescriptor::alloc()->init();
|
||||||
|
if (frameCount == 200) {
|
||||||
|
captureDescriptor->setCaptureObject(device);
|
||||||
|
manager->startCapture(captureDescriptor, nullptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef PANDA3DS_IOS
|
||||||
|
CA::MetalDrawable* drawable = metalDrawable;
|
||||||
|
if (!drawable) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
MTL::Texture* texture = drawableTexture;
|
||||||
|
#else
|
||||||
CA::MetalDrawable* drawable = metalLayer->nextDrawable();
|
CA::MetalDrawable* drawable = metalLayer->nextDrawable();
|
||||||
if (!drawable) {
|
if (!drawable) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
MTL::Texture* texture = drawable->getTexture();
|
||||||
|
#endif
|
||||||
|
|
||||||
using namespace PICA::ExternalRegs;
|
using namespace PICA::ExternalRegs;
|
||||||
|
|
||||||
// Top screen
|
// Top screen
|
||||||
|
@ -91,13 +112,13 @@ void RendererMTL::display() {
|
||||||
|
|
||||||
MTL::RenderPassDescriptor* renderPassDescriptor = MTL::RenderPassDescriptor::alloc()->init();
|
MTL::RenderPassDescriptor* renderPassDescriptor = MTL::RenderPassDescriptor::alloc()->init();
|
||||||
MTL::RenderPassColorAttachmentDescriptor* colorAttachment = renderPassDescriptor->colorAttachments()->object(0);
|
MTL::RenderPassColorAttachmentDescriptor* colorAttachment = renderPassDescriptor->colorAttachments()->object(0);
|
||||||
colorAttachment->setTexture(drawable->texture());
|
colorAttachment->setTexture(texture);
|
||||||
colorAttachment->setLoadAction(MTL::LoadActionClear);
|
colorAttachment->setLoadAction(MTL::LoadActionClear);
|
||||||
colorAttachment->setClearColor(MTL::ClearColor{0.0f, 0.0f, 0.0f, 1.0f});
|
colorAttachment->setClearColor(MTL::ClearColor{0.0f, 0.0f, 0.0f, 1.0f});
|
||||||
colorAttachment->setStoreAction(MTL::StoreActionStore);
|
colorAttachment->setStoreAction(MTL::StoreActionStore);
|
||||||
|
|
||||||
nextRenderPassName = "Display";
|
nextRenderPassName = "Display";
|
||||||
beginRenderPassIfNeeded(renderPassDescriptor, false, drawable->texture());
|
beginRenderPassIfNeeded(renderPassDescriptor, false, texture);
|
||||||
renderCommandEncoder->setRenderPipelineState(displayPipeline);
|
renderCommandEncoder->setRenderPipelineState(displayPipeline);
|
||||||
renderCommandEncoder->setFragmentSamplerState(nearestSampler, 0);
|
renderCommandEncoder->setFragmentSamplerState(nearestSampler, 0);
|
||||||
|
|
||||||
|
@ -124,24 +145,32 @@ void RendererMTL::display() {
|
||||||
// Inform the vertex buffer cache that the frame ended
|
// Inform the vertex buffer cache that the frame ended
|
||||||
vertexBufferCache.endFrame();
|
vertexBufferCache.endFrame();
|
||||||
|
|
||||||
// Release
|
// Release the drawable (not on iOS cause SwiftUI handles it there)
|
||||||
|
#ifndef PANDA3DS_IOS
|
||||||
drawable->release();
|
drawable->release();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (frameCount == 200) {
|
||||||
|
manager->stopCapture();
|
||||||
|
}
|
||||||
|
captureDescriptor->release();
|
||||||
}
|
}
|
||||||
|
|
||||||
void RendererMTL::initGraphicsContext(SDL_Window* window) {
|
void RendererMTL::initGraphicsContext(SDL_Window* window) {
|
||||||
// TODO: what should be the type of the view?
|
// On iOS, the SwiftUI side handles MetalLayer & the CommandQueue
|
||||||
|
|
||||||
#ifdef PANDA3DS_IOS
|
#ifdef PANDA3DS_IOS
|
||||||
// On iOS, the SwiftUI side handles device<->MTKView interaction
|
|
||||||
device = MTL::CreateSystemDefaultDevice();
|
device = MTL::CreateSystemDefaultDevice();
|
||||||
#else
|
#else
|
||||||
|
// TODO: what should be the type of the view?
|
||||||
void* view = SDL_Metal_CreateView(window);
|
void* view = SDL_Metal_CreateView(window);
|
||||||
metalLayer = (CA::MetalLayer*)SDL_Metal_GetLayer(view);
|
metalLayer = (CA::MetalLayer*)SDL_Metal_GetLayer(view);
|
||||||
device = MTL::CreateSystemDefaultDevice();
|
device = MTL::CreateSystemDefaultDevice();
|
||||||
metalLayer->setDevice(device);
|
metalLayer->setDevice(device);
|
||||||
commandQueue = device->newCommandQueue();
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
commandQueue = device->newCommandQueue();
|
||||||
|
printf("C++ device pointer: %p\n", device);
|
||||||
|
|
||||||
// Textures
|
// Textures
|
||||||
MTL::TextureDescriptor* textureDescriptor = MTL::TextureDescriptor::alloc()->init();
|
MTL::TextureDescriptor* textureDescriptor = MTL::TextureDescriptor::alloc()->init();
|
||||||
textureDescriptor->setTextureType(MTL::TextureType2D);
|
textureDescriptor->setTextureType(MTL::TextureType2D);
|
||||||
|
@ -816,7 +845,7 @@ void RendererMTL::beginRenderPassIfNeeded(
|
||||||
) {
|
) {
|
||||||
createCommandBufferIfNeeded();
|
createCommandBufferIfNeeded();
|
||||||
|
|
||||||
if (doesClears || !renderCommandEncoder || colorTexture != lastColorTexture ||
|
if (1 ||doesClears || !renderCommandEncoder || colorTexture != lastColorTexture ||
|
||||||
(depthTexture != lastDepthTexture && !(lastDepthTexture && !depthTexture))) {
|
(depthTexture != lastDepthTexture && !(lastDepthTexture && !depthTexture))) {
|
||||||
endRenderPass();
|
endRenderPass();
|
||||||
|
|
||||||
|
|
|
@ -655,7 +655,7 @@ float4 performLogicOp(LogicOp logicOp, float4 s, float4 d) {
|
||||||
return as_type<float4>(performLogicOpU(logicOp, as_type<uint4>(s), as_type<uint4>(d)));
|
return as_type<float4>(performLogicOpU(logicOp, as_type<uint4>(s), as_type<uint4>(d)));
|
||||||
}
|
}
|
||||||
|
|
||||||
fragment float4 fragmentDraw(DrawVertexOut in [[stage_in]], float4 prevColor [[color(0)]], constant PicaRegs& picaRegs [[buffer(0)]], constant FragTEV& tev [[buffer(1)]], constant LogicOp& logicOp [[buffer(2)]], constant uint2& lutSlices [[buffer(3)]], texture2d<float> tex0 [[texture(0)]], texture2d<float> tex1 [[texture(1)]], texture2d<float> tex2 [[texture(2)]], texture2d_array<float> texLightingLut [[texture(3)]], texture1d_array<float> texFogLut [[texture(4)]], sampler samplr0 [[sampler(0)]], sampler samplr1 [[sampler(1)]], sampler samplr2 [[sampler(2)]], sampler linearSampler [[sampler(3)]]) {
|
fragment float4 fragmentDraw(DrawVertexOut in [[stage_in]], constant PicaRegs& picaRegs [[buffer(0)]], constant FragTEV& tev [[buffer(1)]], constant LogicOp& logicOp [[buffer(2)]], constant uint2& lutSlices [[buffer(3)]], texture2d<float> tex0 [[texture(0)]], texture2d<float> tex1 [[texture(1)]], texture2d<float> tex2 [[texture(2)]], texture2d_array<float> texLightingLut [[texture(3)]], texture1d_array<float> texFogLut [[texture(4)]], sampler samplr0 [[sampler(0)]], sampler samplr1 [[sampler(1)]], sampler samplr2 [[sampler(2)]], sampler linearSampler [[sampler(3)]]) {
|
||||||
Globals globals;
|
Globals globals;
|
||||||
|
|
||||||
// HACK
|
// HACK
|
||||||
|
@ -755,5 +755,5 @@ fragment float4 fragmentDraw(DrawVertexOut in [[stage_in]], float4 prevColor [[c
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return performLogicOp(logicOp, color, prevColor);
|
return performLogicOp(logicOp, color, float4(1.0, 0.0, 0.0, 1.0));
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,7 +15,7 @@ extern "C" {
|
||||||
std::unique_ptr<Emulator> emulator = nullptr;
|
std::unique_ptr<Emulator> emulator = nullptr;
|
||||||
HIDService* hidService = nullptr;
|
HIDService* hidService = nullptr;
|
||||||
|
|
||||||
extern "C" __attribute__((visibility("default"))) void iosCreateEmulator() {
|
IOS_EXPORT void iosCreateEmulator() {
|
||||||
printf("Creating emulator\n");
|
printf("Creating emulator\n");
|
||||||
|
|
||||||
emulator = std::make_unique<Emulator>();
|
emulator = std::make_unique<Emulator>();
|
||||||
|
@ -23,12 +23,14 @@ extern "C" __attribute__((visibility("default"))) void iosCreateEmulator() {
|
||||||
emulator->initGraphicsContext(nullptr);
|
emulator->initGraphicsContext(nullptr);
|
||||||
|
|
||||||
// auto path = emulator->getAppDataRoot() / "Kirb Demo.3ds";
|
// auto path = emulator->getAppDataRoot() / "Kirb Demo.3ds";
|
||||||
|
// auto path = emulator->getAppDataRoot() / "Kirb Demo.3ds";
|
||||||
|
|
||||||
auto path = emulator->getAppDataRoot() / "SimplerTri.elf";
|
auto path = emulator->getAppDataRoot() / "SimplerTri.elf";
|
||||||
emulator->loadROM(path);
|
emulator->loadROM(path);
|
||||||
|
|
||||||
while (1) {
|
|
||||||
emulator->runFrame();
|
|
||||||
}
|
|
||||||
|
|
||||||
printf("Created emulator\n");
|
printf("Created emulator\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
IOS_EXPORT void iosRunFrame(void* drawable, void* drawableTexture) {
|
||||||
|
emulator->getRenderer()->setMTKDrawable(drawable, drawableTexture);
|
||||||
|
emulator->runFrame();
|
||||||
|
}
|
Loading…
Add table
Reference in a new issue