Fix ifaddrs#getifaddrs_INET flakiness.

The interface name wasn't being nul-terminated for the ioctl. Also clean up
the code a bit to give more useful diagnostics on failure.

Bug: http://b/26887941
Change-Id: I30c6bdc1a32733971a27ed1fb7db9d8239b6262b
This commit is contained in:
Elliott Hughes 2016-02-03 07:42:33 -08:00
parent 3e75110bcb
commit 2d5e21f00d
1 changed files with 39 additions and 39 deletions

View File

@ -109,55 +109,55 @@ TEST(ifaddrs, getifaddrs_interfaces) {
} }
TEST(ifaddrs, getifaddrs_INET) { TEST(ifaddrs, getifaddrs_INET) {
std::multimap<std::string,in_addr_t> inetaddrs; std::multimap<std::string, in_addr_t> inetaddrs;
std::multimap<std::string,in_addr_t> broadinetaddrs; std::multimap<std::string, in_addr_t> broadinetaddrs;
{ // Collect the IPv4 addresses for each interface.
ifaddrs* addrs; ifaddrs* addrs;
ASSERT_EQ(0, getifaddrs(&addrs)); ASSERT_EQ(0, getifaddrs(&addrs));
for (ifaddrs* addr = addrs; addr != nullptr; addr = addr->ifa_next) { for (ifaddrs* addr = addrs; addr != nullptr; addr = addr->ifa_next) {
if (addr->ifa_name && addr->ifa_addr && addr->ifa_addr->sa_family == AF_INET) { if (addr->ifa_name && addr->ifa_addr && addr->ifa_addr->sa_family == AF_INET) {
auto sock = reinterpret_cast<sockaddr_in*>(addr->ifa_addr); auto sock = reinterpret_cast<sockaddr_in*>(addr->ifa_addr);
inetaddrs.emplace(std::string(addr->ifa_name), sock->sin_addr.s_addr); inetaddrs.emplace(std::string(addr->ifa_name), sock->sin_addr.s_addr);
} }
if (addr->ifa_name && addr->ifa_broadaddr && addr->ifa_broadaddr->sa_family == AF_INET) { if (addr->ifa_name && addr->ifa_broadaddr && addr->ifa_broadaddr->sa_family == AF_INET) {
auto sock = reinterpret_cast<sockaddr_in*>(addr->ifa_broadaddr); auto sock = reinterpret_cast<sockaddr_in*>(addr->ifa_broadaddr);
broadinetaddrs.emplace(std::string(addr->ifa_name), sock->sin_addr.s_addr); broadinetaddrs.emplace(std::string(addr->ifa_name), sock->sin_addr.s_addr);
}
} }
freeifaddrs(addrs);
} }
freeifaddrs(addrs);
{ // Check that the addresses returned by the SIOCGIFADDR and SIOCGIFBRDADDR ioctls
int fd = socket(AF_INET, SOCK_DGRAM, 0); // are in our collections.
ASSERT_TRUE(fd != -1); auto check_inet_agrees = [&](std::multimap<std::string, in_addr_t> addrs, int request)->void {
for (auto it = addrs.begin(); it != addrs.end(); ) {
std::string if_name(it->first);
auto check_inet_agrees = [&](std::multimap<std::string, in_addr_t> addrs, int request)->bool { ifreq ifr;
for (auto it = addrs.begin(); it != addrs.end(); ) { memset(&ifr, 0, sizeof(ifr));
ifreq ifr; ifr.ifr_addr.sa_family = AF_INET;
ifr.ifr_addr.sa_family = AF_INET; if_name.copy(ifr.ifr_name, IFNAMSIZ - 1);
it->first.copy(ifr.ifr_name, IFNAMSIZ - 1);
ioctl(fd, request, &ifr);
sockaddr_in* sock = reinterpret_cast<sockaddr_in*>(&ifr.ifr_addr); int fd = socket(AF_INET, SOCK_DGRAM, 0);
in_addr_t addr = sock->sin_addr.s_addr; ASSERT_TRUE(fd != -1);
ASSERT_EQ(0, ioctl(fd, request, &ifr)) << if_name << ' ' << strerror(errno);
close(fd);
bool found = false; sockaddr_in* sock = reinterpret_cast<sockaddr_in*>(&ifr.ifr_addr);
for (auto ub = addrs.upper_bound(it->first); it != ub; ++it) { in_addr_t addr = sock->sin_addr.s_addr;
if (it->second == addr) {
found = true; bool found = false;
} for (auto ub = addrs.upper_bound(it->first); it != ub; ++it) {
if (it->second == addr) {
found = true;
} }
if (!found) return false;
} }
return true; EXPECT_TRUE(found) << if_name;
}; }
};
ASSERT_TRUE(check_inet_agrees(inetaddrs, SIOCGIFADDR)); check_inet_agrees(inetaddrs, SIOCGIFADDR);
ASSERT_TRUE(check_inet_agrees(broadinetaddrs, SIOCGIFBRDADDR)); check_inet_agrees(broadinetaddrs, SIOCGIFBRDADDR);
close(fd);
}
} }
static void print_sockaddr_ll(const char* what, const sockaddr* p) { static void print_sockaddr_ll(const char* what, const sockaddr* p) {