esp6.c 16.2 KB
Newer Older
L
Linus Torvalds 已提交
1 2
/*
 * Copyright (C)2002 USAGI/WIDE Project
3
 *
L
Linus Torvalds 已提交
4 5 6 7
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
8
 *
L
Linus Torvalds 已提交
9 10 11 12
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
13
 *
L
Linus Torvalds 已提交
14
 * You should have received a copy of the GNU General Public License
15
 * along with this program; if not, see <http://www.gnu.org/licenses/>.
L
Linus Torvalds 已提交
16 17 18
 *
 * Authors
 *
19
 *	Mitsuru KANDA @USAGI       : IPv6 Support
20 21
 *	Kazunori MIYAZAWA @USAGI   :
 *	Kunihiro Ishiguro <kunihiro@ipinfusion.com>
22
 *
23
 *	This file is derived from net/ipv4/esp.c
L
Linus Torvalds 已提交
24 25
 */

26 27
#define pr_fmt(fmt) "IPv6: " fmt

28 29
#include <crypto/aead.h>
#include <crypto/authenc.h>
30
#include <linux/err.h>
L
Linus Torvalds 已提交
31 32 33 34
#include <linux/module.h>
#include <net/ip.h>
#include <net/xfrm.h>
#include <net/esp.h>
35
#include <linux/scatterlist.h>
H
Herbert Xu 已提交
36
#include <linux/kernel.h>
L
Linus Torvalds 已提交
37 38
#include <linux/pfkeyv2.h>
#include <linux/random.h>
39
#include <linux/slab.h>
40
#include <linux/spinlock.h>
41
#include <net/ip6_route.h>
L
Linus Torvalds 已提交
42 43
#include <net/icmp.h>
#include <net/ipv6.h>
44
#include <net/protocol.h>
L
Linus Torvalds 已提交
45 46
#include <linux/icmpv6.h>

47 48 49 50 51 52 53
struct esp_skb_cb {
	struct xfrm_skb_cb xfrm;
	void *tmp;
};

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

54 55
static u32 esp6_get_mtu(struct xfrm_state *x, int mtu);

56 57 58
/*
 * Allocate an AEAD request structure with extra space for SG and IV.
 *
59 60 61
 * For alignment considerations the upper 32 bits of the sequence number are
 * placed at the front, if present. Followed by the IV, the request and finally
 * the SG list.
62 63 64
 *
 * TODO: Use spare space in skb for this where possible.
 */
65
static void *esp_alloc_tmp(struct crypto_aead *aead, int nfrags, int seqihlen)
66 67 68
{
	unsigned int len;

69 70 71 72
	len = seqihlen;

	len += crypto_aead_ivsize(aead);

73 74 75 76 77 78
	if (len) {
		len += crypto_aead_alignmask(aead) &
		       ~(crypto_tfm_ctx_alignment() - 1);
		len = ALIGN(len, crypto_tfm_ctx_alignment());
	}

H
Herbert Xu 已提交
79
	len += sizeof(struct aead_request) + crypto_aead_reqsize(aead);
80 81 82 83 84 85 86
	len = ALIGN(len, __alignof__(struct scatterlist));

	len += sizeof(struct scatterlist) * nfrags;

	return kmalloc(len, GFP_ATOMIC);
}

87 88 89 90 91 92
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)
93 94
{
	return crypto_aead_ivsize(aead) ?
95 96
	       PTR_ALIGN((u8 *)tmp + seqhilen,
			 crypto_aead_alignmask(aead) + 1) : tmp + seqhilen;
97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124
}

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 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);
}

H
Herbert Xu 已提交
125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148
/* Move ESP header back into place. */
static void esp_restore_header(struct sk_buff *skb, unsigned int offset)
{
	struct ip_esp_hdr *esph = (void *)(skb->data + offset);
	void *tmp = ESP_SKB_CB(skb)->tmp;
	__be32 *seqhi = esp_tmp_seqhi(tmp);

	esph->seq_no = esph->spi;
	esph->spi = *seqhi;
}

static void esp_output_restore_header(struct sk_buff *skb)
{
	esp_restore_header(skb, skb_transport_offset(skb) - sizeof(__be32));
}

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

	esp_output_restore_header(skb);
	esp_output_done(base, err);
}

