ng_ipaddr.c 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  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. /* some getaddrinfo implementations require that ai_socktype is set. */
  27. hints.ai_socktype = SOCK_STREAM;
  28. /* silly, but ngircd stores UINT16 in server config, not string */
  29. snprintf(portstr, sizeof(portstr), "%u", (unsigned int) port);
  30. ret = getaddrinfo(ip_str, portstr, &hints, &res0);
  31. assert(ret == 0);
  32. if (ret != 0)
  33. return false;
  34. assert(sizeof(*addr) >= res0->ai_addrlen);
  35. if (sizeof(*addr) >= res0->ai_addrlen)
  36. memcpy(addr, res0->ai_addr, res0->ai_addrlen);
  37. else
  38. ret = -1;
  39. freeaddrinfo(res0);
  40. return ret == 0;
  41. #else /* HAVE_GETADDRINFO */
  42. assert(ip_str);
  43. addr->sin4.sin_family = AF_INET;
  44. # ifdef HAVE_INET_ATON
  45. if (inet_aton(ip_str, &addr->sin4.sin_addr) == 0)
  46. return false;
  47. # else
  48. addr->sin4.sin_addr.s_addr = inet_addr(ip_str);
  49. if (addr->sin4.sin_addr.s_addr == (unsigned) -1)
  50. return false;
  51. # endif
  52. ng_ipaddr_setport(addr, port);
  53. return true;
  54. #endif /* HAVE_GETADDRINFO */
  55. }
  56. GLOBAL void
  57. ng_ipaddr_setport(ng_ipaddr_t *a, UINT16 port)
  58. {
  59. #ifdef WANT_IPV6
  60. int af;
  61. assert(a != NULL);
  62. af = a->sa.sa_family;
  63. assert(af == AF_INET || af == AF_INET6);
  64. switch (af) {
  65. case AF_INET:
  66. a->sin4.sin_port = htons(port);
  67. break;
  68. case AF_INET6:
  69. a->sin6.sin6_port = htons(port);
  70. break;
  71. }
  72. #else /* WANT_IPV6 */
  73. assert(a != NULL);
  74. assert(a->sin4.sin_family == AF_INET);
  75. a->sin4.sin_port = htons(port);
  76. #endif /* WANT_IPV6 */
  77. }
  78. GLOBAL bool
  79. ng_ipaddr_ipequal(const ng_ipaddr_t *a, const ng_ipaddr_t *b)
  80. {
  81. assert(a != NULL);
  82. assert(b != NULL);
  83. #ifdef WANT_IPV6
  84. if (a->sa.sa_family != b->sa.sa_family)
  85. return false;
  86. assert(ng_ipaddr_salen(a) == ng_ipaddr_salen(b));
  87. switch (a->sa.sa_family) {
  88. case AF_INET6:
  89. return IN6_ARE_ADDR_EQUAL(&a->sin6.sin6_addr, &b->sin6.sin6_addr);
  90. case AF_INET:
  91. return memcmp(&a->sin4.sin_addr, &b->sin4.sin_addr, sizeof(a->sin4.sin_addr)) == 0;
  92. }
  93. return false;
  94. #else
  95. assert(a->sin4.sin_family == AF_INET);
  96. assert(b->sin4.sin_family == AF_INET);
  97. return memcmp(&a->sin4.sin_addr, &b->sin4.sin_addr, sizeof(a->sin4.sin_addr)) == 0;
  98. #endif
  99. }
  100. #ifdef WANT_IPV6
  101. GLOBAL const char *
  102. ng_ipaddr_tostr(const ng_ipaddr_t *addr)
  103. {
  104. static char strbuf[NG_INET_ADDRSTRLEN];
  105. strbuf[0] = 0;
  106. ng_ipaddr_tostr_r(addr, strbuf);
  107. return strbuf;
  108. }
  109. /* str must be at least NG_INET_ADDRSTRLEN bytes long */
  110. GLOBAL bool
  111. ng_ipaddr_tostr_r(const ng_ipaddr_t *addr, char *str)
  112. {
  113. #ifdef HAVE_GETNAMEINFO
  114. const struct sockaddr *sa = (const struct sockaddr *) addr;
  115. int ret;
  116. *str = 0;
  117. ret = getnameinfo(sa, ng_ipaddr_salen(addr),
  118. str, NG_INET_ADDRSTRLEN, NULL, 0, NI_NUMERICHOST);
  119. /*
  120. * avoid leading ':'.
  121. * causes mis-interpretation of client host in e.g. /WHOIS
  122. */
  123. if (*str == ':') {
  124. char tmp[NG_INET_ADDRSTRLEN] = "0";
  125. ret = getnameinfo(sa, ng_ipaddr_salen(addr),
  126. tmp+1, sizeof(tmp) -1, NULL, 0, NI_NUMERICHOST);
  127. if (ret == 0)
  128. strlcpy(str, tmp, NG_INET_ADDRSTRLEN);
  129. }
  130. assert (ret == 0);
  131. return ret == 0;
  132. #else
  133. abort(); /* WANT_IPV6 depends on HAVE_GETNAMEINFO */
  134. #endif
  135. }
  136. #endif /* WANT_IPV6 */
  137. /* -eof- */