/* * Derived from https://github.com/ajrisi/lsif/blob/master/lsif.c * No copyright information in the file, and published publicly, * so presume no rights reserved. * * This method doesn't work properly on OSX, but is for Android where no other * option seems to work. Should work on any linux system. * ******************************** * * Updated code to obtain IP and MAC address for all "up" network * interfaces on a linux system. Now IPv6 friendly and updated to use * inet_ntop instead of the deprecated inet_ntoa function. This version * should not seg fault on newer linux systems * * Version 2.0 * * Authors: * Adam Pierce * Adam Risi * William Schaub * * Date: 11/11/2009 * http://www.adamrisi.com * http://www.doctort.org/adam/ * http://teotwawki.steubentech.com/ * */ #include "serval.h" #include "conf.h" #include #include #include #include #include #include #include #include #include #include #if __MACH__ || __NetBSD__ || __OpenBSD__ || __FreeBSD__ #include #endif /* Include sockio.h if needed */ #ifndef SIOCGIFCONF #include #endif #include #if __MACH__ #include #endif #ifdef HAVE_LINUX_IF_H #include #else #if HAVE_NET_IF_H || __MACH__ || __NetBSD__ || __OpenBSD__ || __FreeBSD__ #include #endif #endif #ifdef HAVE_IFADDRS_H #include #endif /* On platforms that have variable length ifreq use the old fixed length interface instead */ #ifdef OSIOCGIFCONF #undef SIOCGIFCONF #define SIOCGIFCONF OSIOCGIFCONF #undef SIOCGIFADDR #define SIOCGIFADDR OSIOCGIFADDR #undef SIOCGIFBRDADDR #define SIOCGIFBRDADDR OSIOCGIFBRDADDR #endif #ifdef linux /* for when all other options fail, as can happen on Android, if the permissions for the socket-based method are broken. Down side is that it while it gets the interface name and broadcast, it doesn't get the local address for that interface. */ int scrapeProcNetRoute() { if (config.debug.overlayinterfaces) DEBUG("called"); FILE *f=fopen("/proc/net/route","r"); if (!f) return WHY_perror("fopen(\"/proc/net/route\")"); char line[1024],name[1024],dest[1024],mask[1024]; /* skip header line */ line[0] = '\0'; if (fgets(line,1024,f) == NULL) return WHYF_perror("fgets(%p,1024,\"/proc/net/route\")", line); line[0] = '\0'; if (fgets(line,1024,f) == NULL) return WHYF_perror("fgets(%p,1024,\"/proc/net/route\")", line); while(line[0]) { int r; if ((r=sscanf(line,"%s %s %*s %*s %*s %*s %*s %s",name,dest,mask))==3) { struct in_addr addr = {.s_addr=strtol(dest,NULL,16)}; struct in_addr netmask = {.s_addr=strtol(mask,NULL,16)}; overlay_interface_register(name,addr,netmask); } line[0] = '\0'; if (fgets(line,1024,f) == NULL) return WHYF_perror("fgets(%p,1024,\"/proc/net/route\")", line); } fclose(f); return 0; } #endif #ifdef SIOCGIFCONF /* Not present in Linux */ #ifndef _SIZEOF_ADDR_IFREQ #define _SIZEOF_ADDR_IFREQ(x) sizeof(struct ifreq) #endif int lsif(void) { char buf[8192]; struct ifconf ifc; int sck, nInterfaces, ofs; struct ifreq *ifr; struct in_addr addr, netmask; if (config.debug.overlayinterfaces) DEBUG("called"); /* Get a socket handle. */ sck = socket(PF_INET, SOCK_DGRAM, 0); if(sck < 0) { WHY_perror("socket"); return 1; } /* Query available interfaces. */ ifc.ifc_len = sizeof(buf); ifc.ifc_buf = buf; if(ioctl(sck, SIOCGIFCONF, &ifc) < 0) { WHY_perror("ioctl(SIOCGIFCONF)"); close(sck); return 1; } /* Iterate through the list of interfaces. */ nInterfaces = 0; ofs = 0; while (ofs < ifc.ifc_len && ofs < sizeof(buf)) { ifr = (struct ifreq *)(ifc.ifc_ifcu.ifcu_buf + ofs); ofs += _SIZEOF_ADDR_IFREQ(*ifr); /* We're only interested in IPv4 addresses */ if (ifr->ifr_ifru.ifru_addr.sa_family != AF_INET) { if (config.debug.overlayinterfaces) DEBUGF("Skipping non-AF_INET address on %s", ifr->ifr_name); continue; } addr = ((struct sockaddr_in *)&ifr->ifr_ifru.ifru_addr)->sin_addr; /* Get interface flags */ if (ioctl(sck, SIOCGIFFLAGS, ifr) == -1) FATAL_perror("ioctl(SIOCGIFFLAGS)"); /* Not broadcast? Not interested.. */ if ((ifr->ifr_ifru.ifru_flags & IFF_BROADCAST) == 0) { if (config.debug.overlayinterfaces) DEBUGF("Skipping non-broadcast address on %s", ifr->ifr_name); continue; } /* Get netmask */ if (ioctl(sck, SIOCGIFNETMASK, ifr, sizeof(*ifr)) != 0) { WHY_perror("ioctl(SIOCGIFNETMASK)"); continue; } netmask = ((struct sockaddr_in *)&ifr->ifr_ifru.ifru_addr)->sin_addr; overlay_interface_register(ifr->ifr_name, addr, netmask); nInterfaces++; } if (config.debug.overlayinterfaces) DEBUGF("Examined %d interface addresses", nInterfaces); close(sck); return 0; } #endif #ifdef HAVE_IFADDRS_H int doifaddrs(void) { struct ifaddrs *ifaddr, *ifa; char *name; struct in_addr addr, netmask; if (config.debug.overlayinterfaces) DEBUGF("called"); if (getifaddrs(&ifaddr) == -1) return WHY_perror("getifaddr()"); for (ifa = ifaddr; ifa != NULL ; ifa = ifa->ifa_next) { if (!ifa->ifa_addr || !ifa->ifa_netmask) continue; /* We're only interested in IPv4 addresses */ if (ifa->ifa_addr->sa_family != AF_INET) { if (config.debug.overlayinterfaces) DEBUGF("Skipping non-AF_INET address on %s", ifa->ifa_name); continue; } /* Not broadcast? Not interested.. */ if ((ifa->ifa_flags & IFF_BROADCAST) == 0) { if (config.debug.overlayinterfaces) DEBUGF("Skipping non-broadcast address on %s", ifa->ifa_name); continue; } name = ifa->ifa_name; addr = ((struct sockaddr_in *)ifa->ifa_addr)->sin_addr; netmask = ((struct sockaddr_in *)ifa->ifa_netmask)->sin_addr; overlay_interface_register(name, addr, netmask); } freeifaddrs(ifaddr); return 0; } #endif