L
Linus Torvalds 已提交
149 150 151
static int esp6_output(struct xfrm_state *x, struct sk_buff *skb)
{
	int err;
152
	struct ip_esp_hdr *esph;
153
	struct crypto_aead *aead;
H
Herbert Xu 已提交
154
	struct aead_request *req;
155
	struct scatterlist *sg;
L
Linus Torvalds 已提交
156
	struct sk_buff *trailer;
157
	void *tmp;
L
Linus Torvalds 已提交
158 159 160
	int blksize;
	int clen;
	int alen;
161
	int plen;
H
Herbert Xu 已提交
162
	int ivlen;
163
	int tfclen;
L
Linus Torvalds 已提交
164
	int nfrags;
165 166
	int assoclen;
	int seqhilen;
167
	u8 *iv;
168
	u8 *tail;
169
	__be32 *seqhi;
H
Herbert Xu 已提交
170
	__be64 seqno;
L
Linus Torvalds 已提交
171

172
	/* skb is pure payload to encrypt */
173
	aead = x->data;
174
	alen = crypto_aead_authsize(aead);
H
Herbert Xu 已提交
175
	ivlen = crypto_aead_ivsize(aead);
176

177 178 179 180 181 182 183 184 185
	tfclen = 0;
	if (x->tfcpad) {
		struct xfrm_dst *dst = (struct xfrm_dst *)skb_dst(skb);
		u32 padto;

		padto = min(x->tfcpad, esp6_get_mtu(x, dst->child_mtu_cached));
		if (skb->len < padto)
			tfclen = padto - skb->len;
	}
186
	blksize = ALIGN(crypto_aead_blocksize(aead), 4);
187 188
	clen = ALIGN(skb->len + 2 + tfclen, blksize);
	plen = clen - skb->len - tfclen;
L
Linus Torvalds 已提交
189

190 191
	err = skb_cow_data(skb, tfclen + plen + alen, &trailer);
	if (err < 0)
L
Linus Torvalds 已提交
192
		goto error;
193 194
	nfrags = err;

195 196 197 198 199 200 201 202
	assoclen = sizeof(*esph);
	seqhilen = 0;

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

H
Herbert Xu 已提交
203
	tmp = esp_alloc_tmp(aead, nfrags, seqhilen);
J
Julia Lawall 已提交
204 205
	if (!tmp) {
		err = -ENOMEM;
206
		goto error;
J
Julia Lawall 已提交
207
	}
208

209 210
	seqhi = esp_tmp_seqhi(tmp);
	iv = esp_tmp_iv(aead, tmp, seqhilen);
H
Herbert Xu 已提交
211 212
	req = esp_tmp_req(aead, iv);
	sg = esp_req_sg(aead, req);
L
Linus Torvalds 已提交
213 214

	/* Fill padding... */
215
	tail = skb_tail_pointer(trailer);
216 217 218 219
	if (tfclen) {
		memset(tail, 0, tfclen);
		tail += tfclen;
	}
L
Linus Torvalds 已提交
220 221
	do {
		int i;
222
		for (i = 0; i < plen - 2; i++)
223
			tail[i] = i + 1;
L
Linus Torvalds 已提交
224
	} while (0);
225 226
	tail[plen - 2] = plen - 2;
	tail[plen - 1] = *skb_mac_header(skb);
227
	pskb_put(skb, trailer, clen - skb->len + alen);
L
Linus Torvalds 已提交
228

229
	skb_push(skb, -skb_network_offset(skb));
230
	esph = ip_esp_hdr(skb);
231
	*skb_mac_header(skb) = IPPROTO_ESP;
L
Linus Torvalds 已提交
232

233
	esph->seq_no = htonl(XFRM_SKB_CB(skb)->seq.output.low);
L
Linus Torvalds 已提交
234

H
Herbert Xu 已提交
235 236 237 238 239 240 241 242 243 244 245 246 247 248 249
	aead_request_set_callback(req, 0, esp_output_done, skb);

	/* For ESN we move the header forward by 4 bytes to
	 * accomodate the high bits.  We will move it back after
	 * encryption.
	 */
	if ((x->props.flags & XFRM_STATE_ESN)) {
		esph = (void *)(skb_transport_header(skb) - sizeof(__be32));
		*seqhi = esph->spi;
		esph->seq_no = htonl(XFRM_SKB_CB(skb)->seq.output.hi);
		aead_request_set_callback(req, 0, esp_output_done_esn, skb);
	}

	esph->spi = x->id.spi;

250 251
	sg_init_table(sg, nfrags);
	skb_to_sgvec(skb, sg,
H
Herbert Xu 已提交
252 253
		     (unsigned char *)esph - skb->data,
		     assoclen + ivlen + clen + alen);
254

H
Herbert Xu 已提交
255 256 257 258 259 260 261 262 263
	aead_request_set_crypt(req, sg, sg, ivlen + clen, iv);
	aead_request_set_ad(req, assoclen);

	seqno = cpu_to_be64(XFRM_SKB_CB(skb)->seq.output.low +
			    ((u64)XFRM_SKB_CB(skb)->seq.output.hi << 32));

	memset(iv, 0, ivlen);
	memcpy(iv + ivlen - min(ivlen, 8), (u8 *)&seqno + 8 - min(ivlen, 8),
	       min(ivlen, 8));
L
Linus Torvalds 已提交
264

265
	ESP_SKB_CB(skb)->tmp = tmp;
H
Herbert Xu 已提交
266 267 268 269
	err = crypto_aead_encrypt(req);

	switch (err) {
	case -EINPROGRESS:
270
		goto error;
L
Linus Torvalds 已提交
271

H
Herbert Xu 已提交
272
	case -EBUSY:
273
		err = NET_XMIT_DROP;
H
Herbert Xu 已提交
274 275 276 277 278 279
		break;

	case 0:
		if ((x->props.flags & XFRM_STATE_ESN))
			esp_output_restore_header(skb);
	}
280 281 282 283 284 285 286 287 288 289

	kfree(tmp);

error:
	return err;
}

