2b6b0bfccb
It seems that PACKET_TAG_IPSEC_IN_CRYPTO_DONE is for network adapters that have IPsec accelerators; a driver sets the mtag to a packet when its device has already encrypted the packet. Unfortunately no driver implements such offload features for long years and seems unlikely to implement them soon. (Note that neither FreeBSD nor Linux doesn't have such drivers.) Let's remove related (unused) codes and simplify the IPsec code.
398 lines
12 KiB
C
398 lines
12 KiB
C
/* $NetBSD: ipsec.h,v 1.51 2017/07/05 03:44:59 ozaki-r Exp $ */
|
|
/* $FreeBSD: /usr/local/www/cvsroot/FreeBSD/src/sys/netipsec/ipsec.h,v 1.2.4.2 2004/02/14 22:23:23 bms Exp $ */
|
|
/* $KAME: ipsec.h,v 1.53 2001/11/20 08:32:38 itojun Exp $ */
|
|
|
|
/*
|
|
* Copyright (C) 1995, 1996, 1997, and 1998 WIDE Project.
|
|
* 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.
|
|
* 3. Neither the name of the project nor the names of its contributors
|
|
* may be used to endorse or promote products derived from this software
|
|
* without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE PROJECT AND CONTRIBUTORS ``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 PROJECT OR CONTRIBUTORS 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 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.
|
|
*/
|
|
|
|
/*
|
|
* IPsec controller part.
|
|
*/
|
|
|
|
#ifndef _NETIPSEC_IPSEC_H_
|
|
#define _NETIPSEC_IPSEC_H_
|
|
|
|
#if defined(_KERNEL_OPT)
|
|
#include "opt_inet.h"
|
|
#include "opt_ipsec.h"
|
|
#endif
|
|
|
|
#include <net/pfkeyv2.h>
|
|
|
|
#ifdef _KERNEL
|
|
#include <sys/socketvar.h>
|
|
|
|
#include <netinet/in_pcb_hdr.h>
|
|
#include <netipsec/keydb.h>
|
|
|
|
/*
|
|
* Security Policy Index
|
|
* Ensure that both address families in the "src" and "dst" are same.
|
|
* When the value of the ul_proto is ICMPv6, the port field in "src"
|
|
* specifies ICMPv6 type, and the port field in "dst" specifies ICMPv6 code.
|
|
*/
|
|
struct secpolicyindex {
|
|
u_int8_t dir; /* direction of packet flow, see blow */
|
|
union sockaddr_union src; /* IP src address for SP */
|
|
union sockaddr_union dst; /* IP dst address for SP */
|
|
u_int8_t prefs; /* prefix length in bits for src */
|
|
u_int8_t prefd; /* prefix length in bits for dst */
|
|
u_int16_t ul_proto; /* upper layer Protocol */
|
|
#ifdef notyet
|
|
uid_t uids;
|
|
uid_t uidd;
|
|
gid_t gids;
|
|
gid_t gidd;
|
|
#endif
|
|
};
|
|
|
|
/* Security Policy Data Base */
|
|
struct secpolicy {
|
|
LIST_ENTRY(secpolicy) chain;
|
|
|
|
u_int refcnt; /* reference count */
|
|
struct secpolicyindex spidx; /* selector */
|
|
u_int32_t id; /* It's unique number on the system. */
|
|
u_int state; /* 0: dead, others: alive */
|
|
#define IPSEC_SPSTATE_DEAD 0
|
|
#define IPSEC_SPSTATE_ALIVE 1
|
|
|
|
u_int policy; /* DISCARD, NONE or IPSEC, see keyv2.h */
|
|
struct ipsecrequest *req;
|
|
/* pointer to the ipsec request tree, */
|
|
/* if policy == IPSEC else this value == NULL.*/
|
|
|
|
/*
|
|
* lifetime handler.
|
|
* the policy can be used without limitiation if both lifetime and
|
|
* validtime are zero.
|
|
* "lifetime" is passed by sadb_lifetime.sadb_lifetime_addtime.
|
|
* "validtime" is passed by sadb_lifetime.sadb_lifetime_usetime.
|
|
*/
|
|
time_t created; /* time created the policy */
|
|
time_t lastused; /* updated every when kernel sends a packet */
|
|
time_t lifetime; /* duration of the lifetime of this policy */
|
|
time_t validtime; /* duration this policy is valid without use */
|
|
};
|
|
|
|
/* Request for IPsec */
|
|
struct ipsecrequest {
|
|
struct ipsecrequest *next;
|
|
/* pointer to next structure */
|
|
/* If NULL, it means the end of chain. */
|
|
struct secasindex saidx;/* hint for search proper SA */
|
|
/* if __ss_len == 0 then no address specified.*/
|
|
u_int level; /* IPsec level defined below. */
|
|
|
|
struct secasvar *sav; /* place holder of SA for use */
|
|
struct secpolicy *sp; /* back pointer to SP */
|
|
};
|
|
|
|
/* security policy in PCB */
|
|
struct inpcbpolicy {
|
|
struct secpolicy *sp_in;
|
|
struct secpolicy *sp_out;
|
|
int priv; /* privileged socket ? */
|
|
|
|
/* cached policy */
|
|
struct {
|
|
struct secpolicy *cachesp;
|
|
struct secpolicyindex cacheidx;
|
|
int cachehint; /* processing requirement hint: */
|
|
#define IPSEC_PCBHINT_UNKNOWN 0 /* Unknown */
|
|
#define IPSEC_PCBHINT_YES 1 /* IPsec processing is required */
|
|
#define IPSEC_PCBHINT_NO 2 /* IPsec processing not required */
|
|
u_int cachegen; /* spdgen when cache filled */
|
|
} sp_cache[3]; /* XXX 3 == IPSEC_DIR_MAX */
|
|
int sp_cacheflags;
|
|
#define IPSEC_PCBSP_CONNECTED 1
|
|
struct inpcb_hdr *sp_inph; /* back pointer */
|
|
};
|
|
|
|
extern u_int ipsec_spdgen;
|
|
|
|
static inline bool
|
|
ipsec_pcb_skip_ipsec(struct inpcbpolicy *pcbsp, int dir)
|
|
{
|
|
|
|
KASSERT(inph_locked(pcbsp->sp_inph));
|
|
|
|
return pcbsp->sp_cache[(dir)].cachehint == IPSEC_PCBHINT_NO &&
|
|
pcbsp->sp_cache[(dir)].cachegen == ipsec_spdgen;
|
|
}
|
|
|
|
/* SP acquiring list table. */
|
|
struct secspacq {
|
|
LIST_ENTRY(secspacq) chain;
|
|
|
|
struct secpolicyindex spidx;
|
|
|
|
time_t created; /* for lifetime */
|
|
int count; /* for lifetime */
|
|
/* XXX: here is mbuf place holder to be sent ? */
|
|
};
|
|
#endif /* _KERNEL */
|
|
|
|
/* buffer size for formatted output of ipsec address (addr + '%' + scope_id?) */
|
|
#define IPSEC_ADDRSTRLEN (INET6_ADDRSTRLEN + 11)
|
|
/* buffer size for ipsec_logsastr() */
|
|
#define IPSEC_LOGSASTRLEN 192
|
|
|
|
/* according to IANA assignment, port 0x0000 and proto 0xff are reserved. */
|
|
#define IPSEC_PORT_ANY 0
|
|
#define IPSEC_ULPROTO_ANY 255
|
|
#define IPSEC_PROTO_ANY 255
|
|
|
|
/* mode of security protocol */
|
|
/* NOTE: DON'T use IPSEC_MODE_ANY at SPD. It's only use in SAD */
|
|
#define IPSEC_MODE_ANY 0 /* i.e. wildcard. */
|
|
#define IPSEC_MODE_TRANSPORT 1
|
|
#define IPSEC_MODE_TUNNEL 2
|
|
#define IPSEC_MODE_TCPMD5 3 /* TCP MD5 mode */
|
|
|
|
/*
|
|
* Direction of security policy.
|
|
* NOTE: Since INVALID is used just as flag.
|
|
* The other are used for loop counter too.
|
|
*/
|
|
#define IPSEC_DIR_ANY 0
|
|
#define IPSEC_DIR_INBOUND 1
|
|
#define IPSEC_DIR_OUTBOUND 2
|
|
#define IPSEC_DIR_MAX 3
|
|
#define IPSEC_DIR_INVALID 4
|
|
|
|
#define IPSEC_DIR_IS_VALID(dir) ((dir) >= 0 && (dir) <= IPSEC_DIR_MAX)
|
|
#define IPSEC_DIR_IS_INOROUT(dir) ((dir) == IPSEC_DIR_INBOUND || \
|
|
(dir) == IPSEC_DIR_OUTBOUND)
|
|
|
|
/* Policy level */
|
|
/*
|
|
* IPSEC, ENTRUST and BYPASS are allowed for setsockopt() in PCB,
|
|
* DISCARD, IPSEC and NONE are allowed for setkey() in SPD.
|
|
* DISCARD and NONE are allowed for system default.
|
|
*/
|
|
#define IPSEC_POLICY_DISCARD 0 /* discarding packet */
|
|
#define IPSEC_POLICY_NONE 1 /* through IPsec engine */
|
|
#define IPSEC_POLICY_IPSEC 2 /* do IPsec */
|
|
#define IPSEC_POLICY_ENTRUST 3 /* consulting SPD if present. */
|
|
#define IPSEC_POLICY_BYPASS 4 /* only for privileged socket. */
|
|
|
|
/* Security protocol level */
|
|
#define IPSEC_LEVEL_DEFAULT 0 /* reference to system default */
|
|
#define IPSEC_LEVEL_USE 1 /* use SA if present. */
|
|
#define IPSEC_LEVEL_REQUIRE 2 /* require SA. */
|
|
#define IPSEC_LEVEL_UNIQUE 3 /* unique SA. */
|
|
|
|
#define IPSEC_MANUAL_REQID_MAX 0x3fff
|
|
/*
|
|
* if security policy level == unique, this id
|
|
* indicate to a relative SA for use, else is
|
|
* zero.
|
|
* 1 - 0x3fff are reserved for manual keying.
|
|
* 0 are reserved for above reason. Others is
|
|
* for kernel use.
|
|
* Note that this id doesn't identify SA
|
|
* by only itself.
|
|
*/
|
|
#define IPSEC_REPLAYWSIZE 32
|
|
|
|
#ifdef _KERNEL
|
|
struct ipsec_output_state {
|
|
struct mbuf *m;
|
|
struct route *ro;
|
|
struct sockaddr *dst;
|
|
};
|
|
|
|
struct ipsec_history {
|
|
int ih_proto;
|
|
u_int32_t ih_spi;
|
|
};
|
|
|
|
extern int ipsec_debug;
|
|
#ifdef IPSEC_DEBUG
|
|
extern int ipsec_replay;
|
|
extern int ipsec_integrity;
|
|
#endif
|
|
|
|
extern struct secpolicy ip4_def_policy;
|
|
extern int ip4_esp_trans_deflev;
|
|
extern int ip4_esp_net_deflev;
|
|
extern int ip4_ah_trans_deflev;
|
|
extern int ip4_ah_net_deflev;
|
|
extern int ip4_ah_cleartos;
|
|
extern int ip4_ah_offsetmask;
|
|
extern int ip4_ipsec_dfbit;
|
|
extern int ip4_ipsec_ecn;
|
|
extern int ip4_esp_randpad;
|
|
extern int crypto_support;
|
|
|
|
#include <sys/syslog.h>
|
|
#define ipseclog(x) do { if (ipsec_debug) log x; } while (0)
|
|
/* for openbsd compatibility */
|
|
#define DPRINTF(x) do { if (ipsec_debug) printf x; } while (0)
|
|
|
|
#define IPSECLOG(level, fmt, args...) \
|
|
do { \
|
|
if (ipsec_debug) \
|
|
log(level, "%s: " fmt, __func__, ##args); \
|
|
} while (0)
|
|
|
|
void ipsec_pcbconn (struct inpcbpolicy *);
|
|
void ipsec_pcbdisconn (struct inpcbpolicy *);
|
|
void ipsec_invalpcbcacheall (void);
|
|
|
|
struct tdb_ident;
|
|
struct secpolicy *ipsec_getpolicy (const struct tdb_ident*, u_int);
|
|
struct inpcb;
|
|
struct secpolicy *ipsec4_checkpolicy (struct mbuf *, u_int, u_int,
|
|
int *, struct inpcb *);
|
|
struct secpolicy * ipsec_getpolicybyaddr(struct mbuf *, u_int,
|
|
int, int *);
|
|
int ipsec4_output(struct mbuf *, struct inpcb *, int,
|
|
u_long *, bool *, bool *);
|
|
int ipsec4_input(struct mbuf *, int);
|
|
int ipsec4_forward(struct mbuf *, int *);
|
|
#ifdef INET6
|
|
int ipsec6_input(struct mbuf *);
|
|
#endif
|
|
|
|
static __inline struct secpolicy*
|
|
ipsec4_getpolicybysock(
|
|
struct mbuf *m,
|
|
u_int dir,
|
|
const struct socket *so,
|
|
int *err
|
|
)
|
|
{
|
|
panic("ipsec4_getpolicybysock");
|
|
}
|
|
|
|
static __inline int
|
|
ipsec_copy_pcbpolicy(
|
|
struct inpcbpolicy *oldp,
|
|
struct inpcbpolicy *newp
|
|
)
|
|
{
|
|
/*XXX do nothing */
|
|
return (0);
|
|
}
|
|
|
|
struct inpcb;
|
|
#define ipsec_init_pcbpolicy ipsec_init_policy
|
|
int ipsec_init_policy (struct socket *so, struct inpcbpolicy **);
|
|
int ipsec_copy_policy
|
|
(const struct inpcbpolicy *, struct inpcbpolicy *);
|
|
u_int ipsec_get_reqlevel (const struct ipsecrequest *);
|
|
int ipsec_in_reject (const struct secpolicy *, const struct mbuf *);
|
|
|
|
int ipsec4_set_policy (struct inpcb *, int, const void *, size_t, kauth_cred_t);
|
|
int ipsec4_get_policy (struct inpcb *, const void *, size_t, struct mbuf **);
|
|
int ipsec4_delete_pcbpolicy (struct inpcb *);
|
|
int ipsec4_in_reject (struct mbuf *, struct inpcb *);
|
|
|
|
|
|
struct secas;
|
|
struct tcpcb;
|
|
int ipsec_chkreplay (u_int32_t, const struct secasvar *);
|
|
int ipsec_updatereplay (u_int32_t, const struct secasvar *);
|
|
|
|
size_t ipsec4_hdrsiz (struct mbuf *, u_int, struct inpcb *);
|
|
size_t ipsec4_hdrsiz_tcp (struct tcpcb *);
|
|
#define ipsec4_getpolicybyaddr ipsec_getpolicybyaddr
|
|
|
|
union sockaddr_union;
|
|
const char *ipsec_address(const union sockaddr_union* sa, char *, size_t);
|
|
const char *ipsec_logsastr(const struct secasvar *, char *, size_t);
|
|
|
|
void ipsec_dumpmbuf (struct mbuf *);
|
|
|
|
/* NetBSD protosw ctlin entrypoint */
|
|
void *esp4_ctlinput(int, const struct sockaddr *, void *);
|
|
void *ah4_ctlinput(int, const struct sockaddr *, void *);
|
|
|
|
struct m_tag;
|
|
void ipsec4_common_input(struct mbuf *m, ...);
|
|
int ipsec4_common_input_cb(struct mbuf *, struct secasvar *,
|
|
int, int);
|
|
int ipsec4_process_packet(struct mbuf *, struct ipsecrequest *);
|
|
int ipsec_process_done (struct mbuf *, struct ipsecrequest *);
|
|
#define ipsec_indone(m) \
|
|
(m_tag_find((m), PACKET_TAG_IPSEC_IN_DONE, NULL) != NULL)
|
|
|
|
#define ipsec_outdone(m) \
|
|
(m_tag_find((m), PACKET_TAG_IPSEC_OUT_DONE, NULL) != NULL)
|
|
|
|
struct mbuf *ipsec_copypkt (struct mbuf *);
|
|
|
|
void m_checkalignment(const char* , struct mbuf *, int, int);
|
|
struct mbuf *m_clone(struct mbuf *);
|
|
struct mbuf *m_makespace(struct mbuf *, int, int, int *);
|
|
void *m_pad(struct mbuf *, int );
|
|
int m_striphdr(struct mbuf *, int, int);
|
|
|
|
/* Per-socket caching of IPsec output policy */
|
|
static __inline
|
|
int ipsec_clear_socket_cache(struct mbuf *m)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
void nat_t_ports_get(struct mbuf *, u_int16_t *, u_int16_t *);
|
|
|
|
extern int ipsec_used __read_mostly, ipsec_enabled __read_mostly;
|
|
|
|
#endif /* _KERNEL */
|
|
|
|
#ifndef _KERNEL
|
|
char *ipsec_set_policy (const char *, int);
|
|
int ipsec_get_policylen (char *);
|
|
char *ipsec_dump_policy (char *, const char *);
|
|
|
|
const char *ipsec_strerror (void);
|
|
#endif /* !_KERNEL */
|
|
|
|
#ifdef _KERNEL
|
|
/* External declarations of per-file init functions */
|
|
void ah_attach(void);
|
|
void esp_attach(void);
|
|
void ipcomp_attach(void);
|
|
void ipe4_attach(void);
|
|
void ipe4_attach(void);
|
|
void tcpsignature_attach(void);
|
|
|
|
void ipsec_attach(void);
|
|
|
|
void sysctl_net_inet_ipsec_setup(struct sysctllog **);
|
|
#ifdef INET6
|
|
void sysctl_net_inet6_ipsec6_setup(struct sysctllog **);
|
|
#endif
|
|
|
|
#endif /* _KERNEL */
|
|
#endif /* !_NETIPSEC_IPSEC_H_ */
|