diff options
author | Xan Lopez <xlopez@igalia.com> | 2011-06-28 00:18:58 +0800 |
---|---|---|
committer | Xan Lopez <xlopez@igalia.com> | 2011-06-28 01:34:52 +0800 |
commit | 836a072b8a9404be59187f66f46c64998670a534 (patch) | |
tree | 09e8769a144518730455cf40e62679c2e473a7fe | |
parent | fe909aa62629a5994dd643450b0de45e0e3f508d (diff) | |
download | gsoc2013-epiphany-836a072b8a9404be59187f66f46c64998670a534.tar gsoc2013-epiphany-836a072b8a9404be59187f66f46c64998670a534.tar.gz gsoc2013-epiphany-836a072b8a9404be59187f66f46c64998670a534.tar.bz2 gsoc2013-epiphany-836a072b8a9404be59187f66f46c64998670a534.tar.lz gsoc2013-epiphany-836a072b8a9404be59187f66f46c64998670a534.tar.xz gsoc2013-epiphany-836a072b8a9404be59187f66f46c64998670a534.tar.zst gsoc2013-epiphany-836a072b8a9404be59187f66f46c64998670a534.zip |
Implement about:memory
Gives an estimate of how much memory the browser process is using,
extracted from /proc/$PID/smaps. Only works in GNU/Linux systems.
-rw-r--r-- | data/pages/about.css | 33 | ||||
-rw-r--r-- | lib/Makefile.am | 2 | ||||
-rw-r--r-- | lib/ephy-request-about.c | 37 | ||||
-rw-r--r-- | lib/ephy-smaps.c | 337 | ||||
-rw-r--r-- | lib/ephy-smaps.h | 50 |
5 files changed, 456 insertions, 3 deletions
diff --git a/data/pages/about.css b/data/pages/about.css index 37f1cc066..af498283d 100644 --- a/data/pages/about.css +++ b/data/pages/about.css @@ -45,3 +45,36 @@ font-size: 1.5em; text-align: right; } + +/* about:memory */ + +.memory-table caption +{ + margin-bottom: 0.5em; +} + +.memory-table +{ + margin: 14px; + width: 75%; + border-collapse: collapse; +} + +.memory-table th +{ + padding: 2px; + background:#b9c9ff; + border-top: 4px solid #aabcff; + border-bottom: 1px solid #fff; + color: #039; +} + +.memory-table td +{ + padding: 2px; + background: #e8edff; + border-bottom: 1px solid #fff; + color: #669; + border-top: 1px solidtransparent; +} + diff --git a/lib/Makefile.am b/lib/Makefile.am index 0f8dd5c7c..3b4172d28 100644 --- a/lib/Makefile.am +++ b/lib/Makefile.am @@ -25,6 +25,7 @@ NOINST_H_FILES = \ ephy-request-about.h \ ephy-shlib-loader.h \ ephy-signal-accumulator.h \ + ephy-smaps.h \ ephy-string.h \ ephy-stock-icons.h \ ephy-time-helpers.h \ @@ -66,6 +67,7 @@ libephymisc_la_SOURCES = \ ephy-settings.c \ ephy-shlib-loader.c \ ephy-signal-accumulator.c \ + ephy-smaps.c \ ephy-state.c \ ephy-string.c \ ephy-stock-icons.c \ diff --git a/lib/ephy-request-about.c b/lib/ephy-request-about.c index a809f29f5..4601e4787 100644 --- a/lib/ephy-request-about.c +++ b/lib/ephy-request-about.c @@ -1,8 +1,21 @@ /* -*- Mode: C; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* - * ephy-request-about.c: about: URI request object + * Copyright © 2011 Igalia S.L. + * + * 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 Street, Fifth Floor, Boston, MA 02110-1301, USA. * - * Copyright (C) 2011, Igalia S.L. */ #ifdef HAVE_CONFIG_H @@ -16,12 +29,14 @@ #include "ephy-file-helpers.h" #include "ephy-request-about.h" +#include "ephy-smaps.h" G_DEFINE_TYPE (EphyRequestAbout, ephy_request_about, SOUP_TYPE_REQUEST) struct _EphyRequestAboutPrivate { gssize content_length; gchar *css_style; + EphySMaps *smaps; }; static void @@ -30,12 +45,16 @@ ephy_request_about_init (EphyRequestAbout *about) about->priv = G_TYPE_INSTANCE_GET_PRIVATE (about, EPHY_TYPE_REQUEST_ABOUT, EphyRequestAboutPrivate); about->priv->content_length = 0; about->priv->css_style = NULL; + about->priv->smaps = ephy_smaps_new (); } static void ephy_request_about_finalize (GObject *obj) { - g_free (EPHY_REQUEST_ABOUT (obj)->priv->css_style); + EphyRequestAboutPrivate *priv = EPHY_REQUEST_ABOUT (obj)->priv; + + g_object_unref (priv->smaps); + g_free (priv->css_style); G_OBJECT_CLASS (ephy_request_about_parent_class)->finalize (obj); } @@ -114,6 +133,18 @@ ephy_request_about_send (SoupRequest *request, webkit_web_plugin_database_plugins_list_free (plugin_list); g_string_append (data_str, "</body>"); + } else if (!g_strcmp0 (uri->path, "memory")) { + char *memory = ephy_smaps_to_html (EPHY_REQUEST_ABOUT (request)->priv->smaps); + + if (memory) { + g_string_append_printf (data_str, "<head><title>%s</title>" \ + "<style type=\"text/css\">%s</style></head><body>", + _("Memory usage"), + about->priv->css_style); + + g_string_append (data_str, memory); + g_free (memory); + } } else if (!g_strcmp0 (uri->path, "epiphany")) { g_string_append_printf (data_str, "<head><title>Epiphany</title>" \ diff --git a/lib/ephy-smaps.c b/lib/ephy-smaps.c new file mode 100644 index 000000000..eab640094 --- /dev/null +++ b/lib/ephy-smaps.c @@ -0,0 +1,337 @@ +/* -*- Mode: C; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ +/* + * Copyright © 2011 Igalia S.L. + * + * 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 Street, Fifth Floor, Boston, MA 02110-1301, USA. + * + */ + +#include "config.h" +#include "ephy-smaps.h" + +#include <gio/gio.h> +#include <stdio.h> +#include <string.h> + +G_DEFINE_TYPE (EphySMaps, ephy_smaps, G_TYPE_OBJECT) + +struct _EphySMapsPrivate { + GRegex *header; + GRegex *detail; +}; + +typedef struct { + char *start; + char *end; + char *perms; + char *offset; + char *major; + char *minor; + char *inode; + char *filename; + char *size; + char *rss; + char *pss; + char *shared_clean; + char *shared_dirty; + char *private_clean; + char *private_dirty; +} VMA_t; + +typedef struct { + guint shared_clean; + guint shared_dirty; + guint private_clean; + guint private_dirty; +} PermEntry; + +static void vma_free (VMA_t* vma) +{ + g_free (vma->start); + g_free (vma->end); + g_free (vma->perms); + g_free (vma->offset); + g_free (vma->major); + g_free (vma->minor); + g_free (vma->inode); + g_free (vma->filename); + g_free (vma->size); + g_free (vma->rss); + g_free (vma->pss); + g_free (vma->shared_clean); + g_free (vma->shared_dirty); + g_free (vma->private_clean); + g_free (vma->private_dirty); + + g_slice_free (VMA_t, vma); +} + +static void perm_entry_free (PermEntry *entry) +{ + if (entry) + g_slice_free (PermEntry, entry); +} + +static void add_to_perm_entry (GHashTable *hash, VMA_t *entry) +{ + const char *perms = entry->perms; + PermEntry *value; + guint number; + gboolean insert = FALSE; + + value = g_hash_table_lookup (hash, perms); + + if (!value) { + value = g_slice_new0 (PermEntry); + insert = TRUE; + } + + sscanf (entry->shared_clean, "%u", &number); + value->shared_clean += number; + sscanf (entry->shared_dirty, "%u", &number); + value->shared_dirty += number; + sscanf (entry->private_clean, "%u", &number); + value->private_clean += number; + sscanf (entry->private_dirty, "%u", &number); + value->private_dirty += number; + + if (insert) + g_hash_table_insert (hash, g_strdup (perms), value); +} + +static void add_to_totals (PermEntry *entry, PermEntry *totals) +{ + totals->shared_clean += entry->shared_clean; + totals->shared_dirty += entry->shared_dirty; + totals->private_dirty += entry->private_dirty; + totals->private_dirty += entry->private_dirty; +} + +static void print_vma_table (GString *str, GHashTable *hash, const char *caption) +{ + PermEntry *pentry, totals; + + memset (&totals, 0, sizeof (PermEntry)); + + g_string_append_printf (str, "<table class=\"memory-table\"><caption>%s</caption><colgroup><colgroup span=\"2\" align=\"center\"><colgroup span=\"2\" align=\"center\"><colgroup><thead><tr><th><th colspan=\"2\">Shared</th><th colspan=\"2\">Private</th><th></tr></thead>", caption); + g_string_append (str, "<tbody><tr><td></td><td>Clean</td><td>Dirty</td><td>Clean</td><td>Dirty</td><td></td></tr>"); + pentry = g_hash_table_lookup (hash, "r-xp"); + if (pentry) { + g_string_append_printf (str, "<tbody><tr><td>r-xp</td><td>%d</td><td>%d</td><td>%d</td><td>%d</td><td>Code</td></tr>", + pentry->shared_clean, pentry->shared_dirty, pentry->private_clean, pentry->private_dirty); + add_to_totals (pentry, &totals); + } + pentry = g_hash_table_lookup (hash, "rw-p"); + if (pentry) { + g_string_append_printf (str, "<tbody><tr><td>rw-p</td><td>%d</td><td>%d</td><td>%d</td><td>%d</td><td>Data</td></tr>", + pentry->shared_clean, pentry->shared_dirty, pentry->private_clean, pentry->private_dirty); + add_to_totals (pentry, &totals); + } + pentry = g_hash_table_lookup (hash, "r--p"); + if (pentry) { + g_string_append_printf (str, "<tbody><tr><td>r--p</td><td>%d</td><td>%d</td><td>%d</td><td>%d</td><td>Read-only Data</td></tr>", + pentry->shared_clean, pentry->shared_dirty, pentry->private_clean, pentry->private_dirty); + add_to_totals (pentry, &totals); + } + pentry = g_hash_table_lookup (hash, "---p"); + if (pentry) { + g_string_append_printf (str, "<tbody><tr><td>---p</td><td>%d</td><td>%d</td><td>%d</td><td>%d</td><td></td></tr>", + pentry->shared_clean, pentry->shared_dirty, pentry->private_clean, pentry->private_dirty); + add_to_totals (pentry, &totals); + } + pentry = g_hash_table_lookup (hash, "r--s"); + if (pentry) { + g_string_append_printf (str, "<tbody><tr><td>r--s</td><td>%d</td><td>%d</td><td>%d</td><td>%d</td><td></td></tr>", + pentry->shared_clean, pentry->shared_dirty, pentry->private_clean, pentry->private_dirty); + add_to_totals (pentry, &totals); + } + g_string_append_printf (str, "<tbody><tr><td>Total:</td><td>%d kB</td><td>%d kB</td><td>%d kB</td><td>%d kB</td><td></td></tr>", + totals.shared_clean, totals.shared_dirty, totals.private_clean, totals.private_dirty); + g_string_append (str, "</table>"); +} + +char* ephy_smaps_to_html (EphySMaps *smaps) +{ + GFileInputStream *stream; + GDataInputStream *data_stream; + GFile *file = g_file_new_for_path ("/proc/self/smaps"); + char *line; + GString *str; + GError *error = NULL; + VMA_t *vma = NULL; + GHashTable *anon_hash, *mapped_hash; + GSList *vma_entries = NULL, *p; + EphySMapsPrivate *priv = smaps->priv; + + stream = g_file_read (file, NULL, &error); + g_object_unref (file); + + if (error && error->code == G_IO_ERROR_NOT_FOUND ) { + /* This is not GNU/Linux, do nothing. */ + g_error_free (error); + return NULL; + } + + data_stream = g_data_input_stream_new (G_INPUT_STREAM (stream)); + g_object_unref (stream); + + str = g_string_new (""); + + while ((line = g_data_input_stream_read_line (data_stream, NULL, NULL, NULL))) { + GMatchInfo *match_info = NULL; + gboolean matched = FALSE; + + g_regex_match (priv->header, line, 0, &match_info); + if (g_match_info_matches (match_info)) { + matched = TRUE; + + if (vma) + vma_entries = g_slist_append (vma_entries, vma); + + vma = g_slice_new0 (VMA_t); + + vma->start = g_match_info_fetch (match_info, 1); + vma->end = g_match_info_fetch (match_info, 2); + vma->perms = g_match_info_fetch (match_info, 3); + vma->offset = g_match_info_fetch (match_info, 4); + vma->major = g_match_info_fetch (match_info, 5); + vma->minor = g_match_info_fetch (match_info, 6); + vma->inode = g_match_info_fetch (match_info, 7); + vma->filename = g_match_info_fetch (match_info, 8); + } + + g_match_info_free (match_info); + + if (matched) + goto out; + + g_regex_match (priv->detail, line, 0, &match_info); + if (g_match_info_matches (match_info)) { + char *name = g_match_info_fetch (match_info, 1); + char **size = NULL; + + if (g_str_equal (name, "Size")) + size = &vma->size; + else if (g_str_equal (name, "Rss")) + size = &vma->rss; + else if (g_str_equal (name, "Pss")) + size = &vma->pss; + else if (g_str_equal (name, "Shared_Clean")) + size = &vma->shared_clean; + else if (g_str_equal (name, "Shared_Dirty")) + size = &vma->shared_dirty; + else if (g_str_equal (name, "Private_Clean")) + size = &vma->private_clean; + else if (g_str_equal (name, "Private_Dirty")) + size = &vma->private_dirty; + + if (size) + *size = g_match_info_fetch (match_info, 2); + + g_free (name); + } + + g_match_info_free (match_info); + out: + g_free (line); + } + + if (vma) + vma_entries = g_slist_append (vma_entries, vma); + + g_object_unref (data_stream); + + /* All the file is parsed now. We have parsed more stuff than what + * we are going to use, but it might be useful in the future. */ + anon_hash = g_hash_table_new_full (g_str_hash, + g_str_equal, + (GDestroyNotify)g_free, + (GDestroyNotify)perm_entry_free); + mapped_hash = g_hash_table_new_full (g_str_hash, + g_str_equal, + (GDestroyNotify)g_free, + (GDestroyNotify)perm_entry_free); + + for (p = vma_entries; p; p = p->next) { + VMA_t *entry = (VMA_t*)p->data; + + if (g_strcmp0 (entry->major, "00") && g_strcmp0 (entry->minor, "00")) + add_to_perm_entry (anon_hash, entry); + else + add_to_perm_entry (mapped_hash, entry); + + vma_free (entry); + } + + g_slist_free (vma_entries); + + /* Create the page. */ + g_string_append (str, "<body>"); + + /* Anon table. */ + print_vma_table (str, anon_hash, "Anonymous memory"); + + /* Mapped table. */ + print_vma_table (str, mapped_hash, "Mapped memory"); + + g_string_append (str, "</body>"); + + /* Done. */ + g_hash_table_unref (anon_hash); + g_hash_table_unref (mapped_hash); + + return g_string_free (str, FALSE); +} + +static void +ephy_smaps_init (EphySMaps *smaps) +{ + smaps->priv = G_TYPE_INSTANCE_GET_PRIVATE (smaps, EPHY_TYPE_SMAPS, EphySMapsPrivate); + + /* Prepare the regexps for the smaps file. */ + smaps->priv->header = g_regex_new ("^([0-9a-f]+)-([0-9a-f]+) (....) ([0-9a-f]+) (..):(..) (\\d+) *(.*)$", + G_REGEX_OPTIMIZE, + 0, + NULL); + smaps->priv->detail = g_regex_new ("^(.*): +(\\d+) kB", G_REGEX_OPTIMIZE, 0, NULL); +} + +static void +ephy_smaps_finalize (GObject *obj) +{ + EphySMapsPrivate *priv = EPHY_SMAPS (obj)->priv; + + g_regex_unref (priv->header); + g_regex_unref (priv->detail); + + G_OBJECT_CLASS (ephy_smaps_parent_class)->finalize (obj); +} + +static void +ephy_smaps_class_init (EphySMapsClass *smaps_class) +{ + GObjectClass *gobject_class = G_OBJECT_CLASS (smaps_class); + + gobject_class->finalize = ephy_smaps_finalize; + + g_type_class_add_private (smaps_class, sizeof (EphySMapsPrivate)); +} + +EphySMaps *ephy_smaps_new () +{ + return EPHY_SMAPS (g_object_new (EPHY_TYPE_SMAPS, NULL)); +} + + diff --git a/lib/ephy-smaps.h b/lib/ephy-smaps.h new file mode 100644 index 000000000..728198656 --- /dev/null +++ b/lib/ephy-smaps.h @@ -0,0 +1,50 @@ +/* -*- Mode: C; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ +/* + * Copyright © 2011 Igalia S.L. + * + * 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 Street, Fifth Floor, Boston, MA 02110-1301, USA. + * + */ + +#ifndef EPHY_SMAPS_H +#define EPHY_SMAPS_H + +#include <glib-object.h> + +#define EPHY_TYPE_SMAPS (ephy_smaps_get_type ()) +#define EPHY_SMAPS(object) (G_TYPE_CHECK_INSTANCE_CAST ((object), EPHY_TYPE_SMAPS, EphySMaps)) +#define EPHY_SMAPS_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), EPHY_TYPE_SMAPS, EphySMapsClass)) +#define EPHY_IS_SMAPS(object) (G_TYPE_CHECK_INSTANCE_TYPE ((object), EPHY_TYPE_SMAPS)) +#define EPHY_IS_SMAPS_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), EPHY_TYPE_SMAPS)) +#define EPHY_SMAPS_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), EPHY_TYPE_SMAPS, EphySMapsClass)) + +typedef struct _EphySMapsPrivate EphySMapsPrivate; + +typedef struct { + GObject parent; + + EphySMapsPrivate *priv; +} EphySMaps; + +typedef struct { + GObjectClass parent; + +} EphySMapsClass; + +GType ephy_smaps_get_type (void); +EphySMaps * ephy_smaps_new (void); +char * ephy_smaps_to_html (EphySMaps *smaps); + +#endif /* EPHY_SMAPS_H */ |