summaryrefslogtreecommitdiff
path: root/sbin/ipsecctl
diff options
context:
space:
mode:
Diffstat (limited to 'sbin/ipsecctl')
-rw-r--r--sbin/ipsecctl/parse.y79
1 files changed, 72 insertions, 7 deletions
diff --git a/sbin/ipsecctl/parse.y b/sbin/ipsecctl/parse.y
index 6084ee58f14..398c0b3a641 100644
--- a/sbin/ipsecctl/parse.y
+++ b/sbin/ipsecctl/parse.y
@@ -1,4 +1,4 @@
-/* $OpenBSD: parse.y,v 1.88 2006/06/01 17:32:20 naddy Exp $ */
+/* $OpenBSD: parse.y,v 1.89 2006/06/01 22:44:03 hshoexer Exp $ */
/*
* Copyright (c) 2002, 2003, 2004 Henning Brauer <henning@openbsd.org>
@@ -163,13 +163,16 @@ struct ipsec_rule *create_sa(u_int8_t, u_int8_t, struct ipsec_addr_wrap *,
struct ipsec_key *);
struct ipsec_rule *reverse_sa(struct ipsec_rule *, u_int32_t,
struct ipsec_key *, struct ipsec_key *);
+struct ipsec_rule *create_sagroup(struct ipsec_addr_wrap *, u_int8_t,
+ u_int32_t, struct ipsec_addr_wrap *, u_int8_t,
+ u_int32_t);
struct ipsec_rule *create_flow(u_int8_t, u_int8_t,
struct ipsec_addr_wrap *, u_int16_t,
struct ipsec_addr_wrap *, u_int16_t,
struct ipsec_addr_wrap *, struct ipsec_addr_wrap *,
u_int8_t, char *, char *, u_int8_t);
int expand_rule(struct ipsec_rule *, u_int8_t, u_int32_t,
- struct ipsec_key *, struct ipsec_key *);
+ struct ipsec_key *, struct ipsec_key *, int);
struct ipsec_rule *reverse_rule(struct ipsec_rule *);
struct ipsec_rule *create_ike(u_int8_t, struct ipsec_addr_wrap *, struct
ipsec_addr_wrap *, struct ipsec_addr_wrap *,
@@ -177,6 +180,7 @@ struct ipsec_rule *create_ike(u_int8_t, struct ipsec_addr_wrap *, struct
struct ipsec_transforms *, struct
ipsec_transforms *, u_int8_t, u_int8_t, char *,
char *, struct ike_auth *);
+int add_sagroup(struct ipsec_rule *);
struct ipsec_transforms *ipsec_transforms;
@@ -305,7 +309,7 @@ tcpmd5rule : TCPMD5 hosts spispec authkeyspec {
YYERROR;
r->nr = ipsec->rule_nr++;
- if (expand_rule(r, 0, $3.spiin, $4.keyin, NULL))
+ if (expand_rule(r, 0, $3.spiin, $4.keyin, NULL, 0))
errx(1, "tcpmd5rule: expand_rule");
}
;
@@ -320,7 +324,7 @@ sarule : satype tmode hosts spispec transforms authkeyspec
YYERROR;
r->nr = ipsec->rule_nr++;
- if (expand_rule(r, 0, $4.spiin, $6.keyin, $7.keyin))
+ if (expand_rule(r, 0, $4.spiin, $6.keyin, $7.keyin, 1))
errx(1, "sarule: expand_rule");
}
;
@@ -334,7 +338,7 @@ flowrule : FLOW satype dir proto hosts peers ids type {
if (r == NULL)
YYERROR;
- if (expand_rule(r, $3, 0, NULL, NULL))
+ if (expand_rule(r, $3, 0, NULL, NULL, 0))
errx(1, "flowrule: expand_rule");
}
;
@@ -348,7 +352,7 @@ ikerule : IKE ikemode satype proto hosts peers mmxfs qmxfs ids ikeauth {
YYERROR;
r->nr = ipsec->rule_nr++;
- if (expand_rule(r, 0, 0, NULL, NULL))
+ if (expand_rule(r, 0, 0, NULL, NULL, 0))
errx(1, "ikerule: expand_rule");
}
;
@@ -1873,6 +1877,38 @@ validate_sa(u_int32_t spi, u_int8_t satype, struct ipsec_transforms *xfs,
return 1;
}
+int
+add_sagroup(struct ipsec_rule *r)
+{
+ struct ipsec_rule *rp, *last, *group;
+ int found = 0;
+
+ TAILQ_FOREACH(rp, &ipsec->group_queue, group_entry) {
+ if (strcmp(rp->dst->name, r->dst->name) == 0) {
+ found = 1;
+ break;
+ }
+ }
+ if (found) {
+ last = TAILQ_LAST(&rp->dst_group_queue, dst_group_queue);
+ TAILQ_INSERT_TAIL(&rp->dst_group_queue, r, dst_group_entry);
+
+ group = create_sagroup(last->dst, last->satype, last->spi,
+ r->dst, r->satype, r->spi);
+ if (group == NULL)
+ return (1);
+ group->nr = ipsec->rule_nr++;
+ if (ipsecctl_add_rule(ipsec, group))
+ return (1);
+ } else {
+ TAILQ_INSERT_TAIL(&ipsec->group_queue, r, group_entry);
+ TAILQ_INIT(&r->dst_group_queue);
+ TAILQ_INSERT_TAIL(&r->dst_group_queue, r, dst_group_entry);
+ }
+
+ return (0);
+}
+
struct ipsec_rule *
create_sa(u_int8_t satype, u_int8_t tmode, struct ipsec_addr_wrap *src, struct
ipsec_addr_wrap *dst, u_int32_t spi, struct ipsec_transforms *xfs,
@@ -1928,6 +1964,29 @@ reverse_sa(struct ipsec_rule *rule, u_int32_t spi, struct ipsec_key *authkey,
}
struct ipsec_rule *
+create_sagroup(struct ipsec_addr_wrap *dst, u_int8_t proto, u_int32_t spi,
+ struct ipsec_addr_wrap *dst2, u_int8_t proto2, u_int32_t spi2)
+{
+ struct ipsec_rule *r;
+
+ r = calloc(1, sizeof(struct ipsec_rule));
+ if (r == NULL)
+ err(1, "create_sagroup: calloc");
+
+ r->type |= RULE_GROUP;
+
+ r->dst = copyhost(dst);
+ r->dst2 = copyhost(dst2);
+ r->proto = proto;
+ r->proto2 = proto2;
+ r->spi = spi;
+ r->spi2 = spi2;
+ r->satype = proto;
+
+ return (r);
+}
+
+struct ipsec_rule *
create_flow(u_int8_t dir, u_int8_t proto, struct ipsec_addr_wrap *src,
u_int16_t sport, struct ipsec_addr_wrap *dst, u_int16_t dport,
struct ipsec_addr_wrap *local, struct ipsec_addr_wrap *peer,
@@ -2006,7 +2065,7 @@ errout:
int
expand_rule(struct ipsec_rule *rule, u_int8_t direction, u_int32_t spi,
- struct ipsec_key *authkey, struct ipsec_key *enckey)
+ struct ipsec_key *authkey, struct ipsec_key *enckey, int group)
{
struct ipsec_rule *r, *revr;
struct ipsec_addr_wrap *src, *dst;
@@ -2024,6 +2083,8 @@ expand_rule(struct ipsec_rule *rule, u_int8_t direction, u_int32_t spi,
r->nr = ipsec->rule_nr++;
if (ipsecctl_add_rule(ipsec, r))
return (1);
+ if (group && add_sagroup(r))
+ return (1);
if (direction == IPSEC_INOUT) {
/* Create and add reverse flow rule. */
@@ -2034,6 +2095,8 @@ expand_rule(struct ipsec_rule *rule, u_int8_t direction, u_int32_t spi,
revr->nr = ipsec->rule_nr++;
if (ipsecctl_add_rule(ipsec, revr))
return (1);
+ if (group && add_sagroup(revr))
+ return (1);
} else if (spi != 0 || authkey || enckey) {
/* Create and add reverse sa rule. */
revr = reverse_sa(r, spi, authkey, enckey);
@@ -2043,6 +2106,8 @@ expand_rule(struct ipsec_rule *rule, u_int8_t direction, u_int32_t spi,
revr->nr = ipsec->rule_nr++;
if (ipsecctl_add_rule(ipsec, revr))
return (1);
+ if (group && add_sagroup(revr))
+ return (1);
}
added++;
}