summaryrefslogtreecommitdiff
path: root/sbin/photurisd/kernel.c
diff options
context:
space:
mode:
authorNiels Provos <provos@cvs.openbsd.org>2000-12-16 08:31:57 +0000
committerNiels Provos <provos@cvs.openbsd.org>2000-12-16 08:31:57 +0000
commit14bceab423b0fe91b8ce3f6db1287a70cc40305d (patch)
treeaa3e85a1107f2db45a5c864fe5e3fe5be5c136a8 /sbin/photurisd/kernel.c
parent669d1ef3e7479820a6c36483ef74da50437e4f31 (diff)
when a used SA expires and the exchange is gone, start a new one.
Diffstat (limited to 'sbin/photurisd/kernel.c')
-rw-r--r--sbin/photurisd/kernel.c160
1 files changed, 102 insertions, 58 deletions
diff --git a/sbin/photurisd/kernel.c b/sbin/photurisd/kernel.c
index 3b8b2faad0a..90ef9b8e2c4 100644
--- a/sbin/photurisd/kernel.c
+++ b/sbin/photurisd/kernel.c
@@ -39,7 +39,7 @@
*/
#ifndef lint
-static char rcsid[] = "$Id: kernel.c,v 1.19 2000/12/15 07:29:44 provos Exp $";
+static char rcsid[] = "$Id: kernel.c,v 1.20 2000/12/16 08:31:56 provos Exp $";
#endif
#include <time.h>
@@ -1154,7 +1154,7 @@ kernel_handle_queue()
{
struct pfmsg *pfmsg;
- while (pfmsg = TAILQ_FIRST(&pfqueue)) {
+ while ((pfmsg = TAILQ_FIRST(&pfqueue))) {
TAILQ_REMOVE(&pfqueue, pfmsg, next);
kernel_dispatch_notify(pfmsg->smsg);
@@ -1235,6 +1235,8 @@ kernel_handle_expire(struct sadb_msg *sadb)
struct sadb_sa *sa;
struct sadb_address *dst;
char dstbuf[NI_MAXHOST];
+ struct stateob *st;
+ time_t tm;
struct sockaddr *dstaddr;
struct sadb_lifetime *life;
struct sadb_ext *ext = (struct sadb_ext *)(sadb + 1);
@@ -1289,15 +1291,16 @@ kernel_handle_expire(struct sadb_msg *sadb)
: "HARD", dstbuf,
ntohl (sa->sadb_sa_spi), sadb->sadb_msg_satype));
+ spi = spi_find(dstbuf, (u_char *)&sa->sadb_sa_spi);
+ if (spi == NULL) {
+ LOG_DBG((LOG_KERNEL, 35, __FUNCTION__
+ ": can't find %s SPI %x",
+ dstbuf, ntohl(sa->sadb_sa_spi)));
+ return (-1);
+ }
+
switch(life->sadb_lifetime_exttype) {
case SADB_EXT_LIFETIME_HARD:
- spi = spi_find(dstbuf, (u_char *)&sa->sadb_sa_spi);
- if (spi == NULL) {
- LOG_DBG((LOG_KERNEL, 35, __FUNCTION__
- ": can't find %s SPI %x",
- dstbuf, ntohl(sa->sadb_sa_spi)));
- return (-1);
- }
LOG_DBG((LOG_KERNEL, 35, __FUNCTION__": removing %s SPI %x",
dstbuf, ntohl(sa->sadb_sa_spi)));
spi_unlink(spi);
@@ -1318,7 +1321,34 @@ kernel_handle_expire(struct sadb_msg *sadb)
return (0);
}
- spi_update(global_socket, (u_int8_t *)&sa->sadb_sa_spi);
+ if (spi->flags & SPI_OWNER) {
+ spi_update(global_socket,
+ (u_int8_t *)&sa->sadb_sa_spi);
+ return (0);
+ }
+
+ /*
+ * Try to find an already established exchange which is
+ * still valid.
+ */
+
+ st = state_find(dstbuf);
+
+ tm = time(NULL);
+ while (st != NULL &&
+ (st->lifetime <= tm || st->phase < SPI_UPDATE))
+ st = state_find_next(st, dstbuf);
+
+ if (st == NULL) {
+ int type = spi->flags & SPI_ESP ?
+ IPSEC_OPT_ENC : IPSEC_OPT_AUTH;
+
+ LOG_DBG((LOG_KERNEL, 45, __FUNCTION__
+ ": starting new exchange to %s",
+ spi->address));
+ kernel_new_exchange(spi->address, type);
+ }
+
break;
default:
log_print(__FUNCTION__": unknown extension type %d",
@@ -1329,6 +1359,43 @@ kernel_handle_expire(struct sadb_msg *sadb)
return (0);
}
+int
+kernel_new_exchange(char *address, int type)
+{
+ struct stateob *st;
+
+ /* No established exchange found, start a new one */
+ if ((st = state_new()) == NULL) {
+ log_print(__FUNCTION__
+ ": state_new() failed for remote ip %s", address);
+ return (-1);
+ }
+
+ /* Set up the state information */
+ strncpy(st->address, address, sizeof(st->address) - 1);
+ st->port = global_port;
+ st->sport = 0;
+ st->dport = 0;
+ st->protocol = 0;
+
+ st->flags = IPSEC_NOTIFY;
+
+ st->flags |= type;
+
+ if (start_exchange(global_socket, st, st->address,
+ st->port) == -1) {
+ log_print(__FUNCTION__": start_exchange() - informing kernel of failure");
+ /* Inform kernel of our failure */
+ kernel_notify_result(st, NULL, 0);
+ state_value_reset(st);
+ free(st);
+ return (-1);
+ } else
+ state_insert(st);
+
+ return (0);
+}
+
/*
* Tries to establish a new SA according to the information in a
* REQUEST_SA notify message received from the kernel.
@@ -1355,26 +1422,27 @@ kernel_request_sa(struct sadb_msg *sadb)
src = (struct sadb_address *)
pfkey_find_extension(ext, end, SADB_EXT_ADDRESS_SRC);
- if (dst) {
- dstaddr = (struct sockaddr *)(dst + 1);
- switch (dstaddr->sa_family) {
- case AF_INET:
- if (inet_ntop (AF_INET, &((struct sockaddr_in *)dstaddr)->sin_addr,
- dstbuf, sizeof(dstbuf)) == NULL) {
- log_error (__FUNCTION__": inet_ntop failed");
- return (-1);
- }
- break;
- default:
- log_error(__FUNCTION__
- ": unsupported address family %d",
- dstaddr->sa_family);
+ if (!dst)
+ return (-1);
+
+ dstaddr = (struct sockaddr *)(dst + 1);
+ switch (dstaddr->sa_family) {
+ case AF_INET:
+ if (inet_ntop(AF_INET,
+ &((struct sockaddr_in *)dstaddr)->sin_addr,
+ dstbuf, sizeof(dstbuf)) == NULL) {
+ log_error (__FUNCTION__": inet_ntop failed");
return (-1);
}
-
- LOG_DBG((LOG_KERNEL, 20, __FUNCTION__": dst: %s", dstbuf));
- } else
+ break;
+ default:
+ log_error(__FUNCTION__
+ ": unsupported address family %d",
+ dstaddr->sa_family);
return (-1);
+ }
+
+ LOG_DBG((LOG_KERNEL, 20, __FUNCTION__": dst: %s", dstbuf));
/* Try to find an already established exchange which is still valid */
st = state_find(dstbuf);
@@ -1383,37 +1451,7 @@ kernel_request_sa(struct sadb_msg *sadb)
while (st != NULL && (st->lifetime <= tm || st->phase < SPI_UPDATE))
st = state_find_next(st, dstbuf);
- if (st == NULL) {
- /* No established exchange found, start a new one */
- if ((st = state_new()) == NULL) {
- log_print(__FUNCTION__
- ": state_new() failed for remote ip %s",
- dstbuf);
- return (-1);
- }
- /* Set up the state information */
- strncpy(st->address, dstbuf, sizeof(st->address) - 1);
- st->port = global_port;
- st->sport = 0;
- st->dport = 0;
- st->protocol = 0;
-
- st->flags = IPSEC_NOTIFY;
-
- st->flags |= sadb->sadb_msg_satype == SADB_SATYPE_ESP ?
- IPSEC_OPT_ENC : IPSEC_OPT_AUTH;
-
- if (start_exchange(global_socket, st, st->address,
- st->port) == -1) {
- log_print(__FUNCTION__": start_exchange() - informing kernel of failure");
- /* Inform kernel of our failure */
- kernel_notify_result(st, NULL, 0);
- state_value_reset(st);
- free(st);
- return (-1);
- } else
- state_insert(st);
- } else {
+ if (st) {
struct sockaddr_in sin;
/*
@@ -1438,8 +1476,14 @@ kernel_request_sa(struct sadb_msg *sadb)
(struct sockaddr *)&sin, sizeof(sin)) != packet_size) {
log_error(__FUNCTION__": sendto()");
}
+ } else {
+ int type = sadb->sadb_msg_satype == SADB_SATYPE_ESP ?
+ IPSEC_OPT_ENC : IPSEC_OPT_AUTH;
+
+ return (kernel_new_exchange(dstbuf, type));
}
+
return (0);
}