esp4.c 16.4 KB
Newer Older
1 2
#define pr_fmt(fmt) "IPsec: " fmt

3 4
#include <crypto/aead.h>
#include <crypto/authenc.h>
5
#include <linux/err.h>
L
Linus Torvalds 已提交
6 7 8 9
#include <linux/module.h>
#include <net/ip.h>
#include <net/xfrm.h>
#include <net/esp.h>
10
#include <linux/scatterlist.h>
H
Herbert Xu 已提交
11
#include <linux/kernel.h>
L
Linus Torvalds 已提交
12
#include <linux/pfkeyv2.h>
13 14
#include <linux/rtnetlink.h>
#include <linux/slab.h>
15
#include <linux/spinlock.h>
16
#include <linux/in6.h>
L
Linus Torvalds 已提交
17
#include <net/icmp.h>
18
#include <net/protocol.h>
L
Linus Torvalds 已提交
19 20
#include <net/udp.h>

21 22 23 24 25 26 27
struct esp_skb_cb {
	struct xfrm_skb_cb xfrm;
	void *tmp;
};

#define ESP_SKB_CB(__skb) ((struct esp_skb_cb *)&((__skb)->cb[0]))

28 29
static u32 esp4_get_mtu(struct xfrm_state *x, int mtu);

30 31 32 33 34 35 36 37
/*
 * Allocate an AEAD request structure with extra space for SG and IV.
 *
 * For alignment considerations the IV is placed at the front, followed
 * by the request and finally the SG list.
 *
 * TODO: Use spare space in skb for this where possible.
 */
38
static void *esp_alloc_tmp(struct crypto_aead *aead, int nfrags, int seqhilen)
39 40 41
{
	unsigned int len;

42 43 44 45
	len = seqhilen;

	len += crypto_aead_ivsize(aead);

46 47 48 49 50 51 52 53 54 55 56 57 58 59
	if (len) {
		len += crypto_aead_alignmask(aead) &
		       ~(crypto_tfm_ctx_alignment() - 1);
		len = ALIGN(len, crypto_tfm_ctx_alignment());
	}

	len += sizeof(struct aead_givcrypt_request) + crypto_aead_reqsize(aead);
	len = ALIGN(len, __alignof__(struct scatterlist));

	len += sizeof(struct scatterlist) * nfrags;

	return kmalloc(len, GFP_ATOMIC);
}

60 61 62 63 64
static inline __be32 *esp_tmp_seqhi(void *tmp)
{
	return PTR_ALIGN((__be32 *)tmp, __alignof__(__be32));
}
static inline u8 *esp_tmp_iv(struct crypto_aead *aead, void *tmp, int seqhilen)
65 66
{
	return crypto_aead_ivsize(aead) ?
67 68
	       PTR_ALIGN((u8 *)tmp + seqhilen,
			 crypto_aead_alignmask(aead) + 1) : tmp + seqhilen;
69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115
}

static inline struct aead_givcrypt_request *esp_tmp_givreq(
	struct crypto_aead *aead, u8 *iv)
{
	struct aead_givcrypt_request *req;

	req = (void *)PTR_ALIGN(iv + crypto_aead_ivsize(aead),
				crypto_tfm_ctx_alignment());
	aead_givcrypt_set_tfm(req, aead);
	return req;
}

static inline struct aead_request *esp_tmp_req(struct crypto_aead *aead, u8 *iv)
{
	struct aead_request *req;

	req = (void *)PTR_ALIGN(iv + crypto_aead_ivsize(aead),
				crypto_tfm_ctx_alignment());
	aead_request_set_tfm(req, aead);
	return req;
}

static inline struct scatterlist *esp_req_sg(struct crypto_aead *aead,
					     struct aead_request *req)
{
	return (void *)ALIGN((unsigned long)(req + 1) +
			     crypto_aead_reqsize(aead),
			     __alignof__(struct scatterlist));
}

static inline struct scatterlist *esp_givreq_sg(
	struct crypto_aead *aead, struct aead_givcrypt_request *req)
{
	return (void *)ALIGN((unsigned long)(req + 1) +
			     crypto_aead_reqsize(aead),
			     __alignof__(struct scatterlist));
}

static void esp_output_done(struct crypto_async_request *base, int err)
{
	struct sk_buff *skb = base->data;

	kfree(ESP_SKB_CB(skb)->tmp);
	xfrm_output_resume(skb, err);
}

