/* * Copyright (c) 2017 Petr Vorel <pvorel@suse.cz> * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 2 of the License, or * (at your option) any later version. * * This program 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. */ #include <sys/socket.h> #include <linux/rtnetlink.h> #include <net/if.h> #include <stdio.h> #include <stdlib.h> #define TST_NO_DEFAULT_MAIN #include "tst_test.h" #include "tst_net.h" static char *iface; static int prefix; static void usage(const char *cmd) { fprintf(stderr, "USAGE:\n" "%s IP_LHOST[/PREFIX]\n" "%s -r IP_RHOST[/PREFIX]\n" "%s -h\n\n" "Set prefix and interface name for given IP.\n" "Prefix and interface are found from kernel exported info (rtnetlink).\n\n" "EXPORTED VARIABLES:\n" "Export one of the following variables:\n" "IPV4_LPREFIX: IPv4 prefix for IPV4_LNETWORK\n" "IPV4_RPREFIX: IPv4 prefix for IPV4_RNETWORK\n" "IPV6_LPREFIX: IPv6 prefix for IPV6_LNETWORK\n" "IPV6_RPREFIX: IPv6 prefix for IPV6_RNETWORK\n" "Export one of the following variables (if found):\n" "LHOST_IFACES: iface name of the local host\n" "RHOST_IFACES: iface name of the remote host\n\n" "PARAMS:\n" "-h this help\n" "-r export remote environment variables\n", cmd, cmd, cmd); } static int read_iface_prefix(const char *ip_str, int is_ipv6) { uint8_t family = is_ipv6 ? AF_INET6 : AF_INET; char buf[16384]; unsigned int len; struct { struct nlmsghdr nlhdr; struct ifaddrmsg addrmsg; } msg; struct nlmsghdr *retmsg; int sock = SAFE_SOCKET(AF_NETLINK, SOCK_RAW, NETLINK_ROUTE); memset(&msg, 0, sizeof(msg)); msg.nlhdr.nlmsg_len = NLMSG_LENGTH(sizeof(struct ifaddrmsg)); msg.nlhdr.nlmsg_flags = NLM_F_REQUEST | NLM_F_ROOT; msg.nlhdr.nlmsg_type = RTM_GETADDR; msg.addrmsg.ifa_family = family; SAFE_SEND(1, sock, &msg, msg.nlhdr.nlmsg_len, 0); len = recv(sock, buf, sizeof(buf), 0); retmsg = (struct nlmsghdr *)buf; while NLMSG_OK(retmsg, len) { char ifname[IFNAMSIZ]; struct ifaddrmsg *retaddr; struct rtattr *retrta; char pradd[128]; int attlen; retaddr = (struct ifaddrmsg *)NLMSG_DATA(retmsg); retrta = (struct rtattr *)IFA_RTA(retaddr); attlen = IFA_PAYLOAD(retmsg); while RTA_OK(retrta, attlen) { if (retrta->rta_type == IFA_ADDRESS) { inet_ntop(family, RTA_DATA(retrta), pradd, sizeof(pradd)); if_indextoname(retaddr->ifa_index, ifname); if (!strcmp(pradd, ip_str)) { prefix = retaddr->ifa_prefixlen; iface = strdup(ifname); return 0; } } retrta = RTA_NEXT(retrta, attlen); } retmsg = NLMSG_NEXT(retmsg, len); } return -1; } static void print_ivar(const char *name, unsigned int val) { printf("export %s=%d\n", name, val); } int main(int argc, char *argv[]) { char *ip_str = NULL, *prefix_str = NULL; int is_ipv6, is_rhost = 0; struct in_addr ip; struct in6_addr ip6; int is_usage = argc > 1 && (!strcmp(argv[1], "-h") || !strcmp(argv[1], "--help")); if (argc < 2 || is_usage) { usage(argv[0]); exit(is_usage ? EXIT_SUCCESS : EXIT_FAILURE); } if (!strcmp(argv[1], "-r")) is_rhost = 1; ip_str = argv[is_rhost ? 2 : 1]; is_ipv6 = !!strchr(ip_str, ':'); prefix_str = strchr(ip_str, '/'); if (prefix_str) { prefix = get_prefix(ip_str, is_ipv6); tst_res_comment(TINFO, "IP address '%s' contains prefix %d, using it and don't search for iface.\n", ip_str, prefix); } else if (read_iface_prefix(ip_str, is_ipv6)) { tst_res_comment(TINFO, "prefix and interface not found for '%s'.\n", ip_str); exit(EXIT_SUCCESS); } /* checks for validity of IP string */ if (is_ipv6) get_in6_addr(ip_str, &ip6); else get_in_addr(ip_str, &ip); if (is_ipv6) { print_svar_change("LHOST_IFACES", iface); print_ivar(is_rhost ? "IPV6_RPREFIX" : "IPV6_LPREFIX", prefix); } else { print_svar_change("RHOST_IFACES", iface); print_ivar(is_rhost ? "IPV4_RPREFIX" : "IPV4_LPREFIX", prefix); } exit(EXIT_SUCCESS); }