2015-07-21 16:43:56 +08:00
|
|
|
#ifndef __NET_DST_METADATA_H
|
|
|
|
#define __NET_DST_METADATA_H 1
|
|
|
|
|
|
|
|
#include <linux/skbuff.h>
|
|
|
|
#include <net/ip_tunnels.h>
|
|
|
|
#include <net/dst.h>
|
|
|
|
|
|
|
|
struct metadata_dst {
|
|
|
|
struct dst_entry dst;
|
2015-07-21 16:43:58 +08:00
|
|
|
union {
|
|
|
|
struct ip_tunnel_info tun_info;
|
|
|
|
} u;
|
2015-07-21 16:43:56 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
static inline struct metadata_dst *skb_metadata_dst(struct sk_buff *skb)
|
|
|
|
{
|
|
|
|
struct metadata_dst *md_dst = (struct metadata_dst *) skb_dst(skb);
|
|
|
|
|
|
|
|
if (md_dst && md_dst->dst.flags & DST_METADATA)
|
|
|
|
return md_dst;
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2015-08-20 19:56:25 +08:00
|
|
|
static inline struct ip_tunnel_info *skb_tunnel_info(struct sk_buff *skb)
|
2015-07-21 16:43:58 +08:00
|
|
|
{
|
|
|
|
struct metadata_dst *md_dst = skb_metadata_dst(skb);
|
2015-08-20 19:56:25 +08:00
|
|
|
struct dst_entry *dst;
|
2015-07-21 16:43:58 +08:00
|
|
|
|
|
|
|
if (md_dst)
|
|
|
|
return &md_dst->u.tun_info;
|
|
|
|
|
2015-08-20 19:56:25 +08:00
|
|
|
dst = skb_dst(skb);
|
|
|
|
if (dst && dst->lwtstate)
|
|
|
|
return lwt_tun_info(dst->lwtstate);
|
2015-07-21 16:44:00 +08:00
|
|
|
|
2015-07-21 16:43:58 +08:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2015-07-21 16:43:56 +08:00
|
|
|
static inline bool skb_valid_dst(const struct sk_buff *skb)
|
|
|
|
{
|
|
|
|
struct dst_entry *dst = skb_dst(skb);
|
|
|
|
|
|
|
|
return dst && !(dst->flags & DST_METADATA);
|
|
|
|
}
|
|
|
|
|
2016-01-21 09:59:49 +08:00
|
|
|
static inline int skb_metadata_dst_cmp(const struct sk_buff *skb_a,
|
|
|
|
const struct sk_buff *skb_b)
|
|
|
|
{
|
|
|
|
const struct metadata_dst *a, *b;
|
|
|
|
|
|
|
|
if (!(skb_a->_skb_refdst | skb_b->_skb_refdst))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
a = (const struct metadata_dst *) skb_dst(skb_a);
|
|
|
|
b = (const struct metadata_dst *) skb_dst(skb_b);
|
|
|
|
|
|
|
|
if (!a != !b || a->u.tun_info.options_len != b->u.tun_info.options_len)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
return memcmp(&a->u.tun_info, &b->u.tun_info,
|
|
|
|
sizeof(a->u.tun_info) + a->u.tun_info.options_len);
|
|
|
|
}
|
|
|
|
|
2016-02-12 22:43:57 +08:00
|
|
|
void metadata_dst_free(struct metadata_dst *);
|
2015-07-21 16:43:56 +08:00
|
|
|
struct metadata_dst *metadata_dst_alloc(u8 optslen, gfp_t flags);
|
bpf: add helpers to access tunnel metadata
Introduce helpers to let eBPF programs attached to TC manipulate tunnel metadata:
bpf_skb_[gs]et_tunnel_key(skb, key, size, flags)
skb: pointer to skb
key: pointer to 'struct bpf_tunnel_key'
size: size of 'struct bpf_tunnel_key'
flags: room for future extensions
First eBPF program that uses these helpers will allocate per_cpu
metadata_dst structures that will be used on TX.
On RX metadata_dst is allocated by tunnel driver.
Typical usage for TX:
struct bpf_tunnel_key tkey;
... populate tkey ...
bpf_skb_set_tunnel_key(skb, &tkey, sizeof(tkey), 0);
bpf_clone_redirect(skb, vxlan_dev_ifindex, 0);
RX:
struct bpf_tunnel_key tkey = {};
bpf_skb_get_tunnel_key(skb, &tkey, sizeof(tkey), 0);
... lookup or redirect based on tkey ...
'struct bpf_tunnel_key' will be extended in the future by adding
elements to the end and the 'size' argument will indicate which fields
are populated, thereby keeping backwards compatibility.
The 'flags' argument may be used as well when the 'size' is not enough or
to indicate completely different layout of bpf_tunnel_key.
Signed-off-by: Alexei Starovoitov <ast@plumgrid.com>
Acked-by: Thomas Graf <tgraf@suug.ch>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-07-31 06:36:57 +08:00
|
|
|
struct metadata_dst __percpu *metadata_dst_alloc_percpu(u8 optslen, gfp_t flags);
|
2015-07-21 16:43:56 +08:00
|
|
|
|
2015-08-31 09:09:38 +08:00
|
|
|
static inline struct metadata_dst *tun_rx_dst(int md_size)
|
2015-08-27 14:46:50 +08:00
|
|
|
{
|
|
|
|
struct metadata_dst *tun_dst;
|
|
|
|
|
|
|
|
tun_dst = metadata_dst_alloc(md_size, GFP_ATOMIC);
|
|
|
|
if (!tun_dst)
|
|
|
|
return NULL;
|
|
|
|
|
2015-08-31 09:09:38 +08:00
|
|
|
tun_dst->u.tun_info.options_len = 0;
|
|
|
|
tun_dst->u.tun_info.mode = 0;
|
2015-08-27 14:46:50 +08:00
|
|
|
return tun_dst;
|
|
|
|
}
|
|
|
|
|
2015-10-23 09:17:16 +08:00
|
|
|
static inline struct metadata_dst *tun_dst_unclone(struct sk_buff *skb)
|
|
|
|
{
|
|
|
|
struct metadata_dst *md_dst = skb_metadata_dst(skb);
|
2015-11-04 20:49:49 +08:00
|
|
|
int md_size;
|
2015-10-23 09:17:16 +08:00
|
|
|
struct metadata_dst *new_md;
|
|
|
|
|
|
|
|
if (!md_dst)
|
|
|
|
return ERR_PTR(-EINVAL);
|
|
|
|
|
2015-11-04 20:49:49 +08:00
|
|
|
md_size = md_dst->u.tun_info.options_len;
|
2015-10-23 09:17:16 +08:00
|
|
|
new_md = metadata_dst_alloc(md_size, GFP_ATOMIC);
|
|
|
|
if (!new_md)
|
|
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
|
|
|
|
memcpy(&new_md->u.tun_info, &md_dst->u.tun_info,
|
|
|
|
sizeof(struct ip_tunnel_info) + md_size);
|
|
|
|
skb_dst_drop(skb);
|
|
|
|
dst_hold(&new_md->dst);
|
|
|
|
skb_dst_set(skb, &new_md->dst);
|
|
|
|
return new_md;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline struct ip_tunnel_info *skb_tunnel_info_unclone(struct sk_buff *skb)
|
|
|
|
{
|
|
|
|
struct metadata_dst *dst;
|
|
|
|
|
|
|
|
dst = tun_dst_unclone(skb);
|
|
|
|
if (IS_ERR(dst))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return &dst->u.tun_info;
|
|
|
|
}
|
|
|
|
|
2015-08-27 14:46:50 +08:00
|
|
|
static inline struct metadata_dst *ip_tun_rx_dst(struct sk_buff *skb,
|
|
|
|
__be16 flags,
|
|
|
|
__be64 tunnel_id,
|
|
|
|
int md_size)
|
|
|
|
{
|
|
|
|
const struct iphdr *iph = ip_hdr(skb);
|
|
|
|
struct metadata_dst *tun_dst;
|
|
|
|
|
2015-08-31 09:09:38 +08:00
|
|
|
tun_dst = tun_rx_dst(md_size);
|
2015-08-27 14:46:50 +08:00
|
|
|
if (!tun_dst)
|
|
|
|
return NULL;
|
|
|
|
|
2015-08-31 09:09:38 +08:00
|
|
|
ip_tunnel_key_init(&tun_dst->u.tun_info.key,
|
|
|
|
iph->saddr, iph->daddr, iph->tos, iph->ttl,
|
|
|
|
0, 0, tunnel_id, flags);
|
2015-08-27 14:46:50 +08:00
|
|
|
return tun_dst;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline struct metadata_dst *ipv6_tun_rx_dst(struct sk_buff *skb,
|
|
|
|
__be16 flags,
|
|
|
|
__be64 tunnel_id,
|
|
|
|
int md_size)
|
|
|
|
{
|
|
|
|
const struct ipv6hdr *ip6h = ipv6_hdr(skb);
|
|
|
|
struct metadata_dst *tun_dst;
|
|
|
|
struct ip_tunnel_info *info;
|
|
|
|
|
2015-08-31 09:09:38 +08:00
|
|
|
tun_dst = tun_rx_dst(md_size);
|
2015-08-27 14:46:50 +08:00
|
|
|
if (!tun_dst)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
info = &tun_dst->u.tun_info;
|
2015-08-31 09:09:38 +08:00
|
|
|
info->mode = IP_TUNNEL_INFO_IPV6;
|
|
|
|
info->key.tun_flags = flags;
|
|
|
|
info->key.tun_id = tunnel_id;
|
|
|
|
info->key.tp_src = 0;
|
|
|
|
info->key.tp_dst = 0;
|
|
|
|
|
2015-08-27 14:46:50 +08:00
|
|
|
info->key.u.ipv6.src = ip6h->saddr;
|
|
|
|
info->key.u.ipv6.dst = ip6h->daddr;
|
|
|
|
info->key.tos = ipv6_get_dsfield(ip6h);
|
|
|
|
info->key.ttl = ip6h->hop_limit;
|
|
|
|
return tun_dst;
|
|
|
|
}
|
|
|
|
|
2015-07-21 16:43:56 +08:00
|
|
|
#endif /* __NET_DST_METADATA_H */
|