123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641 |
- /*
- Simple DirectMedia Layer
- Copyright (C) 1997-2025 Sam Lantinga <[email protected]>
- 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"
- #include "SDL_dbus.h"
- #include "../../stdlib/SDL_vacopy.h"
- #ifdef SDL_USE_LIBDBUS
- // we never link directly to libdbus.
- static const char *dbus_library = "libdbus-1.so.3";
- static SDL_SharedObject *dbus_handle = NULL;
- static char *inhibit_handle = NULL;
- static unsigned int screensaver_cookie = 0;
- static SDL_DBusContext dbus;
- static bool LoadDBUSSyms(void)
- {
- #define SDL_DBUS_SYM2_OPTIONAL(TYPE, x, y) \
- dbus.x = (TYPE)SDL_LoadFunction(dbus_handle, #y)
- #define SDL_DBUS_SYM2(TYPE, x, y) \
- if (!(dbus.x = (TYPE)SDL_LoadFunction(dbus_handle, #y))) \
- return false
- #define SDL_DBUS_SYM_OPTIONAL(TYPE, x) \
- SDL_DBUS_SYM2_OPTIONAL(TYPE, x, dbus_##x)
- #define SDL_DBUS_SYM(TYPE, x) \
- SDL_DBUS_SYM2(TYPE, x, dbus_##x)
- 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 *, DBusHandleMessageFunction, void *), connection_remove_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(dbus_bool_t (*)(DBusMessage *, const char *), message_has_path);
- 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_OPTIONAL(char *(*)(DBusError *), try_get_local_machine_id);
- SDL_DBUS_SYM(void (*)(void *), free);
- SDL_DBUS_SYM(void (*)(char **), free_string_array);
- SDL_DBUS_SYM(void (*)(void), shutdown);
- #undef SDL_DBUS_SYM
- #undef SDL_DBUS_SYM2
- return true;
- }
- static void UnloadDBUSLibrary(void)
- {
- if (dbus_handle) {
- SDL_UnloadObject(dbus_handle);
- dbus_handle = NULL;
- }
- }
- static bool LoadDBUSLibrary(void)
- {
- bool result = true;
- if (!dbus_handle) {
- dbus_handle = SDL_LoadObject(dbus_library);
- if (!dbus_handle) {
- result = false;
- // Don't call SDL_SetError(): SDL_LoadObject already did.
- } else {
- result = LoadDBUSSyms();
- if (!result) {
- UnloadDBUSLibrary();
- }
- }
- }
- return result;
- }
- static SDL_InitState dbus_init;
- void SDL_DBus_Init(void)
- {
- static bool is_dbus_available = true;
- if (!is_dbus_available) {
- return; // don't keep trying if this fails.
- }
- if (!SDL_ShouldInit(&dbus_init)) {
- return;
- }
- if (!LoadDBUSLibrary()) {
- goto error;
- }
- if (!dbus.threads_init_default()) {
- goto error;
- }
- DBusError err;
- dbus.error_init(&err);
- // session bus is required
- dbus.session_conn = dbus.bus_get_private(DBUS_BUS_SESSION, &err);
- if (dbus.error_is_set(&err)) {
- dbus.error_free(&err);
- goto error;
- }
- dbus.connection_set_exit_on_disconnect(dbus.session_conn, 0);
- // 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);
- SDL_SetInitialized(&dbus_init, true);
- return;
- error:
- is_dbus_available = false;
- SDL_SetInitialized(&dbus_init, true);
- SDL_DBus_Quit();
- }
- void SDL_DBus_Quit(void)
- {
- if (!SDL_ShouldQuit(&dbus_init)) {
- return;
- }
- if (dbus.system_conn) {
- dbus.connection_close(dbus.system_conn);
- dbus.connection_unref(dbus.system_conn);
- }
- if (dbus.session_conn) {
- dbus.connection_close(dbus.session_conn);
- dbus.connection_unref(dbus.session_conn);
- }
- if (SDL_GetHintBoolean(SDL_HINT_SHUTDOWN_DBUS_ON_QUIT, false)) {
- if (dbus.shutdown) {
- dbus.shutdown();
- }
- UnloadDBUSLibrary();
- } else {
- /* Leaving libdbus loaded when skipping dbus_shutdown() avoids
- * spurious leak warnings from LeakSanitizer on internal D-Bus
- * allocations that would be freed by dbus_shutdown(). */
- dbus_handle = NULL;
- }
- SDL_zero(dbus);
- if (inhibit_handle) {
- SDL_free(inhibit_handle);
- inhibit_handle = NULL;
- }
- SDL_SetInitialized(&dbus_init, false);
- }
- SDL_DBusContext *SDL_DBus_GetContext(void)
- {
- if (!dbus_handle || !dbus.session_conn) {
- SDL_DBus_Init();
- }
- return (dbus_handle && dbus.session_conn) ? &dbus : NULL;
- }
- static bool SDL_DBus_CallMethodInternal(DBusConnection *conn, const char *node, const char *path, const char *interface, const char *method, va_list ap)
- {
- bool result = false;
- if (conn) {
- DBusMessage *msg = dbus.message_new_method_call(node, path, interface, method);
- if (msg) {
- int firstarg;
- va_list ap_reply;
- va_copy(ap_reply, ap); // copy the arg list so we don't compete with D-Bus for it
- firstarg = va_arg(ap, int);
- 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) {
- // 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.
- {
- void *dumpptr = va_arg(ap_reply, void *);
- (void)dumpptr;
- }
- if (firstarg == DBUS_TYPE_ARRAY) {
- {
- const int dumpint = va_arg(ap_reply, int);
- (void)dumpint;
- }
- }
- }
- firstarg = va_arg(ap_reply, int);
- if ((firstarg == DBUS_TYPE_INVALID) || dbus.message_get_args_valist(reply, NULL, firstarg, ap_reply)) {
- result = true;
- }
- dbus.message_unref(reply);
- }
- }
- va_end(ap_reply);
- dbus.message_unref(msg);
- }
- }
- return result;
- }
- bool SDL_DBus_CallMethodOnConnection(DBusConnection *conn, const char *node, const char *path, const char *interface, const char *method, ...)
- {
- bool result;
- va_list ap;
- va_start(ap, method);
- result = SDL_DBus_CallMethodInternal(conn, node, path, interface, method, ap);
- va_end(ap);
- return result;
- }
- bool SDL_DBus_CallMethod(const char *node, const char *path, const char *interface, const char *method, ...)
- {
- bool result;
- va_list ap;
- va_start(ap, method);
- result = SDL_DBus_CallMethodInternal(dbus.session_conn, node, path, interface, method, ap);
- va_end(ap);
- return result;
- }
- static bool SDL_DBus_CallVoidMethodInternal(DBusConnection *conn, const char *node, const char *path, const char *interface, const char *method, va_list ap)
- {
- bool result = false;
- 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);
- result = true;
- }
- }
- dbus.message_unref(msg);
- }
- }
- return result;
- }
- static bool SDL_DBus_CallWithBasicReply(DBusConnection *conn, DBusMessage *msg, const int expectedtype, void *result)
- {
- bool retval = 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 = true;
- }
- dbus.message_unref(reply);
- }
- return retval;
- }
- bool SDL_DBus_CallVoidMethodOnConnection(DBusConnection *conn, const char *node, const char *path, const char *interface, const char *method, ...)
- {
- bool result;
- va_list ap;
- va_start(ap, method);
- result = SDL_DBus_CallVoidMethodInternal(conn, node, path, interface, method, ap);
- va_end(ap);
- return result;
- }
- bool SDL_DBus_CallVoidMethod(const char *node, const char *path, const char *interface, const char *method, ...)
- {
- bool result;
- va_list ap;
- va_start(ap, method);
- result = SDL_DBus_CallVoidMethodInternal(dbus.session_conn, node, path, interface, method, ap);
- va_end(ap);
- return result;
- }
- bool SDL_DBus_QueryPropertyOnConnection(DBusConnection *conn, const char *node, const char *path, const char *interface, const char *property, int expectedtype, void *result)
- {
- bool retval = false;
- 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)) {
- retval = SDL_DBus_CallWithBasicReply(conn, msg, expectedtype, result);
- }
- dbus.message_unref(msg);
- }
- }
- return retval;
- }
- bool SDL_DBus_QueryProperty(const char *node, const char *path, const char *interface, const char *property, int expectedtype, void *result)
- {
- return SDL_DBus_QueryPropertyOnConnection(dbus.session_conn, node, path, interface, property, expectedtype, result);
- }
- void SDL_DBus_ScreensaverTickle(void)
- {
- if (screensaver_cookie == 0 && !inhibit_handle) { // no need to tickle if we're inhibiting.
- // 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);
- }
- }
- static bool SDL_DBus_AppendDictWithKeysAndValues(DBusMessageIter *iterInit, const char **keys, const char **values, int count)
- {
- DBusMessageIter iterDict;
- if (!dbus.message_iter_open_container(iterInit, DBUS_TYPE_ARRAY, "{sv}", &iterDict)) {
- goto failed;
- }
- for (int i = 0; i < count; i++) {
- DBusMessageIter iterEntry, iterValue;
- const char *key = keys[i];
- const char *value = values[i];
- if (!dbus.message_iter_open_container(&iterDict, DBUS_TYPE_DICT_ENTRY, NULL, &iterEntry)) {
- goto failed;
- }
- if (!dbus.message_iter_append_basic(&iterEntry, DBUS_TYPE_STRING, &key)) {
- goto failed;
- }
- if (!dbus.message_iter_open_container(&iterEntry, DBUS_TYPE_VARIANT, DBUS_TYPE_STRING_AS_STRING, &iterValue)) {
- goto failed;
- }
- if (!dbus.message_iter_append_basic(&iterValue, DBUS_TYPE_STRING, &value)) {
- goto failed;
- }
- if (!dbus.message_iter_close_container(&iterEntry, &iterValue) || !dbus.message_iter_close_container(&iterDict, &iterEntry)) {
- goto failed;
- }
- }
- if (!dbus.message_iter_close_container(iterInit, &iterDict)) {
- goto failed;
- }
- return 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 false;
- }
- static bool SDL_DBus_AppendDictWithKeyValue(DBusMessageIter *iterInit, const char *key, const char *value)
- {
- const char *keys[1];
- const char *values[1];
- keys[0] = key;
- values[0] = value;
- return SDL_DBus_AppendDictWithKeysAndValues(iterInit, keys, values, 1);
- }
- bool SDL_DBus_ScreensaverInhibit(bool inhibit)
- {
- const char *default_inhibit_reason = "Playing a game";
- if ((inhibit && (screensaver_cookie != 0 || inhibit_handle)) || (!inhibit && (screensaver_cookie == 0 && !inhibit_handle))) {
- return true;
- }
- 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 false;
- }
- if (SDL_GetSandbox() != 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";
- const char *window = ""; // As a future improvement we could gather the X11 XID or Wayland surface identifier
- static const unsigned int INHIBIT_IDLE = 8; // Taken from the portal API reference
- DBusMessageIter iterInit;
- if (inhibit) {
- DBusMessage *msg;
- bool result = false;
- const char *key = "reason";
- const char *reply = NULL;
- const char *reason = SDL_GetHint(SDL_HINT_SCREENSAVER_INHIBIT_ACTIVITY_NAME);
- if (!reason || !reason[0]) {
- reason = default_inhibit_reason;
- }
- msg = dbus.message_new_method_call(bus_name, path, interface, "Inhibit");
- if (!msg) {
- return false;
- }
- if (!dbus.message_append_args(msg, DBUS_TYPE_STRING, &window, DBUS_TYPE_UINT32, &INHIBIT_IDLE, DBUS_TYPE_INVALID)) {
- dbus.message_unref(msg);
- return false;
- }
- dbus.message_iter_init_append(msg, &iterInit);
- // a{sv}
- if (!SDL_DBus_AppendDictWithKeyValue(&iterInit, key, reason)) {
- dbus.message_unref(msg);
- return false;
- }
- if (SDL_DBus_CallWithBasicReply(dbus.session_conn, msg, DBUS_TYPE_OBJECT_PATH, &reply)) {
- inhibit_handle = SDL_strdup(reply);
- result = true;
- }
- dbus.message_unref(msg);
- return result;
- } else {
- if (!SDL_DBus_CallVoidMethod(bus_name, inhibit_handle, "org.freedesktop.portal.Request", "Close", DBUS_TYPE_INVALID)) {
- return false;
- }
- SDL_free(inhibit_handle);
- inhibit_handle = NULL;
- }
- } else {
- const char *bus_name = "org.freedesktop.ScreenSaver";
- const char *path = "/org/freedesktop/ScreenSaver";
- const char *interface = "org.freedesktop.ScreenSaver";
- if (inhibit) {
- const char *app = SDL_GetAppMetadataProperty(SDL_PROP_APP_METADATA_NAME_STRING);
- const char *reason = SDL_GetHint(SDL_HINT_SCREENSAVER_INHIBIT_ACTIVITY_NAME);
- if (!reason || !reason[0]) {
- reason = default_inhibit_reason;
- }
- if (!SDL_DBus_CallMethod(bus_name, path, interface, "Inhibit",
- DBUS_TYPE_STRING, &app, DBUS_TYPE_STRING, &reason, DBUS_TYPE_INVALID,
- DBUS_TYPE_UINT32, &screensaver_cookie, DBUS_TYPE_INVALID)) {
- return false;
- }
- return (screensaver_cookie != 0);
- } else {
- if (!SDL_DBus_CallVoidMethod(bus_name, path, interface, "UnInhibit", DBUS_TYPE_UINT32, &screensaver_cookie, DBUS_TYPE_INVALID)) {
- return false;
- }
- screensaver_cookie = 0;
- }
- }
- return true;
- }
- void SDL_DBus_PumpEvents(void)
- {
- if (dbus.session_conn) {
- dbus.connection_read_write(dbus.session_conn, 0);
- while (dbus.connection_dispatch(dbus.session_conn) == DBUS_DISPATCH_DATA_REMAINS) {
- // Do nothing, actual work happens in DBus_MessageFilter
- SDL_DelayNS(SDL_US_TO_NS(10));
- }
- }
- }
- /*
- * 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;
- }
- /*
- * 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;
- }
- #endif
|