[BACK]Return to mld6.c CVS log [TXT][DIR] Up to [cvs.NetBSD.org] / src / sys / netinet6

Please note that diffs are not public domain; they are subject to the copyright notices on the relevant files.

Diff for /src/sys/netinet6/mld6.c between version 1.31 and 1.38

version 1.31, 2006/03/05 23:47:08 version 1.38, 2007/05/23 17:15:03
Line 181  mld_init()
Line 181  mld_init()
         hbh_buf[3] = 0;          hbh_buf[3] = 0;
         hbh_buf[4] = IP6OPT_RTALERT;          hbh_buf[4] = IP6OPT_RTALERT;
         hbh_buf[5] = IP6OPT_RTALERT_LEN - 2;          hbh_buf[5] = IP6OPT_RTALERT_LEN - 2;
         bcopy((caddr_t)&rtalert_code, &hbh_buf[6], sizeof(u_int16_t));          bcopy((void *)&rtalert_code, &hbh_buf[6], sizeof(u_int16_t));
   
         ip6_opts.ip6po_hbh = hbh;          ip6_opts.ip6po_hbh = hbh;
         /* We will specify the hoplimit by a multicast option. */          /* We will specify the hoplimit by a multicast option. */
Line 189  mld_init()
Line 189  mld_init()
 }  }
   
 static void  static void
 mld_starttimer(in6m)  mld_starttimer(struct in6_multi *in6m)
         struct in6_multi *in6m;  
 {  {
         struct timeval now;          struct timeval now;
   
Line 209  mld_starttimer(in6m)
Line 208  mld_starttimer(in6m)
 }  }
   
 static void  static void
 mld_stoptimer(in6m)  mld_stoptimer(struct in6_multi *in6m)
         struct in6_multi *in6m;  
 {  {
         if (in6m->in6m_timer == IN6M_TIMER_UNDEF)          if (in6m->in6m_timer == IN6M_TIMER_UNDEF)
                 return;                  return;
Line 221  mld_stoptimer(in6m)
Line 219  mld_stoptimer(in6m)
 }  }
   
 static void  static void
 mld_timeo(in6m)  mld_timeo(struct in6_multi *in6m)
         struct in6_multi *in6m;  
 {  {
         int s = splsoftnet();          int s = splsoftnet();
   
Line 243  mld_timeo(in6m)
Line 240  mld_timeo(in6m)
 }  }
   
 static u_long  static u_long
 mld_timerresid(in6m)  mld_timerresid(struct in6_multi *in6m)
         struct in6_multi *in6m;  
 {  {
         struct timeval now, diff;          struct timeval now, diff;
   
Line 268  mld_timerresid(in6m)
Line 264  mld_timerresid(in6m)
 }  }
   
 static void  static void
 mld_start_listening(in6m)  mld_start_listening(struct in6_multi *in6m)
         struct in6_multi *in6m;  
 {  {
         struct in6_addr all_in6;          struct in6_addr all_in6;
   
Line 301  mld_start_listening(in6m)
Line 296  mld_start_listening(in6m)
 }  }
   
 static void  static void
 mld_stop_listening(in6m)  mld_stop_listening(struct in6_multi *in6m)
         struct in6_multi *in6m;  
 {  {
         struct in6_addr allnode, allrouter;          struct in6_addr allnode, allrouter;
   
Line 326  mld_stop_listening(in6m)
Line 320  mld_stop_listening(in6m)
 }  }
   
 void  void
 mld_input(m, off)  mld_input(struct mbuf *m, int off)
         struct mbuf *m;  
         int off;  
 {  {
         struct ip6_hdr *ip6 = mtod(m, struct ip6_hdr *);          struct ip6_hdr *ip6 = mtod(m, struct ip6_hdr *);
         struct mld_hdr *mldh;          struct mld_hdr *mldh;
Line 428  mld_input(m, off)
Line 420  mld_input(m, off)
                 if (ia == NULL)                  if (ia == NULL)
                         break;                          break;
   
                 for (in6m = ia->ia6_multiaddrs.lh_first;                  LIST_FOREACH(in6m, &ia->ia6_multiaddrs, in6m_entry) {
                      in6m;  
                      in6m = in6m->in6m_entry.le_next)  
                 {  
                         if (IN6_ARE_ADDR_EQUAL(&in6m->in6m_addr, &all_in6) ||                          if (IN6_ARE_ADDR_EQUAL(&in6m->in6m_addr, &all_in6) ||
                             IPV6_ADDR_MC_SCOPE(&in6m->in6m_addr) <                              IPV6_ADDR_MC_SCOPE(&in6m->in6m_addr) <
                             IPV6_ADDR_SCOPE_LINKLOCAL)                              IPV6_ADDR_SCOPE_LINKLOCAL)
Line 500  mld_input(m, off)
Line 489  mld_input(m, off)
 }  }
   
 static void  static void
 mld_sendpkt(in6m, type, dst)  mld_sendpkt(struct in6_multi *in6m, int type,
         struct in6_multi *in6m;          const struct in6_addr *dst)
         int type;  
         const struct in6_addr *dst;  
 {  {
         struct mbuf *mh;          struct mbuf *mh;
         struct mld_hdr *mldh;          struct mld_hdr *mldh;
Line 571  mld_sendpkt(in6m, type, dst)
Line 558  mld_sendpkt(in6m, type, dst)
 }  }
   
 static struct mld_hdr *  static struct mld_hdr *
 mld_allocbuf(mh, len, in6m, type)  mld_allocbuf(struct mbuf **mh, int len, struct in6_multi *in6m,
         struct mbuf **mh;      int type)
         int len;  
         struct in6_multi *in6m;  
         int type;  
 {  {
         struct mbuf *md;          struct mbuf *md;
         struct mld_hdr *mldh;          struct mld_hdr *mldh;
Line 626  mld_allocbuf(mh, len, in6m, type)
Line 610  mld_allocbuf(mh, len, in6m, type)
  * Add an address to the list of IP6 multicast addresses for a given interface.   * Add an address to the list of IP6 multicast addresses for a given interface.
  */   */
 struct  in6_multi *  struct  in6_multi *
 in6_addmulti(maddr6, ifp, errorp, delay)  in6_addmulti(struct in6_addr *maddr6, struct ifnet *ifp,
         struct in6_addr *maddr6;          int *errorp, int timer)
         struct ifnet *ifp;  
         int *errorp, delay;  
 {  {
         struct  in6_ifaddr *ia;          struct  in6_ifaddr *ia;
         struct  in6_ifreq ifr;          struct  in6_ifreq ifr;
Line 674  in6_addmulti(maddr6, ifp, errorp, delay)
Line 656  in6_addmulti(maddr6, ifp, errorp, delay)
                 }                  }
                 IFP_TO_IA6(ifp, ia);                  IFP_TO_IA6(ifp, ia);
                 if (ia == NULL) {                  if (ia == NULL) {
                           /* leaks in6m_timer_ch */
                         free(in6m, M_IPMADDR);                          free(in6m, M_IPMADDR);
                         splx(s);                          splx(s);
                         *errorp = EADDRNOTAVAIL; /* appropriate? */                          *errorp = EADDRNOTAVAIL; /* appropriate? */
Line 695  in6_addmulti(maddr6, ifp, errorp, delay)
Line 678  in6_addmulti(maddr6, ifp, errorp, delay)
                         *errorp = ENXIO; /* XXX: appropriate? */                          *errorp = ENXIO; /* XXX: appropriate? */
                 else                  else
                         *errorp = (*ifp->if_ioctl)(ifp, SIOCADDMULTI,                          *errorp = (*ifp->if_ioctl)(ifp, SIOCADDMULTI,
                             (caddr_t)&ifr);                              (void *)&ifr);
                 if (*errorp) {                  if (*errorp) {
                         LIST_REMOVE(in6m, in6m_entry);                          LIST_REMOVE(in6m, in6m_entry);
                           /* leaks in6m_timer_ch */
                         free(in6m, M_IPMADDR);                          free(in6m, M_IPMADDR);
                         IFAFREE(&ia->ia_ifa);                          IFAFREE(&ia->ia_ifa);
                         splx(s);                          splx(s);
Line 705  in6_addmulti(maddr6, ifp, errorp, delay)
Line 689  in6_addmulti(maddr6, ifp, errorp, delay)
                 }                  }
   
                 callout_init(in6m->in6m_timer_ch);                  callout_init(in6m->in6m_timer_ch);
                 in6m->in6m_timer = delay;                  in6m->in6m_timer = timer;
                 if (in6m->in6m_timer > 0) {                  if (in6m->in6m_timer > 0) {
                         in6m->in6m_state = MLD_REPORTPENDING;                          in6m->in6m_state = MLD_REPORTPENDING;
                         mld_starttimer(in6m);                          mld_starttimer(in6m);
Line 728  in6_addmulti(maddr6, ifp, errorp, delay)
Line 712  in6_addmulti(maddr6, ifp, errorp, delay)
  * Delete a multicast address record.   * Delete a multicast address record.
  */   */
 void  void
 in6_delmulti(in6m)  in6_delmulti(struct in6_multi *in6m)
         struct in6_multi *in6m;  
 {  {
         struct  in6_ifreq ifr;          struct  in6_ifreq ifr;
         struct  in6_ifaddr *ia;          struct  in6_ifaddr *ia;
Line 748  in6_delmulti(in6m)
Line 731  in6_delmulti(in6m)
                  * Unlink from list.                   * Unlink from list.
                  */                   */
                 LIST_REMOVE(in6m, in6m_entry);                  LIST_REMOVE(in6m, in6m_entry);
                 if (in6m->in6m_ia) {                  if (in6m->in6m_ia != NULL) {
                         IFAFREE(&in6m->in6m_ia->ia_ifa); /* release reference */                          IFAFREE(&in6m->in6m_ia->ia_ifa); /* release reference */
                           in6m->in6m_ia = NULL;
                 }                  }
   
                 /*                  /*
Line 758  in6_delmulti(in6m)
Line 742  in6_delmulti(in6m)
                  */                   */
                 for (ia = in6_ifaddr; ia; ia = ia->ia_next) {                  for (ia = in6_ifaddr; ia; ia = ia->ia_next) {
                         struct in6_multi_mship *imm;                          struct in6_multi_mship *imm;
                         LIST_FOREACH(imm, &ia->ia6_memberships,                          LIST_FOREACH(imm, &ia->ia6_memberships, i6mm_chain) {
                             i6mm_chain) {  
                                 if (imm->i6mm_maddr == in6m)                                  if (imm->i6mm_maddr == in6m)
                                         imm->i6mm_maddr = NULL;                                          imm->i6mm_maddr = NULL;
                         }                          }
Line 774  in6_delmulti(in6m)
Line 757  in6_delmulti(in6m)
                 ifr.ifr_addr.sin6_len = sizeof(struct sockaddr_in6);                  ifr.ifr_addr.sin6_len = sizeof(struct sockaddr_in6);
                 ifr.ifr_addr.sin6_addr = in6m->in6m_addr;                  ifr.ifr_addr.sin6_addr = in6m->in6m_addr;
                 (*in6m->in6m_ifp->if_ioctl)(in6m->in6m_ifp,                  (*in6m->in6m_ifp->if_ioctl)(in6m->in6m_ifp,
                     SIOCDELMULTI, (caddr_t)&ifr);                      SIOCDELMULTI, (void *)&ifr);
                 free(in6m->in6m_timer_ch, M_IPMADDR);                  free(in6m->in6m_timer_ch, M_IPMADDR);
                 free(in6m, M_IPMADDR);                  free(in6m, M_IPMADDR);
         }          }
Line 783  in6_delmulti(in6m)
Line 766  in6_delmulti(in6m)
   
   
 struct in6_multi_mship *  struct in6_multi_mship *
 in6_joingroup(ifp, addr, errorp, delay)  in6_joingroup(struct ifnet *ifp, struct in6_addr *addr,
         struct ifnet *ifp;          int *errorp, int timer)
         struct in6_addr *addr;  
         int *errorp, delay;  
 {  {
         struct in6_multi_mship *imm;          struct in6_multi_mship *imm;
   
Line 797  in6_joingroup(ifp, addr, errorp, delay)
Line 778  in6_joingroup(ifp, addr, errorp, delay)
         }          }
   
         memset(imm, 0, sizeof(*imm));          memset(imm, 0, sizeof(*imm));
         imm->i6mm_maddr = in6_addmulti(addr, ifp, errorp, delay);          imm->i6mm_maddr = in6_addmulti(addr, ifp, errorp, timer);
         if (!imm->i6mm_maddr) {          if (!imm->i6mm_maddr) {
                 /* *errorp is alrady set */                  /* *errorp is already set */
                 free(imm, M_IPMADDR);                  free(imm, M_IPMADDR);
                 return NULL;                  return NULL;
         }          }
Line 807  in6_joingroup(ifp, addr, errorp, delay)
Line 788  in6_joingroup(ifp, addr, errorp, delay)
 }  }
   
 int  int
 in6_leavegroup(imm)  in6_leavegroup(struct in6_multi_mship *imm)
         struct in6_multi_mship *imm;  
 {  {
   
         if (imm->i6mm_maddr) {          if (imm->i6mm_maddr) {
Line 826  in6_leavegroup(imm)
Line 806  in6_leavegroup(imm)
  * such time as this interface is reconfigured for IPv6.   * such time as this interface is reconfigured for IPv6.
  */   */
 void  void
 in6_savemkludge(oia)  in6_savemkludge(struct in6_ifaddr *oia)
         struct in6_ifaddr *oia;  
 {  {
         struct in6_ifaddr *ia;          struct in6_ifaddr *ia;
         struct in6_multi *in6m, *next;          struct in6_multi *in6m;
   
         IFP_TO_IA6(oia->ia_ifp, ia);          IFP_TO_IA6(oia->ia_ifp, ia);
         if (ia) {       /* there is another address */          if (ia) {       /* there is another address */
                 for (in6m = oia->ia6_multiaddrs.lh_first; in6m; in6m = next) {                  KASSERT(ia != oia);
                         next = in6m->in6m_entry.le_next;                  while ((in6m = LIST_FIRST(&oia->ia6_multiaddrs)) != NULL) {
                         IFAFREE(&in6m->in6m_ia->ia_ifa);                          LIST_REMOVE(in6m, in6m_entry);
                         IFAREF(&ia->ia_ifa);                          IFAREF(&ia->ia_ifa);
                           IFAFREE(&in6m->in6m_ia->ia_ifa);
                         in6m->in6m_ia = ia;                          in6m->in6m_ia = ia;
                         LIST_INSERT_HEAD(&ia->ia6_multiaddrs, in6m, in6m_entry);                          LIST_INSERT_HEAD(&ia->ia6_multiaddrs, in6m, in6m_entry);
                 }                  }
         } else {        /* last address on this if deleted, save */          } else {        /* last address on this if deleted, save */
                 struct multi6_kludge *mk;                  struct multi6_kludge *mk;
   
                 for (mk = in6_mk.lh_first; mk; mk = mk->mk_entry.le_next) {                  LIST_FOREACH(mk, &in6_mk, mk_entry) {
                         if (mk->mk_ifp == oia->ia_ifp)                          if (mk->mk_ifp == oia->ia_ifp)
                                 break;                                  break;
                 }                  }
                 if (mk == NULL) /* this should not happen! */                  if (mk == NULL) /* this should not happen! */
                         panic("in6_savemkludge: no kludge space");                          panic("in6_savemkludge: no kludge space");
   
                 for (in6m = oia->ia6_multiaddrs.lh_first; in6m; in6m = next) {                  while ((in6m = LIST_FIRST(&oia->ia6_multiaddrs)) != NULL) {
                         next = in6m->in6m_entry.le_next;                          LIST_REMOVE(in6m, in6m_entry);
                         IFAFREE(&in6m->in6m_ia->ia_ifa); /* release reference */                          IFAFREE(&in6m->in6m_ia->ia_ifa); /* release reference */
                         in6m->in6m_ia = NULL;                          in6m->in6m_ia = NULL;
                         LIST_INSERT_HEAD(&mk->mk_head, in6m, in6m_entry);                          LIST_INSERT_HEAD(&mk->mk_head, in6m, in6m_entry);
Line 866  in6_savemkludge(oia)
Line 846  in6_savemkludge(oia)
  * then we re-attach it to the first address configured on the i/f.   * then we re-attach it to the first address configured on the i/f.
  */   */
 void  void
 in6_restoremkludge(ia, ifp)  in6_restoremkludge(struct in6_ifaddr *ia, struct ifnet *ifp)
         struct in6_ifaddr *ia;  
         struct ifnet *ifp;  
 {  {
         struct multi6_kludge *mk;          struct multi6_kludge *mk;
           struct in6_multi *in6m;
   
         for (mk = in6_mk.lh_first; mk; mk = mk->mk_entry.le_next) {          LIST_FOREACH(mk, &in6_mk, mk_entry) {
                 if (mk->mk_ifp == ifp) {                  if (mk->mk_ifp == ifp)
                         struct in6_multi *in6m, *next;  
   
                         for (in6m = mk->mk_head.lh_first; in6m; in6m = next) {  
                                 next = in6m->in6m_entry.le_next;  
                                 in6m->in6m_ia = ia;  
                                 IFAREF(&ia->ia_ifa);  
                                 LIST_INSERT_HEAD(&ia->ia6_multiaddrs,  
                                                  in6m, in6m_entry);  
                         }  
                         LIST_INIT(&mk->mk_head);  
                         break;                          break;
                 }          }
           if (mk == NULL)
                   return;
           while ((in6m = LIST_FIRST(&mk->mk_head)) != NULL) {
                   LIST_REMOVE(in6m, in6m_entry);
                   in6m->in6m_ia = ia;
                   IFAREF(&ia->ia_ifa);
                   LIST_INSERT_HEAD(&ia->ia6_multiaddrs, in6m, in6m_entry);
         }          }
 }  }
   
Line 901  in6_restoremkludge(ia, ifp)
Line 877  in6_restoremkludge(ia, ifp)
  * it is a global function.   * it is a global function.
  */   */
 void  void
 in6_createmkludge(ifp)  in6_createmkludge(struct ifnet *ifp)
         struct ifnet *ifp;  
 {  {
         struct multi6_kludge *mk;          struct multi6_kludge *mk;
   
         for (mk = in6_mk.lh_first; mk; mk = mk->mk_entry.le_next) {          LIST_FOREACH(mk, &in6_mk, mk_entry) {
                 /* If we've already had one, do not allocate. */                  /* If we've already had one, do not allocate. */
                 if (mk->mk_ifp == ifp)                  if (mk->mk_ifp == ifp)
                         return;                          return;
Line 921  in6_createmkludge(ifp)
Line 896  in6_createmkludge(ifp)
 }  }
   
 void  void
 in6_purgemkludge(ifp)  in6_purgemkludge(struct ifnet *ifp)
         struct ifnet *ifp;  
 {  {
         struct multi6_kludge *mk;          struct multi6_kludge *mk;
         struct in6_multi *in6m;          struct in6_multi *in6m, *next;
   
         for (mk = in6_mk.lh_first; mk; mk = mk->mk_entry.le_next) {          LIST_FOREACH(mk, &in6_mk, mk_entry) {
                 if (mk->mk_ifp != ifp)                  if (mk->mk_ifp == ifp)
                         continue;                          break;
           }
                 /* leave from all multicast groups joined */          if (mk == NULL)
                 while ((in6m = LIST_FIRST(&mk->mk_head)) != NULL) {                  return;
                         in6_delmulti(in6m);  
                 }          /* leave from all multicast groups joined */
                 LIST_REMOVE(mk, mk_entry);          for (in6m = LIST_FIRST(&mk->mk_head); in6m != NULL; in6m = next) {
                 free(mk, M_IPMADDR);                  next = LIST_NEXT(in6m, in6m_entry);
                 break;                  in6_delmulti(in6m);
         }          }
           LIST_REMOVE(mk, mk_entry);
           free(mk, M_IPMADDR);
 }  }

Legend:
Removed from v.1.31  
changed lines
  Added in v.1.38

CVSweb <webmaster@jp.NetBSD.org>