diff options
author | Angelos D. Keromytis <angelos@cvs.openbsd.org> | 2001-01-14 23:40:02 +0000 |
---|---|---|
committer | Angelos D. Keromytis <angelos@cvs.openbsd.org> | 2001-01-14 23:40:02 +0000 |
commit | 4aa108e08165951b9c666cf5c51aeb5dca9ced1c (patch) | |
tree | 6f7a058a41dfb36868a9841cc4e39ebdb2c7e80f | |
parent | 381ab4ca856ec50be5342a387f0ce3a8ba4b8e6c (diff) |
Handling of Phase 1 DELETE and Phase 2 INVALID_SPI messages
(newsham@lava.net)
-rw-r--r-- | sbin/isakmpd/ipsec.c | 159 | ||||
-rw-r--r-- | sbin/isakmpd/sa.c | 46 | ||||
-rw-r--r-- | sbin/isakmpd/sa.h | 4 |
3 files changed, 163 insertions, 46 deletions
diff --git a/sbin/isakmpd/ipsec.c b/sbin/isakmpd/ipsec.c index c8bb0f9a32a..62b2300e47d 100644 --- a/sbin/isakmpd/ipsec.c +++ b/sbin/isakmpd/ipsec.c @@ -1,4 +1,4 @@ -/* $OpenBSD: ipsec.c,v 1.35 2001/01/11 00:46:28 angelos Exp $ */ +/* $OpenBSD: ipsec.c,v 1.36 2001/01/14 23:40:01 angelos Exp $ */ /* $EOM: ipsec.c,v 1.143 2000/12/11 23:57:42 niklas Exp $ */ /* @@ -193,6 +193,7 @@ struct dst_spi_proto_arg { /* * Check if SA matches what we are asking for through V_ARG. It has to * be a finished phase 2 SA. + * if "proto" arg is 0, match any proto */ static int ipsec_sa_check (struct sa *sa, void *v_arg) @@ -220,8 +221,8 @@ ipsec_sa_check (struct sa *sa, void *v_arg) for (proto = TAILQ_FIRST (&sa->protos); proto; proto = TAILQ_NEXT (proto, link)) - if (proto->proto == arg->proto - && memcmp (proto->spi[incoming], &arg->spi, sizeof arg->spi) == 0) + if ((arg->proto == 0 || proto->proto == arg->proto) + && memcmp (proto->spi[incoming], &arg->spi, sizeof arg->spi) == 0) return 1; return 0; } @@ -758,12 +759,102 @@ ipsec_initiator (struct message *msg) return 0; } +/* + * delete all SA's from addr with the associated proto and SPI's + * + * spis[] is an array of SPIs of size 16-octet for proto ISAKMP + * or 4-octet otherwise. + */ +static void +ipsec_delete_spi_list (struct sockaddr *addr, u_int8_t proto, + u_int8_t *spis, int nspis, char *type) +{ + u_int32_t iaddr = ((struct sockaddr_in *)addr)->sin_addr.s_addr; + struct sa *sa; + int i; + + for (i = 0; i < nspis; i++) + { + if (proto == ISAKMP_PROTO_ISAKMP) + { + u_int8_t *spi = spis + i * ISAKMP_HDR_COOKIES_LEN; + + /* + * This really shouldn't happen in IPSEC DOI + * code, but Cisco VPN 3000 sends ISAKMP DELETE's + * this way. + */ + sa = sa_lookup_isakmp_sa (addr, spi); + } + else + { + u_int32_t spi = ((u_int32_t *)spis)[i]; + + sa = ipsec_sa_lookup (iaddr, spi, proto); + } + + if (sa == NULL) + { + LOG_DBG ((LOG_SA, 30, "ipsec_delete_spi_list: " + "could not locate SA (SPI %08x, proto %u)", + spis[i], proto)); + continue; + } + + /* Delete the SA and search for the next */ + LOG_DBG ((LOG_SA, 30, "ipsec_delete_spi_list: " + "%s made us delete SA %p (%d references) for proto %d", + type, sa, sa->refcnt, proto)); + + sa_reference (sa); + sa_free (sa); + } +} + +/* + * deal with a NOTIFY of INVALID_SPI + */ +static void +ipsec_invalid_spi (struct message *msg, struct payload *p) +{ + struct sockaddr *dst; + int invspisz, off, dstlen; + u_int32_t spi; + u_int16_t totsiz; + u_int8_t spisz; + + /* + * get the invalid spi out of the variable sized notification data + * field, which is after the variable sized SPI field [which specifies + * the receiving entity's phase-1 SPI, not the invalid spi] + */ + totsiz = GET_ISAKMP_GEN_LENGTH (p->p); + spisz = GET_ISAKMP_NOTIFY_SPI_SZ (p->p); + off = ISAKMP_NOTIFY_SPI_OFF + spisz; + invspisz = totsiz - off; + + if(invspisz != sizeof spi) + { + LOG_DBG ((LOG_SA, 40, + "ipsec_invalid_spi: SPI size %d in INVALID_SPI " + "payload unsupported", spisz)); + return; + } + memcpy (&spi, p->p + off, sizeof spi); + + msg->transport->vtbl->get_dst (msg->transport, &dst, &dstlen); + + /* delete matching SPI's from this peer */ + ipsec_delete_spi_list (dst, 0, (u_int8_t *)&spi, 1, "INVALID_SPI"); +} + static int ipsec_responder (struct message *msg) { struct exchange *exchange = msg->exchange; int (**script) (struct message *msg) = 0; struct payload *p; + u_int16_t type; /* Check that a new exchange is coherent with the IKE rules. */ if (exchange->step == 0 @@ -795,10 +886,14 @@ ipsec_responder (struct message *msg) for (p = TAILQ_FIRST (&msg->payload[ISAKMP_PAYLOAD_NOTIFY]); p; p = TAILQ_NEXT (p, link)) { + type = GET_ISAKMP_NOTIFY_MSG_TYPE (p->p); LOG_DBG ((LOG_EXCHANGE, 10, "ipsec_responder: got NOTIFY of type %s", - constant_lookup (isakmp_notify_cst, - GET_ISAKMP_NOTIFY_MSG_TYPE (p->p)))); + constant_lookup (isakmp_notify_cst, type))); + + if(type == ISAKMP_NOTIFY_INVALID_SPI) + ipsec_invalid_spi (msg, p); + p->flags |= PL_MARK; } @@ -1296,11 +1391,11 @@ int ipsec_handle_leftover_payload (struct message *msg, u_int8_t type, struct payload *payload) { - u_int32_t spisz, nspis, *spis; + u_int32_t spisz, nspis; struct sockaddr *dst; socklen_t dstlen; - int flag = 0, i; - u_int8_t proto; + int flag = 0; + u_int8_t *spis, proto; struct sa *sa; switch (type) @@ -1310,14 +1405,6 @@ ipsec_handle_leftover_payload (struct message *msg, u_int8_t type, nspis = GET_ISAKMP_DELETE_NSPIS (payload->p); spisz = GET_ISAKMP_DELETE_SPI_SZ (payload->p); - if (spisz != sizeof (u_int32_t)) - { - LOG_DBG ((LOG_SA, 50, - "ipsec_handle_leftover_payload: SPI size %d in DELETE " - "payload unsupported", spisz)); - return -1; - } - if (nspis == 0) { LOG_DBG ((LOG_SA, 60, "ipsec_handle_leftover_payload: message " @@ -1325,7 +1412,17 @@ ipsec_handle_leftover_payload (struct message *msg, u_int8_t type, return -1; } - spis = (u_int32_t *) malloc (nspis * spisz); + /* verify proper SPI size */ + if ((proto == ISAKMP_PROTO_ISAKMP && spisz != ISAKMP_HDR_COOKIES_LEN) + || (proto != ISAKMP_PROTO_ISAKMP && spisz != sizeof (u_int32_t))) + { + LOG_DBG ((LOG_SA, 50, + "ipsec_handle_leftover_payload: invalid SPI size %d " + "for proto %d in DELETE payload", spisz, proto)); + return -1; + } + + spis = (u_int8_t *) malloc (nspis * spisz); if (spis == NULL) { LOG_DBG ((LOG_SA, 50, @@ -1335,35 +1432,11 @@ ipsec_handle_leftover_payload (struct message *msg, u_int8_t type, return -1; } + /* extract SPI and get dst address */ memcpy (spis, payload->p + ISAKMP_DELETE_SPI_OFF, nspis * spisz); msg->transport->vtbl->get_dst (msg->transport, &dst, &dstlen); - for (i = 0; i < nspis; i++) - { - sa = ipsec_sa_lookup (((struct sockaddr_in *)dst)->sin_addr.s_addr, - spis[i], proto); - if (sa == NULL) - { - LOG_DBG ((LOG_SA, 40, "ipsec_handle_leftover_payload: " - "could not locate SA (SPI %08x, proto %u)", - spis[i], proto)); - continue; - } - - /* Delete the SA and search for the next */ - LOG_DBG ((LOG_SA, 30, "ipsec_handle_leftover_payload: " - "DELETE made us delete SA %p (%d references)", - sa, sa->refcnt)); - - /* - * This SA is still referenced by the software timeout. - * However, sa_free will clean up all timeouts and - * decrement reference counters. We need to reference it - * so that sa_release() works. - */ - sa_reference(sa); - sa_free(sa); - } + ipsec_delete_spi_list (dst, proto, spis, nspis, "DELETE"); free (spis); payload->flags |= PL_MARK; diff --git a/sbin/isakmpd/sa.c b/sbin/isakmpd/sa.c index 1313fd6ef09..d96d529b3a9 100644 --- a/sbin/isakmpd/sa.c +++ b/sbin/isakmpd/sa.c @@ -1,4 +1,4 @@ -/* $OpenBSD: sa.c,v 1.33 2000/12/12 01:47:12 niklas Exp $ */ +/* $OpenBSD: sa.c,v 1.34 2001/01/14 23:40:01 angelos Exp $ */ /* $EOM: sa.c,v 1.112 2000/12/12 00:22:52 niklas Exp $ */ /* @@ -195,6 +195,50 @@ sa_check_peer (struct sa *sa, void *v_addr) return dstlen == addr->len && memcmp (dst, addr->addr, dstlen) == 0; } +struct dst_isakmpspi_arg { + struct sockaddr *dst; + u_int8_t *spi; /* must be ISAKMP_SPI_SIZE octets */ +}; + +/* + * Check if SA matches what we are asking for through V_ARG. It has to + * be a finished phaes 1 (ISAKMP) SA. + */ +static int +isakmp_sa_check (struct sa *sa, void *v_arg) +{ + struct dst_isakmpspi_arg *arg = v_arg; + struct sockaddr *dst, *src; + int dstlen, srclen; + + if (sa->phase != 1 || !(sa->flags & SA_FLAG_READY)) + return 0; + + /* verify address is either src or dst for this sa */ + sa->transport->vtbl->get_dst (sa->transport, &dst, &dstlen); + sa->transport->vtbl->get_src (sa->transport, &src, &srclen); + if (memcmp (src, arg->dst, src->sa_len) && + memcmp (dst, arg->dst, dst->sa_len)) + return 0; + + /* match icookie+rcookie against spi */ + if(memcmp (sa->cookies, arg->spi, ISAKMP_HDR_COOKIES_LEN) == 0) + return 1; + + return 0; +} + +/* + * Find an ISAKMP SA with a "name" of DST & SPI. + */ +struct sa * +sa_lookup_isakmp_sa (struct sockaddr *dst, u_int8_t *spi) +{ + struct dst_isakmpspi_arg arg = { dst, spi }; + + return sa_find (isakmp_sa_check, &arg); +} + /* Lookup a ready SA by the peer's address. */ struct sa * sa_lookup_by_peer (struct sockaddr *dst, socklen_t dstlen) diff --git a/sbin/isakmpd/sa.h b/sbin/isakmpd/sa.h index b38ef0ed532..cb6b51dbc23 100644 --- a/sbin/isakmpd/sa.h +++ b/sbin/isakmpd/sa.h @@ -1,4 +1,4 @@ -/* $OpenBSD: sa.h,v 1.18 2000/10/10 13:35:24 niklas Exp $ */ +/* $OpenBSD: sa.h,v 1.19 2001/01/14 23:40:01 angelos Exp $ */ /* $EOM: sa.h,v 1.58 2000/10/10 12:39:01 provos Exp $ */ /* @@ -199,10 +199,10 @@ extern struct sa *sa_lookup_by_peer (struct sockaddr *, socklen_t); extern struct sa *sa_lookup_by_header (u_int8_t *, int); extern struct sa *sa_lookup_by_name (char *, int); extern struct sa *sa_lookup_from_icookie (u_int8_t *); +extern struct sa *sa_lookup_isakmp_sa (struct sockaddr *, u_int8_t *); extern void sa_mark_replaced (struct sa *); extern void sa_reference (struct sa *); extern void sa_release (struct sa *); extern void sa_report (void); extern int sa_setup_expirations (struct sa *); - #endif /* _SA_H_ */ |