diff options
Diffstat (limited to 'modules/computer.c')
| -rw-r--r-- | modules/computer.c | 1124 | 
1 files changed, 1124 insertions, 0 deletions
| diff --git a/modules/computer.c b/modules/computer.c new file mode 100644 index 00000000..fff8ba36 --- /dev/null +++ b/modules/computer.c @@ -0,0 +1,1124 @@ +/* + *    HardInfo - Displays System Information + *    Copyright (C) 2003-2008 L. A. F. Pereira <l@tia.mat.br> + * + *    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, version 2 or later. + * + *    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 <config.h> +#include <gtk/gtk.h> +#include <stdarg.h> +#include <stdlib.h> +#include <string.h> +#include <string.h> +#include <sys/stat.h> +#include <time.h> + +#include <hardinfo.h> +#include <iconcache.h> +#include <shell.h> + +#include <vendor.h> + +#include "computer.h" + +#include "dmi_util.h" /* for dmi_get_str() */ +#include "dt_util.h" /* for dtr_get_string() */ + +#include "info.h" + +#define THISORUNK(t) ( (t) ? t : _("(Unknown)") ) + +/* Callbacks */ +gchar *callback_summary(void); +gchar *callback_os(void); +gchar *callback_security(void); +gchar *callback_modules(void); +gchar *callback_boots(void); +gchar *callback_locales(void); +gchar *callback_memory_usage(); +gchar *callback_fs(void); +gchar *callback_display(void); +gchar *callback_network(void); +gchar *callback_users(void); +gchar *callback_groups(void); +gchar *callback_env_var(void); +#if GLIB_CHECK_VERSION(2,14,0) +gchar *callback_dev(void); +#endif /* GLIB_CHECK_VERSION(2,14,0) */ + +/* Scan callbacks */ +void scan_summary(gboolean reload); +void scan_os(gboolean reload); +void scan_security(gboolean reload); +void scan_modules(gboolean reload); +void scan_boots(gboolean reload); +void scan_locales(gboolean reload); +void scan_fs(gboolean reload); +void scan_memory_usage(gboolean reload); +void scan_display(gboolean reload); +void scan_network(gboolean reload); +void scan_users(gboolean reload); +void scan_groups(gboolean reload); +void scan_env_var(gboolean reload); +#if GLIB_CHECK_VERSION(2,14,0) +void scan_dev(gboolean reload); +#endif /* GLIB_CHECK_VERSION(2,14,0) */ + +enum { +    ENTRY_SUMMARY, +    ENTRY_OS, +    ENTRY_SECURITY, +    ENTRY_KMOD, +    ENTRY_BOOTS, +    ENTRY_LANGUAGES, +    ENTRY_MEMORY_USAGE, +    ENTRY_FS, +    ENTRY_DISPLAY, +    ENTRY_ENV, +    ENTRY_DEVEL, +    ENTRY_USERS, +    ENTRY_GROUPS +}; + +static ModuleEntry entries[] = { +    [ENTRY_SUMMARY] = {N_("Summary"), "summary.png", callback_summary, scan_summary, MODULE_FLAG_NONE}, +    [ENTRY_OS] = {N_("Operating System"), "os.png", callback_os, scan_os, MODULE_FLAG_NONE}, +    [ENTRY_SECURITY] = {N_("Security"), "security.png", callback_security, scan_security, MODULE_FLAG_NONE}, +    [ENTRY_KMOD] = {N_("Kernel Modules"), "module.png", callback_modules, scan_modules, MODULE_FLAG_NONE}, +    [ENTRY_BOOTS] = {N_("Boots"), "boot.png", callback_boots, scan_boots, MODULE_FLAG_NONE}, +    [ENTRY_LANGUAGES] = {N_("Languages"), "language.png", callback_locales, scan_locales, MODULE_FLAG_NONE}, +    [ENTRY_MEMORY_USAGE] = {N_("Memory Usage"), "memory.png", callback_memory_usage, scan_memory_usage, MODULE_FLAG_NONE}, +    [ENTRY_FS] = {N_("Filesystems"), "dev_removable.png", callback_fs, scan_fs, MODULE_FLAG_NONE}, +    [ENTRY_DISPLAY] = {N_("Display"), "monitor.png", callback_display, scan_display, MODULE_FLAG_NONE}, +    [ENTRY_ENV] = {N_("Environment Variables"), "environment.png", callback_env_var, scan_env_var, MODULE_FLAG_NONE}, +#if GLIB_CHECK_VERSION(2,14,0) +    [ENTRY_DEVEL] = {N_("Development"), "devel.png", callback_dev, scan_dev, MODULE_FLAG_NONE}, +#else +    [ENTRY_DEVEL] = {N_("Development"), "devel.png", callback_dev, scan_dev, MODULE_FLAG_HIDE}, +#endif /* GLIB_CHECK_VERSION(2,14,0) */ +    [ENTRY_USERS] = {N_("Users"), "users.png", callback_users, scan_users, MODULE_FLAG_NONE}, +    [ENTRY_GROUPS] = {N_("Groups"), "users.png", callback_groups, scan_groups, MODULE_FLAG_NONE}, +    {NULL}, +}; + +gchar *module_list = NULL; +Computer *computer = NULL; +gchar *meminfo = NULL; + +gchar *hi_more_info(gchar * entry) +{ +    gchar *info = moreinfo_lookup_with_prefix("COMP", entry); + +    if (info) +        return g_strdup(info); + +    return g_strdup_printf("[%s]", entry); +} + +/* a g_str_equal() where either may be null */ +#define g_str_equal0(a,b) (g_strcmp0(a,b) == 0) + +gchar *hi_get_field(gchar * field) +{ +    gchar *tag, *label; +    key_get_components(field, NULL, &tag, NULL, &label, NULL, TRUE); + +    gchar *tmp; + +    if (g_str_equal0(label, _("Memory"))) { +        MemoryInfo *mi = computer_get_memory(); +        tmp = g_strdup_printf(_("%dMB (%dMB used)"), mi->total, mi->used); +        g_free(mi); +    } else if (g_str_equal0(label, _("Uptime"))) { +        tmp = computer_get_formatted_uptime(); +    } else if (g_str_equal0(label, _("Date/Time"))) { +        time_t t = time(NULL); + +        tmp = g_new0(gchar, 64); +        strftime(tmp, 64, "%c", localtime(&t)); +    } else if (g_str_equal0(label, _("Load Average"))) { +        tmp = computer_get_formatted_loadavg(); +    } else if (g_str_equal0(tag, "entropy")) { +        tmp = computer_get_entropy_avail(); +    } else { +        gchar *info = NULL; +        if (tag) +            info = moreinfo_lookup_with_prefix("DEV", tag); +        else if (label) +            info = moreinfo_lookup_with_prefix("DEV", label); + +        if (info) +            tmp = g_strdup(info); +        else +            tmp = g_strdup_printf("Unknown field: [tag: %s] label: %s", tag ? tag : "(none)", label ? label : "(empty)"); +    } +    return tmp; +} + +void scan_summary(gboolean reload) +{ +    SCAN_START(); +    module_entry_scan_all_except(entries, 0); +    computer->alsa = computer_get_alsainfo(); +    SCAN_END(); +} + +void scan_os(gboolean reload) +{ +    SCAN_START(); +    computer->os = computer_get_os(); +    SCAN_END(); +} + +void scan_security(gboolean reload) +{ +    SCAN_START(); +    //nothing to do here yet +    SCAN_END(); +} + +void scan_modules(gboolean reload) +{ +    SCAN_START(); +    scan_modules_do(); +    SCAN_END(); +} + +void scan_boots(gboolean reload) +{ +    SCAN_START(); +    scan_boots_real(); +    SCAN_END(); +} + +void scan_locales(gboolean reload) +{ +    SCAN_START(); +    scan_os(FALSE); +    scan_languages(computer->os); +    SCAN_END(); +} + +void scan_fs(gboolean reload) +{ +    SCAN_START(); +    scan_filesystems(); +    SCAN_END(); +} + +void scan_memory_usage(gboolean reload) +{ +    SCAN_START(); +    scan_memory_do(); +    SCAN_END(); +} + +void scan_display(gboolean reload) +{ +    SCAN_START(); +    if (computer->display) +        computer_free_display(computer->display); +    computer->display = computer_get_display(); +    SCAN_END(); +} + +void scan_users(gboolean reload) +{ +    SCAN_START(); +    scan_users_do(); +    SCAN_END(); +} + +void scan_groups(gboolean reload) +{ +    SCAN_START(); +    scan_groups_do(); +    SCAN_END(); +} + +#if GLIB_CHECK_VERSION(2,14,0) +static gchar *dev_list = NULL; +void scan_dev(gboolean reload) +{ +    SCAN_START(); + +    guint i; +    struct { +       gchar *compiler_name; +       gchar *version_command; +       gchar *regex; +       gboolean read_stdout; +    } detect_lang[] = { +       { N_("Scripting Languages"), NULL, FALSE }, +       { N_("Gambas3 (gbr3)"), "gbr3 --version", "\\d+\\.\\d+\\.\\d+", TRUE }, +       { N_("Python (default)"), "python -V", "\\d+\\.\\d+\\.\\d+", FALSE }, +       { N_("Python2"), "python2 -V", "\\d+\\.\\d+\\.\\d+", FALSE }, +       { N_("Python3"), "python3 -V", "\\d+\\.\\d+\\.\\d+", TRUE }, +       { N_("Perl"), "perl -v", "\\d+\\.\\d+\\.\\d+", TRUE }, +       { N_("Perl6 (VM)"), "perl6 -v", "(?<=This is ).*", TRUE }, +       { N_("Perl6"), "perl6 -v", "(?<=implementing Perl )\\w*\\.\\w*", TRUE }, +       { N_("PHP"), "php --version", "\\d+\\.\\d+\\.\\S+", TRUE}, +       { N_("Ruby"), "ruby --version", "\\d+\\.\\d+\\.\\d+", TRUE }, +       { N_("Bash"), "bash --version", "\\d+\\.\\d+\\.\\S+", TRUE}, +       { N_("JavaScript (Node.js)"), "node --version", "(?<=v)(\\d\\.?)+", TRUE }, +       { N_("awk"), "awk --version", "GNU Awk \\d+\\.\\d+\\.\\d+", TRUE }, +       { N_("Compilers"), NULL, FALSE }, +       { N_("C (GCC)"), "gcc -v", "\\d+\\.\\d+\\.\\d+", FALSE }, +       { N_("C (Clang)"), "clang -v", "\\d+\\.\\d+", FALSE }, +       { N_("D (dmd)"), "dmd --help", "\\d+\\.\\d+", TRUE }, +       { N_("Gambas3 (gbc3)"), "gbc3 --version", "\\d+\\.\\d+\\.\\d+", TRUE }, +       { N_("Java"), "javac -version", "\\d+\\.\\d+\\.\\d+", FALSE }, +       { N_("C♯ (mcs)"), "mcs --version", "\\d+\\.\\d+\\.\\d+\\.\\d+", TRUE }, +       { N_("Vala"), "valac --version", "\\d+\\.\\d+\\.\\d+", TRUE }, +       { N_("Haskell (GHC)"), "ghc -v", "\\d+\\.\\d+\\.\\d+", FALSE }, +       { N_("FreePascal"), "fpc -iV", "\\d+\\.\\d+\\.?\\d*", TRUE }, +       { N_("Go"), "go version", "\\d+\\.\\d+\\.?\\d* ", TRUE }, +       { N_("Rust"), "rustc --version", "(?<=rustc )(\\d\\.?)+", TRUE }, +       { N_("Tools"), NULL, FALSE }, +       { N_("make"), "make --version", "\\d+\\.\\d+", TRUE }, +       { N_("ninja"), "ninja --version", "\\d+\\.\\d+\\.\\d+", TRUE }, +       { N_("GDB"), "gdb --version", "(?<=^GNU gdb ).*", TRUE }, +       { N_("LLDB"), "lldb --version", "(?<=lldb version )(\\d\\.?)+", TRUE }, +       { N_("strace"), "strace -V", "\\d+\\.\\d+\\.?\\d*", TRUE }, +       { N_("valgrind"), "valgrind --version", "\\d+\\.\\d+\\.\\S+", TRUE }, +       { N_("QMake"), "qmake --version", "\\d+\\.\\S+", TRUE}, +       { N_("CMake"), "cmake --version", "\\d+\\.\\d+\\.?\\d*", TRUE}, +       { N_("Gambas3 IDE"), "gambas3 --version", "\\d+\\.\\d+\\.\\d+", TRUE }, +       { N_("Radare2"), "radare2 -v", "(?<=radare2 )(\\d+\\.?)+(-git)?", TRUE }, +       { N_("ltrace"), "ltrace --version", "(?<=ltrace version )\\d+\\.\\d+\\.\\d+", TRUE }, +       { N_("Powershell"), "pwsh --version", "\\d+\\.\\d+\\.\\d+", TRUE }, +    }; + +    g_free(dev_list); + +    dev_list = g_strdup(""); + +    for (i = 0; i < G_N_ELEMENTS(detect_lang); i++) { +       gchar *version = NULL; +       gchar *output, *ignored; +       gchar *temp; +       GRegex *regex; +       GMatchInfo *match_info; +       gboolean found; + +       if (!detect_lang[i].regex) { +            dev_list = h_strdup_cprintf("[%s]\n", dev_list, _(detect_lang[i].compiler_name)); +            continue; +       } + +       if (detect_lang[i].read_stdout) { +            found = hardinfo_spawn_command_line_sync(detect_lang[i].version_command, &output, &ignored, NULL, NULL); +       } else { +            found = hardinfo_spawn_command_line_sync(detect_lang[i].version_command, &ignored, &output, NULL, NULL); +       } +       g_free(ignored); + +       if (found) { +           regex = g_regex_new(detect_lang[i].regex, 0, 0, NULL); + +           g_regex_match(regex, output, 0, &match_info); +           if (g_match_info_matches(match_info)) { +               version = g_match_info_fetch(match_info, 0); +           } + +           g_match_info_free(match_info); +           g_regex_unref(regex); +           g_free(output); +       } + +       if (version == NULL) +           version = strdup(_("Not found")); + +       dev_list = h_strdup_cprintf("%s=%s\n", dev_list, detect_lang[i].compiler_name, version); +       g_free(version); + +       temp = g_strdup_printf(_("Detecting version: %s"), +                              detect_lang[i].compiler_name); +       shell_status_update(temp); +       g_free(temp); +    } + +    SCAN_END(); +} + +gchar *callback_dev(void) +{ +    return g_strdup_printf( +                "[$ShellParam$]\n" +                "ViewType=5\n" +                "ColumnTitle$TextValue=%s\n" /* Program */ +                "ColumnTitle$Value=%s\n" /* Version */ +                "ShowColumnHeaders=true\n" +                "%s", +                _("Program"), _("Version"), +                dev_list); +} +#endif /* GLIB_CHECK_VERSION(2,14,0) */ + +gchar *callback_memory_usage() +{ +    extern gchar *lginterval; +    return g_strdup_printf("[Memory]\n" +               "%s\n" +               "[$ShellParam$]\n" +               "ViewType=2\n" +               "LoadGraphSuffix= kB\n" +               "RescanInterval=2000\n" +               "ColumnTitle$TextValue=%s\n" +               "ColumnTitle$Extra1=%s\n" +               "ColumnTitle$Value=%s\n" +               "ShowColumnHeaders=true\n" +               "%s\n", meminfo, +               _("Field"), _("Description"), _("Value"), /* column labels */ +               lginterval); +} + +static gchar *detect_machine_type(void) +{ +    GDir *dir; +    gchar *chassis; + +    chassis = dmi_chassis_type_str(-1, 0); +    if (chassis) +        return chassis; + +    chassis = dtr_get_string("/model", 0); +    if (chassis) { +        if (strstr(chassis, "Raspberry Pi") != NULL +            || strstr(chassis, "ODROID") != NULL +            || strstr(chassis, "Firefly ROC") != NULL +            /* FIXME: consider making a table when adding more models */ ) { +                g_free(chassis); +                return g_strdup(_("Single-board computer")); +        } +        g_free(chassis); +    } + +    if (g_file_test("/proc/pmu/info", G_FILE_TEST_EXISTS)) +        return g_strdup(_("Laptop")); + +    dir = g_dir_open("/proc/acpi/battery", 0, NULL); +    if (dir) { +        const gchar *name = g_dir_read_name(dir); + +        g_dir_close(dir); + +        if (name) +            return g_strdup(_("Laptop")); +    } + +    dir = g_dir_open("/sys/class/power_supply", 0, NULL); +    if (dir) { +        const gchar *name; + +        while ((name = g_dir_read_name(dir))) { +            gchar *contents; +            gchar type[PATH_MAX]; +            int r; + +            r = snprintf(type, sizeof(type), "%s/%s/type", +                         "/sys/class/power_supply", name); +            if (r < 0 || r > PATH_MAX) +                continue; + +            if (g_file_get_contents(type, &contents, NULL, NULL)) { +                if (g_str_has_prefix(contents, "Battery")) { +                    g_free(contents); +                    g_dir_close(dir); + +                    return g_strdup(_("Laptop")); +                } + +                g_free(contents); +            } +        } + +        g_dir_close(dir); +    } + +    /* FIXME: check if batteries are found using /proc/apm */ + +    return g_strdup(_("Unknown physical machine type")); +} + +/* Table based off imvirt by Thomas Liske <liske@ibh.de> +   Copyright (c) 2008 IBH IT-Service GmbH under GPLv2. */ +gchar *computer_get_virtualization(void) +{ +    gboolean found = FALSE; +    gint i, j; +    gchar *files[] = { +        "/proc/scsi/scsi", +        "/proc/cpuinfo", +        "/var/log/dmesg", +        NULL +    }; +    static const struct { +        gchar *str; +        gchar *vmtype; +    } vm_types[] = { +        /* VMware */ +        { "VMware", N_("Virtual (VMware)") }, +        { ": VMware Virtual IDE CDROM Drive", N_("Virtual (VMware)") }, +        /* QEMU */ +        { "QEMU", N_("Virtual (QEMU)") }, +        { "QEMU Virtual CPU", N_("Virtual (QEMU)") }, +        { ": QEMU HARDDISK", N_("Virtual (QEMU)") }, +        { ": QEMU CD-ROM", N_("Virtual (QEMU)") }, +        /* Generic Virtual Machine */ +        { ": Virtual HD,", N_("Virtual (Unknown)") }, +        { ": Virtual CD,", N_("Virtual (Unknown)") }, +        /* Virtual Box */ +        { "VBOX", N_("Virtual (VirtualBox)") }, +        { ": VBOX HARDDISK", N_("Virtual (VirtualBox)") }, +        { ": VBOX CD-ROM", N_("Virtual (VirtualBox)") }, +        /* Xen */ +        { "Xen virtual console", N_("Virtual (Xen)") }, +        { "Xen reported: ", N_("Virtual (Xen)") }, +        { "xen-vbd: registered block device", N_("Virtual (Xen)") }, +        /* Generic */ +        { " hypervisor", N_("Virtual (hypervisor present)") } , +        { NULL } +    }; +    gchar *tmp; + +    DEBUG("Detecting virtual machine"); + +    if (g_file_test("/proc/xen", G_FILE_TEST_EXISTS)) { +         DEBUG("/proc/xen found; assuming Xen"); +         return g_strdup(_("Virtual (Xen)")); +    } + +    tmp = module_call_method("devices::getMotherboard"); +    if (strstr(tmp, "VirtualBox") != NULL) { +        g_free(tmp); +        return g_strdup(_("Virtual (VirtualBox)")); +    } +    g_free(tmp); + +    for (i = 0; files[i+1]; i++) { +         gchar buffer[512]; +         FILE *file; + +         if ((file = fopen(files[i], "r"))) { +              while (!found && fgets(buffer, 512, file)) { +                  for (j = 0; vm_types[j+1].str; j++) { +                      if (strstr(buffer, vm_types[j].str)) { +                         found = TRUE; +                         break; +                      } +                  } +              } + +              fclose(file); + +              if (found) { +                  DEBUG("%s found (by reading file %s)", +                        vm_types[j].vmtype, files[i]); +                  return g_strdup(_(vm_types[j].vmtype)); +              } +         } + +    } + +    DEBUG("no virtual machine detected; assuming physical machine"); + +    return detect_machine_type(); +} + +gchar *callback_summary(void) +{ +    struct Info *info = info_new(); + +    info_set_view_type(info, SHELL_VIEW_DETAIL); + +    info_add_group(info, _("Computer"), +        info_field(_("Processor"), +            idle_free(module_call_method("devices::getProcessorNameAndDesc"))), +        info_field_update(_("Memory"), 1000), +        info_field_printf(_("Machine Type"), "%s", +            computer_get_virtualization()), +        info_field(_("Operating System"), computer->os->distro), +        info_field(_("User Name"), computer->os->username), +        info_field_update(_("Date/Time"), 1000), +        info_field_last()); + +    info_add_group(info, _("Display"), +        info_field_printf(_("Resolution"), _(/* label for resolution */ "%dx%d pixels"), +            computer->display->width, computer->display->height), +        info_field(_("Display Adapter"), +            idle_free(module_call_method("devices::getGPUList"))), +        info_field(_("OpenGL Renderer"), THISORUNK(computer->display->xi->glx->ogl_renderer)), +        info_field(_("Session Display Server"), THISORUNK(computer->display->display_server)), +        info_field_last()); + +    info_add_computed_group(info, _("Audio Devices"), +        idle_free(computer_get_alsacards(computer))); +    info_add_computed_group_wo_extra(info, _("Input Devices"), +        idle_free(module_call_method("devices::getInputDevices"))); +    info_add_computed_group(info, NULL, /* getPrinters provides group headers */ +        idle_free(module_call_method("devices::getPrinters"))); +    info_add_computed_group_wo_extra(info, NULL,  /* getStorageDevices provides group headers */ +        idle_free(module_call_method("devices::getStorageDevices"))); + +    return info_flatten(info); +} + +gchar *callback_os(void) +{ +    struct Info *info = info_new(); +    gchar *distro_icon; +    gchar *distro; + +    info_set_view_type(info, SHELL_VIEW_DETAIL); + +    distro_icon = computer->os->distroid +                      ? idle_free(g_strdup_printf("distros/%s.svg", +                                                  computer->os->distroid)) +                      : NULL; +    distro = computer->os->distrocode +                      ? idle_free(g_strdup_printf("%s (%s)", +                                  computer->os->distro, computer->os->distrocode)) +                      : computer->os->distro; + +    struct InfoGroup *version_group = +    info_add_group( +        info, _("Version"), info_field(_("Kernel"), computer->os->kernel), +        info_field(_("Command Line"), computer->os->kcmdline ?: _("Unknown")), +        info_field(_("Version"), computer->os->kernel_version), +        info_field(_("C Library"), computer->os->libc), +        info_field(_("Distribution"), distro, +                   .value_has_vendor = TRUE, +                   .icon = distro_icon), +        info_field_last()); + +    if (computer->os->distro_flavor) { +        info_group_add_field(version_group, +            info_field(_("Spin/Flavor"), computer->os->distro_flavor->name, +                .value_has_vendor = TRUE, +                .icon = computer->os->distro_flavor->icon) ); +    } + +    info_add_group(info, _("Current Session"), +        info_field(_("Computer Name"), computer->os->hostname), +        info_field(_("User Name"), computer->os->username), +        info_field(_("Language"), computer->os->language), +        info_field(_("Home Directory"), computer->os->homedir), +        info_field(_("Desktop Environment"), computer->os->desktop), +        info_field_last()); + +    info_add_group(info, _("Misc"), info_field_update(_("Uptime"), 1000), +                   info_field_update(_("Load Average"), 10000), +                   info_field_last()); + +    return info_flatten(info); +} + +gchar *callback_security(void) +{ +    struct Info *info = info_new(); + +    info_set_view_type(info, SHELL_VIEW_DETAIL); + +    info_add_group(info, _("HardInfo"), +                   info_field(_("HardInfo running as"), +                              (getuid() == 0) ? _("Superuser") : _("User")), +                   info_field_last()); + +    info_add_group( +        info, _("Health"), +        info_field_update(_("Available entropy in /dev/random"), 1000, .tag = g_strdup("entropy") ), +        info_field_last()); + +    info_add_group( +        info, _("Hardening Features"), +        info_field(_("ASLR"), idle_free(computer_get_aslr())), +        info_field(_("dmesg"), idle_free(computer_get_dmesg_status())), +        info_field_last()); + +    info_add_group( +        info, _("Linux Security Modules"), +        info_field(_("Modules available"), idle_free(computer_get_lsm())), +        info_field(_("SELinux status"), computer_get_selinux()), +        info_field_last()); + +    GDir *dir = g_dir_open("/sys/devices/system/cpu/vulnerabilities", 0, NULL); +    if (dir) { +        struct InfoGroup *vulns = +            info_add_group(info, _("CPU Vulnerabilities"), info_field_last()); +        vulns->sort = INFO_GROUP_SORT_NAME_ASCENDING; +        const gchar *vuln; + +        while ((vuln = g_dir_read_name(dir))) { +            gchar *contents = h_sysfs_read_string( +                "/sys/devices/system/cpu/vulnerabilities", vuln); +            if (!contents) +                continue; + +            const gchar *icon = NULL; +	    if (g_strstr_len(contents, -1, "Not affected") ) +	        icon = "circle_green_check.svg"; + +            if (g_str_has_prefix(contents, "Mitigation:") || +                g_str_has_prefix(contents, "mitigation:")) +                icon = "circle_yellow_exclaim.svg"; + +            if (g_strstr_len(contents, -1, "Vulnerable") || +                g_strstr_len(contents, -1, "vulnerable")) +                icon = "circle_red_x.svg"; + +            info_group_add_fields(vulns, +                                  info_field(g_strdup(vuln), +                                             idle_free(contents), .icon = icon, +                                             .free_name_on_flatten = TRUE), +                                  info_field_last()); +        } + +        g_dir_close(dir); +    } + +    return info_flatten(info); +} + +gchar *callback_modules(void) +{ +    struct Info *info = info_new(); + +    info_add_computed_group(info, _("Loaded Modules"), module_list); + +    info_set_column_title(info, "TextValue", _("Name")); +    info_set_column_title(info, "Value", _("Description")); +    info_set_column_headers_visible(info, TRUE); +    info_set_view_type(info, SHELL_VIEW_DUAL); + +    return info_flatten(info); +} + +gchar *callback_boots(void) +{ +    struct Info *info = info_new(); + +    info_add_computed_group(info, _("Boots"), computer->os->boots); + +    info_set_column_title(info, "TextValue", _("Date & Time")); +    info_set_column_title(info, "Value", _("Kernel Version")); +    info_set_column_headers_visible(info, TRUE); + +    return info_flatten(info); +} + +gchar *callback_locales(void) +{ +    struct Info *info = info_new(); + +    info_add_computed_group(info, _("Available Languages"), computer->os->languages); + +    info_set_column_title(info, "TextValue", _("Language Code")); +    info_set_column_title(info, "Value", _("Name")); +    info_set_view_type(info, SHELL_VIEW_DUAL); +    info_set_column_headers_visible(info, TRUE); + +    return info_flatten(info); +} + +gchar *callback_fs(void) +{ +    struct Info *info = info_new(); + +    info_add_computed_group(info, _("Mounted File Systems"), fs_list); + +    info_set_column_title(info, "Extra1", _("Mount Point")); +    info_set_column_title(info, "Progress", _("Usage")); +    info_set_column_title(info, "TextValue", _("Device")); +    info_set_column_headers_visible(info, TRUE); +    info_set_view_type(info, SHELL_VIEW_PROGRESS_DUAL); +    info_set_zebra_visible(info, TRUE); +    info_set_normalize_percentage(info, FALSE); + +    return info_flatten(info); +} + +gchar *callback_display(void) +{ +    int n = 0; +    gchar *screens_str = strdup(""), *outputs_str = strdup(""); +    xinfo *xi = computer->display->xi; +    xrr_info *xrr = xi->xrr; +    glx_info *glx = xi->glx; +    wl_info *wl = computer->display->wl; + +    struct Info *info = info_new(); + +    info_set_view_type(info, SHELL_VIEW_DETAIL); + +    info_add_group(info, _("Session"), +        info_field(_("Type"), THISORUNK(computer->display->session_type)), +        info_field_last()); + +    info_add_group(info, _("Wayland"), +        info_field(_("Current Display Name"), +                    (wl->display_name) ? (wl->display_name) : _("(Not Available)")), +        info_field_last()); + +    info_add_group(info, _("X Server"), +        info_field(_("Current Display Name"), THISORUNK(xi->display_name) ), +        info_field(_("Vendor"), THISORUNK(xi->vendor), .value_has_vendor = TRUE ), +        info_field(_("Version"), THISORUNK(xi->version) ), +        info_field(_("Release Number"), THISORUNK(xi->release_number) ), +        info_field_last()); + +    for (n = 0; n < xrr->screen_count; n++) { +        gchar *dims = g_strdup_printf(_(/* resolution WxH unit */ "%dx%d pixels"), xrr->screens[n].px_width, xrr->screens[n].px_height); +        screens_str = h_strdup_cprintf("Screen %d=%s\n", screens_str, xrr->screens[n].number, dims); +        g_free(dims); +    } +    info_add_computed_group(info, _("Screens"), screens_str); + +    for (n = 0; n < xrr->output_count; n++) { +        gchar *connection = NULL; +        switch (xrr->outputs[n].connected) { +            case 0: +                connection = _("Disconnected"); +                break; +            case 1: +                connection = _("Connected"); +                break; +            case -1: +            default: +                connection = _("Unknown"); +                break; +        } +        gchar *dims = (xrr->outputs[n].screen == -1) +            ? g_strdup(_("Unused")) +            : g_strdup_printf(_("%dx%d pixels, offset (%d, %d)"), +                    xrr->outputs[n].px_width, xrr->outputs[n].px_height, +                    xrr->outputs[n].px_offset_x, xrr->outputs[n].px_offset_y); + +        outputs_str = h_strdup_cprintf("%s=%s; %s\n", outputs_str, +            xrr->outputs[n].name, connection, dims); + +        g_free(dims); +    } +    info_add_computed_group(info, _("Outputs (XRandR)"), outputs_str); + +    info_add_group(info, _("OpenGL (GLX)"), +        info_field(_("Vendor"), THISORUNK(glx->ogl_vendor), .value_has_vendor = TRUE ), +        info_field(_("Renderer"), THISORUNK(glx->ogl_renderer) ), +        info_field(_("Direct Rendering"), +            glx->direct_rendering ? _("Yes") : _("No")), +        info_field(_("Version (Compatibility)"), THISORUNK(glx->ogl_version) ), +        info_field(_("Shading Language Version (Compatibility)"), THISORUNK(glx->ogl_sl_version) ), +        info_field(_("Version (Core)"), THISORUNK(glx->ogl_core_version) ), +        info_field(_("Shading Language Version (Core)"), THISORUNK(glx->ogl_core_sl_version) ), +        info_field(_("Version (ES)"), THISORUNK(glx->ogles_version) ), +        info_field(_("Shading Language Version (ES)"), THISORUNK(glx->ogles_sl_version) ), +        info_field(_("GLX Version"), THISORUNK(glx->glx_version) ), +        info_field_last()); + +    return info_flatten(info); +} + +gchar *callback_users(void) +{ +    struct Info *info = info_new(); + +    info_add_computed_group(info, _("Users"), users); +    info_set_view_type(info, SHELL_VIEW_DUAL); +    info_set_reload_interval(info, 10000); + +    return info_flatten(info); +} + +gchar *callback_groups(void) +{ +    struct Info *info = info_new(); + +    info_add_computed_group(info, _("Group"), groups); + +    info_set_column_title(info, "TextValue", _("Name")); +    info_set_column_title(info, "Value", _("Group ID")); +    info_set_column_headers_visible(info, TRUE); +    info_set_reload_interval(info, 10000); + +    return info_flatten(info); +} + +gchar *get_os_kernel(void) +{ +    scan_os(FALSE); +    return g_strdup(computer->os->kernel); +} + +gchar *get_os(void) +{ +    scan_os(FALSE); +    return g_strdup(computer->os->distro); +} + +gchar *get_ogl_renderer(void) +{ +    scan_display(FALSE); + +    return g_strdup(computer->display->xi->glx->ogl_renderer); +} + +gchar *get_display_summary(void) +{ +    scan_display(FALSE); + +    gchar *gpu_list = module_call_method("devices::getGPUList"); + +    gchar *ret = g_strdup_printf( +                           "%s\n" +                           "%dx%d\n" +                           "%s\n" +                           "%s", +                           gpu_list, +                           computer->display->width, computer->display->height, +                           computer->display->display_server, +                           (computer->display->xi->glx->ogl_renderer) +                              ? computer->display->xi->glx->ogl_renderer +                              : "" ); +    g_free(gpu_list); +    return ret; +} + +gchar *get_kernel_module_description(gchar *module) +{ +    gchar *description; + +    if (!_module_hash_table) { +        scan_modules(FALSE); +    } + +    description = g_hash_table_lookup(_module_hash_table, module); +    if (!description) { +        return NULL; +    } + +    return g_strdup(description); +} + +gchar *get_audio_cards(void) +{ +    if (!computer->alsa) { +      computer->alsa = computer_get_alsainfo(); +    } + +    return computer_get_alsacards(computer); +} + +/* the returned string must stay in kB as it is used + * elsewhere with that expectation */ +gchar *get_memory_total(void) +{ +    scan_memory_usage(FALSE); +    return moreinfo_lookup ("DEV:MemTotal"); +} + +gchar *memory_devices_get_system_memory_str(); /* in dmi_memory.c */ +gchar *memory_devices_get_system_memory_types_str(); +/* Note 1: moreinfo_lookup() results should not be freed because + *         they are pointers into a GHash. + *         module_call_method() g_strdup()s it's return value. */ +const gchar *get_memory_desc(void) // [1] const (as to say "don't free") +{ +    scan_memory_usage(FALSE); +    gchar *avail = g_strdup(moreinfo_lookup("DEV:MemTotal")); // [1] g_strdup() +    double k = avail ? (double)strtol(avail, NULL, 10) : 0; +    if (k) { +        g_free(avail); +        avail = NULL; +        const char *fmt = _(/*/ <value> <unit> "usable memory" */ "%0.1f %s available to Linux"); +        if (k > (2048 * 1024)) +            avail = g_strdup_printf(fmt, k / (1024*1024), _("GiB") ); +        else if (k > 2048) +            avail = g_strdup_printf(fmt, k / 1024, _("MiB") ); +        else +            avail = g_strdup_printf(fmt, k, _("KiB") ); +    } +    gchar *mem = memory_devices_get_system_memory_str(); +    if (mem) { +        gchar *types = memory_devices_get_system_memory_types_str(); +        gchar *ret = g_strdup_printf("%s %s\n%s", mem, types, avail ? avail : ""); +        g_free(avail); +        g_free(mem); +        g_free(types); +        return (gchar*)idle_free(ret); // [1] idle_free() +    } +    return (gchar*)idle_free(avail); // [1] idle_free() +} + +static gchar *get_machine_type(void) +{ +    return computer_get_virtualization(); +} + +const ShellModuleMethod *hi_exported_methods(void) +{ +    static const ShellModuleMethod m[] = { +        {"getOSKernel", get_os_kernel}, +        {"getOS", get_os}, +        {"getDisplaySummary", get_display_summary}, +        {"getOGLRenderer", get_ogl_renderer}, +        {"getAudioCards", get_audio_cards}, +        {"getKernelModuleDescription", get_kernel_module_description}, +        {"getMemoryTotal", get_memory_total}, +        {"getMemoryDesc", get_memory_desc}, +        {"getMachineType", get_machine_type}, +        {NULL}, +    }; + +    return m; +} + +ModuleEntry *hi_module_get_entries(void) +{ +    return entries; +} + +gchar *hi_module_get_name(void) +{ +    return g_strdup(_("Computer")); +} + +guchar hi_module_get_weight(void) +{ +    return 80; +} + +gchar **hi_module_get_dependencies(void) +{ +    static gchar *deps[] = { "devices.so", NULL }; + +    return deps; +} + +gchar *hi_module_get_summary(void) +{ +    gchar *virt = computer_get_virtualization(); +    gchar *machine_type = g_strdup_printf("%s (%s)", +                                          _("Motherboard"), +                                          (char*)idle_free(virt)); + +    return g_strdup_printf("[%s]\n" +                    "Icon=os.png\n" +                    "Method=computer::getOS\n" +                    "[%s]\n" +                    "Icon=processor.png\n" +                    "Method=devices::getProcessorNameAndDesc\n" +                    "[%s]\n" +                    "Icon=memory.png\n" +                    "Method=computer::getMemoryDesc\n" +                    "[%s]\n" +                    "Icon=module.png\n" +                    "Method=devices::getMotherboard\n" +                    "[%s]\n" +                    "Icon=monitor.png\n" +                    "Method=computer::getDisplaySummary\n" +                    "[%s]\n" +                    "Icon=hdd.png\n" +                    "Method=devices::getStorageDevicesSimple\n" +                    "[%s]\n" +                    "Icon=printer.png\n" +                    "Method=devices::getPrinters\n" +                    "[%s]\n" +                    "Icon=audio.png\n" +                    "Method=computer::getAudioCards\n", +                    _("Operating System"), +                    _("Processor"), _("Memory"), (char*)idle_free(machine_type), _("Graphics"), +                    _("Storage"), _("Printers"), _("Audio") +                    ); +} + +void hi_module_deinit(void) +{ +    g_hash_table_destroy(memlabels); + +    if (computer->os) { +        g_free(computer->os->kernel); +        g_free(computer->os->kcmdline); +        g_free(computer->os->libc); +        g_free(computer->os->distrocode); +        g_free(computer->os->distro); +        g_free(computer->os->hostname); +        g_free(computer->os->language); +        g_free(computer->os->homedir); +        g_free(computer->os->kernel_version); +        g_free(computer->os->languages); +        g_free(computer->os->desktop); +        g_free(computer->os->username); +        g_free(computer->os->boots); +        g_free(computer->os); +    } + +    computer_free_display(computer->display); + +    if (computer->alsa) { +        g_slist_free(computer->alsa->cards); +        g_free(computer->alsa); +    } + +    g_free(computer->date_time); +    g_free(computer); + +    moreinfo_del_with_prefix("COMP"); +} + +void hi_module_init(void) +{ +    computer = g_new0(Computer, 1); +    init_memory_labels(); +    kernel_module_icon_init(); +} + +const ModuleAbout *hi_module_get_about(void) +{ +    static const ModuleAbout ma = { +        .author = "L. A. F. Pereira", +        .description = N_("Gathers high-level computer information"), +        .version = VERSION, +        .license = "GNU GPL version 2 or later.",}; + +    return &ma; +} + +static const gchar *hinote_kmod() { +    static gchar note[note_max_len] = ""; +    gboolean ok = TRUE; +    *note = 0; /* clear */ +    ok &= note_require_tool("lsmod", note, _("<i><b>lsmod</b></i> is required.")); +    return ok ? NULL : g_strstrip(note); /* remove last \n */ +} + +static const gchar *hinote_display() { +    static gchar note[note_max_len] = ""; +    gboolean ok = TRUE; +    *note = 0; /* clear */ +    ok &= note_require_tool("xrandr", note, _("X.org's <i><b>xrandr</b></i> utility provides additional details when available.")); +    ok &= note_require_tool("glxinfo", note, _("Mesa's <i><b>glxinfo</b></i> utility is required for OpenGL information.")); +    return ok ? NULL : g_strstrip(note); /* remove last \n */ +} + +const gchar *hi_note_func(gint entry) +{ +    if (entry == ENTRY_KMOD) { +        return hinote_kmod(); +    } +    else if (entry == ENTRY_DISPLAY) { +        return hinote_display(); +    } +    return NULL; +} | 
