/* $Id: hostversion.cpp 82968 2020-02-04 10:35:17Z vboxsync $ */ /** @file * X11 guest client - host version check. */ /* * Copyright (C) 2011-2020 Oracle Corporation * * This file is part of VirtualBox Open Source Edition (OSE), as * available from http://www.virtualbox.org. This file is free software; * you can redistribute it and/or modify it under the terms of the GNU * General Public License (GPL) as published by the Free Software * Foundation, in version 2 as it comes in the "COPYING" file of the * VirtualBox OSE distribution. VirtualBox OSE is distributed in the * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind. */ #include #include #include #include #include #include #include #ifdef VBOX_WITH_DBUS # include #endif #include #include #ifdef VBOX_OSE # include #endif #include "VBoxClient.h" static const char *getName() { return "Host Version Check"; } static const char *getPidFilePath() { return ".vboxclient-hostversion.pid"; } static int showNotify(const char *pszHeader, const char *pszBody) { int rc; # ifdef VBOX_WITH_DBUS DBusConnection *conn; DBusMessage* msg = NULL; conn = dbus_bus_get (DBUS_BUS_SESSION, NULL); if (conn == NULL) { LogRelFlowFunc(("Could not retrieve D-BUS session bus!\n")); rc = VERR_INVALID_HANDLE; } else { msg = dbus_message_new_method_call("org.freedesktop.Notifications", "/org/freedesktop/Notifications", "org.freedesktop.Notifications", "Notify"); if (msg == NULL) { VBClLogError("Could not create D-BUS message!\n"); rc = VERR_INVALID_HANDLE; } else rc = VINF_SUCCESS; } if (RT_SUCCESS(rc)) { uint32_t msg_replace_id = 0; const char *msg_app = "VBoxClient"; const char *msg_icon = ""; const char *msg_summary = pszHeader; const char *msg_body = pszBody; int32_t msg_timeout = -1; /* Let the notification server decide */ DBusMessageIter iter; DBusMessageIter array; /*DBusMessageIter dict; - unused */ /*DBusMessageIter value; - unused */ /*DBusMessageIter variant; - unused */ /*DBusMessageIter data; - unused */ /* Format: UINT32 org.freedesktop.Notifications.Notify * (STRING app_name, UINT32 replaces_id, STRING app_icon, STRING summary, STRING body, * ARRAY actions, DICT hints, INT32 expire_timeout) */ dbus_message_iter_init_append(msg,&iter); dbus_message_iter_append_basic(&iter,DBUS_TYPE_STRING,&msg_app); dbus_message_iter_append_basic(&iter,DBUS_TYPE_UINT32,&msg_replace_id); dbus_message_iter_append_basic(&iter,DBUS_TYPE_STRING,&msg_icon); dbus_message_iter_append_basic(&iter,DBUS_TYPE_STRING,&msg_summary); dbus_message_iter_append_basic(&iter,DBUS_TYPE_STRING,&msg_body); dbus_message_iter_open_container(&iter,DBUS_TYPE_ARRAY,DBUS_TYPE_STRING_AS_STRING,&array); dbus_message_iter_close_container(&iter,&array); dbus_message_iter_open_container(&iter,DBUS_TYPE_ARRAY,"{sv}",&array); dbus_message_iter_close_container(&iter,&array); dbus_message_iter_append_basic(&iter,DBUS_TYPE_INT32,&msg_timeout); DBusError err; dbus_error_init(&err); DBusMessage *reply; reply = dbus_connection_send_with_reply_and_block(conn, msg, 30 * 1000 /* 30 seconds timeout */, &err); if (dbus_error_is_set(&err)) VBClLogError("D-BUS returned an error while sending the notification: %s", err.message); else if (reply) { dbus_connection_flush(conn); dbus_message_unref(reply); } if (dbus_error_is_set(&err)) dbus_error_free(&err); } if (msg != NULL) dbus_message_unref(msg); # else /** @todo Implement me */ RT_NOREF(pszHeader, pszBody); rc = VINF_SUCCESS; # endif /* VBOX_WITH_DBUS */ return rc; } /** @todo Move this part in VbglR3 and just provide a callback for the platform-specific notification stuff, since this is very similar to the VBoxTray code. */ static int run(struct VBCLSERVICE **ppInterface, bool fDaemonised) { int rc; LogFlowFunc(("\n")); RT_NOREF(ppInterface); /* Because we need desktop notifications to be displayed, wait * some time to make the desktop environment load (as a work around). */ if (fDaemonised) RTThreadSleep(30 * 1000 /* Wait 30 seconds */); # ifdef VBOX_WITH_DBUS rc = RTDBusLoadLib(); if (RT_FAILURE(rc)) VBClLogError("D-Bus seems not to be installed; no host version check/notification done\n"); # else rc = VERR_NOT_IMPLEMENTED; # endif /* VBOX_WITH_DBUS */ # ifdef VBOX_WITH_GUEST_PROPS uint32_t uGuestPropSvcClientID; if (RT_SUCCESS(rc)) { rc = VbglR3GuestPropConnect(&uGuestPropSvcClientID); if (RT_FAILURE(rc)) VBClLogError("Cannot connect to guest property service while chcking for host version! rc = %Rrc\n", rc); } if (RT_SUCCESS(rc)) { char *pszHostVersion; char *pszGuestVersion; bool bUpdate; rc = VbglR3HostVersionCheckForUpdate(uGuestPropSvcClientID, &bUpdate, &pszHostVersion, &pszGuestVersion); if (RT_SUCCESS(rc)) { if (bUpdate) { char szMsg[1024]; char szTitle[64]; /** @todo add some translation macros here */ RTStrPrintf(szTitle, sizeof(szTitle), "VirtualBox Guest Additions update available!"); #ifndef VBOX_OSE RTStrPrintf(szMsg, sizeof(szMsg), "Your guest is currently running the Guest Additions version %s. " "We recommend updating to the latest version (%s) by choosing the " "install option from the Devices menu.", pszGuestVersion, pszHostVersion); #else /* This is the message which appears for non-Oracle builds of the * Guest Additions. Distributors are encouraged to customise this. */ RTStrPrintf(szMsg, sizeof(szMsg), "Your virtual machine is currently running the Guest Additions version %s. Since you are running a version of the Guest Additions provided by the operating system you installed in the virtual machine we recommend that you update it to at least version %s using that system's update features, or alternatively that you remove this version and then install the " VBOX_VENDOR_SHORT " Guest Additions package using the install option from the Devices menu. Please consult the documentation for the operating system you are running to find out how to update or remove the current Guest Additions package.", pszGuestVersion, pszHostVersion); #endif rc = showNotify(szTitle, szMsg); VBClLogInfo("VirtualBox Guest Additions update available!\n"); if (RT_FAILURE(rc)) VBClLogError("Could not show version notifier tooltip! rc = %d\n", rc); } /* Store host version to not notify again */ rc = VbglR3HostVersionLastCheckedStore(uGuestPropSvcClientID, pszHostVersion); VbglR3GuestPropReadValueFree(pszHostVersion); VbglR3GuestPropReadValueFree(pszGuestVersion); } VbglR3GuestPropDisconnect(uGuestPropSvcClientID); } # endif /* VBOX_WITH_GUEST_PROPS */ VbglR3Term(); LogFlowFunc(("returning %Rrc\n", rc)); return rc; } struct VBCLSERVICE vbclHostVersionInterface = { getName, getPidFilePath, VBClServiceDefaultHandler, /* init */ run, VBClServiceDefaultCleanup }; struct HOSTVERSIONSERVICE { struct VBCLSERVICE *pInterface; }; /* Static factory */ struct VBCLSERVICE **VBClGetHostVersionService() { struct HOSTVERSIONSERVICE *pService = (struct HOSTVERSIONSERVICE *)RTMemAlloc(sizeof(*pService)); if (!pService) VBClLogFatalError("Out of memory\n"); pService->pInterface = &vbclHostVersionInterface; return &pService->pInterface; }