L
Linus Torvalds 已提交
116 117 118 119
static int esp_output(struct xfrm_state *x, struct sk_buff *skb)
{
	int err;
	struct ip_esp_hdr *esph;
120 121 122 123
	struct crypto_aead *aead;
	struct aead_givcrypt_request *req;
	struct scatterlist *sg;
	struct scatterlist *asg;
L
Linus Torvalds 已提交
124
	struct sk_buff *trailer;
125 126
	void *tmp;
	u8 *iv;
127
	u8 *tail;
L
Linus Torvalds 已提交
128 129 130
	int blksize;
	int clen;
	int alen;
131 132
	int plen;
	int tfclen;
L
Linus Torvalds 已提交
133
	int nfrags;
134 135 136 137
	int assoclen;
	int sglists;
	int seqhilen;
	__be32 *seqhi;
L
Linus Torvalds 已提交
138

139
	/* skb is pure payload to encrypt */
L
Linus Torvalds 已提交
140

141
	aead = x->data;
142 143
	alen = crypto_aead_authsize(aead);

144 145 146 147 148 149 150 151 152
	tfclen = 0;
	if (x->tfcpad) {
		struct xfrm_dst *dst = (struct xfrm_dst *)skb_dst(skb);
		u32 padto;

		padto = min(x->tfcpad, esp4_get_mtu(x, dst->child_mtu_cached));
		if (skb->len < padto)
			tfclen = padto - skb->len;
	}
153
	blksize = ALIGN(crypto_aead_blocksize(aead), 4);
154 155
	clen = ALIGN(skb->len + 2 + tfclen, blksize);
	plen = clen - skb->len - tfclen;
156

157 158
	err = skb_cow_data(skb, tfclen + plen + alen, &trailer);
	if (err < 0)
159 160
		goto error;
	nfrags = err;
L
Linus Torvalds 已提交
161

162 163 164 165 166 167 168 169 170 171 172
	assoclen = sizeof(*esph);
	sglists = 1;
	seqhilen = 0;

	if (x->props.flags & XFRM_STATE_ESN) {
		sglists += 2;
		seqhilen += sizeof(__be32);
		assoclen += seqhilen;
	}

	tmp = esp_alloc_tmp(aead, nfrags + sglists, seqhilen);
173 174
	if (!tmp) {
		err = -ENOMEM;
L
Linus Torvalds 已提交
175
		goto error;
176
	}
L
Linus Torvalds 已提交
177

178 179
	seqhi = esp_tmp_seqhi(tmp);
	iv = esp_tmp_iv(aead, tmp, seqhilen);
180 181
	req = esp_tmp_givreq(aead, iv);
	asg = esp_givreq_sg(aead, req);
182
	sg = asg + sglists;
183

L
Linus Torvalds 已提交
184
	/* Fill padding... */
185
	tail = skb_tail_pointer(trailer);
186 187 188 189
	if (tfclen) {
		memset(tail, 0, tfclen);
		tail += tfclen;
	}
L
Linus Torvalds 已提交
190 191
	do {
		int i;
192
		for (i = 0; i < plen - 2; i++)
193
			tail[i] = i + 1;
L
Linus Torvalds 已提交
194
	} while (0);
195 196
	tail[plen - 2] = plen - 2;
	tail[plen - 1] = *skb_mac_header(skb);
197
	pskb_put(skb, trailer, clen - skb->len + alen);
L
Linus Torvalds 已提交
198

199
	skb_push(skb, -skb_network_offset(skb));
200
	esph = ip_esp_hdr(skb);
201
	*skb_mac_header(skb) = IPPROTO_ESP;
L
Linus Torvalds 已提交
202 203 204 205 206

	/* this is non-NULL only with UDP Encapsulation */
	if (x->encap) {
		struct xfrm_encap_tmpl *encap = x->encap;
		struct udphdr *uh;
A
Al Viro 已提交
207
		__be32 *udpdata32;
A
Al Viro 已提交
208
		__be16 sport, dport;
209 210 211 212 213 214 215
		int encap_type;

		spin_lock_bh(&x->lock);
		sport = encap->encap_sport;
		dport = encap->encap_dport;
		encap_type = encap->encap_type;
		spin_unlock_bh(&x->lock);
L
Linus Torvalds 已提交
216 217

		uh = (struct udphdr *)esph;
218 219 220
		uh->source = sport;
		uh->dest = dport;
		uh->len = htons(skb->len - skb_transport_offset(skb));
L
Linus Torvalds 已提交
221 222
		uh->check = 0;

223
		switch (encap_type) {
L
Linus Torvalds 已提交
224 225 226 227 228
		default:
		case UDP_ENCAP_ESPINUDP:
			esph = (struct ip_esp_hdr *)(uh + 1);
			break;
		case UDP_ENCAP_ESPINUDP_NON_IKE:
A
Al Viro 已提交
229
			udpdata32 = (__be32 *)(uh + 1);
L
Linus Torvalds 已提交
230 231 232 233 234
			udpdata32[0] = udpdata32[1] = 0;
			esph = (struct ip_esp_hdr *)(udpdata32 + 2);
			break;
		}

235 236
		*skb_mac_header(skb) = IPPROTO_UDP;
	}
L
Linus Torvalds 已提交
237 238

	esph->spi = x->id.spi;
239
	esph->seq_no = htonl(XFRM_SKB_CB(skb)->seq.output.low);
L
Linus Torvalds 已提交
240

241 242 243 244
	sg_init_table(sg, nfrags);
	skb_to_sgvec(skb, sg,
		     esph->enc_data + crypto_aead_ivsize(aead) - skb->data,
		     clen + alen);
245 246 247 248 249 250 251 252 253

	if ((x->props.flags & XFRM_STATE_ESN)) {
		sg_init_table(asg, 3);
		sg_set_buf(asg, &esph->spi, sizeof(__be32));
		*seqhi = htonl(XFRM_SKB_CB(skb)->seq.output.hi);
		sg_set_buf(asg + 1, seqhi, seqhilen);
		sg_set_buf(asg + 2, &esph->seq_no, sizeof(__be32));
	} else
		sg_init_one(asg, esph, sizeof(*esph));
254 255 256

	aead_givcrypt_set_callback(req, 0, esp_output_done, skb);
	aead_givcrypt_set_crypt(req, sg, sg, clen, iv);
257
	aead_givcrypt_set_assoc(req, asg, assoclen);
258
	aead_givcrypt_set_giv(req, esph->enc_data,
259
			      XFRM_SKB_CB(skb)->seq.output.low);
260 261 262 263 264

	ESP_SKB_CB(skb)->tmp = tmp;
	err = crypto_aead_givencrypt(req);
	if (err == -EINPROGRESS)
		goto error;
L
Linus Torvalds 已提交
265

266 267
	if (err == -EBUSY)
		err = NET_XMIT_DROP;
L
Linus Torvalds 已提交
268

269
	kfree(tmp);
270

L
Linus Torvalds 已提交
271 272 273 274
error:
	return err;
}

