]> git.sur5r.net Git - i3/i3status/blob - src/print_eth_info.c
af5a75788f7cf17d4d0f3b9154781e83abc0e5d6
[i3/i3status] / src / print_eth_info.c
1 // vim:ts=4:sw=4:expandtab
2 #include <string.h>
3 #include <limits.h>
4 #include <stdio.h>
5 #include <sys/ioctl.h>
6 #include <sys/types.h>
7 #include <sys/socket.h>
8 #include <net/if.h>
9 #include <netinet/in.h>
10 #include <arpa/inet.h>
11 #include <yajl/yajl_gen.h>
12 #include <yajl/yajl_version.h>
13
14 #include "i3status.h"
15
16 #if defined(LINUX)
17 #include <linux/ethtool.h>
18 #include <linux/sockios.h>
19 #define PART_ETHSPEED "E: %s (%d Mbit/s)"
20 #endif
21
22 #if defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || defined(__DragonFly__)
23 #include <net/if_media.h>
24 #define IFM_TYPE_MATCH(dt, t) \
25     (IFM_TYPE((dt)) == 0 || IFM_TYPE((dt)) == IFM_TYPE((t)))
26
27 #define PART_ETHSPEED "E: %s (%s)"
28 #endif
29
30 #if defined(__OpenBSD__) || defined(__NetBSD__)
31 #include <errno.h>
32 #include <net/if_media.h>
33 #endif
34
35 static int print_eth_speed(char *outwalk, const char *interface) {
36 #if defined(LINUX)
37     /* This code path requires root privileges */
38     int ethspeed = 0;
39     struct ifreq ifr;
40     struct ethtool_cmd ecmd;
41
42     ecmd.cmd = ETHTOOL_GSET;
43     (void)memset(&ifr, 0, sizeof(ifr));
44     ifr.ifr_data = (caddr_t)&ecmd;
45     (void)strcpy(ifr.ifr_name, interface);
46     if (ioctl(general_socket, SIOCETHTOOL, &ifr) == 0) {
47         ethspeed = (ecmd.speed == USHRT_MAX ? 0 : ecmd.speed);
48         return sprintf(outwalk, "%d Mbit/s", ethspeed);
49     } else
50         return sprintf(outwalk, "?");
51 #elif defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || defined(__DragonFly__)
52     char *ethspeed;
53     struct ifmediareq ifm;
54     (void)memset(&ifm, 0, sizeof(ifm));
55     (void)strncpy(ifm.ifm_name, interface, sizeof(ifm.ifm_name));
56     int ret = ioctl(general_socket, SIOCGIFMEDIA, (caddr_t)&ifm);
57
58     /* Get the description of the media type, partially taken from
59      * FreeBSD's ifconfig */
60     const struct ifmedia_description *desc;
61     struct ifmedia_description ifm_subtype_descriptions[] =
62         IFM_SUBTYPE_ETHERNET_DESCRIPTIONS;
63
64     for (desc = ifm_subtype_descriptions;
65          desc->ifmt_string != NULL;
66          desc++) {
67         if (IFM_TYPE_MATCH(desc->ifmt_word, ifm.ifm_active) &&
68             IFM_SUBTYPE(desc->ifmt_word) == IFM_SUBTYPE(ifm.ifm_active))
69             break;
70     }
71     ethspeed = (desc->ifmt_string != NULL ? desc->ifmt_string : "?");
72     return sprintf(outwalk, "%s", ethspeed);
73 #elif defined(__OpenBSD__) || defined(__NetBSD__)
74     char *ethspeed;
75     struct ifmediareq ifmr;
76
77     (void)memset(&ifmr, 0, sizeof(ifmr));
78     (void)strlcpy(ifmr.ifm_name, interface, sizeof(ifmr.ifm_name));
79
80     if (ioctl(general_socket, SIOCGIFMEDIA, (caddr_t)&ifmr) < 0) {
81         if (errno != E2BIG)
82             return sprintf(outwalk, "?");
83     }
84
85     struct ifmedia_description *desc;
86     struct ifmedia_description ifm_subtype_descriptions[] =
87         IFM_SUBTYPE_DESCRIPTIONS;
88
89     for (desc = ifm_subtype_descriptions; desc->ifmt_string != NULL; desc++) {
90         /*
91                  * Skip these non-informative values and go right ahead to the
92                  * actual speeds.
93                  */
94         if (BEGINS_WITH(desc->ifmt_string, "autoselect") ||
95             BEGINS_WITH(desc->ifmt_string, "auto"))
96             continue;
97
98         if (IFM_TYPE_MATCH(desc->ifmt_word, ifmr.ifm_active) &&
99             IFM_SUBTYPE(desc->ifmt_word) == IFM_SUBTYPE(ifmr.ifm_active))
100             break;
101     }
102     ethspeed = (desc->ifmt_string != NULL ? desc->ifmt_string : "?");
103     return sprintf(outwalk, "%s", ethspeed);
104
105 #else
106     return sprintf(outwalk, "?");
107 #endif
108 }
109
110 /*
111  * Combines ethernet IP addresses and speed (if requested) for displaying
112  *
113  */
114 void print_eth_info(yajl_gen json_gen, char *buffer, const char *interface, const char *format_up, const char *format_down) {
115     const char *walk;
116     const char *ip_address = get_ip_addr(interface);
117     char *outwalk = buffer;
118
119     INSTANCE(interface);
120
121     if (ip_address == NULL) {
122         START_COLOR("color_bad");
123         outwalk += sprintf(outwalk, "%s", format_down);
124         goto out;
125     }
126
127     START_COLOR("color_good");
128
129     for (walk = format_up; *walk != '\0'; walk++) {
130         if (*walk != '%') {
131             *(outwalk++) = *walk;
132             continue;
133         }
134
135         if (BEGINS_WITH(walk + 1, "ip")) {
136             outwalk += sprintf(outwalk, "%s", ip_address);
137             walk += strlen("ip");
138         } else if (BEGINS_WITH(walk + 1, "speed")) {
139             outwalk += print_eth_speed(outwalk, interface);
140             walk += strlen("speed");
141         }
142     }
143
144 out:
145     END_COLOR;
146     OUTPUT_FULL_TEXT(buffer);
147 }