| 
									
										
										
										
											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>
 | 
					
						
							|  |  |  | #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"
 | 
					
						
							| 
									
										
										
										
											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-01-21 16:40:53 -03:00
										 |  |  | #include "core/frontend/scope_acquire_window_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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-17 18:34:58 -05:00
										 |  |  | EmuThread::EmuThread(GRenderWindow* render_window) : render_window(render_window) {} | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-29 02:05:06 -04:00
										 |  |  | EmuThread::~EmuThread() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-16 23:31:14 -04:00
										 |  |  | void EmuThread::run() { | 
					
						
							| 
									
										
										
										
											2019-01-23 22:09:22 -05:00
										 |  |  |     render_window->MakeCurrent(); | 
					
						
							| 
									
										
										
										
											2015-05-18 21:24:43 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-17 18:25:21 -03:00
										 |  |  |     MicroProfileOnThreadCreate("EmuThread"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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-01-23 22:09:22 -05:00
										 |  |  |     if (Settings::values.use_asynchronous_gpu_emulation) { | 
					
						
							|  |  |  |         // Release OpenGL context for the GPU thread
 | 
					
						
							|  |  |  |         render_window->DoneCurrent(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
 | 
					
						
							| 
									
										
										
										
											2015-08-17 18:25:21 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-25 01:49:34 +10:00
										 |  |  |     render_window->moveContext(); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  | class GGLContext : public Core::Frontend::GraphicsContext { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2019-04-06 17:59:56 -03:00
										 |  |  |     explicit GGLContext(QOpenGLContext* shared_context) : shared_context{shared_context} { | 
					
						
							|  |  |  |         context.setFormat(shared_context->format()); | 
					
						
							|  |  |  |         context.setShareContext(shared_context); | 
					
						
							|  |  |  |         context.create(); | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void MakeCurrent() override { | 
					
						
							| 
									
										
										
										
											2019-04-06 17:59:56 -03:00
										 |  |  |         context.makeCurrent(shared_context->surface()); | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void DoneCurrent() override { | 
					
						
							| 
									
										
										
										
											2019-04-06 17:59:56 -03:00
										 |  |  |         context.doneCurrent(); | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void SwapBuffers() override {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2019-04-06 17:59:56 -03:00
										 |  |  |     QOpenGLContext* shared_context; | 
					
						
							|  |  |  |     QOpenGLContext context; | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | class GWidgetInternal : public QWindow { | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | public: | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  |     GWidgetInternal(GRenderWindow* parent) : parent(parent) {} | 
					
						
							|  |  |  |     virtual ~GWidgetInternal() = default; | 
					
						
							| 
									
										
										
										
											2014-10-12 18:14:57 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-26 02:56:13 -02:00
										 |  |  |     void resizeEvent(QResizeEvent* ev) override { | 
					
						
							| 
									
										
										
										
											2014-10-12 18:14:57 +02:00
										 |  |  |         parent->OnClientAreaResized(ev->size().width(), ev->size().height()); | 
					
						
							|  |  |  |         parent->OnFramebufferSizeChanged(); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-10-12 18:14:57 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-25 17:01:11 -03:00
										 |  |  |     void keyPressEvent(QKeyEvent* event) override { | 
					
						
							|  |  |  |         InputCommon::GetKeyboard()->PressKey(event->key()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void keyReleaseEvent(QKeyEvent* event) override { | 
					
						
							|  |  |  |         InputCommon::GetKeyboard()->ReleaseKey(event->key()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void mousePressEvent(QMouseEvent* event) override { | 
					
						
							|  |  |  |         if (event->source() == Qt::MouseEventSynthesizedBySystem) | 
					
						
							|  |  |  |             return; // touch input is handled in TouchBeginEvent
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const auto pos{event->pos()}; | 
					
						
							|  |  |  |         if (event->button() == Qt::LeftButton) { | 
					
						
							|  |  |  |             const auto [x, y] = parent->ScaleTouch(pos); | 
					
						
							|  |  |  |             parent->TouchPressed(x, y); | 
					
						
							|  |  |  |         } else if (event->button() == Qt::RightButton) { | 
					
						
							|  |  |  |             InputCommon::GetMotionEmu()->BeginTilt(pos.x(), pos.y()); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void mouseMoveEvent(QMouseEvent* event) override { | 
					
						
							|  |  |  |         if (event->source() == Qt::MouseEventSynthesizedBySystem) | 
					
						
							|  |  |  |             return; // touch input is handled in TouchUpdateEvent
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const auto pos{event->pos()}; | 
					
						
							|  |  |  |         const auto [x, y] = parent->ScaleTouch(pos); | 
					
						
							|  |  |  |         parent->TouchMoved(x, y); | 
					
						
							|  |  |  |         InputCommon::GetMotionEmu()->Tilt(pos.x(), pos.y()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void mouseReleaseEvent(QMouseEvent* event) override { | 
					
						
							|  |  |  |         if (event->source() == Qt::MouseEventSynthesizedBySystem) | 
					
						
							|  |  |  |             return; // touch input is handled in TouchEndEvent
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (event->button() == Qt::LeftButton) | 
					
						
							|  |  |  |             parent->TouchReleased(); | 
					
						
							|  |  |  |         else if (event->button() == Qt::RightButton) | 
					
						
							|  |  |  |             InputCommon::GetMotionEmu()->EndTilt(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     void DisablePainting() { | 
					
						
							|  |  |  |         do_painting = false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-03-25 17:01:11 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     void EnablePainting() { | 
					
						
							|  |  |  |         do_painting = true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-08-26 22:04:12 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  |     std::pair<unsigned, unsigned> GetSize() const { | 
					
						
							|  |  |  |         return std::make_pair(width(), height()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | protected: | 
					
						
							|  |  |  |     bool IsPaintingEnabled() const { | 
					
						
							|  |  |  |         return do_painting; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | private: | 
					
						
							| 
									
										
										
										
											2014-10-12 18:14:57 +02:00
										 |  |  |     GRenderWindow* parent; | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  |     bool do_painting = false; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // This class overrides paintEvent and resizeEvent to prevent the GUI thread from stealing GL
 | 
					
						
							|  |  |  | // context.
 | 
					
						
							|  |  |  | // The corresponding functionality is handled in EmuThread instead
 | 
					
						
							|  |  |  | class GGLWidgetInternal final : public GWidgetInternal, public QOpenGLWindow { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     GGLWidgetInternal(GRenderWindow* parent, QOpenGLContext* shared_context) | 
					
						
							|  |  |  |         : GWidgetInternal(parent), QOpenGLWindow(shared_context) {} | 
					
						
							|  |  |  |     ~GGLWidgetInternal() override = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void paintEvent(QPaintEvent* ev) override { | 
					
						
							|  |  |  |         if (IsPaintingEnabled()) { | 
					
						
							|  |  |  |             QPainter painter(this); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-29 19:20:12 -03:00
										 |  |  | #ifdef HAS_VULKAN
 | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | class GVKWidgetInternal final : public GWidgetInternal { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     GVKWidgetInternal(GRenderWindow* parent, QVulkanInstance* instance) : GWidgetInternal(parent) { | 
					
						
							|  |  |  |         setSurfaceType(QSurface::SurfaceType::VulkanSurface); | 
					
						
							|  |  |  |         setVulkanInstance(instance); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ~GVKWidgetInternal() override = default; | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-01-29 19:20:12 -03:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-29 01:53:30 -04:00
										 |  |  | GRenderWindow::GRenderWindow(GMainWindow* parent, EmuThread* emu_thread) | 
					
						
							| 
									
										
										
										
											2019-04-16 23:49:24 -04:00
										 |  |  |     : 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); | 
					
						
							| 
									
										
										
										
											2014-11-13 18:17:39 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-21 11:53:03 +02:00
										 |  |  |     InputCommon::Init(); | 
					
						
							| 
									
										
										
										
											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(); | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Avoid an unordered destruction that generates a segfault
 | 
					
						
							|  |  |  |     delete child; | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void GRenderWindow::moveContext() { | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  |     if (!context) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-08-25 00:47:00 +10:00
										 |  |  |     DoneCurrent(); | 
					
						
							| 
									
										
										
										
											2018-01-17 23:19:41 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     // If the thread started running, move the GL Context to the new thread. Otherwise, move it
 | 
					
						
							|  |  |  |     // back.
 | 
					
						
							|  |  |  |     auto thread = (QThread::currentThread() == qApp->thread() && emu_thread != nullptr) | 
					
						
							|  |  |  |                       ? emu_thread | 
					
						
							|  |  |  |                       : qApp->thread(); | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     context->moveToThread(thread); | 
					
						
							| 
									
										
										
										
											2014-08-25 00:47:00 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void GRenderWindow::SwapBuffers() { | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  |     if (context) { | 
					
						
							|  |  |  |         context->swapBuffers(child); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void GRenderWindow::MakeCurrent() { | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  |     if (context) { | 
					
						
							|  |  |  |         context->makeCurrent(child); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void GRenderWindow::DoneCurrent() { | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  |     if (context) { | 
					
						
							|  |  |  |         context->doneCurrent(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 18:01:46 -07:00
										 |  |  | void GRenderWindow::PollEvents() {} | 
					
						
							| 
									
										
										
										
											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(); | 
					
						
							|  |  |  |     const VkSurfaceKHR surface_copy = vk_instance->surfaceForWindow(child); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     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-01-21 16:40:53 -03:00
										 |  |  |     const qreal pixelRatio{GetWindowPixelRatio()}; | 
					
						
							|  |  |  |     const auto size{child->GetSize()}; | 
					
						
							|  |  |  |     UpdateCurrentFramebufferLayout(size.first * pixelRatio, size.second * pixelRatio); | 
					
						
							| 
									
										
										
										
											2014-08-29 22:23:12 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-25 17:01:11 -03:00
										 |  |  | void GRenderWindow::ForwardKeyPressEvent(QKeyEvent* event) { | 
					
						
							|  |  |  |     if (child) { | 
					
						
							|  |  |  |         child->keyPressEvent(event); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GRenderWindow::ForwardKeyReleaseEvent(QKeyEvent* event) { | 
					
						
							|  |  |  |     if (child) { | 
					
						
							|  |  |  |         child->keyReleaseEvent(event); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-25 17:01:11 -03:00
										 |  |  | qreal GRenderWindow::GetWindowPixelRatio() const { | 
					
						
							| 
									
										
										
										
											2015-09-10 23:42:45 +02:00
										 |  |  |     // windowHandle() might not be accessible until the window is displayed to screen.
 | 
					
						
							|  |  |  |     return windowHandle() ? windowHandle()->screen()->devicePixelRatio() : 1.0f; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-29 02:02:36 -04:00
										 |  |  | std::pair<u32, u32> GRenderWindow::ScaleTouch(const QPointF pos) const { | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  |     const qreal pixel_ratio{GetWindowPixelRatio()}; | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-29 02:02:36 -04:00
										 |  |  | void GRenderWindow::OnClientAreaResized(u32 width, u32 height) { | 
					
						
							| 
									
										
										
										
											2014-10-12 18:14:57 +02:00
										 |  |  |     NotifyClientAreaSizeChanged(std::make_pair(width, height)); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											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 { | 
					
						
							| 
									
										
										
										
											2019-04-06 17:59:56 -03:00
										 |  |  |     return std::make_unique<GGLContext>(context.get()); | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | bool GRenderWindow::InitRenderTarget() { | 
					
						
							| 
									
										
										
										
											2019-03-25 16:41:48 -03:00
										 |  |  |     shared_context.reset(); | 
					
						
							|  |  |  |     context.reset(); | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  |     if (child) { | 
					
						
							|  |  |  |         delete child; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (container) { | 
					
						
							|  |  |  |         delete container; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (layout()) { | 
					
						
							|  |  |  |         delete layout(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-25 16:41:48 -03:00
										 |  |  |     container = QWidget::createWindowContainer(child, this); | 
					
						
							| 
									
										
										
										
											2016-08-29 21:28:58 -04:00
										 |  |  |     QBoxLayout* layout = new QHBoxLayout(this); | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-25 16:46:30 -03:00
										 |  |  |     layout->addWidget(container); | 
					
						
							| 
									
										
										
										
											2016-08-29 21:28:58 -04:00
										 |  |  |     layout->setMargin(0); | 
					
						
							|  |  |  |     setLayout(layout); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     child->resize(Layout::ScreenUndocked::Width, Layout::ScreenUndocked::Height); | 
					
						
							|  |  |  |     container->resize(Layout::ScreenUndocked::Width, Layout::ScreenUndocked::Height); | 
					
						
							| 
									
										
										
										
											2016-08-29 21:28:58 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     OnMinimalClientAreaChangeRequest(GetActiveConfig().min_client_area_size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     OnFramebufferSizeChanged(); | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  |     NotifyClientAreaSizeChanged(child->GetSize()); | 
					
						
							| 
									
										
										
										
											2016-08-29 21:28:58 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							|  |  |  |     shared_context = std::make_unique<QOpenGLContext>(); | 
					
						
							|  |  |  |     shared_context->setFormat(fmt); | 
					
						
							|  |  |  |     shared_context->create(); | 
					
						
							|  |  |  |     context = std::make_unique<QOpenGLContext>(); | 
					
						
							|  |  |  |     context->setShareContext(shared_context.get()); | 
					
						
							|  |  |  |     context->setFormat(fmt); | 
					
						
							|  |  |  |     context->create(); | 
					
						
							|  |  |  |     fmt.setSwapInterval(false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     child = new GGLWidgetInternal(this, shared_context.get()); | 
					
						
							|  |  |  |     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; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     child = new GVKWidgetInternal(this, vk_instance.get()); | 
					
						
							|  |  |  |     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() { | 
					
						
							|  |  |  |     Core::Frontend::ScopeAcquireWindowContext acquire_context{*this}; | 
					
						
							|  |  |  |     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-08-26 22:04:12 +02:00
										 |  |  |     child->DisablePainting(); | 
					
						
							| 
									
										
										
										
											2015-04-29 00:01:41 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-30 19:46:50 -04:00
										 |  |  | void GRenderWindow::OnEmulationStopping() { | 
					
						
							| 
									
										
										
										
											2015-04-29 00:01:41 -04:00
										 |  |  |     emu_thread = nullptr; | 
					
						
							| 
									
										
										
										
											2015-08-26 22:04:12 +02:00
										 |  |  |     child->EnablePainting(); | 
					
						
							| 
									
										
										
										
											2015-04-29 00:01:41 -04:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } |