summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--include/i3status.h3
-rw-r--r--src/first_network_device.c57
2 files changed, 53 insertions, 7 deletions
diff --git a/include/i3status.h b/include/i3status.h
index b4cf83f..4d2d0f1 100644
--- a/include/i3status.h
+++ b/include/i3status.h
@@ -202,7 +202,8 @@ void set_timezone(const char *tz);
/* src/first_network_device.c */
typedef enum {
NET_TYPE_WIRELESS = 0,
- NET_TYPE_ETHERNET = 1
+ NET_TYPE_ETHERNET = 1,
+ NET_TYPE_OTHER = 2
} net_type_t;
const char *first_eth_interface(const net_type_t type);
diff --git a/src/first_network_device.c b/src/first_network_device.c
index 06d8b4a..541aabb 100644
--- a/src/first_network_device.c
+++ b/src/first_network_device.c
@@ -6,11 +6,58 @@
#include "i3status.h"
+static bool sysfs_devtype(char *dest, size_t n, const char *ifnam) {
+ FILE *fp;
+ char buf[1024];
+
+ snprintf(buf, sizeof(buf), "/sys/class/net/%s/uevent", ifnam);
+ if ((fp = fopen(buf, "r")) == NULL)
+ return false;
+
+ dest[0] = '\0';
+
+ while (fgets(buf, sizeof(buf), fp)) {
+ size_t slen;
+ char *s;
+
+ slen = strlen(buf);
+ /* Line is too long to fit in the buffer */
+ if (buf[slen - 1] != '\n' && !feof(fp))
+ break;
+ if ((s = strchr(buf, '='))) {
+ if (strncmp(buf, "DEVTYPE", s - buf))
+ continue;
+ buf[slen - 1] = '\0';
+ strncpy(dest, ++s, n);
+ break;
+ }
+ }
+ fclose(fp);
+ return true;
+}
+
+static net_type_t iface_type(const char *ifname) {
+ char devtype[32];
+
+ if (!sysfs_devtype(devtype, sizeof(devtype), ifname))
+ return NET_TYPE_OTHER;
+
+ if (!devtype[0])
+ return NET_TYPE_ETHERNET;
+
+ if (strcmp(devtype, "wlan") == 0)
+ return NET_TYPE_WIRELESS;
+
+ if (strcmp(devtype, "wwan") == 0)
+ return NET_TYPE_OTHER;
+
+ return NET_TYPE_OTHER;
+}
+
const char *first_eth_interface(const net_type_t type) {
static char *interface = NULL;
struct ifaddrs *ifaddr, *addrp;
- struct stat stbuf;
- static char path[1024];
+ net_type_t iftype;
getifaddrs(&ifaddr);
@@ -32,10 +79,8 @@ const char *first_eth_interface(const net_type_t type) {
addrp->ifa_addr->sa_family != AF_INET6)
continue;
// Skip this interface if it is a wireless interface.
- snprintf(path, sizeof(path), "/sys/class/net/%s/wireless", addrp->ifa_name);
- const bool is_wireless = (stat(path, &stbuf) == 0);
- if ((is_wireless && type == NET_TYPE_ETHERNET) ||
- (!is_wireless && type == NET_TYPE_WIRELESS))
+ iftype = iface_type(addrp->ifa_name);
+ if (iftype != type)
continue;
interface = strdup(addrp->ifa_name);
break;