static int esp_input_done2(struct sk_buff *skb, int err)
{
	struct xfrm_state *x = xfrm_input_state(skb);
290
	struct crypto_aead *aead = x->data;
291 292 293 294 295 296 297 298
	int alen = crypto_aead_authsize(aead);
	int hlen = sizeof(struct ip_esp_hdr) + crypto_aead_ivsize(aead);
	int elen = skb->len - hlen;
	int hdr_len = skb_network_header_len(skb);
	int padlen;
	u8 nexthdr[2];

	kfree(ESP_SKB_CB(skb)->tmp);
L
Linus Torvalds 已提交
299

300
	if (unlikely(err))
301
		goto out;
302

303 304
	if (skb_copy_bits(skb, skb->len - alen - 2, nexthdr, 2))
		BUG();
L
Linus Torvalds 已提交
305

306 307 308
	err = -EINVAL;
	padlen = nexthdr[0];
	if (padlen + 2 + alen >= elen) {
309 310
		net_dbg_ratelimited("ipsec esp packet is garbage padlen=%d, elen=%d\n",
				    padlen + 2, elen - alen);
311
		goto out;
L
Linus Torvalds 已提交
312 313
	}

314
	/* ... check padding bits here. Silly. :-) */
315

316 317
	pskb_trim(skb, skb->len - alen - padlen - 2);
	__skb_pull(skb, hlen);
318 319 320 321
	if (x->props.mode == XFRM_MODE_TUNNEL)
		skb_reset_transport_header(skb);
	else
		skb_set_transport_header(skb, -hdr_len);
322 323 324 325 326 327 328 329

	err = nexthdr[1];

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

out:
L
Linus Torvalds 已提交
330 331 332
	return err;
}

333 334 335 336 337 338 339
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));
}

H
Herbert Xu 已提交
340 341 342 343 344 345 346 347 348 349 350 351 352 353
static void esp_input_restore_header(struct sk_buff *skb)
{
	esp_restore_header(skb, 0);
	__skb_pull(skb, 4);
}

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

	esp_input_restore_header(skb);
	esp_input_done(base, err);
}

