mirror of
https://gitlab.torproject.org/tpo/core/tor.git
synced 2024-11-14 07:03:44 +01:00
Merge remote-tracking branch 'sebastian/bug14875'
This commit is contained in:
commit
5644d92dd7
@ -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;
|
||||||
|
|
||||||
struct ifreq *r = (struct ifreq *)ifr;
|
/* These acrobatics are due to alignment issues which trigger
|
||||||
|
* undefined behaviour traps on OSX. */
|
||||||
|
struct ifreq *r = tor_malloc(IFREQ_SIZE);
|
||||||
|
|
||||||
|
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 ||
|
||||||
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -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 */
|
||||||
@ -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
|
||||||
|
@ -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
|
||||||
|
@ -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);
|
||||||
@ -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…
Reference in New Issue
Block a user