Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
linux
Commits
78bd8961
Commit
78bd8961
authored
May 16, 2003
by
Mitsuru Kanda
Committed by
David S. Miller
May 16, 2003
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
[IPV6]: Add IPCOMP support.
parent
f9c72ecd
Changes
11
Hide whitespace changes
Inline
Side-by-side
Showing
11 changed files
with
417 additions
and
26 deletions
+417
-26
include/linux/ip.h
include/linux/ip.h
+6
-0
include/linux/ipv6.h
include/linux/ipv6.h
+6
-0
include/net/ipcomp.h
include/net/ipcomp.h
+12
-0
include/net/ipv6.h
include/net/ipv6.h
+1
-0
net/ipv4/ipcomp.c
net/ipv4/ipcomp.c
+11
-24
net/ipv6/Kconfig
net/ipv6/Kconfig
+9
-0
net/ipv6/Makefile
net/ipv6/Makefile
+1
-0
net/ipv6/ip6_output.c
net/ipv6/ip6_output.c
+1
-1
net/ipv6/ipcomp6.c
net/ipv6/ipcomp6.c
+368
-0
net/ipv6/ipv6_syms.c
net/ipv6/ipv6_syms.c
+1
-0
net/xfrm/xfrm_input.c
net/xfrm/xfrm_input.c
+1
-1
No files found.
include/linux/ip.h
View file @
78bd8961
...
@@ -198,4 +198,10 @@ struct ip_esp_hdr {
...
@@ -198,4 +198,10 @@ struct ip_esp_hdr {
__u8
enc_data
[
0
];
/* Variable len but >=8. Mind the 64 bit alignment! */
__u8
enc_data
[
0
];
/* Variable len but >=8. Mind the 64 bit alignment! */
};
};
struct
ip_comp_hdr
{
__u8
nexthdr
;
__u8
flags
;
__u16
cpi
;
};
#endif
/* _LINUX_IP_H */
#endif
/* _LINUX_IP_H */
include/linux/ipv6.h
View file @
78bd8961
...
@@ -89,6 +89,12 @@ struct ipv6_esp_hdr {
...
@@ -89,6 +89,12 @@ struct ipv6_esp_hdr {
__u8
enc_data
[
0
];
/* Length variable but >=8. Mind the 64 bit alignment! */
__u8
enc_data
[
0
];
/* Length variable but >=8. Mind the 64 bit alignment! */
};
};
struct
ipv6_comp_hdr
{
__u8
nexthdr
;
__u8
flags
;
__u16
cpi
;
};
/*
/*
* IPv6 fixed header
* IPv6 fixed header
*
*
...
...
include/net/ipcomp.h
0 → 100644
View file @
78bd8961
#ifndef _NET_IPCOMP_H
#define _NET_IPCOMP_H
#define IPCOMP_SCRATCH_SIZE 65400
struct
ipcomp_data
{
u16
threshold
;
u8
*
scratch
;
struct
crypto_tfm
*
tfm
;
};
#endif
include/net/ipv6.h
View file @
78bd8961
...
@@ -315,6 +315,7 @@ extern int ip6_build_xmit(struct sock *sk,
...
@@ -315,6 +315,7 @@ extern int ip6_build_xmit(struct sock *sk,
unsigned
length
,
unsigned
length
,
struct
ipv6_txoptions
*
opt
,
struct
ipv6_txoptions
*
opt
,
int
hlimit
,
int
flags
);
int
hlimit
,
int
flags
);
extern
int
ip6_found_nexthdr
(
struct
sk_buff
*
skb
,
u8
**
nexthdr
);
extern
int
ip6_append_data
(
struct
sock
*
sk
,
extern
int
ip6_append_data
(
struct
sock
*
sk
,
int
getfrag
(
void
*
from
,
char
*
to
,
int
offset
,
int
len
,
int
odd
,
struct
sk_buff
*
skb
),
int
getfrag
(
void
*
from
,
char
*
to
,
int
offset
,
int
len
,
int
odd
,
struct
sk_buff
*
skb
),
...
...
net/ipv4/ipcomp.c
View file @
78bd8961
...
@@ -22,20 +22,7 @@
...
@@ -22,20 +22,7 @@
#include <net/xfrm.h>
#include <net/xfrm.h>
#include <net/icmp.h>
#include <net/icmp.h>
#include <net/esp.h>
#include <net/esp.h>
#include <net/ipcomp.h>
#define IPCOMP_SCRATCH_SIZE 65400
struct
ipcomp_hdr
{
u8
nexthdr
;
u8
flags
;
u16
cpi
;
};
struct
ipcomp_data
{
u16
threshold
;
u8
*
scratch
;
struct
crypto_tfm
*
tfm
;
};
static
int
ipcomp_decompress
(
struct
xfrm_state
*
x
,
struct
sk_buff
*
skb
)
static
int
ipcomp_decompress
(
struct
xfrm_state
*
x
,
struct
sk_buff
*
skb
)
{
{
...
@@ -52,7 +39,7 @@ static int ipcomp_decompress(struct xfrm_state *x, struct sk_buff *skb)
...
@@ -52,7 +39,7 @@ static int ipcomp_decompress(struct xfrm_state *x, struct sk_buff *skb)
if
(
err
)
if
(
err
)
goto
out
;
goto
out
;
if
(
dlen
<
(
plen
+
sizeof
(
struct
ipcomp_hdr
)))
{
if
(
dlen
<
(
plen
+
sizeof
(
struct
ip
_
comp_hdr
)))
{
err
=
-
EINVAL
;
err
=
-
EINVAL
;
goto
out
;
goto
out
;
}
}
...
@@ -93,11 +80,11 @@ static int ipcomp_input(struct xfrm_state *x,
...
@@ -93,11 +80,11 @@ static int ipcomp_input(struct xfrm_state *x,
iph
=
skb
->
nh
.
iph
;
iph
=
skb
->
nh
.
iph
;
memcpy
(
&
tmp_iph
,
iph
,
iph
->
ihl
*
4
);
memcpy
(
&
tmp_iph
,
iph
,
iph
->
ihl
*
4
);
nexthdr
=
*
(
u8
*
)
skb
->
data
;
nexthdr
=
*
(
u8
*
)
skb
->
data
;
skb_pull
(
skb
,
sizeof
(
struct
ipcomp_hdr
));
skb_pull
(
skb
,
sizeof
(
struct
ip
_
comp_hdr
));
skb
->
nh
.
raw
+=
sizeof
(
struct
ipcomp_hdr
);
skb
->
nh
.
raw
+=
sizeof
(
struct
ip
_
comp_hdr
);
memcpy
(
skb
->
nh
.
raw
,
&
tmp_iph
,
tmp_iph
.
iph
.
ihl
*
4
);
memcpy
(
skb
->
nh
.
raw
,
&
tmp_iph
,
tmp_iph
.
iph
.
ihl
*
4
);
iph
=
skb
->
nh
.
iph
;
iph
=
skb
->
nh
.
iph
;
iph
->
tot_len
=
htons
(
ntohs
(
iph
->
tot_len
)
-
sizeof
(
struct
ipcomp_hdr
));
iph
->
tot_len
=
htons
(
ntohs
(
iph
->
tot_len
)
-
sizeof
(
struct
ip
_
comp_hdr
));
iph
->
protocol
=
nexthdr
;
iph
->
protocol
=
nexthdr
;
skb
->
h
.
raw
=
skb
->
data
;
skb
->
h
.
raw
=
skb
->
data
;
err
=
ipcomp_decompress
(
x
,
skb
);
err
=
ipcomp_decompress
(
x
,
skb
);
...
@@ -122,7 +109,7 @@ static int ipcomp_compress(struct xfrm_state *x, struct sk_buff *skb)
...
@@ -122,7 +109,7 @@ static int ipcomp_compress(struct xfrm_state *x, struct sk_buff *skb)
if
(
err
)
if
(
err
)
goto
out
;
goto
out
;
if
((
dlen
+
sizeof
(
struct
ipcomp_hdr
))
>=
plen
)
{
if
((
dlen
+
sizeof
(
struct
ip
_
comp_hdr
))
>=
plen
)
{
err
=
-
EMSGSIZE
;
err
=
-
EMSGSIZE
;
goto
out
;
goto
out
;
}
}
...
@@ -162,7 +149,7 @@ static int ipcomp_output(struct sk_buff *skb)
...
@@ -162,7 +149,7 @@ static int ipcomp_output(struct sk_buff *skb)
struct
dst_entry
*
dst
=
skb
->
dst
;
struct
dst_entry
*
dst
=
skb
->
dst
;
struct
xfrm_state
*
x
=
dst
->
xfrm
;
struct
xfrm_state
*
x
=
dst
->
xfrm
;
struct
iphdr
*
iph
,
*
top_iph
;
struct
iphdr
*
iph
,
*
top_iph
;
struct
ipcomp_hdr
*
ipch
;
struct
ip
_
comp_hdr
*
ipch
;
struct
ipcomp_data
*
ipcd
=
x
->
data
;
struct
ipcomp_data
*
ipcd
=
x
->
data
;
union
{
union
{
struct
iphdr
iph
;
struct
iphdr
iph
;
...
@@ -215,13 +202,13 @@ static int ipcomp_output(struct sk_buff *skb)
...
@@ -215,13 +202,13 @@ static int ipcomp_output(struct sk_buff *skb)
/* Install ipcomp header, convert into ipcomp datagram. */
/* Install ipcomp header, convert into ipcomp datagram. */
iph
=
skb
->
nh
.
iph
;
iph
=
skb
->
nh
.
iph
;
memcpy
(
&
tmp_iph
,
iph
,
iph
->
ihl
*
4
);
memcpy
(
&
tmp_iph
,
iph
,
iph
->
ihl
*
4
);
top_iph
=
(
struct
iphdr
*
)
skb_push
(
skb
,
sizeof
(
struct
ipcomp_hdr
));
top_iph
=
(
struct
iphdr
*
)
skb_push
(
skb
,
sizeof
(
struct
ip
_
comp_hdr
));
memcpy
(
top_iph
,
&
tmp_iph
,
iph
->
ihl
*
4
);
memcpy
(
top_iph
,
&
tmp_iph
,
iph
->
ihl
*
4
);
iph
=
top_iph
;
iph
=
top_iph
;
iph
->
tot_len
=
htons
(
skb
->
len
);
iph
->
tot_len
=
htons
(
skb
->
len
);
iph
->
protocol
=
IPPROTO_COMP
;
iph
->
protocol
=
IPPROTO_COMP
;
iph
->
check
=
0
;
iph
->
check
=
0
;
ipch
=
(
struct
ipcomp_hdr
*
)((
char
*
)
iph
+
iph
->
ihl
*
4
);
ipch
=
(
struct
ip
_
comp_hdr
*
)((
char
*
)
iph
+
iph
->
ihl
*
4
);
ipch
->
nexthdr
=
x
->
props
.
mode
?
IPPROTO_IPIP
:
tmp_iph
.
iph
.
protocol
;
ipch
->
nexthdr
=
x
->
props
.
mode
?
IPPROTO_IPIP
:
tmp_iph
.
iph
.
protocol
;
ipch
->
flags
=
0
;
ipch
->
flags
=
0
;
ipch
->
cpi
=
htons
((
u16
)
ntohl
(
x
->
id
.
spi
));
ipch
->
cpi
=
htons
((
u16
)
ntohl
(
x
->
id
.
spi
));
...
@@ -252,7 +239,7 @@ static void ipcomp4_err(struct sk_buff *skb, u32 info)
...
@@ -252,7 +239,7 @@ static void ipcomp4_err(struct sk_buff *skb, u32 info)
{
{
u32
spi
;
u32
spi
;
struct
iphdr
*
iph
=
(
struct
iphdr
*
)
skb
->
data
;
struct
iphdr
*
iph
=
(
struct
iphdr
*
)
skb
->
data
;
struct
ip
comp_hdr
*
ipch
=
(
struct
ip
comp_hdr
*
)(
skb
->
data
+
(
iph
->
ihl
<<
2
));
struct
ip
_comp_hdr
*
ipch
=
(
struct
ip_
comp_hdr
*
)(
skb
->
data
+
(
iph
->
ihl
<<
2
));
struct
xfrm_state
*
x
;
struct
xfrm_state
*
x
;
if
(
skb
->
h
.
icmph
->
type
!=
ICMP_DEST_UNREACH
||
if
(
skb
->
h
.
icmph
->
type
!=
ICMP_DEST_UNREACH
||
...
@@ -356,7 +343,7 @@ static int ipcomp_init_state(struct xfrm_state *x, void *args)
...
@@ -356,7 +343,7 @@ static int ipcomp_init_state(struct xfrm_state *x, void *args)
goto
error
;
goto
error
;
memset
(
ipcd
,
0
,
sizeof
(
*
ipcd
));
memset
(
ipcd
,
0
,
sizeof
(
*
ipcd
));
x
->
props
.
header_len
=
sizeof
(
struct
ipcomp_hdr
);
x
->
props
.
header_len
=
sizeof
(
struct
ip
_
comp_hdr
);
if
(
x
->
props
.
mode
)
if
(
x
->
props
.
mode
)
x
->
props
.
header_len
+=
sizeof
(
struct
iphdr
);
x
->
props
.
header_len
+=
sizeof
(
struct
iphdr
);
x
->
data
=
ipcd
;
x
->
data
=
ipcd
;
...
...
net/ipv6/Kconfig
View file @
78bd8961
...
@@ -33,4 +33,13 @@ config INET6_ESP
...
@@ -33,4 +33,13 @@ config INET6_ESP
If unsure, say Y.
If unsure, say Y.
config INET6_IPCOMP
tristate "IPv6: IPComp transformation"
depends on IPV6
---help---
Support for IP Paylod Compression (RFC3173), typically needed
for IPsec.
If unsure, say Y.
source "net/ipv6/netfilter/Kconfig"
source "net/ipv6/netfilter/Kconfig"
net/ipv6/Makefile
View file @
78bd8961
...
@@ -13,4 +13,5 @@ ipv6-objs := af_inet6.o anycast.o ip6_output.o ip6_input.o addrconf.o sit.o \
...
@@ -13,4 +13,5 @@ ipv6-objs := af_inet6.o anycast.o ip6_output.o ip6_input.o addrconf.o sit.o \
obj-$(CONFIG_INET6_AH)
+=
ah6.o
obj-$(CONFIG_INET6_AH)
+=
ah6.o
obj-$(CONFIG_INET6_ESP)
+=
esp6.o
obj-$(CONFIG_INET6_ESP)
+=
esp6.o
obj-$(CONFIG_INET6_IPCOMP)
+=
ipcomp6.o
obj-$(CONFIG_NETFILTER)
+=
netfilter/
obj-$(CONFIG_NETFILTER)
+=
netfilter/
net/ipv6/ip6_output.c
View file @
78bd8961
...
@@ -887,7 +887,7 @@ static void ip6_copy_metadata(struct sk_buff *to, struct sk_buff *from)
...
@@ -887,7 +887,7 @@ static void ip6_copy_metadata(struct sk_buff *to, struct sk_buff *from)
#endif
#endif
}
}
static
int
ip6_found_nexthdr
(
struct
sk_buff
*
skb
,
u8
**
nexthdr
)
int
ip6_found_nexthdr
(
struct
sk_buff
*
skb
,
u8
**
nexthdr
)
{
{
u16
offset
=
sizeof
(
struct
ipv6hdr
);
u16
offset
=
sizeof
(
struct
ipv6hdr
);
struct
ipv6_opt_hdr
*
exthdr
=
(
struct
ipv6_opt_hdr
*
)(
skb
->
nh
.
ipv6h
+
1
);
struct
ipv6_opt_hdr
*
exthdr
=
(
struct
ipv6_opt_hdr
*
)(
skb
->
nh
.
ipv6h
+
1
);
...
...
net/ipv6/ipcomp6.c
0 → 100644
View file @
78bd8961
/*
* IP Payload Compression Protocol (IPComp) for IPv6 - RFC3713
*
* Copyright (C)2003 USAGI/WIDE Project
*
* Author Mitsuru KANDA <mk@linux-ipv6.org>
*
* 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.
*
* 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.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
/*
* [Memo]
*
* Outbound:
* The compression of IP datagram MUST be done before AH/ESP processing,
* fragmentation, and the addition of Hop-by-Hop/Routing header.
*
* Inbound:
* The decompression of IP datagram MUST be done after the reassembly,
* AH/ESP processing.
*/
#include <linux/config.h>
#include <linux/module.h>
#include <net/ip.h>
#include <net/xfrm.h>
#include <net/ipcomp.h>
#include <asm/scatterlist.h>
#include <linux/crypto.h>
#include <linux/pfkeyv2.h>
#include <linux/random.h>
#include <net/icmp.h>
#include <net/ipv6.h>
#include <linux/ipv6.h>
#include <linux/icmpv6.h>
/* XXX no ipv6 ipcomp specific */
#define NIP6(addr) \
ntohs((addr).s6_addr16[0]),\
ntohs((addr).s6_addr16[1]),\
ntohs((addr).s6_addr16[2]),\
ntohs((addr).s6_addr16[3]),\
ntohs((addr).s6_addr16[4]),\
ntohs((addr).s6_addr16[5]),\
ntohs((addr).s6_addr16[6]),\
ntohs((addr).s6_addr16[7])
static
int
ipcomp6_input
(
struct
xfrm_state
*
x
,
struct
xfrm_decap_state
*
decap
,
struct
sk_buff
*
skb
)
{
int
err
=
0
;
u8
nexthdr
=
0
;
u8
*
prevhdr
;
int
hdr_len
=
skb
->
h
.
raw
-
skb
->
nh
.
raw
;
unsigned
char
*
tmp_hdr
=
NULL
;
struct
ipv6hdr
*
iph
;
int
plen
,
dlen
;
struct
ipcomp_data
*
ipcd
=
x
->
data
;
u8
*
start
,
*
scratch
=
ipcd
->
scratch
;
if
((
skb_is_nonlinear
(
skb
)
||
skb_cloned
(
skb
))
&&
skb_linearize
(
skb
,
GFP_ATOMIC
)
!=
0
)
{
err
=
-
ENOMEM
;
goto
out
;
}
skb
->
ip_summed
=
CHECKSUM_NONE
;
/* Remove ipcomp header and decompress original payload */
iph
=
skb
->
nh
.
ipv6h
;
tmp_hdr
=
kmalloc
(
hdr_len
,
GFP_ATOMIC
);
if
(
!
tmp_hdr
)
goto
out
;
memcpy
(
tmp_hdr
,
iph
,
hdr_len
);
nexthdr
=
*
(
u8
*
)
skb
->
data
;
skb_pull
(
skb
,
sizeof
(
struct
ipv6_comp_hdr
));
skb
->
nh
.
raw
+=
sizeof
(
struct
ipv6_comp_hdr
);
memcpy
(
skb
->
nh
.
raw
,
tmp_hdr
,
hdr_len
);
iph
=
skb
->
nh
.
ipv6h
;
iph
->
payload_len
=
htons
(
ntohs
(
iph
->
payload_len
)
-
sizeof
(
struct
ipv6_comp_hdr
));
skb
->
h
.
raw
=
skb
->
data
;
/* decompression */
plen
=
skb
->
len
;
dlen
=
IPCOMP_SCRATCH_SIZE
;
start
=
skb
->
data
;
err
=
crypto_comp_decompress
(
ipcd
->
tfm
,
start
,
plen
,
scratch
,
&
dlen
);
if
(
err
)
{
err
=
-
EINVAL
;
goto
out
;
}
if
(
dlen
<
(
plen
+
sizeof
(
struct
ipv6_comp_hdr
)))
{
err
=
-
EINVAL
;
goto
out
;
}
err
=
pskb_expand_head
(
skb
,
0
,
dlen
-
plen
,
GFP_ATOMIC
);
if
(
err
)
{
goto
out
;
}
skb_put
(
skb
,
dlen
-
plen
);
memcpy
(
skb
->
data
,
scratch
,
dlen
);
iph
=
skb
->
nh
.
ipv6h
;
iph
->
payload_len
=
htons
(
skb
->
len
);
ip6_found_nexthdr
(
skb
,
&
prevhdr
);
*
prevhdr
=
nexthdr
;
out:
if
(
tmp_hdr
)
kfree
(
tmp_hdr
);
if
(
err
)
goto
error_out
;
return
nexthdr
;
error_out:
return
err
;
}
static
int
ipcomp6_output
(
struct
sk_buff
*
skb
)
{
int
err
;
struct
dst_entry
*
dst
=
skb
->
dst
;
struct
xfrm_state
*
x
=
dst
->
xfrm
;
struct
ipv6hdr
*
tmp_iph
=
NULL
,
*
iph
,
*
top_iph
;
int
hdr_len
=
0
;
struct
ipv6_comp_hdr
*
ipch
;
struct
ipcomp_data
*
ipcd
=
x
->
data
;
u8
*
prevhdr
;
u8
nexthdr
=
0
;
int
plen
,
dlen
;
u8
*
start
,
*
scratch
=
ipcd
->
scratch
;
if
(
skb
->
ip_summed
==
CHECKSUM_HW
&&
skb_checksum_help
(
skb
)
==
NULL
)
{
err
=
-
EINVAL
;
goto
error_nolock
;
}
spin_lock_bh
(
&
x
->
lock
);
err
=
xfrm_check_output
(
x
,
skb
,
AF_INET6
);
if
(
err
)
goto
error
;
if
(
x
->
props
.
mode
)
{
hdr_len
=
sizeof
(
struct
ipv6hdr
);
nexthdr
=
IPPROTO_IPV6
;
iph
=
skb
->
nh
.
ipv6h
;
top_iph
=
(
struct
ipv6hdr
*
)
skb_push
(
skb
,
sizeof
(
struct
ipv6hdr
));
top_iph
->
version
=
6
;
top_iph
->
priority
=
iph
->
priority
;
top_iph
->
flow_lbl
[
0
]
=
iph
->
flow_lbl
[
0
];
top_iph
->
flow_lbl
[
1
]
=
iph
->
flow_lbl
[
1
];
top_iph
->
flow_lbl
[
2
]
=
iph
->
flow_lbl
[
2
];
top_iph
->
nexthdr
=
IPPROTO_IPV6
;
/* initial */
top_iph
->
payload_len
=
htons
(
skb
->
len
-
sizeof
(
struct
ipv6hdr
));
top_iph
->
hop_limit
=
iph
->
hop_limit
;
memcpy
(
&
top_iph
->
saddr
,
(
struct
in6_addr
*
)
&
x
->
props
.
saddr
,
sizeof
(
struct
in6_addr
));
memcpy
(
&
top_iph
->
daddr
,
(
struct
in6_addr
*
)
&
x
->
id
.
daddr
,
sizeof
(
struct
in6_addr
));
skb
->
nh
.
raw
=
skb
->
data
;
/* == top_iph */
skb
->
h
.
raw
=
skb
->
nh
.
raw
+
hdr_len
;
}
else
{
hdr_len
=
ip6_found_nexthdr
(
skb
,
&
prevhdr
);
nexthdr
=
*
prevhdr
;
}
/* check whether datagram len is larger than threshold */
if
((
skb
->
len
-
hdr_len
)
<
ipcd
->
threshold
)
{
goto
out_ok
;
}
if
((
skb_is_nonlinear
(
skb
)
||
skb_cloned
(
skb
))
&&
skb_linearize
(
skb
,
GFP_ATOMIC
)
!=
0
)
{
err
=
-
ENOMEM
;
goto
error
;
}
/* compression */
plen
=
skb
->
len
-
hdr_len
;
dlen
=
IPCOMP_SCRATCH_SIZE
;
start
=
skb
->
data
+
hdr_len
;
err
=
crypto_comp_compress
(
ipcd
->
tfm
,
start
,
plen
,
scratch
,
&
dlen
);
if
(
err
)
{
goto
error
;
}
if
((
dlen
+
sizeof
(
struct
ipv6_comp_hdr
))
>=
plen
)
{
goto
out_ok
;
}
memcpy
(
start
,
scratch
,
dlen
);
pskb_trim
(
skb
,
hdr_len
+
dlen
);
/* insert ipcomp header and replace datagram */
tmp_iph
=
kmalloc
(
hdr_len
,
GFP_ATOMIC
);
if
(
!
tmp_iph
)
{
err
=
-
ENOMEM
;
goto
error
;
}
memcpy
(
tmp_iph
,
skb
->
nh
.
raw
,
hdr_len
);
top_iph
=
(
struct
ipv6hdr
*
)
skb_push
(
skb
,
sizeof
(
struct
ipv6_comp_hdr
));
memcpy
(
top_iph
,
tmp_iph
,
hdr_len
);
kfree
(
tmp_iph
);
top_iph
->
payload_len
=
htons
(
skb
->
len
-
sizeof
(
struct
ipv6hdr
));
skb
->
nh
.
raw
=
skb
->
data
;
/* top_iph */
ip6_found_nexthdr
(
skb
,
&
prevhdr
);
*
prevhdr
=
IPPROTO_COMP
;
ipch
=
(
struct
ipv6_comp_hdr
*
)((
unsigned
char
*
)
top_iph
+
hdr_len
);
ipch
->
nexthdr
=
nexthdr
;
ipch
->
flags
=
0
;
ipch
->
cpi
=
htons
((
u16
)
ntohl
(
x
->
id
.
spi
));
skb
->
h
.
raw
=
(
unsigned
char
*
)
ipch
;
out_ok:
x
->
curlft
.
bytes
+=
skb
->
len
;
x
->
curlft
.
packets
++
;
spin_unlock_bh
(
&
x
->
lock
);
if
((
skb
->
dst
=
dst_pop
(
dst
))
==
NULL
)
{
err
=
-
EHOSTUNREACH
;
goto
error_nolock
;
}
err
=
NET_XMIT_BYPASS
;
out_exit:
return
err
;
error:
spin_unlock_bh
(
&
x
->
lock
);
error_nolock:
kfree_skb
(
skb
);
goto
out_exit
;
}
static
void
ipcomp6_err
(
struct
sk_buff
*
skb
,
struct
inet6_skb_parm
*
opt
,
int
type
,
int
code
,
int
offset
,
__u32
info
)
{
u32
spi
;
struct
ipv6hdr
*
iph
=
(
struct
ipv6hdr
*
)
skb
->
data
;
struct
ipv6_comp_hdr
*
ipcomph
=
(
struct
ipv6_comp_hdr
*
)(
skb
->
data
+
offset
);
struct
xfrm_state
*
x
;
if
(
type
!=
ICMPV6_DEST_UNREACH
||
type
!=
ICMPV6_PKT_TOOBIG
)
return
;
spi
=
ntohl
(
ntohs
(
ipcomph
->
cpi
));
x
=
xfrm_state_lookup
((
xfrm_address_t
*
)
&
iph
->
daddr
,
spi
,
IPPROTO_COMP
,
AF_INET6
);
if
(
!
x
)
return
;
printk
(
KERN_DEBUG
"pmtu discvovery on SA IPCOMP/%08x/"
"%04x:%04x:%04x:%04x:%04x:%04x:%04x:%04x
\n
"
,
spi
,
NIP6
(
iph
->
daddr
));
xfrm_state_put
(
x
);
}
static
void
ipcomp6_free_data
(
struct
ipcomp_data
*
ipcd
)
{
if
(
ipcd
->
tfm
)
crypto_free_tfm
(
ipcd
->
tfm
);
if
(
ipcd
->
scratch
)
kfree
(
ipcd
->
scratch
);
}
static
void
ipcomp6_destroy
(
struct
xfrm_state
*
x
)
{
struct
ipcomp_data
*
ipcd
=
x
->
data
;
ipcomp6_free_data
(
ipcd
);
kfree
(
ipcd
);
}
static
int
ipcomp6_init_state
(
struct
xfrm_state
*
x
,
void
*
args
)
{
int
err
=
-
ENOMEM
;
struct
ipcomp_data
*
ipcd
;
struct
xfrm_algo_desc
*
calg_desc
;
ipcd
=
kmalloc
(
sizeof
(
*
ipcd
),
GFP_KERNEL
);
if
(
!
ipcd
)
goto
error
;
memset
(
ipcd
,
0
,
sizeof
(
*
ipcd
));
x
->
props
.
header_len
=
sizeof
(
struct
ipv6_comp_hdr
);
if
(
x
->
props
.
mode
)
x
->
props
.
header_len
+=
sizeof
(
struct
ipv6hdr
);
x
->
data
=
ipcd
;
ipcd
->
scratch
=
kmalloc
(
IPCOMP_SCRATCH_SIZE
,
GFP_KERNEL
);
if
(
!
ipcd
->
scratch
)
goto
error
;
ipcd
->
tfm
=
crypto_alloc_tfm
(
x
->
calg
->
alg_name
,
0
);
if
(
!
ipcd
->
tfm
)
goto
error
;
calg_desc
=
xfrm_calg_get_byname
(
x
->
calg
->
alg_name
);
BUG_ON
(
!
calg_desc
);
ipcd
->
threshold
=
calg_desc
->
uinfo
.
comp
.
threshold
;
err
=
0
;
out:
return
err
;
error:
if
(
ipcd
)
{
ipcomp6_free_data
(
ipcd
);
kfree
(
ipcd
);
}
goto
out
;
}
static
struct
xfrm_type
ipcomp6_type
=
{
.
description
=
"IPCOMP6"
,
.
owner
=
THIS_MODULE
,
.
proto
=
IPPROTO_COMP
,
.
init_state
=
ipcomp6_init_state
,
.
destructor
=
ipcomp6_destroy
,
.
input
=
ipcomp6_input
,
.
output
=
ipcomp6_output
,
};
static
struct
inet6_protocol
ipcomp6_protocol
=
{
.
handler
=
xfrm6_rcv
,
.
err_handler
=
ipcomp6_err
,
.
flags
=
INET6_PROTO_NOPOLICY
,
};
static
int
__init
ipcomp6_init
(
void
)
{
if
(
xfrm_register_type
(
&
ipcomp6_type
,
AF_INET6
)
<
0
)
{
printk
(
KERN_INFO
"ipcomp6 init: can't add xfrm type
\n
"
);
return
-
EAGAIN
;
}
if
(
inet6_add_protocol
(
&
ipcomp6_protocol
,
IPPROTO_COMP
)
<
0
)
{
printk
(
KERN_INFO
"ipcomp6 init: can't add protocol
\n
"
);
xfrm_unregister_type
(
&
ipcomp6_type
,
AF_INET6
);
return
-
EAGAIN
;
}
return
0
;
}
static
void
__exit
ipcomp6_fini
(
void
)
{
if
(
inet6_del_protocol
(
&
ipcomp6_protocol
,
IPPROTO_COMP
)
<
0
)
printk
(
KERN_INFO
"ipv6 ipcomp close: can't remove protocol
\n
"
);
if
(
xfrm_unregister_type
(
&
ipcomp6_type
,
AF_INET6
)
<
0
)
printk
(
KERN_INFO
"ipv6 ipcomp close: can't remove xfrm type
\n
"
);
}
module_init
(
ipcomp6_init
);
module_exit
(
ipcomp6_fini
);
MODULE_LICENSE
(
"GPL"
);
MODULE_DESCRIPTION
(
"IP Payload Compression Protocol (IPComp) for IPv6 - RFC3173"
);
MODULE_AUTHOR
(
"Mitsuru KANDA <mk@linux-ipv6.org>"
);
net/ipv6/ipv6_syms.c
View file @
78bd8961
...
@@ -35,5 +35,6 @@ EXPORT_SYMBOL(ipv6_chk_addr);
...
@@ -35,5 +35,6 @@ EXPORT_SYMBOL(ipv6_chk_addr);
EXPORT_SYMBOL
(
in6addr_any
);
EXPORT_SYMBOL
(
in6addr_any
);
EXPORT_SYMBOL
(
in6addr_loopback
);
EXPORT_SYMBOL
(
in6addr_loopback
);
EXPORT_SYMBOL
(
in6_dev_finish_destroy
);
EXPORT_SYMBOL
(
in6_dev_finish_destroy
);
EXPORT_SYMBOL
(
ip6_found_nexthdr
);
EXPORT_SYMBOL
(
xfrm6_rcv
);
EXPORT_SYMBOL
(
xfrm6_rcv
);
EXPORT_SYMBOL
(
xfrm6_clear_mutable_options
);
EXPORT_SYMBOL
(
xfrm6_clear_mutable_options
);
net/xfrm/xfrm_input.c
View file @
78bd8961
...
@@ -34,7 +34,7 @@ int xfrm_parse_spi(struct sk_buff *skb, u8 nexthdr, u32 *spi, u32 *seq)
...
@@ -34,7 +34,7 @@ int xfrm_parse_spi(struct sk_buff *skb, u8 nexthdr, u32 *spi, u32 *seq)
offset_seq
=
offsetof
(
struct
ip_esp_hdr
,
seq_no
);
offset_seq
=
offsetof
(
struct
ip_esp_hdr
,
seq_no
);
break
;
break
;
case
IPPROTO_COMP
:
case
IPPROTO_COMP
:
if
(
!
pskb_may_pull
(
skb
,
4
))
if
(
!
pskb_may_pull
(
skb
,
sizeof
(
struct
ip_comp_hdr
)
))
return
-
EINVAL
;
return
-
EINVAL
;
*
spi
=
ntohl
(
ntohs
(
*
(
u16
*
)(
skb
->
h
.
raw
+
2
)));
*
spi
=
ntohl
(
ntohs
(
*
(
u16
*
)(
skb
->
h
.
raw
+
2
)));
*
seq
=
0
;
*
seq
=
0
;
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment