2015-06-21 17:33:46 +02:00
/*
Simple DirectMedia Layer
2023-01-09 09:41:41 -08:00
Copyright ( C ) 1997 - 2023 Sam Lantinga < slouken @ libsdl . org >
2015-06-21 17:33:46 +02:00
This software is provided ' as - is ' , without any express or implied
warranty . In no event will the authors be held liable for any damages
arising from the use of this software .
Permission is granted to anyone to use this software for any purpose ,
including commercial applications , and to alter it and redistribute it
freely , subject to the following restrictions :
1. The origin of this software must not be misrepresented ; you must not
claim that you wrote the original software . If you use this software
in a product , an acknowledgment in the product documentation would be
appreciated but is not required .
2. Altered source versions must be plainly marked as such , and must not be
misrepresented as being the original software .
3. This notice may not be removed or altered from any source distribution .
*/
2022-11-29 18:34:15 -08:00
# include "SDL_internal.h"
2015-06-21 17:33:46 +02:00
# include "SDL_dbus.h"
2022-08-29 11:08:24 +02:00
# include "SDL_sandbox.h"
2022-05-06 10:51:55 -07:00
# include "../../stdlib/SDL_vacopy.h"
2015-06-21 17:33:46 +02:00
2023-03-29 21:49:01 +00:00
# ifdef SDL_USE_LIBDBUS
2015-06-21 17:33:46 +02:00
/* we never link directly to libdbus. */
static const char * dbus_library = " libdbus-1.so.3 " ;
static void * dbus_handle = NULL ;
2022-08-29 11:08:24 +02:00
static char * inhibit_handle = NULL ;
2015-06-21 17:33:46 +02:00
static unsigned int screensaver_cookie = 0 ;
2016-10-14 08:20:40 -07:00
static SDL_DBusContext dbus ;
2015-06-21 17:33:46 +02:00
2022-11-30 12:51:59 -08:00
static int LoadDBUSSyms ( void )
2015-06-21 17:33:46 +02:00
{
2023-01-09 14:55:12 -08:00
# define SDL_DBUS_SYM2(TYPE, x, y) \
if ( ! ( dbus . x = ( TYPE ) SDL_LoadFunction ( dbus_handle , # y ) ) ) \
2022-11-30 12:51:59 -08:00
return - 1
2023-01-09 14:55:12 -08:00
# define SDL_DBUS_SYM(TYPE, x) \
SDL_DBUS_SYM2 ( TYPE , x , dbus_ # # x )
2015-06-21 17:33:46 +02:00
2023-01-09 14:55:12 -08:00
SDL_DBUS_SYM ( DBusConnection * ( * ) ( DBusBusType , DBusError * ) , bus_get_private ) ;
SDL_DBUS_SYM ( dbus_bool_t ( * ) ( DBusConnection * , DBusError * ) , bus_register ) ;
SDL_DBUS_SYM ( void ( * ) ( DBusConnection * , const char * , DBusError * ) , bus_add_match ) ;
SDL_DBUS_SYM ( DBusConnection * ( * ) ( const char * , DBusError * ) , connection_open_private ) ;
SDL_DBUS_SYM ( void ( * ) ( DBusConnection * , dbus_bool_t ) , connection_set_exit_on_disconnect ) ;
SDL_DBUS_SYM ( dbus_bool_t ( * ) ( DBusConnection * ) , connection_get_is_connected ) ;
SDL_DBUS_SYM ( dbus_bool_t ( * ) ( DBusConnection * , DBusHandleMessageFunction , void * , DBusFreeFunction ) , connection_add_filter ) ;
SDL_DBUS_SYM ( dbus_bool_t ( * ) ( DBusConnection * , const char * , const DBusObjectPathVTable * , void * , DBusError * ) , connection_try_register_object_path ) ;
SDL_DBUS_SYM ( dbus_bool_t ( * ) ( DBusConnection * , DBusMessage * , dbus_uint32_t * ) , connection_send ) ;
SDL_DBUS_SYM ( DBusMessage * ( * ) ( DBusConnection * , DBusMessage * , int , DBusError * ) , connection_send_with_reply_and_block ) ;
SDL_DBUS_SYM ( void ( * ) ( DBusConnection * ) , connection_close ) ;
SDL_DBUS_SYM ( void ( * ) ( DBusConnection * ) , connection_ref ) ;
SDL_DBUS_SYM ( void ( * ) ( DBusConnection * ) , connection_unref ) ;
SDL_DBUS_SYM ( void ( * ) ( DBusConnection * ) , connection_flush ) ;
SDL_DBUS_SYM ( dbus_bool_t ( * ) ( DBusConnection * , int ) , connection_read_write ) ;
SDL_DBUS_SYM ( DBusDispatchStatus ( * ) ( DBusConnection * ) , connection_dispatch ) ;
SDL_DBUS_SYM ( dbus_bool_t ( * ) ( DBusMessage * , const char * , const char * ) , message_is_signal ) ;
SDL_DBUS_SYM ( DBusMessage * ( * ) ( const char * , const char * , const char * , const char * ) , message_new_method_call ) ;
SDL_DBUS_SYM ( dbus_bool_t ( * ) ( DBusMessage * , int , . . . ) , message_append_args ) ;
SDL_DBUS_SYM ( dbus_bool_t ( * ) ( DBusMessage * , int , va_list ) , message_append_args_valist ) ;
SDL_DBUS_SYM ( void ( * ) ( DBusMessage * , DBusMessageIter * ) , message_iter_init_append ) ;
SDL_DBUS_SYM ( dbus_bool_t ( * ) ( DBusMessageIter * , int , const char * , DBusMessageIter * ) , message_iter_open_container ) ;
SDL_DBUS_SYM ( dbus_bool_t ( * ) ( DBusMessageIter * , int , const void * ) , message_iter_append_basic ) ;
SDL_DBUS_SYM ( dbus_bool_t ( * ) ( DBusMessageIter * , DBusMessageIter * ) , message_iter_close_container ) ;
SDL_DBUS_SYM ( dbus_bool_t ( * ) ( DBusMessage * , DBusError * , int , . . . ) , message_get_args ) ;
SDL_DBUS_SYM ( dbus_bool_t ( * ) ( DBusMessage * , DBusError * , int , va_list ) , message_get_args_valist ) ;
SDL_DBUS_SYM ( dbus_bool_t ( * ) ( DBusMessage * , DBusMessageIter * ) , message_iter_init ) ;
SDL_DBUS_SYM ( dbus_bool_t ( * ) ( DBusMessageIter * ) , message_iter_next ) ;
SDL_DBUS_SYM ( void ( * ) ( DBusMessageIter * , void * ) , message_iter_get_basic ) ;
SDL_DBUS_SYM ( int ( * ) ( DBusMessageIter * ) , message_iter_get_arg_type ) ;
SDL_DBUS_SYM ( void ( * ) ( DBusMessageIter * , DBusMessageIter * ) , message_iter_recurse ) ;
SDL_DBUS_SYM ( void ( * ) ( DBusMessage * ) , message_unref ) ;
SDL_DBUS_SYM ( dbus_bool_t ( * ) ( void ) , threads_init_default ) ;
SDL_DBUS_SYM ( void ( * ) ( DBusError * ) , error_init ) ;
SDL_DBUS_SYM ( dbus_bool_t ( * ) ( const DBusError * ) , error_is_set ) ;
SDL_DBUS_SYM ( void ( * ) ( DBusError * ) , error_free ) ;
SDL_DBUS_SYM ( char * ( * ) ( void ) , get_local_machine_id ) ;
SDL_DBUS_SYM ( void ( * ) ( void * ) , free ) ;
SDL_DBUS_SYM ( void ( * ) ( char * * ) , free_string_array ) ;
SDL_DBUS_SYM ( void ( * ) ( void ) , shutdown ) ;
2015-06-21 17:33:46 +02:00
2022-11-30 12:51:59 -08:00
# undef SDL_DBUS_SYM
# undef SDL_DBUS_SYM2
2015-06-21 17:33:46 +02:00
return 0 ;
}
2022-11-30 12:51:59 -08:00
static void UnloadDBUSLibrary ( void )
2015-06-21 17:33:46 +02:00
{
if ( dbus_handle ! = NULL ) {
SDL_UnloadObject ( dbus_handle ) ;
dbus_handle = NULL ;
}
}
2022-11-30 12:51:59 -08:00
static int LoadDBUSLibrary ( void )
2015-06-21 17:33:46 +02:00
{
int retval = 0 ;
if ( dbus_handle = = NULL ) {
dbus_handle = SDL_LoadObject ( dbus_library ) ;
if ( dbus_handle = = NULL ) {
retval = - 1 ;
/* Don't call SDL_SetError(): SDL_LoadObject already did. */
} else {
retval = LoadDBUSSyms ( ) ;
if ( retval < 0 ) {
UnloadDBUSLibrary ( ) ;
}
}
}
return retval ;
}
2021-08-04 13:30:12 -04:00
static SDL_SpinLock spinlock_dbus_init = 0 ;
/* you must hold spinlock_dbus_init before calling this! */
2022-11-30 12:51:59 -08:00
static void SDL_DBus_Init_Spinlocked ( void )
2015-06-21 17:33:46 +02:00
{
2019-10-09 19:38:16 -04:00
static SDL_bool is_dbus_available = SDL_TRUE ;
if ( ! is_dbus_available ) {
2022-11-30 12:51:59 -08:00
return ; /* don't keep trying if this fails. */
2019-10-09 19:38:16 -04:00
}
if ( ! dbus . session_conn ) {
2015-06-21 17:33:46 +02:00
DBusError err ;
2019-10-09 19:38:16 -04:00
if ( LoadDBUSLibrary ( ) = = - 1 ) {
2022-11-30 12:51:59 -08:00
is_dbus_available = SDL_FALSE ; /* can't load at all? Don't keep trying. */
return ;
2019-10-09 19:38:16 -04:00
}
2020-03-30 14:26:15 -07:00
if ( ! dbus . threads_init_default ( ) ) {
is_dbus_available = SDL_FALSE ;
return ;
}
2015-06-21 17:33:46 +02:00
dbus . error_init ( & err ) ;
2020-03-26 20:28:04 -04:00
/* session bus is required */
2015-06-21 17:33:46 +02:00
dbus . session_conn = dbus . bus_get_private ( DBUS_BUS_SESSION , & err ) ;
if ( dbus . error_is_set ( & err ) ) {
dbus . error_free ( & err ) ;
2017-05-28 07:08:10 -04:00
SDL_DBus_Quit ( ) ;
2019-10-09 19:38:16 -04:00
is_dbus_available = SDL_FALSE ;
2022-11-30 12:51:59 -08:00
return ; /* oh well */
2015-06-21 17:33:46 +02:00
}
dbus . connection_set_exit_on_disconnect ( dbus . session_conn , 0 ) ;
2020-03-26 20:28:04 -04:00
/* system bus is optional */
dbus . system_conn = dbus . bus_get_private ( DBUS_BUS_SYSTEM , & err ) ;
if ( ! dbus . error_is_set ( & err ) ) {
dbus . connection_set_exit_on_disconnect ( dbus . system_conn , 0 ) ;
}
dbus . error_free ( & err ) ;
2015-06-21 17:33:46 +02:00
}
}
2022-11-30 12:51:59 -08:00
void SDL_DBus_Init ( void )
2021-08-04 13:30:12 -04:00
{
2022-11-30 12:51:59 -08:00
SDL_AtomicLock ( & spinlock_dbus_init ) ; /* make sure two threads can't init at same time, since this can happen before SDL_Init. */
2021-08-04 13:30:12 -04:00
SDL_DBus_Init_Spinlocked ( ) ;
SDL_AtomicUnlock ( & spinlock_dbus_init ) ;
}
2022-11-30 12:51:59 -08:00
void SDL_DBus_Quit ( void )
2015-06-21 17:33:46 +02:00
{
2017-05-28 07:08:10 -04:00
if ( dbus . system_conn ) {
dbus . connection_close ( dbus . system_conn ) ;
dbus . connection_unref ( dbus . system_conn ) ;
}
2015-06-21 17:33:46 +02:00
if ( dbus . session_conn ) {
dbus . connection_close ( dbus . session_conn ) ;
dbus . connection_unref ( dbus . session_conn ) ;
2017-05-28 07:08:10 -04:00
}
2018-02-13 08:07:52 -08:00
/* Don't do this - bug 3950
dbus_shutdown ( ) is a debug feature which closes all global resources in the dbus library . Calling this should be done by the app , not a library , because if there are multiple users of dbus in the process then SDL could shut it down even though another part is using it .
*/
#if 0
2017-05-28 07:08:10 -04:00
if ( dbus . shutdown ) {
2015-06-21 17:33:46 +02:00
dbus . shutdown ( ) ;
}
2018-02-13 08:07:52 -08:00
# endif
2017-05-28 07:08:10 -04:00
SDL_zero ( dbus ) ;
2015-06-21 17:33:46 +02:00
UnloadDBUSLibrary ( ) ;
2022-08-29 11:08:24 +02:00
SDL_free ( inhibit_handle ) ;
inhibit_handle = NULL ;
2015-06-21 17:33:46 +02:00
}
SDL_DBusContext *
SDL_DBus_GetContext ( void )
{
2022-11-27 17:38:43 +01:00
if ( dbus_handle = = NULL | | ! dbus . session_conn ) {
2015-06-21 17:33:46 +02:00
SDL_DBus_Init ( ) ;
}
2022-11-30 12:51:59 -08:00
2019-10-09 19:38:16 -04:00
return ( dbus_handle & & dbus . session_conn ) ? & dbus : NULL ;
2015-06-21 17:33:46 +02:00
}
2022-11-30 12:51:59 -08:00
static SDL_bool SDL_DBus_CallMethodInternal ( DBusConnection * conn , const char * node , const char * path , const char * interface , const char * method , va_list ap )
2015-06-21 17:33:46 +02:00
{
2017-05-28 07:11:52 -04:00
SDL_bool retval = SDL_FALSE ;
if ( conn ) {
DBusMessage * msg = dbus . message_new_method_call ( node , path , interface , method ) ;
if ( msg ) {
2018-07-13 17:53:24 -04:00
int firstarg ;
va_list ap_reply ;
2022-11-30 12:51:59 -08:00
va_copy ( ap_reply , ap ) ; /* copy the arg list so we don't compete with D-Bus for it */
2018-07-13 17:53:24 -04:00
firstarg = va_arg ( ap , int ) ;
2017-05-28 07:11:52 -04:00
if ( ( firstarg = = DBUS_TYPE_INVALID ) | | dbus . message_append_args_valist ( msg , firstarg , ap ) ) {
DBusMessage * reply = dbus . connection_send_with_reply_and_block ( conn , msg , 300 , NULL ) ;
if ( reply ) {
2018-07-13 17:53:24 -04:00
/* skip any input args, get to output args. */
while ( ( firstarg = va_arg ( ap_reply , int ) ) ! = DBUS_TYPE_INVALID ) {
/* we assume D-Bus already validated all this. */
2022-11-30 12:51:59 -08:00
{
void * dumpptr = va_arg ( ap_reply , void * ) ;
( void ) dumpptr ;
}
2018-07-13 17:53:24 -04:00
if ( firstarg = = DBUS_TYPE_ARRAY ) {
2022-11-30 12:51:59 -08:00
{
const int dumpint = va_arg ( ap_reply , int ) ;
( void ) dumpint ;
}
2018-07-13 17:53:24 -04:00
}
}
firstarg = va_arg ( ap_reply , int ) ;
if ( ( firstarg = = DBUS_TYPE_INVALID ) | | dbus . message_get_args_valist ( reply , NULL , firstarg , ap_reply ) ) {
2017-05-28 07:11:52 -04:00
retval = SDL_TRUE ;
}
dbus . message_unref ( reply ) ;
}
2015-06-21 17:33:46 +02:00
}
2018-07-13 17:53:24 -04:00
va_end ( ap_reply ) ;
2015-06-21 17:33:46 +02:00
dbus . message_unref ( msg ) ;
}
}
2017-05-28 07:11:52 -04:00
return retval ;
2015-06-21 17:33:46 +02:00
}
SDL_bool
2017-05-28 07:11:52 -04:00
SDL_DBus_CallMethodOnConnection ( DBusConnection * conn , const char * node , const char * path , const char * interface , const char * method , . . . )
2015-06-21 17:33:46 +02:00
{
2017-05-28 07:11:52 -04:00
SDL_bool retval ;
va_list ap ;
va_start ( ap , method ) ;
retval = SDL_DBus_CallMethodInternal ( conn , node , path , interface , method , ap ) ;
va_end ( ap ) ;
return retval ;
}
2015-06-21 17:33:46 +02:00
2017-05-28 07:11:52 -04:00
SDL_bool
SDL_DBus_CallMethod ( const char * node , const char * path , const char * interface , const char * method , . . . )
{
SDL_bool retval ;
va_list ap ;
va_start ( ap , method ) ;
retval = SDL_DBus_CallMethodInternal ( dbus . session_conn , node , path , interface , method , ap ) ;
va_end ( ap ) ;
return retval ;
}
2015-06-21 17:33:46 +02:00
2022-11-30 12:51:59 -08:00
static SDL_bool SDL_DBus_CallVoidMethodInternal ( DBusConnection * conn , const char * node , const char * path , const char * interface , const char * method , va_list ap )
2017-05-28 07:11:52 -04:00
{
SDL_bool retval = SDL_FALSE ;
2015-06-21 17:33:46 +02:00
2017-05-28 07:11:52 -04:00
if ( conn ) {
DBusMessage * msg = dbus . message_new_method_call ( node , path , interface , method ) ;
if ( msg ) {
int firstarg = va_arg ( ap , int ) ;
if ( ( firstarg = = DBUS_TYPE_INVALID ) | | dbus . message_append_args_valist ( msg , firstarg , ap ) ) {
if ( dbus . connection_send ( conn , msg , NULL ) ) {
dbus . connection_flush ( conn ) ;
retval = SDL_TRUE ;
}
}
dbus . message_unref ( msg ) ;
2015-06-21 17:33:46 +02:00
}
2017-05-28 07:11:52 -04:00
}
2015-06-21 17:33:46 +02:00
2017-05-28 07:11:52 -04:00
return retval ;
}
2015-06-21 17:33:46 +02:00
2022-11-30 12:51:59 -08:00
static SDL_bool SDL_DBus_CallWithBasicReply ( DBusConnection * conn , DBusMessage * msg , const int expectedtype , void * result )
2022-08-25 16:18:45 +02:00
{
SDL_bool retval = SDL_FALSE ;
DBusMessage * reply = dbus . connection_send_with_reply_and_block ( conn , msg , 300 , NULL ) ;
if ( reply ) {
DBusMessageIter iter , actual_iter ;
dbus . message_iter_init ( reply , & iter ) ;
if ( dbus . message_iter_get_arg_type ( & iter ) = = DBUS_TYPE_VARIANT ) {
dbus . message_iter_recurse ( & iter , & actual_iter ) ;
} else {
actual_iter = iter ;
}
if ( dbus . message_iter_get_arg_type ( & actual_iter ) = = expectedtype ) {
dbus . message_iter_get_basic ( & actual_iter , result ) ;
retval = SDL_TRUE ;
}
dbus . message_unref ( reply ) ;
}
return retval ;
}
2017-05-28 07:11:52 -04:00
SDL_bool
SDL_DBus_CallVoidMethodOnConnection ( DBusConnection * conn , const char * node , const char * path , const char * interface , const char * method , . . . )
{
SDL_bool retval ;
va_list ap ;
va_start ( ap , method ) ;
retval = SDL_DBus_CallVoidMethodInternal ( conn , node , path , interface , method , ap ) ;
va_end ( ap ) ;
return retval ;
}
SDL_bool
SDL_DBus_CallVoidMethod ( const char * node , const char * path , const char * interface , const char * method , . . . )
{
SDL_bool retval ;
va_list ap ;
va_start ( ap , method ) ;
retval = SDL_DBus_CallVoidMethodInternal ( dbus . session_conn , node , path , interface , method , ap ) ;
va_end ( ap ) ;
return retval ;
}
SDL_bool
SDL_DBus_QueryPropertyOnConnection ( DBusConnection * conn , const char * node , const char * path , const char * interface , const char * property , const int expectedtype , void * result )
{
SDL_bool retval = SDL_FALSE ;
2015-06-21 17:33:46 +02:00
2017-05-28 07:11:52 -04:00
if ( conn ) {
DBusMessage * msg = dbus . message_new_method_call ( node , path , " org.freedesktop.DBus.Properties " , " Get " ) ;
if ( msg ) {
if ( dbus . message_append_args ( msg , DBUS_TYPE_STRING , & interface , DBUS_TYPE_STRING , & property , DBUS_TYPE_INVALID ) ) {
2022-08-25 16:18:45 +02:00
retval = SDL_DBus_CallWithBasicReply ( conn , msg , expectedtype , result ) ;
2017-05-28 07:11:52 -04:00
}
2015-06-21 17:33:46 +02:00
dbus . message_unref ( msg ) ;
}
2017-05-28 07:11:52 -04:00
}
2015-06-21 17:33:46 +02:00
2017-05-28 07:11:52 -04:00
return retval ;
}
SDL_bool
SDL_DBus_QueryProperty ( const char * node , const char * path , const char * interface , const char * property , const int expectedtype , void * result )
{
return SDL_DBus_QueryPropertyOnConnection ( dbus . session_conn , node , path , interface , property , expectedtype , result ) ;
}
2022-11-30 12:51:59 -08:00
void SDL_DBus_ScreensaverTickle ( void )
2017-05-28 07:11:52 -04:00
{
2022-11-30 12:51:59 -08:00
if ( screensaver_cookie = = 0 & & inhibit_handle = = NULL ) { /* no need to tickle if we're inhibiting. */
2019-07-02 09:43:26 -04:00
/* org.gnome.ScreenSaver is the legacy interface, but it'll either do nothing or just be a second harmless tickle on newer systems, so we leave it for now. */
SDL_DBus_CallVoidMethod ( " org.gnome.ScreenSaver " , " /org/gnome/ScreenSaver " , " org.gnome.ScreenSaver " , " SimulateUserActivity " , DBUS_TYPE_INVALID ) ;
SDL_DBus_CallVoidMethod ( " org.freedesktop.ScreenSaver " , " /org/freedesktop/ScreenSaver " , " org.freedesktop.ScreenSaver " , " SimulateUserActivity " , DBUS_TYPE_INVALID ) ;
}
2017-05-28 07:11:52 -04:00
}
2022-11-30 12:51:59 -08:00
static SDL_bool SDL_DBus_AppendDictWithKeyValue ( DBusMessageIter * iterInit , const char * key , const char * value )
2022-08-29 11:08:24 +02:00
{
DBusMessageIter iterDict , iterEntry , iterValue ;
2022-11-27 17:38:43 +01:00
if ( ! dbus . message_iter_open_container ( iterInit , DBUS_TYPE_ARRAY , " {sv} " , & iterDict ) ) {
2022-08-29 11:08:24 +02:00
goto failed ;
2022-11-27 17:38:43 +01:00
}
2022-08-29 11:08:24 +02:00
2022-11-27 17:38:43 +01:00
if ( ! dbus . message_iter_open_container ( & iterDict , DBUS_TYPE_DICT_ENTRY , NULL , & iterEntry ) ) {
2022-08-29 11:08:24 +02:00
goto failed ;
2022-11-27 17:38:43 +01:00
}
2022-08-29 11:08:24 +02:00
2022-11-27 17:38:43 +01:00
if ( ! dbus . message_iter_append_basic ( & iterEntry , DBUS_TYPE_STRING , & key ) ) {
2022-08-29 11:08:24 +02:00
goto failed ;
2022-11-27 17:38:43 +01:00
}
2022-08-29 11:08:24 +02:00
2022-11-27 17:38:43 +01:00
if ( ! dbus . message_iter_open_container ( & iterEntry , DBUS_TYPE_VARIANT , DBUS_TYPE_STRING_AS_STRING , & iterValue ) ) {
2022-08-29 11:08:24 +02:00
goto failed ;
2022-11-27 17:38:43 +01:00
}
2022-08-29 11:08:24 +02:00
2022-11-27 17:38:43 +01:00
if ( ! dbus . message_iter_append_basic ( & iterValue , DBUS_TYPE_STRING , & value ) ) {
2022-08-29 11:08:24 +02:00
goto failed ;
2022-11-27 17:38:43 +01:00
}
2022-08-29 11:08:24 +02:00
2022-11-30 12:51:59 -08:00
if ( ! dbus . message_iter_close_container ( & iterEntry , & iterValue ) | | ! dbus . message_iter_close_container ( & iterDict , & iterEntry ) | | ! dbus . message_iter_close_container ( iterInit , & iterDict ) ) {
2022-08-29 11:08:24 +02:00
goto failed ;
}
return SDL_TRUE ;
failed :
/* message_iter_abandon_container_if_open() and message_iter_abandon_container() might be
* missing if libdbus is too old . Instead , we just return without cleaning up any eventual
* open container */
return SDL_FALSE ;
}
2017-05-28 07:11:52 -04:00
SDL_bool
SDL_DBus_ScreensaverInhibit ( SDL_bool inhibit )
{
2022-08-29 11:08:24 +02:00
const char * default_inhibit_reason = " Playing a game " ;
2022-11-30 12:51:59 -08:00
if ( ( inhibit & & ( screensaver_cookie ! = 0 | | inhibit_handle ! = NULL ) ) | | ( ! inhibit & & ( screensaver_cookie = = 0 & & inhibit_handle = = NULL ) ) ) {
2015-06-21 17:33:46 +02:00
return SDL_TRUE ;
2022-08-29 11:08:24 +02:00
}
2022-09-28 10:25:24 +02:00
if ( ! dbus . session_conn ) {
/* We either lost connection to the session bus or were not able to
* load the D - Bus library at all . */
return SDL_FALSE ;
}
2022-08-29 11:08:24 +02:00
if ( SDL_DetectSandbox ( ) ! = SDL_SANDBOX_NONE ) {
const char * bus_name = " org.freedesktop.portal.Desktop " ;
const char * path = " /org/freedesktop/portal/desktop " ;
const char * interface = " org.freedesktop.portal.Inhibit " ;
2022-11-30 12:51:59 -08:00
const char * window = " " ; /* As a future improvement we could gather the X11 XID or Wayland surface identifier */
2022-08-29 11:08:24 +02:00
static const unsigned int INHIBIT_IDLE = 8 ; /* Taken from the portal API reference */
DBusMessageIter iterInit ;
if ( inhibit ) {
DBusMessage * msg ;
SDL_bool retval = SDL_FALSE ;
const char * key = " reason " ;
const char * reply = NULL ;
const char * reason = SDL_GetHint ( SDL_HINT_SCREENSAVER_INHIBIT_ACTIVITY_NAME ) ;
2022-11-27 17:38:43 +01:00
if ( reason = = NULL | | ! reason [ 0 ] ) {
2022-08-29 11:08:24 +02:00
reason = default_inhibit_reason ;
}
msg = dbus . message_new_method_call ( bus_name , path , interface , " Inhibit " ) ;
2022-11-27 17:38:43 +01:00
if ( msg = = NULL ) {
2022-08-29 11:08:24 +02:00
return SDL_FALSE ;
}
if ( ! dbus . message_append_args ( msg , DBUS_TYPE_STRING , & window , DBUS_TYPE_UINT32 , & INHIBIT_IDLE , DBUS_TYPE_INVALID ) ) {
dbus . message_unref ( msg ) ;
return SDL_FALSE ;
}
dbus . message_iter_init_append ( msg , & iterInit ) ;
if ( ! SDL_DBus_AppendDictWithKeyValue ( & iterInit , key , reason ) ) {
dbus . message_unref ( msg ) ;
return SDL_FALSE ;
}
if ( SDL_DBus_CallWithBasicReply ( dbus . session_conn , msg , DBUS_TYPE_OBJECT_PATH , & reply ) ) {
inhibit_handle = SDL_strdup ( reply ) ;
retval = SDL_TRUE ;
}
dbus . message_unref ( msg ) ;
return retval ;
} else {
if ( ! SDL_DBus_CallVoidMethod ( bus_name , inhibit_handle , " org.freedesktop.portal.Request " , " Close " , DBUS_TYPE_INVALID ) ) {
return SDL_FALSE ;
}
SDL_free ( inhibit_handle ) ;
inhibit_handle = NULL ;
}
2015-06-21 17:33:46 +02:00
} else {
2022-08-29 11:08:24 +02:00
const char * bus_name = " org.freedesktop.ScreenSaver " ;
2017-05-28 07:11:52 -04:00
const char * path = " /org/freedesktop/ScreenSaver " ;
const char * interface = " org.freedesktop.ScreenSaver " ;
if ( inhibit ) {
2021-08-28 22:52:13 +08:00
const char * app = SDL_GetHint ( SDL_HINT_APP_NAME ) ;
const char * reason = SDL_GetHint ( SDL_HINT_SCREENSAVER_INHIBIT_ACTIVITY_NAME ) ;
2022-11-27 17:38:43 +01:00
if ( app = = NULL | | ! app [ 0 ] ) {
2022-11-30 12:51:59 -08:00
app = " My SDL application " ;
2021-08-28 22:52:13 +08:00
}
2022-11-27 17:38:43 +01:00
if ( reason = = NULL | | ! reason [ 0 ] ) {
2022-08-29 11:08:24 +02:00
reason = default_inhibit_reason ;
2021-08-28 22:52:13 +08:00
}
2022-08-29 11:08:24 +02:00
if ( ! SDL_DBus_CallMethod ( bus_name , path , interface , " Inhibit " ,
2022-11-30 12:51:59 -08:00
DBUS_TYPE_STRING , & app , DBUS_TYPE_STRING , & reason , DBUS_TYPE_INVALID ,
DBUS_TYPE_UINT32 , & screensaver_cookie , DBUS_TYPE_INVALID ) ) {
2017-05-28 07:11:52 -04:00
return SDL_FALSE ;
2015-06-21 17:33:46 +02:00
}
2017-05-28 07:11:52 -04:00
return ( screensaver_cookie ! = 0 ) ? SDL_TRUE : SDL_FALSE ;
} else {
2022-08-29 11:08:24 +02:00
if ( ! SDL_DBus_CallVoidMethod ( bus_name , path , interface , " UnInhibit " , DBUS_TYPE_UINT32 , & screensaver_cookie , DBUS_TYPE_INVALID ) ) {
2017-05-28 07:11:52 -04:00
return SDL_FALSE ;
}
screensaver_cookie = 0 ;
2015-06-21 17:33:46 +02:00
}
}
2017-05-28 07:11:52 -04:00
return SDL_TRUE ;
2015-06-21 17:33:46 +02:00
}
# endif