summaryrefslogtreecommitdiff
path: root/sys/net/pfkeyv2_convert.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/net/pfkeyv2_convert.c')
-rw-r--r--sys/net/pfkeyv2_convert.c94
1 files changed, 93 insertions, 1 deletions
diff --git a/sys/net/pfkeyv2_convert.c b/sys/net/pfkeyv2_convert.c
index e33642b8590..2254b60e25c 100644
--- a/sys/net/pfkeyv2_convert.c
+++ b/sys/net/pfkeyv2_convert.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: pfkeyv2_convert.c,v 1.7 2001/12/12 04:46:42 angelos Exp $ */
+/* $OpenBSD: pfkeyv2_convert.c,v 1.8 2002/05/31 01:38:04 angelos Exp $ */
/*
* The author of this code is Angelos D. Keromytis (angelos@keromytis.org)
*
@@ -97,6 +97,7 @@
#include <sys/mbuf.h>
#include <sys/kernel.h>
#include <sys/socket.h>
+#include <net/route.h>
#include <netinet/ip_ipsp.h>
#include <net/pfkeyv2.h>
#include <crypto/cryptodev.h>
@@ -381,6 +382,97 @@ export_lifetime(void **p, struct tdb *tdb, int type)
}
/*
+ * Import flow information to two struct sockaddr_encap's. Either
+ * all or none of the address arguments are NULL.
+ */
+void
+import_flow(struct sockaddr_encap *flow, struct sockaddr_encap *flowmask,
+ struct sadb_address *ssrc, struct sadb_address *ssrcmask,
+ struct sadb_address *ddst, struct sadb_address *ddstmask,
+ struct sadb_protocol *sab)
+{
+ u_int8_t transproto = 0;
+ union sockaddr_union *src = (union sockaddr_union *)(ssrc + 1);
+ union sockaddr_union *dst = (union sockaddr_union *)(ddst + 1);
+ union sockaddr_union *srcmask = (union sockaddr_union *)(ssrcmask + 1);
+ union sockaddr_union *dstmask = (union sockaddr_union *)(ddstmask + 1);
+
+ if (src == NULL)
+ return; /* There wasn't any information to begin with. */
+
+ bzero(flow, sizeof(*flow));
+ bzero(flowmask, sizeof(*flowmask));
+
+ if (sab != NULL)
+ transproto = sab->sadb_protocol_proto;
+
+ /*
+ * Check that all the address families match. We know they are
+ * valid and supported because pfkeyv2_parsemessage() checked that.
+ */
+ if ((src->sa.sa_family != dst->sa.sa_family) ||
+ (src->sa.sa_family != srcmask->sa.sa_family) ||
+ (src->sa.sa_family != dstmask->sa.sa_family))
+ return;
+
+ /* Generic netmask handling, works for IPv4 and IPv6. */
+ rt_maskedcopy(&src->sa, &src->sa, &srcmask->sa);
+ rt_maskedcopy(&dst->sa, &dst->sa, &dstmask->sa);
+
+ /*
+ * We set these as an indication that tdb_filter/tdb_filtermask are
+ * in fact initialized.
+ */
+ flow->sen_family = flowmask->sen_family = PF_KEY;
+ flow->sen_len = flowmask->sen_len = SENT_LEN;
+
+ switch (src->sa.sa_family)
+ {
+#ifdef INET
+ case AF_INET:
+ flow->sen_type = SENT_IP4;
+ flow->sen_direction = sab->sadb_protocol_direction;
+ flow->sen_ip_src = src->sin.sin_addr;
+ flow->sen_ip_dst = dst->sin.sin_addr;
+ flow->sen_proto = transproto;
+ flow->sen_sport = src->sin.sin_port;
+ flow->sen_dport = dst->sin.sin_port;
+
+ flowmask->sen_type = SENT_IP4;
+ flowmask->sen_direction = 0xff;
+ flowmask->sen_ip_src = srcmask->sin.sin_addr;
+ flowmask->sen_ip_dst = dstmask->sin.sin_addr;
+ flowmask->sen_sport = srcmask->sin.sin_port;
+ flowmask->sen_dport = dstmask->sin.sin_port;
+ if (transproto)
+ flowmask->sen_proto = 0xff;
+ break;
+#endif /* INET */
+
+#ifdef INET6
+ case AF_INET6:
+ flow->sen_type = SENT_IP6;
+ flow->sen_ip6_direction = sab->sadb_protocol_direction;
+ flow->sen_ip6_src = src->sin6.sin6_addr;
+ flow->sen_ip6_dst = dst->sin6.sin6_addr;
+ flow->sen_ip6_proto = transproto;
+ flow->sen_ip6_sport = src->sin6.sin6_port;
+ flow->sen_ip6_dport = dst->sin6.sin6_port;
+
+ flowmask->sen_type = SENT_IP6;
+ flowmask->sen_ip6_direction = 0xff;
+ flowmask->sen_ip6_src = srcmask->sin6.sin6_addr;
+ flowmask->sen_ip6_dst = dstmask->sin6.sin6_addr;
+ flowmask->sen_ip6_sport = srcmask->sin6.sin6_port;
+ flowmask->sen_ip6_dport = dstmask->sin6.sin6_port;
+ if (transproto)
+ flowmask->sen_ip6_proto = 0xff;
+ break;
+#endif /* INET6 */
+ }
+}
+
+/*
* Copy an SADB_ADDRESS payload to a struct sockaddr.
*/
void