diff options
author | Niklas Hallqvist <niklas@cvs.openbsd.org> | 1999-07-15 14:15:42 +0000 |
---|---|---|
committer | Niklas Hallqvist <niklas@cvs.openbsd.org> | 1999-07-15 14:15:42 +0000 |
commit | 893ba3a842c40c7f5d6f9f14837e0d432dd7767f (patch) | |
tree | b143d604f311079eb8a3f83241ef170196a8f67a /sys/netinet | |
parent | 7bdb8bc6dd617efd36012f956c9992b1cfa8e85c (diff) |
From angelos@, edits by me, demand keying for PF_KEY
Diffstat (limited to 'sys/netinet')
-rw-r--r-- | sys/netinet/ip_ipsp.c | 59 | ||||
-rw-r--r-- | sys/netinet/ip_ipsp.h | 7 | ||||
-rw-r--r-- | sys/netinet/ip_output.c | 41 |
3 files changed, 76 insertions, 31 deletions
diff --git a/sys/netinet/ip_ipsp.c b/sys/netinet/ip_ipsp.c index 54d22e3d5f6..111afee9b8c 100644 --- a/sys/netinet/ip_ipsp.c +++ b/sys/netinet/ip_ipsp.c @@ -1,4 +1,4 @@ -/* $OpenBSD: ip_ipsp.c,v 1.48 1999/07/06 20:54:03 ho Exp $ */ +/* $OpenBSD: ip_ipsp.c,v 1.49 1999/07/15 14:15:41 niklas Exp $ */ /* * The authors of this code are John Ioannidis (ji@tla.org), @@ -243,7 +243,8 @@ check_ipsec_policy(struct inpcb *inp, u_int32_t daddr) gw = (struct sockaddr_encap *) (re->re_rt->rt_gateway); - if (gw->sen_type == SENT_IPSP) { + if (gw->sen_type == SENT_IPSP) + { bzero(&sunion, sizeof(sunion)); sunion.sin.sin_family = AF_INET; sunion.sin.sin_len = sizeof(struct sockaddr_in); @@ -337,9 +338,9 @@ tdb_add_inp(struct tdb *tdb, struct inpcb *inp) } /* - * Reserve an SPI; the SA is not valid yet though. Zero is reserved as - * an error return value. If tspi is not zero, we try to allocate that - * SPI. + * Reserve an SPI; the SA is not valid yet though. We use SPI_LOCAL_USE as + * an error return value. It'll not be a problem that we also use that + * for demand-keying as that is manually specified. */ u_int32_t @@ -350,33 +351,35 @@ reserve_spi(u_int32_t sspi, u_int32_t tspi, union sockaddr_union *src, u_int32_t spi; int nums; - if (tspi <= 255) /* We don't reserve 0 < SPI <= 255 */ + /* Don't accept ranges only encompassing reserved SPIs. */ + if (tspi < sspi || tspi <= SPI_RESERVED_MAX) { - (*errval) = EEXIST; + (*errval) = EINVAL; return 0; } - - if ((sspi == tspi) && (sspi != 0)) /* Asking for a specific SPI */ + + /* Limit the range to not include reserved areas. */ + if (sspi <= SPI_RESERVED_MAX) + sspi = SPI_RESERVED_MAX + 1; + + if (sspi == tspi) /* Asking for a specific SPI */ nums = 1; else nums = 50; /* XXX figure out some good value */ while (nums--) { - if (tspi != 0) /* SPIRANGE was defined */ + if (sspi == tspi) /* Specific SPI asked */ + spi = tspi; + else /* Range specified */ { - if (sspi == tspi) /* Specific SPI asked */ - spi = tspi; - else /* Range specified */ - { - get_random_bytes((void *) &spi, sizeof(spi)); - spi = sspi + (spi % (tspi - sspi)); - } + get_random_bytes((void *) &spi, sizeof(spi)); + spi = sspi + (spi % (tspi - sspi)); } - else /* Some SPI */ - get_random_bytes((void *) &spi, sizeof(spi)); - if (spi <= 255) /* Don't allocate SPI <= 255, they're reserved */ + /* Don't allocate reserved SPIs. */ + if (spi == SPI_LOCAL_USE || + (spi >= SPI_RESERVED_MIN && spi <= SPI_RESERVED_MAX)) continue; else spi = htonl(spi); @@ -520,6 +523,16 @@ tdb_expiration(struct tdb *tdb, int flags) int will_be_first, sole_reason, early; int s = spltdb(); + /* + * If this is the local use SPI, this is an SPD entry, so don't setup any + * timers. + */ + if (ntohl(tdb->tdb_spi) == SPI_LOCAL_USE) + { + splx(s); + return; + } + /* Find the earliest expiration. */ if ((tdb->tdb_flags & TDBF_FIRSTUSE) && tdb->tdb_first_use != 0 && (next_timeout == 0 || @@ -537,7 +550,8 @@ tdb_expiration(struct tdb *tdb, int flags) next_timeout = tdb->tdb_soft_timeout; /* No change? */ - if (next_timeout == tdb->tdb_timeout) { + if (next_timeout == tdb->tdb_timeout) + { splx(s); return; } @@ -633,7 +647,8 @@ tdb_expiration(struct tdb *tdb, int flags) /* * Check various invariants. */ - if (tdb->tdb_expnext.tqe_prev != NULL) { + if (tdb->tdb_expnext.tqe_prev != NULL) + { t = TAILQ_FIRST(&expclusterlist); if (t != tdb && t->tdb_timeout >= tdb->tdb_timeout) panic("tdb_expiration: " diff --git a/sys/netinet/ip_ipsp.h b/sys/netinet/ip_ipsp.h index 7c256a5767c..54164a7fd89 100644 --- a/sys/netinet/ip_ipsp.h +++ b/sys/netinet/ip_ipsp.h @@ -1,4 +1,4 @@ -/* $OpenBSD: ip_ipsp.h,v 1.38 1999/07/06 20:17:52 cmetz Exp $ */ +/* $OpenBSD: ip_ipsp.h,v 1.39 1999/07/15 14:15:41 niklas Exp $ */ /* * The authors of this code are John Ioannidis (ji@tla.org), @@ -99,6 +99,11 @@ union sockaddr_union #define AH_RMD160_ALEN 20 #define AH_ALEN_MAX 20 /* Keep updated */ +/* Reserved SPI numbers */ +#define SPI_LOCAL_USE 0 +#define SPI_RESERVED_MIN 1 +#define SPI_RESERVED_MAX 255 + struct sockaddr_encap { u_int8_t sen_len; /* length */ diff --git a/sys/netinet/ip_output.c b/sys/netinet/ip_output.c index 5b9e9e85fcb..cef8b3570e2 100644 --- a/sys/netinet/ip_output.c +++ b/sys/netinet/ip_output.c @@ -1,4 +1,4 @@ -/* $OpenBSD: ip_output.c,v 1.48 1999/06/15 02:24:02 deraadt Exp $ */ +/* $OpenBSD: ip_output.c,v 1.49 1999/07/15 14:15:41 niklas Exp $ */ /* $NetBSD: ip_output.c,v 1.28 1996/02/13 23:43:07 christos Exp $ */ /* @@ -90,6 +90,8 @@ int (*fr_checkp) __P((struct ip *, int, struct ifnet *, int, struct mbuf **)); extern int ipsec_auth_default_level; extern int ipsec_esp_trans_default_level; extern int ipsec_esp_network_default_level; + +extern int pfkeyv2_acquire(struct tdb *, int); #endif /* @@ -275,16 +277,39 @@ ip_output(m0, va_alist) } /* - * For VPNs a route with a reserved SPI of 1 is used to + * For VPNs a route with a reserved SPI of 0 is used to * indicate the need for an SA when none is established. */ - if (ntohl(gw->sen_ipsp_spi) == 0x1) { - sa_require = NOTIFY_SATYPE_AUTH | NOTIFY_SATYPE_TUNNEL; - if (gw->sen_ipsp_sproto == IPPROTO_ESP) - sa_require |= NOTIFY_SATYPE_CONF; + if (ntohl(gw->sen_ipsp_spi) == SPI_LOCAL_USE) { + bzero(&sunion, sizeof(sunion)); + sunion.sin.sin_family = AF_INET; + sunion.sin.sin_len = sizeof(struct sockaddr_in); + sunion.sin.sin_addr = gw->sen_ipsp_dst; + tdb = (struct tdb *) gettdb(gw->sen_ipsp_spi, &sunion, + gw->sen_ipsp_sproto); + + if (tdb) + { + if (tdb->tdb_authalgxform) + sa_require = NOTIFY_SATYPE_AUTH; + if (tdb->tdb_encalgxform) + sa_require |= NOTIFY_SATYPE_CONF; + if (tdb->tdb_flags & TDBF_TUNNELING) + sa_require |= NOTIFY_SATYPE_TUNNEL; + } + else /* No TDB found */ + { + /* + * XXX We should construct a TDB from system + * default (which should be tunable via sysctl). + * For now, drop packet and ignore SPD entry. + */ + goto no_encap; + } + + /* PF_KEYv2 notification message */ + pfkeyv2_acquire(tdb, 0); /* XXX Check for errors */ - /* XXX PF_KEYv2 notification message */ - splx(s); /* |