2015-06-21 17:33:46 +02:00
/*
Simple DirectMedia Layer
2025-01-01 07:47:53 -08:00
Copyright ( C ) 1997 - 2025 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 .
*/
# include "../../SDL_internal.h"
2021-08-28 22:52:13 +08:00
# include "SDL_hints.h"
2015-06-21 17:33:46 +02:00
# include "SDL_dbus.h"
2021-08-04 13:30:12 -04:00
# include "SDL_atomic.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-11-17 04:06:26 +01:00
# ifdef SDL_USE_LIBDBUS
2015-06-21 17:33:46 +02:00
/* we never link directly to libdbus. */
# include "SDL_loadso.h"
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-06-20 13:22:55 +01:00
# define SDL_DBUS_SYM2_OPTIONAL(x, y) \
dbus . x = SDL_LoadFunction ( dbus_handle , # y )
2022-11-30 12:51:59 -08:00
# define SDL_DBUS_SYM2(x, y) \
if ( ! ( dbus . x = SDL_LoadFunction ( dbus_handle , # y ) ) ) \
return - 1
# define SDL_DBUS_SYM(x) \
SDL_DBUS_SYM2 ( x , dbus_ # # x )
2015-06-21 17:33:46 +02:00
2023-06-20 13:22:55 +01:00
# define SDL_DBUS_SYM_OPTIONAL(x) \
SDL_DBUS_SYM2_OPTIONAL ( x , dbus_ # # x )
2015-06-21 17:33:46 +02:00
SDL_DBUS_SYM ( bus_get_private ) ;
SDL_DBUS_SYM ( bus_register ) ;
SDL_DBUS_SYM ( bus_add_match ) ;
SDL_DBUS_SYM ( connection_open_private ) ;
SDL_DBUS_SYM ( connection_set_exit_on_disconnect ) ;
SDL_DBUS_SYM ( connection_get_is_connected ) ;
SDL_DBUS_SYM ( connection_add_filter ) ;
SDL_DBUS_SYM ( connection_try_register_object_path ) ;
SDL_DBUS_SYM ( connection_send ) ;
SDL_DBUS_SYM ( connection_send_with_reply_and_block ) ;
SDL_DBUS_SYM ( connection_close ) ;
2022-12-14 17:41:35 +01:00
SDL_DBUS_SYM ( connection_ref ) ;
2015-06-21 17:33:46 +02:00
SDL_DBUS_SYM ( connection_unref ) ;
SDL_DBUS_SYM ( connection_flush ) ;
SDL_DBUS_SYM ( connection_read_write ) ;
SDL_DBUS_SYM ( connection_dispatch ) ;
SDL_DBUS_SYM ( message_is_signal ) ;
SDL_DBUS_SYM ( message_new_method_call ) ;
SDL_DBUS_SYM ( message_append_args ) ;
2017-05-28 07:11:52 -04:00
SDL_DBUS_SYM ( message_append_args_valist ) ;
2020-05-11 14:31:04 -07:00
SDL_DBUS_SYM ( message_iter_init_append ) ;
SDL_DBUS_SYM ( message_iter_open_container ) ;
SDL_DBUS_SYM ( message_iter_append_basic ) ;
SDL_DBUS_SYM ( message_iter_close_container ) ;
2015-06-21 17:33:46 +02:00
SDL_DBUS_SYM ( message_get_args ) ;
2017-05-28 07:11:52 -04:00
SDL_DBUS_SYM ( message_get_args_valist ) ;
2015-06-21 17:33:46 +02:00
SDL_DBUS_SYM ( message_iter_init ) ;
SDL_DBUS_SYM ( message_iter_next ) ;
SDL_DBUS_SYM ( message_iter_get_basic ) ;
SDL_DBUS_SYM ( message_iter_get_arg_type ) ;
SDL_DBUS_SYM ( message_iter_recurse ) ;
SDL_DBUS_SYM ( message_unref ) ;
2020-03-30 14:26:15 -07:00
SDL_DBUS_SYM ( threads_init_default ) ;
2015-06-21 17:33:46 +02:00
SDL_DBUS_SYM ( error_init ) ;
SDL_DBUS_SYM ( error_is_set ) ;
SDL_DBUS_SYM ( error_free ) ;
SDL_DBUS_SYM ( get_local_machine_id ) ;
2023-06-20 13:22:55 +01:00
SDL_DBUS_SYM_OPTIONAL ( try_get_local_machine_id ) ;
2015-06-21 17:33:46 +02:00
SDL_DBUS_SYM ( free ) ;
2017-05-28 07:14:11 -04:00
SDL_DBUS_SYM ( free_string_array ) ;
2015-06-21 17:33:46 +02:00
SDL_DBUS_SYM ( shutdown ) ;
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
{
2023-11-10 15:30:56 +01:00
if ( dbus_handle ) {
2015-06-21 17:33:46 +02:00
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 ;
2023-11-10 15:30:56 +01:00
if ( ! dbus_handle ) {
2015-06-21 17:33:46 +02:00
dbus_handle = SDL_LoadObject ( dbus_library ) ;
2023-11-10 15:30:56 +01:00
if ( ! dbus_handle ) {
2015-06-21 17:33:46 +02:00
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
}
2024-01-02 13:15:32 -05:00
2024-01-07 14:20:16 -08:00
if ( SDL_GetHintBoolean ( SDL_HINT_SHUTDOWN_DBUS_ON_QUIT , SDL_FALSE ) ) {
if ( dbus . shutdown ) {
dbus . shutdown ( ) ;
}
2015-06-21 17:33:46 +02:00
}
2024-01-02 13:15:32 -05:00
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
}
2023-05-23 10:59:03 -07:00
SDL_DBusContext * SDL_DBus_GetContext ( void )
2015-06-21 17:33:46 +02:00
{
2023-11-10 15:30:56 +01:00
if ( ! dbus_handle | | ! 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
}
2023-05-23 10:59:03 -07:00
SDL_bool 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
2023-05-23 10:59:03 -07:00
SDL_bool SDL_DBus_CallMethod ( const char * node , const char * path , const char * interface , const char * method , . . . )
2017-05-28 07:11:52 -04:00
{
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 ;
}
2023-05-23 10:59:03 -07:00
SDL_bool SDL_DBus_CallVoidMethodOnConnection ( DBusConnection * conn , const char * node , const char * path , const char * interface , const char * method , . . . )
2017-05-28 07:11:52 -04:00
{
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 ;
}
2023-05-23 10:59:03 -07:00
SDL_bool SDL_DBus_CallVoidMethod ( const char * node , const char * path , const char * interface , const char * method , . . . )
2017-05-28 07:11:52 -04:00
{
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 ;
}
2023-05-23 10:59:03 -07:00
SDL_bool SDL_DBus_QueryPropertyOnConnection ( DBusConnection * conn , const char * node , const char * path , const char * interface , const char * property , const int expectedtype , void * result )
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 , " 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 ;
}
2023-05-23 10:59:03 -07:00
SDL_bool SDL_DBus_QueryProperty ( const char * node , const char * path , const char * interface , const char * property , const int expectedtype , void * result )
2017-05-28 07:11:52 -04:00
{
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
{
2023-11-10 15:30:56 +01:00
if ( screensaver_cookie = = 0 & & ! inhibit_handle ) { /* 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 ;
}
2023-05-23 10:59:03 -07:00
SDL_bool SDL_DBus_ScreensaverInhibit ( SDL_bool inhibit )
2017-05-28 07:11:52 -04:00
{
2022-08-29 11:08:24 +02:00
const char * default_inhibit_reason = " Playing a game " ;
2023-11-10 15:30:56 +01:00
if ( ( inhibit & & ( screensaver_cookie ! = 0 | | inhibit_handle ) ) | | ( ! inhibit & & ( screensaver_cookie = = 0 & & ! inhibit_handle ) ) ) {
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 ) ;
2023-11-10 15:30:56 +01:00
if ( ! reason | | ! 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 " ) ;
2023-11-10 15:30:56 +01:00
if ( ! msg ) {
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 ) ;
2023-11-10 15:30:56 +01:00
if ( ! app | | ! app [ 0 ] ) {
2022-11-30 12:51:59 -08:00
app = " My SDL application " ;
2021-08-28 22:52:13 +08:00
}
2023-11-10 15:30:56 +01:00
if ( ! reason | | ! 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
}
2023-06-20 13:22:55 +01:00
/*
* Get the machine ID if possible . Result must be freed with dbus - > free ( ) .
*/
char * SDL_DBus_GetLocalMachineId ( void )
{
DBusError err ;
char * result ;
dbus . error_init ( & err ) ;
if ( dbus . try_get_local_machine_id ) {
/* Available since dbus 1.12.0, has proper error-handling */
result = dbus . try_get_local_machine_id ( & err ) ;
} else {
/* Available since time immemorial, but has no error-handling:
* if the machine ID can ' t be read , many versions of libdbus will
* treat that as a fatal mis - installation and abort ( ) */
result = dbus . get_local_machine_id ( ) ;
}
if ( result ) {
return result ;
}
if ( dbus . error_is_set ( & err ) ) {
SDL_SetError ( " %s: %s " , err . name , err . message ) ;
dbus . error_free ( & err ) ;
} else {
SDL_SetError ( " Error getting D-Bus machine ID " ) ;
}
return NULL ;
}
2023-09-16 19:08:16 +01:00
/*
* Convert file drops with mime type " application/vnd.portal.filetransfer " to file paths
* Result must be freed with dbus - > free_string_array ( ) .
* https : //flatpak.github.io/xdg-desktop-portal/#gdbus-method-org-freedesktop-portal-FileTransfer.RetrieveFiles
*/
char * * SDL_DBus_DocumentsPortalRetrieveFiles ( const char * key , int * path_count )
{
DBusError err ;
DBusMessageIter iter , iterDict ;
char * * paths = NULL ;
DBusMessage * reply = NULL ;
DBusMessage * msg = dbus . message_new_method_call ( " org.freedesktop.portal.Documents " , /* Node */
" /org/freedesktop/portal/documents " , /* Path */
" org.freedesktop.portal.FileTransfer " , /* Interface */
" RetrieveFiles " ) ; /* Method */
/* Make sure we have a connection to the dbus session bus */
if ( ! SDL_DBus_GetContext ( ) | | ! dbus . session_conn ) {
/* We either cannot connect to the session bus or were unable to
* load the D - Bus library at all . */
return NULL ;
}
dbus . error_init ( & err ) ;
/* First argument is a "application/vnd.portal.filetransfer" key from a DnD or clipboard event */
if ( ! dbus . message_append_args ( msg , DBUS_TYPE_STRING , & key , DBUS_TYPE_INVALID ) ) {
SDL_OutOfMemory ( ) ;
dbus . message_unref ( msg ) ;
goto failed ;
}
/* Second argument is a variant dictionary for options.
* The spec doesn ' t define any entries yet so it ' s empty . */
dbus . message_iter_init_append ( msg , & iter ) ;
if ( ! dbus . message_iter_open_container ( & iter , DBUS_TYPE_ARRAY , " {sv} " , & iterDict ) | |
! dbus . message_iter_close_container ( & iter , & iterDict ) ) {
SDL_OutOfMemory ( ) ;
dbus . message_unref ( msg ) ;
goto failed ;
}
reply = dbus . connection_send_with_reply_and_block ( dbus . session_conn , msg , DBUS_TIMEOUT_USE_DEFAULT , & err ) ;
dbus . message_unref ( msg ) ;
if ( reply ) {
dbus . message_get_args ( reply , & err , DBUS_TYPE_ARRAY , DBUS_TYPE_STRING , & paths , path_count , DBUS_TYPE_INVALID ) ;
dbus . message_unref ( reply ) ;
}
if ( paths ) {
return paths ;
}
failed :
if ( dbus . error_is_set ( & err ) ) {
SDL_SetError ( " %s: %s " , err . name , err . message ) ;
dbus . error_free ( & err ) ;
} else {
SDL_SetError ( " Error retrieving paths for documents portal \" %s \" " , key ) ;
}
return NULL ;
}
2015-06-21 17:33:46 +02:00
# endif
2016-11-29 05:34:20 -08:00
/* vi: set ts=4 sw=4 expandtab: */