ng_ipaddr.c 3.7 KB

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