summaryrefslogtreecommitdiff
path: root/sys
diff options
context:
space:
mode:
Diffstat (limited to 'sys')
-rw-r--r--sys/net/pf.c196
1 files changed, 164 insertions, 32 deletions
diff --git a/sys/net/pf.c b/sys/net/pf.c
index 0179876ae9a..3b7cbaa1f6c 100644
--- a/sys/net/pf.c
+++ b/sys/net/pf.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: pf.c,v 1.118 2001/07/21 23:26:41 dhartmei Exp $ */
+/* $OpenBSD: pf.c,v 1.119 2001/07/25 12:22:28 dhartmei Exp $ */
/*
* Copyright (c) 2001, Daniel Hartmeier
@@ -57,6 +57,7 @@
#include <netinet/udp.h>
#include <netinet/ip_icmp.h>
+#include <dev/rndvar.h>
#include <net/pfvar.h>
#include "bpfilter.h"
@@ -78,6 +79,12 @@ struct pf_tree_node {
int balance;
};
+struct pf_port_node {
+ LIST_ENTRY(pf_port_node) next;
+ u_int16_t port;
+};
+LIST_HEAD(pf_port_list, pf_port_node);
+
/*
* Global variables
*/
@@ -103,10 +110,10 @@ u_int32_t ticket_nats_active;
u_int32_t ticket_nats_inactive;
u_int32_t ticket_rdrs_active;
u_int32_t ticket_rdrs_inactive;
-u_int16_t pf_next_port_tcp = 50001;
-u_int16_t pf_next_port_udp = 50001;
+struct pf_port_list pf_tcp_ports;
+struct pf_port_list pf_udp_ports;
-struct pool pf_tree_pl, pf_rule_pl, pf_nat_pl;
+struct pool pf_tree_pl, pf_rule_pl, pf_nat_pl, pf_sport_pl;
struct pool pf_rdr_pl, pf_state_pl;
int pf_tree_key_compare(struct pf_tree_key *,
@@ -165,6 +172,12 @@ int pf_test_state_icmp(struct pf_state **, int,
void *pf_pull_hdr(struct mbuf *, int, void *, int,
u_short *, u_short *);
+int pf_get_sport(u_int8_t, u_int16_t, u_int16_t,
+ u_int16_t *);
+void pf_put_sport(u_int8_t, u_int16_t);
+int pf_add_sport(struct pf_port_list *, u_int16_t);
+int pf_chk_sport(struct pf_port_list *, u_int16_t);
+
#if NPFLOG > 0
#define PFLOG_PACKET(x,a,b,c,d,e) \
do { \
@@ -178,6 +191,10 @@ void *pf_pull_hdr(struct mbuf *, int, void *, int,
#define PFLOG_PACKET
#endif
+#define STATE_TRANSLATE(s) \
+ ( (s)->lan.addr != (s)->gwy.addr || \
+ (s)->lan.port != (s)->gwy.port )
+
int
pf_tree_key_compare(struct pf_tree_key *a, struct pf_tree_key *b)
{
@@ -540,6 +557,9 @@ pf_purge_expired_states(void)
pf_tree_remove(&tree_lan_ext, NULL, &key);
if (pf_find_state(tree_lan_ext, &key) != NULL)
printf("pf: ERROR: remove failed\n");
+ if (STATE_TRANSLATE(cur->state))
+ pf_put_sport(cur->state->proto,
+ htons(cur->state->gwy.port));
/* free state */
pool_put(&pf_state_pl, cur->state);
/*
@@ -637,6 +657,8 @@ pfattach(int num)
0, NULL, NULL, 0);
pool_init(&pf_state_pl, sizeof(struct pf_state), 0, 0, 0, "pfstatepl",
0, NULL, NULL, 0);
+ pool_init(&pf_sport_pl, sizeof(struct pf_port_node), 0, 0, 0, "pfsport",
+ 0, NULL, NULL, 0);
TAILQ_INIT(&pf_rules[0]);
TAILQ_INIT(&pf_rules[1]);
@@ -651,6 +673,9 @@ pfattach(int num)
pf_rdrs_active = &pf_rdrs[0];
pf_rdrs_inactive = &pf_rdrs[1];
+ LIST_INIT(&pf_tcp_ports);
+ LIST_INIT(&pf_udp_ports);
+
pf_normalize_init();
}
@@ -1363,6 +1388,105 @@ pf_match_port(u_int8_t op, u_int16_t a1, u_int16_t a2, u_int16_t p)
return (0); /* never reached */
}
+int
+pf_chk_sport(struct pf_port_list *plist, u_int16_t port)
+{
+ struct pf_port_node *pnode;
+
+ LIST_FOREACH(pnode, plist, next) {
+ if (pnode->port == port)
+ return (1);
+ }
+
+ return (0);
+}
+
+int
+pf_add_sport(struct pf_port_list *plist, u_int16_t port)
+{
+ struct pf_port_node *pnode;
+
+ pnode = pool_get(&pf_sport_pl, M_NOWAIT);
+ if (pnode == NULL)
+ return (ENOMEM);
+
+ pnode->port = port;
+ LIST_INSERT_HEAD(plist, pnode, next);
+
+ return (0);
+}
+
+void
+pf_put_sport(u_int8_t proto, u_int16_t port)
+{
+ struct pf_port_list *plist;
+ struct pf_port_node *pnode;
+
+ if (proto == IPPROTO_TCP)
+ plist = &pf_tcp_ports;
+ else if (proto == IPPROTO_UDP)
+ plist = &pf_udp_ports;
+ else
+ return;
+
+ LIST_FOREACH(pnode, plist, next) {
+ if (pnode->port == port) {
+ LIST_REMOVE(pnode, next);
+ pool_put(&pf_sport_pl, pnode);
+ break;
+ }
+ }
+}
+
+int
+pf_get_sport(u_int8_t proto, u_int16_t low, u_int16_t high, u_int16_t *port)
+{
+ struct pf_port_list *plist;
+ int step;
+ u_int16_t cut;
+
+ if (proto == IPPROTO_TCP)
+ plist = &pf_tcp_ports;
+ else if (proto == IPPROTO_UDP)
+ plist = &pf_udp_ports;
+ else
+ return (EINVAL);
+
+ /* port search; start random, step; similar 2 portloop in in_pcbbind */
+ if (low == high) {
+ *port = low;
+ if (! pf_chk_sport(plist, *port))
+ goto found;
+ return (1);
+ } else if (low < high) {
+ step = 1;
+ cut = arc4random() % (high - low) + low;
+ } else {
+ step = -1;
+ cut = arc4random() % (low - high) + high;
+ }
+
+ *port = cut - step;
+ do {
+ *port += step;
+ if (! pf_chk_sport(plist, *port))
+ goto found;
+ } while (*port != low && *port != high);
+
+ step = -step;
+ *port = cut;
+ do {
+ *port += step;
+ if (! pf_chk_sport(plist, *port))
+ goto found;
+ } while (*port != low && *port != high);
+
+ return (1); /* none available */
+
+found:
+ return (pf_add_sport(plist, *port));
+}
+
struct pf_nat *
pf_get_nat(struct ifnet *ifp, u_int8_t proto, u_int32_t saddr, u_int32_t daddr)
{
@@ -1424,10 +1548,10 @@ pf_test_tcp(int direction, struct ifnet *ifp, struct mbuf *m,
struct pf_nat *nat = NULL;
struct pf_rdr *rdr = NULL;
u_int32_t baddr;
- u_int16_t bport, nport;
+ u_int16_t bport, nport = 0;
struct pf_rule *r, *rm = NULL;
u_short reason;
- int rewrite = 0;
+ int rewrite = 0, error;
if (direction == PF_OUT) {
/* check outgoing packet for NAT */
@@ -1435,9 +1559,12 @@ pf_test_tcp(int direction, struct ifnet *ifp, struct mbuf *m,
h->ip_src.s_addr, h->ip_dst.s_addr)) != NULL) {
baddr = h->ip_src.s_addr;
bport = th->th_sport;
+ error = pf_get_sport(IPPROTO_TCP, 50001,
+ 65535, &nport);
+ if (error)
+ return (PF_DROP);
pf_change_ap(&h->ip_src.s_addr, &th->th_sport,
- &h->ip_sum, &th->th_sum, nat->raddr,
- htons(pf_next_port_tcp));
+ &h->ip_sum, &th->th_sum, nat->raddr, htons(nport));
rewrite++;
}
} else {
@@ -1498,19 +1625,25 @@ pf_test_tcp(int direction, struct ifnet *ifp, struct mbuf *m,
rm->return_icmp & 255);
}
- if (rm->action == PF_DROP)
+ if (rm->action == PF_DROP) {
+ if (nport && nat != NULL)
+ pf_put_sport(IPPROTO_TCP, nport);
return (PF_DROP);
+ }
}
- if (((rm != NULL) && rm->keep_state) || (nat != NULL) || (rdr != NULL)) {
+ if (((rm != NULL) && rm->keep_state) || nat != NULL || rdr != NULL) {
/* create new state */
u_int16_t len;
struct pf_state *s;
len = h->ip_len - off - (th->th_off << 2);
s = pool_get(&pf_state_pl, PR_NOWAIT);
- if (s == NULL)
+ if (s == NULL) {
+ if (nport && nat != NULL)
+ pf_put_sport(IPPROTO_TCP, nport);
return (PF_DROP);
+ }
s->rule = rm;
s->log = rm && (rm->log & 2);
@@ -1518,15 +1651,12 @@ pf_test_tcp(int direction, struct ifnet *ifp, struct mbuf *m,
s->direction = direction;
if (direction == PF_OUT) {
s->gwy.addr = h->ip_src.s_addr;
- s->gwy.port = th->th_sport;
+ s->gwy.port = th->th_sport; /* sport */
s->ext.addr = h->ip_dst.s_addr;
s->ext.port = th->th_dport;
if (nat != NULL) {
s->lan.addr = baddr;
s->lan.port = bport;
- pf_next_port_tcp++;
- if (pf_next_port_tcp == 65535)
- pf_next_port_tcp = 50001;
} else {
s->lan.addr = s->gwy.addr;
s->lan.port = s->gwy.port;
@@ -1578,10 +1708,10 @@ pf_test_udp(int direction, struct ifnet *ifp, struct mbuf *m,
struct pf_nat *nat = NULL;
struct pf_rdr *rdr = NULL;
u_int32_t baddr;
- u_int16_t bport, nport;
+ u_int16_t bport, nport = 0;
struct pf_rule *r, *rm = NULL;
u_short reason;
- int rewrite = 0;
+ int rewrite = 0, error;
if (direction == PF_OUT) {
/* check outgoing packet for NAT */
@@ -1589,9 +1719,12 @@ pf_test_udp(int direction, struct ifnet *ifp, struct mbuf *m,
h->ip_src.s_addr, h->ip_dst.s_addr)) != NULL) {
baddr = h->ip_src.s_addr;
bport = uh->uh_sport;
+ error = pf_get_sport(IPPROTO_UDP, 50001,
+ 65535, &nport);
+ if (error)
+ return (PF_DROP);
pf_change_ap(&h->ip_src.s_addr, &uh->uh_sport,
- &h->ip_sum, &uh->uh_sum, nat->raddr,
- htons(pf_next_port_udp));
+ &h->ip_sum, &uh->uh_sum, nat->raddr, htons(nport));
rewrite++;
}
} else {
@@ -1649,8 +1782,11 @@ pf_test_udp(int direction, struct ifnet *ifp, struct mbuf *m,
rm->return_icmp & 255);
}
- if (rm->action == PF_DROP)
+ if (rm->action == PF_DROP) {
+ if (nport && nat != NULL)
+ pf_put_sport(IPPROTO_UDP, nport);
return (PF_DROP);
+ }
}
if ((rm != NULL && rm->keep_state) || nat != NULL || rdr != NULL) {
@@ -1660,8 +1796,11 @@ pf_test_udp(int direction, struct ifnet *ifp, struct mbuf *m,
len = h->ip_len - off - sizeof(*uh);
s = pool_get(&pf_state_pl, PR_NOWAIT);
- if (s == NULL)
+ if (s == NULL) {
+ if (nport && nat != NULL)
+ pf_put_sport(IPPROTO_UDP, nport);
return (PF_DROP);
+ }
s->rule = rm;
s->log = rm && (rm->log & 2);
@@ -1675,9 +1814,6 @@ pf_test_udp(int direction, struct ifnet *ifp, struct mbuf *m,
if (nat != NULL) {
s->lan.addr = baddr;
s->lan.port = bport;
- pf_next_port_udp++;
- if (pf_next_port_udp == 65535)
- pf_next_port_udp = 50001;
} else {
s->lan.addr = s->gwy.addr;
s->lan.port = s->gwy.port;
@@ -1965,8 +2101,7 @@ pf_test_state_tcp(struct pf_state **state, int direction, struct ifnet *ifp,
(*state)->expire = pftv.tv_sec + 24*60*60;
/* translate source/destination address, if needed */
- if ((*state)->lan.addr != (*state)->gwy.addr ||
- (*state)->lan.port != (*state)->gwy.port) {
+ if (STATE_TRANSLATE(*state)) {
if (direction == PF_OUT)
pf_change_ap(&h->ip_src.s_addr,
&th->th_sport, &h->ip_sum,
@@ -2044,8 +2179,7 @@ pf_test_state_udp(struct pf_state **state, int direction, struct ifnet *ifp,
(*state)->expire = pftv.tv_sec + 20;
/* translate source/destination address, if necessary */
- if ((*state)->lan.addr != (*state)->gwy.addr ||
- (*state)->lan.port != (*state)->gwy.port) {
+ if (STATE_TRANSLATE(*state)) {
if (direction == PF_OUT)
pf_change_ap(&h->ip_src.s_addr, &uh->uh_sport,
&h->ip_sum, &uh->uh_sum,
@@ -2180,8 +2314,7 @@ pf_test_state_icmp(struct pf_state **state, int direction, struct ifnet *ifp,
return (PF_DROP);
}
- if ((*state)->lan.addr != (*state)->gwy.addr ||
- (*state)->lan.port != (*state)->gwy.port) {
+ if (STATE_TRANSLATE(*state)) {
if (direction == PF_IN) {
pf_change_icmp(&h2.ip_src.s_addr,
&th.th_sport, &h->ip_dst.s_addr,
@@ -2230,8 +2363,7 @@ pf_test_state_icmp(struct pf_state **state, int direction, struct ifnet *ifp,
if (*state == NULL)
return (PF_DROP);
- if ((*state)->lan.addr != (*state)->gwy.addr ||
- (*state)->lan.port != (*state)->gwy.port) {
+ if (STATE_TRANSLATE(*state)) {
if (direction == PF_IN) {
pf_change_icmp(&h2.ip_src.s_addr,
&uh.uh_sport, &h->ip_dst.s_addr,