2018-09-29 23:18:46 +00:00
# include "session.h"
2018-06-28 08:44:43 +00:00
# include "settings/streamingpreferences.h"
2018-09-04 04:21:37 +00:00
# include "streaming/streamutils.h"
2019-06-30 00:40:30 +00:00
# include "backend/richpresencemanager.h"
2018-06-28 08:44:43 +00:00
# include <Limelight.h>
# include <SDL.h>
2018-07-08 04:52:20 +00:00
# include "utils.h"
2018-06-28 08:44:43 +00:00
2018-07-22 00:32:00 +00:00
# ifdef HAVE_FFMPEG
2018-07-18 03:00:16 +00:00
# include "video/ffmpeg.h"
2018-07-22 00:32:00 +00:00
# endif
2018-07-18 03:00:16 +00:00
2018-07-22 03:22:00 +00:00
# ifdef HAVE_SLVIDEO
2019-03-23 05:51:08 +00:00
# include "video/slvid.h"
2018-07-22 03:22:00 +00:00
# endif
2018-07-23 00:42:31 +00:00
# ifdef Q_OS_WIN32
// Scaling the icon down on Win32 looks dreadful, so render at lower res
2018-07-23 00:07:45 +00:00
# define ICON_SIZE 32
2018-07-23 00:42:31 +00:00
# else
# define ICON_SIZE 64
# endif
2018-07-23 00:07:45 +00:00
2022-10-27 23:12:26 +00:00
// HACK: Remove once proper Dark Mode support lands in SDL
# ifdef Q_OS_WIN32
# include <SDL_syswm.h>
# include <dwmapi.h>
# ifndef DWMWA_USE_IMMERSIVE_DARK_MODE_OLD
# define DWMWA_USE_IMMERSIVE_DARK_MODE_OLD 19
# endif
# ifndef DWMWA_USE_IMMERSIVE_DARK_MODE
# define DWMWA_USE_IMMERSIVE_DARK_MODE 20
# endif
# endif
2021-01-09 23:51:25 +00:00
# define SDL_CODE_FLUSH_WINDOW_EVENT_BARRIER 100
2022-03-05 19:32:38 +00:00
# define SDL_CODE_GAMECONTROLLER_RUMBLE 101
2023-06-18 21:04:49 +00:00
# define SDL_CODE_GAMECONTROLLER_RUMBLE_TRIGGERS 102
2023-06-18 21:08:32 +00:00
# define SDL_CODE_GAMECONTROLLER_SET_MOTION_EVENT_STATE 103
2023-07-02 21:44:41 +00:00
# define SDL_CODE_GAMECONTROLLER_SET_CONTROLLER_LED 104
2021-01-09 23:51:25 +00:00
2018-07-22 01:47:41 +00:00
# include <openssl/rand.h>
2018-06-28 08:44:43 +00:00
# include <QtEndian>
# include <QCoreApplication>
2018-07-09 07:09:06 +00:00
# include <QThreadPool>
2018-07-23 00:07:45 +00:00
# include <QSvgRenderer>
# include <QPainter>
# include <QImage>
2020-01-11 10:06:08 +00:00
# include <QGuiApplication>
# include <QCursor>
2022-10-27 23:12:26 +00:00
# include <QWindow>
2018-06-28 08:44:43 +00:00
2020-08-11 05:21:54 +00:00
# define CONN_TEST_SERVER "qt.conntest.moonlight-stream.org"
2018-06-28 08:44:43 +00:00
CONNECTION_LISTENER_CALLBACKS Session : : k_ConnCallbacks = {
Session : : clStageStarting ,
nullptr ,
Session : : clStageFailed ,
nullptr ,
Session : : clConnectionTerminated ,
2019-02-12 05:39:55 +00:00
Session : : clLogMessage ,
2019-03-17 22:08:21 +00:00
Session : : clRumble ,
2022-01-29 04:10:50 +00:00
Session : : clConnectionStatusUpdate ,
Session : : clSetHdrMode ,
2023-06-18 21:04:49 +00:00
Session : : clRumbleTriggers ,
2023-06-18 21:08:32 +00:00
Session : : clSetMotionEventState ,
2023-07-02 21:44:41 +00:00
Session : : clSetControllerLED ,
2018-06-28 08:44:43 +00:00
} ;
Session * Session : : s_ActiveSession ;
2018-07-09 07:09:06 +00:00
QSemaphore Session : : s_ActiveSessionSemaphore ( 1 ) ;
2018-06-28 08:44:43 +00:00
void Session : : clStageStarting ( int stage )
{
// We know this is called on the same thread as LiStartConnection()
// which happens to be the main thread, so it's cool to interact
// with the GUI in these callbacks.
2018-07-07 23:30:26 +00:00
emit s_ActiveSession - > stageStarting ( QString : : fromLocal8Bit ( LiGetStageName ( stage ) ) ) ;
2018-06-28 08:44:43 +00:00
}
2020-02-25 07:03:34 +00:00
void Session : : clStageFailed ( int stage , int errorCode )
2018-06-28 08:44:43 +00:00
{
2020-08-11 05:21:54 +00:00
// Perform the port test now, while we're on the async connection thread and not blocking the UI.
2020-12-23 19:56:15 +00:00
unsigned int portFlags = LiGetPortFlagsFromStage ( stage ) ;
s_ActiveSession - > m_PortTestResults = LiTestClientConnectivity ( CONN_TEST_SERVER , 443 , portFlags ) ;
2020-08-11 05:21:54 +00:00
2020-12-23 19:56:15 +00:00
char failingPorts [ 128 ] ;
LiStringifyPortFlags ( portFlags , " , " , failingPorts , sizeof ( failingPorts ) ) ;
emit s_ActiveSession - > stageFailed ( QString : : fromLocal8Bit ( LiGetStageName ( stage ) ) , errorCode , QString ( failingPorts ) ) ;
2018-06-28 08:44:43 +00:00
}
2020-02-25 07:03:34 +00:00
void Session : : clConnectionTerminated ( int errorCode )
2018-06-28 08:44:43 +00:00
{
2020-12-23 19:56:15 +00:00
unsigned int portFlags = LiGetPortFlagsFromTerminationErrorCode ( errorCode ) ;
s_ActiveSession - > m_PortTestResults = LiTestClientConnectivity ( CONN_TEST_SERVER , 443 , portFlags ) ;
2020-08-09 00:59:26 +00:00
2019-02-10 10:16:05 +00:00
// Display the termination dialog if this was not intended
2020-05-02 04:28:48 +00:00
switch ( errorCode ) {
case ML_ERROR_GRACEFUL_TERMINATION :
break ;
case ML_ERROR_NO_VIDEO_TRAFFIC :
s_ActiveSession - > m_UnexpectedTermination = true ;
2020-12-23 19:56:15 +00:00
char ports [ 128 ] ;
SDL_assert ( portFlags ! = 0 ) ;
LiStringifyPortFlags ( portFlags , " , " , ports , sizeof ( ports ) ) ;
emit s_ActiveSession - > displayLaunchError ( tr ( " No video received from host. " ) + " \n \n " +
tr ( " Check your firewall and port forwarding rules for port(s): %1 " ) . arg ( ports ) ) ;
2020-05-02 04:28:48 +00:00
break ;
2020-08-30 04:02:22 +00:00
case ML_ERROR_NO_VIDEO_FRAME :
s_ActiveSession - > m_UnexpectedTermination = true ;
2020-11-21 19:15:54 +00:00
emit s_ActiveSession - > displayLaunchError ( tr ( " Your network connection isn't performing well. Reduce your video bitrate setting or try a faster connection. " ) ) ;
2020-08-30 04:02:22 +00:00
break ;
2022-10-05 03:02:44 +00:00
case ML_ERROR_PROTECTED_CONTENT :
2020-12-24 17:24:01 +00:00
case ML_ERROR_UNEXPECTED_EARLY_TERMINATION :
s_ActiveSession - > m_UnexpectedTermination = true ;
emit s_ActiveSession - > displayLaunchError ( tr ( " Something went wrong on your host PC when starting the stream. " ) + " \n \n " +
2023-03-18 19:20:51 +00:00
tr ( " Make sure you don't have any DRM-protected content open on your host PC. You can also try restarting your host PC. " ) ) ;
2020-12-24 17:24:01 +00:00
break ;
2022-10-05 03:05:18 +00:00
case ML_ERROR_FRAME_CONVERSION :
s_ActiveSession - > m_UnexpectedTermination = true ;
emit s_ActiveSession - > displayLaunchError ( tr ( " The host PC reported a fatal video encoding error. " ) + " \n \n " +
tr ( " Try disabling HDR mode, changing the streaming resolution, or changing your host PC's display resolution. " ) ) ;
break ;
2020-05-02 04:28:48 +00:00
default :
2019-02-10 10:16:05 +00:00
s_ActiveSession - > m_UnexpectedTermination = true ;
2020-11-21 19:15:54 +00:00
emit s_ActiveSession - > displayLaunchError ( tr ( " Connection terminated " ) ) ;
2020-05-02 04:28:48 +00:00
break ;
2019-02-10 10:16:05 +00:00
}
2018-09-02 22:34:10 +00:00
2018-06-28 08:44:43 +00:00
SDL_LogError ( SDL_LOG_CATEGORY_APPLICATION ,
2020-02-25 07:03:34 +00:00
" Connection terminated: %d " ,
2018-06-28 08:44:43 +00:00
errorCode ) ;
// Push a quit event to the main loop
SDL_Event event ;
event . type = SDL_QUIT ;
event . quit . timestamp = SDL_GetTicks ( ) ;
SDL_PushEvent ( & event ) ;
}
void Session : : clLogMessage ( const char * format , . . . )
{
va_list ap ;
va_start ( ap , format ) ;
SDL_LogMessageV ( SDL_LOG_CATEGORY_APPLICATION ,
SDL_LOG_PRIORITY_INFO ,
format ,
ap ) ;
va_end ( ap ) ;
}
2019-02-12 05:39:55 +00:00
void Session : : clRumble ( unsigned short controllerNumber , unsigned short lowFreqMotor , unsigned short highFreqMotor )
{
2022-03-05 19:32:38 +00:00
// We push an event for the main thread to handle in order to properly synchronize
// with the removal of game controllers that could result in our game controller
// going away during this callback.
SDL_Event rumbleEvent = { } ;
rumbleEvent . type = SDL_USEREVENT ;
rumbleEvent . user . code = SDL_CODE_GAMECONTROLLER_RUMBLE ;
rumbleEvent . user . data1 = ( void * ) ( uintptr_t ) controllerNumber ;
rumbleEvent . user . data2 = ( void * ) ( uintptr_t ) ( ( lowFreqMotor < < 16 ) | highFreqMotor ) ;
SDL_PushEvent ( & rumbleEvent ) ;
2019-02-12 05:39:55 +00:00
}
2019-03-17 22:08:21 +00:00
void Session : : clConnectionStatusUpdate ( int connectionStatus )
{
2019-03-19 03:21:52 +00:00
SDL_LogInfo ( SDL_LOG_CATEGORY_APPLICATION ,
" Connection status update: %d " ,
connectionStatus ) ;
if ( ! s_ActiveSession - > m_Preferences - > connectionWarnings ) {
return ;
}
2019-05-19 20:10:42 +00:00
if ( s_ActiveSession - > m_MouseEmulationRefCount > 0 ) {
// Don't display the overlay if mouse emulation is already using it
return ;
}
2019-03-17 22:08:21 +00:00
switch ( connectionStatus )
{
case CONN_STATUS_POOR :
2023-09-23 15:15:21 +00:00
s_ActiveSession - > m_OverlayManager . updateOverlayText ( Overlay : : OverlayStatusUpdate ,
s_ActiveSession - > m_StreamConfig . bitrate > 5000 ?
" Slow connection to PC \n Reduce your bitrate " : " Poor connection to PC " ) ;
2019-03-17 22:08:21 +00:00
s_ActiveSession - > m_OverlayManager . setOverlayState ( Overlay : : OverlayStatusUpdate , true ) ;
break ;
case CONN_STATUS_OKAY :
s_ActiveSession - > m_OverlayManager . setOverlayState ( Overlay : : OverlayStatusUpdate , false ) ;
break ;
}
}
2022-01-29 04:10:50 +00:00
void Session : : clSetHdrMode ( bool enabled )
{
// If we're in the process of recreating our decoder when we get
// this callback, we'll drop it. The main thread will make the
// callback when it finishes creating the new decoder.
if ( SDL_AtomicTryLock ( & s_ActiveSession - > m_DecoderLock ) ) {
IVideoDecoder * decoder = s_ActiveSession - > m_VideoDecoder ;
if ( decoder ! = nullptr ) {
decoder - > setHdrMode ( enabled ) ;
}
SDL_AtomicUnlock ( & s_ActiveSession - > m_DecoderLock ) ;
}
}
2023-06-18 21:04:49 +00:00
void Session : : clRumbleTriggers ( uint16_t controllerNumber , uint16_t leftTrigger , uint16_t rightTrigger )
{
// We push an event for the main thread to handle in order to properly synchronize
// with the removal of game controllers that could result in our game controller
// going away during this callback.
SDL_Event rumbleEvent = { } ;
rumbleEvent . type = SDL_USEREVENT ;
rumbleEvent . user . code = SDL_CODE_GAMECONTROLLER_RUMBLE_TRIGGERS ;
rumbleEvent . user . data1 = ( void * ) ( uintptr_t ) controllerNumber ;
rumbleEvent . user . data2 = ( void * ) ( uintptr_t ) ( ( leftTrigger < < 16 ) | rightTrigger ) ;
SDL_PushEvent ( & rumbleEvent ) ;
}
2023-06-18 21:08:32 +00:00
void Session : : clSetMotionEventState ( uint16_t controllerNumber , uint8_t motionType , uint16_t reportRateHz )
{
// We push an event for the main thread to handle in order to properly synchronize
// with the removal of game controllers that could result in our game controller
// going away during this callback.
SDL_Event setMotionEventStateEvent = { } ;
setMotionEventStateEvent . type = SDL_USEREVENT ;
setMotionEventStateEvent . user . code = SDL_CODE_GAMECONTROLLER_SET_MOTION_EVENT_STATE ;
setMotionEventStateEvent . user . data1 = ( void * ) ( uintptr_t ) controllerNumber ;
setMotionEventStateEvent . user . data2 = ( void * ) ( uintptr_t ) ( ( motionType < < 16 ) | reportRateHz ) ;
SDL_PushEvent ( & setMotionEventStateEvent ) ;
}
2023-07-02 21:44:41 +00:00
void Session : : clSetControllerLED ( uint16_t controllerNumber , uint8_t r , uint8_t g , uint8_t b )
{
// We push an event for the main thread to handle in order to properly synchronize
// with the removal of game controllers that could result in our game controller
// going away during this callback.
SDL_Event setControllerLEDEvent = { } ;
setControllerLEDEvent . type = SDL_USEREVENT ;
setControllerLEDEvent . user . code = SDL_CODE_GAMECONTROLLER_SET_CONTROLLER_LED ;
setControllerLEDEvent . user . data1 = ( void * ) ( uintptr_t ) controllerNumber ;
setControllerLEDEvent . user . data2 = ( void * ) ( uintptr_t ) ( r < < 16 | g < < 8 | b ) ;
SDL_PushEvent ( & setControllerLEDEvent ) ;
}
2018-07-18 03:00:16 +00:00
bool Session : : chooseDecoder ( StreamingPreferences : : VideoDecoderSelection vds ,
SDL_Window * window , int videoFormat , int width , int height ,
2019-02-13 02:42:53 +00:00
int frameRate , bool enableVsync , bool enableFramePacing , bool testOnly , IVideoDecoder * & chosenDecoder )
2018-07-18 03:00:16 +00:00
{
2019-04-12 05:27:20 +00:00
DECODER_PARAMETERS params ;
2021-04-30 23:12:56 +00:00
// We should never have vsync enabled for test-mode.
// It introduces unnecessary delay for renderers that may
// block while waiting for a backbuffer swap.
SDL_assert ( ! enableVsync | | ! testOnly ) ;
2019-04-12 05:27:20 +00:00
params . width = width ;
params . height = height ;
params . frameRate = frameRate ;
params . videoFormat = videoFormat ;
params . window = window ;
params . enableVsync = enableVsync ;
params . enableFramePacing = enableFramePacing ;
2022-08-21 23:38:09 +00:00
params . testOnly = testOnly ;
2019-04-12 05:27:20 +00:00
params . vds = vds ;
2018-12-26 06:19:23 +00:00
SDL_LogInfo ( SDL_LOG_CATEGORY_APPLICATION ,
" V-sync %s " ,
enableVsync ? " enabled " : " disabled " ) ;
2018-07-22 03:22:00 +00:00
# ifdef HAVE_SLVIDEO
2019-02-13 02:42:53 +00:00
chosenDecoder = new SLVideoDecoder ( testOnly ) ;
2019-04-12 05:27:20 +00:00
if ( chosenDecoder - > initialize ( & params ) ) {
2018-07-22 03:22:00 +00:00
SDL_LogInfo ( SDL_LOG_CATEGORY_APPLICATION ,
" SLVideo video decoder chosen " ) ;
return true ;
}
else {
SDL_LogError ( SDL_LOG_CATEGORY_APPLICATION ,
" Unable to load SLVideo decoder " ) ;
delete chosenDecoder ;
chosenDecoder = nullptr ;
}
# endif
2018-07-22 00:00:09 +00:00
# ifdef HAVE_FFMPEG
2019-02-13 02:42:53 +00:00
chosenDecoder = new FFmpegVideoDecoder ( testOnly ) ;
2019-04-12 05:27:20 +00:00
if ( chosenDecoder - > initialize ( & params ) ) {
2018-07-18 03:00:16 +00:00
SDL_LogInfo ( SDL_LOG_CATEGORY_APPLICATION ,
" FFmpeg-based video decoder chosen " ) ;
return true ;
}
else {
SDL_LogError ( SDL_LOG_CATEGORY_APPLICATION ,
" Unable to load FFmpeg decoder " ) ;
delete chosenDecoder ;
chosenDecoder = nullptr ;
}
2018-07-22 00:00:09 +00:00
# endif
2018-07-18 03:00:16 +00:00
2018-08-03 09:11:44 +00:00
# if !defined(HAVE_FFMPEG) && !defined(HAVE_SLVIDEO)
# error No video decoding libraries available!
# endif
2018-07-18 03:00:16 +00:00
// If we reach this, we didn't initialize any decoders successfully
return false ;
}
int Session : : drSetup ( int videoFormat , int width , int height , int frameRate , void * , int )
{
2018-07-20 22:31:57 +00:00
s_ActiveSession - > m_ActiveVideoFormat = videoFormat ;
s_ActiveSession - > m_ActiveVideoWidth = width ;
s_ActiveSession - > m_ActiveVideoHeight = height ;
s_ActiveSession - > m_ActiveVideoFrameRate = frameRate ;
2018-07-21 02:55:07 +00:00
// Defer decoder setup until we've started streaming so we
// don't have to hide and show the SDL window (which seems to
// cause pointer hiding to break on Windows).
2018-07-18 03:00:16 +00:00
2018-07-28 23:06:26 +00:00
SDL_LogInfo ( SDL_LOG_CATEGORY_APPLICATION , " Video stream is %dx%dx%d (format 0x%x) " ,
width , height , frameRate , videoFormat ) ;
2018-07-18 03:00:16 +00:00
return 0 ;
}
int Session : : drSubmitDecodeUnit ( PDECODE_UNIT du )
{
// Use a lock since we'll be yanking this decoder out
// from underneath the session when we initiate destruction.
// We need to destroy the decoder on the main thread to satisfy
2018-09-23 03:03:41 +00:00
// some API constraints (like DXVA2). If we can't acquire it,
// that means the decoder is about to be destroyed, so we can
2022-10-05 05:53:31 +00:00
// safely return DR_OK and wait for the IDR frame request by
2018-09-23 03:03:41 +00:00
// the decoder reinitialization code.
2022-01-17 20:30:12 +00:00
if ( SDL_AtomicTryLock ( & s_ActiveSession - > m_DecoderLock ) ) {
2018-09-23 03:03:41 +00:00
IVideoDecoder * decoder = s_ActiveSession - > m_VideoDecoder ;
if ( decoder ! = nullptr ) {
int ret = decoder - > submitDecodeUnit ( du ) ;
SDL_AtomicUnlock ( & s_ActiveSession - > m_DecoderLock ) ;
return ret ;
}
else {
SDL_AtomicUnlock ( & s_ActiveSession - > m_DecoderLock ) ;
return DR_OK ;
}
2018-07-18 03:00:16 +00:00
}
else {
2018-09-23 03:03:41 +00:00
// Decoder is going away. Ignore anything coming in until
// the lock is released.
2018-07-18 03:00:16 +00:00
return DR_OK ;
}
}
2020-02-09 19:35:05 +00:00
void Session : : getDecoderInfo ( SDL_Window * window ,
2022-01-29 06:59:04 +00:00
bool & isHardwareAccelerated , bool & isFullScreenOnly ,
bool & isHdrSupported , QSize & maxResolution )
2020-02-09 19:35:05 +00:00
{
IVideoDecoder * decoder ;
2023-07-16 20:38:58 +00:00
// Since AV1 support on the host side is in its infancy, let's not consider
// _only_ a working AV1 decoder to be acceptable and still show the warning
// dialog indicating lack of hardware decoding support.
2022-01-29 06:59:04 +00:00
// Try an HEVC Main10 decoder first to see if we have HDR support
if ( chooseDecoder ( StreamingPreferences : : VDS_FORCE_HARDWARE ,
window , VIDEO_FORMAT_H265_MAIN10 , 1920 , 1080 , 60 ,
false , false , true , decoder ) ) {
isHardwareAccelerated = decoder - > isHardwareAccelerated ( ) ;
isFullScreenOnly = decoder - > isAlwaysFullScreen ( ) ;
isHdrSupported = decoder - > isHdrSupported ( ) ;
maxResolution = decoder - > getDecoderMaxResolution ( ) ;
delete decoder ;
2020-02-09 19:35:05 +00:00
return ;
}
2023-07-16 20:38:58 +00:00
// Try an AV1 Main10 decoder next to see if we have HDR support
if ( chooseDecoder ( StreamingPreferences : : VDS_FORCE_HARDWARE ,
window , VIDEO_FORMAT_AV1_MAIN10 , 1920 , 1080 , 60 ,
false , false , true , decoder ) ) {
// If we've got a working AV1 Main 10-bit decoder, we'll enable the HDR checkbox
// but we will still continue probing to get other attributes for HEVC or H.264
// decoders. See the AV1 comment at the top of the function for more info.
isHdrSupported = decoder - > isHdrSupported ( ) ;
delete decoder ;
}
else {
// HDR can only be supported by a hardware codec that can handle 10-bit video.
// If we made it this far, we don't have one, so HDR will not be available.
isHdrSupported = false ;
}
2020-02-09 19:35:05 +00:00
2022-01-29 06:59:04 +00:00
// Try a regular hardware accelerated HEVC decoder now
if ( chooseDecoder ( StreamingPreferences : : VDS_FORCE_HARDWARE ,
window , VIDEO_FORMAT_H265 , 1920 , 1080 , 60 ,
false , false , true , decoder ) ) {
isHardwareAccelerated = decoder - > isHardwareAccelerated ( ) ;
isFullScreenOnly = decoder - > isAlwaysFullScreen ( ) ;
maxResolution = decoder - > getDecoderMaxResolution ( ) ;
delete decoder ;
2022-01-07 04:07:31 +00:00
2022-01-29 06:59:04 +00:00
return ;
}
2023-07-16 20:38:58 +00:00
#if 0 // See AV1 comment at the top of this function
if ( chooseDecoder ( StreamingPreferences : : VDS_FORCE_HARDWARE ,
window , VIDEO_FORMAT_AV1_MAIN8 , 1920 , 1080 , 60 ,
false , false , true , decoder ) ) {
isHardwareAccelerated = decoder - > isHardwareAccelerated ( ) ;
isFullScreenOnly = decoder - > isAlwaysFullScreen ( ) ;
maxResolution = decoder - > getDecoderMaxResolution ( ) ;
delete decoder ;
return ;
}
# endif
2022-01-29 06:59:04 +00:00
// If we still didn't find a hardware decoder, try H.264 now.
// This will fall back to software decoding, so it should always work.
if ( chooseDecoder ( StreamingPreferences : : VDS_AUTO ,
window , VIDEO_FORMAT_H264 , 1920 , 1080 , 60 ,
false , false , true , decoder ) ) {
isHardwareAccelerated = decoder - > isHardwareAccelerated ( ) ;
isFullScreenOnly = decoder - > isAlwaysFullScreen ( ) ;
maxResolution = decoder - > getDecoderMaxResolution ( ) ;
delete decoder ;
return ;
2022-01-07 04:07:31 +00:00
}
2022-01-29 06:59:04 +00:00
SDL_LogError ( SDL_LOG_CATEGORY_APPLICATION ,
" Failed to find ANY working H.264 or HEVC decoder! " ) ;
2020-02-09 19:35:05 +00:00
}
2019-03-24 00:46:42 +00:00
bool Session : : isHardwareDecodeAvailable ( SDL_Window * window ,
StreamingPreferences : : VideoDecoderSelection vds ,
2018-07-18 03:00:16 +00:00
int videoFormat , int width , int height , int frameRate )
{
IVideoDecoder * decoder ;
2021-04-30 23:12:56 +00:00
if ( ! chooseDecoder ( vds , window , videoFormat , width , height , frameRate , false , false , true , decoder ) ) {
2018-07-18 03:00:16 +00:00
return false ;
}
bool ret = decoder - > isHardwareAccelerated ( ) ;
2018-07-21 01:15:46 +00:00
2018-07-18 03:00:16 +00:00
delete decoder ;
2018-11-01 01:20:39 +00:00
2018-07-18 03:00:16 +00:00
return ret ;
}
2019-12-14 23:20:44 +00:00
bool Session : : populateDecoderProperties ( SDL_Window * window )
2018-08-25 20:36:54 +00:00
{
IVideoDecoder * decoder ;
2023-07-03 05:45:36 +00:00
int videoFormat ;
2023-07-16 20:38:58 +00:00
if ( m_StreamConfig . supportedVideoFormats & VIDEO_FORMAT_AV1_MAIN10 ) {
videoFormat = VIDEO_FORMAT_AV1_MAIN10 ;
}
else if ( m_StreamConfig . supportedVideoFormats & VIDEO_FORMAT_AV1_MAIN8 ) {
videoFormat = VIDEO_FORMAT_AV1_MAIN8 ;
}
else if ( m_StreamConfig . supportedVideoFormats & VIDEO_FORMAT_H265_MAIN10 ) {
2023-07-03 05:45:36 +00:00
videoFormat = VIDEO_FORMAT_H265_MAIN10 ;
}
else if ( m_StreamConfig . supportedVideoFormats & VIDEO_FORMAT_H265 ) {
videoFormat = VIDEO_FORMAT_H265 ;
}
else {
videoFormat = VIDEO_FORMAT_H264 ;
}
2019-12-14 23:20:44 +00:00
if ( ! chooseDecoder ( m_Preferences - > videoDecoderSelection ,
window ,
2023-07-03 05:45:36 +00:00
videoFormat ,
2019-12-14 23:20:44 +00:00
m_StreamConfig . width ,
m_StreamConfig . height ,
m_StreamConfig . fps ,
2021-04-30 23:12:56 +00:00
false , false , true , decoder ) ) {
2018-08-25 20:36:54 +00:00
return false ;
}
2019-12-14 23:34:48 +00:00
m_VideoCallbacks . capabilities = decoder - > getDecoderCapabilities ( ) ;
2022-01-17 21:06:12 +00:00
if ( m_VideoCallbacks . capabilities & CAPABILITY_PULL_RENDERER ) {
// It is an error to pass a push callback when in pull mode
m_VideoCallbacks . submitDecodeUnit = nullptr ;
}
else {
m_VideoCallbacks . submitDecodeUnit = drSubmitDecodeUnit ;
}
2018-08-25 20:36:54 +00:00
2022-02-17 06:26:56 +00:00
{
bool ok ;
m_StreamConfig . colorSpace = qEnvironmentVariableIntValue ( " COLOR_SPACE_OVERRIDE " , & ok ) ;
if ( ok ) {
SDL_LogWarn ( SDL_LOG_CATEGORY_APPLICATION ,
" Using colorspace override: %d " ,
m_StreamConfig . colorSpace ) ;
}
else {
2022-10-13 06:18:57 +00:00
m_StreamConfig . colorSpace = decoder - > getDecoderColorspace ( ) ;
2022-02-17 06:26:56 +00:00
}
m_StreamConfig . colorRange = qEnvironmentVariableIntValue ( " COLOR_RANGE_OVERRIDE " , & ok ) ;
if ( ok ) {
SDL_LogWarn ( SDL_LOG_CATEGORY_APPLICATION ,
" Using color range override: %d " ,
m_StreamConfig . colorRange ) ;
}
else {
2022-10-13 04:59:01 +00:00
m_StreamConfig . colorRange = decoder - > getDecoderColorRange ( ) ;
2022-02-17 06:26:56 +00:00
}
}
2019-12-14 23:25:56 +00:00
2021-02-07 16:49:29 +00:00
if ( decoder - > isAlwaysFullScreen ( ) ) {
m_IsFullScreen = true ;
}
2018-11-01 01:49:37 +00:00
delete decoder ;
2019-12-14 23:20:44 +00:00
return true ;
2018-08-25 20:36:54 +00:00
}
2018-09-29 21:06:55 +00:00
Session : : Session ( NvComputer * computer , NvApp & app , StreamingPreferences * preferences )
: m_Preferences ( preferences ? preferences : new StreamingPreferences ( this ) ) ,
2021-03-13 21:20:58 +00:00
m_IsFullScreen ( m_Preferences - > windowMode ! = StreamingPreferences : : WM_WINDOWED | | ! WMUtils : : isRunningDesktopEnvironment ( ) ) ,
2018-09-29 21:06:55 +00:00
m_Computer ( computer ) ,
2018-07-09 07:09:06 +00:00
m_App ( app ) ,
2018-07-18 03:00:16 +00:00
m_Window ( nullptr ) ,
m_VideoDecoder ( nullptr ) ,
2018-07-20 22:31:57 +00:00
m_DecoderLock ( 0 ) ,
2018-09-05 22:15:53 +00:00
m_AudioDisabled ( false ) ,
2020-12-25 21:32:11 +00:00
m_AudioMuted ( false ) ,
2018-09-05 22:15:53 +00:00
m_DisplayOriginX ( 0 ) ,
2018-09-08 23:01:35 +00:00
m_DisplayOriginY ( 0 ) ,
2019-02-13 03:30:02 +00:00
m_UnexpectedTermination ( true ) , // Failure prior to streaming is unexpected
2019-02-12 05:39:55 +00:00
m_InputHandler ( nullptr ) ,
2019-05-19 20:10:42 +00:00
m_MouseEmulationRefCount ( 0 ) ,
2021-01-15 01:28:21 +00:00
m_FlushingWindowEventsRef ( 0 ) ,
2020-08-11 05:21:54 +00:00
m_AsyncConnectionSuccess ( false ) ,
m_PortTestResults ( 0 ) ,
2018-09-13 13:23:06 +00:00
m_OpusDecoder ( nullptr ) ,
2018-10-02 08:21:42 +00:00
m_AudioRenderer ( nullptr ) ,
2019-02-15 06:32:54 +00:00
m_AudioSampleCount ( 0 ) ,
m_DropAudioEndTime ( 0 )
2018-09-09 17:17:32 +00:00
{
}
2019-03-24 00:46:42 +00:00
bool Session : : initialize ( )
2018-06-28 08:44:43 +00:00
{
2019-04-22 00:43:38 +00:00
if ( SDL_InitSubSystem ( SDL_INIT_VIDEO ) ! = 0 ) {
SDL_LogError ( SDL_LOG_CATEGORY_APPLICATION ,
" SDL_InitSubSystem(SDL_INIT_VIDEO) failed: %s " ,
SDL_GetError ( ) ) ;
return false ;
}
2019-03-24 00:46:42 +00:00
// Create a hidden window to use for decoder initialization tests
2020-09-05 21:06:56 +00:00
SDL_Window * testWindow = SDL_CreateWindow ( " " , 0 , 0 , 1280 , 720 ,
SDL_WINDOW_HIDDEN | StreamUtils : : getPlatformWindowFlags ( ) ) ;
2019-03-24 00:46:42 +00:00
if ( ! testWindow ) {
2020-09-05 21:06:56 +00:00
SDL_LogWarn ( SDL_LOG_CATEGORY_APPLICATION ,
" Failed to create test window with platform flags: %s " ,
SDL_GetError ( ) ) ;
testWindow = SDL_CreateWindow ( " " , 0 , 0 , 1280 , 720 , SDL_WINDOW_HIDDEN ) ;
if ( ! testWindow ) {
SDL_LogError ( SDL_LOG_CATEGORY_APPLICATION ,
" Failed to create window for hardware decode test: %s " ,
SDL_GetError ( ) ) ;
SDL_QuitSubSystem ( SDL_INIT_VIDEO ) ;
return false ;
}
2019-03-24 00:46:42 +00:00
}
2018-09-15 01:35:28 +00:00
qInfo ( ) < < " Server GPU: " < < m_Computer - > gpuModel ;
qInfo ( ) < < " Server GFE version: " < < m_Computer - > gfeVersion ;
2018-08-05 20:32:04 +00:00
2018-07-08 04:52:20 +00:00
LiInitializeVideoCallbacks ( & m_VideoCallbacks ) ;
m_VideoCallbacks . setup = drSetup ;
2018-07-18 03:00:16 +00:00
2018-06-28 08:44:43 +00:00
LiInitializeStreamConfiguration ( & m_StreamConfig ) ;
2018-09-29 21:06:55 +00:00
m_StreamConfig . width = m_Preferences - > width ;
m_StreamConfig . height = m_Preferences - > height ;
m_StreamConfig . fps = m_Preferences - > fps ;
m_StreamConfig . bitrate = m_Preferences - > bitrateKbps ;
2018-06-28 08:44:43 +00:00
m_StreamConfig . hevcBitratePercentageMultiplier = 75 ;
2023-10-15 17:51:23 +00:00
m_StreamConfig . av1BitratePercentageMultiplier = 75 ;
2021-04-24 00:49:03 +00:00
# ifndef STEAM_LINK
// Enable audio encryption as long as we're not on Steam Link.
// That hardware can hardly handle Opus decoding at all.
m_StreamConfig . encryptionFlags = ENCFLG_AUDIO ;
# endif
2018-08-05 20:32:04 +00:00
SDL_LogInfo ( SDL_LOG_CATEGORY_APPLICATION ,
" Video bitrate: %d kbps " ,
m_StreamConfig . bitrate ) ;
2018-07-22 01:47:41 +00:00
RAND_bytes ( reinterpret_cast < unsigned char * > ( m_StreamConfig . remoteInputAesKey ) ,
sizeof ( m_StreamConfig . remoteInputAesKey ) ) ;
// Only the first 4 bytes are populated in the RI key IV
RAND_bytes ( reinterpret_cast < unsigned char * > ( m_StreamConfig . remoteInputAesIv ) , 4 ) ;
2018-09-29 21:06:55 +00:00
switch ( m_Preferences - > audioConfig )
2018-06-28 08:44:43 +00:00
{
2018-10-02 22:30:22 +00:00
case StreamingPreferences : : AC_STEREO :
2020-04-03 07:12:52 +00:00
m_StreamConfig . audioConfiguration = AUDIO_CONFIGURATION_STEREO ;
2018-06-28 08:44:43 +00:00
break ;
2018-10-02 22:30:22 +00:00
case StreamingPreferences : : AC_51_SURROUND :
2020-04-03 07:12:52 +00:00
m_StreamConfig . audioConfiguration = AUDIO_CONFIGURATION_51_SURROUND ;
break ;
case StreamingPreferences : : AC_71_SURROUND :
m_StreamConfig . audioConfiguration = AUDIO_CONFIGURATION_71_SURROUND ;
2018-06-28 08:44:43 +00:00
break ;
}
2018-07-13 09:28:10 +00:00
2019-07-26 16:50:45 +00:00
LiInitializeAudioCallbacks ( & m_AudioCallbacks ) ;
m_AudioCallbacks . init = arInit ;
m_AudioCallbacks . cleanup = arCleanup ;
m_AudioCallbacks . decodeAndPlaySample = arDecodeAndPlaySample ;
m_AudioCallbacks . capabilities = getAudioRendererCapabilities ( m_StreamConfig . audioConfiguration ) ;
2018-07-28 23:06:26 +00:00
SDL_LogInfo ( SDL_LOG_CATEGORY_APPLICATION ,
2020-04-04 19:53:03 +00:00
" Audio channel count: %d " ,
CHANNEL_COUNT_FROM_AUDIO_CONFIGURATION ( m_StreamConfig . audioConfiguration ) ) ;
SDL_LogInfo ( SDL_LOG_CATEGORY_APPLICATION ,
" Audio channel mask: %X " ,
CHANNEL_MASK_FROM_AUDIO_CONFIGURATION ( m_StreamConfig . audioConfiguration ) ) ;
2018-07-28 23:06:26 +00:00
2023-07-03 05:45:36 +00:00
// H.264 is always supported
m_StreamConfig . supportedVideoFormats = VIDEO_FORMAT_H264 ;
2018-09-29 21:06:55 +00:00
switch ( m_Preferences - > videoCodecConfig )
2018-06-28 08:44:43 +00:00
{
case StreamingPreferences : : VCC_AUTO :
2023-07-16 20:38:58 +00:00
#if 0
// TODO: Determine if AV1 is better depending on the decoder
if ( m_Preferences - > enableHdr & & isHardwareDecodeAvailable ( testWindow ,
m_Preferences - > videoDecoderSelection ,
VIDEO_FORMAT_AV1_MAIN10 ,
m_StreamConfig . width ,
m_StreamConfig . height ,
m_StreamConfig . fps ) ) {
m_StreamConfig . supportedVideoFormats | = VIDEO_FORMAT_AV1_MAIN8 | VIDEO_FORMAT_AV1_MAIN10 ;
}
else if ( isHardwareDecodeAvailable ( testWindow ,
m_Preferences - > videoDecoderSelection ,
VIDEO_FORMAT_AV1_MAIN8 ,
m_StreamConfig . width ,
m_StreamConfig . height ,
m_StreamConfig . fps ) ) {
m_StreamConfig . supportedVideoFormats | = VIDEO_FORMAT_AV1_MAIN8 ;
}
# endif
2018-06-28 08:44:43 +00:00
// TODO: Determine if HEVC is better depending on the decoder
2023-07-16 20:38:58 +00:00
if ( m_Preferences - > enableHdr & & isHardwareDecodeAvailable ( testWindow ,
m_Preferences - > videoDecoderSelection ,
VIDEO_FORMAT_H265_MAIN10 ,
m_StreamConfig . width ,
m_StreamConfig . height ,
m_StreamConfig . fps ) ) {
m_StreamConfig . supportedVideoFormats | = VIDEO_FORMAT_H265 | VIDEO_FORMAT_H265_MAIN10 ;
}
else if ( isHardwareDecodeAvailable ( testWindow ,
m_Preferences - > videoDecoderSelection ,
VIDEO_FORMAT_H265 ,
m_StreamConfig . width ,
m_StreamConfig . height ,
m_StreamConfig . fps ) ) {
2023-07-03 05:45:36 +00:00
m_StreamConfig . supportedVideoFormats | = VIDEO_FORMAT_H265 ;
}
2018-10-14 18:28:52 +00:00
# ifdef Q_OS_DARWIN
2018-10-13 00:59:53 +00:00
{
// Prior to GFE 3.11, GFE did not allow us to constrain
// the number of reference frames, so we have to fixup the SPS
// to allow decoding via VideoToolbox on macOS. Since we don't
// have fixup code for HEVC, just avoid it if GFE is too old.
QVector < int > gfeVersion = NvHTTP : : parseQuad ( m_Computer - > gfeVersion ) ;
if ( gfeVersion . isEmpty ( ) | | // Very old versions don't have GfeVersion at all
gfeVersion [ 0 ] < 3 | |
( gfeVersion [ 0 ] = = 3 & & gfeVersion [ 1 ] < 11 ) ) {
SDL_LogWarn ( SDL_LOG_CATEGORY_APPLICATION ,
" Disabling HEVC on macOS due to old GFE version " ) ;
2023-07-16 20:38:58 +00:00
m_StreamConfig . supportedVideoFormats & = ~ VIDEO_FORMAT_MASK_H265 ;
2018-10-13 00:59:53 +00:00
}
}
2018-10-14 18:28:52 +00:00
# endif
2018-06-28 08:44:43 +00:00
break ;
case StreamingPreferences : : VCC_FORCE_H264 :
break ;
case StreamingPreferences : : VCC_FORCE_HEVC :
2023-07-16 20:38:58 +00:00
case StreamingPreferences : : VCC_FORCE_HEVC_HDR_DEPRECATED :
2023-07-03 05:45:36 +00:00
m_StreamConfig . supportedVideoFormats | = VIDEO_FORMAT_H265 ;
2023-07-16 20:38:58 +00:00
if ( m_Preferences - > enableHdr ) {
m_StreamConfig . supportedVideoFormats | = VIDEO_FORMAT_H265_MAIN10 ;
}
2018-06-28 08:44:43 +00:00
break ;
2023-07-16 20:38:58 +00:00
case StreamingPreferences : : VCC_FORCE_AV1 :
m_StreamConfig . supportedVideoFormats | = VIDEO_FORMAT_AV1_MAIN8 ;
if ( m_Preferences - > enableHdr ) {
m_StreamConfig . supportedVideoFormats | = VIDEO_FORMAT_AV1_MAIN10 ;
}
2018-06-28 08:44:43 +00:00
break ;
}
2018-07-16 08:12:53 +00:00
2018-09-29 21:06:55 +00:00
switch ( m_Preferences - > windowMode )
2018-09-04 02:17:34 +00:00
{
2019-06-23 20:29:57 +00:00
default :
2018-09-04 02:17:34 +00:00
case StreamingPreferences : : WM_FULLSCREEN_DESKTOP :
2021-03-13 21:20:58 +00:00
// Only use full-screen desktop mode if we're running a desktop environment
if ( WMUtils : : isRunningDesktopEnvironment ( ) ) {
2021-02-07 16:49:29 +00:00
m_FullScreenFlag = SDL_WINDOW_FULLSCREEN_DESKTOP ;
break ;
}
// Fall-through
2018-09-04 02:17:34 +00:00
case StreamingPreferences : : WM_FULLSCREEN :
m_FullScreenFlag = SDL_WINDOW_FULLSCREEN ;
break ;
}
2019-03-24 00:46:42 +00:00
2020-02-25 03:55:16 +00:00
# if !SDL_VERSION_ATLEAST(2, 0, 11)
2020-02-09 20:00:30 +00:00
// HACK: Using a full-screen window breaks mouse capture on the Pi's LXDE
// GUI environment. Force the session to use windowed mode (which won't
// really matter anyway because the MMAL renderer always draws full-screen).
if ( qgetenv ( " DESKTOP_SESSION " ) = = " LXDE-pi " ) {
SDL_LogInfo ( SDL_LOG_CATEGORY_APPLICATION ,
" Forcing windowed mode on LXDE-Pi " ) ;
m_FullScreenFlag = 0 ;
}
2020-02-25 03:55:16 +00:00
# endif
2020-02-09 20:00:30 +00:00
2019-03-24 00:46:42 +00:00
// Check for validation errors/warnings and emit
// signals for them, if appropriate
bool ret = validateLaunch ( testWindow ) ;
2019-12-14 23:20:44 +00:00
if ( ret ) {
// Populate decoder-dependent properties.
// Must be done after validateLaunch() since m_StreamConfig is finalized.
ret = populateDecoderProperties ( testWindow ) ;
}
2019-03-24 00:46:42 +00:00
SDL_DestroyWindow ( testWindow ) ;
2019-04-22 00:43:38 +00:00
if ( ! ret ) {
SDL_QuitSubSystem ( SDL_INIT_VIDEO ) ;
return false ;
}
return true ;
2018-06-28 08:44:43 +00:00
}
2018-08-04 23:05:37 +00:00
void Session : : emitLaunchWarning ( QString text )
{
// Emit the warning to the UI
emit displayLaunchWarning ( text ) ;
// Wait a little bit so the user can actually read what we just said.
// This wait is a little longer than the actual toast timeout (3 seconds)
// to allow it to transition off the screen before continuing.
uint32_t start = SDL_GetTicks ( ) ;
while ( ! SDL_TICKS_PASSED ( SDL_GetTicks ( ) , start + 3500 ) ) {
SDL_Delay ( 5 ) ;
2021-03-07 16:06:12 +00:00
if ( ! m_ThreadedExec ) {
// Pump the UI loop while we wait if we're on the main thread
QCoreApplication : : processEvents ( QEventLoop : : ExcludeUserInputEvents ) ;
QCoreApplication : : sendPostedEvents ( ) ;
}
2018-08-04 23:05:37 +00:00
}
}
2019-03-24 00:46:42 +00:00
bool Session : : validateLaunch ( SDL_Window * testWindow )
2018-06-28 08:44:43 +00:00
{
2021-05-01 01:05:38 +00:00
if ( ! m_Computer - > isSupportedServerVersion ) {
emit displayLaunchError ( tr ( " The version of GeForce Experience on %1 is not supported by this build of Moonlight. You must update Moonlight to stream from %1. " ) . arg ( m_Computer - > name ) ) ;
return false ;
}
2020-04-26 06:50:12 +00:00
if ( m_Preferences - > absoluteMouseMode & & ! m_App . isAppCollectorGame ) {
2020-11-21 19:15:54 +00:00
emitLaunchWarning ( tr ( " Your selection to enable remote desktop mouse mode may cause problems in games. " ) ) ;
2020-04-26 06:50:12 +00:00
}
2018-09-29 21:06:55 +00:00
if ( m_Preferences - > videoDecoderSelection = = StreamingPreferences : : VDS_FORCE_SOFTWARE ) {
2023-07-16 20:38:58 +00:00
if ( m_Preferences - > enableHdr ) {
2020-11-21 19:15:54 +00:00
emitLaunchWarning ( tr ( " HDR is not supported with software decoding. " ) ) ;
2023-07-03 05:45:36 +00:00
m_StreamConfig . supportedVideoFormats & = ~ VIDEO_FORMAT_MASK_10BIT ;
2019-12-31 00:05:43 +00:00
}
else {
2020-11-21 19:15:54 +00:00
emitLaunchWarning ( tr ( " Your settings selection to force software decoding may cause poor streaming performance. " ) ) ;
2019-12-31 00:05:43 +00:00
}
2018-08-04 23:15:13 +00:00
}
2023-08-06 03:56:12 +00:00
if ( m_StreamConfig . supportedVideoFormats & VIDEO_FORMAT_MASK_AV1 ) {
if ( ! ( m_Computer - > serverCodecModeSupport & SCM_MASK_AV1 ) ) {
if ( m_Preferences - > videoCodecConfig = = StreamingPreferences : : VCC_FORCE_AV1 ) {
emitLaunchWarning ( tr ( " Your host software or GPU doesn't support encoding AV1. " ) ) ;
}
// We'll try to fall back to HEVC first if AV1 fails. We'd rather not fall back
// straight to H.264 if the user asked for AV1 and the host doesn't support it.
if ( m_StreamConfig . supportedVideoFormats & VIDEO_FORMAT_AV1_MAIN8 ) {
m_StreamConfig . supportedVideoFormats | = VIDEO_FORMAT_H265 ;
}
if ( m_StreamConfig . supportedVideoFormats & VIDEO_FORMAT_AV1_MAIN10 ) {
m_StreamConfig . supportedVideoFormats | = VIDEO_FORMAT_H265_MAIN10 ;
2018-07-27 02:26:22 +00:00
}
2022-01-07 04:08:43 +00:00
// Moonlight-common-c will handle this case already, but we want
// to set this explicitly here so we can do our hardware acceleration
// check below.
2023-08-06 03:56:12 +00:00
m_StreamConfig . supportedVideoFormats & = ~ VIDEO_FORMAT_MASK_AV1 ;
2018-07-13 09:28:10 +00:00
}
2022-01-07 05:15:19 +00:00
else if ( m_Preferences - > videoDecoderSelection = = StreamingPreferences : : VDS_AUTO & & // Force hardware decoding checked below
2023-08-06 03:56:12 +00:00
m_Preferences - > videoCodecConfig ! = StreamingPreferences : : VCC_AUTO & & // Auto VCC is already checked in initialize()
! isHardwareDecodeAvailable ( testWindow ,
m_Preferences - > videoDecoderSelection ,
VIDEO_FORMAT_AV1_MAIN8 ,
m_StreamConfig . width ,
m_StreamConfig . height ,
m_StreamConfig . fps ) ) {
emitLaunchWarning ( tr ( " Using software decoding due to your selection to force AV1 without GPU support. This may cause poor streaming performance. " ) ) ;
2022-01-07 05:15:19 +00:00
}
}
2023-08-06 03:56:12 +00:00
if ( m_StreamConfig . supportedVideoFormats & VIDEO_FORMAT_MASK_H265 ) {
if ( m_Computer - > maxLumaPixelsHEVC = = 0 ) {
if ( m_Preferences - > videoCodecConfig = = StreamingPreferences : : VCC_FORCE_HEVC ) {
emitLaunchWarning ( tr ( " Your host PC doesn't support encoding HEVC. " ) ) ;
2023-07-16 20:38:58 +00:00
}
// Moonlight-common-c will handle this case already, but we want
// to set this explicitly here so we can do our hardware acceleration
// check below.
2023-08-06 03:56:12 +00:00
m_StreamConfig . supportedVideoFormats & = ~ VIDEO_FORMAT_MASK_H265 ;
2023-07-16 20:38:58 +00:00
}
else if ( m_Preferences - > videoDecoderSelection = = StreamingPreferences : : VDS_AUTO & & // Force hardware decoding checked below
2023-08-06 03:56:12 +00:00
m_Preferences - > videoCodecConfig ! = StreamingPreferences : : VCC_AUTO & & // Auto VCC is already checked in initialize()
2023-07-16 20:38:58 +00:00
! isHardwareDecodeAvailable ( testWindow ,
m_Preferences - > videoDecoderSelection ,
2023-08-06 03:56:12 +00:00
VIDEO_FORMAT_H265 ,
2023-07-16 20:38:58 +00:00
m_StreamConfig . width ,
m_StreamConfig . height ,
m_StreamConfig . fps ) ) {
2023-08-06 03:56:12 +00:00
emitLaunchWarning ( tr ( " Using software decoding due to your selection to force HEVC without GPU support. This may cause poor streaming performance. " ) ) ;
2023-07-16 20:38:58 +00:00
}
}
2023-07-03 05:45:36 +00:00
if ( ! ( m_StreamConfig . supportedVideoFormats & VIDEO_FORMAT_MASK_H265 ) & &
2022-01-07 05:15:19 +00:00
m_Preferences - > videoDecoderSelection = = StreamingPreferences : : VDS_AUTO & &
! isHardwareDecodeAvailable ( testWindow ,
m_Preferences - > videoDecoderSelection ,
VIDEO_FORMAT_H264 ,
m_StreamConfig . width ,
m_StreamConfig . height ,
m_StreamConfig . fps ) ) {
if ( m_Preferences - > videoCodecConfig = = StreamingPreferences : : VCC_FORCE_H264 ) {
emitLaunchWarning ( tr ( " Using software decoding due to your selection to force H.264 without GPU support. This may cause poor streaming performance. " ) ) ;
}
else {
if ( m_Computer - > maxLumaPixelsHEVC = = 0 & &
isHardwareDecodeAvailable ( testWindow ,
m_Preferences - > videoDecoderSelection ,
VIDEO_FORMAT_H265 ,
m_StreamConfig . width ,
m_StreamConfig . height ,
m_StreamConfig . fps ) ) {
emitLaunchWarning ( tr ( " Your host PC and client PC don't support the same video codecs. This may cause poor streaming performance. " ) ) ;
}
else {
emitLaunchWarning ( tr ( " Your client GPU doesn't support H.264 decoding. This may cause poor streaming performance. " ) ) ;
}
}
2018-07-09 03:53:24 +00:00
}
2023-07-16 20:38:58 +00:00
if ( m_Preferences - > enableHdr ) {
2018-06-28 08:44:43 +00:00
// Check that the server GPU supports HDR
2023-07-16 20:38:58 +00:00
if ( ! ( m_Computer - > serverCodecModeSupport & SCM_MASK_10BIT ) ) {
2023-03-18 19:20:51 +00:00
emitLaunchWarning ( tr ( " Your host PC doesn't support HDR streaming. " ) ) ;
2023-07-03 05:45:36 +00:00
m_StreamConfig . supportedVideoFormats & = ~ VIDEO_FORMAT_MASK_10BIT ;
2018-06-28 08:44:43 +00:00
}
2023-07-16 20:38:58 +00:00
else if ( m_Preferences - > videoCodecConfig = = StreamingPreferences : : VCC_FORCE_H264 ) {
emitLaunchWarning ( tr ( " HDR is not supported using the H.264 codec. " ) ) ;
2023-07-03 05:45:36 +00:00
m_StreamConfig . supportedVideoFormats & = ~ VIDEO_FORMAT_MASK_10BIT ;
2018-07-13 09:28:10 +00:00
}
2023-07-16 20:38:58 +00:00
else if ( m_Preferences - > videoCodecConfig ! = StreamingPreferences : : VCC_AUTO ) { // Auto was already checked during init
// Check that the available HDR-capable codecs on the client and server are compatible
if ( ( m_Computer - > serverCodecModeSupport & SCM_AV1_MAIN10 ) & & ( m_StreamConfig . supportedVideoFormats & VIDEO_FORMAT_AV1_MAIN10 ) ) {
if ( ! isHardwareDecodeAvailable ( testWindow ,
m_Preferences - > videoDecoderSelection ,
VIDEO_FORMAT_AV1_MAIN10 ,
m_StreamConfig . width ,
m_StreamConfig . height ,
m_StreamConfig . fps ) ) {
emitLaunchWarning ( tr ( " This PC's GPU doesn't support AV1 Main10 decoding for HDR streaming. " ) ) ;
m_StreamConfig . supportedVideoFormats & = ~ VIDEO_FORMAT_AV1_MAIN10 ;
}
}
if ( ( m_Computer - > serverCodecModeSupport & SCM_HEVC_MAIN10 ) & & ( m_StreamConfig . supportedVideoFormats & VIDEO_FORMAT_H265_MAIN10 ) ) {
if ( ! isHardwareDecodeAvailable ( testWindow ,
m_Preferences - > videoDecoderSelection ,
VIDEO_FORMAT_H265_MAIN10 ,
m_StreamConfig . width ,
m_StreamConfig . height ,
m_StreamConfig . fps ) ) {
emitLaunchWarning ( tr ( " This PC's GPU doesn't support HEVC Main10 decoding for HDR streaming. " ) ) ;
m_StreamConfig . supportedVideoFormats & = ~ VIDEO_FORMAT_H265_MAIN10 ;
}
}
}
else if ( ! ( m_StreamConfig . supportedVideoFormats & VIDEO_FORMAT_MASK_10BIT ) ) {
emitLaunchWarning ( tr ( " This PC's GPU doesn't support 10-bit HEVC or AV1 decoding for HDR streaming. " ) ) ;
}
// Check for compatibility between server and client codecs
if ( ( m_StreamConfig . supportedVideoFormats & VIDEO_FORMAT_MASK_10BIT ) & & // Ignore this check if we already failed one above
! ( ( ( m_StreamConfig . supportedVideoFormats & VIDEO_FORMAT_H265_MAIN10 ) & & ( m_Computer - > serverCodecModeSupport & SCM_HEVC_MAIN10 ) ) | |
( ( m_StreamConfig . supportedVideoFormats & VIDEO_FORMAT_AV1_MAIN10 ) & & ( m_Computer - > serverCodecModeSupport & SCM_AV1_MAIN10 ) ) ) ) {
emitLaunchWarning ( tr ( " Your host PC and client PC don't support the same HDR video codecs. " ) ) ;
m_StreamConfig . supportedVideoFormats & = ~ VIDEO_FORMAT_MASK_10BIT ;
2018-06-28 08:44:43 +00:00
}
}
if ( m_StreamConfig . width > = 3840 ) {
2018-07-09 03:53:24 +00:00
// Only allow 4K on GFE 3.x+
2018-09-15 01:36:15 +00:00
if ( m_Computer - > gfeVersion . isEmpty ( ) | | m_Computer - > gfeVersion . startsWith ( " 2. " ) ) {
2020-11-21 19:15:54 +00:00
emitLaunchWarning ( tr ( " GeForce Experience 3.0 or higher is required for 4K streaming. " ) ) ;
2018-07-09 03:53:24 +00:00
m_StreamConfig . width = 1920 ;
m_StreamConfig . height = 1080 ;
2018-06-28 08:44:43 +00:00
}
}
2018-09-22 23:04:36 +00:00
// Test if audio works at the specified audio configuration
bool audioTestPassed = testAudio ( m_StreamConfig . audioConfiguration ) ;
2020-04-01 04:36:16 +00:00
// Gracefully degrade to stereo if surround sound doesn't work
if ( ! audioTestPassed & & CHANNEL_COUNT_FROM_AUDIO_CONFIGURATION ( m_StreamConfig . audioConfiguration ) > 2 ) {
2020-04-03 07:12:52 +00:00
audioTestPassed = testAudio ( AUDIO_CONFIGURATION_STEREO ) ;
2018-09-22 23:04:36 +00:00
if ( audioTestPassed ) {
2020-04-03 07:12:52 +00:00
m_StreamConfig . audioConfiguration = AUDIO_CONFIGURATION_STEREO ;
2020-11-21 19:15:54 +00:00
emitLaunchWarning ( tr ( " Your selected surround sound setting is not supported by the current audio device. " ) ) ;
2018-09-22 23:04:36 +00:00
}
}
// If nothing worked, warn the user that audio will not work
m_AudioDisabled = ! audioTestPassed ;
2018-08-31 04:09:31 +00:00
if ( m_AudioDisabled ) {
2020-11-21 19:15:54 +00:00
emitLaunchWarning ( tr ( " Failed to open audio device. Audio will be unavailable during this session. " ) ) ;
2018-08-31 04:09:31 +00:00
}
2018-09-30 02:14:52 +00:00
// Check for unmapped gamepads
if ( ! SdlInputHandler : : getUnmappedGamepads ( ) . isEmpty ( ) ) {
2020-11-21 19:15:54 +00:00
emitLaunchWarning ( tr ( " An attached gamepad has no mapping and won't be usable. Visit the Moonlight help to resolve this. " ) ) ;
2018-09-30 02:14:52 +00:00
}
2022-10-25 06:24:57 +00:00
// NVENC will fail to initialize when any dimension exceeds 4096 using:
// - H.264 on all versions of NVENC
// - HEVC prior to Pascal
2022-10-25 06:18:10 +00:00
//
2022-10-25 06:24:57 +00:00
// However, if we aren't using Nvidia hosting software, don't assume anything about
// encoding capabilities by using HEVC Main 10 support. It will likely be wrong.
2022-10-25 06:18:10 +00:00
if ( ( m_StreamConfig . width > 4096 | | m_StreamConfig . height > 4096 ) & & m_Computer - > isNvidiaServerSoftware ) {
2020-12-23 00:57:45 +00:00
// Pascal added support for 8K HEVC encoding support. Maxwell 2 could encode HEVC but only up to 4K.
// We can't directly identify Pascal, but we can look for HEVC Main10 which was added in the same generation.
2023-07-16 20:38:58 +00:00
if ( m_Computer - > maxLumaPixelsHEVC = = 0 | | ! ( m_Computer - > serverCodecModeSupport & SCM_HEVC_MAIN10 ) ) {
2020-11-21 19:15:54 +00:00
emit displayLaunchError ( tr ( " Your host PC's GPU doesn't support streaming video resolutions over 4K. " ) ) ;
2020-04-04 19:47:44 +00:00
return false ;
}
2023-07-03 05:45:36 +00:00
else if ( ( m_StreamConfig . supportedVideoFormats & ~ VIDEO_FORMAT_MASK_H264 ) = = 0 ) {
emit displayLaunchError ( tr ( " Video resolutions over 4K are not supported by the H.264 codec. " ) ) ;
2020-04-04 19:47:44 +00:00
return false ;
}
}
2018-09-29 21:06:55 +00:00
if ( m_Preferences - > videoDecoderSelection = = StreamingPreferences : : VDS_FORCE_HARDWARE & &
2023-07-03 05:45:36 +00:00
! ( m_StreamConfig . supportedVideoFormats & VIDEO_FORMAT_MASK_10BIT ) & & // HDR was already checked for hardware decode support above
2019-03-24 00:46:42 +00:00
! isHardwareDecodeAvailable ( testWindow ,
m_Preferences - > videoDecoderSelection ,
2023-07-16 20:38:58 +00:00
( m_StreamConfig . supportedVideoFormats & VIDEO_FORMAT_MASK_AV1 ) ? VIDEO_FORMAT_AV1_MAIN8 :
( ( m_StreamConfig . supportedVideoFormats & VIDEO_FORMAT_MASK_H265 ) ? VIDEO_FORMAT_H265 : VIDEO_FORMAT_H264 ) ,
2018-08-10 02:37:49 +00:00
m_StreamConfig . width ,
m_StreamConfig . height ,
m_StreamConfig . fps ) ) {
2018-09-29 21:06:55 +00:00
if ( m_Preferences - > videoCodecConfig = = StreamingPreferences : : VCC_AUTO ) {
2020-11-21 19:15:54 +00:00
emit displayLaunchError ( tr ( " Your selection to force hardware decoding cannot be satisfied due to missing hardware decoding support on this PC's GPU. " ) ) ;
2018-08-10 02:37:49 +00:00
}
else {
2020-11-21 19:15:54 +00:00
emit displayLaunchError ( tr ( " Your codec selection and force hardware decoding setting are not compatible. This PC's GPU lacks support for decoding your chosen codec. " ) ) ;
2018-08-10 02:37:49 +00:00
}
// Fail the launch, because we won't manage to get a decoder for the actual stream
return false ;
}
2018-07-07 23:30:26 +00:00
return true ;
2018-06-28 08:44:43 +00:00
}
2018-07-09 07:09:06 +00:00
class DeferredSessionCleanupTask : public QRunnable
{
2018-12-06 03:49:06 +00:00
public :
DeferredSessionCleanupTask ( Session * session ) :
m_Session ( session ) { }
private :
virtual ~ DeferredSessionCleanupTask ( ) override
{
// Allow another session to start now that we're cleaned up
Session : : s_ActiveSession = nullptr ;
Session : : s_ActiveSessionSemaphore . release ( ) ;
2022-03-06 19:11:36 +00:00
// Notify that the session is ready to be cleaned up
emit m_Session - > readyForDeletion ( ) ;
2018-12-06 03:49:06 +00:00
}
2018-07-09 07:09:06 +00:00
void run ( ) override
{
2018-12-06 04:17:00 +00:00
// Only quit the running app if our session terminated gracefully
bool shouldQuit =
2018-12-06 06:17:26 +00:00
! m_Session - > m_UnexpectedTermination & &
2018-12-06 04:17:00 +00:00
m_Session - > m_Preferences - > quitAppAfter ;
2018-12-06 03:49:06 +00:00
// Notify the UI
2018-12-06 04:17:00 +00:00
if ( shouldQuit ) {
emit m_Session - > quitStarting ( ) ;
2018-12-06 03:49:06 +00:00
}
else {
2020-08-11 05:21:54 +00:00
emit m_Session - > sessionFinished ( m_Session - > m_PortTestResults ) ;
2018-12-06 03:49:06 +00:00
}
2022-01-19 00:19:28 +00:00
// The video decoder must already be destroyed, since it could
// try to interact with APIs that can only be called between
// LiStartConnection() and LiStopConnection().
SDL_assert ( m_Session - > m_VideoDecoder = = nullptr ) ;
2018-07-09 07:09:06 +00:00
// Finish cleanup of the connection state
LiStopConnection ( ) ;
2018-12-06 03:49:06 +00:00
// Perform a best-effort app quit
2018-12-06 04:17:00 +00:00
if ( shouldQuit ) {
2021-07-02 22:14:48 +00:00
NvHTTP http ( m_Session - > m_Computer ) ;
2018-12-06 03:49:06 +00:00
// Logging is already done inside NvHTTP
try {
http . quitApp ( ) ;
} catch ( const GfeHttpResponseException & ) {
} catch ( const QtNetworkReplyException & ) {
}
// Session is finished now
2020-08-11 05:21:54 +00:00
emit m_Session - > sessionFinished ( m_Session - > m_PortTestResults ) ;
2018-12-06 03:49:06 +00:00
}
2018-07-09 07:09:06 +00:00
}
2018-12-06 03:49:06 +00:00
Session * m_Session ;
2018-07-09 07:09:06 +00:00
} ;
2018-09-05 22:45:36 +00:00
void Session : : getWindowDimensions ( int & x , int & y ,
2018-07-20 23:01:22 +00:00
int & width , int & height )
{
int displayIndex = 0 ;
2018-08-17 05:25:14 +00:00
if ( m_Window ! = nullptr ) {
displayIndex = SDL_GetWindowDisplayIndex ( m_Window ) ;
SDL_assert ( displayIndex > = 0 ) ;
}
2018-09-05 22:15:53 +00:00
// Create our window on the same display that Qt's UI
// was being displayed on.
else {
SDL_LogInfo ( SDL_LOG_CATEGORY_APPLICATION ,
" Qt UI screen is at (%d,%d) " ,
m_DisplayOriginX , m_DisplayOriginY ) ;
2018-08-05 21:55:26 +00:00
for ( int i = 0 ; i < SDL_GetNumVideoDisplays ( ) ; i + + ) {
2018-09-05 22:15:53 +00:00
SDL_Rect displayBounds ;
if ( SDL_GetDisplayBounds ( i , & displayBounds ) = = 0 ) {
if ( displayBounds . x = = m_DisplayOriginX & &
displayBounds . y = = m_DisplayOriginY ) {
SDL_LogInfo ( SDL_LOG_CATEGORY_APPLICATION ,
" SDL found matching display %d " ,
i ) ;
displayIndex = i ;
break ;
}
}
else {
SDL_LogWarn ( SDL_LOG_CATEGORY_APPLICATION ,
" SDL_GetDisplayBounds(%d) failed: %s " ,
i , SDL_GetError ( ) ) ;
2018-08-05 21:55:26 +00:00
}
}
}
2018-08-17 05:25:14 +00:00
SDL_Rect usableBounds ;
2022-10-15 01:24:12 +00:00
if ( SDL_GetDisplayUsableBounds ( displayIndex , & usableBounds ) = = 0 ) {
// Don't use more than 80% of the display to leave room for system UI
2022-10-15 02:09:32 +00:00
// and ensure the target size is not odd (otherwise one of the sides
// of the image will have a one-pixel black bar next to it).
2022-10-15 01:24:12 +00:00
SDL_Rect src , dst ;
src . x = src . y = dst . x = dst . y = 0 ;
src . w = m_StreamConfig . width ;
src . h = m_StreamConfig . height ;
2022-10-15 20:36:09 +00:00
dst . w = ( ( int ) SDL_ceilf ( usableBounds . w * 0.80f ) & ~ 0x1 ) ;
dst . h = ( ( int ) SDL_ceilf ( usableBounds . h * 0.80f ) & ~ 0x1 ) ;
2022-10-15 01:24:12 +00:00
// Scale the window size while preserving aspect ratio
StreamUtils : : scaleSourceToDestinationSurface ( & src , & dst ) ;
// If the stream window can fit within the usable drawing area with 1:1
// scaling, do that rather than filling the screen.
if ( m_StreamConfig . width < dst . w & & m_StreamConfig . height < dst . h ) {
width = m_StreamConfig . width ;
height = m_StreamConfig . height ;
}
else {
width = dst . w ;
height = dst . h ;
2018-07-20 23:01:22 +00:00
}
2018-08-17 05:25:14 +00:00
}
else {
SDL_LogError ( SDL_LOG_CATEGORY_APPLICATION ,
" SDL_GetDisplayUsableBounds() failed: %s " ,
SDL_GetError ( ) ) ;
2018-07-20 23:01:22 +00:00
2018-08-17 05:25:14 +00:00
width = m_StreamConfig . width ;
height = m_StreamConfig . height ;
2018-07-20 23:01:22 +00:00
}
2018-09-07 01:53:00 +00:00
x = y = SDL_WINDOWPOS_CENTERED_DISPLAY ( displayIndex ) ;
2018-07-20 23:01:22 +00:00
}
2018-09-04 02:54:41 +00:00
void Session : : updateOptimalWindowDisplayMode ( )
{
SDL_DisplayMode desktopMode , bestMode , mode ;
int displayIndex = SDL_GetWindowDisplayIndex ( m_Window ) ;
2018-09-07 08:12:18 +00:00
// Try the current display mode first. On macOS, this will be the normal
// scaled desktop resolution setting.
if ( SDL_GetDesktopDisplayMode ( displayIndex , & desktopMode ) = = 0 ) {
// If this doesn't fit the selected resolution, use the native
// resolution of the panel (unscaled).
if ( desktopMode . w < m_ActiveVideoWidth | | desktopMode . h < m_ActiveVideoHeight ) {
2022-08-16 05:25:49 +00:00
if ( ! StreamUtils : : getNativeDesktopMode ( displayIndex , & desktopMode ) ) {
2018-09-07 08:12:18 +00:00
return ;
}
}
}
else {
SDL_LogWarn ( SDL_LOG_CATEGORY_APPLICATION ,
" SDL_GetDesktopDisplayMode() failed: %s " ,
SDL_GetError ( ) ) ;
return ;
}
// Start with the native desktop resolution and try to find
// the highest refresh rate that our stream FPS evenly divides.
bestMode = desktopMode ;
bestMode . refresh_rate = 0 ;
for ( int i = 0 ; i < SDL_GetNumDisplayModes ( displayIndex ) ; i + + ) {
if ( SDL_GetDisplayMode ( displayIndex , i , & mode ) = = 0 ) {
if ( mode . w = = desktopMode . w & & mode . h = = desktopMode . h & &
mode . refresh_rate % m_StreamConfig . fps = = 0 ) {
2022-04-24 22:11:10 +00:00
SDL_LogInfo ( SDL_LOG_CATEGORY_APPLICATION ,
" Found display mode with desktop resolution: %dx%dx%d " ,
mode . w , mode . h , mode . refresh_rate ) ;
2018-09-07 08:12:18 +00:00
if ( mode . refresh_rate > bestMode . refresh_rate ) {
bestMode = mode ;
2018-09-04 02:54:41 +00:00
}
}
}
2018-09-07 08:12:18 +00:00
}
2018-09-04 02:54:41 +00:00
2022-04-22 04:57:17 +00:00
// If we didn't find a mode that matched the current resolution and
// had a high enough refresh rate, start looking for lower resolution
// modes that can meet the required refresh rate and minimum video
// resolution. We will also try to pick a display mode that matches
// aspect ratio closest to the video stream.
if ( bestMode . refresh_rate = = 0 ) {
float bestModeAspectRatio = 0 ;
float videoAspectRatio = ( float ) m_ActiveVideoWidth / ( float ) m_ActiveVideoHeight ;
2021-12-19 03:07:35 +00:00
for ( int i = 0 ; i < SDL_GetNumDisplayModes ( displayIndex ) ; i + + ) {
if ( SDL_GetDisplayMode ( displayIndex , i , & mode ) = = 0 ) {
2022-04-22 04:57:17 +00:00
float modeAspectRatio = ( float ) mode . w / ( float ) mode . h ;
2021-12-19 03:07:35 +00:00
if ( mode . w > = m_ActiveVideoWidth & & mode . h > = m_ActiveVideoHeight & &
mode . refresh_rate % m_StreamConfig . fps = = 0 ) {
2022-04-24 22:11:10 +00:00
SDL_LogInfo ( SDL_LOG_CATEGORY_APPLICATION ,
" Found display mode with video resolution: %dx%dx%d " ,
mode . w , mode . h , mode . refresh_rate ) ;
2022-04-22 04:57:17 +00:00
if ( mode . refresh_rate > = bestMode . refresh_rate & &
( bestModeAspectRatio = = 0 | | fabs ( videoAspectRatio - modeAspectRatio ) < = fabs ( videoAspectRatio - bestModeAspectRatio ) ) ) {
2021-12-19 03:07:35 +00:00
bestMode = mode ;
2022-04-22 04:57:17 +00:00
bestModeAspectRatio = modeAspectRatio ;
2021-12-19 03:07:35 +00:00
}
}
}
}
}
2018-09-07 08:12:18 +00:00
if ( bestMode . refresh_rate = = 0 ) {
// We may find no match if the user has moved a 120 FPS
// stream onto a 60 Hz monitor (since no refresh rate can
// divide our FPS setting). We'll stick to the default in
// this case.
SDL_LogWarn ( SDL_LOG_CATEGORY_APPLICATION ,
2022-04-24 22:11:10 +00:00
" No matching display mode found; using desktop mode " ) ;
2018-09-07 08:12:18 +00:00
bestMode = desktopMode ;
}
2018-09-07 02:16:59 +00:00
2018-10-13 03:02:54 +00:00
if ( ( SDL_GetWindowFlags ( m_Window ) & SDL_WINDOW_FULLSCREEN_DESKTOP ) = = SDL_WINDOW_FULLSCREEN ) {
// Only print when the window is actually in full-screen exclusive mode,
// otherwise we're not actually using the mode we've set here
SDL_LogInfo ( SDL_LOG_CATEGORY_APPLICATION ,
" Chosen best display mode: %dx%dx%d " ,
bestMode . w , bestMode . h , bestMode . refresh_rate ) ;
}
2018-09-04 02:54:41 +00:00
2018-09-07 08:12:18 +00:00
SDL_SetWindowDisplayMode ( m_Window , & bestMode ) ;
2018-09-04 02:54:41 +00:00
}
2018-07-20 23:01:22 +00:00
void Session : : toggleFullscreen ( )
{
2018-09-04 02:17:34 +00:00
bool fullScreen = ! ( SDL_GetWindowFlags ( m_Window ) & m_FullScreenFlag ) ;
2018-07-20 23:01:22 +00:00
2023-09-12 02:07:54 +00:00
# if defined(Q_OS_WIN32) || defined(Q_OS_DARWIN)
2022-08-29 01:10:26 +00:00
// Destroy the video decoder before toggling full-screen because D3D9 can try
// to put the window back into full-screen before we've managed to destroy
// the renderer. This leads to excessive flickering and can cause the window
// decorations to get messed up as SDL and D3D9 fight over the window style.
2023-09-12 02:07:54 +00:00
//
// On Apple Silicon Macs, the AVSampleBufferDisplayLayer may cause WindowServer
// to deadlock when transitioning out of fullscreen. Destroy the decoder before
// exiting fullscreen as a workaround. See issue #973.
2022-08-29 01:10:26 +00:00
SDL_AtomicLock ( & m_DecoderLock ) ;
delete m_VideoDecoder ;
m_VideoDecoder = nullptr ;
SDL_AtomicUnlock ( & m_DecoderLock ) ;
# endif
2022-10-15 01:24:12 +00:00
// Actually enter/leave fullscreen
SDL_SetWindowFullscreen ( m_Window , fullScreen ? m_FullScreenFlag : 0 ) ;
2021-02-27 22:47:38 +00:00
2022-10-15 04:01:55 +00:00
# ifdef Q_OS_DARWIN
// SDL on macOS has a bug that causes the window size to be reset to crazy
// large dimensions when exiting out of true fullscreen mode. We can work
// around the issue by manually resetting the position and size here.
if ( ! fullScreen & & m_FullScreenFlag = = SDL_WINDOW_FULLSCREEN ) {
int x , y , width , height ;
getWindowDimensions ( x , y , width , height ) ;
SDL_SetWindowSize ( m_Window , width , height ) ;
SDL_SetWindowPosition ( m_Window , x , y ) ;
}
# endif
2021-02-27 22:47:38 +00:00
// Input handler might need to start/stop keyboard grab after changing modes
m_InputHandler - > updateKeyboardGrabState ( ) ;
2022-05-20 00:14:55 +00:00
// Input handler might need stop/stop mouse grab after changing modes
m_InputHandler - > updatePointerRegionLock ( ) ;
2018-07-20 23:01:22 +00:00
}
2019-05-19 20:10:42 +00:00
void Session : : notifyMouseEmulationMode ( bool enabled )
{
m_MouseEmulationRefCount + = enabled ? 1 : - 1 ;
SDL_assert ( m_MouseEmulationRefCount > = 0 ) ;
// We re-use the status update overlay for mouse mode notification
if ( m_MouseEmulationRefCount > 0 ) {
2023-09-23 15:15:21 +00:00
m_OverlayManager . updateOverlayText ( Overlay : : OverlayStatusUpdate , " Gamepad mouse mode active \n Long press Start to deactivate " ) ;
2019-05-19 20:10:42 +00:00
m_OverlayManager . setOverlayState ( Overlay : : OverlayStatusUpdate , true ) ;
}
else {
m_OverlayManager . setOverlayState ( Overlay : : OverlayStatusUpdate , false ) ;
}
}
2020-08-11 05:21:54 +00:00
class AsyncConnectionStartThread : public QThread
2018-06-28 08:44:43 +00:00
{
2020-08-11 05:21:54 +00:00
public :
AsyncConnectionStartThread ( Session * session ) :
QThread ( nullptr ) ,
2021-10-24 21:22:49 +00:00
m_Session ( session )
{
setObjectName ( " Async Conn Start " ) ;
}
2018-07-07 23:30:26 +00:00
2020-08-11 05:21:54 +00:00
void run ( ) override
{
m_Session - > m_AsyncConnectionSuccess = m_Session - > startConnectionAsync ( ) ;
2018-09-15 02:11:06 +00:00
}
2018-07-07 23:30:26 +00:00
2020-08-11 05:21:54 +00:00
Session * m_Session ;
} ;
2018-06-28 08:44:43 +00:00
2020-08-11 05:21:54 +00:00
// Called in a non-main thread
bool Session : : startConnectionAsync ( )
{
// Wait 1.5 seconds before connecting to let the user
// have time to read any messages present on the segue
SDL_Delay ( 1500 ) ;
2018-06-28 08:44:43 +00:00
// The UI should have ensured the old game was already quit
// if we decide to stream a different game.
Q_ASSERT ( m_Computer - > currentGameId = = 0 | |
m_Computer - > currentGameId = = m_App . id ) ;
2023-09-12 02:25:29 +00:00
bool enableGameOptimizations ;
if ( m_Computer - > isNvidiaServerSoftware ) {
// GFE will set all settings to 720p60 if it doesn't recognize
// the chosen resolution. Avoid that by disabling SOPS when it
// is not streaming a supported resolution.
enableGameOptimizations = false ;
for ( const NvDisplayMode & mode : m_Computer - > displayModes ) {
if ( mode . width = = m_StreamConfig . width & &
mode . height = = m_StreamConfig . height ) {
SDL_LogInfo ( SDL_LOG_CATEGORY_APPLICATION ,
" Found host supported resolution: %dx%d " ,
mode . width , mode . height ) ;
enableGameOptimizations = m_Preferences - > gameOptimizations ;
break ;
}
2019-02-10 04:37:11 +00:00
}
}
2023-09-12 02:25:29 +00:00
else {
// Always send SOPS to Sunshine because we may repurpose the
// option to control whether the display mode is adjusted
enableGameOptimizations = m_Preferences - > gameOptimizations ;
}
2019-02-10 04:37:11 +00:00
2021-07-02 06:51:08 +00:00
QString rtspSessionUrl ;
2018-06-28 09:10:31 +00:00
try {
2021-07-02 22:14:48 +00:00
NvHTTP http ( m_Computer ) ;
2023-02-03 06:36:30 +00:00
http . startApp ( m_Computer - > currentGameId ! = 0 ? " resume " : " launch " ,
m_Computer - > isNvidiaServerSoftware ,
m_App . id , & m_StreamConfig ,
enableGameOptimizations ,
m_Preferences - > playAudioOnHost ,
m_InputHandler - > getAttachedGamepadMask ( ) ,
2023-02-20 22:52:28 +00:00
! m_Preferences - > multiController ,
2023-02-03 06:36:30 +00:00
rtspSessionUrl ) ;
2018-06-28 09:10:31 +00:00
} catch ( const GfeHttpResponseException & e ) {
2023-03-18 19:20:51 +00:00
emit displayLaunchError ( tr ( " Host returned error: %1 " ) . arg ( e . toQString ( ) ) ) ;
2020-08-11 05:21:54 +00:00
return false ;
2018-09-29 10:01:49 +00:00
} catch ( const QtNetworkReplyException & e ) {
emit displayLaunchError ( e . toQString ( ) ) ;
2020-08-11 05:21:54 +00:00
return false ;
2018-06-28 08:44:43 +00:00
}
2021-07-03 04:34:54 +00:00
QByteArray hostnameStr = m_Computer - > activeAddress . address ( ) . toLatin1 ( ) ;
2018-06-28 08:44:43 +00:00
QByteArray siAppVersion = m_Computer - > appVersion . toLatin1 ( ) ;
SERVER_INFORMATION hostInfo ;
hostInfo . address = hostnameStr . data ( ) ;
hostInfo . serverInfoAppVersion = siAppVersion . data ( ) ;
2023-07-03 05:45:36 +00:00
hostInfo . serverCodecModeSupport = m_Computer - > serverCodecModeSupport ;
2018-06-28 08:44:43 +00:00
// Older GFE versions didn't have this field
QByteArray siGfeVersion ;
2018-09-15 01:36:15 +00:00
if ( ! m_Computer - > gfeVersion . isEmpty ( ) ) {
2018-06-28 08:44:43 +00:00
siGfeVersion = m_Computer - > gfeVersion . toLatin1 ( ) ;
}
2018-09-15 01:36:15 +00:00
if ( ! siGfeVersion . isEmpty ( ) ) {
2018-06-28 08:44:43 +00:00
hostInfo . serverInfoGfeVersion = siGfeVersion . data ( ) ;
}
2021-07-02 06:51:08 +00:00
// Older GFE and Sunshine versions didn't have this field
QByteArray rtspSessionUrlStr ;
if ( ! rtspSessionUrl . isEmpty ( ) ) {
rtspSessionUrlStr = rtspSessionUrl . toLatin1 ( ) ;
hostInfo . rtspSessionUrl = rtspSessionUrlStr . data ( ) ;
}
2020-01-22 03:10:10 +00:00
if ( m_Preferences - > packetSize ! = 0 ) {
// Override default packet size and remote streaming detection
// NB: Using STREAM_CFG_AUTO will cap our packet size at 1024 for remote hosts.
m_StreamConfig . streamingRemotely = STREAM_CFG_LOCAL ;
m_StreamConfig . packetSize = m_Preferences - > packetSize ;
SDL_LogWarn ( SDL_LOG_CATEGORY_APPLICATION ,
" Using custom packet size: %d bytes " ,
m_Preferences - > packetSize ) ;
2019-12-02 05:50:36 +00:00
}
else {
2022-11-03 01:35:41 +00:00
// Use 1392 byte video packets by default
m_StreamConfig . packetSize = 1392 ;
// getActiveAddressReachability() does network I/O, so we only attempt to check
// reachability if we've already contacted the PC successfully.
switch ( m_Computer - > getActiveAddressReachability ( ) ) {
case NvComputer : : RI_LAN :
// This address is on-link, so treat it as a local address
// even if it's not in RFC 1918 space or it's an IPv6 address.
m_StreamConfig . streamingRemotely = STREAM_CFG_LOCAL ;
break ;
case NvComputer : : RI_VPN :
// It looks like our route to this PC is over a VPN, so cap at 1024 bytes.
2020-01-22 03:10:10 +00:00
// Treat it as remote even if the target address is in RFC 1918 address space.
m_StreamConfig . streamingRemotely = STREAM_CFG_REMOTE ;
m_StreamConfig . packetSize = 1024 ;
2022-11-03 01:35:41 +00:00
break ;
default :
// If we don't have reachability info, let moonlight-common-c decide.
2020-01-22 03:10:10 +00:00
m_StreamConfig . streamingRemotely = STREAM_CFG_AUTO ;
2022-11-03 01:35:41 +00:00
break ;
2020-01-22 03:10:10 +00:00
}
2019-12-02 05:50:36 +00:00
}
2018-06-28 08:44:43 +00:00
int err = LiStartConnection ( & hostInfo , & m_StreamConfig , & k_ConnCallbacks ,
2018-08-31 04:09:31 +00:00
& m_VideoCallbacks ,
2019-06-23 19:49:37 +00:00
m_AudioDisabled ? nullptr : & m_AudioCallbacks ,
2018-06-28 08:44:43 +00:00
NULL , 0 , NULL , 0 ) ;
if ( err ! = 0 ) {
// We already displayed an error dialog in the stage failure
// listener.
2020-08-11 05:21:54 +00:00
return false ;
}
emit connectionStarted ( ) ;
return true ;
}
2021-01-09 23:51:25 +00:00
void Session : : flushWindowEvents ( )
{
// Pump events to ensure all pending OS events are posted
SDL_PumpEvents ( ) ;
// Insert a barrier to discard any additional window events.
// We don't use SDL_FlushEvent() here because it could cause
// important events to be lost.
2021-01-15 01:28:21 +00:00
m_FlushingWindowEventsRef + + ;
2021-01-09 23:51:25 +00:00
// This event will cause us to set m_FlushingWindowEvents back to false.
SDL_Event flushEvent = { } ;
flushEvent . type = SDL_USEREVENT ;
flushEvent . user . code = SDL_CODE_FLUSH_WINDOW_EVENT_BARRIER ;
SDL_PushEvent ( & flushEvent ) ;
}
2021-03-07 16:06:12 +00:00
class ExecThread : public QThread
{
public :
ExecThread ( Session * session ) :
QThread ( nullptr ) ,
2021-10-24 21:22:49 +00:00
m_Session ( session )
{
setObjectName ( " Session Exec " ) ;
}
2021-03-07 16:06:12 +00:00
void run ( ) override
{
m_Session - > execInternal ( ) ;
}
Session * m_Session ;
} ;
2020-08-11 05:21:54 +00:00
void Session : : exec ( int displayOriginX , int displayOriginY )
{
m_DisplayOriginX = displayOriginX ;
m_DisplayOriginY = displayOriginY ;
2021-03-07 16:06:12 +00:00
// Use a separate thread for the streaming session on X11 or Wayland
// to ensure we don't stomp on Qt's GL context. This breaks when using
// the Qt EGLFS backend, so we will restrict this to X11
m_ThreadedExec = WMUtils : : isRunningX11 ( ) | | WMUtils : : isRunningWayland ( ) ;
if ( m_ThreadedExec ) {
// Run the streaming session on a separate thread for Linux/BSD
ExecThread execThread ( this ) ;
execThread . start ( ) ;
// Until the SDL streaming window is created, we should continue
// to update the Qt UI to allow warning messages to display and
// make sure that the Qt window can hide itself.
while ( ! execThread . wait ( 10 ) & & m_Window = = nullptr ) {
QCoreApplication : : processEvents ( QEventLoop : : ExcludeUserInputEvents ) ;
QCoreApplication : : sendPostedEvents ( ) ;
}
QCoreApplication : : processEvents ( QEventLoop : : ExcludeUserInputEvents ) ;
QCoreApplication : : sendPostedEvents ( ) ;
// SDL is in charge now. Wait until the streaming thread exits
// to further update the Qt window.
execThread . wait ( ) ;
}
else {
// Run the streaming session on the main thread for Windows and macOS
execInternal ( ) ;
}
}
void Session : : execInternal ( )
{
2020-08-11 05:21:54 +00:00
// Complete initialization in this deferred context to avoid
// calling expensive functions in the constructor (during the
// process of loading the StreamSegue).
//
// NB: This initializes the SDL video subsystem, so it must be
// called on the main thread.
if ( ! initialize ( ) ) {
emit sessionFinished ( 0 ) ;
2022-03-06 19:11:36 +00:00
emit readyForDeletion ( ) ;
2020-08-11 05:21:54 +00:00
return ;
}
// Wait for any old session to finish cleanup
s_ActiveSessionSemaphore . acquire ( ) ;
// We're now active
s_ActiveSession = this ;
// Initialize the gamepad code with our preferences
// NB: m_InputHandler must be initialize before starting the connection.
2023-03-18 19:29:45 +00:00
m_InputHandler = new SdlInputHandler ( * m_Preferences , m_StreamConfig . width , m_StreamConfig . height ) ;
2020-08-11 05:21:54 +00:00
AsyncConnectionStartThread asyncConnThread ( this ) ;
2021-03-07 16:06:12 +00:00
if ( ! m_ThreadedExec ) {
// Kick off the async connection thread while we sit here and pump the event loop
asyncConnThread . start ( ) ;
while ( ! asyncConnThread . wait ( 10 ) ) {
QCoreApplication : : processEvents ( QEventLoop : : ExcludeUserInputEvents ) ;
QCoreApplication : : sendPostedEvents ( ) ;
}
// Pump the event loop one last time to ensure we pick up any events from
// the thread that happened while it was in the final successful QThread::wait().
2020-08-11 05:21:54 +00:00
QCoreApplication : : processEvents ( QEventLoop : : ExcludeUserInputEvents ) ;
2020-09-05 05:38:54 +00:00
QCoreApplication : : sendPostedEvents ( ) ;
2020-08-11 05:21:54 +00:00
}
2021-03-07 16:06:12 +00:00
else {
// We're already in a separate thread so run the connection operations
// synchronously and don't pump the event loop. The main thread is already
// pumping the event loop for us.
asyncConnThread . run ( ) ;
}
2020-09-05 05:38:54 +00:00
2020-08-11 05:21:54 +00:00
// If the connection failed, clean up and abort the connection.
if ( ! m_AsyncConnectionSuccess ) {
2019-02-16 09:39:31 +00:00
delete m_InputHandler ;
m_InputHandler = nullptr ;
2019-04-22 00:43:38 +00:00
SDL_QuitSubSystem ( SDL_INIT_VIDEO ) ;
2018-12-06 06:17:26 +00:00
QThreadPool : : globalInstance ( ) - > start ( new DeferredSessionCleanupTask ( this ) ) ;
2018-06-28 08:44:43 +00:00
return ;
}
2018-07-21 02:55:07 +00:00
int x , y , width , height ;
2018-09-05 22:45:36 +00:00
getWindowDimensions ( x , y , width , height ) ;
2018-07-21 02:55:07 +00:00
2020-03-19 05:26:15 +00:00
# ifdef STEAM_LINK
// We need a little delay before creating the window or we will trigger some kind
// of graphics driver bug on Steam Link that causes a jagged overlay to appear in
// the top right corner randomly.
SDL_Delay ( 500 ) ;
# endif
2022-01-28 03:49:03 +00:00
// Request at least 8 bits per color for GL
SDL_GL_SetAttribute ( SDL_GL_RED_SIZE , 8 ) ;
SDL_GL_SetAttribute ( SDL_GL_GREEN_SIZE , 8 ) ;
SDL_GL_SetAttribute ( SDL_GL_BLUE_SIZE , 8 ) ;
2022-10-15 01:24:12 +00:00
// We always want a resizable window with High DPI enabled
2022-10-30 16:56:15 +00:00
Uint32 defaultWindowFlags = SDL_WINDOW_ALLOW_HIGHDPI | SDL_WINDOW_RESIZABLE ;
// If we're starting in windowed mode and the Moonlight GUI is maximized,
// match that with the streaming window.
if ( ! m_IsFullScreen ) {
QWindowList windows = QGuiApplication : : topLevelWindows ( ) ;
for ( const QWindow * window : windows ) {
if ( window - > windowState ( ) & Qt : : WindowMaximized ) {
defaultWindowFlags | = SDL_WINDOW_MAXIMIZED ;
break ;
}
}
}
2022-10-15 01:24:12 +00:00
2023-10-15 18:15:38 +00:00
// We use only the computer name on macOS to match Apple conventions where the
// app name is featured in the menu bar and the document name is in the title bar.
# ifdef Q_OS_DARWIN
std : : string windowName = QString ( m_Computer - > name ) . toStdString ( ) ;
# else
2023-07-23 06:16:24 +00:00
std : : string windowName = QString ( m_Computer - > name + " - Moonlight " ) . toStdString ( ) ;
2023-10-15 18:15:38 +00:00
# endif
2023-07-23 06:16:24 +00:00
m_Window = SDL_CreateWindow ( windowName . c_str ( ) ,
2018-07-21 02:55:07 +00:00
x ,
y ,
width ,
height ,
2022-10-15 01:24:12 +00:00
defaultWindowFlags | StreamUtils : : getPlatformWindowFlags ( ) ) ;
2018-07-21 02:55:07 +00:00
if ( ! m_Window ) {
2020-09-05 21:06:56 +00:00
SDL_LogWarn ( SDL_LOG_CATEGORY_APPLICATION ,
" SDL_CreateWindow() failed with platform flags: %s " ,
SDL_GetError ( ) ) ;
2023-07-23 06:16:24 +00:00
m_Window = SDL_CreateWindow ( windowName . c_str ( ) ,
2020-09-05 21:06:56 +00:00
x ,
y ,
width ,
height ,
2022-10-15 01:24:12 +00:00
defaultWindowFlags ) ;
2020-09-05 21:06:56 +00:00
if ( ! m_Window ) {
SDL_LogError ( SDL_LOG_CATEGORY_APPLICATION ,
" SDL_CreateWindow() failed: %s " ,
SDL_GetError ( ) ) ;
delete m_InputHandler ;
m_InputHandler = nullptr ;
SDL_QuitSubSystem ( SDL_INIT_VIDEO ) ;
QThreadPool : : globalInstance ( ) - > start ( new DeferredSessionCleanupTask ( this ) ) ;
return ;
}
2018-07-21 02:55:07 +00:00
}
2022-10-27 23:12:26 +00:00
// HACK: Remove once proper Dark Mode support lands in SDL
# ifdef Q_OS_WIN32
{
BOOL darkModeEnabled = FALSE ;
// Query whether dark mode is enabled for our Qt window (which tracks the OS dark mode state)
QWindowList windows = QGuiApplication : : topLevelWindows ( ) ;
for ( const QWindow * window : windows ) {
if ( SUCCEEDED ( DwmGetWindowAttribute ( ( HWND ) window - > winId ( ) , DWMWA_USE_IMMERSIVE_DARK_MODE , & darkModeEnabled , sizeof ( darkModeEnabled ) ) ) | |
SUCCEEDED ( DwmGetWindowAttribute ( ( HWND ) window - > winId ( ) , DWMWA_USE_IMMERSIVE_DARK_MODE_OLD , & darkModeEnabled , sizeof ( darkModeEnabled ) ) ) ) {
break ;
}
}
2022-10-27 23:18:47 +00:00
SDL_SysWMinfo info ;
SDL_VERSION ( & info . version ) ;
if ( SDL_GetWindowWMInfo ( m_Window , & info ) & & info . subsystem = = SDL_SYSWM_WINDOWS ) {
RECT clientRect ;
HBRUSH blackBrush ;
// Draw a black background (otherwise our window will be bright white).
//
// TODO: Remove when SDL does this itself
blackBrush = CreateSolidBrush ( 0 ) ;
GetClientRect ( info . info . win . window , & clientRect ) ;
FillRect ( info . info . win . hdc , & clientRect , blackBrush ) ;
DeleteObject ( blackBrush ) ;
// If dark mode is enabled, propagate that to our SDL window
if ( darkModeEnabled ) {
2022-10-27 23:12:26 +00:00
if ( FAILED ( DwmSetWindowAttribute ( info . info . win . window , DWMWA_USE_IMMERSIVE_DARK_MODE , & darkModeEnabled , sizeof ( darkModeEnabled ) ) ) ) {
DwmSetWindowAttribute ( info . info . win . window , DWMWA_USE_IMMERSIVE_DARK_MODE_OLD , & darkModeEnabled , sizeof ( darkModeEnabled ) ) ;
}
2022-10-28 00:26:51 +00:00
// Toggle non-client rendering off and back on to ensure dark mode takes effect on Windows 10.
// DWM doesn't seem to correctly invalidate the non-client area after enabling dark mode.
DWMNCRENDERINGPOLICY ncPolicy = DWMNCRP_DISABLED ;
DwmSetWindowAttribute ( info . info . win . window , DWMWA_NCRENDERING_POLICY , & ncPolicy , sizeof ( ncPolicy ) ) ;
ncPolicy = DWMNCRP_ENABLED ;
DwmSetWindowAttribute ( info . info . win . window , DWMWA_NCRENDERING_POLICY , & ncPolicy , sizeof ( ncPolicy ) ) ;
2022-10-27 23:12:26 +00:00
}
}
}
# endif
2020-04-29 03:24:23 +00:00
m_InputHandler - > setWindow ( m_Window ) ;
2018-10-13 03:34:58 +00:00
QSvgRenderer svgIconRenderer ( QString ( " :/res/moonlight.svg " ) ) ;
QImage svgImage ( ICON_SIZE , ICON_SIZE , QImage : : Format_RGBA8888 ) ;
svgImage . fill ( 0 ) ;
QPainter svgPainter ( & svgImage ) ;
svgIconRenderer . render ( & svgPainter ) ;
SDL_Surface * iconSurface = SDL_CreateRGBSurfaceWithFormatFrom ( ( void * ) svgImage . constBits ( ) ,
svgImage . width ( ) ,
svgImage . height ( ) ,
32 ,
4 * svgImage . width ( ) ,
SDL_PIXELFORMAT_RGBA32 ) ;
# ifndef Q_OS_DARWIN
// Other platforms seem to preserve our Qt icon when creating a new window.
if ( iconSurface ! = nullptr ) {
// This must be called before entering full-screen mode on Windows
// or our icon will not persist when toggling to windowed mode
SDL_SetWindowIcon ( m_Window , iconSurface ) ;
}
# endif
2022-10-15 01:24:12 +00:00
// Update the window display mode based on our current monitor
// for if/when we enter full-screen mode.
updateOptimalWindowDisplayMode ( ) ;
2022-04-24 21:37:57 +00:00
2022-10-15 01:24:12 +00:00
// Enter full screen if requested
if ( m_IsFullScreen ) {
2018-09-04 02:17:34 +00:00
SDL_SetWindowFullscreen ( m_Window , m_FullScreenFlag ) ;
2022-10-27 23:18:47 +00:00
# ifdef Q_OS_WIN32
SDL_SysWMinfo info ;
SDL_VERSION ( & info . version ) ;
// Draw a black background again after entering full-screen to avoid visual artifacts
// where the old window decorations were before.
//
// TODO: Remove when SDL does this itself
if ( SDL_GetWindowWMInfo ( m_Window , & info ) & & info . subsystem = = SDL_SYSWM_WINDOWS ) {
RECT clientRect ;
HBRUSH blackBrush ;
blackBrush = CreateSolidBrush ( 0 ) ;
GetClientRect ( info . info . win . window , & clientRect ) ;
FillRect ( info . info . win . hdc , & clientRect , blackBrush ) ;
DeleteObject ( blackBrush ) ;
}
# endif
2018-08-17 05:25:14 +00:00
}
2018-07-08 04:52:20 +00:00
2020-03-14 07:33:53 +00:00
bool needsFirstEnterCapture = false ;
2020-05-10 02:48:13 +00:00
bool needsPostDecoderCreationCapture = false ;
2020-03-14 07:33:53 +00:00
2020-05-02 01:43:54 +00:00
// HACK: For Wayland, we wait until we get the first SDL_WINDOWEVENT_ENTER
2020-05-10 02:48:13 +00:00
// event where it seems to work consistently on GNOME. For other platforms,
// especially where SDL may call SDL_RecreateWindow(), we must only capture
// after the decoder is created.
if ( strcmp ( SDL_GetCurrentVideoDriver ( ) , " wayland " ) = = 0 ) {
2020-05-02 01:43:54 +00:00
// Native Wayland: Capture on SDL_WINDOWEVENT_ENTER
needsFirstEnterCapture = true ;
}
2020-05-10 02:48:13 +00:00
else {
// X11/XWayland: Capture after decoder creation
needsPostDecoderCreationCapture = true ;
}
2018-06-28 08:44:43 +00:00
2018-07-31 05:44:19 +00:00
// Stop text input. SDL enables it by default
// when we initialize the video subsystem, but this
// causes an IME popup when certain keys are held down
// on macOS.
SDL_StopTextInput ( ) ;
2022-03-17 02:57:03 +00:00
// Disable the screen saver if requested
if ( m_Preferences - > keepAwake ) {
2022-03-17 02:48:56 +00:00
SDL_DisableScreenSaver ( ) ;
}
2018-07-13 09:28:10 +00:00
2020-01-16 02:23:02 +00:00
// Hide Qt's fake mouse cursor on EGLFS systems
if ( QGuiApplication : : platformName ( ) = = " eglfs " ) {
QGuiApplication : : setOverrideCursor ( QCursor ( Qt : : BlankCursor ) ) ;
}
2020-01-11 10:06:08 +00:00
2018-08-14 05:23:05 +00:00
// Set timer resolution to 1 ms on Windows for greater
// sleep precision and more accurate callback timing.
SDL_SetHint ( SDL_HINT_TIMER_RESOLUTION , " 1 " ) ;
2018-09-07 02:42:53 +00:00
int currentDisplayIndex = SDL_GetWindowDisplayIndex ( m_Window ) ;
2018-12-06 06:17:26 +00:00
// Now that we're about to stream, any SDL_QUIT event is expected
// unless it comes from the connection termination callback where
// (m_UnexpectedTermination is set back to true).
m_UnexpectedTermination = false ;
2019-06-30 00:40:30 +00:00
// Start rich presence to indicate we're in game
2020-08-30 08:38:26 +00:00
RichPresenceManager presence ( * m_Preferences , m_App . name ) ;
2019-06-30 00:40:30 +00:00
2018-09-13 19:14:56 +00:00
// Hijack this thread to be the SDL main thread. We have to do this
// because we want to suspend all Qt processing until the stream is over.
2018-06-28 08:44:43 +00:00
SDL_Event event ;
2018-08-02 01:26:50 +00:00
for ( ; ; ) {
2022-01-17 22:59:45 +00:00
# if SDL_VERSION_ATLEAST(2, 0, 18) && !defined(STEAM_LINK)
// SDL 2.0.18 has a proper wait event implementation that uses platform
// support to block on events rather than polling on Windows, macOS, X11,
// and Wayland. It will fall back to 1 ms polling if a joystick is
// connected, so we don't use it for STEAM_LINK to ensure we only poll
// every 10 ms.
//
// NB: This behavior was introduced in SDL 2.0.16, but had a few critical
// issues that could cause indefinite timeouts, delayed joystick detection,
// and other problems.
2021-08-15 20:16:51 +00:00
if ( ! SDL_WaitEventTimeout ( & event , 1000 ) ) {
presence . runCallbacks ( ) ;
continue ;
}
# else
2019-01-22 04:10:13 +00:00
// We explicitly use SDL_PollEvent() and SDL_Delay() because
// SDL_WaitEvent() has an internal SDL_Delay(10) inside which
// blocks this thread too long for high polling rate mice and high
// refresh rate displays.
if ( ! SDL_PollEvent ( & event ) ) {
2019-04-28 20:04:58 +00:00
# ifndef STEAM_LINK
2019-01-22 04:10:13 +00:00
SDL_Delay ( 1 ) ;
2019-04-28 20:04:58 +00:00
# else
// Waking every 1 ms to process input is too much for the low performance
// ARM core in the Steam Link, so we will wait 10 ms instead.
SDL_Delay ( 10 ) ;
# endif
2019-06-30 00:40:30 +00:00
presence . runCallbacks ( ) ;
2019-01-22 04:10:13 +00:00
continue ;
}
2021-08-15 20:16:51 +00:00
# endif
2018-06-28 08:44:43 +00:00
switch ( event . type ) {
case SDL_QUIT :
SDL_LogInfo ( SDL_LOG_CATEGORY_APPLICATION ,
" Quit event received " ) ;
2018-07-09 07:09:06 +00:00
goto DispatchDeferredCleanup ;
2018-07-08 04:52:20 +00:00
2019-04-10 04:46:14 +00:00
case SDL_USEREVENT :
2020-07-12 22:06:36 +00:00
switch ( event . user . code ) {
case SDL_CODE_FRAME_READY :
2022-08-29 01:10:26 +00:00
if ( m_VideoDecoder ! = nullptr ) {
m_VideoDecoder - > renderFrameOnMainThread ( ) ;
}
2020-07-12 22:06:36 +00:00
break ;
2021-01-09 23:51:25 +00:00
case SDL_CODE_FLUSH_WINDOW_EVENT_BARRIER :
2021-01-15 01:28:21 +00:00
m_FlushingWindowEventsRef - - ;
2021-01-09 23:51:25 +00:00
break ;
2022-03-05 19:32:38 +00:00
case SDL_CODE_GAMECONTROLLER_RUMBLE :
2023-06-18 21:04:49 +00:00
m_InputHandler - > rumble ( ( uint16_t ) ( uintptr_t ) event . user . data1 ,
( uint16_t ) ( ( uintptr_t ) event . user . data2 > > 16 ) ,
( uint16_t ) ( ( uintptr_t ) event . user . data2 & 0xFFFF ) ) ;
break ;
case SDL_CODE_GAMECONTROLLER_RUMBLE_TRIGGERS :
m_InputHandler - > rumbleTriggers ( ( uint16_t ) ( uintptr_t ) event . user . data1 ,
( uint16_t ) ( ( uintptr_t ) event . user . data2 > > 16 ) ,
( uint16_t ) ( ( uintptr_t ) event . user . data2 & 0xFFFF ) ) ;
2022-03-05 19:32:38 +00:00
break ;
2023-06-18 21:08:32 +00:00
case SDL_CODE_GAMECONTROLLER_SET_MOTION_EVENT_STATE :
m_InputHandler - > setMotionEventState ( ( uint16_t ) ( uintptr_t ) event . user . data1 ,
( uint8_t ) ( ( uintptr_t ) event . user . data2 > > 16 ) ,
( uint16_t ) ( ( uintptr_t ) event . user . data2 & 0xFFFF ) ) ;
break ;
2023-07-02 21:44:41 +00:00
case SDL_CODE_GAMECONTROLLER_SET_CONTROLLER_LED :
m_InputHandler - > setControllerLED ( ( uint16_t ) ( uintptr_t ) event . user . data1 ,
( uint8_t ) ( ( uintptr_t ) event . user . data2 > > 16 ) ,
( uint8_t ) ( ( uintptr_t ) event . user . data2 > > 8 ) ,
( uint8_t ) ( ( uintptr_t ) event . user . data2 ) ) ;
break ;
2020-07-12 22:06:36 +00:00
default :
SDL_assert ( false ) ;
}
2018-07-08 04:52:20 +00:00
break ;
2018-07-20 22:31:57 +00:00
2018-07-21 02:37:54 +00:00
case SDL_WINDOWEVENT :
2020-12-25 21:32:11 +00:00
// Early handling of some events
switch ( event . window . event ) {
case SDL_WINDOWEVENT_FOCUS_LOST :
2021-01-09 23:56:23 +00:00
if ( m_Preferences - > muteOnFocusLoss ) {
2020-12-25 21:32:11 +00:00
m_AudioMuted = true ;
}
2021-01-09 23:56:23 +00:00
m_InputHandler - > notifyFocusLost ( ) ;
2020-12-25 21:32:11 +00:00
break ;
2021-01-09 23:56:23 +00:00
case SDL_WINDOWEVENT_FOCUS_GAINED :
if ( m_Preferences - > muteOnFocusLoss ) {
2020-12-25 21:32:11 +00:00
m_AudioMuted = false ;
}
break ;
2021-01-09 23:56:23 +00:00
case SDL_WINDOWEVENT_LEAVE :
m_InputHandler - > notifyMouseLeave ( ) ;
break ;
2020-05-08 02:26:02 +00:00
}
2018-09-01 21:31:37 +00:00
2021-08-15 20:16:51 +00:00
presence . runCallbacks ( ) ;
2020-03-14 07:33:53 +00:00
// Capture the mouse on SDL_WINDOWEVENT_ENTER if needed
if ( needsFirstEnterCapture & & event . window . event = = SDL_WINDOWEVENT_ENTER ) {
m_InputHandler - > setCaptureActive ( true ) ;
needsFirstEnterCapture = false ;
}
2018-07-21 07:16:03 +00:00
// We want to recreate the decoder for resizes (full-screen toggles) and the initial shown event.
// We use SDL_WINDOWEVENT_SIZE_CHANGED rather than SDL_WINDOWEVENT_RESIZED because the latter doesn't
// seem to fire when switching from windowed to full-screen on X11.
if ( event . window . event ! = SDL_WINDOWEVENT_SIZE_CHANGED & & event . window . event ! = SDL_WINDOWEVENT_SHOWN ) {
2018-09-07 02:42:53 +00:00
// Check that the window display hasn't changed. If it has, we want
// to recreate the decoder to allow it to adapt to the new display.
// This will allow Pacer to pull the new display refresh rate.
2022-05-17 22:13:07 +00:00
# if SDL_VERSION_ATLEAST(2, 0, 18)
// On SDL 2.0.18+, there's an event for this specific situation
if ( event . window . event ! = SDL_WINDOWEVENT_DISPLAY_CHANGED ) {
break ;
}
# else
// Prior to SDL 2.0.18, we must check the display index for each window event
2018-09-07 02:42:53 +00:00
if ( SDL_GetWindowDisplayIndex ( m_Window ) = = currentDisplayIndex ) {
break ;
}
2022-05-17 22:13:07 +00:00
# endif
2018-07-21 02:37:54 +00:00
}
2022-03-06 21:21:43 +00:00
# ifdef Q_OS_WIN32
// We can get a resize event after being minimized. Recreating the renderer at that time can cause
// us to start drawing on the screen even while our window is minimized. Minimizing on Windows also
// moves the window to -32000, -32000 which can cause a false window display index change. Avoid
// that whole mess by never recreating the decoder if we're minimized.
else if ( SDL_GetWindowFlags ( m_Window ) & SDL_WINDOW_MINIMIZED ) {
break ;
}
# endif
2018-07-21 02:37:54 +00:00
2021-01-15 01:28:21 +00:00
if ( m_FlushingWindowEventsRef > 0 ) {
2021-01-09 23:51:25 +00:00
// Ignore window events for renderer reset if flushing
2021-01-15 01:28:21 +00:00
SDL_LogInfo ( SDL_LOG_CATEGORY_APPLICATION ,
" Dropping window event during flush: %d (%d %d) " ,
event . window . event ,
event . window . data1 ,
event . window . data2 ) ;
2021-01-09 23:51:25 +00:00
break ;
}
2020-05-13 01:41:36 +00:00
SDL_LogInfo ( SDL_LOG_CATEGORY_APPLICATION ,
" Recreating renderer for window event: %d (%d %d) " ,
event . window . event ,
event . window . data1 ,
event . window . data2 ) ;
2018-07-21 02:55:07 +00:00
// Fall through
2018-07-20 22:31:57 +00:00
case SDL_RENDER_DEVICE_RESET :
case SDL_RENDER_TARGETS_RESET :
2018-08-17 05:25:14 +00:00
2022-05-17 22:10:40 +00:00
if ( event . type ! = SDL_WINDOWEVENT ) {
SDL_LogWarn ( SDL_LOG_CATEGORY_APPLICATION ,
" Recreating renderer by internal request: %d " ,
event . type ) ;
}
2018-08-19 08:19:23 +00:00
SDL_AtomicLock ( & m_DecoderLock ) ;
// Destroy the old decoder
delete m_VideoDecoder ;
2021-01-09 23:51:25 +00:00
// Insert a barrier to discard any additional window events
// that could cause the renderer to be and recreated again.
// We don't use SDL_FlushEvent() here because it could cause
// important events to be lost.
flushWindowEvents ( ) ;
2018-08-19 08:19:23 +00:00
2022-04-24 21:12:12 +00:00
// Update the window display mode based on our current monitor
2022-04-24 21:48:21 +00:00
// NB: Avoid a useless modeset by only doing this if it changed.
if ( currentDisplayIndex ! = SDL_GetWindowDisplayIndex ( m_Window ) ) {
currentDisplayIndex = SDL_GetWindowDisplayIndex ( m_Window ) ;
updateOptimalWindowDisplayMode ( ) ;
}
2018-08-17 05:25:14 +00:00
2018-08-19 08:19:23 +00:00
// Now that the old decoder is dead, flush any events it may
// have queued to reset itself (if this reset was the result
// of state loss).
SDL_PumpEvents ( ) ;
SDL_FlushEvent ( SDL_RENDER_DEVICE_RESET ) ;
SDL_FlushEvent ( SDL_RENDER_TARGETS_RESET ) ;
2018-07-20 22:31:57 +00:00
2018-09-08 22:09:46 +00:00
{
// If the stream exceeds the display refresh rate (plus some slack),
// forcefully disable V-sync to allow the stream to render faster
// than the display.
int displayHz = StreamUtils : : getDisplayRefreshRate ( m_Window ) ;
2018-09-29 21:06:55 +00:00
bool enableVsync = m_Preferences - > enableVsync ;
2018-09-08 22:09:46 +00:00
if ( displayHz + 5 < m_StreamConfig . fps ) {
SDL_LogWarn ( SDL_LOG_CATEGORY_APPLICATION ,
" Disabling V-sync because refresh rate limit exceeded " ) ;
enableVsync = false ;
}
// Choose a new decoder (hopefully the same one, but possibly
// not if a GPU was removed or something).
2018-09-29 21:06:55 +00:00
if ( ! chooseDecoder ( m_Preferences - > videoDecoderSelection ,
2018-09-08 22:09:46 +00:00
m_Window , m_ActiveVideoFormat , m_ActiveVideoWidth ,
m_ActiveVideoHeight , m_ActiveVideoFrameRate ,
enableVsync ,
2018-12-25 20:57:00 +00:00
enableVsync & & m_Preferences - > framePacing ,
2019-02-13 02:42:53 +00:00
false ,
2018-09-08 22:09:46 +00:00
s_ActiveSession - > m_VideoDecoder ) ) {
SDL_AtomicUnlock ( & m_DecoderLock ) ;
SDL_LogError ( SDL_LOG_CATEGORY_APPLICATION ,
" Failed to recreate decoder after reset " ) ;
2020-11-21 19:15:54 +00:00
emit displayLaunchError ( tr ( " Unable to initialize video decoder. Please check your streaming settings and try again. " ) ) ;
2018-09-08 22:09:46 +00:00
goto DispatchDeferredCleanup ;
}
2020-05-10 02:48:13 +00:00
// As of SDL 2.0.12, SDL_RecreateWindow() doesn't carry over mouse capture
// or mouse hiding state to the new window. By capturing after the decoder
// is set up, this ensures the window re-creation is already done.
if ( needsPostDecoderCreationCapture ) {
m_InputHandler - > setCaptureActive ( true ) ;
2021-01-15 01:30:30 +00:00
needsPostDecoderCreationCapture = false ;
2020-05-10 02:48:13 +00:00
}
2018-07-20 22:31:57 +00:00
}
// Request an IDR frame to complete the reset
2022-10-05 05:53:31 +00:00
LiRequestIdrFrame ( ) ;
2018-07-20 22:31:57 +00:00
2022-01-29 04:10:50 +00:00
// Set HDR mode. We may miss the callback if we're in the middle
// of recreating our decoder at the time the HDR transition happens.
m_VideoDecoder - > setHdrMode ( LiGetCurrentHostDisplayHdrMode ( ) ) ;
2022-03-29 23:26:09 +00:00
// After a window resize, we need to reset the pointer lock region
m_InputHandler - > updatePointerRegionLock ( ) ;
2018-07-20 22:31:57 +00:00
SDL_AtomicUnlock ( & m_DecoderLock ) ;
break ;
2018-06-28 08:44:43 +00:00
case SDL_KEYUP :
case SDL_KEYDOWN :
2021-08-15 20:16:51 +00:00
presence . runCallbacks ( ) ;
2019-02-12 05:39:55 +00:00
m_InputHandler - > handleKeyEvent ( & event . key ) ;
2018-06-28 08:44:43 +00:00
break ;
case SDL_MOUSEBUTTONDOWN :
case SDL_MOUSEBUTTONUP :
2021-08-15 20:16:51 +00:00
presence . runCallbacks ( ) ;
2019-02-12 05:39:55 +00:00
m_InputHandler - > handleMouseButtonEvent ( & event . button ) ;
2018-06-28 08:44:43 +00:00
break ;
case SDL_MOUSEMOTION :
2020-04-29 03:24:23 +00:00
m_InputHandler - > handleMouseMotionEvent ( & event . motion ) ;
2018-06-28 08:44:43 +00:00
break ;
case SDL_MOUSEWHEEL :
2019-02-12 05:39:55 +00:00
m_InputHandler - > handleMouseWheelEvent ( & event . wheel ) ;
2018-06-28 08:44:43 +00:00
break ;
case SDL_CONTROLLERAXISMOTION :
2019-02-12 05:39:55 +00:00
m_InputHandler - > handleControllerAxisEvent ( & event . caxis ) ;
2018-06-28 08:44:43 +00:00
break ;
case SDL_CONTROLLERBUTTONDOWN :
case SDL_CONTROLLERBUTTONUP :
2021-08-15 20:16:51 +00:00
presence . runCallbacks ( ) ;
2019-02-12 05:39:55 +00:00
m_InputHandler - > handleControllerButtonEvent ( & event . cbutton ) ;
2018-06-28 08:44:43 +00:00
break ;
2023-06-18 21:08:32 +00:00
# if SDL_VERSION_ATLEAST(2, 0, 14)
case SDL_CONTROLLERSENSORUPDATE :
m_InputHandler - > handleControllerSensorEvent ( & event . csensor ) ;
break ;
case SDL_CONTROLLERTOUCHPADDOWN :
case SDL_CONTROLLERTOUCHPADUP :
case SDL_CONTROLLERTOUCHPADMOTION :
m_InputHandler - > handleControllerTouchpadEvent ( & event . ctouchpad ) ;
break ;
2023-07-02 21:44:41 +00:00
# endif
# if SDL_VERSION_ATLEAST(2, 24, 0)
case SDL_JOYBATTERYUPDATED :
m_InputHandler - > handleJoystickBatteryEvent ( & event . jbattery ) ;
break ;
2023-06-18 21:08:32 +00:00
# endif
2018-06-28 08:44:43 +00:00
case SDL_CONTROLLERDEVICEADDED :
case SDL_CONTROLLERDEVICEREMOVED :
2019-02-12 05:39:55 +00:00
m_InputHandler - > handleControllerDeviceEvent ( & event . cdevice ) ;
2018-06-28 08:44:43 +00:00
break ;
2018-09-30 02:14:52 +00:00
case SDL_JOYDEVICEADDED :
2019-02-12 05:39:55 +00:00
m_InputHandler - > handleJoystickArrivalEvent ( & event . jdevice ) ;
2018-09-30 02:14:52 +00:00
break ;
2018-09-30 05:43:28 +00:00
case SDL_FINGERDOWN :
case SDL_FINGERMOTION :
case SDL_FINGERUP :
2020-04-29 03:24:23 +00:00
m_InputHandler - > handleTouchFingerEvent ( & event . tfinger ) ;
2018-09-30 05:43:28 +00:00
break ;
2018-06-28 08:44:43 +00:00
}
}
2018-07-09 07:09:06 +00:00
DispatchDeferredCleanup :
// Uncapture the mouse and hide the window immediately,
// so we can return to the Qt GUI ASAP.
2019-07-03 05:17:18 +00:00
m_InputHandler - > setCaptureActive ( false ) ;
2018-07-13 09:28:10 +00:00
SDL_EnableScreenSaver ( ) ;
2018-08-14 05:23:05 +00:00
SDL_SetHint ( SDL_HINT_TIMER_RESOLUTION , " 0 " ) ;
2020-01-16 02:23:02 +00:00
if ( QGuiApplication : : platformName ( ) = = " eglfs " ) {
QGuiApplication : : restoreOverrideCursor ( ) ;
}
2018-07-09 07:09:06 +00:00
2018-10-04 01:27:12 +00:00
// Raise any keys that are still down
2019-02-12 05:39:55 +00:00
m_InputHandler - > raiseAllKeys ( ) ;
2018-10-04 01:27:12 +00:00
2022-03-05 19:32:38 +00:00
// Destroy the input handler now. This must be destroyed
// before allowwing the UI to continue execution or it could
// interfere with SDLGamepadKeyNavigation.
2019-02-13 03:30:02 +00:00
delete m_InputHandler ;
m_InputHandler = nullptr ;
2018-07-18 03:00:16 +00:00
// Destroy the decoder, since this must be done on the main thread
2022-01-19 00:19:28 +00:00
// NB: This must happen before LiStopConnection() for pull-based
// decoders.
2018-07-18 03:00:16 +00:00
SDL_AtomicLock ( & m_DecoderLock ) ;
delete m_VideoDecoder ;
m_VideoDecoder = nullptr ;
SDL_AtomicUnlock ( & m_DecoderLock ) ;
2018-11-01 01:20:39 +00:00
// This must be called after the decoder is deleted, because
// the renderer may want to interact with the window
2018-07-21 01:15:46 +00:00
SDL_DestroyWindow ( m_Window ) ;
2018-11-01 01:20:39 +00:00
2018-07-23 00:07:45 +00:00
if ( iconSurface ! = nullptr ) {
SDL_FreeSurface ( iconSurface ) ;
}
2018-07-21 01:15:46 +00:00
2019-04-22 00:43:38 +00:00
SDL_QuitSubSystem ( SDL_INIT_VIDEO ) ;
2018-07-09 07:09:06 +00:00
// Cleanup can take a while, so dispatch it to a worker thread.
// When it is complete, it will release our s_ActiveSessionSemaphore
// reference.
2018-12-06 03:49:06 +00:00
QThreadPool : : globalInstance ( ) - > start ( new DeferredSessionCleanupTask ( this ) ) ;
2018-06-28 08:44:43 +00:00
}
2018-07-09 07:09:06 +00:00