275
static int esp_input_done2(struct sk_buff *skb, int err)
L
Linus Torvalds 已提交
276
{
277
	const struct iphdr *iph;
278
	struct xfrm_state *x = xfrm_input_state(skb);
279
	struct crypto_aead *aead = x->data;
280 281 282
	int alen = crypto_aead_authsize(aead);
	int hlen = sizeof(struct ip_esp_hdr) + crypto_aead_ivsize(aead);
	int elen = skb->len - hlen;
283
	int ihl;
284 285
	u8 nexthdr[2];
	int padlen;
L
Linus Torvalds 已提交
286

287
	kfree(ESP_SKB_CB(skb)->tmp);
288

289
	if (unlikely(err))
290
		goto out;
L
Linus Torvalds 已提交
291

292 293
	if (skb_copy_bits(skb, skb->len-alen-2, nexthdr, 2))
		BUG();
L
Linus Torvalds 已提交
294

295
	err = -EINVAL;
296
	padlen = nexthdr[0];
297
	if (padlen + 2 + alen >= elen)
298
		goto out;
L
Linus Torvalds 已提交
299

300
	/* ... check padding bits here. Silly. :-) */
L
Linus Torvalds 已提交
301

302
	iph = ip_hdr(skb);
303 304
	ihl = iph->ihl * 4;

305 306
	if (x->encap) {
		struct xfrm_encap_tmpl *encap = x->encap;
307
		struct udphdr *uh = (void *)(skb_network_header(skb) + ihl);
308 309 310 311 312 313 314 315 316 317 318 319 320

		/*
		 * 1) if the NAT-T peer's IP or port changed then
		 *    advertize the change to the keying daemon.
		 *    This is an inbound SA, so just compare
		 *    SRC ports.
		 */
		if (iph->saddr != x->props.saddr.a4 ||
		    uh->source != encap->encap_sport) {
			xfrm_address_t ipaddr;

			ipaddr.a4 = iph->saddr;
			km_new_mapping(x, &ipaddr, uh->source);
321

322 323 324 325 326 327 328
			/* XXX: perhaps add an extra
			 * policy check here, to see
			 * if we should allow or
			 * reject a packet from a
			 * different source
			 * address/port.
			 */
L
Linus Torvalds 已提交
329
		}
330

331 332 333 334 335 336 337
		/*
		 * 2) ignore UDP/TCP checksums in case
		 *    of NAT-T in Transport Mode, or
		 *    perform other post-processing fixes
		 *    as per draft-ietf-ipsec-udp-encaps-06,
		 *    section 3.1.2
		 */
338
		if (x->props.mode == XFRM_MODE_TRANSPORT)
339
			skb->ip_summed = CHECKSUM_UNNECESSARY;
L
Linus Torvalds 已提交
340 341
	}

342
	pskb_trim(skb, skb->len - alen - padlen - 2);
343
	__skb_pull(skb, hlen);
344 345 346 347
	if (x->props.mode == XFRM_MODE_TUNNEL)
		skb_reset_transport_header(skb);
	else
		skb_set_transport_header(skb, -ihl);
348

349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373
	err = nexthdr[1];

	/* RFC4303: Drop dummy packets without any error */
	if (err == IPPROTO_NONE)
		err = -EINVAL;

out:
	return err;
}

