| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  | // Copyright 2014 Citra Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | #include <glad/glad.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-11 00:23:00 -04:00
										 |  |  | #include <QApplication>
 | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | #include <QHBoxLayout>
 | 
					
						
							|  |  |  | #include <QKeyEvent>
 | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | #include <QMessageBox>
 | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  | #include <QOffscreenSurface>
 | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | #include <QOpenGLContext>
 | 
					
						
							|  |  |  | #include <QOpenGLFunctions>
 | 
					
						
							|  |  |  | #include <QOpenGLFunctions_4_3_Core>
 | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  | #include <QOpenGLWindow>
 | 
					
						
							|  |  |  | #include <QPainter>
 | 
					
						
							| 
									
										
										
										
											2014-08-29 22:23:12 -07:00
										 |  |  | #include <QScreen>
 | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | #include <QStringList>
 | 
					
						
							| 
									
										
										
										
											2014-08-29 22:23:12 -07:00
										 |  |  | #include <QWindow>
 | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | #ifdef HAS_VULKAN
 | 
					
						
							|  |  |  | #include <QVulkanWindow>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-29 18:37:15 -04:00
										 |  |  | #include <fmt/format.h>
 | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "common/assert.h"
 | 
					
						
							| 
									
										
										
										
											2015-08-17 18:25:21 -03:00
										 |  |  | #include "common/microprofile.h"
 | 
					
						
							| 
									
										
										
										
											2015-09-11 00:23:00 -04:00
										 |  |  | #include "common/scm_rev.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | #include "common/scope_exit.h"
 | 
					
						
							| 
									
										
										
										
											2014-04-10 20:50:10 -04:00
										 |  |  | #include "core/core.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-09 22:36:07 -05:00
										 |  |  | #include "core/frontend/framebuffer_layout.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-17 15:38:56 -05:00
										 |  |  | #include "core/frontend/scope_acquire_context.h"
 | 
					
						
							| 
									
										
										
										
											2017-01-28 12:33:35 +02:00
										 |  |  | #include "core/settings.h"
 | 
					
						
							| 
									
										
										
										
											2017-01-21 11:53:03 +02:00
										 |  |  | #include "input_common/keyboard.h"
 | 
					
						
							|  |  |  | #include "input_common/main.h"
 | 
					
						
							| 
									
										
										
										
											2017-08-07 00:04:06 +03:00
										 |  |  | #include "input_common/motion_emu.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-31 14:16:16 +08:00
										 |  |  | #include "video_core/renderer_base.h"
 | 
					
						
							|  |  |  | #include "video_core/video_core.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-11 20:33:56 -07:00
										 |  |  | #include "yuzu/bootmanager.h"
 | 
					
						
							| 
									
										
										
										
											2019-01-17 00:01:00 -07:00
										 |  |  | #include "yuzu/main.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-11 20:33:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | EmuThread::EmuThread(Core::Frontend::GraphicsContext& core_context) : core_context(core_context) {} | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-29 02:05:06 -04:00
										 |  |  | EmuThread::~EmuThread() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | static GMainWindow* GetMainWindow() { | 
					
						
							|  |  |  |     for (QWidget* w : qApp->topLevelWidgets()) { | 
					
						
							|  |  |  |         if (GMainWindow* main = qobject_cast<GMainWindow*>(w)) { | 
					
						
							|  |  |  |             return main; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return nullptr; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-05-18 21:24:43 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | void EmuThread::run() { | 
					
						
							| 
									
										
										
										
											2015-08-17 18:25:21 -03:00
										 |  |  |     MicroProfileOnThreadCreate("EmuThread"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  |     // Acquire render context for duration of the thread if this is the rendering thread
 | 
					
						
							|  |  |  |     if (!Settings::values.use_asynchronous_gpu_emulation) { | 
					
						
							|  |  |  |         core_context.MakeCurrent(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     SCOPE_EXIT({ | 
					
						
							|  |  |  |         if (!Settings::values.use_asynchronous_gpu_emulation) { | 
					
						
							|  |  |  |             core_context.DoneCurrent(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-21 16:38:23 -03:00
										 |  |  |     emit LoadProgress(VideoCore::LoadCallbackStage::Prepare, 0, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Core::System::GetInstance().Renderer().Rasterizer().LoadDiskResources( | 
					
						
							|  |  |  |         stop_run, [this](VideoCore::LoadCallbackStage stage, std::size_t value, std::size_t total) { | 
					
						
							|  |  |  |             emit LoadProgress(stage, value, total); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     emit LoadProgress(VideoCore::LoadCallbackStage::Complete, 0, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-10 18:06:49 -06:00
										 |  |  |     // Holds whether the cpu was running during the last iteration,
 | 
					
						
							| 
									
										
										
										
											2015-01-07 12:14:23 +01:00
										 |  |  |     // so that the DebugModeLeft signal can be emitted before the
 | 
					
						
							|  |  |  |     // next execution step
 | 
					
						
							|  |  |  |     bool was_active = false; | 
					
						
							| 
									
										
										
										
											2015-04-16 23:31:14 -04:00
										 |  |  |     while (!stop_run) { | 
					
						
							| 
									
										
										
										
											2015-04-28 19:03:01 -04:00
										 |  |  |         if (running) { | 
					
						
							| 
									
										
										
										
											2015-01-07 12:14:23 +01:00
										 |  |  |             if (!was_active) | 
					
						
							|  |  |  |                 emit DebugModeLeft(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-08 16:28:30 -05:00
										 |  |  |             Core::System::ResultStatus result = Core::System::GetInstance().RunLoop(); | 
					
						
							|  |  |  |             if (result != Core::System::ResultStatus::Success) { | 
					
						
							| 
									
										
										
										
											2018-01-16 17:32:27 +01:00
										 |  |  |                 this->SetRunning(false); | 
					
						
							| 
									
										
										
										
											2017-04-13 01:15:23 -04:00
										 |  |  |                 emit ErrorThrown(result, Core::System::GetInstance().GetStatusDetails()); | 
					
						
							| 
									
										
										
										
											2017-03-08 16:28:30 -05:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2015-01-07 12:14:23 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-28 19:03:01 -04:00
										 |  |  |             was_active = running || exec_step; | 
					
						
							| 
									
										
										
										
											2015-04-30 19:46:50 -04:00
										 |  |  |             if (!was_active && !stop_run) | 
					
						
							| 
									
										
										
										
											2015-01-07 12:14:23 +01:00
										 |  |  |                 emit DebugModeEntered(); | 
					
						
							| 
									
										
										
										
											2015-04-28 19:03:01 -04:00
										 |  |  |         } else if (exec_step) { | 
					
						
							| 
									
										
										
										
											2015-01-07 12:14:23 +01:00
										 |  |  |             if (!was_active) | 
					
						
							|  |  |  |                 emit DebugModeLeft(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-28 19:03:01 -04:00
										 |  |  |             exec_step = false; | 
					
						
							| 
									
										
										
										
											2016-12-15 19:01:48 -05:00
										 |  |  |             Core::System::GetInstance().SingleStep(); | 
					
						
							| 
									
										
										
										
											2015-01-07 12:14:23 +01:00
										 |  |  |             emit DebugModeEntered(); | 
					
						
							| 
									
										
										
										
											2014-11-09 16:56:29 -05:00
										 |  |  |             yieldCurrentThread(); | 
					
						
							| 
									
										
										
										
											2015-05-25 20:34:09 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-07 12:14:23 +01:00
										 |  |  |             was_active = false; | 
					
						
							| 
									
										
										
										
											2015-05-16 11:56:00 -06:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2019-04-01 12:29:59 -04:00
										 |  |  |             std::unique_lock lock{running_mutex}; | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |             running_cv.wait(lock, [this] { return IsRunning() || exec_step || stop_run; }); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-04-16 23:31:14 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-30 08:47:50 -03:00
										 |  |  |     // Shutdown the core emulation
 | 
					
						
							| 
									
										
										
										
											2016-11-04 23:14:38 -04:00
										 |  |  |     Core::System::GetInstance().Shutdown(); | 
					
						
							| 
									
										
										
										
											2015-08-30 08:47:50 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-29 02:17:31 +02:00
										 |  |  | #if MICROPROFILE_ENABLED
 | 
					
						
							| 
									
										
										
										
											2015-08-17 18:25:21 -03:00
										 |  |  |     MicroProfileOnThreadExit(); | 
					
						
							| 
									
										
										
										
											2016-04-29 02:17:31 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  | class GGLContext : public Core::Frontend::GraphicsContext { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  |     explicit GGLContext(QOpenGLContext* shared_context) | 
					
						
							|  |  |  |         : context(new QOpenGLContext(shared_context->parent())), | 
					
						
							|  |  |  |           surface(new QOffscreenSurface(nullptr)) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // disable vsync for any shared contexts
 | 
					
						
							|  |  |  |         auto format = shared_context->format(); | 
					
						
							|  |  |  |         format.setSwapInterval(0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         context->setShareContext(shared_context); | 
					
						
							|  |  |  |         context->setFormat(format); | 
					
						
							|  |  |  |         context->create(); | 
					
						
							|  |  |  |         surface->setParent(shared_context->parent()); | 
					
						
							|  |  |  |         surface->setFormat(format); | 
					
						
							|  |  |  |         surface->create(); | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void MakeCurrent() override { | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  |         context->makeCurrent(surface); | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void DoneCurrent() override { | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  |         context->doneCurrent(); | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  |     QOpenGLContext* context; | 
					
						
							|  |  |  |     QOffscreenSurface* surface; | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  | class ChildRenderWindow : public QWindow { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     ChildRenderWindow(QWindow* parent, QWidget* event_handler) | 
					
						
							|  |  |  |         : QWindow{parent}, event_handler{event_handler} {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     virtual ~ChildRenderWindow() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     virtual void Present() = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | protected: | 
					
						
							|  |  |  |     bool event(QEvent* event) override { | 
					
						
							|  |  |  |         switch (event->type()) { | 
					
						
							|  |  |  |         case QEvent::UpdateRequest: | 
					
						
							|  |  |  |             Present(); | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         case QEvent::MouseButtonPress: | 
					
						
							|  |  |  |         case QEvent::MouseButtonRelease: | 
					
						
							|  |  |  |         case QEvent::MouseButtonDblClick: | 
					
						
							|  |  |  |         case QEvent::MouseMove: | 
					
						
							|  |  |  |         case QEvent::KeyPress: | 
					
						
							|  |  |  |         case QEvent::KeyRelease: | 
					
						
							|  |  |  |         case QEvent::FocusIn: | 
					
						
							|  |  |  |         case QEvent::FocusOut: | 
					
						
							|  |  |  |         case QEvent::FocusAboutToChange: | 
					
						
							|  |  |  |         case QEvent::Enter: | 
					
						
							|  |  |  |         case QEvent::Leave: | 
					
						
							|  |  |  |         case QEvent::Wheel: | 
					
						
							|  |  |  |         case QEvent::TabletMove: | 
					
						
							|  |  |  |         case QEvent::TabletPress: | 
					
						
							|  |  |  |         case QEvent::TabletRelease: | 
					
						
							|  |  |  |         case QEvent::TabletEnterProximity: | 
					
						
							|  |  |  |         case QEvent::TabletLeaveProximity: | 
					
						
							|  |  |  |         case QEvent::TouchBegin: | 
					
						
							|  |  |  |         case QEvent::TouchUpdate: | 
					
						
							|  |  |  |         case QEvent::TouchEnd: | 
					
						
							|  |  |  |         case QEvent::InputMethodQuery: | 
					
						
							|  |  |  |         case QEvent::TouchCancel: | 
					
						
							|  |  |  |             return QCoreApplication::sendEvent(event_handler, event); | 
					
						
							|  |  |  |         case QEvent::Drop: | 
					
						
							|  |  |  |             GetMainWindow()->DropAction(static_cast<QDropEvent*>(event)); | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         case QEvent::DragResponse: | 
					
						
							|  |  |  |         case QEvent::DragEnter: | 
					
						
							|  |  |  |         case QEvent::DragLeave: | 
					
						
							|  |  |  |         case QEvent::DragMove: | 
					
						
							|  |  |  |             GetMainWindow()->AcceptDropEvent(static_cast<QDropEvent*>(event)); | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             return QWindow::event(event); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-03-25 17:01:11 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  |     void exposeEvent(QExposeEvent* event) override { | 
					
						
							|  |  |  |         QWindow::requestUpdate(); | 
					
						
							|  |  |  |         QWindow::exposeEvent(event); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-03-25 17:01:11 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  | private: | 
					
						
							|  |  |  |     QWidget* event_handler{}; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-03-25 17:01:11 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  | class OpenGLWindow final : public ChildRenderWindow { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     OpenGLWindow(QWindow* parent, QWidget* event_handler, QOpenGLContext* shared_context) | 
					
						
							|  |  |  |         : ChildRenderWindow{parent, event_handler}, | 
					
						
							|  |  |  |           context(new QOpenGLContext(shared_context->parent())) { | 
					
						
							| 
									
										
										
										
											2015-08-26 22:04:12 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  |         // disable vsync for any shared contexts
 | 
					
						
							|  |  |  |         auto format = shared_context->format(); | 
					
						
							|  |  |  |         format.setSwapInterval(Settings::values.use_vsync ? 1 : 0); | 
					
						
							|  |  |  |         this->setFormat(format); | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  |         context->setShareContext(shared_context); | 
					
						
							|  |  |  |         context->setScreen(this->screen()); | 
					
						
							|  |  |  |         context->setFormat(format); | 
					
						
							|  |  |  |         context->create(); | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  |         setSurfaceType(QWindow::OpenGLSurface); | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  |         // TODO: One of these flags might be interesting: WA_OpaquePaintEvent, WA_NoBackground,
 | 
					
						
							|  |  |  |         // WA_DontShowOnScreen, WA_DeleteOnClose
 | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  |     ~OpenGLWindow() override { | 
					
						
							|  |  |  |         context->doneCurrent(); | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  |     void Present() override { | 
					
						
							|  |  |  |         if (!isExposed()) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         context->makeCurrent(this); | 
					
						
							|  |  |  |         Core::System::GetInstance().Renderer().TryPresent(100); | 
					
						
							|  |  |  |         context->swapBuffers(this); | 
					
						
							|  |  |  |         auto f = context->versionFunctions<QOpenGLFunctions_4_3_Core>(); | 
					
						
							|  |  |  |         f->glFinish(); | 
					
						
							|  |  |  |         QWindow::requestUpdate(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     QOpenGLContext* context{}; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef HAS_VULKAN
 | 
					
						
							|  |  |  | class VulkanWindow final : public ChildRenderWindow { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     VulkanWindow(QWindow* parent, QWidget* event_handler, QVulkanInstance* instance) | 
					
						
							|  |  |  |         : ChildRenderWindow{parent, event_handler} { | 
					
						
							|  |  |  |         setSurfaceType(QSurface::SurfaceType::VulkanSurface); | 
					
						
							|  |  |  |         setVulkanInstance(instance); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ~VulkanWindow() override = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void Present() override { | 
					
						
							|  |  |  |         // TODO(bunnei): ImplementMe
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     QWidget* event_handler{}; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | GRenderWindow::GRenderWindow(QWidget* parent_, EmuThread* emu_thread) | 
					
						
							|  |  |  |     : QWidget(parent_), emu_thread(emu_thread) { | 
					
						
							| 
									
										
										
										
											2018-10-24 08:10:56 -04:00
										 |  |  |     setWindowTitle(QStringLiteral("yuzu %1 | %2-%3") | 
					
						
							| 
									
										
										
										
											2019-05-20 14:50:05 -04:00
										 |  |  |                        .arg(QString::fromUtf8(Common::g_build_name), | 
					
						
							|  |  |  |                             QString::fromUtf8(Common::g_scm_branch), | 
					
						
							|  |  |  |                             QString::fromUtf8(Common::g_scm_desc))); | 
					
						
							| 
									
										
										
										
											2018-10-01 22:42:49 +03:00
										 |  |  |     setAttribute(Qt::WA_AcceptTouchEvents); | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  |     auto layout = new QHBoxLayout(this); | 
					
						
							|  |  |  |     layout->setMargin(0); | 
					
						
							|  |  |  |     setLayout(layout); | 
					
						
							| 
									
										
										
										
											2017-01-21 11:53:03 +02:00
										 |  |  |     InputCommon::Init(); | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     GMainWindow* parent = GetMainWindow(); | 
					
						
							| 
									
										
										
										
											2019-05-29 01:53:30 -04:00
										 |  |  |     connect(this, &GRenderWindow::FirstFrameDisplayed, parent, &GMainWindow::OnLoadComplete); | 
					
						
							| 
									
										
										
										
											2017-01-21 11:53:03 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GRenderWindow::~GRenderWindow() { | 
					
						
							|  |  |  |     InputCommon::Shutdown(); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | void GRenderWindow::MakeCurrent() { | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  |     if (core_context) { | 
					
						
							|  |  |  |         core_context->MakeCurrent(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-01-17 23:19:41 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | void GRenderWindow::DoneCurrent() { | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  |     if (core_context) { | 
					
						
							|  |  |  |         core_context->DoneCurrent(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-08-25 00:47:00 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | void GRenderWindow::PollEvents() { | 
					
						
							| 
									
										
										
										
											2019-01-17 00:01:00 -07:00
										 |  |  |     if (!first_frame) { | 
					
						
							|  |  |  |         first_frame = true; | 
					
						
							| 
									
										
										
										
											2020-01-06 14:02:47 -03:00
										 |  |  |         emit FirstFrameDisplayed(); | 
					
						
							| 
									
										
										
										
											2019-01-17 00:01:00 -07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | bool GRenderWindow::IsShown() const { | 
					
						
							|  |  |  |     return !isMinimized(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GRenderWindow::RetrieveVulkanHandlers(void* get_instance_proc_addr, void* instance, | 
					
						
							|  |  |  |                                            void* surface) const { | 
					
						
							|  |  |  | #ifdef HAS_VULKAN
 | 
					
						
							|  |  |  |     const auto instance_proc_addr = vk_instance->getInstanceProcAddr("vkGetInstanceProcAddr"); | 
					
						
							|  |  |  |     const VkInstance instance_copy = vk_instance->vkInstance(); | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  |     const VkSurfaceKHR surface_copy = vk_instance->surfaceForWindow(child_window); | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     std::memcpy(get_instance_proc_addr, &instance_proc_addr, sizeof(instance_proc_addr)); | 
					
						
							|  |  |  |     std::memcpy(instance, &instance_copy, sizeof(instance_copy)); | 
					
						
							|  |  |  |     std::memcpy(surface, &surface_copy, sizeof(surface_copy)); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     UNREACHABLE_MSG("Executing Vulkan code without compiling Vulkan"); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-12 18:14:57 +02:00
										 |  |  | // On Qt 5.0+, this correctly gets the size of the framebuffer (pixels).
 | 
					
						
							| 
									
										
										
										
											2014-08-29 22:23:12 -07:00
										 |  |  | //
 | 
					
						
							|  |  |  | // Older versions get the window size (density independent pixels),
 | 
					
						
							|  |  |  | // and hence, do not support DPI scaling ("retina" displays).
 | 
					
						
							|  |  |  | // The result will be a viewport that is smaller than the extent of the window.
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void GRenderWindow::OnFramebufferSizeChanged() { | 
					
						
							|  |  |  |     // Screen changes potentially incur a change in screen DPI, hence we should update the
 | 
					
						
							|  |  |  |     // framebuffer size
 | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  |     const qreal pixel_ratio = windowPixelRatio(); | 
					
						
							|  |  |  |     const u32 width = this->width() * pixel_ratio; | 
					
						
							|  |  |  |     const u32 height = this->height() * pixel_ratio; | 
					
						
							|  |  |  |     UpdateCurrentFramebufferLayout(width, height); | 
					
						
							| 
									
										
										
										
											2019-03-25 17:01:11 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void GRenderWindow::BackupGeometry() { | 
					
						
							| 
									
										
										
										
											2019-05-29 01:49:38 -04:00
										 |  |  |     geometry = QWidget::saveGeometry(); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void GRenderWindow::RestoreGeometry() { | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |     // We don't want to back up the geometry here (obviously)
 | 
					
						
							|  |  |  |     QWidget::restoreGeometry(geometry); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void GRenderWindow::restoreGeometry(const QByteArray& geometry) { | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |     // Make sure users of this class don't need to deal with backing up the geometry themselves
 | 
					
						
							|  |  |  |     QWidget::restoreGeometry(geometry); | 
					
						
							|  |  |  |     BackupGeometry(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | QByteArray GRenderWindow::saveGeometry() { | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |     // If we are a top-level widget, store the current geometry
 | 
					
						
							|  |  |  |     // otherwise, store the last backup
 | 
					
						
							| 
									
										
										
										
											2019-05-29 01:49:38 -04:00
										 |  |  |     if (parent() == nullptr) { | 
					
						
							|  |  |  |         return QWidget::saveGeometry(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return geometry; | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | qreal GRenderWindow::windowPixelRatio() const { | 
					
						
							|  |  |  |     return devicePixelRatio(); | 
					
						
							| 
									
										
										
										
											2015-09-10 23:42:45 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-29 02:02:36 -04:00
										 |  |  | std::pair<u32, u32> GRenderWindow::ScaleTouch(const QPointF pos) const { | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  |     const qreal pixel_ratio = windowPixelRatio(); | 
					
						
							| 
									
										
										
										
											2019-05-29 02:02:36 -04:00
										 |  |  |     return {static_cast<u32>(std::max(std::round(pos.x() * pixel_ratio), qreal{0.0})), | 
					
						
							|  |  |  |             static_cast<u32>(std::max(std::round(pos.y() * pixel_ratio), qreal{0.0}))}; | 
					
						
							| 
									
										
										
										
											2018-10-01 22:42:49 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-05 12:29:44 +02:00
										 |  |  | void GRenderWindow::closeEvent(QCloseEvent* event) { | 
					
						
							|  |  |  |     emit Closed(); | 
					
						
							|  |  |  |     QWidget::closeEvent(event); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | void GRenderWindow::keyPressEvent(QKeyEvent* event) { | 
					
						
							|  |  |  |     InputCommon::GetKeyboard()->PressKey(event->key()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GRenderWindow::keyReleaseEvent(QKeyEvent* event) { | 
					
						
							|  |  |  |     InputCommon::GetKeyboard()->ReleaseKey(event->key()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GRenderWindow::mousePressEvent(QMouseEvent* event) { | 
					
						
							|  |  |  |     if (event->source() == Qt::MouseEventSynthesizedBySystem) | 
					
						
							|  |  |  |         return; // touch input is handled in TouchBeginEvent
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto pos = event->pos(); | 
					
						
							|  |  |  |     if (event->button() == Qt::LeftButton) { | 
					
						
							|  |  |  |         const auto [x, y] = ScaleTouch(pos); | 
					
						
							|  |  |  |         this->TouchPressed(x, y); | 
					
						
							|  |  |  |     } else if (event->button() == Qt::RightButton) { | 
					
						
							|  |  |  |         InputCommon::GetMotionEmu()->BeginTilt(pos.x(), pos.y()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GRenderWindow::mouseMoveEvent(QMouseEvent* event) { | 
					
						
							|  |  |  |     if (event->source() == Qt::MouseEventSynthesizedBySystem) | 
					
						
							|  |  |  |         return; // touch input is handled in TouchUpdateEvent
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto pos = event->pos(); | 
					
						
							|  |  |  |     const auto [x, y] = ScaleTouch(pos); | 
					
						
							|  |  |  |     this->TouchMoved(x, y); | 
					
						
							|  |  |  |     InputCommon::GetMotionEmu()->Tilt(pos.x(), pos.y()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GRenderWindow::mouseReleaseEvent(QMouseEvent* event) { | 
					
						
							|  |  |  |     if (event->source() == Qt::MouseEventSynthesizedBySystem) | 
					
						
							|  |  |  |         return; // touch input is handled in TouchEndEvent
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (event->button() == Qt::LeftButton) | 
					
						
							|  |  |  |         this->TouchReleased(); | 
					
						
							|  |  |  |     else if (event->button() == Qt::RightButton) | 
					
						
							|  |  |  |         InputCommon::GetMotionEmu()->EndTilt(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-01 22:42:49 +03:00
										 |  |  | void GRenderWindow::TouchBeginEvent(const QTouchEvent* event) { | 
					
						
							|  |  |  |     // TouchBegin always has exactly one touch point, so take the .first()
 | 
					
						
							|  |  |  |     const auto [x, y] = ScaleTouch(event->touchPoints().first().pos()); | 
					
						
							|  |  |  |     this->TouchPressed(x, y); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GRenderWindow::TouchUpdateEvent(const QTouchEvent* event) { | 
					
						
							|  |  |  |     QPointF pos; | 
					
						
							|  |  |  |     int active_points = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // average all active touch points
 | 
					
						
							|  |  |  |     for (const auto tp : event->touchPoints()) { | 
					
						
							|  |  |  |         if (tp.state() & (Qt::TouchPointPressed | Qt::TouchPointMoved | Qt::TouchPointStationary)) { | 
					
						
							|  |  |  |             active_points++; | 
					
						
							|  |  |  |             pos += tp.pos(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pos /= active_points; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const auto [x, y] = ScaleTouch(pos); | 
					
						
							|  |  |  |     this->TouchMoved(x, y); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GRenderWindow::TouchEndEvent() { | 
					
						
							|  |  |  |     this->TouchReleased(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool GRenderWindow::event(QEvent* event) { | 
					
						
							|  |  |  |     if (event->type() == QEvent::TouchBegin) { | 
					
						
							|  |  |  |         TouchBeginEvent(static_cast<QTouchEvent*>(event)); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } else if (event->type() == QEvent::TouchUpdate) { | 
					
						
							|  |  |  |         TouchUpdateEvent(static_cast<QTouchEvent*>(event)); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } else if (event->type() == QEvent::TouchEnd || event->type() == QEvent::TouchCancel) { | 
					
						
							|  |  |  |         TouchEndEvent(); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return QWidget::event(event); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-17 21:41:25 +02:00
										 |  |  | void GRenderWindow::focusOutEvent(QFocusEvent* event) { | 
					
						
							|  |  |  |     QWidget::focusOutEvent(event); | 
					
						
							|  |  |  |     InputCommon::GetKeyboard()->ReleaseAllKeys(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | void GRenderWindow::resizeEvent(QResizeEvent* event) { | 
					
						
							|  |  |  |     QWidget::resizeEvent(event); | 
					
						
							|  |  |  |     OnFramebufferSizeChanged(); | 
					
						
							| 
									
										
										
										
											2014-10-12 18:14:57 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2014-10-12 22:46:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  | std::unique_ptr<Core::Frontend::GraphicsContext> GRenderWindow::CreateSharedContext() const { | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  |     if (Settings::values.renderer_backend == Settings::RendererBackend::OpenGL) { | 
					
						
							|  |  |  |         return std::make_unique<GGLContext>(QOpenGLContext::globalShareContext()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return {}; | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | bool GRenderWindow::InitRenderTarget() { | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  |     ReleaseRenderTarget(); | 
					
						
							| 
									
										
										
										
											2016-08-29 21:28:58 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-17 00:01:00 -07:00
										 |  |  |     first_frame = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  |     switch (Settings::values.renderer_backend) { | 
					
						
							|  |  |  |     case Settings::RendererBackend::OpenGL: | 
					
						
							|  |  |  |         if (!InitializeOpenGL()) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Settings::RendererBackend::Vulkan: | 
					
						
							|  |  |  |         if (!InitializeVulkan()) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-08-29 21:28:58 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  |     // Reset minimum required size to avoid resizing issues on the main window after restarting.
 | 
					
						
							| 
									
										
										
										
											2019-03-25 16:46:30 -03:00
										 |  |  |     setMinimumSize(1, 1); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  |     // Show causes the window to actually be created and the gl context as well, but we don't want
 | 
					
						
							|  |  |  |     // the widget to be shown yet, so immediately hide it.
 | 
					
						
							| 
									
										
										
										
											2019-03-25 16:46:30 -03:00
										 |  |  |     show(); | 
					
						
							|  |  |  |     hide(); | 
					
						
							| 
									
										
										
										
											2016-08-29 21:28:58 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-09 22:36:07 -05:00
										 |  |  |     resize(Layout::ScreenUndocked::Width, Layout::ScreenUndocked::Height); | 
					
						
							| 
									
										
										
										
											2016-08-29 21:28:58 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     OnMinimalClientAreaChangeRequest(GetActiveConfig().min_client_area_size); | 
					
						
							|  |  |  |     OnFramebufferSizeChanged(); | 
					
						
							|  |  |  |     BackupGeometry(); | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (Settings::values.renderer_backend == Settings::RendererBackend::OpenGL) { | 
					
						
							|  |  |  |         if (!LoadOpenGL()) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2016-08-29 21:28:58 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | void GRenderWindow::ReleaseRenderTarget() { | 
					
						
							|  |  |  |     if (child_widget) { | 
					
						
							|  |  |  |         layout()->removeWidget(child_widget); | 
					
						
							|  |  |  |         delete child_widget; | 
					
						
							|  |  |  |         child_widget = nullptr; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-29 02:14:24 -04:00
										 |  |  | void GRenderWindow::CaptureScreenshot(u32 res_scale, const QString& screenshot_path) { | 
					
						
							| 
									
										
										
										
											2018-08-31 14:16:16 +08:00
										 |  |  |     auto& renderer = Core::System::GetInstance().Renderer(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-29 02:14:24 -04:00
										 |  |  |     if (res_scale == 0) { | 
					
						
							| 
									
										
										
										
											2018-08-31 14:16:16 +08:00
										 |  |  |         res_scale = VideoCore::GetResolutionScaleFactor(renderer); | 
					
						
							| 
									
										
										
										
											2019-05-29 02:14:24 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-31 14:16:16 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const Layout::FramebufferLayout layout{Layout::FrameLayoutFromResolutionScale(res_scale)}; | 
					
						
							|  |  |  |     screenshot_image = QImage(QSize(layout.width, layout.height), QImage::Format_RGB32); | 
					
						
							| 
									
										
										
										
											2019-05-29 02:24:29 -04:00
										 |  |  |     renderer.RequestScreenshot( | 
					
						
							|  |  |  |         screenshot_image.bits(), | 
					
						
							|  |  |  |         [=] { | 
					
						
							|  |  |  |             const std::string std_screenshot_path = screenshot_path.toStdString(); | 
					
						
							|  |  |  |             if (screenshot_image.mirrored(false, true).save(screenshot_path)) { | 
					
						
							|  |  |  |                 LOG_INFO(Frontend, "Screenshot saved to \"{}\"", std_screenshot_path); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 LOG_ERROR(Frontend, "Failed to save screenshot to \"{}\"", std_screenshot_path); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         layout); | 
					
						
							| 
									
										
										
										
											2018-08-31 14:16:16 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-29 02:02:36 -04:00
										 |  |  | void GRenderWindow::OnMinimalClientAreaChangeRequest(std::pair<u32, u32> minimal_size) { | 
					
						
							| 
									
										
										
										
											2014-11-19 09:02:05 +00:00
										 |  |  |     setMinimumSize(minimal_size.first, minimal_size.second); | 
					
						
							| 
									
										
										
										
											2014-10-12 22:46:33 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-04-29 00:01:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | bool GRenderWindow::InitializeOpenGL() { | 
					
						
							|  |  |  |     // TODO: One of these flags might be interesting: WA_OpaquePaintEvent, WA_NoBackground,
 | 
					
						
							|  |  |  |     // WA_DontShowOnScreen, WA_DeleteOnClose
 | 
					
						
							|  |  |  |     QSurfaceFormat fmt; | 
					
						
							|  |  |  |     fmt.setVersion(4, 3); | 
					
						
							|  |  |  |     fmt.setProfile(QSurfaceFormat::CompatibilityProfile); | 
					
						
							|  |  |  |     fmt.setOption(QSurfaceFormat::FormatOption::DeprecatedFunctions); | 
					
						
							|  |  |  |     // TODO: expose a setting for buffer value (ie default/single/double/triple)
 | 
					
						
							|  |  |  |     fmt.setSwapBehavior(QSurfaceFormat::DefaultSwapBehavior); | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  |     fmt.setSwapInterval(0); | 
					
						
							|  |  |  |     QSurfaceFormat::setDefaultFormat(fmt); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     GMainWindow* parent = GetMainWindow(); | 
					
						
							|  |  |  |     QWindow* parent_win_handle = parent ? parent->windowHandle() : nullptr; | 
					
						
							|  |  |  |     child_window = new OpenGLWindow(parent_win_handle, this, QOpenGLContext::globalShareContext()); | 
					
						
							|  |  |  |     child_window->create(); | 
					
						
							|  |  |  |     child_widget = createWindowContainer(child_window, this); | 
					
						
							|  |  |  |     child_widget->resize(Layout::ScreenUndocked::Width, Layout::ScreenUndocked::Height); | 
					
						
							|  |  |  |     layout()->addWidget(child_widget); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     core_context = CreateSharedContext(); | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool GRenderWindow::InitializeVulkan() { | 
					
						
							|  |  |  | #ifdef HAS_VULKAN
 | 
					
						
							|  |  |  |     vk_instance = std::make_unique<QVulkanInstance>(); | 
					
						
							|  |  |  |     vk_instance->setApiVersion(QVersionNumber(1, 1, 0)); | 
					
						
							|  |  |  |     vk_instance->setFlags(QVulkanInstance::Flag::NoDebugOutputRedirect); | 
					
						
							|  |  |  |     if (Settings::values.renderer_debug) { | 
					
						
							|  |  |  |         const auto supported_layers{vk_instance->supportedLayers()}; | 
					
						
							|  |  |  |         const bool found = | 
					
						
							|  |  |  |             std::find_if(supported_layers.begin(), supported_layers.end(), [](const auto& layer) { | 
					
						
							|  |  |  |                 constexpr const char searched_layer[] = "VK_LAYER_LUNARG_standard_validation"; | 
					
						
							|  |  |  |                 return layer.name == searched_layer; | 
					
						
							|  |  |  |             }); | 
					
						
							|  |  |  |         if (found) { | 
					
						
							|  |  |  |             vk_instance->setLayers(QByteArrayList() << "VK_LAYER_LUNARG_standard_validation"); | 
					
						
							|  |  |  |             vk_instance->setExtensions(QByteArrayList() << VK_EXT_DEBUG_UTILS_EXTENSION_NAME); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!vk_instance->create()) { | 
					
						
							|  |  |  |         QMessageBox::critical( | 
					
						
							|  |  |  |             this, tr("Error while initializing Vulkan 1.1!"), | 
					
						
							|  |  |  |             tr("Your OS doesn't seem to support Vulkan 1.1 instances, or you do not have the " | 
					
						
							|  |  |  |                "latest graphics drivers.")); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  |     GMainWindow* parent = GetMainWindow(); | 
					
						
							|  |  |  |     QWindow* parent_win_handle = parent ? parent->windowHandle() : nullptr; | 
					
						
							|  |  |  |     child_window = new VulkanWindow(parent_win_handle, this, vk_instance.get()); | 
					
						
							|  |  |  |     child_window->create(); | 
					
						
							|  |  |  |     child_widget = createWindowContainer(child_window, this); | 
					
						
							|  |  |  |     child_widget->resize(Layout::ScreenUndocked::Width, Layout::ScreenUndocked::Height); | 
					
						
							|  |  |  |     layout()->addWidget(child_widget); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  |     return true; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     QMessageBox::critical(this, tr("Vulkan not available!"), | 
					
						
							|  |  |  |                           tr("yuzu has not been compiled with Vulkan support.")); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool GRenderWindow::LoadOpenGL() { | 
					
						
							| 
									
										
										
										
											2020-02-17 15:38:56 -05:00
										 |  |  |     Core::Frontend::ScopeAcquireContext acquire_context{*this}; | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  |     if (!gladLoadGL()) { | 
					
						
							|  |  |  |         QMessageBox::critical(this, tr("Error while initializing OpenGL 4.3!"), | 
					
						
							|  |  |  |                               tr("Your GPU may not support OpenGL 4.3, or you do not have the " | 
					
						
							|  |  |  |                                  "latest graphics driver.")); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QStringList unsupported_gl_extensions = GetUnsupportedGLExtensions(); | 
					
						
							|  |  |  |     if (!unsupported_gl_extensions.empty()) { | 
					
						
							|  |  |  |         QMessageBox::critical( | 
					
						
							|  |  |  |             this, tr("Error while initializing OpenGL!"), | 
					
						
							|  |  |  |             tr("Your GPU may not support one or more required OpenGL extensions. Please ensure you " | 
					
						
							|  |  |  |                "have the latest graphics driver.<br><br>Unsupported extensions:<br>") + | 
					
						
							|  |  |  |                 unsupported_gl_extensions.join(QStringLiteral("<br>"))); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QStringList GRenderWindow::GetUnsupportedGLExtensions() const { | 
					
						
							|  |  |  |     QStringList unsupported_ext; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!GLAD_GL_ARB_buffer_storage) | 
					
						
							|  |  |  |         unsupported_ext.append(QStringLiteral("ARB_buffer_storage")); | 
					
						
							|  |  |  |     if (!GLAD_GL_ARB_direct_state_access) | 
					
						
							|  |  |  |         unsupported_ext.append(QStringLiteral("ARB_direct_state_access")); | 
					
						
							|  |  |  |     if (!GLAD_GL_ARB_vertex_type_10f_11f_11f_rev) | 
					
						
							|  |  |  |         unsupported_ext.append(QStringLiteral("ARB_vertex_type_10f_11f_11f_rev")); | 
					
						
							|  |  |  |     if (!GLAD_GL_ARB_texture_mirror_clamp_to_edge) | 
					
						
							|  |  |  |         unsupported_ext.append(QStringLiteral("ARB_texture_mirror_clamp_to_edge")); | 
					
						
							|  |  |  |     if (!GLAD_GL_ARB_multi_bind) | 
					
						
							|  |  |  |         unsupported_ext.append(QStringLiteral("ARB_multi_bind")); | 
					
						
							|  |  |  |     if (!GLAD_GL_ARB_clip_control) | 
					
						
							|  |  |  |         unsupported_ext.append(QStringLiteral("ARB_clip_control")); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Extensions required to support some texture formats.
 | 
					
						
							|  |  |  |     if (!GLAD_GL_EXT_texture_compression_s3tc) | 
					
						
							|  |  |  |         unsupported_ext.append(QStringLiteral("EXT_texture_compression_s3tc")); | 
					
						
							|  |  |  |     if (!GLAD_GL_ARB_texture_compression_rgtc) | 
					
						
							|  |  |  |         unsupported_ext.append(QStringLiteral("ARB_texture_compression_rgtc")); | 
					
						
							|  |  |  |     if (!GLAD_GL_ARB_depth_buffer_float) | 
					
						
							|  |  |  |         unsupported_ext.append(QStringLiteral("ARB_depth_buffer_float")); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (const QString& ext : unsupported_ext) | 
					
						
							|  |  |  |         LOG_CRITICAL(Frontend, "Unsupported GL extension: {}", ext.toStdString()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return unsupported_ext; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-30 19:46:50 -04:00
										 |  |  | void GRenderWindow::OnEmulationStarting(EmuThread* emu_thread) { | 
					
						
							| 
									
										
										
										
											2015-04-29 00:01:41 -04:00
										 |  |  |     this->emu_thread = emu_thread; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-30 19:46:50 -04:00
										 |  |  | void GRenderWindow::OnEmulationStopping() { | 
					
						
							| 
									
										
										
										
											2015-04-29 00:01:41 -04:00
										 |  |  |     emu_thread = nullptr; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-09-04 15:55:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void GRenderWindow::showEvent(QShowEvent* event) { | 
					
						
							| 
									
										
										
										
											2015-09-04 15:55:48 +02:00
										 |  |  |     QWidget::showEvent(event); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 18:01:46 -07:00
										 |  |  |     // windowHandle() is not initialized until the Window is shown, so we connect it here.
 | 
					
						
							| 
									
										
										
										
											2018-01-18 20:03:13 -05:00
										 |  |  |     connect(windowHandle(), &QWindow::screenChanged, this, &GRenderWindow::OnFramebufferSizeChanged, | 
					
						
							|  |  |  |             Qt::UniqueConnection); | 
					
						
							| 
									
										
										
										
											2015-09-04 15:55:48 +02:00
										 |  |  | } |