/* * virarptable.c Linux ARP table handling * * Copyright (C) 2018 Chen Hanxiao * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library. If not, see * . */ #include #include #ifdef __linux__ # include #endif #include "viralloc.h" #include "virarptable.h" #include "virfile.h" #include "virlog.h" #include "virnetlink.h" #include "virsocketaddr.h" #include "virstring.h" #define VIR_FROM_THIS VIR_FROM_NONE VIR_LOG_INIT("util.arptable"); #ifdef __linux__ # define NDA_RTA(r) \ ((struct rtattr*)(((char*)(r)) + NLMSG_ALIGN(sizeof(struct ndmsg)))) static int parse_rtattr(struct rtattr *tb[], int max, struct rtattr *rta, int len) { memset(tb, 0, sizeof(struct rtattr *) * (max + 1)); VIR_WARNINGS_NO_CAST_ALIGN for (; RTA_OK(rta, len); rta = RTA_NEXT(rta, len)) { VIR_WARNINGS_RESET if ((rta->rta_type <= max) && (!tb[rta->rta_type])) tb[rta->rta_type] = rta; } if (len) VIR_WARN("malformed netlink message: Deficit %d, rta_len=%d", len, rta->rta_len); return 0; } virArpTablePtr virArpTableGet(void) { int num = 0; int msglen; g_autofree void *nlData = NULL; virArpTablePtr table = NULL; struct nlmsghdr* nh; struct rtattr * tb[NDA_MAX+1]; msglen = virNetlinkGetNeighbor(&nlData, 0, 0); if (msglen < 0) return NULL; if (VIR_ALLOC(table) < 0) return NULL; nh = (struct nlmsghdr*)nlData; VIR_WARNINGS_NO_CAST_ALIGN for (; NLMSG_OK(nh, msglen); nh = NLMSG_NEXT(nh, msglen)) { VIR_WARNINGS_RESET struct ndmsg *r = NLMSG_DATA(nh); int len = nh->nlmsg_len; void *addr; if ((len -= NLMSG_LENGTH(sizeof(*nh))) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("wrong nlmsg len")); goto cleanup; } if (r->ndm_family && (r->ndm_family != AF_INET)) continue; /* catch stale and reachalbe arp entry only */ if (r->ndm_state && (!(r->ndm_state == NUD_STALE || r->ndm_state == NUD_REACHABLE))) continue; if (nh->nlmsg_type == NLMSG_DONE) return table; VIR_WARNINGS_NO_CAST_ALIGN parse_rtattr(tb, NDA_MAX, NDA_RTA(r), nh->nlmsg_len - NLMSG_LENGTH(sizeof(*r))); VIR_WARNINGS_RESET if (tb[NDA_DST] == NULL || tb[NDA_LLADDR] == NULL) continue; if (tb[NDA_DST]) { g_autofree char *ipstr = NULL; virSocketAddr virAddr; if (VIR_REALLOC_N(table->t, num + 1) < 0) goto cleanup; table->n = num + 1; addr = RTA_DATA(tb[NDA_DST]); bzero(&virAddr, sizeof(virAddr)); virAddr.len = sizeof(virAddr.data.inet4); virAddr.data.inet4.sin_family = AF_INET; virAddr.data.inet4.sin_addr = *(struct in_addr *)addr; ipstr = virSocketAddrFormat(&virAddr); if (VIR_STRDUP(table->t[num].ipaddr, ipstr) < 0) goto cleanup; } if (tb[NDA_LLADDR]) { virMacAddr macaddr; char ifmac[VIR_MAC_STRING_BUFLEN]; addr = RTA_DATA(tb[NDA_LLADDR]); memcpy(macaddr.addr, addr, VIR_MAC_BUFLEN); virMacAddrFormat(&macaddr, ifmac); if (VIR_STRDUP(table->t[num].mac, ifmac) < 0) goto cleanup; num++; } } return table; cleanup: virArpTableFree(table); return NULL; } #else virArpTablePtr virArpTableGet(void) { virReportError(VIR_ERR_NO_SUPPORT, "%s", _("get arp table not implemented on this platform")); return NULL; } #endif /* __linux__ */ void virArpTableFree(virArpTablePtr table) { size_t i; if (!table) return; for (i = 0; i < table->n; i++) { VIR_FREE(table->t[i].ipaddr); VIR_FREE(table->t[i].mac); } VIR_FREE(table->t); VIR_FREE(table); }