# HG changeset patch # User Mark Doliner # Date 1089042206 0 # Node ID ad93d76666dda4ecd4e7a928cfe2a6411f11d780 # Parent 18f760b306ad4aefdde21e0b6375cf905ea04cf3 [gaim-migrate @ 10273] Minor changes to the listen function in network.c We were using strerror() to print the string form of errno when getaddrinfo() failed, but that is incorrect. getaddrinfo() does not use errno or strerror. It retuns an integer error code which can be converted to a string with gai_strerror(errorcode) When we didn't have getaddrinfo, and we were unable to set SO_REUSEADDR, we would abort the listen attempt. That's not necessary becuase SO_REUSEADDR isn't vital, it's just nice to have (right?) struct addrinfo *res was not getting freed when we did not have any interfaces to listen on (incredibly rare) committer: Tailor Script diff -r 18f760b306ad -r ad93d76666dd src/network.c --- a/src/network.c Mon Jul 05 15:28:40 2004 +0000 +++ b/src/network.c Mon Jul 05 15:43:26 2004 +0000 @@ -149,39 +149,50 @@ static int gaim_network_do_listen(unsigned short port) { -#if HAVE_GETADDRINFO int listenfd; const int on = 1; - struct addrinfo hints, *res, *ressave; +#if HAVE_GETADDRINFO + int errnum; + struct addrinfo hints, *res, *next; char serv[5]; - snprintf(serv, sizeof(serv), "%d", port); + /* + * Get a list of addresses on this machine. + */ + snprintf(serv, sizeof(serv), "%hu", port); memset(&hints, 0, sizeof(struct addrinfo)); hints.ai_flags = AI_PASSIVE; hints.ai_family = AF_UNSPEC; hints.ai_socktype = SOCK_STREAM; - if (getaddrinfo(NULL /* any IP */, serv, &hints, &res) != 0) { - gaim_debug_warning("network", "getaddrinfo: %s\n", strerror(errno)); + errnum = getaddrinfo(NULL /* any IP */, serv, &hints, &res); + if (errnum != 0) { + gaim_debug_warning("network", "getaddrinfo: %s\n", gai_strerror(errnum)); + if (errnum == EAI_SYSTEM) + gaim_debug_warning("network", "getaddrinfo: system error: %s\n", strerror(errno)); return -1; } - ressave = res; - do { + + /* + * Go through the list of addresses and attempt to listen on + * one of them. + * XXX - Try IPv6 addresses first? + */ + for (next = res; next != NULL; next = next->ai_next) { listenfd = socket(res->ai_family, res->ai_socktype, res->ai_protocol); if (listenfd < 0) continue; - setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on)); + if (setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on)) != 0) + gaim_debug_warning("network", "setsockopt: %s\n", strerror(errno)); if (bind(listenfd, res->ai_addr, res->ai_addrlen) == 0) break; /* success */ close(listenfd); - } while ( (res = res->ai_next) ); + } - if (!res) - return -1; + freeaddrinfo(res); - freeaddrinfo(ressave); + if (next == NULL) + return -1; #else - int listenfd; - const int on = 1; struct sockaddr_in sockin; if ((listenfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) { @@ -189,14 +200,11 @@ return -1; } - if (setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, (char *)&on, sizeof(on)) != 0) { + if (setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on)) != 0) gaim_debug_warning("network", "setsockopt: %s\n", strerror(errno)); - close(listenfd); - return -1; - } memset(&sockin, 0, sizeof(struct sockaddr_in)); - sockin.sin_family = AF_INET; + sockin.sin_family = PF_INET; sockin.sin_port = htons(port); if (bind(listenfd, (struct sockaddr *)&sockin, sizeof(struct sockaddr_in)) != 0) { @@ -253,7 +261,7 @@ struct sockaddr_in addr; socklen_t len; - g_return_val_if_fail(fd > 0, 0); + g_return_val_if_fail(fd >= 0, 0); len = sizeof(addr); if (getsockname(fd, (struct sockaddr *) &addr, &len) == -1) {