354
static int esp6_input(struct xfrm_state *x, struct sk_buff *skb)
L
Linus Torvalds 已提交
355
{
356
	struct ip_esp_hdr *esph;
357
	struct crypto_aead *aead = x->data;
358
	struct aead_request *req;
L
Linus Torvalds 已提交
359
	struct sk_buff *trailer;
H
Herbert Xu 已提交
360 361
	int ivlen = crypto_aead_ivsize(aead);
	int elen = skb->len - sizeof(*esph) - ivlen;
L
Linus Torvalds 已提交
362
	int nfrags;
363 364
	int assoclen;
	int seqhilen;
L
Linus Torvalds 已提交
365
	int ret = 0;
366
	void *tmp;
367
	__be32 *seqhi;
368 369
	u8 *iv;
	struct scatterlist *sg;
L
Linus Torvalds 已提交
370

H
Herbert Xu 已提交
371
	if (!pskb_may_pull(skb, sizeof(*esph) + ivlen)) {
L
Linus Torvalds 已提交
372
		ret = -EINVAL;
373
		goto out;
L
Linus Torvalds 已提交
374 375
	}

376
	if (elen <= 0) {
L
Linus Torvalds 已提交
377
		ret = -EINVAL;
378
		goto out;
L
Linus Torvalds 已提交
379 380
	}

381 382
	nfrags = skb_cow_data(skb, 0, &trailer);
	if (nfrags < 0) {
383 384 385 386
		ret = -EINVAL;
		goto out;
	}

387
	ret = -ENOMEM;
388 389 390 391 392 393 394 395 396

	assoclen = sizeof(*esph);
	seqhilen = 0;

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

H
Herbert Xu 已提交
397
	tmp = esp_alloc_tmp(aead, nfrags, seqhilen);
398 399
	if (!tmp)
		goto out;
L
Linus Torvalds 已提交
400

401
	ESP_SKB_CB(skb)->tmp = tmp;
402 403
	seqhi = esp_tmp_seqhi(tmp);
	iv = esp_tmp_iv(aead, tmp, seqhilen);
404
	req = esp_tmp_req(aead, iv);
H
Herbert Xu 已提交
405
	sg = esp_req_sg(aead, req);
L
Linus Torvalds 已提交
406

407
	skb->ip_summed = CHECKSUM_NONE;
L
Linus Torvalds 已提交
408

409
	esph = (struct ip_esp_hdr *)skb->data;
L
Linus Torvalds 已提交
410

H
Herbert Xu 已提交
411
	aead_request_set_callback(req, 0, esp_input_done, skb);
412

H
Herbert Xu 已提交
413 414 415 416
	/* For ESN we move the header forward by 4 bytes to
	 * accomodate the high bits.  We will move it back after
	 * decryption.
	 */
417
	if ((x->props.flags & XFRM_STATE_ESN)) {
H
Herbert Xu 已提交
418 419 420
		esph = (void *)skb_push(skb, 4);
		*seqhi = esph->spi;
		esph->spi = esph->seq_no;
421
		esph->seq_no = XFRM_SKB_CB(skb)->seq.input.hi;
H
Herbert Xu 已提交
422 423
		aead_request_set_callback(req, 0, esp_input_done_esn, skb);
	}
424

H
Herbert Xu 已提交
425 426 427 428 429
	sg_init_table(sg, nfrags);
	skb_to_sgvec(skb, sg, 0, skb->len);

	aead_request_set_crypt(req, sg, sg, elen + ivlen, iv);
	aead_request_set_ad(req, assoclen);
L
Linus Torvalds 已提交
430

431 432 433
	ret = crypto_aead_decrypt(req);
	if (ret == -EINPROGRESS)
		goto out;
434

H
Herbert Xu 已提交
435 436 437
	if ((x->props.flags & XFRM_STATE_ESN))
		esp_input_restore_header(skb);

438
	ret = esp_input_done2(skb, ret);
L
Linus Torvalds 已提交
439 440 441 442 443

out:
	return ret;
}

444
static u32 esp6_get_mtu(struct xfrm_state *x, int mtu)
L
Linus Torvalds 已提交
445
{
446 447
	struct crypto_aead *aead = x->data;
	u32 blksize = ALIGN(crypto_aead_blocksize(aead), 4);
448
	unsigned int net_adj;
L
Linus Torvalds 已提交
449

450 451 452 453
	if (x->props.mode != XFRM_MODE_TUNNEL)
		net_adj = sizeof(struct ipv6hdr);
	else
		net_adj = 0;
454

455
	return ((mtu - x->props.header_len - crypto_aead_authsize(aead) -
456
		 net_adj) & ~(blksize - 1)) + net_adj - 2;
L
Linus Torvalds 已提交
457 458
}

459 460
static int esp6_err(struct sk_buff *skb, struct inet6_skb_parm *opt,
		    u8 type, u8 code, int offset, __be32 info)
