ila_lwt.c 5.6 KB
Newer Older
1 2 3 4 5 6 7 8
#include <linux/errno.h>
#include <linux/ip.h>
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/skbuff.h>
#include <linux/socket.h>
#include <linux/types.h>
#include <net/checksum.h>
9
#include <net/dst_cache.h>
10 11
#include <net/ip.h>
#include <net/ip6_fib.h>
12
#include <net/ip6_route.h>
13 14 15
#include <net/lwtunnel.h>
#include <net/protocol.h>
#include <uapi/linux/ila.h>
T
Tom Herbert 已提交
16
#include "ila.h"
17

18 19 20 21 22 23 24 25 26 27 28 29
struct ila_lwt {
	struct ila_params p;
	struct dst_cache dst_cache;
	u32 connected : 1;
};

static inline struct ila_lwt *ila_lwt_lwtunnel(
	struct lwtunnel_state *lwt)
{
	return (struct ila_lwt *)lwt->data;
}

30
static inline struct ila_params *ila_params_lwtunnel(
31
	struct lwtunnel_state *lwt)
32
{
33
	return &ila_lwt_lwtunnel(lwt)->p;
34 35
}

E
Eric W. Biederman 已提交
36
static int ila_output(struct net *net, struct sock *sk, struct sk_buff *skb)
37
{
38 39 40 41
	struct dst_entry *orig_dst = skb_dst(skb);
	struct ila_lwt *ilwt = ila_lwt_lwtunnel(orig_dst->lwtstate);
	struct dst_entry *dst;
	int err = -EINVAL;
42 43 44 45

	if (skb->protocol != htons(ETH_P_IPV6))
		goto drop;

46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79
	ila_update_ipv6_locator(skb, ila_params_lwtunnel(orig_dst->lwtstate),
				true);

	dst = dst_cache_get(&ilwt->dst_cache);
	if (unlikely(!dst)) {
		struct ipv6hdr *ip6h = ipv6_hdr(skb);
		struct flowi6 fl6;

		/* Lookup a route for the new destination. Take into
		 * account that the base route may already have a gateway.
		 */

		memset(&fl6, 0, sizeof(fl6));
		fl6.flowi6_oif = orig_dst->dev->ifindex;
		fl6.flowi6_iif = LOOPBACK_IFINDEX;
		fl6.daddr = *rt6_nexthop((struct rt6_info *)orig_dst,
					 &ip6h->daddr);

		dst = ip6_route_output(net, NULL, &fl6);
		if (dst->error) {
			err = -EHOSTUNREACH;
			dst_release(dst);
			goto drop;
		}

		dst = xfrm_lookup(net, dst, flowi6_to_flowi(&fl6), NULL, 0);
		if (IS_ERR(dst)) {
			err = PTR_ERR(dst);
			goto drop;
		}

		if (ilwt->connected)
			dst_cache_set_ip6(&ilwt->dst_cache, dst, &fl6.saddr);
	}
80

81 82
	skb_dst_set(skb, dst);
	return dst_output(net, sk, skb);
83 84 85 86 87 88 89 90 91 92 93 94 95

drop:
	kfree_skb(skb);
	return -EINVAL;
}

static int ila_input(struct sk_buff *skb)
{
	struct dst_entry *dst = skb_dst(skb);

	if (skb->protocol != htons(ETH_P_IPV6))
		goto drop;

96
	ila_update_ipv6_locator(skb, ila_params_lwtunnel(dst->lwtstate), false);
97

98
	return dst->lwtstate->orig_input(skb);
99 100 101 102 103 104

drop:
	kfree_skb(skb);
	return -EINVAL;
}

S
stephen hemminger 已提交
105
static const struct nla_policy ila_nl_policy[ILA_ATTR_MAX + 1] = {
106
	[ILA_ATTR_LOCATOR] = { .type = NLA_U64, },
107
	[ILA_ATTR_CSUM_MODE] = { .type = NLA_U8, },
108 109 110
};

static int ila_build_state(struct net_device *dev, struct nlattr *nla,
111
			   unsigned int family, const void *cfg,
112 113
			   struct lwtunnel_state **ts)
{
114
	struct ila_lwt *ilwt;
115 116 117 118
	struct ila_params *p;
	struct nlattr *tb[ILA_ATTR_MAX + 1];
	size_t encap_len = sizeof(*p);
	struct lwtunnel_state *newts;
119
	const struct fib6_config *cfg6 = cfg;
120
	struct ila_addr *iaddr;
121 122
	int ret;

123 124 125
	if (family != AF_INET6)
		return -EINVAL;

126
	if (cfg6->fc_dst_len < 8 * sizeof(struct ila_locator) + 3) {
127 128 129 130 131 132 133 134
		/* Need to have full locator and at least type field
		 * included in destination
		 */
		return -EINVAL;
	}

