summaryrefslogtreecommitdiff
path: root/sys/net/if_gre.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/net/if_gre.c')
-rw-r--r--sys/net/if_gre.c50
1 files changed, 37 insertions, 13 deletions
diff --git a/sys/net/if_gre.c b/sys/net/if_gre.c
index fc20588a14c..f84091156a1 100644
--- a/sys/net/if_gre.c
+++ b/sys/net/if_gre.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_gre.c,v 1.32 2004/08/18 18:37:53 canacar Exp $ */
+/* $OpenBSD: if_gre.c,v 1.33 2005/05/14 19:24:23 brad Exp $ */
/* $NetBSD: if_gre.c,v 1.9 1999/10/25 19:18:11 drochner Exp $ */
/*
@@ -8,6 +8,8 @@
* This code is derived from software contributed to The NetBSD Foundation
* by Heiko W.Rupp <hwr@pilhuhn.de>
*
+ * IPv6-over-GRE contributed by Gert Doering <gert@greenie.muc.de>
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
@@ -93,11 +95,12 @@
#define GRE_RECURSION_LIMIT 3 /* How many levels of recursion allowed */
#endif /* GRE_RECURSION_LIMIT */
-#define GREMTU 1450 /* XXX this is below the standard MTU of
- 1500 Bytes, allowing for headers,
- but we should possibly do path mtu discovery
- before changing if state to up to find the
- correct value */
+/*
+ * It is not easy to calculate the right value for a GRE MTU.
+ * We leave this task to the admin and use the same default that
+ * other vendors use.
+ */
+#define GREMTU 1476
int gre_clone_create(struct if_clone *, int);
int gre_clone_destroy(struct ifnet *);
@@ -141,7 +144,7 @@ gre_clone_create(struct if_clone *ifc, int unit)
snprintf(sc->sc_if.if_xname, sizeof sc->sc_if.if_xname, "%s%d",
ifc->ifc_name, unit);
sc->sc_if.if_softc = sc;
- sc->sc_if.if_type = IFT_OTHER;
+ sc->sc_if.if_type = IFT_TUNNEL;
sc->sc_if.if_addrlen = 0;
sc->sc_if.if_hdrlen = 24; /* IP + GRE */
sc->sc_if.if_mtu = GREMTU;
@@ -203,7 +206,8 @@ gre_output(struct ifnet *ifp, struct mbuf *m, struct sockaddr *dst,
struct gre_softc *sc = (struct gre_softc *) (ifp->if_softc);
struct greip *gh = NULL;
struct ip *inp = NULL;
- u_short etype = 0;
+ u_int8_t ip_tos = 0;
+ u_int16_t etype = 0;
struct mobile_h mob_h;
struct m_tag *mtag;
@@ -314,7 +318,7 @@ gre_output(struct ifnet *ifp, struct mbuf *m, struct sockaddr *dst,
}
HTONS(mob_h.proto);
- mob_h.hcrc = gre_in_cksum((u_short *) &mob_h, msiz);
+ mob_h.hcrc = gre_in_cksum((u_int16_t *) &mob_h, msiz);
/* Squeeze in the mobility header */
if ((m->m_data - msiz) < m->m_pktdat) {
@@ -377,6 +381,7 @@ gre_output(struct ifnet *ifp, struct mbuf *m, struct sockaddr *dst,
}
inp = mtod(m, struct ip *);
+ ip_tos = inp->ip_tos;
etype = ETHERTYPE_IP;
break;
#ifdef NETATALK
@@ -389,6 +394,11 @@ gre_output(struct ifnet *ifp, struct mbuf *m, struct sockaddr *dst,
etype = ETHERTYPE_NS;
break;
#endif
+#ifdef INET6
+ case AF_INET6:
+ etype = ETHERTYPE_IPV6;
+ break;
+#endif
default:
IF_DROP(&ifp->if_snd);
m_freem(m);
@@ -424,7 +434,7 @@ gre_output(struct ifnet *ifp, struct mbuf *m, struct sockaddr *dst,
gh->gi_dst = sc->g_dst;
((struct ip *) gh)->ip_hl = (sizeof(struct ip)) >> 2;
((struct ip *) gh)->ip_ttl = ip_defttl;
- ((struct ip *) gh)->ip_tos = inp->ip_tos;
+ ((struct ip *) gh)->ip_tos = ip_tos;
gh->gi_len = htons(m->m_pkthdr.len);
}
@@ -465,6 +475,16 @@ gre_ioctl(struct ifnet *ifp, u_long cmd, caddr_t data)
else
sc->g_proto = IPPROTO_MOBILE;
break;
+ case SIOCSIFMTU:
+ if (ifr->ifr_mtu < 576) {
+ error = EINVAL;
+ break;
+ }
+ ifp->if_mtu = ifr->ifr_mtu;
+ break;
+ case SIOCGIFMTU:
+ ifr->ifr_mtu = sc->sc_if.if_mtu;
+ break;
case SIOCADDMULTI:
case SIOCDELMULTI:
if (ifr == 0) {
@@ -476,6 +496,10 @@ gre_ioctl(struct ifnet *ifp, u_long cmd, caddr_t data)
case AF_INET:
break;
#endif
+#ifdef INET6
+ case AF_INET6:
+ break;
+#endif
default:
error = EAFNOSUPPORT;
break;
@@ -665,10 +689,10 @@ gre_compute_route(struct gre_softc *sc)
* do a checksum of a buffer - much like in_cksum, which operates on
* mbufs.
*/
-u_short
-gre_in_cksum(u_short *p, u_int len)
+u_int16_t
+gre_in_cksum(u_int16_t *p, u_int len)
{
- u_int sum = 0;
+ u_int32_t sum = 0;
int nwords = len >> 1;
while (nwords-- != 0)