L
Linus Torvalds 已提交
461
{
A
Alexey Dobriyan 已提交
462
	struct net *net = dev_net(skb->dev);
463
	const struct ipv6hdr *iph = (const struct ipv6hdr *)skb->data;
464
	struct ip_esp_hdr *esph = (struct ip_esp_hdr *)(skb->data + offset);
L
Linus Torvalds 已提交
465 466
	struct xfrm_state *x;

467
	if (type != ICMPV6_PKT_TOOBIG &&
468
	    type != NDISC_REDIRECT)
469
		return 0;
L
Linus Torvalds 已提交
470

471 472
	x = xfrm_state_lookup(net, skb->mark, (const xfrm_address_t *)&iph->daddr,
			      esph->spi, IPPROTO_ESP, AF_INET6);
L
Linus Torvalds 已提交
473
	if (!x)
474
		return 0;
475 476

	if (type == NDISC_REDIRECT)
477 478
		ip6_redirect(skb, net, skb->dev->ifindex, 0,
			     sock_net_uid(net, NULL));
479
	else
480
		ip6_update_pmtu(skb, net, info, 0, 0, sock_net_uid(net, NULL));
L
Linus Torvalds 已提交
481
	xfrm_state_put(x);
482 483

	return 0;
L
Linus Torvalds 已提交
484 485 486 487
}

static void esp6_destroy(struct xfrm_state *x)
{
488
	struct crypto_aead *aead = x->data;
L
Linus Torvalds 已提交
489

490
	if (!aead)
L
Linus Torvalds 已提交
491 492
		return;

493
	crypto_free_aead(aead);
L
Linus Torvalds 已提交
494 495
}

496 497
static int esp_init_aead(struct xfrm_state *x)
{
H
Herbert Xu 已提交
498
	char aead_name[CRYPTO_MAX_ALG_NAME];
499 500 501
	struct crypto_aead *aead;
	int err;

H
Herbert Xu 已提交
502 503 504 505 506 507
	err = -ENAMETOOLONG;
	if (snprintf(aead_name, CRYPTO_MAX_ALG_NAME, "%s(%s)",
		     x->geniv, x->aead->alg_name) >= CRYPTO_MAX_ALG_NAME)
		goto error;

	aead = crypto_alloc_aead(aead_name, 0, 0);
508 509 510 511
	err = PTR_ERR(aead);
	if (IS_ERR(aead))
		goto error;

512
	x->data = aead;
513 514 515 516 517 518 519 520 521 522 523 524 525 526 527

	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)
L
Linus Torvalds 已提交
528
{
529 530 531 532 533 534 535 536
	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 已提交
537

538
	err = -EINVAL;
539
	if (!x->ealg)
540
		goto error;
541

542
	err = -ENAMETOOLONG;
543 544 545

	if ((x->props.flags & XFRM_STATE_ESN)) {
		if (snprintf(authenc_name, CRYPTO_MAX_ALG_NAME,
H
Herbert Xu 已提交
546 547
			     "%s%sauthencesn(%s,%s)%s",
			     x->geniv ?: "", x->geniv ? "(" : "",
548
			     x->aalg ? x->aalg->alg_name : "digest_null",
H
Herbert Xu 已提交
549 550
			     x->ealg->alg_name,
			     x->geniv ? ")" : "") >= CRYPTO_MAX_ALG_NAME)
551 552 553
			goto error;
	} else {
		if (snprintf(authenc_name, CRYPTO_MAX_ALG_NAME,
H
Herbert Xu 已提交
554 555
			     "%s%sauthenc(%s,%s)%s",
			     x->geniv ?: "", x->geniv ? "(" : "",
556
			     x->aalg ? x->aalg->alg_name : "digest_null",
H
Herbert Xu 已提交
557 558
			     x->ealg->alg_name,
			     x->geniv ? ")" : "") >= CRYPTO_MAX_ALG_NAME)
559 560
			goto error;
	}
561 562 563 564 565 566

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

567
	x->data = aead;
568 569 570 571 572 573 574 575 576 577 578 579 580 581 582

	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 已提交
583 584 585
	if (x->aalg) {
		struct xfrm_algo_desc *aalg_desc;

586 587
		memcpy(p, x->aalg->alg_key, (x->aalg->alg_key_len + 7) / 8);
		p += (x->aalg->alg_key_len + 7) / 8;
588

L
Linus Torvalds 已提交
589 590
		aalg_desc = xfrm_aalg_get_byname(x->aalg->alg_name, 0);
		BUG_ON(!aalg_desc);
591

592
		err = -EINVAL;
593
		if (aalg_desc->uinfo.auth.icv_fullbits / 8 !=
594
		    crypto_aead_authsize(aead)) {
595 596 597 598
			pr_info("ESP: %s digestsize %u != %hu\n",
				x->aalg->alg_name,
				crypto_aead_authsize(aead),
				aalg_desc->uinfo.auth.icv_fullbits / 8);
599
			goto free_key;
L
Linus Torvalds 已提交
600
		}
601

602
		err = crypto_aead_setauthsize(
603
			aead, x->aalg->alg_trunc_len / 8);
604 605
		if (err)
			goto free_key;
L
Linus Torvalds 已提交
606
	}
607 608 609 610 611 612 613 614 615

	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);