	iaddr = (struct ila_addr *)&cfg6->fc_dst;

135 136 137 138
	if (!ila_addr_is_ila(iaddr) || ila_csum_neutral_set(iaddr->ident)) {
		/* Don't allow translation for a non-ILA address or checksum
		 * neutral flag to be set.
		 */
139 140 141
		return -EINVAL;
	}

142 143 144 145 146 147 148 149 150 151 152 153
	ret = nla_parse_nested(tb, ILA_ATTR_MAX, nla,
			       ila_nl_policy);
	if (ret < 0)
		return ret;

	if (!tb[ILA_ATTR_LOCATOR])
		return -EINVAL;

	newts = lwtunnel_state_alloc(encap_len);
	if (!newts)
		return -ENOMEM;

154 155 156 157 158 159 160
	ilwt = ila_lwt_lwtunnel(newts);
	ret = dst_cache_init(&ilwt->dst_cache, GFP_ATOMIC);
	if (ret) {
		kfree(newts);
		return ret;
	}

161 162 163
	newts->len = encap_len;
	p = ila_params_lwtunnel(newts);

164
	p->locator.v64 = (__force __be64)nla_get_u64(tb[ILA_ATTR_LOCATOR]);
165

166 167 168 169 170 171
	/* Precompute checksum difference for translation since we
	 * know both the old locator and the new one.
	 */
	p->locator_match = iaddr->loc;
	p->csum_diff = compute_csum_diff8(
		(__be32 *)&p->locator_match, (__be32 *)&p->locator);
172

173 174 175 176 177
	if (tb[ILA_ATTR_CSUM_MODE])
		p->csum_mode = nla_get_u8(tb[ILA_ATTR_CSUM_MODE]);

	ila_init_saved_csum(p);

178 179 180 181
	newts->type = LWTUNNEL_ENCAP_ILA;
	newts->flags |= LWTUNNEL_STATE_OUTPUT_REDIRECT |
			LWTUNNEL_STATE_INPUT_REDIRECT;

182 183 184
	if (cfg6->fc_dst_len == 8 * sizeof(struct in6_addr))
		ilwt->connected = 1;

185 186 187 188 189
	*ts = newts;

	return 0;
}

190 191 192 193 194
static void ila_destroy_state(struct lwtunnel_state *lwt)
{
	dst_cache_destroy(&ila_lwt_lwtunnel(lwt)->dst_cache);
}

195 196 197 198 199
static int ila_fill_encap_info(struct sk_buff *skb,
			       struct lwtunnel_state *lwtstate)
{
	struct ila_params *p = ila_params_lwtunnel(lwtstate);

200
	if (nla_put_u64_64bit(skb, ILA_ATTR_LOCATOR, (__force u64)p->locator.v64,
N
Nicolas Dichtel 已提交
201
			      ILA_ATTR_PAD))
202
		goto nla_put_failure;
203
	if (nla_put_u8(skb, ILA_ATTR_CSUM_MODE, (__force u8)p->csum_mode))
204
		goto nla_put_failure;
205 206 207 208 209 210 211 212 213

	return 0;

nla_put_failure:
	return -EMSGSIZE;
}

static int ila_encap_nlsize(struct lwtunnel_state *lwtstate)
{
214 215 216
	return nla_total_size_64bit(sizeof(u64)) + /* ILA_ATTR_LOCATOR */
	       nla_total_size(sizeof(u8)) +        /* ILA_ATTR_CSUM_MODE */
	       0;
217 218 219 220 221 222 223
}

static int ila_encap_cmp(struct lwtunnel_state *a, struct lwtunnel_state *b)
{
	struct ila_params *a_p = ila_params_lwtunnel(a);
	struct ila_params *b_p = ila_params_lwtunnel(b);

224
	return (a_p->locator.v64 != b_p->locator.v64);
225 226 227 228
}

static const struct lwtunnel_encap_ops ila_encap_ops = {
	.build_state = ila_build_state,
229
	.destroy_state = ila_destroy_state,
230 231 232 233 234 235 236
	.output = ila_output,
	.input = ila_input,
	.fill_encap = ila_fill_encap_info,
	.get_encap_size = ila_encap_nlsize,
	.cmp_encap = ila_encap_cmp,
};

T
Tom Herbert 已提交
237
int ila_lwt_init(void)
238 239 240 241
{
	return lwtunnel_encap_add_ops(&ila_encap_ops, LWTUNNEL_ENCAP_ILA);
}

T
Tom Herbert 已提交
242
void ila_lwt_fini(void)
243 244 245
{
	lwtunnel_encap_del_ops(&ila_encap_ops, LWTUNNEL_ENCAP_ILA);
}