/* * Copyright (C) 2013 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include <netdb.h> #include <gtest/gtest.h> #include <arpa/inet.h> #include <string.h> #include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> // https://code.google.com/p/android/issues/detail?id=13228 TEST(netdb, freeaddrinfo_NULL) { freeaddrinfo(NULL); } TEST(netdb, getaddrinfo_NULL_host) { // It's okay for the host argument to be NULL, as long as service isn't. addrinfo* ai = NULL; ASSERT_EQ(0, getaddrinfo(NULL, "smtp", NULL, &ai)); // (sockaddr_in::sin_port and sockaddr_in6::sin6_port overlap.) ASSERT_EQ(25U, ntohs(reinterpret_cast<sockaddr_in*>(ai->ai_addr)->sin_port)); freeaddrinfo(ai); } TEST(netdb, getaddrinfo_NULL_service) { // It's okay for the service argument to be NULL, as long as host isn't. addrinfo* ai = NULL; ASSERT_EQ(0, getaddrinfo("localhost", NULL, NULL, &ai)); ASSERT_TRUE(ai != NULL); freeaddrinfo(ai); } TEST(netdb, getaddrinfo_NULL_hints) { addrinfo* ai = NULL; ASSERT_EQ(0, getaddrinfo("localhost", "9999", NULL, &ai)); bool saw_tcp = false; bool saw_udp = false; for (addrinfo* p = ai; p != NULL; p = p->ai_next) { ASSERT_TRUE(p->ai_family == AF_INET || p->ai_family == AF_INET6); if (p->ai_socktype == SOCK_STREAM) { ASSERT_EQ(IPPROTO_TCP, p->ai_protocol); saw_tcp = true; } else if (p->ai_socktype == SOCK_DGRAM) { ASSERT_EQ(IPPROTO_UDP, p->ai_protocol); saw_udp = true; } } ASSERT_TRUE(saw_tcp); ASSERT_TRUE(saw_udp); freeaddrinfo(ai); } TEST(netdb, getaddrinfo_service_lookup) { addrinfo* ai = NULL; ASSERT_EQ(0, getaddrinfo("localhost", "smtp", NULL, &ai)); ASSERT_EQ(SOCK_STREAM, ai->ai_socktype); ASSERT_EQ(IPPROTO_TCP, ai->ai_protocol); ASSERT_EQ(25, ntohs(reinterpret_cast<sockaddr_in*>(ai->ai_addr)->sin_port)); freeaddrinfo(ai); } TEST(netdb, getaddrinfo_hints) { addrinfo hints; memset(&hints, 0, sizeof(hints)); hints.ai_family = AF_INET; hints.ai_socktype = SOCK_STREAM; hints.ai_protocol = IPPROTO_TCP; addrinfo* ai = NULL; ASSERT_EQ(0, getaddrinfo( "localhost", "9999", &hints, &ai)); ASSERT_TRUE(ai != NULL); // In glibc, getaddrinfo() converts ::1 to 127.0.0.1 for localhost, // so one or two addrinfo may be returned. addrinfo* tai = ai; while (tai != NULL) { ASSERT_EQ(AF_INET, tai->ai_family); ASSERT_EQ(SOCK_STREAM, tai->ai_socktype); ASSERT_EQ(IPPROTO_TCP, tai->ai_protocol); tai = tai->ai_next; } freeaddrinfo(ai); } TEST(netdb, getaddrinfo_ip6_localhost) { addrinfo* ai = NULL; ASSERT_EQ(0, getaddrinfo("ip6-localhost", NULL, NULL, &ai)); ASSERT_TRUE(ai != NULL); ASSERT_GE(ai->ai_addrlen, static_cast<socklen_t>(sizeof(sockaddr_in6))); ASSERT_TRUE(ai->ai_addr != NULL); sockaddr_in6 *addr = reinterpret_cast<sockaddr_in6*>(ai->ai_addr); ASSERT_EQ(addr->sin6_family, AF_INET6); ASSERT_EQ(0, memcmp(&addr->sin6_addr, &in6addr_loopback, sizeof(in6_addr))); freeaddrinfo(ai); } TEST(netdb, getnameinfo_salen) { sockaddr_storage ss; memset(&ss, 0, sizeof(ss)); sockaddr* sa = reinterpret_cast<sockaddr*>(&ss); char tmp[16]; ss.ss_family = AF_INET; socklen_t too_much = sizeof(ss); socklen_t just_right = sizeof(sockaddr_in); socklen_t too_little = sizeof(sockaddr_in) - 1; ASSERT_EQ(0, getnameinfo(sa, too_much, tmp, sizeof(tmp), NULL, 0, NI_NUMERICHOST)); ASSERT_STREQ("0.0.0.0", tmp); ASSERT_EQ(0, getnameinfo(sa, just_right, tmp, sizeof(tmp), NULL, 0, NI_NUMERICHOST)); ASSERT_STREQ("0.0.0.0", tmp); ASSERT_EQ(EAI_FAMILY, getnameinfo(sa, too_little, tmp, sizeof(tmp), NULL, 0, NI_NUMERICHOST)); ss.ss_family = AF_INET6; just_right = sizeof(sockaddr_in6); too_little = sizeof(sockaddr_in6) - 1; too_much = just_right + 1; ASSERT_EQ(0, getnameinfo(sa, too_much, tmp, sizeof(tmp), NULL, 0, NI_NUMERICHOST)); ASSERT_STREQ("::", tmp); ASSERT_EQ(0, getnameinfo(sa, just_right, tmp, sizeof(tmp), NULL, 0, NI_NUMERICHOST)); ASSERT_STREQ("::", tmp); ASSERT_EQ(EAI_FAMILY, getnameinfo(sa, too_little, tmp, sizeof(tmp), NULL, 0, NI_NUMERICHOST)); } TEST(netdb, getnameinfo_localhost) { sockaddr_in addr; char host[NI_MAXHOST]; memset(&addr, 0, sizeof(sockaddr_in)); addr.sin_family = AF_INET; addr.sin_addr.s_addr = htonl(0x7f000001); ASSERT_EQ(0, getnameinfo(reinterpret_cast<sockaddr*>(&addr), sizeof(addr), host, sizeof(host), NULL, 0, 0)); ASSERT_STREQ(host, "localhost"); } static void VerifyLocalhostName(const char* name) { // Test possible localhost name and aliases, which depend on /etc/hosts or /system/etc/hosts. ASSERT_TRUE(strcmp(name, "localhost") == 0 || strcmp(name, "ip6-localhost") == 0 || strcmp(name, "ip6-loopback") == 0) << name; } TEST(netdb, getnameinfo_ip6_localhost) { sockaddr_in6 addr; char host[NI_MAXHOST]; memset(&addr, 0, sizeof(sockaddr_in6)); addr.sin6_family = AF_INET6; addr.sin6_addr = in6addr_loopback; ASSERT_EQ(0, getnameinfo(reinterpret_cast<sockaddr*>(&addr), sizeof(addr), host, sizeof(host), NULL, 0, 0)); VerifyLocalhostName(host); } static void VerifyLocalhost(hostent *hent) { ASSERT_TRUE(hent != NULL); VerifyLocalhostName(hent->h_name); for (size_t i = 0; hent->h_aliases[i] != NULL; ++i) { VerifyLocalhostName(hent->h_aliases[i]); } ASSERT_EQ(hent->h_addrtype, AF_INET); ASSERT_EQ(hent->h_addr[0], 127); ASSERT_EQ(hent->h_addr[1], 0); ASSERT_EQ(hent->h_addr[2], 0); ASSERT_EQ(hent->h_addr[3], 1); } TEST(netdb, gethostbyname) { hostent* hp = gethostbyname("localhost"); VerifyLocalhost(hp); } TEST(netdb, gethostbyname2) { hostent* hp = gethostbyname2("localhost", AF_INET); VerifyLocalhost(hp); } TEST(netdb, gethostbyname_r) { hostent hent; hostent *hp; char buf[512]; int err; int result = gethostbyname_r("localhost", &hent, buf, sizeof(buf), &hp, &err); ASSERT_EQ(0, result); VerifyLocalhost(hp); // Change hp->h_addr to test reentrancy. hp->h_addr[0] = 0; hostent hent2; hostent *hp2; char buf2[512]; result = gethostbyname_r("localhost", &hent2, buf2, sizeof(buf2), &hp2, &err); ASSERT_EQ(0, result); VerifyLocalhost(hp2); ASSERT_EQ(0, hp->h_addr[0]); } TEST(netdb, gethostbyname2_r) { hostent hent; hostent *hp; char buf[512]; int err; int result = gethostbyname2_r("localhost", AF_INET, &hent, buf, sizeof(buf), &hp, &err); ASSERT_EQ(0, result); VerifyLocalhost(hp); // Change hp->h_addr to test reentrancy. hp->h_addr[0] = 0; hostent hent2; hostent *hp2; char buf2[512]; result = gethostbyname2_r("localhost", AF_INET, &hent2, buf2, sizeof(buf2), &hp2, &err); ASSERT_EQ(0, result); VerifyLocalhost(hp2); ASSERT_EQ(0, hp->h_addr[0]); } TEST(netdb, gethostbyaddr) { in_addr addr = { htonl(0x7f000001) }; hostent *hp = gethostbyaddr(&addr, sizeof(addr), AF_INET); VerifyLocalhost(hp); } TEST(netdb, gethostbyaddr_r) { in_addr addr = { htonl(0x7f000001) }; hostent hent; hostent *hp; char buf[512]; int err; int result = gethostbyaddr_r(&addr, sizeof(addr), AF_INET, &hent, buf, sizeof(buf), &hp, &err); ASSERT_EQ(0, result); VerifyLocalhost(hp); // Change hp->h_addr to test reentrancy. hp->h_addr[0] = 0; hostent hent2; hostent *hp2; char buf2[512]; result = gethostbyaddr_r(&addr, sizeof(addr), AF_INET, &hent2, buf2, sizeof(buf2), &hp2, &err); ASSERT_EQ(0, result); VerifyLocalhost(hp2); ASSERT_EQ(0, hp->h_addr[0]); } TEST(netdb, gethostbyname_r_ERANGE) { hostent hent; hostent *hp; char buf[4]; // Use too small buffer. int err; int result = gethostbyname_r("localhost", &hent, buf, sizeof(buf), &hp, &err); ASSERT_EQ(ERANGE, result); ASSERT_EQ(NULL, hp); } TEST(netdb, gethostbyname2_r_ERANGE) { hostent hent; hostent *hp; char buf[4]; // Use too small buffer. int err; int result = gethostbyname2_r("localhost", AF_INET, &hent, buf, sizeof(buf), &hp, &err); ASSERT_EQ(ERANGE, result); ASSERT_EQ(NULL, hp); } TEST(netdb, gethostbyaddr_r_ERANGE) { in_addr addr = { htonl(0x7f000001) }; hostent hent; hostent *hp; char buf[4]; // Use too small buffer. int err; int result = gethostbyaddr_r(&addr, sizeof(addr), AF_INET, &hent, buf, sizeof(buf), &hp, &err); ASSERT_EQ(ERANGE, result); ASSERT_EQ(NULL, hp); } TEST(netdb, getservbyname) { // smtp is TCP-only, so we know we'll get 25/tcp back. servent* s = getservbyname("smtp", NULL); ASSERT_TRUE(s != NULL); ASSERT_EQ(25, ntohs(s->s_port)); ASSERT_STREQ("tcp", s->s_proto); // We get the same result by explicitly asking for tcp. s = getservbyname("smtp", "tcp"); ASSERT_TRUE(s != NULL); ASSERT_EQ(25, ntohs(s->s_port)); ASSERT_STREQ("tcp", s->s_proto); // And we get a failure if we explicitly ask for udp. s = getservbyname("smtp", "udp"); ASSERT_TRUE(s == NULL); // But there are actually udp services. s = getservbyname("echo", "udp"); ASSERT_TRUE(s != NULL); ASSERT_EQ(7, ntohs(s->s_port)); ASSERT_STREQ("udp", s->s_proto); }