| 
									
										
										
										
											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>
 | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  | #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-04-02 02:32:58 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if !defined(WIN32) && HAS_VULKAN
 | 
					
						
							|  |  |  | #include <qpa/qplatformnativeinterface.h>
 | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | #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"
 | 
					
						
							| 
									
										
										
										
											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-03-24 20:58:49 -06:00
										 |  |  | EmuThread::EmuThread() = default; | 
					
						
							| 
									
										
										
										
											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
										 |  |  | void EmuThread::run() { | 
					
						
							| 
									
										
										
										
											2015-08-17 18:25:21 -03:00
										 |  |  |     MicroProfileOnThreadCreate("EmuThread"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 22:57:36 -06:00
										 |  |  |     // Main process has been loaded. Make the context current to this thread and begin GPU and CPU
 | 
					
						
							|  |  |  |     // execution.
 | 
					
						
							|  |  |  |     Core::System::GetInstance().GPU().Start(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  | class OpenGLSharedContext : public Core::Frontend::GraphicsContext { | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  | public: | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |     /// Create the original context that should be shared from
 | 
					
						
							|  |  |  |     explicit OpenGLSharedContext(QSurface* surface) : surface(surface) { | 
					
						
							|  |  |  |         QSurfaceFormat format; | 
					
						
							|  |  |  |         format.setVersion(4, 3); | 
					
						
							|  |  |  |         format.setProfile(QSurfaceFormat::CompatibilityProfile); | 
					
						
							|  |  |  |         format.setOption(QSurfaceFormat::FormatOption::DeprecatedFunctions); | 
					
						
							|  |  |  |         // TODO: expose a setting for buffer value (ie default/single/double/triple)
 | 
					
						
							|  |  |  |         format.setSwapBehavior(QSurfaceFormat::DefaultSwapBehavior); | 
					
						
							|  |  |  |         format.setSwapInterval(0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         context = std::make_unique<QOpenGLContext>(); | 
					
						
							|  |  |  |         context->setFormat(format); | 
					
						
							|  |  |  |         if (!context->create()) { | 
					
						
							|  |  |  |             LOG_ERROR(Frontend, "Unable to create main openGL context"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Create the shared contexts for rendering and presentation
 | 
					
						
							|  |  |  |     explicit OpenGLSharedContext(QOpenGLContext* share_context, QSurface* main_surface = nullptr) { | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // disable vsync for any shared contexts
 | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |         auto format = share_context->format(); | 
					
						
							|  |  |  |         format.setSwapInterval(main_surface ? Settings::values.use_vsync : 0); | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |         context = std::make_unique<QOpenGLContext>(); | 
					
						
							|  |  |  |         context->setShareContext(share_context); | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  |         context->setFormat(format); | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |         if (!context->create()) { | 
					
						
							|  |  |  |             LOG_ERROR(Frontend, "Unable to create shared openGL context"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!main_surface) { | 
					
						
							|  |  |  |             offscreen_surface = std::make_unique<QOffscreenSurface>(nullptr); | 
					
						
							|  |  |  |             offscreen_surface->setFormat(format); | 
					
						
							|  |  |  |             offscreen_surface->create(); | 
					
						
							|  |  |  |             surface = offscreen_surface.get(); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             surface = main_surface; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ~OpenGLSharedContext() { | 
					
						
							| 
									
										
										
										
											2020-03-30 14:52:46 -06:00
										 |  |  |         DoneCurrent(); | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void SwapBuffers() override { | 
					
						
							|  |  |  |         context->swapBuffers(surface); | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void MakeCurrent() override { | 
					
						
							| 
									
										
										
										
											2020-05-11 23:50:03 -06:00
										 |  |  |         // We can't track the current state of the underlying context in this wrapper class because
 | 
					
						
							|  |  |  |         // Qt may make the underlying context not current for one reason or another. In particular,
 | 
					
						
							|  |  |  |         // the WebBrowser uses GL, so it seems to conflict if we aren't careful.
 | 
					
						
							|  |  |  |         // Instead of always just making the context current (which does not have any caching to
 | 
					
						
							|  |  |  |         // check if the underlying context is already current) we can check for the current context
 | 
					
						
							|  |  |  |         // in the thread local data by calling `currentContext()` and checking if its ours.
 | 
					
						
							|  |  |  |         if (QOpenGLContext::currentContext() != context.get()) { | 
					
						
							|  |  |  |             context->makeCurrent(surface); | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void DoneCurrent() override { | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  |         context->doneCurrent(); | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 22:57:36 -06:00
										 |  |  |     QOpenGLContext* GetShareContext() { | 
					
						
							|  |  |  |         return context.get(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const QOpenGLContext* GetShareContext() const { | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |         return context.get(); | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |     // Avoid using Qt parent system here since we might move the QObjects to new threads
 | 
					
						
							|  |  |  |     // As a note, this means we should avoid using slots/signals with the objects too
 | 
					
						
							|  |  |  |     std::unique_ptr<QOpenGLContext> context; | 
					
						
							|  |  |  |     std::unique_ptr<QOffscreenSurface> offscreen_surface{}; | 
					
						
							|  |  |  |     QSurface* surface; | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  | class DummyContext : public Core::Frontend::GraphicsContext {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class RenderWidget : public QWidget { | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  | public: | 
					
						
							| 
									
										
										
										
											2020-03-24 22:57:36 -06:00
										 |  |  |     explicit RenderWidget(GRenderWindow* parent) : QWidget(parent), render_window(parent) { | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |         setAttribute(Qt::WA_NativeWindow); | 
					
						
							|  |  |  |         setAttribute(Qt::WA_PaintOnScreen); | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-03-25 17:01:11 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |     virtual ~RenderWidget() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 22:57:36 -06:00
										 |  |  |     /// Called on the UI thread when this Widget is ready to draw
 | 
					
						
							|  |  |  |     /// Dervied classes can override this to draw the latest frame.
 | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |     virtual void Present() {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void paintEvent(QPaintEvent* event) override { | 
					
						
							|  |  |  |         Present(); | 
					
						
							|  |  |  |         update(); | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-03-25 17:01:11 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |     QPaintEngine* paintEngine() const override { | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |     GRenderWindow* render_window; | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  | class OpenGLRenderWidget : public RenderWidget { | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  | public: | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |     explicit OpenGLRenderWidget(GRenderWindow* parent) : RenderWidget(parent) { | 
					
						
							|  |  |  |         windowHandle()->setSurfaceType(QWindow::OpenGLSurface); | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |     void SetContext(std::unique_ptr<Core::Frontend::GraphicsContext>&& context_) { | 
					
						
							|  |  |  |         context = std::move(context_); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     void Present() override { | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |         if (!isVisible()) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         context->MakeCurrent(); | 
					
						
							|  |  |  |         if (Core::System::GetInstance().Renderer().TryPresent(100)) { | 
					
						
							|  |  |  |             context->SwapBuffers(); | 
					
						
							|  |  |  |             glFinish(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |     std::unique_ptr<Core::Frontend::GraphicsContext> context{}; | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 22:57:36 -06:00
										 |  |  | #ifdef HAS_VULKAN
 | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  | class VulkanRenderWidget : public RenderWidget { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2020-04-02 02:32:58 -03:00
										 |  |  |     explicit VulkanRenderWidget(GRenderWindow* parent) : RenderWidget(parent) { | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |         windowHandle()->setSurfaceType(QWindow::VulkanSurface); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-03-24 22:57:36 -06:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-02 02:32:58 -03:00
										 |  |  | static Core::Frontend::WindowSystemType GetWindowSystemType() { | 
					
						
							|  |  |  |     // Determine WSI type based on Qt platform.
 | 
					
						
							|  |  |  |     QString platform_name = QGuiApplication::platformName(); | 
					
						
							|  |  |  |     if (platform_name == QStringLiteral("windows")) | 
					
						
							|  |  |  |         return Core::Frontend::WindowSystemType::Windows; | 
					
						
							|  |  |  |     else if (platform_name == QStringLiteral("xcb")) | 
					
						
							|  |  |  |         return Core::Frontend::WindowSystemType::X11; | 
					
						
							|  |  |  |     else if (platform_name == QStringLiteral("wayland")) | 
					
						
							|  |  |  |         return Core::Frontend::WindowSystemType::Wayland; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     LOG_CRITICAL(Frontend, "Unknown Qt platform!"); | 
					
						
							|  |  |  |     return Core::Frontend::WindowSystemType::Windows; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Core::Frontend::EmuWindow::WindowSystemInfo GetWindowSystemInfo(QWindow* window) { | 
					
						
							|  |  |  |     Core::Frontend::EmuWindow::WindowSystemInfo wsi; | 
					
						
							|  |  |  |     wsi.type = GetWindowSystemType(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef HAS_VULKAN
 | 
					
						
							|  |  |  |     // Our Win32 Qt external doesn't have the private API.
 | 
					
						
							|  |  |  | #if defined(WIN32) || defined(__APPLE__)
 | 
					
						
							|  |  |  |     wsi.render_surface = window ? reinterpret_cast<void*>(window->winId()) : nullptr; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     QPlatformNativeInterface* pni = QGuiApplication::platformNativeInterface(); | 
					
						
							|  |  |  |     wsi.display_connection = pni->nativeResourceForWindow("display", window); | 
					
						
							|  |  |  |     if (wsi.type == Core::Frontend::WindowSystemType::Wayland) | 
					
						
							|  |  |  |         wsi.render_surface = window ? pni->nativeResourceForWindow("surface", window) : nullptr; | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         wsi.render_surface = window ? reinterpret_cast<void*>(window->winId()) : nullptr; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     wsi.render_surface_scale = window ? static_cast<float>(window->devicePixelRatio()) : 1.0f; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return wsi; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GRenderWindow::GRenderWindow(GMainWindow* 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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-11 04:22:50 +02:00
										 |  |  |     this->setMouseTracking(true); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06: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::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(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 22:57:36 -06: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) { | 
					
						
							| 
									
										
										
										
											2020-03-24 22:57:36 -06:00
										 |  |  |     // touch input is handled in TouchBeginEvent
 | 
					
						
							|  |  |  |     if (event->source() == Qt::MouseEventSynthesizedBySystem) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     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()); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-04-11 04:22:50 +02:00
										 |  |  |     QWidget::mousePressEvent(event); | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GRenderWindow::mouseMoveEvent(QMouseEvent* event) { | 
					
						
							| 
									
										
										
										
											2020-03-24 22:57:36 -06:00
										 |  |  |     // touch input is handled in TouchUpdateEvent
 | 
					
						
							|  |  |  |     if (event->source() == Qt::MouseEventSynthesizedBySystem) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     auto pos = event->pos(); | 
					
						
							|  |  |  |     const auto [x, y] = ScaleTouch(pos); | 
					
						
							|  |  |  |     this->TouchMoved(x, y); | 
					
						
							|  |  |  |     InputCommon::GetMotionEmu()->Tilt(pos.x(), pos.y()); | 
					
						
							| 
									
										
										
										
											2020-04-11 04:22:50 +02:00
										 |  |  |     QWidget::mouseMoveEvent(event); | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GRenderWindow::mouseReleaseEvent(QMouseEvent* event) { | 
					
						
							| 
									
										
										
										
											2020-03-24 22:57:36 -06:00
										 |  |  |     // touch input is handled in TouchEndEvent
 | 
					
						
							|  |  |  |     if (event->source() == Qt::MouseEventSynthesizedBySystem) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 22:57:36 -06:00
										 |  |  |     if (event->button() == Qt::LeftButton) { | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  |         this->TouchReleased(); | 
					
						
							| 
									
										
										
										
											2020-03-24 22:57:36 -06:00
										 |  |  |     } else if (event->button() == Qt::RightButton) { | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  |         InputCommon::GetMotionEmu()->EndTilt(); | 
					
						
							| 
									
										
										
										
											2020-03-24 22:57:36 -06:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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) { | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |         auto c = static_cast<OpenGLSharedContext*>(main_context.get()); | 
					
						
							|  |  |  |         // Bind the shared contexts to the main surface in case the backend wants to take over
 | 
					
						
							|  |  |  |         // presentation
 | 
					
						
							|  |  |  |         return std::make_unique<OpenGLSharedContext>(c->GetShareContext(), | 
					
						
							|  |  |  |                                                      child_widget->windowHandle()); | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |     return std::make_unique<DummyContext>(); | 
					
						
							| 
									
										
										
										
											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-04-02 02:32:58 -03:00
										 |  |  |     // Update the Window System information with the new render target
 | 
					
						
							|  |  |  |     window_info = GetWindowSystemInfo(child_widget->windowHandle()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |     child_widget->resize(Layout::ScreenUndocked::Width, Layout::ScreenUndocked::Height); | 
					
						
							|  |  |  |     layout()->addWidget(child_widget); | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |         child_widget->deleteLater(); | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  |         child_widget = nullptr; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |     main_context.reset(); | 
					
						
							| 
									
										
										
										
											2020-02-17 15:53:21 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
 | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |     auto child = new OpenGLRenderWidget(this); | 
					
						
							|  |  |  |     child_widget = child; | 
					
						
							|  |  |  |     child_widget->windowHandle()->create(); | 
					
						
							|  |  |  |     auto context = std::make_shared<OpenGLSharedContext>(child->windowHandle()); | 
					
						
							|  |  |  |     main_context = context; | 
					
						
							|  |  |  |     child->SetContext( | 
					
						
							|  |  |  |         std::make_unique<OpenGLSharedContext>(context->GetShareContext(), child->windowHandle())); | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool GRenderWindow::InitializeVulkan() { | 
					
						
							|  |  |  | #ifdef HAS_VULKAN
 | 
					
						
							| 
									
										
										
										
											2020-04-02 02:32:58 -03:00
										 |  |  |     auto child = new VulkanRenderWidget(this); | 
					
						
							| 
									
										
										
										
											2020-03-24 20:58:49 -06:00
										 |  |  |     child_widget = child; | 
					
						
							|  |  |  |     child_widget->windowHandle()->create(); | 
					
						
							|  |  |  |     main_context = std::make_unique<DummyContext>(); | 
					
						
							| 
									
										
										
										
											2020-02-17 21:29:12 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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-03-24 20:58:49 -06:00
										 |  |  |     auto context = CreateSharedContext(); | 
					
						
							|  |  |  |     auto scope = context->Acquire(); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } |