Merge branch 'collectd-5.5' into collectd-5.6
[collectd.git] / src / irq.c
1 /**
2  * collectd - src/irq.c
3  * Copyright (C) 2007  Peter Holik
4  * Copyright (C) 2011  Florian Forster
5  *
6  * This program is free software; you can redistribute it and/or modify it
7  * under the terms of the GNU General Public License as published by the
8  * Free Software Foundation; either version 2 of the License, or (at your
9  * option) any later version.
10  *
11  * This program is distributed in the hope that it will be useful, but
12  * WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License along
17  * with this program; if not, write to the Free Software Foundation, Inc.,
18  * 51 Franklin St, Fifth Floor, Boston, MA  02110-1301 USA
19  *
20  * Authors:
21  *   Peter Holik <peter at holik.at>
22  **/
23
24 #include "collectd.h"
25
26 #include "common.h"
27 #include "plugin.h"
28 #include "utils_ignorelist.h"
29
30 #if !KERNEL_LINUX
31 #error "No applicable input method."
32 #endif
33
34 /*
35  * (Module-)Global variables
36  */
37 static const char *config_keys[] = {"Irq", "IgnoreSelected"};
38 static int config_keys_num = STATIC_ARRAY_SIZE(config_keys);
39
40 static ignorelist_t *ignorelist = NULL;
41
42 /*
43  * Private functions
44  */
45 static int irq_config(const char *key, const char *value) {
46   if (ignorelist == NULL)
47     ignorelist = ignorelist_create(/* invert = */ 1);
48
49   if (strcasecmp(key, "Irq") == 0) {
50     ignorelist_add(ignorelist, value);
51   } else if (strcasecmp(key, "IgnoreSelected") == 0) {
52     int invert = 1;
53     if (IS_TRUE(value))
54       invert = 0;
55     ignorelist_set_invert(ignorelist, invert);
56   } else {
57     return (-1);
58   }
59
60   return (0);
61 }
62
63 static void irq_submit(const char *irq_name, derive_t value) {
64   value_t values[1];
65   value_list_t vl = VALUE_LIST_INIT;
66
67   if (ignorelist_match(ignorelist, irq_name) != 0)
68     return;
69
70   values[0].derive = value;
71
72   vl.values = values;
73   vl.values_len = 1;
74   sstrncpy(vl.host, hostname_g, sizeof(vl.host));
75   sstrncpy(vl.plugin, "irq", sizeof(vl.plugin));
76   sstrncpy(vl.type, "irq", sizeof(vl.type));
77   sstrncpy(vl.type_instance, irq_name, sizeof(vl.type_instance));
78
79   plugin_dispatch_values(&vl);
80 } /* void irq_submit */
81
82 static int irq_read(void) {
83   FILE *fh;
84   char buffer[1024];
85   int cpu_count;
86   char *fields[256];
87
88   /*
89    * Example content:
90    *         CPU0       CPU1       CPU2       CPU3
91    * 0:       2574          1          3          2   IO-APIC-edge      timer
92    * 1:     102553     158669     218062      70587   IO-APIC-edge      i8042
93    * 8:          0          0          0          1   IO-APIC-edge      rtc0
94    */
95   fh = fopen("/proc/interrupts", "r");
96   if (fh == NULL) {
97     char errbuf[1024];
98     ERROR("irq plugin: fopen (/proc/interrupts): %s",
99           sstrerror(errno, errbuf, sizeof(errbuf)));
100     return (-1);
101   }
102
103   /* Get CPU count from the first line */
104   if (fgets(buffer, sizeof(buffer), fh) != NULL) {
105     cpu_count = strsplit(buffer, fields, STATIC_ARRAY_SIZE(fields));
106   } else {
107     ERROR("irq plugin: unable to get CPU count from first line "
108           "of /proc/interrupts");
109     fclose(fh);
110     return (-1);
111   }
112
113   while (fgets(buffer, sizeof(buffer), fh) != NULL) {
114     char *irq_name;
115     size_t irq_name_len;
116     derive_t irq_value;
117     int i;
118     int fields_num;
119     int irq_values_to_parse;
120
121     fields_num = strsplit(buffer, fields, STATIC_ARRAY_SIZE(fields));
122     if (fields_num < 2)
123       continue;
124
125     /* Parse this many numeric fields, skip the rest
126      * (+1 because first there is a name of irq in each line) */
127     if (fields_num >= cpu_count + 1)
128       irq_values_to_parse = cpu_count;
129     else
130       irq_values_to_parse = fields_num - 1;
131
132     /* First field is irq name and colon */
133     irq_name = fields[0];
134     irq_name_len = strlen(irq_name);
135     if (irq_name_len < 2)
136       continue;
137
138     /* Check if irq name ends with colon.
139      * Otherwise it's a header. */
140     if (irq_name[irq_name_len - 1] != ':')
141       continue;
142
143     /* Is it the the ARM fast interrupt (FIQ)? */
144     if (irq_name_len == 4 && (strncmp(irq_name, "FIQ:", 4) == 0))
145       continue;
146
147     irq_name[irq_name_len - 1] = 0;
148     irq_name_len--;
149
150     irq_value = 0;
151     for (i = 1; i <= irq_values_to_parse; i++) {
152       /* Per-CPU value */
153       value_t v;
154       int status;
155
156       status = parse_value(fields[i], &v, DS_TYPE_DERIVE);
157       if (status != 0)
158         break;
159
160       irq_value += v.derive;
161     } /* for (i) */
162
163     /* No valid fields -> do not submit anything. */
164     if (i <= 1)
165       continue;
166
167     irq_submit(irq_name, irq_value);
168   }
169
170   fclose(fh);
171
172   return (0);
173 } /* int irq_read */
174
175 void module_register(void) {
176   plugin_register_config("irq", irq_config, config_keys, config_keys_num);
177   plugin_register_read("irq", irq_read);
178 } /* void module_register */