616 617 618 619 620 621 622 623 624 625 626 627 628
error:
	return err;
}

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

	if (x->encap)
		return -EINVAL;

629
	x->data = NULL;
630 631 632 633 634 635

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

636
	if (err)
L
Linus Torvalds 已提交
637
		goto error;
638

639
	aead = x->data;
640

641 642
	x->props.header_len = sizeof(struct ip_esp_hdr) +
			      crypto_aead_ivsize(aead);
643 644
	switch (x->props.mode) {
	case XFRM_MODE_BEET:
645 646
		if (x->sel.family != AF_INET6)
			x->props.header_len += IPV4_BEET_PHMAXLEN +
647
					       (sizeof(struct ipv6hdr) - sizeof(struct iphdr));
648
		break;
649 650 651
	case XFRM_MODE_TRANSPORT:
		break;
	case XFRM_MODE_TUNNEL:
L
Linus Torvalds 已提交
652
		x->props.header_len += sizeof(struct ipv6hdr);
653
		break;
654 655 656
	default:
		goto error;
	}
657 658

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

error:
662
	return err;
L
Linus Torvalds 已提交
663 664
}

665 666 667 668 669
static int esp6_rcv_cb(struct sk_buff *skb, int err)
{
	return 0;
}

670
static const struct xfrm_type esp6_type = {
L
Linus Torvalds 已提交
671
	.description	= "ESP6",
672 673
	.owner		= THIS_MODULE,
	.proto		= IPPROTO_ESP,
674
	.flags		= XFRM_TYPE_REPLAY_PROT,
L
Linus Torvalds 已提交
675 676
	.init_state	= esp6_init_state,
	.destructor	= esp6_destroy,
677
	.get_mtu	= esp6_get_mtu,
L
Linus Torvalds 已提交
678
	.input		= esp6_input,
679 680
	.output		= esp6_output,
	.hdr_offset	= xfrm6_find_1stfragopt,
L
Linus Torvalds 已提交
681 682
};

683 684 685
static struct xfrm6_protocol esp6_protocol = {
	.handler	=	xfrm6_rcv,
	.cb_handler	=	esp6_rcv_cb,
L
Linus Torvalds 已提交
686
	.err_handler	=	esp6_err,
687
	.priority	=	0,
L
Linus Torvalds 已提交
688 689 690 691 692
};

static int __init esp6_init(void)
{
	if (xfrm_register_type(&esp6_type, AF_INET6) < 0) {
693
		pr_info("%s: can't add xfrm type\n", __func__);
L
Linus Torvalds 已提交
694 695
		return -EAGAIN;
	}
696
	if (xfrm6_protocol_register(&esp6_protocol, IPPROTO_ESP) < 0) {
697
		pr_info("%s: can't add protocol\n", __func__);
L
Linus Torvalds 已提交
698 699 700 701 702 703 704 705 706
		xfrm_unregister_type(&esp6_type, AF_INET6);
		return -EAGAIN;
	}

	return 0;
}

static void __exit esp6_fini(void)
{
707
	if (xfrm6_protocol_deregister(&esp6_protocol, IPPROTO_ESP) < 0)
708
		pr_info("%s: can't remove protocol\n", __func__);
L
Linus Torvalds 已提交
709
	if (xfrm_unregister_type(&esp6_type, AF_INET6) < 0)
710
		pr_info("%s: can't remove xfrm type\n", __func__);
L
Linus Torvalds 已提交
711 712 713 714 715 716
}

module_init(esp6_init);
module_exit(esp6_fini);

MODULE_LICENSE("GPL");
717
MODULE_ALIAS_XFRM_TYPE(AF_INET6, XFRM_PROTO_ESP);