Commit 5644d92d authored by Nick Mathewson's avatar Nick Mathewson 🦀
Browse files

Merge remote-tracking branch 'sebastian/bug14875'

parents f794a091 c8ce973d
Loading
Loading
Loading
Loading
+35 −18
Original line number Original line Diff line number Diff line
@@ -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 ||
@@ -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
+1 −1
Original line number Original line Diff line number Diff line
@@ -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
+2 −2
Original line number Original line Diff line number Diff line
@@ -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);