Loading src/common/address.c +35 −18 Original line number Original line Diff line number Diff line Loading @@ -1374,22 +1374,33 @@ get_interface_addresses_win32(int severity) #ifdef HAVE_IFCONF_TO_SMARTLIST #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 */ /* This is defined on Mac OS X */ #ifndef _SIZEOF_ADDR_IFREQ #ifndef _SIZEOF_ADDR_IFREQ #define _SIZEOF_ADDR_IFREQ sizeof #define _SIZEOF_ADDR_IFREQ sizeof #endif #endif /** Convert <b>*ifr</b>, an ifreq structure array of size <b>buflen</b> /** Convert <b>*buf</b>, an ifreq structure array of size <b>buflen</b>, * into smartlist of <b>tor_addr_t</b> structures. * into smartlist of <b>tor_addr_t</b> structures. */ */ STATIC smartlist_t * STATIC smartlist_t * ifreq_to_smartlist(const struct ifreq *ifr, size_t buflen) ifreq_to_smartlist(char *buf, size_t buflen) { { smartlist_t *result = smartlist_new(); smartlist_t *result = smartlist_new(); char *end = buf + buflen; /* These acrobatics are due to alignment issues which trigger * undefined behaviour traps on OSX. */ struct ifreq *r = tor_malloc(IFREQ_SIZE); struct ifreq *r = (struct ifreq *)ifr; while (buf < end) { /* Copy up to IFREQ_SIZE bytes into the struct ifreq, but don't overrun * buf. */ memcpy(r, buf, end - buf < IFREQ_SIZE ? end - buf : IFREQ_SIZE); while ((char *)r < (char *)ifr+buflen) { const struct sockaddr *sa = &r->ifr_addr; const struct sockaddr *sa = &r->ifr_addr; tor_addr_t tmp; tor_addr_t tmp; int valid_sa_family = (sa->sa_family == AF_INET || int valid_sa_family = (sa->sa_family == AF_INET || Loading @@ -1400,9 +1411,10 @@ ifreq_to_smartlist(const struct ifreq *ifr, size_t buflen) if (valid_sa_family && conversion_success) if (valid_sa_family && conversion_success) smartlist_add(result, tor_memdup(&tmp, sizeof(tmp))); smartlist_add(result, tor_memdup(&tmp, sizeof(tmp))); r = (struct ifreq *)((char *)r + _SIZEOF_ADDR_IFREQ(*r)); buf += _SIZEOF_ADDR_IFREQ(*r); } } tor_free(r); return result; return result; } } Loading @@ -1415,8 +1427,7 @@ get_interface_addresses_ioctl(int severity) { { /* Some older unixy systems make us use ioctl(SIOCGIFCONF) */ /* Some older unixy systems make us use ioctl(SIOCGIFCONF) */ struct ifconf ifc; struct ifconf ifc; int fd, sz; int fd; void *databuf = NULL; smartlist_t *result = NULL; smartlist_t *result = NULL; /* This interface, AFAICT, only supports AF_INET addresses */ /* This interface, AFAICT, only supports AF_INET addresses */ Loading @@ -1426,22 +1437,28 @@ get_interface_addresses_ioctl(int severity) goto done; goto done; } } /* Guess how much space we need. */ int mult = 1; ifc.ifc_len = sz = 4096; ifc.ifc_buf = NULL; databuf = tor_malloc_zero(sz); do { ifc.ifc_buf = databuf; mult *= 2; ifc.ifc_len = mult * IFREQ_SIZE; ifc.ifc_buf = tor_realloc(ifc.ifc_buf, ifc.ifc_len); tor_assert(ifc.ifc_buf); if (ioctl(fd, SIOCGIFCONF, &ifc) < 0) { if (ioctl(fd, SIOCGIFCONF, &ifc) < 0) { tor_log(severity, LD_NET, "ioctl failed: %s", strerror(errno)); tor_log(severity, LD_NET, "ioctl failed: %s", strerror(errno)); close(fd); close(fd); goto done; goto done; } } /* Ensure we have least IFREQ_SIZE bytes unused at the end. Otherwise, we result = ifreq_to_smartlist(databuf, ifc.ifc_len); * 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: done: close(fd); close(fd); tor_free(databuf); tor_free(ifc.ifc_buf); return result; return result; } } #endif #endif Loading src/common/address.h +1 −1 Original line number Original line Diff line number Diff line Loading @@ -287,7 +287,7 @@ STATIC smartlist_t *get_interface_addresses_win32(int severity); #endif #endif #ifdef HAVE_IFCONF_TO_SMARTLIST #ifdef HAVE_IFCONF_TO_SMARTLIST STATIC smartlist_t *ifreq_to_smartlist(const struct ifreq *ifr, STATIC smartlist_t *ifreq_to_smartlist(char *ifr, size_t buflen); size_t buflen); STATIC smartlist_t *get_interface_addresses_ioctl(int severity); STATIC smartlist_t *get_interface_addresses_ioctl(int severity); #endif #endif Loading src/test/test_address.c +2 −2 Original line number Original line Diff line number Diff line Loading @@ -376,7 +376,7 @@ test_address_ifreq_to_smartlist(void *arg) ifc->ifc_len = sizeof(struct ifreq); ifc->ifc_len = sizeof(struct ifreq); ifc->ifc_ifcu.ifcu_req = ifr; ifc->ifc_ifcu.ifcu_req = ifr; results = ifreq_to_smartlist((struct ifreq *)ifc->ifc_buf,ifc->ifc_len); results = ifreq_to_smartlist(ifc->ifc_buf,ifc->ifc_len); tt_int_op(smartlist_len(results),==,1); tt_int_op(smartlist_len(results),==,1); tor_addr = smartlist_get(results, 0); tor_addr = smartlist_get(results, 0); Loading @@ -399,7 +399,7 @@ test_address_ifreq_to_smartlist(void *arg) SMARTLIST_FOREACH(results, tor_addr_t *, t, tor_free(t)); SMARTLIST_FOREACH(results, tor_addr_t *, t, tor_free(t)); smartlist_free(results); smartlist_free(results); results = ifreq_to_smartlist((struct ifreq *)ifc->ifc_buf,ifc->ifc_len); results = ifreq_to_smartlist(ifc->ifc_buf,ifc->ifc_len); tt_int_op(smartlist_len(results),==,2); tt_int_op(smartlist_len(results),==,2); tor_addr = smartlist_get(results, 0); tor_addr = smartlist_get(results, 0); Loading Loading
src/common/address.c +35 −18 Original line number Original line Diff line number Diff line Loading @@ -1374,22 +1374,33 @@ get_interface_addresses_win32(int severity) #ifdef HAVE_IFCONF_TO_SMARTLIST #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 */ /* This is defined on Mac OS X */ #ifndef _SIZEOF_ADDR_IFREQ #ifndef _SIZEOF_ADDR_IFREQ #define _SIZEOF_ADDR_IFREQ sizeof #define _SIZEOF_ADDR_IFREQ sizeof #endif #endif /** Convert <b>*ifr</b>, an ifreq structure array of size <b>buflen</b> /** Convert <b>*buf</b>, an ifreq structure array of size <b>buflen</b>, * into smartlist of <b>tor_addr_t</b> structures. * into smartlist of <b>tor_addr_t</b> structures. */ */ STATIC smartlist_t * STATIC smartlist_t * ifreq_to_smartlist(const struct ifreq *ifr, size_t buflen) ifreq_to_smartlist(char *buf, size_t buflen) { { smartlist_t *result = smartlist_new(); smartlist_t *result = smartlist_new(); char *end = buf + buflen; /* These acrobatics are due to alignment issues which trigger * undefined behaviour traps on OSX. */ struct ifreq *r = tor_malloc(IFREQ_SIZE); struct ifreq *r = (struct ifreq *)ifr; while (buf < end) { /* Copy up to IFREQ_SIZE bytes into the struct ifreq, but don't overrun * buf. */ memcpy(r, buf, end - buf < IFREQ_SIZE ? end - buf : IFREQ_SIZE); while ((char *)r < (char *)ifr+buflen) { const struct sockaddr *sa = &r->ifr_addr; const struct sockaddr *sa = &r->ifr_addr; tor_addr_t tmp; tor_addr_t tmp; int valid_sa_family = (sa->sa_family == AF_INET || int valid_sa_family = (sa->sa_family == AF_INET || Loading @@ -1400,9 +1411,10 @@ ifreq_to_smartlist(const struct ifreq *ifr, size_t buflen) if (valid_sa_family && conversion_success) if (valid_sa_family && conversion_success) smartlist_add(result, tor_memdup(&tmp, sizeof(tmp))); smartlist_add(result, tor_memdup(&tmp, sizeof(tmp))); r = (struct ifreq *)((char *)r + _SIZEOF_ADDR_IFREQ(*r)); buf += _SIZEOF_ADDR_IFREQ(*r); } } tor_free(r); return result; return result; } } Loading @@ -1415,8 +1427,7 @@ get_interface_addresses_ioctl(int severity) { { /* Some older unixy systems make us use ioctl(SIOCGIFCONF) */ /* Some older unixy systems make us use ioctl(SIOCGIFCONF) */ struct ifconf ifc; struct ifconf ifc; int fd, sz; int fd; void *databuf = NULL; smartlist_t *result = NULL; smartlist_t *result = NULL; /* This interface, AFAICT, only supports AF_INET addresses */ /* This interface, AFAICT, only supports AF_INET addresses */ Loading @@ -1426,22 +1437,28 @@ get_interface_addresses_ioctl(int severity) goto done; goto done; } } /* Guess how much space we need. */ int mult = 1; ifc.ifc_len = sz = 4096; ifc.ifc_buf = NULL; databuf = tor_malloc_zero(sz); do { ifc.ifc_buf = databuf; mult *= 2; ifc.ifc_len = mult * IFREQ_SIZE; ifc.ifc_buf = tor_realloc(ifc.ifc_buf, ifc.ifc_len); tor_assert(ifc.ifc_buf); if (ioctl(fd, SIOCGIFCONF, &ifc) < 0) { if (ioctl(fd, SIOCGIFCONF, &ifc) < 0) { tor_log(severity, LD_NET, "ioctl failed: %s", strerror(errno)); tor_log(severity, LD_NET, "ioctl failed: %s", strerror(errno)); close(fd); close(fd); goto done; goto done; } } /* Ensure we have least IFREQ_SIZE bytes unused at the end. Otherwise, we result = ifreq_to_smartlist(databuf, ifc.ifc_len); * 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: done: close(fd); close(fd); tor_free(databuf); tor_free(ifc.ifc_buf); return result; return result; } } #endif #endif Loading
src/common/address.h +1 −1 Original line number Original line Diff line number Diff line Loading @@ -287,7 +287,7 @@ STATIC smartlist_t *get_interface_addresses_win32(int severity); #endif #endif #ifdef HAVE_IFCONF_TO_SMARTLIST #ifdef HAVE_IFCONF_TO_SMARTLIST STATIC smartlist_t *ifreq_to_smartlist(const struct ifreq *ifr, STATIC smartlist_t *ifreq_to_smartlist(char *ifr, size_t buflen); size_t buflen); STATIC smartlist_t *get_interface_addresses_ioctl(int severity); STATIC smartlist_t *get_interface_addresses_ioctl(int severity); #endif #endif Loading
src/test/test_address.c +2 −2 Original line number Original line Diff line number Diff line Loading @@ -376,7 +376,7 @@ test_address_ifreq_to_smartlist(void *arg) ifc->ifc_len = sizeof(struct ifreq); ifc->ifc_len = sizeof(struct ifreq); ifc->ifc_ifcu.ifcu_req = ifr; ifc->ifc_ifcu.ifcu_req = ifr; results = ifreq_to_smartlist((struct ifreq *)ifc->ifc_buf,ifc->ifc_len); results = ifreq_to_smartlist(ifc->ifc_buf,ifc->ifc_len); tt_int_op(smartlist_len(results),==,1); tt_int_op(smartlist_len(results),==,1); tor_addr = smartlist_get(results, 0); tor_addr = smartlist_get(results, 0); Loading @@ -399,7 +399,7 @@ test_address_ifreq_to_smartlist(void *arg) SMARTLIST_FOREACH(results, tor_addr_t *, t, tor_free(t)); SMARTLIST_FOREACH(results, tor_addr_t *, t, tor_free(t)); smartlist_free(results); smartlist_free(results); results = ifreq_to_smartlist((struct ifreq *)ifc->ifc_buf,ifc->ifc_len); results = ifreq_to_smartlist(ifc->ifc_buf,ifc->ifc_len); tt_int_op(smartlist_len(results),==,2); tt_int_op(smartlist_len(results),==,2); tor_addr = smartlist_get(results, 0); tor_addr = smartlist_get(results, 0); Loading