diff options
author | Sebastian Hahn <sebastian@torproject.org> | 2015-02-12 13:59:31 +0100 |
---|---|---|
committer | Sebastian Hahn <sebastian@torproject.org> | 2015-02-12 14:54:39 +0100 |
commit | 87a95b02363508dd6d9ce4094801c55d3b364ebf (patch) | |
tree | a5d52ac0f3e71a3189dbb5c4b2ed68b2426f9f84 /src/common/address.c | |
parent | 9667b2b88a1f8e9af08704c8bc95b8399a6e4145 (diff) | |
download | tor-87a95b02363508dd6d9ce4094801c55d3b364ebf.tar.gz tor-87a95b02363508dd6d9ce4094801c55d3b364ebf.zip |
Actually get all interface addresses
If we guessed a buffer size too small, we never increased the buffer and
tried again
Also simplify the interface of ifreq_to_smartlist a little
Diffstat (limited to 'src/common/address.c')
-rw-r--r-- | src/common/address.c | 37 |
1 files changed, 23 insertions, 14 deletions
diff --git a/src/common/address.c b/src/common/address.c index 3b4be1d601..8cdcd84945 100644 --- a/src/common/address.c +++ b/src/common/address.c @@ -1374,6 +1374,10 @@ get_interface_addresses_win32(int severity) #ifdef HAVE_IFCONF_TO_SMARTLIST +/* Guess how much space we need. There shouldn't be any struct ifreqs + * larger than this, even on OS X where the struct's size is dynamic. */ +#define IFREQ_SIZE 4096 + /* This is defined on Mac OS X */ #ifndef _SIZEOF_ADDR_IFREQ #define _SIZEOF_ADDR_IFREQ sizeof @@ -1383,7 +1387,7 @@ get_interface_addresses_win32(int severity) * into smartlist of <b>tor_addr_t</b> structures. */ STATIC smartlist_t * -ifreq_to_smartlist(const struct ifreq *ifr, size_t buflen) +ifreq_to_smartlist(char *ifr, size_t buflen) { smartlist_t *result = smartlist_new(); @@ -1415,8 +1419,7 @@ get_interface_addresses_ioctl(int severity) { /* Some older unixy systems make us use ioctl(SIOCGIFCONF) */ struct ifconf ifc; - int fd, sz; - void *databuf = NULL; + int fd; smartlist_t *result = NULL; /* This interface, AFAICT, only supports AF_INET addresses */ @@ -1426,22 +1429,28 @@ get_interface_addresses_ioctl(int severity) goto done; } - /* Guess how much space we need. */ - ifc.ifc_len = sz = 4096; - databuf = tor_malloc_zero(sz); - ifc.ifc_buf = databuf; + int mult = 1; + ifc.ifc_buf = NULL; + do { + mult *= 2; + ifc.ifc_len = mult * IFREQ_SIZE; + ifc.ifc_buf = tor_realloc(ifc.ifc_buf, ifc.ifc_len); - if (ioctl(fd, SIOCGIFCONF, &ifc) < 0) { - tor_log(severity, LD_NET, "ioctl failed: %s", strerror(errno)); - close(fd); - goto done; - } + tor_assert(ifc.ifc_buf); - result = ifreq_to_smartlist(databuf, ifc.ifc_len); + if (ioctl(fd, SIOCGIFCONF, &ifc) < 0) { + tor_log(severity, LD_NET, "ioctl failed: %s", strerror(errno)); + close(fd); + goto done; + } + /* Ensure we have least IFREQ_SIZE bytes unused at the end. Otherwise, we + * don't know if we got everything during ioctl. */ + } while (mult * IFREQ_SIZE - ifc.ifc_len <= IFREQ_SIZE); + result = ifreq_to_smartlist(ifc.ifc_buf, ifc.ifc_len); done: close(fd); - tor_free(databuf); + tor_free(ifc.ifc_buf); return result; } #endif |