ng_ipaddr.c 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177
  1. /*
  2. * (c) 2008 Florian Westphal <fw@strlen.de>, public domain.
  3. */
  4. #include "portab.h"
  5. /**
  6. * @file
  7. * Functions for AF_ agnostic ipv4/ipv6 handling.
  8. */
  9. #include <assert.h>
  10. #include <stdio.h>
  11. #include <string.h>
  12. #ifdef HAVE_GETADDRINFO
  13. #include <netdb.h>
  14. #include <sys/types.h>
  15. #endif
  16. #include "ng_ipaddr.h"
  17. GLOBAL bool
  18. ng_ipaddr_init(ng_ipaddr_t *addr, const char *ip_str, UINT16 port)
  19. {
  20. #ifdef HAVE_GETADDRINFO
  21. int ret;
  22. char portstr[64];
  23. struct addrinfo *res0;
  24. struct addrinfo hints;
  25. assert(ip_str);
  26. memset(&hints, 0, sizeof(hints));
  27. hints.ai_flags = AI_NUMERICHOST;
  28. #ifndef WANT_IPV6 /* do not convert ipv6 addresses */
  29. hints.ai_family = AF_INET;
  30. #endif
  31. /* some getaddrinfo implementations require that ai_socktype is set. */
  32. hints.ai_socktype = SOCK_STREAM;
  33. /* silly, but ngircd stores UINT16 in server config, not string */
  34. snprintf(portstr, sizeof(portstr), "%u", (unsigned int) port);
  35. ret = getaddrinfo(ip_str, portstr, &hints, &res0);
  36. if (ret != 0)
  37. return false;
  38. assert(sizeof(*addr) >= res0->ai_addrlen);
  39. if (sizeof(*addr) >= res0->ai_addrlen)
  40. memcpy(addr, res0->ai_addr, res0->ai_addrlen);
  41. else
  42. ret = -1;
  43. freeaddrinfo(res0);
  44. return ret == 0;
  45. #else /* HAVE_GETADDRINFO */
  46. assert(ip_str);
  47. memset(addr, 0, sizeof *addr);
  48. #ifdef HAVE_sockaddr_in_len
  49. addr->sin4.sin_len = sizeof(addr->sin4);
  50. #endif
  51. addr->sin4.sin_family = AF_INET;
  52. # ifdef HAVE_INET_ATON
  53. if (inet_aton(ip_str, &addr->sin4.sin_addr) == 0)
  54. return false;
  55. # else
  56. addr->sin4.sin_addr.s_addr = inet_addr(ip_str);
  57. if (addr->sin4.sin_addr.s_addr == (unsigned) -1)
  58. return false;
  59. # endif
  60. ng_ipaddr_setport(addr, port);
  61. return true;
  62. #endif /* HAVE_GETADDRINFO */
  63. }
  64. GLOBAL void
  65. ng_ipaddr_setport(ng_ipaddr_t *a, UINT16 port)
  66. {
  67. #ifdef WANT_IPV6
  68. int af;
  69. assert(a != NULL);
  70. af = a->sa.sa_family;
  71. assert(af == AF_INET || af == AF_INET6);
  72. switch (af) {
  73. case AF_INET:
  74. a->sin4.sin_port = htons(port);
  75. break;
  76. case AF_INET6:
  77. a->sin6.sin6_port = htons(port);
  78. break;
  79. }
  80. #else /* WANT_IPV6 */
  81. assert(a != NULL);
  82. assert(a->sin4.sin_family == AF_INET);
  83. a->sin4.sin_port = htons(port);
  84. #endif /* WANT_IPV6 */
  85. }
  86. GLOBAL bool
  87. ng_ipaddr_ipequal(const ng_ipaddr_t *a, const ng_ipaddr_t *b)
  88. {
  89. assert(a != NULL);
  90. assert(b != NULL);
  91. #ifdef WANT_IPV6
  92. if (a->sa.sa_family != b->sa.sa_family)
  93. return false;
  94. assert(ng_ipaddr_salen(a) == ng_ipaddr_salen(b));
  95. switch (a->sa.sa_family) {
  96. case AF_INET6:
  97. return IN6_ARE_ADDR_EQUAL(&a->sin6.sin6_addr, &b->sin6.sin6_addr);
  98. case AF_INET:
  99. return memcmp(&a->sin4.sin_addr, &b->sin4.sin_addr, sizeof(a->sin4.sin_addr)) == 0;
  100. }
  101. return false;
  102. #else
  103. assert(a->sin4.sin_family == AF_INET);
  104. assert(b->sin4.sin_family == AF_INET);
  105. return memcmp(&a->sin4.sin_addr, &b->sin4.sin_addr, sizeof(a->sin4.sin_addr)) == 0;
  106. #endif
  107. }
  108. #ifdef WANT_IPV6
  109. GLOBAL const char *
  110. ng_ipaddr_tostr(const ng_ipaddr_t *addr)
  111. {
  112. static char strbuf[NG_INET_ADDRSTRLEN];
  113. strbuf[0] = 0;
  114. ng_ipaddr_tostr_r(addr, strbuf);
  115. return strbuf;
  116. }
  117. /* str must be at least NG_INET_ADDRSTRLEN bytes long */
  118. GLOBAL bool
  119. ng_ipaddr_tostr_r(const ng_ipaddr_t *addr, char *str)
  120. {
  121. #ifdef HAVE_GETNAMEINFO
  122. const struct sockaddr *sa = (const struct sockaddr *) addr;
  123. int ret;
  124. *str = 0;
  125. ret = getnameinfo(sa, ng_ipaddr_salen(addr),
  126. str, NG_INET_ADDRSTRLEN, NULL, 0, NI_NUMERICHOST);
  127. /*
  128. * avoid leading ':'.
  129. * causes mis-interpretation of client host in e.g. /WHOIS
  130. */
  131. if (*str == ':') {
  132. char tmp[NG_INET_ADDRSTRLEN] = "0";
  133. ret = getnameinfo(sa, ng_ipaddr_salen(addr),
  134. tmp + 1, (socklen_t)sizeof(tmp) - 1,
  135. NULL, 0, NI_NUMERICHOST);
  136. if (ret == 0)
  137. strlcpy(str, tmp, NG_INET_ADDRSTRLEN);
  138. }
  139. assert (ret == 0);
  140. return ret == 0;
  141. #else
  142. abort(); /* WANT_IPV6 depends on HAVE_GETNAMEINFO */
  143. #endif
  144. }
  145. #endif /* WANT_IPV6 */
  146. /* -eof- */