static void esp_input_done(struct crypto_async_request *base, int err)
{
	struct sk_buff *skb = base->data;

	xfrm_input_resume(skb, esp_input_done2(skb, err));
}

/*
 * Note: detecting truncated vs. non-truncated authentication data is very
 * expensive, so we only support truncated data, which is the recommended
 * and common case.
 */
static int esp_input(struct xfrm_state *x, struct sk_buff *skb)
{
	struct ip_esp_hdr *esph;
374
	struct crypto_aead *aead = x->data;
375 376 377 378
	struct aead_request *req;
	struct sk_buff *trailer;
	int elen = skb->len - sizeof(*esph) - crypto_aead_ivsize(aead);
	int nfrags;
379 380 381 382
	int assoclen;
	int sglists;
	int seqhilen;
	__be32 *seqhi;
383 384 385 386 387 388
	void *tmp;
	u8 *iv;
	struct scatterlist *sg;
	struct scatterlist *asg;
	int err = -EINVAL;

389
	if (!pskb_may_pull(skb, sizeof(*esph) + crypto_aead_ivsize(aead)))
390 391 392 393 394 395 396 397 398
		goto out;

	if (elen <= 0)
		goto out;

	if ((err = skb_cow_data(skb, 0, &trailer)) < 0)
		goto out;
	nfrags = err;

399 400 401 402 403 404 405 406 407 408
	assoclen = sizeof(*esph);
	sglists = 1;
	seqhilen = 0;

	if (x->props.flags & XFRM_STATE_ESN) {
		sglists += 2;
		seqhilen += sizeof(__be32);
		assoclen += seqhilen;
	}

409
	err = -ENOMEM;
410
	tmp = esp_alloc_tmp(aead, nfrags + sglists, seqhilen);
411 412 413 414
	if (!tmp)
		goto out;

	ESP_SKB_CB(skb)->tmp = tmp;
415 416
	seqhi = esp_tmp_seqhi(tmp);
	iv = esp_tmp_iv(aead, tmp, seqhilen);
417 418
	req = esp_tmp_req(aead, iv);
	asg = esp_req_sg(aead, req);
419
	sg = asg + sglists;
420 421 422 423 424 425 426 427 428 429

	skb->ip_summed = CHECKSUM_NONE;

	esph = (struct ip_esp_hdr *)skb->data;

	/* Get ivec. This can be wrong, check against another impls. */
	iv = esph->enc_data;

	sg_init_table(sg, nfrags);
	skb_to_sgvec(skb, sg, sizeof(*esph) + crypto_aead_ivsize(aead), elen);
430 431 432 433 434 435 436 437 438

	if ((x->props.flags & XFRM_STATE_ESN)) {
		sg_init_table(asg, 3);
		sg_set_buf(asg, &esph->spi, sizeof(__be32));
		*seqhi = XFRM_SKB_CB(skb)->seq.input.hi;
		sg_set_buf(asg + 1, seqhi, seqhilen);
		sg_set_buf(asg + 2, &esph->seq_no, sizeof(__be32));
	} else
		sg_init_one(asg, esph, sizeof(*esph));
439 440 441

	aead_request_set_callback(req, 0, esp_input_done, skb);
	aead_request_set_crypt(req, sg, sg, elen, iv);
442
	aead_request_set_assoc(req, asg, assoclen);
443 444 445 446 447 448

	err = crypto_aead_decrypt(req);
	if (err == -EINPROGRESS)
		goto out;

	err = esp_input_done2(skb, err);
L
Linus Torvalds 已提交
449 450

out:
451
	return err;
L
Linus Torvalds 已提交
452 453
}

