summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/get_eth_info.c74
1 files changed, 63 insertions, 11 deletions
diff --git a/src/get_eth_info.c b/src/get_eth_info.c
index 5e9ab54..9172abb 100644
--- a/src/get_eth_info.c
+++ b/src/get_eth_info.c
@@ -4,14 +4,41 @@
#include <stdio.h>
#include <sys/ioctl.h>
#include <sys/types.h>
+#include <sys/socket.h>
#include <net/if.h>
+#include <netinet/in.h>
+#include <arpa/inet.h>
-#ifdef LINUX
+#include "i3status.h"
+
+#if defined(LINUX)
#include <linux/ethtool.h>
#include <linux/sockios.h>
+#define PART_ETHSPEED "E: %s (%d Mbit/s)"
#endif
-#include "i3status.h"
+#if defined(__FreeBSD__)
+#include <net/if_media.h>
+#define IFM_TYPE_MATCH(dt, t) \
+ (IFM_TYPE((dt)) == 0 || IFM_TYPE((dt)) == IFM_TYPE((t)))
+
+#define PART_ETHSPEED "E: %s (%s)"
+
+struct ifmedia_type_to_subtype {
+ struct {
+ struct ifmedia_description *desc;
+ int alias;
+ } subtypes[5];
+ struct {
+ struct ifmedia_description *desc;
+ int alias;
+ } options[3];
+ struct {
+ struct ifmedia_description *desc;
+ int alias;
+ } modes[3];
+};
+#endif
/*
* Combines ethernet IP addresses and speed (if requested) for displaying
@@ -19,11 +46,20 @@
*/
const char *get_eth_info() {
static char part[512];
+#if defined(LINUX)
+ int ethspeed=0;
+#elif defined(__FreeBSD__)
+ char *ethspeed;
+#endif
const char *ip_address = get_ip_addr(eth_interface);
- int ethspeed = 0;
+
+ if (ip_address == NULL) {
+ (void)snprintf(part, sizeof(part), "E: down");
+ return part;
+ }
if (get_ethspeed) {
-#ifdef LINUX
+#if defined(LINUX)
/* This code path requires root privileges */
struct ifreq ifr;
struct ethtool_cmd ecmd;
@@ -35,16 +71,32 @@ const char *get_eth_info() {
if (ioctl(general_socket, SIOCETHTOOL, &ifr) == 0)
ethspeed = (ecmd.speed == USHRT_MAX ? 0 : ecmd.speed);
else get_ethspeed = false;
+#elif defined(__FreeBSD__)
+ struct ifmediareq ifm;
+ (void)memset(&ifm, 0, sizeof(ifm));
+ (void)strncpy(ifm.ifm_name, ifaddr->ifa_name, sizeof(ifm.ifm_name));
+ int ret = ioctl(general_socket, SIOCGIFMEDIA, (caddr_t)&ifm);
+
+ /* Get the description of the media type, partially taken from
+ * FreeBSD's ifconfig */
+ const struct ifmedia_description *desc;
+ struct ifmedia_description ifm_subtype_descriptions[] =
+ IFM_SUBTYPE_ETHERNET_DESCRIPTIONS;
+
+ for (desc = ifm_subtype_descriptions;
+ desc->ifmt_string != NULL;
+ desc++) {
+ if (IFM_TYPE_MATCH(desc->ifmt_word, ifm.ifm_active) &&
+ IFM_SUBTYPE(desc->ifmt_word) == IFM_SUBTYPE(ifm.ifm_active))
+ break;
+ }
+ ethspeed = (desc->ifmt_string != NULL ? desc->ifmt_string : "?");
#endif
}
- if (ip_address == NULL)
- (void)snprintf(part, sizeof(part), "E: down");
- else {
- if (get_ethspeed)
- (void)snprintf(part, sizeof(part), "E: %s (%d Mbit/s)", ip_address, ethspeed);
- else (void)snprintf(part, sizeof(part), "E: %s", ip_address);
- }
+ if (get_ethspeed)
+ (void)snprintf(part, sizeof(part), PART_ETHSPEED, ip_address, ethspeed);
+ else (void)snprintf(part, sizeof(part), "E: %s", ip_address);
return part;
}