Blame view
net/ipv4/esp4.c
11.8 KB
6b7326c84 [IPSEC] ESP: Use ... |
1 |
#include <linux/err.h> |
1da177e4c Linux-2.6.12-rc2 |
2 3 4 5 6 7 |
#include <linux/module.h> #include <net/ip.h> #include <net/xfrm.h> #include <net/esp.h> #include <asm/scatterlist.h> #include <linux/crypto.h> |
a02a64223 [IPSEC]: Use ALIG... |
8 |
#include <linux/kernel.h> |
1da177e4c Linux-2.6.12-rc2 |
9 10 11 |
#include <linux/pfkeyv2.h> #include <linux/random.h> #include <net/icmp.h> |
14c850212 [INET_SOCK]: Move... |
12 |
#include <net/protocol.h> |
1da177e4c Linux-2.6.12-rc2 |
13 |
#include <net/udp.h> |
1da177e4c Linux-2.6.12-rc2 |
14 15 16 17 18 |
static int esp_output(struct xfrm_state *x, struct sk_buff *skb) { int err; struct iphdr *top_iph; struct ip_esp_hdr *esph; |
6b7326c84 [IPSEC] ESP: Use ... |
19 20 |
struct crypto_blkcipher *tfm; struct blkcipher_desc desc; |
1da177e4c Linux-2.6.12-rc2 |
21 22 |
struct esp_data *esp; struct sk_buff *trailer; |
27a884dc3 [SK_BUFF]: Conver... |
23 |
u8 *tail; |
1da177e4c Linux-2.6.12-rc2 |
24 25 26 27 28 29 |
int blksize; int clen; int alen; int nfrags; /* Strip IP+ESP header. */ |
ea2ae17d6 [SK_BUFF]: Introd... |
30 |
__skb_pull(skb, skb_transport_offset(skb)); |
1da177e4c Linux-2.6.12-rc2 |
31 32 33 34 35 36 37 38 39 40 |
/* Now skb is pure payload to encrypt */ err = -ENOMEM; /* Round to block size */ clen = skb->len; esp = x->data; alen = esp->auth.icv_trunc_len; tfm = esp->conf.tfm; |
6b7326c84 [IPSEC] ESP: Use ... |
41 42 43 |
desc.tfm = tfm; desc.flags = 0; blksize = ALIGN(crypto_blkcipher_blocksize(tfm), 4); |
a02a64223 [IPSEC]: Use ALIG... |
44 |
clen = ALIGN(clen + 2, blksize); |
1da177e4c Linux-2.6.12-rc2 |
45 |
if (esp->conf.padlen) |
a02a64223 [IPSEC]: Use ALIG... |
46 |
clen = ALIGN(clen, esp->conf.padlen); |
1da177e4c Linux-2.6.12-rc2 |
47 48 49 50 51 |
if ((nfrags = skb_cow_data(skb, clen-skb->len+alen, &trailer)) < 0) goto error; /* Fill padding... */ |
27a884dc3 [SK_BUFF]: Conver... |
52 |
tail = skb_tail_pointer(trailer); |
1da177e4c Linux-2.6.12-rc2 |
53 54 55 |
do { int i; for (i=0; i<clen-skb->len - 2; i++) |
27a884dc3 [SK_BUFF]: Conver... |
56 |
tail[i] = i + 1; |
1da177e4c Linux-2.6.12-rc2 |
57 |
} while (0); |
27a884dc3 [SK_BUFF]: Conver... |
58 |
tail[clen - skb->len - 2] = (clen - skb->len) - 2; |
1da177e4c Linux-2.6.12-rc2 |
59 |
pskb_put(skb, trailer, clen - skb->len); |
d56f90a7c [SK_BUFF]: Introd... |
60 |
__skb_push(skb, skb->data - skb_network_header(skb)); |
eddc9ec53 [SK_BUFF]: Introd... |
61 |
top_iph = ip_hdr(skb); |
d56f90a7c [SK_BUFF]: Introd... |
62 63 |
esph = (struct ip_esp_hdr *)(skb_network_header(skb) + top_iph->ihl * 4); |
1da177e4c Linux-2.6.12-rc2 |
64 |
top_iph->tot_len = htons(skb->len + alen); |
557922584 [XFRM]: esp: fix ... |
65 |
*(skb_tail_pointer(trailer) - 1) = top_iph->protocol; |
1da177e4c Linux-2.6.12-rc2 |
66 67 68 69 70 |
/* this is non-NULL only with UDP Encapsulation */ if (x->encap) { struct xfrm_encap_tmpl *encap = x->encap; struct udphdr *uh; |
d5a0a1e31 [IPV4]: encapsula... |
71 |
__be32 *udpdata32; |
1da177e4c Linux-2.6.12-rc2 |
72 73 74 75 76 77 78 79 80 81 82 83 84 |
uh = (struct udphdr *)esph; uh->source = encap->encap_sport; uh->dest = encap->encap_dport; uh->len = htons(skb->len + alen - top_iph->ihl*4); uh->check = 0; switch (encap->encap_type) { default: case UDP_ENCAP_ESPINUDP: esph = (struct ip_esp_hdr *)(uh + 1); break; case UDP_ENCAP_ESPINUDP_NON_IKE: |
d5a0a1e31 [IPV4]: encapsula... |
85 |
udpdata32 = (__be32 *)(uh + 1); |
1da177e4c Linux-2.6.12-rc2 |
86 87 88 89 90 91 92 93 94 95 96 |
udpdata32[0] = udpdata32[1] = 0; esph = (struct ip_esp_hdr *)(udpdata32 + 2); break; } top_iph->protocol = IPPROTO_UDP; } else top_iph->protocol = IPPROTO_ESP; esph->spi = x->id.spi; esph->seq_no = htonl(++x->replay.oseq); |
9500e8a81 [IPSEC]: Sync ser... |
97 |
xfrm_aevent_doreplay(x); |
1da177e4c Linux-2.6.12-rc2 |
98 |
|
e4bec827f [IPSEC] esp: Defe... |
99 100 101 102 103 |
if (esp->conf.ivlen) { if (unlikely(!esp->conf.ivinitted)) { get_random_bytes(esp->conf.ivec, esp->conf.ivlen); esp->conf.ivinitted = 1; } |
6b7326c84 [IPSEC] ESP: Use ... |
104 |
crypto_blkcipher_set_iv(tfm, esp->conf.ivec, esp->conf.ivlen); |
e4bec827f [IPSEC] esp: Defe... |
105 |
} |
1da177e4c Linux-2.6.12-rc2 |
106 107 108 109 110 111 112 113 114 115 |
do { struct scatterlist *sg = &esp->sgbuf[0]; if (unlikely(nfrags > ESP_NUM_FAST_SG)) { sg = kmalloc(sizeof(struct scatterlist)*nfrags, GFP_ATOMIC); if (!sg) goto error; } skb_to_sgvec(skb, sg, esph->enc_data+esp->conf.ivlen-skb->data, clen); |
6b7326c84 [IPSEC] ESP: Use ... |
116 |
err = crypto_blkcipher_encrypt(&desc, sg, sg, clen); |
1da177e4c Linux-2.6.12-rc2 |
117 118 119 |
if (unlikely(sg != &esp->sgbuf[0])) kfree(sg); } while (0); |
6b7326c84 [IPSEC] ESP: Use ... |
120 121 |
if (unlikely(err)) goto error; |
1da177e4c Linux-2.6.12-rc2 |
122 |
if (esp->conf.ivlen) { |
6b7326c84 [IPSEC] ESP: Use ... |
123 124 |
memcpy(esph->enc_data, esp->conf.ivec, esp->conf.ivlen); crypto_blkcipher_get_iv(tfm, esp->conf.ivec, esp->conf.ivlen); |
1da177e4c Linux-2.6.12-rc2 |
125 126 127 |
} if (esp->auth.icv_full_len) { |
07d4ee583 [IPSEC]: Use HMAC... |
128 129 130 |
err = esp_mac_digest(esp, skb, (u8 *)esph - skb->data, sizeof(*esph) + esp->conf.ivlen + clen); memcpy(pskb_put(skb, trailer, alen), esp->auth.work_icv, alen); |
1da177e4c Linux-2.6.12-rc2 |
131 132 133 |
} ip_send_check(top_iph); |
1da177e4c Linux-2.6.12-rc2 |
134 135 136 137 138 139 140 141 142 |
error: return 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. */ |
e695633e2 [IPSEC]: Kill unu... |
143 |
static int esp_input(struct xfrm_state *x, struct sk_buff *skb) |
1da177e4c Linux-2.6.12-rc2 |
144 145 146 147 |
{ struct iphdr *iph; struct ip_esp_hdr *esph; struct esp_data *esp = x->data; |
6b7326c84 [IPSEC] ESP: Use ... |
148 149 |
struct crypto_blkcipher *tfm = esp->conf.tfm; struct blkcipher_desc desc = { .tfm = tfm }; |
1da177e4c Linux-2.6.12-rc2 |
150 |
struct sk_buff *trailer; |
6b7326c84 [IPSEC] ESP: Use ... |
151 |
int blksize = ALIGN(crypto_blkcipher_blocksize(tfm), 4); |
1da177e4c Linux-2.6.12-rc2 |
152 153 154 |
int alen = esp->auth.icv_trunc_len; int elen = skb->len - sizeof(struct ip_esp_hdr) - esp->conf.ivlen - alen; int nfrags; |
31a4ab930 [IPSEC] proto: Mo... |
155 |
int ihl; |
4bf05ecee [IPSEC] esp: Kill... |
156 157 |
u8 nexthdr[2]; struct scatterlist *sg; |
4bf05ecee [IPSEC] esp: Kill... |
158 |
int padlen; |
6b7326c84 [IPSEC] ESP: Use ... |
159 |
int err; |
1da177e4c Linux-2.6.12-rc2 |
160 161 162 163 164 165 166 167 168 |
if (!pskb_may_pull(skb, sizeof(struct ip_esp_hdr))) goto out; if (elen <= 0 || (elen & (blksize-1))) goto out; /* If integrity check is required, do this. */ if (esp->auth.icv_full_len) { |
07d4ee583 [IPSEC]: Use HMAC... |
169 |
u8 sum[alen]; |
1da177e4c Linux-2.6.12-rc2 |
170 |
|
07d4ee583 [IPSEC]: Use HMAC... |
171 172 173 174 175 |
err = esp_mac_digest(esp, skb, 0, skb->len - alen); if (err) goto out; if (skb_copy_bits(skb, skb->len - alen, sum, alen)) |
1da177e4c Linux-2.6.12-rc2 |
176 |
BUG(); |
07d4ee583 [IPSEC]: Use HMAC... |
177 |
if (unlikely(memcmp(esp->auth.work_icv, sum, alen))) { |
1da177e4c Linux-2.6.12-rc2 |
178 179 180 181 182 183 184 185 186 187 188 |
x->stats.integrity_failed++; goto out; } } if ((nfrags = skb_cow_data(skb, 0, &trailer)) < 0) goto out; skb->ip_summed = CHECKSUM_NONE; esph = (struct ip_esp_hdr*)skb->data; |
1da177e4c Linux-2.6.12-rc2 |
189 190 191 |
/* Get ivec. This can be wrong, check against another impls. */ if (esp->conf.ivlen) |
6b7326c84 [IPSEC] ESP: Use ... |
192 |
crypto_blkcipher_set_iv(tfm, esph->enc_data, esp->conf.ivlen); |
1da177e4c Linux-2.6.12-rc2 |
193 |
|
4bf05ecee [IPSEC] esp: Kill... |
194 |
sg = &esp->sgbuf[0]; |
1da177e4c Linux-2.6.12-rc2 |
195 |
|
4bf05ecee [IPSEC] esp: Kill... |
196 197 198 199 200 201 |
if (unlikely(nfrags > ESP_NUM_FAST_SG)) { sg = kmalloc(sizeof(struct scatterlist)*nfrags, GFP_ATOMIC); if (!sg) goto out; } skb_to_sgvec(skb, sg, sizeof(struct ip_esp_hdr) + esp->conf.ivlen, elen); |
6b7326c84 [IPSEC] ESP: Use ... |
202 |
err = crypto_blkcipher_decrypt(&desc, sg, sg, elen); |
4bf05ecee [IPSEC] esp: Kill... |
203 204 |
if (unlikely(sg != &esp->sgbuf[0])) kfree(sg); |
6b7326c84 [IPSEC] ESP: Use ... |
205 206 |
if (unlikely(err)) return err; |
1da177e4c Linux-2.6.12-rc2 |
207 |
|
4bf05ecee [IPSEC] esp: Kill... |
208 209 |
if (skb_copy_bits(skb, skb->len-alen-2, nexthdr, 2)) BUG(); |
1da177e4c Linux-2.6.12-rc2 |
210 |
|
4bf05ecee [IPSEC] esp: Kill... |
211 212 213 |
padlen = nexthdr[0]; if (padlen+2 >= elen) goto out; |
1da177e4c Linux-2.6.12-rc2 |
214 |
|
e905a9eda [NET] IPV4: Fix w... |
215 |
/* ... check padding bits here. Silly. :-) */ |
1da177e4c Linux-2.6.12-rc2 |
216 |
|
eddc9ec53 [SK_BUFF]: Introd... |
217 |
iph = ip_hdr(skb); |
31a4ab930 [IPSEC] proto: Mo... |
218 |
ihl = iph->ihl * 4; |
752c1f4c7 [IPSEC]: Kill pos... |
219 220 |
if (x->encap) { struct xfrm_encap_tmpl *encap = x->encap; |
d56f90a7c [SK_BUFF]: Introd... |
221 |
struct udphdr *uh = (void *)(skb_network_header(skb) + ihl); |
752c1f4c7 [IPSEC]: Kill pos... |
222 223 224 225 226 227 228 229 230 231 232 233 234 |
/* * 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); |
e905a9eda [NET] IPV4: Fix w... |
235 |
|
752c1f4c7 [IPSEC]: Kill pos... |
236 237 238 239 240 241 242 |
/* XXX: perhaps add an extra * policy check here, to see * if we should allow or * reject a packet from a * different source * address/port. */ |
1da177e4c Linux-2.6.12-rc2 |
243 |
} |
e905a9eda [NET] IPV4: Fix w... |
244 |
|
752c1f4c7 [IPSEC]: Kill pos... |
245 246 247 248 249 250 251 |
/* * 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 */ |
0a69452cb [XFRM]: BEET mode |
252 253 |
if (x->props.mode == XFRM_MODE_TRANSPORT || x->props.mode == XFRM_MODE_BEET) |
752c1f4c7 [IPSEC]: Kill pos... |
254 |
skb->ip_summed = CHECKSUM_UNNECESSARY; |
1da177e4c Linux-2.6.12-rc2 |
255 |
} |
4bf05ecee [IPSEC] esp: Kill... |
256 257 |
iph->protocol = nexthdr[1]; pskb_trim(skb, skb->len - alen - padlen - 2); |
967b05f64 [SK_BUFF]: Introd... |
258 259 |
__skb_pull(skb, sizeof(*esph) + esp->conf.ivlen); skb_set_transport_header(skb, -ihl); |
4bf05ecee [IPSEC] esp: Kill... |
260 |
|
1da177e4c Linux-2.6.12-rc2 |
261 262 263 264 265 |
return 0; out: return -EINVAL; } |
c5c252389 [XFRM]: Optimize ... |
266 |
static u32 esp4_get_mtu(struct xfrm_state *x, int mtu) |
1da177e4c Linux-2.6.12-rc2 |
267 268 |
{ struct esp_data *esp = x->data; |
6b7326c84 [IPSEC] ESP: Use ... |
269 |
u32 blksize = ALIGN(crypto_blkcipher_blocksize(esp->conf.tfm), 4); |
c5c252389 [XFRM]: Optimize ... |
270 271 272 273 274 275 |
u32 align = max_t(u32, blksize, esp->conf.padlen); u32 rem; mtu -= x->props.header_len + esp->auth.icv_trunc_len; rem = mtu & (align - 1); mtu &= ~(align - 1); |
0a69452cb [XFRM]: BEET mode |
276 277 278 |
switch (x->props.mode) { case XFRM_MODE_TUNNEL: |
0a69452cb [XFRM]: BEET mode |
279 280 281 282 |
break; default: case XFRM_MODE_TRANSPORT: /* The worst case */ |
c5c252389 [XFRM]: Optimize ... |
283 284 |
mtu -= blksize - 4; mtu += min_t(u32, blksize - 4, rem); |
0a69452cb [XFRM]: BEET mode |
285 286 |
break; case XFRM_MODE_BEET: |
e905a9eda [NET] IPV4: Fix w... |
287 |
/* The worst case. */ |
c5c252389 [XFRM]: Optimize ... |
288 |
mtu += min_t(u32, IPV4_BEET_PHMAXLEN, rem); |
0a69452cb [XFRM]: BEET mode |
289 |
break; |
1da177e4c Linux-2.6.12-rc2 |
290 |
} |
0a69452cb [XFRM]: BEET mode |
291 |
|
c5c252389 [XFRM]: Optimize ... |
292 |
return mtu - 2; |
1da177e4c Linux-2.6.12-rc2 |
293 294 295 296 297 298 299 |
} static void esp4_err(struct sk_buff *skb, u32 info) { struct iphdr *iph = (struct iphdr*)skb->data; struct ip_esp_hdr *esph = (struct ip_esp_hdr*)(skb->data+(iph->ihl<<2)); struct xfrm_state *x; |
88c7664f1 [SK_BUFF]: Introd... |
300 301 |
if (icmp_hdr(skb)->type != ICMP_DEST_UNREACH || icmp_hdr(skb)->code != ICMP_FRAG_NEEDED) |
1da177e4c Linux-2.6.12-rc2 |
302 303 304 305 306 |
return; x = xfrm_state_lookup((xfrm_address_t *)&iph->daddr, esph->spi, IPPROTO_ESP, AF_INET); if (!x) return; |
64ce20730 [NET]: Make NETDE... |
307 308 309 |
NETDEBUG(KERN_DEBUG "pmtu discovery on SA ESP/%08x/%08x ", ntohl(esph->spi), ntohl(iph->daddr)); |
1da177e4c Linux-2.6.12-rc2 |
310 311 312 313 314 315 316 317 318 |
xfrm_state_put(x); } static void esp_destroy(struct xfrm_state *x) { struct esp_data *esp = x->data; if (!esp) return; |
6b7326c84 [IPSEC] ESP: Use ... |
319 |
crypto_free_blkcipher(esp->conf.tfm); |
573dbd959 [CRYPTO]: crypto_... |
320 321 322 |
esp->conf.tfm = NULL; kfree(esp->conf.ivec); esp->conf.ivec = NULL; |
07d4ee583 [IPSEC]: Use HMAC... |
323 |
crypto_free_hash(esp->auth.tfm); |
573dbd959 [CRYPTO]: crypto_... |
324 325 326 |
esp->auth.tfm = NULL; kfree(esp->auth.work_icv); esp->auth.work_icv = NULL; |
1da177e4c Linux-2.6.12-rc2 |
327 328 |
kfree(esp); } |
72cb6962a [IPSEC]: Add xfrm... |
329 |
static int esp_init_state(struct xfrm_state *x) |
1da177e4c Linux-2.6.12-rc2 |
330 331 |
{ struct esp_data *esp = NULL; |
6b7326c84 [IPSEC] ESP: Use ... |
332 |
struct crypto_blkcipher *tfm; |
c5c252389 [XFRM]: Optimize ... |
333 |
u32 align; |
1da177e4c Linux-2.6.12-rc2 |
334 335 336 337 338 339 340 341 |
/* null auth and encryption can have zero length keys */ if (x->aalg) { if (x->aalg->alg_key_len > 512) goto error; } if (x->ealg == NULL) goto error; |
0da974f4f [NET]: Conversion... |
342 |
esp = kzalloc(sizeof(*esp), GFP_KERNEL); |
1da177e4c Linux-2.6.12-rc2 |
343 344 |
if (esp == NULL) return -ENOMEM; |
1da177e4c Linux-2.6.12-rc2 |
345 346 |
if (x->aalg) { struct xfrm_algo_desc *aalg_desc; |
07d4ee583 [IPSEC]: Use HMAC... |
347 |
struct crypto_hash *hash; |
1da177e4c Linux-2.6.12-rc2 |
348 349 350 |
esp->auth.key = x->aalg->alg_key; esp->auth.key_len = (x->aalg->alg_key_len+7)/8; |
07d4ee583 [IPSEC]: Use HMAC... |
351 352 353 354 355 356 357 |
hash = crypto_alloc_hash(x->aalg->alg_name, 0, CRYPTO_ALG_ASYNC); if (IS_ERR(hash)) goto error; esp->auth.tfm = hash; if (crypto_hash_setkey(hash, esp->auth.key, esp->auth.key_len)) |
1da177e4c Linux-2.6.12-rc2 |
358 |
goto error; |
1da177e4c Linux-2.6.12-rc2 |
359 360 361 362 363 |
aalg_desc = xfrm_aalg_get_byname(x->aalg->alg_name, 0); BUG_ON(!aalg_desc); if (aalg_desc->uinfo.auth.icv_fullbits/8 != |
07d4ee583 [IPSEC]: Use HMAC... |
364 |
crypto_hash_digestsize(hash)) { |
64ce20730 [NET]: Make NETDE... |
365 366 367 |
NETDEBUG(KERN_INFO "ESP: %s digestsize %u != %hu ", x->aalg->alg_name, |
07d4ee583 [IPSEC]: Use HMAC... |
368 |
crypto_hash_digestsize(hash), |
64ce20730 [NET]: Make NETDE... |
369 |
aalg_desc->uinfo.auth.icv_fullbits/8); |
1da177e4c Linux-2.6.12-rc2 |
370 371 372 373 374 375 376 377 378 379 380 381 |
goto error; } esp->auth.icv_full_len = aalg_desc->uinfo.auth.icv_fullbits/8; esp->auth.icv_trunc_len = aalg_desc->uinfo.auth.icv_truncbits/8; esp->auth.work_icv = kmalloc(esp->auth.icv_full_len, GFP_KERNEL); if (!esp->auth.work_icv) goto error; } esp->conf.key = x->ealg->alg_key; esp->conf.key_len = (x->ealg->alg_key_len+7)/8; |
6b7326c84 [IPSEC] ESP: Use ... |
382 383 |
tfm = crypto_alloc_blkcipher(x->ealg->alg_name, 0, CRYPTO_ALG_ASYNC); if (IS_ERR(tfm)) |
1da177e4c Linux-2.6.12-rc2 |
384 |
goto error; |
6b7326c84 [IPSEC] ESP: Use ... |
385 386 |
esp->conf.tfm = tfm; esp->conf.ivlen = crypto_blkcipher_ivsize(tfm); |
1da177e4c Linux-2.6.12-rc2 |
387 388 389 390 391 |
esp->conf.padlen = 0; if (esp->conf.ivlen) { esp->conf.ivec = kmalloc(esp->conf.ivlen, GFP_KERNEL); if (unlikely(esp->conf.ivec == NULL)) goto error; |
e4bec827f [IPSEC] esp: Defe... |
392 |
esp->conf.ivinitted = 0; |
1da177e4c Linux-2.6.12-rc2 |
393 |
} |
6b7326c84 [IPSEC] ESP: Use ... |
394 |
if (crypto_blkcipher_setkey(tfm, esp->conf.key, esp->conf.key_len)) |
1da177e4c Linux-2.6.12-rc2 |
395 396 |
goto error; x->props.header_len = sizeof(struct ip_esp_hdr) + esp->conf.ivlen; |
7e49e6de3 [XFRM]: Add XFRM_... |
397 |
if (x->props.mode == XFRM_MODE_TUNNEL) |
1da177e4c Linux-2.6.12-rc2 |
398 |
x->props.header_len += sizeof(struct iphdr); |
ac758e3c5 [XFRM]: beet: fix... |
399 400 |
else if (x->props.mode == XFRM_MODE_BEET) x->props.header_len += IPV4_BEET_PHMAXLEN; |
1da177e4c Linux-2.6.12-rc2 |
401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 |
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; } } x->data = esp; |
c5c252389 [XFRM]: Optimize ... |
416 417 418 419 |
align = ALIGN(crypto_blkcipher_blocksize(esp->conf.tfm), 4); if (esp->conf.padlen) align = max_t(u32, align, esp->conf.padlen); x->props.trailer_len = align + 1 + esp->auth.icv_trunc_len; |
1da177e4c Linux-2.6.12-rc2 |
420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 |
return 0; error: x->data = esp; esp_destroy(x); x->data = NULL; return -EINVAL; } static struct xfrm_type esp_type = { .description = "ESP4", .owner = THIS_MODULE, .proto = IPPROTO_ESP, .init_state = esp_init_state, .destructor = esp_destroy, |
c5c252389 [XFRM]: Optimize ... |
436 |
.get_mtu = esp4_get_mtu, |
1da177e4c Linux-2.6.12-rc2 |
437 |
.input = esp_input, |
1da177e4c Linux-2.6.12-rc2 |
438 439 440 441 442 443 444 445 446 447 448 |
.output = esp_output }; static struct net_protocol esp4_protocol = { .handler = xfrm4_rcv, .err_handler = esp4_err, .no_policy = 1, }; static int __init esp4_init(void) { |
1da177e4c Linux-2.6.12-rc2 |
449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 |
if (xfrm_register_type(&esp_type, AF_INET) < 0) { printk(KERN_INFO "ip esp init: can't add xfrm type "); return -EAGAIN; } if (inet_add_protocol(&esp4_protocol, IPPROTO_ESP) < 0) { printk(KERN_INFO "ip esp init: can't add protocol "); xfrm_unregister_type(&esp_type, AF_INET); return -EAGAIN; } return 0; } static void __exit esp4_fini(void) { if (inet_del_protocol(&esp4_protocol, IPPROTO_ESP) < 0) printk(KERN_INFO "ip esp close: can't remove protocol "); if (xfrm_unregister_type(&esp_type, AF_INET) < 0) printk(KERN_INFO "ip esp close: can't remove xfrm type "); } module_init(esp4_init); module_exit(esp4_fini); MODULE_LICENSE("GPL"); |
d3d6dd3ad [XFRM]: Add modul... |
476 |
MODULE_ALIAS_XFRM_TYPE(AF_INET, XFRM_PROTO_ESP); |