Looks for smaps member using a gperf table instead of using
bsearch. Brings a 25% boost.
This commit is contained in:
@@ -2,6 +2,8 @@ INCLUDES = @INCLUDES@
|
|||||||
|
|
||||||
noinst_LTLIBRARIES = libgtop_sysdeps-2.0.la
|
noinst_LTLIBRARIES = libgtop_sysdeps-2.0.la
|
||||||
|
|
||||||
|
EXTRA_DIST = procmap_smaps.gperf procmap_smaps.c
|
||||||
|
|
||||||
libgtop_sysdeps_2_0_la_SOURCES = open.c close.c cpu.c mem.c swap.c \
|
libgtop_sysdeps_2_0_la_SOURCES = open.c close.c cpu.c mem.c swap.c \
|
||||||
uptime.c loadavg.c shm_limits.c msg_limits.c \
|
uptime.c loadavg.c shm_limits.c msg_limits.c \
|
||||||
sem_limits.c proclist.c procstate.c procuid.c \
|
sem_limits.c proclist.c procstate.c procuid.c \
|
||||||
|
@@ -31,6 +31,7 @@
|
|||||||
|
|
||||||
#include "glibtop_private.h"
|
#include "glibtop_private.h"
|
||||||
|
|
||||||
|
#include "procmap_smaps.c"
|
||||||
|
|
||||||
#define MAPS_FILE "/proc/%u/maps"
|
#define MAPS_FILE "/proc/%u/maps"
|
||||||
#define SMAPS_FILE "/proc/%u/smaps"
|
#define SMAPS_FILE "/proc/%u/smaps"
|
||||||
@@ -66,35 +67,19 @@ _glibtop_init_proc_map_s (glibtop *server)
|
|||||||
/* Provides detailed information about a process. */
|
/* Provides detailed information about a process. */
|
||||||
|
|
||||||
|
|
||||||
|
static const char*
|
||||||
struct smap_value {
|
|
||||||
char name[16];
|
|
||||||
size_t name_len;
|
|
||||||
ptrdiff_t offset;
|
|
||||||
};
|
|
||||||
|
|
||||||
static int
|
|
||||||
compare(const void* a_key, const void* a_smap)
|
|
||||||
{
|
|
||||||
const char* key = a_key;
|
|
||||||
const struct smap_value* smap = a_smap;
|
|
||||||
return strncmp(key, smap->name, smap->name_len);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static gboolean
|
|
||||||
is_smap_value(const char* s)
|
is_smap_value(const char* s)
|
||||||
{
|
{
|
||||||
for ( ; *s; ++s) {
|
for ( ; *s; ++s) {
|
||||||
|
|
||||||
if (isspace(*s))
|
if (isspace(*s))
|
||||||
return FALSE;
|
return NULL;
|
||||||
|
|
||||||
if (*s == ':')
|
if (*s == ':')
|
||||||
return TRUE;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
return FALSE;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -105,25 +90,18 @@ is_smap_value(const char* s)
|
|||||||
static gboolean
|
static gboolean
|
||||||
parse_smaps(glibtop_map_entry *entry, const char* line)
|
parse_smaps(glibtop_map_entry *entry, const char* line)
|
||||||
{
|
{
|
||||||
#define SMAP_OFFSET(MEMBER) offsetof(glibtop_map_entry, MEMBER)
|
|
||||||
#define STR_AND_LEN(X) (X), (sizeof X - 1)
|
|
||||||
|
|
||||||
/* keep sorted */
|
const struct smap_value* smap;
|
||||||
const struct smap_value values[] = {
|
size_t len;
|
||||||
{ STR_AND_LEN("Private_Clean:"), SMAP_OFFSET(private_clean) },
|
const char* colon;
|
||||||
{ STR_AND_LEN("Private_Dirty:"), SMAP_OFFSET(private_dirty) },
|
|
||||||
{ STR_AND_LEN("Rss:"), SMAP_OFFSET(rss) },
|
|
||||||
{ STR_AND_LEN("Shared_Clean:"), SMAP_OFFSET(shared_clean) },
|
|
||||||
{ STR_AND_LEN("Shared_Dirty:"), SMAP_OFFSET(shared_dirty) },
|
|
||||||
{ STR_AND_LEN("Size:"), SMAP_OFFSET(size) }
|
|
||||||
};
|
|
||||||
|
|
||||||
#undef STR_AND_LEN
|
if ((colon = is_smap_value(line)) == NULL)
|
||||||
#undef SMAP_OFFSET
|
return FALSE;
|
||||||
|
|
||||||
struct smap_value* smap;
|
len = colon - line + 1;
|
||||||
|
smap = _glibtop_find_smap(line, len);
|
||||||
|
|
||||||
smap = bsearch(line, values, G_N_ELEMENTS(values), sizeof values[0], compare);
|
// g_debug("smap %s -> %p", line, smap);
|
||||||
|
|
||||||
if (smap) {
|
if (smap) {
|
||||||
char *offset;
|
char *offset;
|
||||||
@@ -133,11 +111,10 @@ parse_smaps(glibtop_map_entry *entry, const char* line)
|
|||||||
offset += smap->offset;
|
offset += smap->offset;
|
||||||
value = (void*) offset;
|
value = (void*) offset;
|
||||||
|
|
||||||
*value = get_scaled(line + smap->name_len, NULL);
|
*value = get_scaled(line + len, NULL);
|
||||||
return TRUE;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return is_smap_value(line);
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -215,6 +192,7 @@ glibtop_get_proc_map_s (glibtop *server, glibtop_proc_map *buf, pid_t pid)
|
|||||||
It's the average number of entry per process on my laptop
|
It's the average number of entry per process on my laptop
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
size_t added = 0;
|
||||||
GArray *entry_list = g_array_sized_new(FALSE, FALSE,
|
GArray *entry_list = g_array_sized_new(FALSE, FALSE,
|
||||||
sizeof(glibtop_map_entry),
|
sizeof(glibtop_map_entry),
|
||||||
100);
|
100);
|
||||||
@@ -242,7 +220,6 @@ glibtop_get_proc_map_s (glibtop *server, glibtop_proc_map *buf, pid_t pid)
|
|||||||
while(TRUE)
|
while(TRUE)
|
||||||
{
|
{
|
||||||
unsigned long perm;
|
unsigned long perm;
|
||||||
guint len;
|
|
||||||
/* int line_end; */
|
/* int line_end; */
|
||||||
|
|
||||||
unsigned short dev_major, dev_minor;
|
unsigned short dev_major, dev_minor;
|
||||||
@@ -293,9 +270,12 @@ glibtop_get_proc_map_s (glibtop *server, glibtop_proc_map *buf, pid_t pid)
|
|||||||
avoid copying the entry, grow by 1 and point to the last
|
avoid copying the entry, grow by 1 and point to the last
|
||||||
element.
|
element.
|
||||||
*/
|
*/
|
||||||
len = entry_list->len;
|
|
||||||
g_array_set_size(entry_list, len + 1);
|
if (G_UNLIKELY(added >= entry_list->len)) {
|
||||||
entry = &g_array_index(entry_list, glibtop_map_entry, len);
|
g_array_set_size(entry_list, 2 * entry_list->len);
|
||||||
|
}
|
||||||
|
|
||||||
|
entry = &g_array_index(entry_list, glibtop_map_entry, added++);
|
||||||
|
|
||||||
entry->flags = _glibtop_sysdeps_map_entry;
|
entry->flags = _glibtop_sysdeps_map_entry;
|
||||||
entry->start = start;
|
entry->start = start;
|
||||||
@@ -322,6 +302,7 @@ glibtop_get_proc_map_s (glibtop *server, glibtop_proc_map *buf, pid_t pid)
|
|||||||
|
|
||||||
eof:
|
eof:
|
||||||
|
|
||||||
|
g_array_set_size(entry_list, added);
|
||||||
free(line);
|
free(line);
|
||||||
fclose (maps);
|
fclose (maps);
|
||||||
|
|
||||||
|
172
sysdeps/linux/procmap_smaps.c
Normal file
172
sysdeps/linux/procmap_smaps.c
Normal file
@@ -0,0 +1,172 @@
|
|||||||
|
/* ANSI-C code produced by gperf version 3.0.3 */
|
||||||
|
/* Command-line: gperf sysdeps/linux/procmap_smaps.gperf */
|
||||||
|
/* Computed positions: -k'9' */
|
||||||
|
|
||||||
|
#if !((' ' == 32) && ('!' == 33) && ('"' == 34) && ('#' == 35) \
|
||||||
|
&& ('%' == 37) && ('&' == 38) && ('\'' == 39) && ('(' == 40) \
|
||||||
|
&& (')' == 41) && ('*' == 42) && ('+' == 43) && (',' == 44) \
|
||||||
|
&& ('-' == 45) && ('.' == 46) && ('/' == 47) && ('0' == 48) \
|
||||||
|
&& ('1' == 49) && ('2' == 50) && ('3' == 51) && ('4' == 52) \
|
||||||
|
&& ('5' == 53) && ('6' == 54) && ('7' == 55) && ('8' == 56) \
|
||||||
|
&& ('9' == 57) && (':' == 58) && (';' == 59) && ('<' == 60) \
|
||||||
|
&& ('=' == 61) && ('>' == 62) && ('?' == 63) && ('A' == 65) \
|
||||||
|
&& ('B' == 66) && ('C' == 67) && ('D' == 68) && ('E' == 69) \
|
||||||
|
&& ('F' == 70) && ('G' == 71) && ('H' == 72) && ('I' == 73) \
|
||||||
|
&& ('J' == 74) && ('K' == 75) && ('L' == 76) && ('M' == 77) \
|
||||||
|
&& ('N' == 78) && ('O' == 79) && ('P' == 80) && ('Q' == 81) \
|
||||||
|
&& ('R' == 82) && ('S' == 83) && ('T' == 84) && ('U' == 85) \
|
||||||
|
&& ('V' == 86) && ('W' == 87) && ('X' == 88) && ('Y' == 89) \
|
||||||
|
&& ('Z' == 90) && ('[' == 91) && ('\\' == 92) && (']' == 93) \
|
||||||
|
&& ('^' == 94) && ('_' == 95) && ('a' == 97) && ('b' == 98) \
|
||||||
|
&& ('c' == 99) && ('d' == 100) && ('e' == 101) && ('f' == 102) \
|
||||||
|
&& ('g' == 103) && ('h' == 104) && ('i' == 105) && ('j' == 106) \
|
||||||
|
&& ('k' == 107) && ('l' == 108) && ('m' == 109) && ('n' == 110) \
|
||||||
|
&& ('o' == 111) && ('p' == 112) && ('q' == 113) && ('r' == 114) \
|
||||||
|
&& ('s' == 115) && ('t' == 116) && ('u' == 117) && ('v' == 118) \
|
||||||
|
&& ('w' == 119) && ('x' == 120) && ('y' == 121) && ('z' == 122) \
|
||||||
|
&& ('{' == 123) && ('|' == 124) && ('}' == 125) && ('~' == 126))
|
||||||
|
/* The character set is not based on ISO-646. */
|
||||||
|
#error "gperf generated tables don't work with this execution character set. Please report a bug to <bug-gnu-gperf@gnu.org>."
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#line 8 "sysdeps/linux/procmap_smaps.gperf"
|
||||||
|
|
||||||
|
#include "glibtop_private.h"
|
||||||
|
#include <glibtop/procmap.h>
|
||||||
|
#include <stddef.h>
|
||||||
|
#define SMAP_OFFSET(MEMBER) offsetof(glibtop_map_entry, MEMBER)
|
||||||
|
#line 14 "sysdeps/linux/procmap_smaps.gperf"
|
||||||
|
struct smap_value { int name; ptrdiff_t offset; };
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#define TOTAL_KEYWORDS 6
|
||||||
|
#define MIN_WORD_LENGTH 4
|
||||||
|
#define MAX_WORD_LENGTH 14
|
||||||
|
#define MIN_HASH_VALUE 4
|
||||||
|
#define MAX_HASH_VALUE 19
|
||||||
|
/* maximum key range = 16, duplicates = 0 */
|
||||||
|
|
||||||
|
#ifdef __GNUC__
|
||||||
|
__inline
|
||||||
|
#else
|
||||||
|
#ifdef __cplusplus
|
||||||
|
inline
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
static unsigned int
|
||||||
|
hash (register const char *str, register unsigned int len)
|
||||||
|
{
|
||||||
|
static const unsigned char asso_values[] =
|
||||||
|
{
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 5, 0, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 5, 20, 20, 0, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
|
||||||
|
20, 20, 20, 20, 20, 20
|
||||||
|
};
|
||||||
|
register int hval = len;
|
||||||
|
|
||||||
|
switch (hval)
|
||||||
|
{
|
||||||
|
default:
|
||||||
|
hval += asso_values[(unsigned char)str[8]];
|
||||||
|
/*FALLTHROUGH*/
|
||||||
|
case 8:
|
||||||
|
case 7:
|
||||||
|
case 6:
|
||||||
|
case 5:
|
||||||
|
case 4:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return hval;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct stringpool_t
|
||||||
|
{
|
||||||
|
char stringpool_str4[sizeof("Rss:")];
|
||||||
|
char stringpool_str5[sizeof("Size:")];
|
||||||
|
char stringpool_str13[sizeof("Shared_Clean:")];
|
||||||
|
char stringpool_str14[sizeof("Private_Dirty:")];
|
||||||
|
char stringpool_str18[sizeof("Shared_Dirty:")];
|
||||||
|
char stringpool_str19[sizeof("Private_Clean:")];
|
||||||
|
};
|
||||||
|
static const struct stringpool_t stringpool_contents =
|
||||||
|
{
|
||||||
|
"Rss:",
|
||||||
|
"Size:",
|
||||||
|
"Shared_Clean:",
|
||||||
|
"Private_Dirty:",
|
||||||
|
"Shared_Dirty:",
|
||||||
|
"Private_Clean:"
|
||||||
|
};
|
||||||
|
#define stringpool ((const char *) &stringpool_contents)
|
||||||
|
#ifdef __GNUC__
|
||||||
|
__inline
|
||||||
|
#ifdef __GNUC_STDC_INLINE__
|
||||||
|
__attribute__ ((__gnu_inline__))
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
static /* manually added */
|
||||||
|
const struct smap_value *
|
||||||
|
_glibtop_find_smap (register const char *str, register unsigned int len)
|
||||||
|
{
|
||||||
|
static const unsigned char lengthtable[] =
|
||||||
|
{
|
||||||
|
0, 0, 0, 0, 4, 5, 0, 0, 0, 0, 0, 0, 0, 13,
|
||||||
|
14, 0, 0, 0, 13, 14
|
||||||
|
};
|
||||||
|
static const struct smap_value wordlist[] =
|
||||||
|
{
|
||||||
|
{-1}, {-1}, {-1}, {-1},
|
||||||
|
#line 18 "sysdeps/linux/procmap_smaps.gperf"
|
||||||
|
{(int)(long)&((struct stringpool_t *)0)->stringpool_str4, SMAP_OFFSET(rss)},
|
||||||
|
#line 21 "sysdeps/linux/procmap_smaps.gperf"
|
||||||
|
{(int)(long)&((struct stringpool_t *)0)->stringpool_str5, SMAP_OFFSET(size)},
|
||||||
|
{-1}, {-1}, {-1}, {-1}, {-1}, {-1}, {-1},
|
||||||
|
#line 19 "sysdeps/linux/procmap_smaps.gperf"
|
||||||
|
{(int)(long)&((struct stringpool_t *)0)->stringpool_str13, SMAP_OFFSET(shared_clean)},
|
||||||
|
#line 17 "sysdeps/linux/procmap_smaps.gperf"
|
||||||
|
{(int)(long)&((struct stringpool_t *)0)->stringpool_str14, SMAP_OFFSET(private_dirty)},
|
||||||
|
{-1}, {-1}, {-1},
|
||||||
|
#line 20 "sysdeps/linux/procmap_smaps.gperf"
|
||||||
|
{(int)(long)&((struct stringpool_t *)0)->stringpool_str18, SMAP_OFFSET(shared_dirty)},
|
||||||
|
#line 16 "sysdeps/linux/procmap_smaps.gperf"
|
||||||
|
{(int)(long)&((struct stringpool_t *)0)->stringpool_str19, SMAP_OFFSET(private_clean)}
|
||||||
|
};
|
||||||
|
|
||||||
|
if (len <= MAX_WORD_LENGTH && len >= MIN_WORD_LENGTH)
|
||||||
|
{
|
||||||
|
register int key = hash (str, len);
|
||||||
|
|
||||||
|
if (key <= MAX_HASH_VALUE && key >= 0)
|
||||||
|
if (len == lengthtable[key])
|
||||||
|
{
|
||||||
|
register const char *s = wordlist[key].name + stringpool;
|
||||||
|
|
||||||
|
if (*str == *s && !memcmp (str + 1, s + 1, len - 1))
|
||||||
|
return &wordlist[key];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
21
sysdeps/linux/procmap_smaps.gperf
Normal file
21
sysdeps/linux/procmap_smaps.gperf
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
%language=ANSI-C
|
||||||
|
%includes
|
||||||
|
%struct-type
|
||||||
|
%readonly-tables
|
||||||
|
%pic
|
||||||
|
%define lookup-function-name _glibtop_find_smap
|
||||||
|
%compare-lengths
|
||||||
|
%{
|
||||||
|
#include "glibtop_private.h"
|
||||||
|
#include <glibtop/procmap.h>
|
||||||
|
#include <stddef.h>
|
||||||
|
#define SMAP_OFFSET(MEMBER) offsetof(glibtop_map_entry, MEMBER)
|
||||||
|
%}
|
||||||
|
struct smap_value { int name; ptrdiff_t offset; };
|
||||||
|
%%
|
||||||
|
Private_Clean:, SMAP_OFFSET(private_clean)
|
||||||
|
Private_Dirty:, SMAP_OFFSET(private_dirty)
|
||||||
|
Rss:, SMAP_OFFSET(rss)
|
||||||
|
Shared_Clean:, SMAP_OFFSET(shared_clean)
|
||||||
|
Shared_Dirty:, SMAP_OFFSET(shared_dirty)
|
||||||
|
Size:, SMAP_OFFSET(size)
|
Reference in New Issue
Block a user