diff options
Diffstat (limited to 'sys/net')
-rw-r--r-- | sys/net/if_gif.c | 34 |
1 files changed, 21 insertions, 13 deletions
diff --git a/sys/net/if_gif.c b/sys/net/if_gif.c index b6d58459a11..a048d13b077 100644 --- a/sys/net/if_gif.c +++ b/sys/net/if_gif.c @@ -1,4 +1,4 @@ -/* $OpenBSD: if_gif.c,v 1.18 2001/06/15 03:38:34 itojun Exp $ */ +/* $OpenBSD: if_gif.c,v 1.19 2001/06/25 06:33:48 angelos Exp $ */ /* $KAME: if_gif.c,v 1.43 2001/02/20 08:51:07 itojun Exp $ */ /* @@ -148,24 +148,33 @@ gif_output(ifp, m, dst, rt) { register struct gif_softc *sc = (struct gif_softc*)ifp; int error = 0; - static int called = 0; /* XXX: MUTEX */ - int calllimit = 3; /* XXX: adhoc */ + struct m_tag *mtag; /* * gif may cause infinite recursion calls when misconfigured. - * We'll prevent this by introducing upper limit. - * XXX: this mechanism may introduce another problem about - * mutual exclusion of the variable CALLED, especially if we - * use kernel thread. + * We'll prevent this by detecting loops. */ - if (++called >= calllimit) { - log(LOG_NOTICE, - "gif_output: recursively called too many times(%d)\n", - called); + for (mtag = m_tag_find(m, PACKET_TAG_GIF, NULL); mtag; + mtag = m_tag_find(m, PACKET_TAG_GIF, mtag)) { + if (!bcmp((caddr_t)(mtag + 1), &ifp, sizeof(struct ifnet *))) { + IF_DROP(&ifp->if_snd); + log(LOG_NOTICE, + "gif_output: recursively called too many times\n"); + m_freem(m); + error = EIO; /* is there better errno? */ + goto end; + } + } + + mtag = m_tag_get(PACKET_TAG_GIF, sizeof(struct ifnet *), M_NOWAIT); + if (mtag == NULL) { + IF_DROP(&ifp->if_snd); m_freem(m); - error = EIO; /* is there better errno? */ + error = ENOMEM; goto end; } + bcopy(&ifp, (caddr_t)(mtag + 1), sizeof(struct ifnet *)); + m_tag_prepend(m, mtag); m->m_flags &= ~(M_BCAST|M_MCAST); if (!(ifp->if_flags & IFF_UP) || @@ -217,7 +226,6 @@ gif_output(ifp, m, dst, rt) } end: - called = 0; /* reset recursion counter */ if (error) ifp->if_oerrors++; return error; } |