getaddrinfo.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439
  1. /* Get address information (partial implementation).
  2. Copyright (C) 1997, 2001-2002, 2004-2010 Free Software Foundation, Inc.
  3. Contributed by Simon Josefsson <simon@josefsson.org>.
  4. This program is free software; you can redistribute it and/or modify
  5. it under the terms of the GNU General Public License as published by
  6. the Free Software Foundation; either version 3, or (at your option)
  7. any later version.
  8. This program is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. GNU General Public License for more details.
  12. You should have received a copy of the GNU General Public License
  13. along with this program; if not, write to the Free Software Foundation,
  14. Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */
  15. #include <config.h>
  16. /* Don't use __attribute__ __nonnull__ in this compilation unit. Otherwise gcc
  17. optimizes away the sa == NULL test below. */
  18. #define _GL_ARG_NONNULL(params)
  19. #include <netdb.h>
  20. #if HAVE_NETINET_IN_H
  21. # include <netinet/in.h>
  22. #endif
  23. /* Get inet_ntop. */
  24. #include <arpa/inet.h>
  25. /* Get calloc. */
  26. #include <stdlib.h>
  27. /* Get memcpy, strdup. */
  28. #include <string.h>
  29. /* Get snprintf. */
  30. #include <stdio.h>
  31. #include <stdbool.h>
  32. #include "gettext.h"
  33. #define _(String) gettext (String)
  34. #define N_(String) String
  35. /* BeOS has AF_INET, but not PF_INET. */
  36. #ifndef PF_INET
  37. # define PF_INET AF_INET
  38. #endif
  39. /* BeOS also lacks PF_UNSPEC. */
  40. #ifndef PF_UNSPEC
  41. # define PF_UNSPEC 0
  42. #endif
  43. #if defined _WIN32 || defined __WIN32__
  44. # define WIN32_NATIVE
  45. #endif
  46. #ifdef WIN32_NATIVE
  47. typedef int (WSAAPI *getaddrinfo_func) (const char*, const char*,
  48. const struct addrinfo*,
  49. struct addrinfo**);
  50. typedef void (WSAAPI *freeaddrinfo_func) (struct addrinfo*);
  51. typedef int (WSAAPI *getnameinfo_func) (const struct sockaddr*,
  52. socklen_t, char*, DWORD,
  53. char*, DWORD, int);
  54. static getaddrinfo_func getaddrinfo_ptr = NULL;
  55. static freeaddrinfo_func freeaddrinfo_ptr = NULL;
  56. static getnameinfo_func getnameinfo_ptr = NULL;
  57. static int
  58. use_win32_p (void)
  59. {
  60. static int done = 0;
  61. HMODULE h;
  62. if (done)
  63. return getaddrinfo_ptr ? 1 : 0;
  64. done = 1;
  65. h = GetModuleHandle ("ws2_32.dll");
  66. if (h)
  67. {
  68. getaddrinfo_ptr = (getaddrinfo_func) GetProcAddress (h, "getaddrinfo");
  69. freeaddrinfo_ptr = (freeaddrinfo_func) GetProcAddress (h, "freeaddrinfo");
  70. getnameinfo_ptr = (getnameinfo_func) GetProcAddress (h, "getnameinfo");
  71. }
  72. /* If either is missing, something is odd. */
  73. if (!getaddrinfo_ptr || !freeaddrinfo_ptr || !getnameinfo_ptr)
  74. {
  75. getaddrinfo_ptr = NULL;
  76. freeaddrinfo_ptr = NULL;
  77. getnameinfo_ptr = NULL;
  78. return 0;
  79. }
  80. return 1;
  81. }
  82. #endif
  83. static inline bool
  84. validate_family (int family)
  85. {
  86. /* FIXME: Support more families. */
  87. #if HAVE_IPV4
  88. if (family == PF_INET)
  89. return true;
  90. #endif
  91. #if HAVE_IPV6
  92. if (family == PF_INET6)
  93. return true;
  94. #endif
  95. if (family == PF_UNSPEC)
  96. return true;
  97. return false;
  98. }
  99. /* Translate name of a service location and/or a service name to set of
  100. socket addresses. */
  101. int
  102. getaddrinfo (const char *restrict nodename,
  103. const char *restrict servname,
  104. const struct addrinfo *restrict hints,
  105. struct addrinfo **restrict res)
  106. {
  107. struct addrinfo *tmp;
  108. int port = 0;
  109. struct hostent *he;
  110. void *storage;
  111. size_t size;
  112. #if HAVE_IPV6
  113. struct v6_pair {
  114. struct addrinfo addrinfo;
  115. struct sockaddr_in6 sockaddr_in6;
  116. };
  117. #endif
  118. #if HAVE_IPV4
  119. struct v4_pair {
  120. struct addrinfo addrinfo;
  121. struct sockaddr_in sockaddr_in;
  122. };
  123. #endif
  124. #ifdef WIN32_NATIVE
  125. if (use_win32_p ())
  126. return getaddrinfo_ptr (nodename, servname, hints, res);
  127. #endif
  128. if (hints && (hints->ai_flags & ~(AI_CANONNAME|AI_PASSIVE)))
  129. /* FIXME: Support more flags. */
  130. return EAI_BADFLAGS;
  131. if (hints && !validate_family (hints->ai_family))
  132. return EAI_FAMILY;
  133. if (hints &&
  134. hints->ai_socktype != SOCK_STREAM && hints->ai_socktype != SOCK_DGRAM)
  135. /* FIXME: Support other socktype. */
  136. return EAI_SOCKTYPE; /* FIXME: Better return code? */
  137. if (!nodename)
  138. {
  139. if (!(hints->ai_flags & AI_PASSIVE))
  140. return EAI_NONAME;
  141. #ifdef HAVE_IPV6
  142. nodename = (hints->ai_family == AF_INET6) ? "::" : "0.0.0.0";
  143. #else
  144. nodename = "0.0.0.0";
  145. #endif
  146. }
  147. if (servname)
  148. {
  149. struct servent *se = NULL;
  150. const char *proto =
  151. (hints && hints->ai_socktype == SOCK_DGRAM) ? "udp" : "tcp";
  152. if (hints == NULL || !(hints->ai_flags & AI_NUMERICSERV))
  153. /* FIXME: Use getservbyname_r if available. */
  154. se = getservbyname (servname, proto);
  155. if (!se)
  156. {
  157. char *c;
  158. if (!(*servname >= '0' && *servname <= '9'))
  159. return EAI_NONAME;
  160. port = strtoul (servname, &c, 10);
  161. if (*c || port > 0xffff)
  162. return EAI_NONAME;
  163. port = htons (port);
  164. }
  165. else
  166. port = se->s_port;
  167. }
  168. /* FIXME: Use gethostbyname_r if available. */
  169. he = gethostbyname (nodename);
  170. if (!he || he->h_addr_list[0] == NULL)
  171. return EAI_NONAME;
  172. switch (he->h_addrtype)
  173. {
  174. #if HAVE_IPV6
  175. case PF_INET6:
  176. size = sizeof (struct v6_pair);
  177. break;
  178. #endif
  179. #if HAVE_IPV4
  180. case PF_INET:
  181. size = sizeof (struct v4_pair);
  182. break;
  183. #endif
  184. default:
  185. return EAI_NODATA;
  186. }
  187. storage = calloc (1, size);
  188. if (!storage)
  189. return EAI_MEMORY;
  190. switch (he->h_addrtype)
  191. {
  192. #if HAVE_IPV6
  193. case PF_INET6:
  194. {
  195. struct v6_pair *p = storage;
  196. struct sockaddr_in6 *sinp = &p->sockaddr_in6;
  197. tmp = &p->addrinfo;
  198. if (port)
  199. sinp->sin6_port = port;
  200. if (he->h_length != sizeof (sinp->sin6_addr))
  201. {
  202. free (storage);
  203. return EAI_SYSTEM; /* FIXME: Better return code? Set errno? */
  204. }
  205. memcpy (&sinp->sin6_addr, he->h_addr_list[0], sizeof sinp->sin6_addr);
  206. tmp->ai_addr = (struct sockaddr *) sinp;
  207. tmp->ai_addrlen = sizeof *sinp;
  208. }
  209. break;
  210. #endif
  211. #if HAVE_IPV4
  212. case PF_INET:
  213. {
  214. struct v4_pair *p = storage;
  215. struct sockaddr_in *sinp = &p->sockaddr_in;
  216. tmp = &p->addrinfo;
  217. if (port)
  218. sinp->sin_port = port;
  219. if (he->h_length != sizeof (sinp->sin_addr))
  220. {
  221. free (storage);
  222. return EAI_SYSTEM; /* FIXME: Better return code? Set errno? */
  223. }
  224. memcpy (&sinp->sin_addr, he->h_addr_list[0], sizeof sinp->sin_addr);
  225. tmp->ai_addr = (struct sockaddr *) sinp;
  226. tmp->ai_addrlen = sizeof *sinp;
  227. }
  228. break;
  229. #endif
  230. default:
  231. free (storage);
  232. return EAI_NODATA;
  233. }
  234. if (hints && hints->ai_flags & AI_CANONNAME)
  235. {
  236. const char *cn;
  237. if (he->h_name)
  238. cn = he->h_name;
  239. else
  240. cn = nodename;
  241. tmp->ai_canonname = strdup (cn);
  242. if (!tmp->ai_canonname)
  243. {
  244. free (storage);
  245. return EAI_MEMORY;
  246. }
  247. }
  248. tmp->ai_protocol = (hints) ? hints->ai_protocol : 0;
  249. tmp->ai_socktype = (hints) ? hints->ai_socktype : 0;
  250. tmp->ai_addr->sa_family = he->h_addrtype;
  251. tmp->ai_family = he->h_addrtype;
  252. #ifdef HAVE_STRUCT_SOCKADDR_SA_LEN
  253. switch (he->h_addrtype)
  254. {
  255. #if HAVE_IPV4
  256. case AF_INET:
  257. tmp->ai_addr->sa_len = sizeof (struct sockaddr_in);
  258. break;
  259. #endif
  260. #if HAVE_IPV6
  261. case AF_INET6:
  262. tmp->ai_addr->sa_len = sizeof (struct sockaddr_in6);
  263. break;
  264. #endif
  265. }
  266. #endif
  267. /* FIXME: If more than one address, create linked list of addrinfo's. */
  268. *res = tmp;
  269. return 0;
  270. }
  271. /* Free `addrinfo' structure AI including associated storage. */
  272. void
  273. freeaddrinfo (struct addrinfo *ai)
  274. {
  275. #ifdef WIN32_NATIVE
  276. if (use_win32_p ())
  277. {
  278. freeaddrinfo_ptr (ai);
  279. return;
  280. }
  281. #endif
  282. while (ai)
  283. {
  284. struct addrinfo *cur;
  285. cur = ai;
  286. ai = ai->ai_next;
  287. free (cur->ai_canonname);
  288. free (cur);
  289. }
  290. }
  291. int
  292. getnameinfo (const struct sockaddr *restrict sa, socklen_t salen,
  293. char *restrict node, socklen_t nodelen,
  294. char *restrict service, socklen_t servicelen,
  295. int flags)
  296. {
  297. #ifdef WIN32_NATIVE
  298. if (use_win32_p ())
  299. return getnameinfo_ptr (sa, salen, node, nodelen,
  300. service, servicelen, flags);
  301. #endif
  302. /* FIXME: Support other flags. */
  303. if ((node && nodelen > 0 && !(flags & NI_NUMERICHOST)) ||
  304. (service && servicelen > 0 && !(flags & NI_NUMERICHOST)) ||
  305. (flags & ~(NI_NUMERICHOST|NI_NUMERICSERV)))
  306. return EAI_BADFLAGS;
  307. if (sa == NULL || salen < sizeof (sa->sa_family))
  308. return EAI_FAMILY;
  309. switch (sa->sa_family)
  310. {
  311. #if HAVE_IPV4
  312. case AF_INET:
  313. if (salen < sizeof (struct sockaddr_in))
  314. return EAI_FAMILY;
  315. break;
  316. #endif
  317. #if HAVE_IPV6
  318. case AF_INET6:
  319. if (salen < sizeof (struct sockaddr_in6))
  320. return EAI_FAMILY;
  321. break;
  322. #endif
  323. default:
  324. return EAI_FAMILY;
  325. }
  326. if (node && nodelen > 0 && flags & NI_NUMERICHOST)
  327. {
  328. switch (sa->sa_family)
  329. {
  330. #if HAVE_IPV4
  331. case AF_INET:
  332. if (!inet_ntop (AF_INET,
  333. &(((const struct sockaddr_in *) sa)->sin_addr),
  334. node, nodelen))
  335. return EAI_SYSTEM;
  336. break;
  337. #endif
  338. #if HAVE_IPV6
  339. case AF_INET6:
  340. if (!inet_ntop (AF_INET6,
  341. &(((const struct sockaddr_in6 *) sa)->sin6_addr),
  342. node, nodelen))
  343. return EAI_SYSTEM;
  344. break;
  345. #endif
  346. default:
  347. return EAI_FAMILY;
  348. }
  349. }
  350. if (service && servicelen > 0 && flags & NI_NUMERICSERV)
  351. switch (sa->sa_family)
  352. {
  353. #if HAVE_IPV4
  354. case AF_INET:
  355. #endif
  356. #if HAVE_IPV6
  357. case AF_INET6:
  358. #endif
  359. {
  360. unsigned short int port
  361. = ntohs (((const struct sockaddr_in *) sa)->sin_port);
  362. if (servicelen <= snprintf (service, servicelen, "%u", port))
  363. return EAI_OVERFLOW;
  364. }
  365. break;
  366. }
  367. return 0;
  368. }