c9a03d91ad
Approved by: re (kensmith)
376 lines
10 KiB
C
376 lines
10 KiB
C
/* $FreeBSD$ */
|
|
/* $OpenBSD: if_pfsync.h,v 1.30 2006/10/31 14:49:01 henning Exp $ */
|
|
|
|
/*
|
|
* Copyright (c) 2001 Michael Shalayeff
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
* IN NO EVENT SHALL THE AUTHOR OR HIS RELATIVES BE LIABLE FOR ANY DIRECT,
|
|
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
* SERVICES; LOSS OF MIND, USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
|
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
|
|
* IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
|
|
* THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#ifndef _NET_IF_PFSYNC_H_
|
|
#define _NET_IF_PFSYNC_H_
|
|
|
|
|
|
#define PFSYNC_ID_LEN sizeof(u_int64_t)
|
|
|
|
struct pfsync_state_scrub {
|
|
u_int16_t pfss_flags;
|
|
u_int8_t pfss_ttl; /* stashed TTL */
|
|
#define PFSYNC_SCRUB_FLAG_VALID 0x01
|
|
u_int8_t scrub_flag;
|
|
u_int32_t pfss_ts_mod; /* timestamp modulation */
|
|
} __packed;
|
|
|
|
struct pfsync_state_host {
|
|
struct pf_addr addr;
|
|
u_int16_t port;
|
|
u_int16_t pad[3];
|
|
} __packed;
|
|
|
|
struct pfsync_state_peer {
|
|
struct pfsync_state_scrub scrub; /* state is scrubbed */
|
|
u_int32_t seqlo; /* Max sequence number sent */
|
|
u_int32_t seqhi; /* Max the other end ACKd + win */
|
|
u_int32_t seqdiff; /* Sequence number modulator */
|
|
u_int16_t max_win; /* largest window (pre scaling) */
|
|
u_int16_t mss; /* Maximum segment size option */
|
|
u_int8_t state; /* active state level */
|
|
u_int8_t wscale; /* window scaling factor */
|
|
u_int8_t pad[6];
|
|
} __packed;
|
|
|
|
struct pfsync_state {
|
|
u_int32_t id[2];
|
|
char ifname[IFNAMSIZ];
|
|
struct pfsync_state_host lan;
|
|
struct pfsync_state_host gwy;
|
|
struct pfsync_state_host ext;
|
|
struct pfsync_state_peer src;
|
|
struct pfsync_state_peer dst;
|
|
struct pf_addr rt_addr;
|
|
u_int32_t rule;
|
|
u_int32_t anchor;
|
|
u_int32_t nat_rule;
|
|
u_int32_t creation;
|
|
u_int32_t expire;
|
|
u_int32_t packets[2][2];
|
|
u_int32_t bytes[2][2];
|
|
u_int32_t creatorid;
|
|
sa_family_t af;
|
|
u_int8_t proto;
|
|
u_int8_t direction;
|
|
u_int8_t log;
|
|
u_int8_t allow_opts;
|
|
u_int8_t timeout;
|
|
u_int8_t sync_flags;
|
|
u_int8_t updates;
|
|
} __packed;
|
|
|
|
#define PFSYNC_FLAG_COMPRESS 0x01
|
|
#define PFSYNC_FLAG_STALE 0x02
|
|
|
|
#ifdef PFSYNC_TDB
|
|
struct pfsync_tdb {
|
|
u_int32_t spi;
|
|
union sockaddr_union dst;
|
|
u_int32_t rpl;
|
|
u_int64_t cur_bytes;
|
|
u_int8_t sproto;
|
|
u_int8_t updates;
|
|
u_int8_t pad[2];
|
|
} __packed;
|
|
#endif
|
|
|
|
struct pfsync_state_upd {
|
|
u_int32_t id[2];
|
|
struct pfsync_state_peer src;
|
|
struct pfsync_state_peer dst;
|
|
u_int32_t creatorid;
|
|
u_int32_t expire;
|
|
u_int8_t timeout;
|
|
u_int8_t updates;
|
|
u_int8_t pad[6];
|
|
} __packed;
|
|
|
|
struct pfsync_state_del {
|
|
u_int32_t id[2];
|
|
u_int32_t creatorid;
|
|
struct {
|
|
u_int8_t state;
|
|
} src;
|
|
struct {
|
|
u_int8_t state;
|
|
} dst;
|
|
u_int8_t pad[2];
|
|
} __packed;
|
|
|
|
struct pfsync_state_upd_req {
|
|
u_int32_t id[2];
|
|
u_int32_t creatorid;
|
|
u_int32_t pad;
|
|
} __packed;
|
|
|
|
struct pfsync_state_clr {
|
|
char ifname[IFNAMSIZ];
|
|
u_int32_t creatorid;
|
|
u_int32_t pad;
|
|
} __packed;
|
|
|
|
struct pfsync_state_bus {
|
|
u_int32_t creatorid;
|
|
u_int32_t endtime;
|
|
u_int8_t status;
|
|
#define PFSYNC_BUS_START 1
|
|
#define PFSYNC_BUS_END 2
|
|
u_int8_t pad[7];
|
|
} __packed;
|
|
|
|
#ifdef _KERNEL
|
|
|
|
union sc_statep {
|
|
struct pfsync_state *s;
|
|
struct pfsync_state_upd *u;
|
|
struct pfsync_state_del *d;
|
|
struct pfsync_state_clr *c;
|
|
struct pfsync_state_bus *b;
|
|
struct pfsync_state_upd_req *r;
|
|
};
|
|
|
|
#ifdef PFSYNC_TDB
|
|
union sc_tdb_statep {
|
|
struct pfsync_tdb *t;
|
|
};
|
|
#endif
|
|
|
|
extern int pfsync_sync_ok;
|
|
|
|
struct pfsync_softc {
|
|
#ifdef __FreeBSD__
|
|
struct ifnet *sc_ifp;
|
|
#else
|
|
struct ifnet sc_if;
|
|
#endif
|
|
struct ifnet *sc_sync_ifp;
|
|
|
|
struct ip_moptions sc_imo;
|
|
#ifdef __FreeBSD__
|
|
struct callout sc_tmo;
|
|
#ifdef PFSYNC_TDB
|
|
struct callout sc_tdb_tmo;
|
|
#endif
|
|
struct callout sc_bulk_tmo;
|
|
struct callout sc_bulkfail_tmo;
|
|
#else
|
|
struct timeout sc_tmo;
|
|
struct timeout sc_tdb_tmo;
|
|
struct timeout sc_bulk_tmo;
|
|
struct timeout sc_bulkfail_tmo;
|
|
#endif
|
|
struct in_addr sc_sync_peer;
|
|
struct in_addr sc_sendaddr;
|
|
struct mbuf *sc_mbuf; /* current cumulative mbuf */
|
|
struct mbuf *sc_mbuf_net; /* current cumulative mbuf */
|
|
#ifdef PFSYNC_TDB
|
|
struct mbuf *sc_mbuf_tdb; /* dito for TDB updates */
|
|
#endif
|
|
#ifdef __FreeBSD__
|
|
struct ifqueue sc_ifq;
|
|
struct task sc_send_task;
|
|
#endif
|
|
union sc_statep sc_statep;
|
|
union sc_statep sc_statep_net;
|
|
#ifdef PFSYNC_TDB
|
|
union sc_tdb_statep sc_statep_tdb;
|
|
#endif
|
|
u_int32_t sc_ureq_received;
|
|
u_int32_t sc_ureq_sent;
|
|
struct pf_state *sc_bulk_send_next;
|
|
struct pf_state *sc_bulk_terminator;
|
|
int sc_bulk_tries;
|
|
int sc_maxcount; /* number of states in mtu */
|
|
int sc_maxupdates; /* number of updates/state */
|
|
#ifdef __FreeBSD__
|
|
eventhandler_tag sc_detachtag;
|
|
#endif
|
|
};
|
|
|
|
extern struct pfsync_softc *pfsyncif;
|
|
#endif
|
|
|
|
|
|
struct pfsync_header {
|
|
u_int8_t version;
|
|
#define PFSYNC_VERSION 3
|
|
u_int8_t af;
|
|
u_int8_t action;
|
|
#define PFSYNC_ACT_CLR 0 /* clear all states */
|
|
#define PFSYNC_ACT_INS 1 /* insert state */
|
|
#define PFSYNC_ACT_UPD 2 /* update state */
|
|
#define PFSYNC_ACT_DEL 3 /* delete state */
|
|
#define PFSYNC_ACT_UPD_C 4 /* "compressed" state update */
|
|
#define PFSYNC_ACT_DEL_C 5 /* "compressed" state delete */
|
|
#define PFSYNC_ACT_INS_F 6 /* insert fragment */
|
|
#define PFSYNC_ACT_DEL_F 7 /* delete fragments */
|
|
#define PFSYNC_ACT_UREQ 8 /* request "uncompressed" state */
|
|
#define PFSYNC_ACT_BUS 9 /* Bulk Update Status */
|
|
#define PFSYNC_ACT_TDB_UPD 10 /* TDB replay counter update */
|
|
#define PFSYNC_ACT_MAX 11
|
|
u_int8_t count;
|
|
u_int8_t pf_chksum[PF_MD5_DIGEST_LENGTH];
|
|
} __packed;
|
|
|
|
#define PFSYNC_BULKPACKETS 1 /* # of packets per timeout */
|
|
#define PFSYNC_MAX_BULKTRIES 12
|
|
#define PFSYNC_HDRLEN sizeof(struct pfsync_header)
|
|
#define PFSYNC_ACTIONS \
|
|
"CLR ST", "INS ST", "UPD ST", "DEL ST", \
|
|
"UPD ST COMP", "DEL ST COMP", "INS FR", "DEL FR", \
|
|
"UPD REQ", "BLK UPD STAT", "TDB UPD"
|
|
|
|
#define PFSYNC_DFLTTL 255
|
|
|
|
struct pfsyncstats {
|
|
u_int64_t pfsyncs_ipackets; /* total input packets, IPv4 */
|
|
u_int64_t pfsyncs_ipackets6; /* total input packets, IPv6 */
|
|
u_int64_t pfsyncs_badif; /* not the right interface */
|
|
u_int64_t pfsyncs_badttl; /* TTL is not PFSYNC_DFLTTL */
|
|
u_int64_t pfsyncs_hdrops; /* packets shorter than hdr */
|
|
u_int64_t pfsyncs_badver; /* bad (incl unsupp) version */
|
|
u_int64_t pfsyncs_badact; /* bad action */
|
|
u_int64_t pfsyncs_badlen; /* data length does not match */
|
|
u_int64_t pfsyncs_badauth; /* bad authentication */
|
|
u_int64_t pfsyncs_stale; /* stale state */
|
|
u_int64_t pfsyncs_badval; /* bad values */
|
|
u_int64_t pfsyncs_badstate; /* insert/lookup failed */
|
|
|
|
u_int64_t pfsyncs_opackets; /* total output packets, IPv4 */
|
|
u_int64_t pfsyncs_opackets6; /* total output packets, IPv6 */
|
|
u_int64_t pfsyncs_onomem; /* no memory for an mbuf */
|
|
u_int64_t pfsyncs_oerrors; /* ip output error */
|
|
};
|
|
|
|
/*
|
|
* Configuration structure for SIOCSETPFSYNC SIOCGETPFSYNC
|
|
*/
|
|
struct pfsyncreq {
|
|
char pfsyncr_syncdev[IFNAMSIZ];
|
|
struct in_addr pfsyncr_syncpeer;
|
|
int pfsyncr_maxupdates;
|
|
int pfsyncr_authlevel;
|
|
};
|
|
|
|
#ifdef __FreeBSD__
|
|
#define SIOCSETPFSYNC _IOW('i', 247, struct ifreq)
|
|
#define SIOCGETPFSYNC _IOWR('i', 248, struct ifreq)
|
|
#endif
|
|
|
|
#define pf_state_peer_hton(s,d) do { \
|
|
(d)->seqlo = htonl((s)->seqlo); \
|
|
(d)->seqhi = htonl((s)->seqhi); \
|
|
(d)->seqdiff = htonl((s)->seqdiff); \
|
|
(d)->max_win = htons((s)->max_win); \
|
|
(d)->mss = htons((s)->mss); \
|
|
(d)->state = (s)->state; \
|
|
(d)->wscale = (s)->wscale; \
|
|
if ((s)->scrub) { \
|
|
(d)->scrub.pfss_flags = \
|
|
htons((s)->scrub->pfss_flags & PFSS_TIMESTAMP); \
|
|
(d)->scrub.pfss_ttl = (s)->scrub->pfss_ttl; \
|
|
(d)->scrub.pfss_ts_mod = htonl((s)->scrub->pfss_ts_mod);\
|
|
(d)->scrub.scrub_flag = PFSYNC_SCRUB_FLAG_VALID; \
|
|
} \
|
|
} while (0)
|
|
|
|
#define pf_state_peer_ntoh(s,d) do { \
|
|
(d)->seqlo = ntohl((s)->seqlo); \
|
|
(d)->seqhi = ntohl((s)->seqhi); \
|
|
(d)->seqdiff = ntohl((s)->seqdiff); \
|
|
(d)->max_win = ntohs((s)->max_win); \
|
|
(d)->mss = ntohs((s)->mss); \
|
|
(d)->state = (s)->state; \
|
|
(d)->wscale = (s)->wscale; \
|
|
if ((s)->scrub.scrub_flag == PFSYNC_SCRUB_FLAG_VALID && \
|
|
(d)->scrub != NULL) { \
|
|
(d)->scrub->pfss_flags = \
|
|
ntohs((s)->scrub.pfss_flags) & PFSS_TIMESTAMP; \
|
|
(d)->scrub->pfss_ttl = (s)->scrub.pfss_ttl; \
|
|
(d)->scrub->pfss_ts_mod = ntohl((s)->scrub.pfss_ts_mod);\
|
|
} \
|
|
} while (0)
|
|
|
|
#define pf_state_host_hton(s,d) do { \
|
|
bcopy(&(s)->addr, &(d)->addr, sizeof((d)->addr)); \
|
|
(d)->port = (s)->port; \
|
|
} while (0)
|
|
|
|
#define pf_state_host_ntoh(s,d) do { \
|
|
bcopy(&(s)->addr, &(d)->addr, sizeof((d)->addr)); \
|
|
(d)->port = (s)->port; \
|
|
} while (0)
|
|
|
|
#define pf_state_counter_hton(s,d) do { \
|
|
d[0] = htonl((s>>32)&0xffffffff); \
|
|
d[1] = htonl(s&0xffffffff); \
|
|
} while (0)
|
|
|
|
#define pf_state_counter_ntoh(s,d) do { \
|
|
d = ntohl(s[0]); \
|
|
d = d<<32; \
|
|
d += ntohl(s[1]); \
|
|
} while (0)
|
|
|
|
#ifdef _KERNEL
|
|
#ifdef __FreeBSD__
|
|
void pfsync_input(struct mbuf *, __unused int);
|
|
#else
|
|
void pfsync_input(struct mbuf *, ...);
|
|
#endif
|
|
int pfsync_clear_states(u_int32_t, char *);
|
|
int pfsync_pack_state(u_int8_t, struct pf_state *, int);
|
|
#define pfsync_insert_state(st) do { \
|
|
if ((st->rule.ptr->rule_flag & PFRULE_NOSYNC) || \
|
|
(st->proto == IPPROTO_PFSYNC)) \
|
|
st->sync_flags |= PFSTATE_NOSYNC; \
|
|
else if (!st->sync_flags) \
|
|
pfsync_pack_state(PFSYNC_ACT_INS, (st), \
|
|
PFSYNC_FLAG_COMPRESS); \
|
|
st->sync_flags &= ~PFSTATE_FROMSYNC; \
|
|
} while (0)
|
|
#define pfsync_update_state(st) do { \
|
|
if (!st->sync_flags) \
|
|
pfsync_pack_state(PFSYNC_ACT_UPD, (st), \
|
|
PFSYNC_FLAG_COMPRESS); \
|
|
st->sync_flags &= ~PFSTATE_FROMSYNC; \
|
|
} while (0)
|
|
#define pfsync_delete_state(st) do { \
|
|
if (!st->sync_flags) \
|
|
pfsync_pack_state(PFSYNC_ACT_DEL, (st), \
|
|
PFSYNC_FLAG_COMPRESS); \
|
|
} while (0)
|
|
#ifdef PFSYNC_TDB
|
|
int pfsync_update_tdb(struct tdb *, int);
|
|
#endif
|
|
#endif
|
|
|
|
#endif /* _NET_IF_PFSYNC_H_ */
|