diff options
Diffstat (limited to 'capplets/system-info')
-rw-r--r-- | capplets/system-info/Makefile.am | 45 | ||||
-rw-r--r-- | capplets/system-info/info-cleanup.c | 133 | ||||
-rw-r--r-- | capplets/system-info/info-cleanup.h | 23 | ||||
-rw-r--r-- | capplets/system-info/main.c | 35 | ||||
-rw-r--r-- | capplets/system-info/mate-system-info.c | 758 | ||||
-rw-r--r-- | capplets/system-info/mate-system-info.desktop.in | 13 | ||||
-rw-r--r-- | capplets/system-info/mate-system-info.h | 41 | ||||
-rw-r--r-- | capplets/system-info/mate-system-info.ui | 156 | ||||
-rw-r--r-- | capplets/system-info/system-info.gresource.xml | 6 |
9 files changed, 1210 insertions, 0 deletions
diff --git a/capplets/system-info/Makefile.am b/capplets/system-info/Makefile.am new file mode 100644 index 00000000..2e87da31 --- /dev/null +++ b/capplets/system-info/Makefile.am @@ -0,0 +1,45 @@ +# This is used in MATECC_CAPPLETS_CFLAGS +cappletname = system-info + +ui_files = mate-system-info.ui + +bin_PROGRAMS = mate-system-info + +mate_system_info_LDADD = $(MATECC_CAPPLETS_LIBS) +BUILT_SOURCES = mate-system-info-resources.h mate-system-info-resources.c +nodist_mate_system_info_SOURCES = \ + $(BUILT_SOURCES) +mate_system_info_SOURCES = \ + mate-system-info.c \ + mate-system-info.h \ + info-cleanup.c \ + info-cleanup.h \ + main.c + +include $(top_srcdir)/gla11y.mk + +desktopdir = $(datadir)/applications +Desktop_in_files = mate-system-info.desktop.in +desktop_DATA = $(Desktop_in_files:.desktop.in=.desktop) + +$(desktop_DATA): $(Desktop_in_files) +if USE_NLS + $(AM_V_GEN) $(MSGFMT) --desktop --keyword= --keyword=Name --keyword=Comment --keyword=Keywords --template $< -d $(top_srcdir)/po -o $@ +else + $(AM_V_GEN) sed '/^# Translators/d' < $< > $@ +endif + +mate-system-info-resources.h mate-system-info-resources.c: system-info.gresource.xml Makefile $(shell $(GLIB_COMPILE_RESOURCES) --generate-dependencies --sourcedir $(srcdir) $(srcdir)/system-info.gresource.xml) + $(AM_V_GEN) XMLLINT=$(XMLLINT) $(GLIB_COMPILE_RESOURCES) --target $@ --sourcedir $(srcdir) --generate --c-name mate_system_info $< + +AM_CPPFLAGS = \ + $(WARN_CFLAGS) \ + $(MATECC_CAPPLETS_CFLAGS) \ + -DMATECC_DATA_DIR="\"$(pkgdatadir)\"" +CLEANFILES = $(MATECC_CAPPLETS_CLEANFILES) $(desktop_DATA) $(BUILT_SOURCES) +EXTRA_DIST = \ + $(ui_files) \ + $(Desktop_in_files) \ + system-info.gresource.xml + +-include $(top_srcdir)/git.mk diff --git a/capplets/system-info/info-cleanup.c b/capplets/system-info/info-cleanup.c new file mode 100644 index 00000000..588ed4f3 --- /dev/null +++ b/capplets/system-info/info-cleanup.c @@ -0,0 +1,133 @@ +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- + * + * Copyright (C) 2010 Red Hat, Inc + * Copyright (C) 2008 William Jon McCann <[email protected]> + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, see <http://www.gnu.org/licenses/>. + * + */ + +#include <glib.h> +#include "info-cleanup.h" + +typedef struct +{ + char *regex; + char *replacement; +} ReplaceStrings; + +static char * +prettify_info (const char *info) +{ + g_autofree char *escaped = NULL; + g_autofree gchar *pretty = NULL; + int i; + static const ReplaceStrings rs[] = { + { "Mesa DRI ", ""}, + { "Mesa Intel", "Intel"}, + { "[(]R[)]", "\302\256"}, + { "[(](tm|TM)[)]", "\342\204\242"}, + { "(ATI|EPYC|AMD FX|Radeon|Ryzen|Threadripper|GeForce RTX) ", "\\1\342\204\242 "}, + { "Gallium \\d+\\.\\d+ on (.*)", "\\1"}, + { " CPU| Processor| \\S+-Core| @ \\d+\\.\\d+GHz", ""}, + { " x86|/MMX|/SSE2|/PCIe", ""}, + { " [(][^)]*(DRM|MESA|LLVM)[^)]*[)]?", ""}, + { "Graphics Controller", "Graphics"}, + { ".*llvmpipe.*", "Software Rendering"}, + { "(AMD .*) [(].*", "\\1"}, + { "(AMD [A-Z])(.*)", "\\1\\L\\2\\E"}, + { "AMD", "AMD<sup>\302\256</sup>"}, + + }; + + if (*info == '\0') + return NULL; + + escaped = g_markup_escape_text (info, -1); + pretty = g_strdup (g_strstrip (escaped)); + + for (i = 0; i < G_N_ELEMENTS (rs); i++) + { + g_autoptr(GError) error = NULL; + g_autoptr(GRegex) re = NULL; + g_autofree gchar *new = NULL; + + re = g_regex_new (rs[i].regex, 0, 0, &error); + if (re == NULL) + { + g_warning ("Error building regex: %s", error->message); + continue; + } + + new = g_regex_replace (re, + pretty, + -1, + 0, + rs[i].replacement, + 0, + &error); + + if (error != NULL) + { + g_warning ("Error replacing %s: %s", rs[i].regex, error->message); + continue; + } + + g_free (pretty); + pretty = g_steal_pointer (&new); + } + + return g_steal_pointer (&pretty); +} + +static char * +remove_duplicate_whitespace (const char *old) +{ + g_autofree gchar *new = NULL; + g_autoptr(GRegex) re = NULL; + g_autoptr(GError) error = NULL; + + if (old == NULL) + return NULL; + + re = g_regex_new ("[ \t\n\r]+", G_REGEX_MULTILINE, 0, &error); + if (re == NULL) + { + g_warning ("Error building regex: %s", error->message); + return g_strdup (old); + } + new = g_regex_replace (re, + old, + -1, + 0, + " ", + 0, + &error); + if (new == NULL) + { + g_warning ("Error replacing string: %s", error->message); + return g_strdup (old); + } + + return g_steal_pointer (&new); +} + +char * +info_cleanup (const char *input) +{ + g_autofree char *pretty = NULL; + + pretty = prettify_info (input); + return remove_duplicate_whitespace (pretty); +} diff --git a/capplets/system-info/info-cleanup.h b/capplets/system-info/info-cleanup.h new file mode 100644 index 00000000..cbc8d541 --- /dev/null +++ b/capplets/system-info/info-cleanup.h @@ -0,0 +1,23 @@ +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- + * + * Copyright (C) 2010 Red Hat, Inc + * Copyright (C) 2008 William Jon McCann <[email protected]> + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, see <http://www.gnu.org/licenses/>. + * + */ + +#include <glib.h> + +char *info_cleanup (const char *input); diff --git a/capplets/system-info/main.c b/capplets/system-info/main.c new file mode 100644 index 00000000..41db34fe --- /dev/null +++ b/capplets/system-info/main.c @@ -0,0 +1,35 @@ +/************************************************************************* + File Name: main.c + Copyright (C) 2023 MATE Developers + This program is free software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see <https://www.gnu.org/licenses/>. + +************************************************************************/ +#include "mate-system-info.h" +#include "capplet-util.h" + +int main (int argc, char *argv[]) +{ + GtkWidget *dialog; + + capplet_init (NULL, &argc, &argv); + + dialog = mate_system_info_new (); + mate_system_info_setup (MATE_SYSTEM_INFO (dialog)); + g_signal_connect (dialog, "response", G_CALLBACK (gtk_main_quit), NULL); + gtk_widget_show (dialog); + + gtk_main (); + + return 0; +} diff --git a/capplets/system-info/mate-system-info.c b/capplets/system-info/mate-system-info.c new file mode 100644 index 00000000..36380678 --- /dev/null +++ b/capplets/system-info/mate-system-info.c @@ -0,0 +1,758 @@ +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- + * + * Copyright (C) 2023 MATE Developers + * Copyright (C) 2019 Purism SPC + * Copyright (C) 2017 Mohammed Sadiq <[email protected]> + * Copyright (C) 2010 Red Hat, Inc + * Copyright (C) 2008 William Jon McCann <[email protected]> + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, see <http://www.gnu.org/licenses/>. + * + */ +#include <gtk/gtk.h> +#include <glibtop/fsusage.h> +#include <glibtop/mountlist.h> +#include <glibtop/mem.h> +#include <glibtop/sysinfo.h> +#include <udisks/udisks.h> +#include <sys/utsname.h> + +#ifdef GDK_WINDOWING_WAYLAND +#include <gdk/gdkwayland.h> +#endif +#ifdef GDK_WINDOWING_X11 +#include <gdk/gdkx.h> +#endif + +#include "info-cleanup.h" +#include "mate-system-info.h" +#include "mate-system-info-resources.h" + +struct _MateSystemInfo +{ + GtkDialog parent_instance; + GtkWidget *logo_image; + GtkWidget *hostname_row; + GtkListBox *hardware_box; + GtkWidget *hardware_model_row; + GtkWidget *memory_row; + GtkWidget *processor_row; + GtkWidget *graphics_row; + GtkWidget *disk_row; + GtkListBox *os_box; + GtkWidget *kernel_row; + GtkWidget *virtualization_row; + GtkWidget *windowing_system_row; + GtkWidget *mate_version_row; + GtkWidget *os_name_row; + GtkWidget *os_type_row; +}; + +G_DEFINE_TYPE (MateSystemInfo, mate_system_info, GTK_TYPE_DIALOG) + +static void +set_lable_style (GtkWidget *lable, + const char *color, + int font_szie, + const char *text, + gboolean blod) +{ + g_autofree gchar *lable_text = NULL; + + if (color == NULL) + { + lable_text = g_strdup_printf ("<span weight=\'light\'font_desc=\'%d\'><b>%s</b></span>", font_szie, text); + } + else + { + if(blod) + { + lable_text = g_strdup_printf ("<span foreground=\'%s\'weight=\'light\'font_desc=\'%d\'><b>%s</b></span>", + color, + font_szie, + text); + } + else + { + lable_text = g_strdup_printf ("<span foreground=\'%s\'weight=\'light\'font_desc=\'%d\'>%s</span>", + color, + font_szie, + text); + } + } + + gtk_label_set_markup (GTK_LABEL(lable), lable_text); +} + +static void +mate_system_info_row_fill (GtkWidget *row, + const char *labelname, + gboolean is_separator) +{ + GtkWidget *vbox; + GtkWidget *box; + GtkWidget *label; + GtkWidget *separator; + + vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0); + box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0); + gtk_box_set_spacing (GTK_BOX (box), 12); + g_object_set (box, "margin", 12, NULL); + gtk_box_pack_start (GTK_BOX (vbox), box, FALSE, FALSE, 3); + + label = gtk_label_new (NULL); + set_lable_style (label, NULL, 12, labelname, TRUE); + gtk_label_set_ellipsize (GTK_LABEL (label), PANGO_ELLIPSIZE_END); + gtk_box_pack_start (GTK_BOX (box), label, FALSE, FALSE, 6); + + label = gtk_label_new (NULL); + gtk_box_pack_end (GTK_BOX (box), label, FALSE, FALSE, 6); + g_object_set_data (G_OBJECT (row), "labelvalue", label); + + separator = gtk_separator_new (GTK_ORIENTATION_HORIZONTAL); + if (is_separator) + gtk_box_pack_start (GTK_BOX (vbox), separator, FALSE, FALSE, 0); + + gtk_container_add (GTK_CONTAINER (row), vbox); +} + +static void +mate_system_info_set_row (MateSystemInfo *info) +{ + mate_system_info_row_fill (info->hostname_row, _("Device Name"), FALSE); + mate_system_info_row_fill (info->hardware_model_row, _("Hardware Model"), TRUE); + mate_system_info_row_fill (info->memory_row, _("Memory"), TRUE); + mate_system_info_row_fill (info->processor_row, _("Processor"), TRUE); + mate_system_info_row_fill (info->graphics_row, _("Graphics"), TRUE); + mate_system_info_row_fill (info->disk_row, _("Disk Capacity"), FALSE); + mate_system_info_row_fill (info->kernel_row, _("Kernel Version"), FALSE); + mate_system_info_row_fill (info->virtualization_row, _("Virtualization"), TRUE); + mate_system_info_row_fill (info->windowing_system_row, _("Windowing System"), TRUE); + mate_system_info_row_fill (info->mate_version_row, _("MATE Version"), TRUE); + mate_system_info_row_fill (info->os_name_row, _("OS Name"), TRUE); + mate_system_info_row_fill (info->os_type_row, _("OS Type"), TRUE); + + gtk_widget_show (info->logo_image); + gtk_widget_show_all (info->hostname_row); + gtk_widget_show_all (info->memory_row); + gtk_widget_show_all (info->processor_row); + gtk_widget_show_all (info->graphics_row); + gtk_widget_show_all (info->disk_row); + gtk_widget_show_all (info->kernel_row); + gtk_widget_show_all (info->windowing_system_row); + gtk_widget_show_all (info->mate_version_row); + gtk_widget_show_all (info->os_type_row); + gtk_widget_show_all (info->os_name_row); +} + +static char * +get_system_hostname (void) +{ + GDBusProxy *hostnamed_proxy; + g_autoptr(GVariant) variant = NULL; + g_autoptr(GError) error = NULL; + + hostnamed_proxy = g_dbus_proxy_new_for_bus_sync (G_BUS_TYPE_SYSTEM, + G_DBUS_PROXY_FLAGS_NONE, + NULL, + "org.freedesktop.hostname1", + "/org/freedesktop/hostname1", + "org.freedesktop.hostname1", + NULL, + &error); + + if (!hostnamed_proxy) + return g_strdup (""); + + variant = g_dbus_proxy_get_cached_property (hostnamed_proxy, "Hostname"); + if (!variant) + { + g_autoptr(GError) error = NULL; + g_autoptr(GVariant) inner = NULL; + + /* Work around systemd-hostname not sending us back + * the property value when changing values */ + variant = g_dbus_proxy_call_sync (hostnamed_proxy, + "org.freedesktop.DBus.Properties.Get", + g_variant_new ("(ss)", "org.freedesktop.hostname1", "Hostname"), + G_DBUS_CALL_FLAGS_NONE, + -1, + NULL, + &error); + if (variant == NULL) + { + g_warning ("Failed to get property '%s': %s", "Hostname", error->message); + g_object_unref (hostnamed_proxy); + return NULL; + } + + g_variant_get (variant, "(v)", &inner); + g_object_unref (hostnamed_proxy); + return g_variant_dup_string (inner, NULL); + } + else + { + g_object_unref (hostnamed_proxy); + return g_variant_dup_string (variant, NULL); + } +} + +static char * +get_hardware_model (void) +{ + g_autoptr(GDBusProxy) hostnamed_proxy = NULL; + g_autoptr(GVariant) vendor_variant = NULL; + g_autoptr(GVariant) model_variant = NULL; + const char *vendor_string, *model_string; + g_autoptr(GError) error = NULL; + + hostnamed_proxy = g_dbus_proxy_new_for_bus_sync (G_BUS_TYPE_SYSTEM, + G_DBUS_PROXY_FLAGS_NONE, + NULL, + "org.freedesktop.hostname1", + "/org/freedesktop/hostname1", + "org.freedesktop.hostname1", + NULL, + &error); + if (hostnamed_proxy == NULL) + { + g_debug ("Couldn't get hostnamed to start, bailing: %s", error->message); + return NULL; + } + + vendor_variant = g_dbus_proxy_get_cached_property (hostnamed_proxy, "HardwareVendor"); + if (!vendor_variant) + { + g_debug ("Unable to retrieve org.freedesktop.hostname1.HardwareVendor property"); + return NULL; + } + + model_variant = g_dbus_proxy_get_cached_property (hostnamed_proxy, "HardwareModel"); + if (!model_variant) + { + g_debug ("Unable to retrieve org.freedesktop.hostname1.HardwareModel property"); + return NULL; + } + + vendor_string = g_variant_get_string (vendor_variant, NULL), + model_string = g_variant_get_string (model_variant, NULL); + + if (vendor_string && g_strcmp0 (vendor_string, "") != 0) + { + gchar *vendor_model = NULL; + + vendor_model = g_strdup_printf ("%s %s", vendor_string, model_string); + return vendor_model; + } + + return NULL; +} + +static char * +get_cpu_info (void) +{ + g_autoptr(GHashTable) counts = NULL; + const glibtop_sysinfo *info; + g_autoptr(GString) cpu = NULL; + GHashTableIter iter; + gpointer key, value; + int i; + int j; + + counts = g_hash_table_new (g_str_hash, g_str_equal); + info = glibtop_get_sysinfo (); + + /* count duplicates */ + for (i = 0; i != info->ncpu; ++i) + { + const char * const keys[] = { "model name", "cpu", "Processor" ,"Model Name"}; + char *model; + int *count; + model = NULL; + + for (j = 0; model == NULL && j != G_N_ELEMENTS (keys); ++j) + { + model = g_hash_table_lookup (info->cpuinfo[i].values, + keys[j]); + } + + if (model == NULL) + continue; + count = g_hash_table_lookup (counts, model); + if (count == NULL) + g_hash_table_insert (counts, model, GINT_TO_POINTER (1)); + else + g_hash_table_replace (counts, model, GINT_TO_POINTER (GPOINTER_TO_INT (count) + 1)); + } + + cpu = g_string_new (NULL); + g_hash_table_iter_init (&iter, counts); + while (g_hash_table_iter_next (&iter, &key, &value)) + { + g_autofree char *cleanedup = NULL; + int count; + + count = GPOINTER_TO_INT (value); + cleanedup = info_cleanup ((const char *) key); + if (count > 1) + g_string_append_printf (cpu, "%s \303\227 %d ", cleanedup, count); + else + g_string_append_printf (cpu, "%s ", cleanedup); + } + + return g_strdup (cpu->str); +} + +static char * +get_renderer_from_session (void) +{ + g_autoptr(GDBusProxy) session_proxy = NULL; + g_autoptr(GVariant) renderer_variant = NULL; + char *renderer; + g_autoptr(GError) error = NULL; + + session_proxy = g_dbus_proxy_new_for_bus_sync (G_BUS_TYPE_SESSION, + G_DBUS_PROXY_FLAGS_NONE, + NULL, + "org.gnome.SessionManager", + "/org/gnome/SessionManager", + "org.gnome.SessionManager", + NULL, &error); + if (error != NULL) + { + g_warning ("Unable to connect to create a proxy for org.gnome.SessionManager: %s", + error->message); + return NULL; + } + + renderer_variant = g_dbus_proxy_get_cached_property (session_proxy, "Renderer"); + + if (!renderer_variant) + { + g_warning ("Unable to retrieve org.gnome.SessionManager.Renderer property"); + return NULL; + } + + renderer = info_cleanup (g_variant_get_string (renderer_variant, NULL)); + + return renderer; +} + +static gchar * +get_graphics_hardware_string (void) +{ + g_autofree char *discrete_renderer = NULL; + g_autofree char *renderer = NULL; + + renderer = get_renderer_from_session (); + if (!renderer) + return g_strdup ("Unknown"); + + return g_strdup (renderer); +} + +static char * +get_primary_disk_size (void) +{ + g_autoptr(UDisksClient) client = NULL; + GDBusObjectManager *manager; + g_autolist(GDBusObject) objects = NULL; + GList *l; + gchar *size; + guint64 total_size; + g_autoptr(GError) error = NULL; + + total_size = 0; + + client = udisks_client_new_sync (NULL, &error); + if (client == NULL) + { + g_warning ("Unable to get UDisks client: %s. Disk information will not be available.", + error->message); + return g_strdup ("Unknown"); + } + + manager = udisks_client_get_object_manager (client); + objects = g_dbus_object_manager_get_objects (manager); + + for (l = objects; l != NULL; l = l->next) + { + UDisksDrive *drive; + drive = udisks_object_peek_drive (UDISKS_OBJECT (l->data)); + + /* Skip removable devices */ + if (drive == NULL || + udisks_drive_get_removable (drive) || + udisks_drive_get_ejectable (drive)) + { + continue; + } + + total_size += udisks_drive_get_size (drive); + } + if (total_size > 0) + { + size = g_format_size (total_size); + } + else + { + size = g_strdup ("Unknown"); + } + + return size; +} + +static char * +get_os_name (void) +{ + g_autofree gchar *name = NULL; + g_autofree gchar *version_id = NULL; + g_autofree gchar *pretty_name = NULL; + g_autofree gchar *name_version = NULL; + gchar *result = NULL; + + name = g_get_os_info (G_OS_INFO_KEY_NAME); + version_id = g_get_os_info (G_OS_INFO_KEY_VERSION_ID); + pretty_name = g_get_os_info (G_OS_INFO_KEY_PRETTY_NAME); + + if (pretty_name) + name_version = g_strdup (pretty_name); + else if (name && version_id) + name_version = g_strdup_printf ("%s %s", name, version_id); + else + name_version = g_strdup (("Unknown")); + + result = g_strdup (name_version); + + return result; +} + +static char * +get_os_type (void) +{ + if (GLIB_SIZEOF_VOID_P == 8) + /* translators: This is the type of architecture for the OS */ + return g_strdup_printf ("64-bit"); + else + /* translators: This is the type of architecture for the OS */ + return g_strdup_printf ("32-bit"); +} + +static char * +get_windowing_system (void) +{ + GdkDisplay *display; + + display = gdk_display_get_default (); + +#if defined(GDK_WINDOWING_X11) + if (GDK_IS_X11_DISPLAY (display)) + return g_strdup ("X11"); +#endif /* GDK_WINDOWING_X11 */ +#if defined(GDK_WINDOWING_WAYLAND) + if (GDK_IS_WAYLAND_DISPLAY (display)) + return g_strdup ("Wayland"); +#endif /* GDK_WINDOWING_WAYLAND */ + return g_strdup (C_("Windowing system (Wayland, X11, or Unknown)", "Unknown")); +} + +static char * +get_kernel_vesrion (void) +{ + struct utsname un; + + if (uname (&un) < 0) + return NULL; + + return g_strdup_printf ("%s %s", un.sysname, un.release); +} + +static struct { + const char *id; + const char *display; +} const virt_tech[] = { + { "kvm", "KVM" }, + { "qemu", "QEmu" }, + { "vmware", "VMware" }, + { "microsoft", "Microsoft" }, + { "oracle", "Oracle" }, + { "xen", "Xen" }, + { "bochs", "Bochs" }, + { "chroot", "chroot" }, + { "openvz", "OpenVZ" }, + { "lxc", "LXC" }, + { "lxc-libvirt", "LXC (libvirt)" }, + { "systemd-nspawn", "systemd (nspawn)" } +}; + +static char * +get_virtualization_label (const char *virt) +{ + const char *display_name; + guint i; + + if (virt == NULL || *virt == '\0') + { + return NULL; + } + + display_name = NULL; + for (i = 0; i < G_N_ELEMENTS (virt_tech); i++) + { + if (g_str_equal (virt_tech[i].id, virt)) + { + display_name = virt_tech[i].display; + break; + } + } + + return display_name ? g_strdup (display_name) : g_strdup (virt); +} + +static char * +get_system_virt (void) +{ + g_autoptr(GError) error = NULL; + g_autoptr(GDBusProxy) systemd_proxy = NULL; + g_autoptr(GVariant) variant = NULL; + GVariant *inner; + + systemd_proxy = g_dbus_proxy_new_for_bus_sync (G_BUS_TYPE_SYSTEM, + G_DBUS_PROXY_FLAGS_NONE, + NULL, + "org.freedesktop.systemd1", + "/org/freedesktop/systemd1", + "org.freedesktop.systemd1", + NULL, + &error); + + if (systemd_proxy == NULL) + { + g_debug ("systemd not available, bailing: %s", error->message); + return NULL; + } + + variant = g_dbus_proxy_call_sync (systemd_proxy, + "org.freedesktop.DBus.Properties.Get", + g_variant_new ("(ss)", "org.freedesktop.systemd1.Manager", "Virtualization"), + G_DBUS_CALL_FLAGS_NONE, + -1, + NULL, + &error); + if (variant == NULL) + { + g_debug ("Failed to get property '%s': %s", "Virtualization", error->message); + return NULL; + } + + g_variant_get (variant, "(v)", &inner); + + return get_virtualization_label (g_variant_get_string (inner, NULL)); +} + +static char * +get_mate_desktop_version () +{ + int status; + int i = 0; + char *version = NULL; + char *argv[] = {"/usr/bin/mate-about", "-v", NULL}; + g_autoptr(GError) error = NULL; + + g_debug ("About to launch '%s'", argv[0]); + + if (!g_spawn_sync (NULL, (char **) argv, NULL, 0, NULL, NULL, &version, NULL, &status, &error)) + { + g_warning ("Failed to get GPU: %s", error->message); + return NULL; + } +#if GLIB_CHECK_VERSION(2, 70, 0) + if (!g_spawn_check_wait_status (status, NULL)) +#else + if (!g_spawn_check_exit_status (status, NULL)) +#endif + return NULL; + + while (version[i] != '\0') + { + if (version[i] == '\n') + { + version[i] = '\0'; + break; + } + i++; + } + + return version; +} + +static char * +get_logo_name (void) +{ + char *logo_name = NULL; + + logo_name = g_get_os_info ("LOGO"); + + if (logo_name == NULL) + logo_name = g_strdup ("mate-desktop"); + + return logo_name; +} + +void +mate_system_info_setup (MateSystemInfo *info) +{ + g_autofree char *logo_name = NULL; + g_autofree char *hostname_text = NULL; + g_autofree char *hw_model_text = NULL; + g_autofree char *memory_text = NULL; + g_autofree char *cpu_text = NULL; + g_autofree char *os_type_text = NULL; + g_autofree char *os_name_text = NULL; + g_autofree char *disk_text = NULL; + g_autofree char *kernel_text = NULL; + g_autofree char *windowing_system_text = NULL; + g_autofree char *virt_text = NULL; + g_autofree char *de_text = NULL; + g_autofree char *graphics_hardware_string = NULL; + + GtkWidget *label; + glibtop_mem mem; + + logo_name = get_logo_name (); + gtk_image_set_from_icon_name (GTK_IMAGE (info->logo_image), logo_name, GTK_ICON_SIZE_INVALID); + gtk_image_set_pixel_size (GTK_IMAGE (info->logo_image), 128); + + hostname_text = get_system_hostname (); + label = g_object_get_data (G_OBJECT (info->hostname_row), "labelvalue"); + set_lable_style (label, "gray", 12, hostname_text, FALSE); + + hw_model_text = get_hardware_model (); + if (hw_model_text != NULL) + { + gtk_widget_show_all (info->hardware_model_row); + label = g_object_get_data (G_OBJECT (info->hardware_model_row), "labelvalue"); + set_lable_style (label, "gray", 12, hw_model_text, FALSE); + } + + glibtop_get_mem (&mem); + memory_text = g_format_size_full (mem.total, G_FORMAT_SIZE_IEC_UNITS); + label = g_object_get_data (G_OBJECT (info->memory_row), "labelvalue"); + set_lable_style (label, "gray", 12, memory_text, FALSE); + + cpu_text = get_cpu_info (); + label = g_object_get_data (G_OBJECT (info->processor_row), "labelvalue"); + set_lable_style (label, "gray", 12, cpu_text, FALSE); + + graphics_hardware_string = get_graphics_hardware_string (); + label = g_object_get_data (G_OBJECT (info->graphics_row), "labelvalue"); + set_lable_style (label, "gray", 12, graphics_hardware_string, FALSE); + + disk_text = get_primary_disk_size (); + label = g_object_get_data (G_OBJECT (info->disk_row), "labelvalue"); + set_lable_style (label, "gray", 12, disk_text, FALSE); + + kernel_text = get_kernel_vesrion (); + label = g_object_get_data (G_OBJECT (info->kernel_row), "labelvalue"); + set_lable_style (label, "gray", 12, kernel_text, FALSE); + + virt_text = get_system_virt (); + if (virt_text != NULL) + { + gtk_widget_show_all (info->virtualization_row); + label = g_object_get_data (G_OBJECT (info->virtualization_row), "labelvalue"); + set_lable_style (label, "gray", 12, virt_text, FALSE); + } + windowing_system_text = get_windowing_system (); + label = g_object_get_data (G_OBJECT (info->windowing_system_row), "labelvalue"); + set_lable_style (label, "gray", 12, windowing_system_text, FALSE); + + de_text = get_mate_desktop_version (); + label = g_object_get_data (G_OBJECT (info->mate_version_row), "labelvalue"); + set_lable_style (label, "gray", 12, de_text, FALSE); + + os_type_text = get_os_type (); + label = g_object_get_data (G_OBJECT (info->os_type_row), "labelvalue"); + set_lable_style (label, "gray", 12, os_type_text, FALSE); + + os_name_text = get_os_name (); + label = g_object_get_data (G_OBJECT (info->os_name_row), "labelvalue"); + set_lable_style (label, "gray", 12, os_name_text, FALSE); +} + +static void +mate_system_info_destroy (GtkWidget *widget) +{ + GTK_WIDGET_CLASS (mate_system_info_parent_class)->destroy (widget); +} + +static void +mate_system_info_class_init (MateSystemInfoClass *klass) +{ + GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass); + + widget_class->destroy = mate_system_info_destroy; + gtk_widget_class_set_template_from_resource (widget_class, "/org/mate/control-center/system-info/mate-system-info.ui"); + + gtk_widget_class_bind_template_child (widget_class, MateSystemInfo, hostname_row); + gtk_widget_class_bind_template_child (widget_class, MateSystemInfo, hardware_box); + gtk_widget_class_bind_template_child (widget_class, MateSystemInfo, disk_row); + gtk_widget_class_bind_template_child (widget_class, MateSystemInfo, mate_version_row); + gtk_widget_class_bind_template_child (widget_class, MateSystemInfo, graphics_row); + gtk_widget_class_bind_template_child (widget_class, MateSystemInfo, hardware_model_row); + gtk_widget_class_bind_template_child (widget_class, MateSystemInfo, memory_row); + gtk_widget_class_bind_template_child (widget_class, MateSystemInfo, os_box); + gtk_widget_class_bind_template_child (widget_class, MateSystemInfo, logo_image); + gtk_widget_class_bind_template_child (widget_class, MateSystemInfo, os_name_row); + gtk_widget_class_bind_template_child (widget_class, MateSystemInfo, os_type_row); + gtk_widget_class_bind_template_child (widget_class, MateSystemInfo, processor_row); + gtk_widget_class_bind_template_child (widget_class, MateSystemInfo, virtualization_row); + gtk_widget_class_bind_template_child (widget_class, MateSystemInfo, kernel_row); + gtk_widget_class_bind_template_child (widget_class, MateSystemInfo, windowing_system_row); +} + +static void +mate_system_info_init (MateSystemInfo *self) +{ + gtk_widget_init_template (GTK_WIDGET (self)); + g_resources_register (mate_system_info_get_resource ()); + mate_system_info_set_row (MATE_SYSTEM_INFO (self)); +} + +GtkWidget * +mate_system_info_new (void) +{ + GtkWidget *dialog; + gboolean use_header; + GdkDisplay *display; + + g_object_get (gtk_settings_get_default (), + "gtk-dialogs-use-header", &use_header, + NULL); + + display = gdk_display_get_default (); +#if defined(GDK_WINDOWING_WAYLAND) + if (GDK_IS_WAYLAND_DISPLAY (display)) + use_header = FALSE; +#endif /* GDK_WINDOWING_WAYLAND */ + + dialog = g_object_new (MATE_TYPE_SYSTEM_INFO, + "use-header-bar", use_header, + NULL); + gtk_window_set_title (GTK_WINDOW (dialog), _("Mate System Info")); + gtk_widget_set_size_request (GTK_WIDGET (dialog), 600, 500); + gtk_window_set_resizable (GTK_WINDOW (dialog), FALSE); + + return dialog; +} diff --git a/capplets/system-info/mate-system-info.desktop.in b/capplets/system-info/mate-system-info.desktop.in new file mode 100644 index 00000000..96408538 --- /dev/null +++ b/capplets/system-info/mate-system-info.desktop.in @@ -0,0 +1,13 @@ +[Desktop Entry] +Name=system-info +Comment=View mate desktop system info +Exec=mate-system-info +# Translators: Do NOT translate or transliterate this text (this is an icon file name)! +Icon=mate-desktop +Terminal=false +Type=Application +StartupNotify=true +Categories=GTK;Settings;HardwareSettings; +# Translators: Search terms to find this application. Do NOT translate or localize the semicolons! The list MUST also end with a semicolon! +Keywords=mate-control-center;MATE;system;preferences; +OnlyShowIn=MATE; diff --git a/capplets/system-info/mate-system-info.h b/capplets/system-info/mate-system-info.h new file mode 100644 index 00000000..72d89ece --- /dev/null +++ b/capplets/system-info/mate-system-info.h @@ -0,0 +1,41 @@ +/* + * Copyright (C) 2023 MATE Developers + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2, or (at your option) + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +#include <gtk/gtk.h> +#include <fcntl.h> +#include <sys/types.h> +#include <libintl.h> +#include <locale.h> +#include <glib/gi18n.h> + +#ifndef __MATE_SYSTEM_INFO_H__ +#define __MATE_SYSTEM_INFO_H__ 1 + +G_BEGIN_DECLS + +#define MATE_TYPE_SYSTEM_INFO (mate_system_info_get_type ()) +G_DECLARE_FINAL_TYPE (MateSystemInfo, mate_system_info, MATE, SYSTEM_INFO, GtkDialog) + +GtkWidget *mate_system_info_new (void); + +void mate_system_info_setup (MateSystemInfo *info); + +G_END_DECLS + +#endif diff --git a/capplets/system-info/mate-system-info.ui b/capplets/system-info/mate-system-info.ui new file mode 100644 index 00000000..d4cafa76 --- /dev/null +++ b/capplets/system-info/mate-system-info.ui @@ -0,0 +1,156 @@ +<?xml version="1.0" encoding="UTF-8"?> +<interface> + <!-- interface-requires gtk+ 3.0 --> + <template class="MateSystemInfo" parent="GtkDialog"> + <property name="can-focus">False</property> + <property name="icon-name">mate-desktop</property> + <property name="type_hint">dialog</property> + + <child internal-child="vbox"> + <object class="GtkBox"> + <property name="visible">True</property> + <property name="can_focus">False</property> + <property name="valign">center</property> + <property name="spacing">30</property> + <property name="orientation">vertical</property> + <child> + <object class="GtkImage" id="logo_image"> + <property name="visible">True</property> + <property name="can_focus">False</property> + </object> + <packing> + <property name="expand">False</property> + <property name="fill">False</property> + </packing> + </child> + + <child> + <object class="GtkListBox"> + <property name="visible">True</property> + <property name="selection-mode">none</property> + <style> + <class name="frame" /> + </style> + + <!-- Device name --> + <child> + <object class="GtkListBoxRow" id="hostname_row"> + <property name="visible">True</property> + </object> + </child> + + </object> + </child> + + <child> + <object class="GtkListBox" id="hardware_box"> + <property name="visible">True</property> + <property name="selection-mode">none</property> + <style> + <class name="frame" /> + </style> + + <!-- Hardware Model --> + <child> + <object class="GtkListBoxRow" id="hardware_model_row"> + <property name="visible">False</property> + <property name="activatable">False</property> + </object> + </child> + + <!-- Memory --> + <child> + <object class="GtkListBoxRow" id="memory_row"> + <property name="visible">True</property> + <property name="activatable">False</property> + </object> + </child> + + <!-- Processor --> + <child> + <object class="GtkListBoxRow" id="processor_row"> + <property name="visible">True</property> + <property name="activatable">False</property> + </object> + </child> + + <!-- Graphics --> + <child> + <object class="GtkListBoxRow" id="graphics_row"> + <property name="visible">True</property> + <property name="activatable">False</property> + </object> + </child> + + <!-- Disk Capacity --> + <child> + <object class="GtkListBoxRow" id="disk_row"> + <property name="visible">True</property> + <property name="activatable">False</property> + </object> + </child> + + </object> + </child> + + <child> + <object class="GtkListBox" id="os_box"> + <property name="visible">True</property> + <property name="selection-mode">none</property> + <style> + <class name="frame" /> + </style> + + <!-- OS Name --> + <child> + <object class="GtkListBoxRow" id="os_name_row"> + <property name="visible">True</property> + <property name="activatable">False</property> + </object> + </child> + + <!-- OS Type --> + <child> + <object class="GtkListBoxRow" id="os_type_row"> + <property name="visible">True</property> + <property name="activatable">False</property> + </object> + </child> + + <!-- MATE Version --> + <child> + <object class="GtkListBoxRow" id="mate_version_row"> + <property name="visible">True</property> + <property name="activatable">False</property> + </object> + </child> + + <!-- Windowing System --> + <child> + <object class="GtkListBoxRow" id="windowing_system_row"> + <property name="visible">True</property> + <property name="activatable">False</property> + </object> + </child> + + <!-- Virtualization --> + <child> + <object class="GtkListBoxRow" id="virtualization_row"> + <property name="visible">False</property> + <property name="activatable">False</property> + </object> + </child> + + <!-- Kernel --> + <child> + <object class="GtkListBoxRow" id="kernel_row"> + <property name="visible">True</property> + <property name="activatable">False</property> + </object> + </child> + </object> + </child> + </object> + </child> + </template> +</interface> diff --git a/capplets/system-info/system-info.gresource.xml b/capplets/system-info/system-info.gresource.xml new file mode 100644 index 00000000..f234e079 --- /dev/null +++ b/capplets/system-info/system-info.gresource.xml @@ -0,0 +1,6 @@ +<?xml version="1.0" encoding="UTF-8"?> +<gresources> + <gresource prefix="/org/mate/control-center/system-info"> + <file preprocess="xml-stripblanks">mate-system-info.ui</file> + </gresource> +</gresources> |