454
static u32 esp4_get_mtu(struct xfrm_state *x, int mtu)
L
Linus Torvalds 已提交
455
{
456 457
	struct crypto_aead *aead = x->data;
	u32 blksize = ALIGN(crypto_aead_blocksize(aead), 4);
458
	unsigned int net_adj;
D
Diego Beltrami 已提交
459 460 461 462

	switch (x->props.mode) {
	case XFRM_MODE_TRANSPORT:
	case XFRM_MODE_BEET:
463
		net_adj = sizeof(struct iphdr);
D
Diego Beltrami 已提交
464
		break;
465 466 467 468 469
	case XFRM_MODE_TUNNEL:
		net_adj = 0;
		break;
	default:
		BUG();
L
Linus Torvalds 已提交
470
	}
D
Diego Beltrami 已提交
471

472
	return ((mtu - x->props.header_len - crypto_aead_authsize(aead) -
473
		 net_adj) & ~(blksize - 1)) + net_adj - 2;
L
Linus Torvalds 已提交
474 475
}

476
static int esp4_err(struct sk_buff *skb, u32 info)
L
Linus Torvalds 已提交
477
{
A
Alexey Dobriyan 已提交
478
	struct net *net = dev_net(skb->dev);
479
	const struct iphdr *iph = (const struct iphdr *)skb->data;
480
	struct ip_esp_hdr *esph = (struct ip_esp_hdr *)(skb->data+(iph->ihl<<2));
L
Linus Torvalds 已提交
481 482
	struct xfrm_state *x;

483 484 485
	switch (icmp_hdr(skb)->type) {
	case ICMP_DEST_UNREACH:
		if (icmp_hdr(skb)->code != ICMP_FRAG_NEEDED)
486
			return 0;
487 488 489
	case ICMP_REDIRECT:
		break;
	default:
490
		return 0;
491
	}
L
Linus Torvalds 已提交
492

493 494
	x = xfrm_state_lookup(net, skb->mark, (const xfrm_address_t *)&iph->daddr,
			      esph->spi, IPPROTO_ESP, AF_INET);
L
Linus Torvalds 已提交
495
	if (!x)
496
		return 0;
497

498
	if (icmp_hdr(skb)->type == ICMP_DEST_UNREACH)
499
		ipv4_update_pmtu(skb, net, info, 0, 0, IPPROTO_ESP, 0);
500
	else
501
		ipv4_redirect(skb, net, 0, 0, IPPROTO_ESP, 0);
L
Linus Torvalds 已提交
502
	xfrm_state_put(x);
503 504

	return 0;
L
Linus Torvalds 已提交
505 506 507 508
}

static void esp_destroy(struct xfrm_state *x)
{
509
	struct crypto_aead *aead = x->data;
L
Linus Torvalds 已提交
510

511
	if (!aead)
L
Linus Torvalds 已提交
512 513
		return;

514
	crypto_free_aead(aead);
L
Linus Torvalds 已提交
515 516
}

