Merge pull request #3329 from efuss/fix-3311
[collectd.git] / src / serial.c
1 /**
2  * collectd - src/serial.c
3  * Copyright (C) 2005,2006  David Bacher
4  *
5  * This program is free software; you can redistribute it and/or modify it
6  * under the terms of the GNU General Public License as published by the
7  * Free Software Foundation; either version 2 of the License, or (at your
8  * option) any later version.
9  *
10  * This program is distributed in the hope that it will be useful, but
11  * WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13  * General Public License for more details.
14  *
15  * You should have received a copy of the GNU General Public License along
16  * with this program; if not, write to the Free Software Foundation, Inc.,
17  * 51 Franklin St, Fifth Floor, Boston, MA  02110-1301 USA
18  *
19  * Authors:
20  *   David Bacher <drbacher at gmail.com>
21  *   Florian octo Forster <octo at collectd.org>
22  **/
23
24 #include "collectd.h"
25
26 #include "plugin.h"
27 #include "utils/common/common.h"
28
29 #if !KERNEL_LINUX
30 #error "No applicable input method."
31 #endif
32
33 static void serial_submit(const char *type_instance, derive_t rx, derive_t tx) {
34   value_list_t vl = VALUE_LIST_INIT;
35   value_t values[] = {
36       {.derive = rx},
37       {.derive = tx},
38   };
39
40   vl.values = values;
41   vl.values_len = STATIC_ARRAY_SIZE(values);
42   sstrncpy(vl.plugin, "serial", sizeof(vl.plugin));
43   sstrncpy(vl.type, "serial_octets", sizeof(vl.type));
44   sstrncpy(vl.type_instance, type_instance, sizeof(vl.type_instance));
45
46   plugin_dispatch_values(&vl);
47 }
48
49 static int serial_read(void) {
50   FILE *fh;
51   char buffer[1024];
52
53   /* there are a variety of names for the serial device */
54   if ((fh = fopen("/proc/tty/driver/serial", "r")) == NULL &&
55       (fh = fopen("/proc/tty/driver/ttyS", "r")) == NULL) {
56     WARNING("serial: fopen: %s", STRERRNO);
57     return -1;
58   }
59
60   while (fgets(buffer, sizeof(buffer), fh) != NULL) {
61     derive_t rx = 0;
62     derive_t tx = 0;
63     bool have_rx = false;
64     bool have_tx = false;
65     size_t len;
66
67     char *fields[16];
68     int numfields;
69
70     numfields = strsplit(buffer, fields, STATIC_ARRAY_SIZE(fields));
71     if (numfields < 6)
72       continue;
73
74     /*
75      * 0: uart:16550A port:000003F8 irq:4 tx:0 rx:0
76      * 1: uart:16550A port:000002F8 irq:3 tx:0 rx:0
77      */
78     len = strlen(fields[0]);
79     if (len < 2)
80       continue;
81     if (fields[0][len - 1] != ':')
82       continue;
83     fields[0][len - 1] = '\0';
84
85     for (int i = 1; i < numfields; i++) {
86       len = strlen(fields[i]);
87       if (len < 4)
88         continue;
89
90       if (strncmp(fields[i], "tx:", 3) == 0) {
91         if (strtoderive(fields[i] + 3, &tx) == 0)
92           have_tx = true;
93       } else if (strncmp(fields[i], "rx:", 3) == 0) {
94         if (strtoderive(fields[i] + 3, &rx) == 0)
95           have_rx = true;
96       }
97     }
98
99     if (have_rx && have_tx)
100       serial_submit(fields[0], rx, tx);
101   }
102
103   fclose(fh);
104   return 0;
105 } /* int serial_read */
106
107 void module_register(void) {
108   plugin_register_read("serial", serial_read);
109 } /* void module_register */