summaryrefslogtreecommitdiff
path: root/proto_nlmsg.c
diff options
context:
space:
mode:
Diffstat (limited to 'proto_nlmsg.c')
-rw-r--r--proto_nlmsg.c233
1 files changed, 233 insertions, 0 deletions
diff --git a/proto_nlmsg.c b/proto_nlmsg.c
index 1b0eed7..89b48b2 100644
--- a/proto_nlmsg.c
+++ b/proto_nlmsg.c
@@ -9,6 +9,9 @@
#include <limits.h>
#include <libgen.h>
#include <netlink/msg.h>
+#include <netlink/route/link.h>
+#include <linux/if_arp.h>
+#include <arpa/inet.h>
#include "pkt_buff.h"
#include "proto.h"
@@ -120,6 +123,115 @@ static const char *nlmsg_rtnl_type2str(uint16_t type)
}
}
+static char *if_type2str(uint16_t type)
+{
+ switch (type) {
+ case ARPHRD_ETHER: return "ether";
+ case ARPHRD_EETHER: return "eether";
+ case ARPHRD_AX25: return "ax25";
+ case ARPHRD_PRONET: return "pronet";
+ case ARPHRD_CHAOS: return "chaos";
+ case ARPHRD_IEEE802: return "ieee802";
+ case ARPHRD_ARCNET: return "arcnet";
+ case ARPHRD_APPLETLK: return "appletlk";
+ case ARPHRD_DLCI: return "dlci";
+ case ARPHRD_ATM: return "atm";
+ case ARPHRD_METRICOM: return "metricom";
+ case ARPHRD_IEEE1394: return "ieee1394";
+ case ARPHRD_INFINIBAND: return "infiniband";
+ case ARPHRD_SLIP: return "slip";
+ case ARPHRD_CSLIP: return "cslip";
+ case ARPHRD_SLIP6: return "slip6";
+ case ARPHRD_CSLIP6: return "cslip6";
+ case ARPHRD_RSRVD: return "RSRVD";
+ case ARPHRD_ADAPT: return "adapt";
+ case ARPHRD_ROSE: return "rose";
+ case ARPHRD_X25: return "x25";
+ case ARPHRD_HWX25: return "hwx25";
+ case ARPHRD_CAN: return "can";
+ case ARPHRD_PPP: return "ppp";
+ case ARPHRD_HDLC: return "hdlc";
+ case ARPHRD_LAPB: return "lapb";
+ case ARPHRD_DDCMP: return "ddcmp";
+ case ARPHRD_RAWHDLC: return "rawhdlc";
+ case ARPHRD_TUNNEL: return "tunnel";
+ case ARPHRD_TUNNEL6: return "tunnel6";
+ case ARPHRD_FRAD: return "frad";
+ case ARPHRD_SKIP: return "skip";
+ case ARPHRD_LOOPBACK: return "loopback";
+ case ARPHRD_LOCALTLK: return "localtlk";
+ case ARPHRD_FDDI: return "fddi";
+ case ARPHRD_BIF: return "bif";
+ case ARPHRD_SIT: return "sit";
+ case ARPHRD_IPDDP: return "ipddp";
+ case ARPHRD_IPGRE: return "ipgre";
+ case ARPHRD_PIMREG: return "pimreg";
+ case ARPHRD_HIPPI: return "hippi";
+ case ARPHRD_ASH: return "ash";
+ case ARPHRD_ECONET: return "econet";
+ case ARPHRD_IRDA: return "irda";
+ case ARPHRD_FCPP: return "fcpp";
+ case ARPHRD_FCAL: return "fcal";
+ case ARPHRD_FCPL: return "fcpl";
+ case ARPHRD_FCFABRIC: return "fcfb0";
+ case ARPHRD_FCFABRIC + 1: return "fcfb1";
+ case ARPHRD_FCFABRIC + 2: return "fcfb2";
+ case ARPHRD_FCFABRIC + 3: return "fcfb3";
+ case ARPHRD_FCFABRIC + 4: return "fcfb4";
+ case ARPHRD_FCFABRIC + 5: return "fcfb5";
+ case ARPHRD_FCFABRIC + 6: return "fcfb6";
+ case ARPHRD_FCFABRIC + 7: return "fcfb7";
+ case ARPHRD_FCFABRIC + 8: return "fcfb8";
+ case ARPHRD_FCFABRIC + 9: return "fcfb9";
+ case ARPHRD_FCFABRIC + 10: return "fcfb10";
+ case ARPHRD_FCFABRIC + 11: return "fcfb11";
+ case ARPHRD_FCFABRIC + 12: return "fcfb12";
+ case ARPHRD_IEEE802_TR: return "ieee802_tr";
+ case ARPHRD_IEEE80211: return "ieee80211";
+ case ARPHRD_IEEE80211_PRISM: return "ieee80211_prism";
+ case ARPHRD_IEEE80211_RADIOTAP: return "ieee80211_radiotap";
+ case ARPHRD_IEEE802154: return "ieee802154";
+ case ARPHRD_PHONET: return "phonet";
+ case ARPHRD_PHONET_PIPE: return "phonet_pipe";
+ case ARPHRD_CAIF: return "caif";
+ case ARPHRD_IP6GRE: return "ip6gre";
+ case ARPHRD_NETLINK: return "netlink";
+ case ARPHRD_NONE: return "none";
+ case ARPHRD_VOID: return "void";
+
+ default: return "Unknown";
+ }
+}
+
+/* Taken from iproute2 */
+static const char *ll_addr_n2a(const unsigned char *addr, int alen, int type,
+ char *buf, int blen)
+{
+ int i;
+ int l;
+
+ if (alen == 4 &&
+ (type == ARPHRD_TUNNEL || type == ARPHRD_SIT || type == ARPHRD_IPGRE)) {
+ return inet_ntop(AF_INET, addr, buf, blen);
+ }
+ if (alen == 16 && type == ARPHRD_TUNNEL6) {
+ return inet_ntop(AF_INET6, addr, buf, blen);
+ }
+ l = 0;
+ for (i=0; i<alen; i++) {
+ if (i==0) {
+ snprintf(buf+l, blen, "%02x", addr[i]);
+ blen -= 2;
+ l += 2;
+ } else {
+ snprintf(buf+l, blen, ":%02x", addr[i]);
+ blen -= 3;
+ l += 3;
+ }
+ }
+ return buf;
+}
+
static char *nlmsg_type2str(uint16_t proto, uint16_t type, char *buf, int len)
{
if (proto == NETLINK_ROUTE && type < RTM_MAX) {
@@ -133,6 +245,124 @@ static char *nlmsg_type2str(uint16_t proto, uint16_t type, char *buf, int len)
return nl_nlmsgtype2str(type, buf, len);
}
+static void rtnl_print_ifinfo(struct nlmsghdr *hdr)
+{
+ struct ifinfomsg *ifi = NLMSG_DATA(hdr);
+ struct rtattr *attr = IFLA_RTA(ifi);
+ uint32_t attrs_len = IFLA_PAYLOAD(hdr);
+ char flags[256];
+ char if_addr[64] = {};
+ char *af_link = "Unknown";
+
+ if (ifi->ifi_family == AF_UNSPEC)
+ af_link = "unspec";
+ else if (ifi->ifi_family == AF_BRIDGE)
+ af_link = "bridge";
+
+ tprintf(" [ Link Family %d (%s%s%s)", ifi->ifi_family,
+ colorize_start(bold), af_link, colorize_end());
+ tprintf(", Type %d (%s%s%s)", ifi->ifi_type,
+ colorize_start(bold),
+ if_type2str(ifi->ifi_type),
+ colorize_end());
+ tprintf(", Index %d", ifi->ifi_index);
+ tprintf(", Flags 0x%x (%s%s%s)", ifi->ifi_flags,
+ colorize_start(bold),
+ rtnl_link_flags2str(ifi->ifi_flags, flags,
+ sizeof(flags)),
+ colorize_end());
+ tprintf(", Change 0x%x (%s%s%s) ]\n", ifi->ifi_change,
+ colorize_start(bold),
+ rtnl_link_flags2str(ifi->ifi_change, flags,
+ sizeof(flags)),
+ colorize_end());
+
+ for (; RTA_OK(attr, attrs_len); attr = RTA_NEXT(attr, attrs_len)) {
+ switch (attr->rta_type) {
+ case IFLA_ADDRESS:
+ tprintf("\tA: Address %s\n",
+ ll_addr_n2a(RTA_DATA(attr),
+ RTA_PAYLOAD(attr),
+ ifi->ifi_type, if_addr,
+ sizeof(if_addr)));
+ break;
+ case IFLA_BROADCAST:
+ tprintf("\tA: Broadcast %s\n",
+ ll_addr_n2a(RTA_DATA(attr),
+ RTA_PAYLOAD(attr),
+ ifi->ifi_type, if_addr,
+ sizeof(if_addr)));
+ break;
+ case IFLA_IFNAME:
+ tprintf("\tA: Name %s%s%s\n",
+ colorize_start(bold),
+ (char *)(RTA_DATA(attr)),
+ colorize_end());
+ break;
+ case IFLA_MTU:
+ tprintf("\tA: MTU %d\n", *(int *)(RTA_DATA(attr)));
+ break;
+ case IFLA_LINK:
+ tprintf("\tA: Link %d\n", *(int *)(RTA_DATA(attr)));
+ break;
+ case IFLA_QDISC:
+ tprintf("\tA: QDisc %s\n", (char *)(RTA_DATA(attr)));
+ break;
+ case IFLA_OPERSTATE:
+ {
+ uint8_t st = *(uint8_t *)(RTA_DATA(attr));
+ char states[256];
+
+ tprintf("\tA: Operation state 0x%x (%s%s%s)\n",
+ st,
+ colorize_start(bold),
+ rtnl_link_operstate2str(st,
+ states, sizeof(states)),
+ colorize_end());
+ }
+ break;
+ case IFLA_LINKMODE:
+ {
+ uint8_t mode = *(uint8_t *)(RTA_DATA(attr));
+ char str[32];
+
+ tprintf("\tA: Mode 0x%x (%s%s%s)\n", mode,
+ colorize_start(bold),
+ rtnl_link_mode2str(mode, str,
+ sizeof(str)),
+ colorize_end());
+ }
+ break;
+ case IFLA_GROUP:
+ tprintf("\tA: Group %d\n", *(int *)(RTA_DATA(attr)));
+ break;
+ case IFLA_TXQLEN:
+ tprintf("\tA: Tx queue len %d\n",
+ *(int *)(RTA_DATA(attr)));
+ break;
+ case IFLA_NET_NS_PID:
+ tprintf("\tA: Network namespace pid %d\n",
+ *(int *)(RTA_DATA(attr)));
+ break;
+ case IFLA_NET_NS_FD:
+ tprintf("\tA: Network namespace fd %d\n",
+ *(int *)(RTA_DATA(attr)));
+ break;
+ }
+ }
+}
+
+static void rtnl_msg_print(struct nlmsghdr *hdr)
+{
+ switch (hdr->nlmsg_type) {
+ case RTM_NEWLINK:
+ case RTM_DELLINK:
+ case RTM_GETLINK:
+ case RTM_SETLINK:
+ rtnl_print_ifinfo(hdr);
+ }
+}
+
static void nlmsg_print(uint16_t family, struct nlmsghdr *hdr)
{
char type[32];
@@ -178,6 +408,9 @@ static void nlmsg_print(uint16_t family, struct nlmsghdr *hdr)
tprintf(" (%s%s%s)", colorize_start(bold), basename(procname),
colorize_end());
tprintf(" ]\n");
+
+ if (family == NETLINK_ROUTE)
+ rtnl_msg_print(hdr);
}
static void nlmsg(struct pkt_buff *pkt)