InetAddress.cc revision e0ccc29f
1efbfdb0cSShuo Chen#include "InetAddress.h"
2efbfdb0cSShuo Chen
324ca08a8SShuo Chen#include <memory>
424ca08a8SShuo Chen
55b543c9cSShuo Chen#include <assert.h>
65b543c9cSShuo Chen#include <arpa/inet.h>
724ca08a8SShuo Chen#include <netdb.h>
824ca08a8SShuo Chen
924ca08a8SShuo Chenstatic_assert(sizeof(InetAddress) == sizeof(struct sockaddr_in6),
1024ca08a8SShuo Chen              "InetAddress is same size as sockaddr_in6");
1124ca08a8SShuo Chenstatic_assert(offsetof(sockaddr_in, sin_family) == 0, "sin_family offset 0");
1224ca08a8SShuo Chenstatic_assert(offsetof(sockaddr_in6, sin6_family) == 0, "sin6_family offset 0");
1324ca08a8SShuo Chenstatic_assert(offsetof(sockaddr_in, sin_port) == 2, "sin_port offset 2");
1424ca08a8SShuo Chenstatic_assert(offsetof(sockaddr_in6, sin6_port) == 2, "sin6_port offset 2");
155b543c9cSShuo Chen
16aafef3ccSShuo ChenInetAddress::InetAddress(StringArg ip, uint16_t port)
17aafef3ccSShuo Chen{
1824ca08a8SShuo Chen  setPort(port);
1924ca08a8SShuo Chen
2024ca08a8SShuo Chen  int result = 0;
2124ca08a8SShuo Chen  if (strchr(ip.c_str(), ':') == NULL)
2224ca08a8SShuo Chen  {
2324ca08a8SShuo Chen    result = ::inet_pton(AF_INET, ip.c_str(), &addr_.sin_addr);
2424ca08a8SShuo Chen  }
2524ca08a8SShuo Chen  else
2624ca08a8SShuo Chen  {
2724ca08a8SShuo Chen    result = ::inet_pton(AF_INET6, ip.c_str(), &addr6_.sin6_addr);
2824ca08a8SShuo Chen  }
2924ca08a8SShuo Chen
3024ca08a8SShuo Chen  assert(result == 1 && "Invalid IP format");
31aafef3ccSShuo Chen}
32efbfdb0cSShuo Chen
3324ca08a8SShuo ChenInetAddress::InetAddress(uint16_t port, bool ipv6)
34efbfdb0cSShuo Chen{
3524ca08a8SShuo Chen  static_assert(offsetof(InetAddress, addr6_) == 0, "addr6_ offset 0");
3624ca08a8SShuo Chen  static_assert(offsetof(InetAddress, addr_) == 0, "addr_ offset 0");
3724ca08a8SShuo Chen  bool loopbackOnly = false;
3824ca08a8SShuo Chen  if (ipv6)
3924ca08a8SShuo Chen  {
4024ca08a8SShuo Chen    memZero(&addr6_, sizeof addr6_);
4124ca08a8SShuo Chen    addr6_.sin6_family = AF_INET6;
4224ca08a8SShuo Chen    in6_addr ip = loopbackOnly ? in6addr_loopback : in6addr_any;
4324ca08a8SShuo Chen    addr6_.sin6_addr = ip;
4424ca08a8SShuo Chen    addr6_.sin6_port = htons(port);
4524ca08a8SShuo Chen  }
4624ca08a8SShuo Chen  else
4724ca08a8SShuo Chen  {
4824ca08a8SShuo Chen    memZero(&addr_, sizeof addr_);
4924ca08a8SShuo Chen    addr_.sin_family = AF_INET;
5024ca08a8SShuo Chen    in_addr_t ip = loopbackOnly ? INADDR_LOOPBACK : INADDR_ANY;
5124ca08a8SShuo Chen    addr_.sin_addr.s_addr = htonl(ip);
5224ca08a8SShuo Chen    addr_.sin_port = htons(port);
5324ca08a8SShuo Chen  }
5424ca08a8SShuo Chen}
5524ca08a8SShuo Chen
5624ca08a8SShuo ChenInetAddress::InetAddress(const struct sockaddr& saddr)
5724ca08a8SShuo Chen{
5824ca08a8SShuo Chen  if (saddr.sa_family == AF_INET)
5924ca08a8SShuo Chen  {
6024ca08a8SShuo Chen    memcpy(&addr_, &saddr, sizeof addr_);
6124ca08a8SShuo Chen  }
6224ca08a8SShuo Chen  else if (saddr.sa_family == AF_INET6)
6324ca08a8SShuo Chen  {
6424ca08a8SShuo Chen    memcpy(&addr6_, &saddr, sizeof addr6_);
6524ca08a8SShuo Chen  }
6624ca08a8SShuo Chen  else
6724ca08a8SShuo Chen  {
6824ca08a8SShuo Chen    assert(false);
6924ca08a8SShuo Chen  }
70efbfdb0cSShuo Chen}
715b543c9cSShuo Chen
725b543c9cSShuo Chenstd::string InetAddress::toIp() const
735b543c9cSShuo Chen{
7424ca08a8SShuo Chen  char buf[64] = "";
7524ca08a8SShuo Chen  static_assert(sizeof buf >= INET_ADDRSTRLEN);
7624ca08a8SShuo Chen  static_assert(sizeof buf >= INET6_ADDRSTRLEN);
7724ca08a8SShuo Chen
7824ca08a8SShuo Chen  if (family() == AF_INET)
7924ca08a8SShuo Chen  {
8024ca08a8SShuo Chen    ::inet_ntop(AF_INET, &addr_.sin_addr, buf, static_cast<socklen_t>(sizeof buf));
8124ca08a8SShuo Chen  }
8224ca08a8SShuo Chen  else if (family() == AF_INET6)
8324ca08a8SShuo Chen  {
8424ca08a8SShuo Chen    ::inet_ntop(AF_INET6, &addr6_.sin6_addr, buf, static_cast<socklen_t>(sizeof buf));
8524ca08a8SShuo Chen  }
8624ca08a8SShuo Chen
875b543c9cSShuo Chen  return buf;
885b543c9cSShuo Chen}
895b543c9cSShuo Chen
905b543c9cSShuo Chenstd::string InetAddress::toIpPort() const
915b543c9cSShuo Chen{
9224ca08a8SShuo Chen  char buf[32] = "";
9324ca08a8SShuo Chen  snprintf(buf, sizeof buf, ":%u", port());
94e0ccc29fSShuo Chen
95e0ccc29fSShuo Chen  if (family() == AF_INET6)
96e0ccc29fSShuo Chen    return "[" + toIp() + "]" + buf;
97e0ccc29fSShuo Chen
9824ca08a8SShuo Chen  return toIp() + buf;
995b543c9cSShuo Chen}
1005b543c9cSShuo Chen
10124ca08a8SShuo Chenbool InetAddress::operator==(const InetAddress& rhs) const
102503bffb0SShuo Chen{
10324ca08a8SShuo Chen  if (family() == rhs.family())
104503bffb0SShuo Chen  {
10524ca08a8SShuo Chen    if (family() == AF_INET)
106503bffb0SShuo Chen    {
10724ca08a8SShuo Chen      return addr_.sin_port == rhs.addr_.sin_port &&
10824ca08a8SShuo Chen          addr_.sin_addr.s_addr == rhs.addr_.sin_addr.s_addr;
109503bffb0SShuo Chen    }
11024ca08a8SShuo Chen    else if (family() == AF_INET6)
111503bffb0SShuo Chen    {
11224ca08a8SShuo Chen      return addr6_.sin6_port == rhs.addr6_.sin6_port &&
11324ca08a8SShuo Chen          memcmp(&addr6_.sin6_addr, &rhs.addr6_.sin6_addr, sizeof addr6_.sin6_addr) == 0;
114503bffb0SShuo Chen    }
115503bffb0SShuo Chen  }
116503bffb0SShuo Chen  return false;
117503bffb0SShuo Chen}
118503bffb0SShuo Chen
11924ca08a8SShuo Chen// static
12024ca08a8SShuo Chenbool InetAddress::resolve(StringArg hostname, uint16_t port, InetAddress* out)
1215b543c9cSShuo Chen{
1225b543c9cSShuo Chen  assert(out);
12324ca08a8SShuo Chen  std::vector<InetAddress> addrs = resolveAll(hostname, port);
12424ca08a8SShuo Chen
12524ca08a8SShuo Chen  if (addrs.empty())
12624ca08a8SShuo Chen    return false;
12724ca08a8SShuo Chen
12824ca08a8SShuo Chen  // Read the first result
12924ca08a8SShuo Chen  *out = addrs[0];
13024ca08a8SShuo Chen
13124ca08a8SShuo Chen  return true;
13224ca08a8SShuo Chen}
13324ca08a8SShuo Chen
13424ca08a8SShuo Chen// static
13524ca08a8SShuo Chenstd::vector<InetAddress> InetAddress::resolveAll(StringArg hostname, uint16_t port)
13624ca08a8SShuo Chen{
13724ca08a8SShuo Chen  std::vector<InetAddress> addrs;
13824ca08a8SShuo Chen
13924ca08a8SShuo Chen  struct addrinfo* result = NULL;
14024ca08a8SShuo Chen  int error = getaddrinfo(hostname.c_str(), NULL, NULL, &result);
14124ca08a8SShuo Chen  if (error != 0)
1425b543c9cSShuo Chen  {
14324ca08a8SShuo Chen    if (error == EAI_SYSTEM)
14424ca08a8SShuo Chen    {
14524ca08a8SShuo Chen      perror("InetAddress::resolve");
14624ca08a8SShuo Chen    }
14724ca08a8SShuo Chen    else
14824ca08a8SShuo Chen    {
14924ca08a8SShuo Chen      fprintf(stderr, "InetAddress::resolve: %s\n", gai_strerror(error));
15024ca08a8SShuo Chen    }
15124ca08a8SShuo Chen    return addrs;
1525b543c9cSShuo Chen  }
15324ca08a8SShuo Chen
15424ca08a8SShuo Chen  assert(result);
15524ca08a8SShuo Chen  std::unique_ptr<struct addrinfo, decltype(&freeaddrinfo)> freeResult(result, freeaddrinfo);
15624ca08a8SShuo Chen
15724ca08a8SShuo Chen  for (struct addrinfo* ai = result; ai != NULL; ai = ai->ai_next)
1585b543c9cSShuo Chen  {
15924ca08a8SShuo Chen    InetAddress addr(*ai->ai_addr);
16024ca08a8SShuo Chen    addr.setPort(port);
16124ca08a8SShuo Chen    addrs.push_back(addr);
1625b543c9cSShuo Chen  }
16324ca08a8SShuo Chen  return addrs;
1645b543c9cSShuo Chen}
165