summaryrefslogtreecommitdiff
path: root/src/pmdas/linux/linux_table.c
blob: d04f45488dc1d0b71dcb1bc332c79191fea2a7e2 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
/*
 * Copyright (c) 2012 Red Hat.
 * Copyright (c) 2004 Silicon Graphics, Inc.  All Rights Reserved.
 *
 * 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.
 */

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <stdint.h>

#include "linux_table.h"

extern int linux_table_lookup(const char *field, struct linux_table *table, uint64_t *val);
extern struct linux_table *linux_table_clone(struct linux_table *table);
extern int linux_table_scan(FILE *fp, struct linux_table *table);

inline int
linux_table_lookup(const char *field, struct linux_table *table, uint64_t *val)
{
    struct linux_table *t;

    for (t=table; t && t->field; t++) {
	if (strncmp(field, t->field, t->field_len) == 0) {
	    if (t->valid) {
		*val = t->val;
		return 1;
	    }
	    /* Invalid */
	    return 0;
	}
    }

    fprintf(stderr, "Warning: linux_table_lookup failed for \"%s\"\n", field);
    return 0;
}

inline struct linux_table *
linux_table_clone(struct linux_table *table)
{
    struct linux_table *ret;
    struct linux_table *t;
    int len;

    if (!table)
	return NULL;
    for (len=1, t=table; t->field; t++)
    	len++;
    ret = (struct linux_table *)malloc(len * sizeof(struct linux_table));
    if (!ret)
	return NULL;
    memcpy(ret, table, len * sizeof(struct linux_table));

    /* Initialize the table */
    for (t=ret; t && t->field; t++) {
	if (!t->field_len)
	    t->field_len = strlen(t->field);
	t->valid = LINUX_TABLE_INVALID;
    }

    return ret;
}

inline int
linux_table_scan(FILE *fp, struct linux_table *table)
{
    char *p;
    struct linux_table *t;
    char buf[1024];
    int ret = 0;

    while(fgets(buf, sizeof(buf), fp) != NULL) {
	for (t=table; t && t->field; t++) {
	    if ((p = strstr(buf, t->field)) != NULL) {
		/* first digit after the matched field */
		for (p += t->field_len; *p; p++) {
		    if (isdigit((int)*p))
			break;
		}
		if (isdigit((int)*p)) {
		    t->this = strtoul(p, NULL, 10);
		    t->valid = LINUX_TABLE_VALID;
		    ret++;
		    break;
		}
	    }
	}
    }

    /* calculate current value, accounting for counter wrap */
    for (t=table; t && t->field; t++) {
    	if (t->maxval == 0)
	    /* instantaneous value */
	    t->val = t->this;
	else {
	    /* counter value */
	    if (t->this >= t->prev)
		t->val += t->this - t->prev;
	    else
	        t->val += t->this + (t->maxval - t->prev);
	    t->prev = t->this;
	}
    }

    return ret;
}