pmap vmflags support

Some new kernel version added a line in /proc/pid/smaps listing a processes vmflags. This broke pmap such that pmap -X and pmap -XX would always fail.

This patch adds support for the vmflags field so that -X and -XX work again AND they display the flags.

Merge commit 'refs/merge-requests/8' of git://gitorious.org/procps/procps into merge-requests/8
This commit is contained in:
Craig Small 2013-01-24 22:39:42 +11:00
commit dca3c6d08f

30
pmap.c
View File

@ -172,6 +172,8 @@ static char *mapping_name(proc_t * p, unsigned KLONG addr,
#define DETL "31" /* for format strings */ #define DETL "31" /* for format strings */
#define NUM_LENGTH 21 /* python says: len(str(2**64)) == 20 */ #define NUM_LENGTH 21 /* python says: len(str(2**64)) == 20 */
#define NUML "20" /* for format strings */ #define NUML "20" /* for format strings */
#define VMFLAGS_LENGTH 81 /* There are 27 posible 2 character vmflags
as of this patch */
struct listnode { struct listnode {
char description[DETAIL_LENGTH]; char description[DETAIL_LENGTH];
@ -204,12 +206,14 @@ static void print_extended_maps (FILE *f)
detail_desc[DETAIL_LENGTH], value_str[NUM_LENGTH], detail_desc[DETAIL_LENGTH], value_str[NUM_LENGTH],
start[NUM_LENGTH], end[NUM_LENGTH], start[NUM_LENGTH], end[NUM_LENGTH],
offset[NUM_LENGTH], inode[NUM_LENGTH], offset[NUM_LENGTH], inode[NUM_LENGTH],
dev[64], fmt_str[64]; dev[64], fmt_str[64],
int maxw1=0, maxw2=0, maxw3=0, maxw4=0, maxw5=0; vmflags[VMFLAGS_LENGTH];
int maxw1=0, maxw2=0, maxw3=0, maxw4=0, maxw5=0, maxwv=0;
int nfields, firstmapping, footer_gap, i; int nfields, firstmapping, footer_gap, i;
unsigned KLONG value; unsigned KLONG value;
char *ret; char *ret;
char c; char c;
char has_vmflags = 0;
ret = fgets(mapbuf, sizeof mapbuf, f); ret = fgets(mapbuf, sizeof mapbuf, f);
firstmapping = 1; firstmapping = 1;
@ -284,6 +288,15 @@ loop_end:
nfields = sscanf(mapbuf, "%"DETL"[^:]: %"NUML"[0-9] kB %c", nfields = sscanf(mapbuf, "%"DETL"[^:]: %"NUML"[0-9] kB %c",
detail_desc, value_str, &c); detail_desc, value_str, &c);
} }
/* === GET VMFLAGS === */
nfields = sscanf(mapbuf, "VmFlags: %[a-z ]", vmflags);
if (nfields == 1) {
if (! has_vmflags) has_vmflags = 1;
ret = fgets(mapbuf, sizeof mapbuf, f);
if (strlen(vmflags) > maxwv) maxwv = strlen(vmflags);
}
/* === PRINT THIS MAPPING === */ /* === PRINT THIS MAPPING === */
/* Print header */ /* Print header */
if (firstmapping && !q_option) { if (firstmapping && !q_option) {
@ -292,6 +305,7 @@ loop_end:
if (strlen("Offset") > maxw3) maxw3 = strlen("Offset"); if (strlen("Offset") > maxw3) maxw3 = strlen("Offset");
if (strlen("Device") > maxw4) maxw4 = strlen("Device"); if (strlen("Device") > maxw4) maxw4 = strlen("Device");
if (strlen("Inode") > maxw5) maxw5 = strlen("Inode"); if (strlen("Inode") > maxw5) maxw5 = strlen("Inode");
if (has_vmflags && strlen("VmFlags") > maxwv) maxwv = strlen("VmFlags");
sprintf(fmt_str, "%%%ds %%%ds %%%ds %%%ds %%%ds", sprintf(fmt_str, "%%%ds %%%ds %%%ds %%%ds %%%ds",
maxw1, maxw2, maxw3, maxw4, maxw5); maxw1, maxw2, maxw3, maxw4, maxw5);
printf(fmt_str, "Address", "Flags", "Offset", "Device", "Inode"); printf(fmt_str, "Address", "Flags", "Offset", "Device", "Inode");
@ -301,6 +315,12 @@ loop_end:
sprintf(fmt_str, " %%%ds", listnode->max_width); sprintf(fmt_str, " %%%ds", listnode->max_width);
printf(fmt_str, listnode->description); printf(fmt_str, listnode->description);
} }
if (has_vmflags) {
sprintf(fmt_str, " %%%ds", maxwv);
printf(fmt_str, "VmFlags");
}
printf(" %s\n", "Description"); printf(" %s\n", "Description");
} }
/* Print data */ /* Print data */
@ -313,6 +333,12 @@ loop_end:
sprintf(fmt_str, " %%%ds", listnode->max_width); sprintf(fmt_str, " %%%ds", listnode->max_width);
printf(fmt_str, listnode->value_str); printf(fmt_str, listnode->value_str);
} }
if (has_vmflags) {
sprintf(fmt_str, " %%%ds", maxwv);
printf(fmt_str, vmflags);
}
printf(" %s\n", map_desc); printf(" %s\n", map_desc);
firstmapping = 0; firstmapping = 0;