517
static int esp_init_aead(struct xfrm_state *x)
L
Linus Torvalds 已提交
518
{
519 520 521 522 523 524 525 526
	struct crypto_aead *aead;
	int err;

	aead = crypto_alloc_aead(x->aead->alg_name, 0, 0);
	err = PTR_ERR(aead);
	if (IS_ERR(aead))
		goto error;

527
	x->data = aead;
528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543

	err = crypto_aead_setkey(aead, x->aead->alg_key,
				 (x->aead->alg_key_len + 7) / 8);
	if (err)
		goto error;

	err = crypto_aead_setauthsize(aead, x->aead->alg_icv_len / 8);
	if (err)
		goto error;

error:
	return err;
}

static int esp_init_authenc(struct xfrm_state *x)
{
544 545 546 547 548 549 550 551
	struct crypto_aead *aead;
	struct crypto_authenc_key_param *param;
	struct rtattr *rta;
	char *key;
	char *p;
	char authenc_name[CRYPTO_MAX_ALG_NAME];
	unsigned int keylen;
	int err;
L
Linus Torvalds 已提交
552

553
	err = -EINVAL;
L
Linus Torvalds 已提交
554
	if (x->ealg == NULL)
555
		goto error;
556

557
	err = -ENAMETOOLONG;
558 559 560 561 562 563 564 565 566 567 568 569 570 571

	if ((x->props.flags & XFRM_STATE_ESN)) {
		if (snprintf(authenc_name, CRYPTO_MAX_ALG_NAME,
			     "authencesn(%s,%s)",
			     x->aalg ? x->aalg->alg_name : "digest_null",
			     x->ealg->alg_name) >= CRYPTO_MAX_ALG_NAME)
			goto error;
	} else {
		if (snprintf(authenc_name, CRYPTO_MAX_ALG_NAME,
			     "authenc(%s,%s)",
			     x->aalg ? x->aalg->alg_name : "digest_null",
			     x->ealg->alg_name) >= CRYPTO_MAX_ALG_NAME)
			goto error;
	}
572 573 574 575 576 577

	aead = crypto_alloc_aead(authenc_name, 0, 0);
	err = PTR_ERR(aead);
	if (IS_ERR(aead))
		goto error;

578
	x->data = aead;
579 580 581 582 583 584 585 586 587 588 589 590 591 592 593

	keylen = (x->aalg ? (x->aalg->alg_key_len + 7) / 8 : 0) +
		 (x->ealg->alg_key_len + 7) / 8 + RTA_SPACE(sizeof(*param));
	err = -ENOMEM;
	key = kmalloc(keylen, GFP_KERNEL);
	if (!key)
		goto error;

	p = key;
	rta = (void *)p;
	rta->rta_type = CRYPTO_AUTHENC_KEYA_PARAM;
	rta->rta_len = RTA_LENGTH(sizeof(*param));
	param = RTA_DATA(rta);
	p += RTA_SPACE(sizeof(*param));

L
Linus Torvalds 已提交
594 595 596
	if (x->aalg) {
		struct xfrm_algo_desc *aalg_desc;

597 598
		memcpy(p, x->aalg->alg_key, (x->aalg->alg_key_len + 7) / 8);
		p += (x->aalg->alg_key_len + 7) / 8;
L
Linus Torvalds 已提交
599 600 601 602

		aalg_desc = xfrm_aalg_get_byname(x->aalg->alg_name, 0);
		BUG_ON(!aalg_desc);

603
		err = -EINVAL;
L
Linus Torvalds 已提交
604
		if (aalg_desc->uinfo.auth.icv_fullbits/8 !=
605
		    crypto_aead_authsize(aead)) {
606 607
			NETDEBUG(KERN_INFO "ESP: %s digestsize %u != %hu\n",
				 x->aalg->alg_name,
608
				 crypto_aead_authsize(aead),
609
				 aalg_desc->uinfo.auth.icv_fullbits/8);
610
			goto free_key;
L
Linus Torvalds 已提交
611 612
		}

613
		err = crypto_aead_setauthsize(
614
			aead, x->aalg->alg_trunc_len / 8);
615 616
		if (err)
			goto free_key;
L
Linus Torvalds 已提交
617
	}
618

619 620 621 622 623 624 625 626
	param->enckeylen = cpu_to_be32((x->ealg->alg_key_len + 7) / 8);
	memcpy(p, x->ealg->alg_key, (x->ealg->alg_key_len + 7) / 8);

	err = crypto_aead_setkey(aead, key, keylen);

free_key:
	kfree(key);

627 628 629 630 631 632 633 634 635 636
error:
	return err;
}

static int esp_init_state(struct xfrm_state *x)
{
	struct crypto_aead *aead;
	u32 align;
	int err;

637
	x->data = NULL;
638 639 640 641 642 643

	if (x->aead)
		err = esp_init_aead(x);
	else
		err = esp_init_authenc(x);

644
	if (err)
L
Linus Torvalds 已提交
645
		goto error;
646

647
	aead = x->data;
648

649 650
	x->props.header_len = sizeof(struct ip_esp_hdr) +
			      crypto_aead_ivsize(aead);
651
	if (x->props.mode == XFRM_MODE_TUNNEL)
L
Linus Torvalds 已提交
652
		x->props.header_len += sizeof(struct iphdr);
J
Joakim Koskela 已提交
653
	else if (x->props.mode == XFRM_MODE_BEET && x->sel.family != AF_INET6)
654
		x->props.header_len += IPV4_BEET_PHMAXLEN;
L
Linus Torvalds 已提交
655 656 657 658 659 660 661 662 663 664 665 666 667 668
	if (x->encap) {
		struct xfrm_encap_tmpl *encap = x->encap;

		switch (encap->encap_type) {
		default:
			goto error;
		case UDP_ENCAP_ESPINUDP:
			x->props.header_len += sizeof(struct udphdr);
			break;
		case UDP_ENCAP_ESPINUDP_NON_IKE:
			x->props.header_len += sizeof(struct udphdr) + 2 * sizeof(u32);
			break;
		}
	}
669 670

	align = ALIGN(crypto_aead_blocksize(aead), 4);
671
	x->props.trailer_len = align + 1 + crypto_aead_authsize(aead);
L
Linus Torvalds 已提交
672 673

error:
674
	return err;
L
Linus Torvalds 已提交
675 676
}

677 678 679 680 681
static int esp4_rcv_cb(struct sk_buff *skb, int err)
{
	return 0;
}

682
static const struct xfrm_type esp_type =
L
Linus Torvalds 已提交
683 684 685 686
{
	.description	= "ESP4",
	.owner		= THIS_MODULE,
	.proto	     	= IPPROTO_ESP,
687
	.flags		= XFRM_TYPE_REPLAY_PROT,
L
Linus Torvalds 已提交
688 689
	.init_state	= esp_init_state,
	.destructor	= esp_destroy,
690
	.get_mtu	= esp4_get_mtu,
L
Linus Torvalds 已提交
691 692 693 694
	.input		= esp_input,
	.output		= esp_output
};

695
static struct xfrm4_protocol esp4_protocol = {
L
Linus Torvalds 已提交
696
	.handler	=	xfrm4_rcv,
697 698
	.input_handler	=	xfrm_input,
	.cb_handler	=	esp4_rcv_cb,
L
Linus Torvalds 已提交
699
	.err_handler	=	esp4_err,
700
	.priority	=	0,
L
Linus Torvalds 已提交
701 702 703 704 705
};

static int __init esp4_init(void)
{
	if (xfrm_register_type(&esp_type, AF_INET) < 0) {
J
Joe Perches 已提交
706
		pr_info("%s: can't add xfrm type\n", __func__);
L
Linus Torvalds 已提交
707 708
		return -EAGAIN;
	}
709
	if (xfrm4_protocol_register(&esp4_protocol, IPPROTO_ESP) < 0) {
J
Joe Perches 已提交
710
		pr_info("%s: can't add protocol\n", __func__);
L
Linus Torvalds 已提交
711 712 713 714 715 716 717 718
		xfrm_unregister_type(&esp_type, AF_INET);
		return -EAGAIN;
	}
	return 0;
}

static void __exit esp4_fini(void)
{
719
	if (xfrm4_protocol_deregister(&esp4_protocol, IPPROTO_ESP) < 0)
J
Joe Perches 已提交
720
		pr_info("%s: can't remove protocol\n", __func__);
L
Linus Torvalds 已提交
721
	if (xfrm_unregister_type(&esp_type, AF_INET) < 0)
J
Joe Perches 已提交
722
		pr_info("%s: can't remove xfrm type\n", __func__);
L
Linus Torvalds 已提交
723 724 725 726 727
}

module_init(esp4_init);
module_exit(esp4_fini);
MODULE_LICENSE("GPL");
728
MODULE_ALIAS_XFRM_TYPE(AF_INET, XFRM_PROTO_ESP);