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
Kirill Smelkov
linux
Commits
6bac9098
Commit
6bac9098
authored
May 12, 2003
by
Kazunori Miyazawa
Committed by
David S. Miller
May 12, 2003
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
[IPV4]: Introduce ip6_append_data.
parent
d983ec93
Changes
8
Show whitespace changes
Inline
Side-by-side
Showing
8 changed files
with
1222 additions
and
253 deletions
+1222
-253
include/linux/ipv6.h
include/linux/ipv6.h
+7
-0
include/linux/skbuff.h
include/linux/skbuff.h
+9
-0
include/net/ipv6.h
include/net/ipv6.h
+20
-0
net/ipv4/ip_output.c
net/ipv4/ip_output.c
+0
-10
net/ipv6/icmp.c
net/ipv6/icmp.c
+130
-80
net/ipv6/ip6_output.c
net/ipv6/ip6_output.c
+668
-1
net/ipv6/raw.c
net/ipv6/raw.c
+148
-82
net/ipv6/udp.c
net/ipv6/udp.c
+240
-80
No files found.
include/linux/ipv6.h
View file @
6bac9098
...
@@ -121,6 +121,7 @@ struct ipv6hdr {
...
@@ -121,6 +121,7 @@ struct ipv6hdr {
#include <linux/icmpv6.h>
#include <linux/icmpv6.h>
#include <net/if_inet6.h>
/* struct ipv6_mc_socklist */
#include <net/if_inet6.h>
/* struct ipv6_mc_socklist */
#include <linux/tcp.h>
#include <linux/tcp.h>
#include <linux/udp.h>
/*
/*
This structure contains results of exthdrs parsing
This structure contains results of exthdrs parsing
...
@@ -178,6 +179,11 @@ struct ipv6_pinfo {
...
@@ -178,6 +179,11 @@ struct ipv6_pinfo {
struct
ipv6_txoptions
*
opt
;
struct
ipv6_txoptions
*
opt
;
struct
sk_buff
*
pktoptions
;
struct
sk_buff
*
pktoptions
;
struct
{
struct
ipv6_txoptions
*
opt
;
struct
rt6_info
*
rt
;
struct
flowi
*
fl
;
}
cork
;
};
};
struct
raw6_opt
{
struct
raw6_opt
{
...
@@ -200,6 +206,7 @@ struct udp6_sock {
...
@@ -200,6 +206,7 @@ struct udp6_sock {
struct
sock
sk
;
struct
sock
sk
;
struct
ipv6_pinfo
*
pinet6
;
struct
ipv6_pinfo
*
pinet6
;
struct
inet_opt
inet
;
struct
inet_opt
inet
;
struct
udp_opt
udp
;
struct
ipv6_pinfo
inet6
;
struct
ipv6_pinfo
inet6
;
};
};
...
...
include/linux/skbuff.h
View file @
6bac9098
...
@@ -792,6 +792,15 @@ static inline int skb_pagelen(const struct sk_buff *skb)
...
@@ -792,6 +792,15 @@ static inline int skb_pagelen(const struct sk_buff *skb)
return
len
+
skb_headlen
(
skb
);
return
len
+
skb_headlen
(
skb
);
}
}
static
inline
void
skb_fill_page_desc
(
struct
sk_buff
*
skb
,
int
i
,
struct
page
*
page
,
int
off
,
int
size
)
{
skb_frag_t
*
frag
=
&
skb_shinfo
(
skb
)
->
frags
[
i
];
frag
->
page
=
page
;
frag
->
page_offset
=
off
;
frag
->
size
=
size
;
skb_shinfo
(
skb
)
->
nr_frags
=
i
+
1
;
}
#define SKB_PAGE_ASSERT(skb) do { if (skb_shinfo(skb)->nr_frags) \
#define SKB_PAGE_ASSERT(skb) do { if (skb_shinfo(skb)->nr_frags) \
BUG(); } while (0)
BUG(); } while (0)
#define SKB_FRAG_ASSERT(skb) do { if (skb_shinfo(skb)->frag_list) \
#define SKB_FRAG_ASSERT(skb) do { if (skb_shinfo(skb)->frag_list) \
...
...
include/net/ipv6.h
View file @
6bac9098
...
@@ -316,6 +316,26 @@ extern int ip6_build_xmit(struct sock *sk,
...
@@ -316,6 +316,26 @@ extern int ip6_build_xmit(struct sock *sk,
struct
ipv6_txoptions
*
opt
,
struct
ipv6_txoptions
*
opt
,
int
hlimit
,
int
flags
);
int
hlimit
,
int
flags
);
extern
int
ip6_append_data
(
struct
sock
*
sk
,
int
getfrag
(
void
*
from
,
char
*
to
,
int
offset
,
int
len
,
int
odd
,
struct
sk_buff
*
skb
),
void
*
from
,
int
length
,
int
transhdrlen
,
int
hlimit
,
struct
ipv6_txoptions
*
opt
,
struct
flowi
*
fl
,
struct
rt6_info
*
rt
,
unsigned
int
flags
);
extern
int
ip6_push_pending_frames
(
struct
sock
*
sk
);
extern
void
ip6_flush_pending_frames
(
struct
sock
*
sk
);
extern
int
ip6_dst_lookup
(
struct
sock
*
sk
,
struct
dst_entry
**
dst
,
struct
flowi
*
fl
,
struct
in6_addr
**
saddr
);
/*
/*
* skb processing functions
* skb processing functions
*/
*/
...
...
net/ipv4/ip_output.c
View file @
6bac9098
...
@@ -685,16 +685,6 @@ skb_can_coalesce(struct sk_buff *skb, int i, struct page *page, int off)
...
@@ -685,16 +685,6 @@ skb_can_coalesce(struct sk_buff *skb, int i, struct page *page, int off)
return
0
;
return
0
;
}
}
static
void
skb_fill_page_desc
(
struct
sk_buff
*
skb
,
int
i
,
struct
page
*
page
,
int
off
,
int
size
)
{
skb_frag_t
*
frag
=
&
skb_shinfo
(
skb
)
->
frags
[
i
];
frag
->
page
=
page
;
frag
->
page_offset
=
off
;
frag
->
size
=
size
;
skb_shinfo
(
skb
)
->
nr_frags
=
i
+
1
;
}
static
inline
unsigned
int
static
inline
unsigned
int
csum_page
(
struct
page
*
page
,
int
offset
,
int
copy
)
csum_page
(
struct
page
*
page
,
int
offset
,
int
copy
)
{
{
...
...
net/ipv6/icmp.c
View file @
6bac9098
...
@@ -28,6 +28,7 @@
...
@@ -28,6 +28,7 @@
* YOSHIFUJI Hideaki @USAGI: added sysctl for icmp rate limit.
* YOSHIFUJI Hideaki @USAGI: added sysctl for icmp rate limit.
* Randy Dunlap and
* Randy Dunlap and
* YOSHIFUJI Hideaki @USAGI: Per-interface statistics support
* YOSHIFUJI Hideaki @USAGI: Per-interface statistics support
* Kazunori MIYAZAWA @USAGI: change output process to use ip6_append_data
*/
*/
#include <linux/module.h>
#include <linux/module.h>
...
@@ -104,42 +105,6 @@ static __inline__ void icmpv6_xmit_unlock(void)
...
@@ -104,42 +105,6 @@ static __inline__ void icmpv6_xmit_unlock(void)
spin_unlock_bh
(
&
icmpv6_socket
->
sk
->
lock
.
slock
);
spin_unlock_bh
(
&
icmpv6_socket
->
sk
->
lock
.
slock
);
}
}
/*
* getfrag callback
*/
static
int
icmpv6_getfrag
(
const
void
*
data
,
struct
in6_addr
*
saddr
,
char
*
buff
,
unsigned
int
offset
,
unsigned
int
len
)
{
struct
icmpv6_msg
*
msg
=
(
struct
icmpv6_msg
*
)
data
;
struct
icmp6hdr
*
icmph
;
__u32
csum
;
if
(
offset
)
{
csum
=
skb_copy_and_csum_bits
(
msg
->
skb
,
msg
->
offset
+
(
offset
-
sizeof
(
struct
icmp6hdr
)),
buff
,
len
,
msg
->
csum
);
msg
->
csum
=
csum
;
return
0
;
}
csum
=
csum_partial_copy_nocheck
((
void
*
)
&
msg
->
icmph
,
buff
,
sizeof
(
struct
icmp6hdr
),
msg
->
csum
);
csum
=
skb_copy_and_csum_bits
(
msg
->
skb
,
msg
->
offset
,
buff
+
sizeof
(
struct
icmp6hdr
),
len
-
sizeof
(
struct
icmp6hdr
),
csum
);
icmph
=
(
struct
icmp6hdr
*
)
buff
;
icmph
->
icmp6_cksum
=
csum_ipv6_magic
(
saddr
,
msg
->
daddr
,
msg
->
len
,
IPPROTO_ICMPV6
,
csum
);
return
0
;
}
/*
/*
* Slightly more convenient version of icmpv6_send.
* Slightly more convenient version of icmpv6_send.
*/
*/
...
@@ -242,22 +207,74 @@ static __inline__ int opt_unrec(struct sk_buff *skb, __u32 offset)
...
@@ -242,22 +207,74 @@ static __inline__ int opt_unrec(struct sk_buff *skb, __u32 offset)
return
(
optval
&
0xC0
)
==
0x80
;
return
(
optval
&
0xC0
)
==
0x80
;
}
}
int
icmpv6_push_pending_frames
(
struct
sock
*
sk
,
struct
flowi
*
fl
,
struct
icmp6hdr
*
thdr
,
int
len
)
{
struct
sk_buff
*
skb
;
struct
icmp6hdr
*
icmp6h
;
int
err
=
0
;
if
((
skb
=
skb_peek
(
&
sk
->
write_queue
))
==
NULL
)
goto
out
;
icmp6h
=
(
struct
icmp6hdr
*
)
skb
->
h
.
raw
;
memcpy
(
icmp6h
,
thdr
,
sizeof
(
struct
icmp6hdr
));
icmp6h
->
icmp6_cksum
=
0
;
if
(
skb_queue_len
(
&
sk
->
write_queue
)
==
1
)
{
skb
->
csum
=
csum_partial
((
char
*
)
icmp6h
,
sizeof
(
struct
icmp6hdr
),
skb
->
csum
);
icmp6h
->
icmp6_cksum
=
csum_ipv6_magic
(
fl
->
fl6_src
,
fl
->
fl6_dst
,
len
,
fl
->
proto
,
skb
->
csum
);
}
else
{
u32
tmp_csum
=
0
;
skb_queue_walk
(
&
sk
->
write_queue
,
skb
)
{
tmp_csum
=
csum_add
(
tmp_csum
,
skb
->
csum
);
}
tmp_csum
=
csum_partial
((
char
*
)
icmp6h
,
sizeof
(
struct
icmp6hdr
),
tmp_csum
);
tmp_csum
=
csum_ipv6_magic
(
fl
->
fl6_src
,
fl
->
fl6_dst
,
len
,
fl
->
proto
,
tmp_csum
);
icmp6h
->
icmp6_cksum
=
tmp_csum
;
}
if
(
icmp6h
->
icmp6_cksum
==
0
)
icmp6h
->
icmp6_cksum
=
-
1
;
ip6_push_pending_frames
(
sk
);
out:
return
err
;
}
static
int
icmpv6_getfrag
(
void
*
from
,
char
*
to
,
int
offset
,
int
len
,
int
odd
,
struct
sk_buff
*
skb
)
{
struct
sk_buff
*
org_skb
=
(
struct
sk_buff
*
)
from
;
__u32
csum
=
0
;
csum
=
skb_copy_and_csum_bits
(
org_skb
,
offset
,
to
,
len
,
csum
);
skb
->
csum
=
csum_block_add
(
skb
->
csum
,
csum
,
odd
);
return
0
;
}
/*
/*
* Send an ICMP message in response to a packet in error
* Send an ICMP message in response to a packet in error
*/
*/
void
icmpv6_send
(
struct
sk_buff
*
skb
,
int
type
,
int
code
,
__u32
info
,
void
icmpv6_send
(
struct
sk_buff
*
skb
,
int
type
,
int
code
,
__u32
info
,
struct
net_device
*
dev
)
struct
net_device
*
dev
)
{
{
struct
inet6_dev
*
idev
;
struct
inet6_dev
*
idev
;
struct
ipv6hdr
*
hdr
=
skb
->
nh
.
ipv6h
;
struct
ipv6hdr
*
hdr
=
skb
->
nh
.
ipv6h
;
struct
sock
*
sk
=
icmpv6_socket
->
sk
;
struct
sock
*
sk
=
icmpv6_socket
->
sk
;
struct
in6_addr
*
saddr
=
NULL
;
struct
ipv6_pinfo
*
np
=
inet6_sk
(
sk
);
int
iif
=
0
;
struct
in6_addr
*
saddr
=
NULL
,
*
tmp_saddr
=
NULL
;
struct
icmpv6_msg
msg
;
struct
dst_entry
*
dst
;
struct
icmp6hdr
tmp_hdr
;
struct
flowi
fl
;
struct
flowi
fl
;
int
iif
=
0
;
int
addr_type
=
0
;
int
addr_type
=
0
;
int
len
;
int
len
,
plen
;
int
hlimit
=
-
1
;
int
err
=
0
;
if
((
u8
*
)
hdr
<
skb
->
head
||
(
u8
*
)(
hdr
+
1
)
>
skb
->
tail
)
if
((
u8
*
)
hdr
<
skb
->
head
||
(
u8
*
)(
hdr
+
1
)
>
skb
->
tail
)
return
;
return
;
...
@@ -328,36 +345,48 @@ void icmpv6_send(struct sk_buff *skb, int type, int code, __u32 info,
...
@@ -328,36 +345,48 @@ void icmpv6_send(struct sk_buff *skb, int type, int code, __u32 info,
if
(
!
icmpv6_xrlim_allow
(
sk
,
type
,
&
fl
))
if
(
!
icmpv6_xrlim_allow
(
sk
,
type
,
&
fl
))
goto
out
;
goto
out
;
/*
tmp_hdr
.
icmp6_type
=
type
;
* ok. kick it. checksum will be provided by the
tmp_hdr
.
icmp6_code
=
code
;
* getfrag_t callback.
tmp_hdr
.
icmp6_cksum
=
0
;
*/
tmp_hdr
.
icmp6_pointer
=
htonl
(
info
);
msg
.
icmph
.
icmp6_type
=
type
;
if
(
!
fl
.
oif
&&
ipv6_addr_is_multicast
(
fl
.
fl6_dst
))
msg
.
icmph
.
icmp6_code
=
code
;
fl
.
oif
=
np
->
mcast_oif
;
msg
.
icmph
.
icmp6_cksum
=
0
;
msg
.
icmph
.
icmp6_pointer
=
htonl
(
info
);
msg
.
skb
=
skb
;
err
=
ip6_dst_lookup
(
sk
,
&
dst
,
&
fl
,
&
tmp_saddr
);
msg
.
offset
=
skb
->
nh
.
raw
-
skb
->
data
;
if
(
err
)
goto
out
;
msg
.
csum
=
0
;
msg
.
daddr
=
&
hdr
->
saddr
;
len
=
skb
->
len
-
msg
.
offset
+
sizeof
(
struct
icmp6hdr
);
if
(
hlimit
<
0
)
{
len
=
min_t
(
unsigned
int
,
len
,
IPV6_MIN_MTU
-
sizeof
(
struct
ipv6hdr
));
if
(
ipv6_addr_is_multicast
(
fl
.
fl6_dst
))
hlimit
=
np
->
mcast_hops
;
else
hlimit
=
np
->
hop_limit
;
if
(
hlimit
<
0
)
hlimit
=
((
struct
rt6_info
*
)
dst
)
->
rt6i_hoplimit
;
}
plen
=
skb
->
nh
.
raw
-
skb
->
data
;
__skb_pull
(
skb
,
plen
);
len
=
skb
->
len
;
len
=
min_t
(
unsigned
int
,
len
,
IPV6_MIN_MTU
-
sizeof
(
struct
ipv6hdr
)
-
sizeof
(
struct
icmp6hdr
));
if
(
len
<
0
)
{
if
(
len
<
0
)
{
if
(
net_ratelimit
())
if
(
net_ratelimit
())
printk
(
KERN_DEBUG
"icmp: len problem
\n
"
);
printk
(
KERN_DEBUG
"icmp: len problem
\n
"
);
__skb_push
(
skb
,
plen
);
goto
out
;
goto
out
;
}
}
msg
.
len
=
len
;
idev
=
in6_dev_get
(
skb
->
dev
);
idev
=
in6_dev_get
(
skb
->
dev
);
ip6_build_xmit
(
sk
,
icmpv6_getfrag
,
&
msg
,
&
fl
,
len
,
NULL
,
-
1
,
err
=
ip6_append_data
(
sk
,
icmpv6_getfrag
,
skb
,
len
+
sizeof
(
struct
icmp6hdr
),
sizeof
(
struct
icmp6hdr
),
MSG_DONTWAIT
);
hlimit
,
NULL
,
&
fl
,
(
struct
rt6_info
*
)
dst
,
MSG_DONTWAIT
);
if
(
err
)
{
ip6_flush_pending_frames
(
sk
);
goto
out
;
}
err
=
icmpv6_push_pending_frames
(
sk
,
&
fl
,
&
tmp_hdr
,
len
+
sizeof
(
struct
icmp6hdr
));
__skb_push
(
skb
,
plen
);
if
(
type
>=
ICMPV6_DEST_UNREACH
&&
type
<=
ICMPV6_PARAMPROB
)
if
(
type
>=
ICMPV6_DEST_UNREACH
&&
type
<=
ICMPV6_PARAMPROB
)
ICMP6_INC_STATS_OFFSET_BH
(
idev
,
Icmp6OutDestUnreachs
,
type
-
ICMPV6_DEST_UNREACH
);
ICMP6_INC_STATS_OFFSET_BH
(
idev
,
Icmp6OutDestUnreachs
,
type
-
ICMPV6_DEST_UNREACH
);
ICMP6_INC_STATS_BH
(
idev
,
Icmp6OutMsgs
);
ICMP6_INC_STATS_BH
(
idev
,
Icmp6OutMsgs
);
...
@@ -365,6 +394,7 @@ void icmpv6_send(struct sk_buff *skb, int type, int code, __u32 info,
...
@@ -365,6 +394,7 @@ void icmpv6_send(struct sk_buff *skb, int type, int code, __u32 info,
if
(
likely
(
idev
!=
NULL
))
if
(
likely
(
idev
!=
NULL
))
in6_dev_put
(
idev
);
in6_dev_put
(
idev
);
out:
out:
if
(
tmp_saddr
)
kfree
(
tmp_saddr
);
icmpv6_xmit_unlock
();
icmpv6_xmit_unlock
();
}
}
...
@@ -372,10 +402,14 @@ static void icmpv6_echo_reply(struct sk_buff *skb)
...
@@ -372,10 +402,14 @@ static void icmpv6_echo_reply(struct sk_buff *skb)
{
{
struct
sock
*
sk
=
icmpv6_socket
->
sk
;
struct
sock
*
sk
=
icmpv6_socket
->
sk
;
struct
inet6_dev
*
idev
;
struct
inet6_dev
*
idev
;
struct
ipv6_pinfo
*
np
=
inet6_sk
(
sk
);
struct
in6_addr
*
saddr
=
NULL
,
*
tmp_saddr
=
NULL
;
struct
icmp6hdr
*
icmph
=
(
struct
icmp6hdr
*
)
skb
->
h
.
raw
;
struct
icmp6hdr
*
icmph
=
(
struct
icmp6hdr
*
)
skb
->
h
.
raw
;
struct
in6_addr
*
saddr
;
struct
icmp6hdr
tmp_hdr
;
struct
icmpv6_msg
msg
;
struct
flowi
fl
;
struct
flowi
fl
;
struct
dst_entry
*
dst
;
int
err
=
0
;
int
hlimit
=
-
1
;
saddr
=
&
skb
->
nh
.
ipv6h
->
daddr
;
saddr
=
&
skb
->
nh
.
ipv6h
->
daddr
;
...
@@ -383,39 +417,55 @@ static void icmpv6_echo_reply(struct sk_buff *skb)
...
@@ -383,39 +417,55 @@ static void icmpv6_echo_reply(struct sk_buff *skb)
ipv6_chk_acast_addr
(
0
,
saddr
))
ipv6_chk_acast_addr
(
0
,
saddr
))
saddr
=
NULL
;
saddr
=
NULL
;
msg
.
icmph
.
icmp6_type
=
ICMPV6_ECHO_REPLY
;
memcpy
(
&
tmp_hdr
,
icmph
,
sizeof
(
tmp_hdr
));
msg
.
icmph
.
icmp6_code
=
0
;
tmp_hdr
.
icmp6_type
=
ICMPV6_ECHO_REPLY
;
msg
.
icmph
.
icmp6_cksum
=
0
;
msg
.
icmph
.
icmp6_identifier
=
icmph
->
icmp6_identifier
;
msg
.
icmph
.
icmp6_sequence
=
icmph
->
icmp6_sequence
;
msg
.
skb
=
skb
;
msg
.
offset
=
0
;
msg
.
csum
=
0
;
msg
.
len
=
skb
->
len
+
sizeof
(
struct
icmp6hdr
);
msg
.
daddr
=
&
skb
->
nh
.
ipv6h
->
saddr
;
fl
.
proto
=
IPPROTO_ICMPV6
;
fl
.
proto
=
IPPROTO_ICMPV6
;
fl
.
fl6_dst
=
msg
.
d
addr
;
fl
.
fl6_dst
=
&
skb
->
nh
.
ipv6h
->
s
addr
;
fl
.
fl6_src
=
saddr
;
fl
.
fl6_src
=
saddr
;
fl
.
oif
=
skb
->
dev
->
ifindex
;
fl
.
oif
=
skb
->
dev
->
ifindex
;
fl
.
fl6_flowlabel
=
0
;
fl
.
fl6_flowlabel
=
0
;
fl
.
fl_icmp_type
=
ICMPV6_ECHO_REPLY
;
fl
.
fl_icmp_type
=
ICMPV6_ECHO_REPLY
;
fl
.
fl_icmp_code
=
0
;
fl
.
fl_icmp_code
=
0
;
icmpv6_xmit_lock
();
if
(
!
fl
.
oif
&&
ipv6_addr_is_multicast
(
fl
.
nl_u
.
ip6_u
.
daddr
))
fl
.
oif
=
np
->
mcast_oif
;
err
=
ip6_dst_lookup
(
sk
,
&
dst
,
&
fl
,
&
tmp_saddr
);
if
(
err
)
goto
out
;
if
(
hlimit
<
0
)
{
if
(
ipv6_addr_is_multicast
(
fl
.
fl6_dst
))
hlimit
=
np
->
mcast_hops
;
else
hlimit
=
np
->
hop_limit
;
if
(
hlimit
<
0
)
hlimit
=
((
struct
rt6_info
*
)
dst
)
->
rt6i_hoplimit
;
}
idev
=
in6_dev_get
(
skb
->
dev
);
idev
=
in6_dev_get
(
skb
->
dev
);
icmpv6_xmit_lock
();
err
=
ip6_append_data
(
sk
,
icmpv6_getfrag
,
skb
,
skb
->
len
+
sizeof
(
struct
icmp6hdr
),
sizeof
(
struct
icmp6hdr
),
hlimit
,
NULL
,
&
fl
,
(
struct
rt6_info
*
)
dst
,
MSG_DONTWAIT
);
if
(
err
)
{
ip6_flush_pending_frames
(
sk
);
goto
out
;
}
err
=
icmpv6_push_pending_frames
(
sk
,
&
fl
,
&
tmp_hdr
,
skb
->
len
+
sizeof
(
struct
icmp6hdr
));
ip6_build_xmit
(
sk
,
icmpv6_getfrag
,
&
msg
,
&
fl
,
msg
.
len
,
NULL
,
-
1
,
MSG_DONTWAIT
);
ICMP6_INC_STATS_BH
(
idev
,
Icmp6OutEchoReplies
);
ICMP6_INC_STATS_BH
(
idev
,
Icmp6OutEchoReplies
);
ICMP6_INC_STATS_BH
(
idev
,
Icmp6OutMsgs
);
ICMP6_INC_STATS_BH
(
idev
,
Icmp6OutMsgs
);
icmpv6_xmit_unlock
();
if
(
likely
(
idev
!=
NULL
))
if
(
likely
(
idev
!=
NULL
))
in6_dev_put
(
idev
);
in6_dev_put
(
idev
);
out:
if
(
tmp_saddr
)
kfree
(
tmp_saddr
);
icmpv6_xmit_unlock
();
}
}
static
void
icmpv6_notify
(
struct
sk_buff
*
skb
,
int
type
,
int
code
,
u32
info
)
static
void
icmpv6_notify
(
struct
sk_buff
*
skb
,
int
type
,
int
code
,
u32
info
)
...
...
net/ipv6/ip6_output.c
View file @
6bac9098
...
@@ -23,6 +23,9 @@
...
@@ -23,6 +23,9 @@
*
*
* H. von Brand : Added missing #include <linux/string.h>
* H. von Brand : Added missing #include <linux/string.h>
* Imran Patel : frag id should be in NBO
* Imran Patel : frag id should be in NBO
* Kazunori MIYAZAWA @USAGI
* : add ip6_append_data and related functions
* for datagram xmit
*/
*/
#include <linux/config.h>
#include <linux/config.h>
...
@@ -52,6 +55,8 @@
...
@@ -52,6 +55,8 @@
#include <net/icmp.h>
#include <net/icmp.h>
#include <net/xfrm.h>
#include <net/xfrm.h>
static
int
ip6_fragment
(
struct
sk_buff
*
skb
,
int
(
*
output
)(
struct
sk_buff
*
));
static
__inline__
void
ipv6_select_ident
(
struct
sk_buff
*
skb
,
struct
frag_hdr
*
fhdr
)
static
__inline__
void
ipv6_select_ident
(
struct
sk_buff
*
skb
,
struct
frag_hdr
*
fhdr
)
{
{
static
u32
ipv6_fragmentation_id
=
1
;
static
u32
ipv6_fragmentation_id
=
1
;
...
@@ -98,7 +103,7 @@ static int ip6_dev_loopback_xmit(struct sk_buff *newskb)
...
@@ -98,7 +103,7 @@ static int ip6_dev_loopback_xmit(struct sk_buff *newskb)
}
}
int
ip6_output
(
struct
sk_buff
*
skb
)
int
ip6_output
2
(
struct
sk_buff
*
skb
)
{
{
struct
dst_entry
*
dst
=
skb
->
dst
;
struct
dst_entry
*
dst
=
skb
->
dst
;
struct
net_device
*
dev
=
dst
->
dev
;
struct
net_device
*
dev
=
dst
->
dev
;
...
@@ -134,6 +139,13 @@ int ip6_output(struct sk_buff *skb)
...
@@ -134,6 +139,13 @@ int ip6_output(struct sk_buff *skb)
return
NF_HOOK
(
PF_INET6
,
NF_IP6_POST_ROUTING
,
skb
,
NULL
,
skb
->
dev
,
ip6_output_finish
);
return
NF_HOOK
(
PF_INET6
,
NF_IP6_POST_ROUTING
,
skb
,
NULL
,
skb
->
dev
,
ip6_output_finish
);
}
}
int
ip6_output
(
struct
sk_buff
*
skb
)
{
if
((
skb
->
len
>
skb
->
dst
->
dev
->
mtu
||
skb_shinfo
(
skb
)
->
frag_list
))
return
ip6_fragment
(
skb
,
ip6_output2
);
else
return
ip6_output2
(
skb
);
}
#ifdef CONFIG_NETFILTER
#ifdef CONFIG_NETFILTER
int
ip6_route_me_harder
(
struct
sk_buff
*
skb
)
int
ip6_route_me_harder
(
struct
sk_buff
*
skb
)
...
@@ -847,3 +859,658 @@ int ip6_forward(struct sk_buff *skb)
...
@@ -847,3 +859,658 @@ int ip6_forward(struct sk_buff *skb)
kfree_skb
(
skb
);
kfree_skb
(
skb
);
return
-
EINVAL
;
return
-
EINVAL
;
}
}
static
void
ip6_copy_metadata
(
struct
sk_buff
*
to
,
struct
sk_buff
*
from
)
{
to
->
pkt_type
=
from
->
pkt_type
;
to
->
priority
=
from
->
priority
;
to
->
protocol
=
from
->
protocol
;
to
->
security
=
from
->
security
;
to
->
dst
=
dst_clone
(
from
->
dst
);
to
->
dev
=
from
->
dev
;
#ifdef CONFIG_NET_SCHED
to
->
tc_index
=
from
->
tc_index
;
#endif
#ifdef CONFIG_NETFILTER
to
->
nfmark
=
from
->
nfmark
;
/* Connection association is same as pre-frag packet */
to
->
nfct
=
from
->
nfct
;
nf_conntrack_get
(
to
->
nfct
);
#if defined(CONFIG_BRIDGE) || defined(CONFIG_BRIDGE_MODULE)
to
->
nf_bridge
=
from
->
nf_bridge
;
nf_bridge_get
(
to
->
nf_bridge
);
#endif
#ifdef CONFIG_NETFILTER_DEBUG
to
->
nf_debug
=
from
->
nf_debug
;
#endif
#endif
}
static
int
ip6_found_nexthdr
(
struct
sk_buff
*
skb
,
u8
**
nexthdr
)
{
u16
offset
=
sizeof
(
struct
ipv6hdr
);
struct
ipv6_opt_hdr
*
exthdr
=
(
struct
ipv6_opt_hdr
*
)(
skb
->
nh
.
ipv6h
+
1
);
unsigned
int
packet_len
=
skb
->
tail
-
skb
->
nh
.
raw
;
int
found_rhdr
=
0
;
*
nexthdr
=
&
skb
->
nh
.
ipv6h
->
nexthdr
;
while
(
offset
+
1
<=
packet_len
)
{
switch
(
**
nexthdr
)
{
case
NEXTHDR_HOP
:
case
NEXTHDR_ROUTING
:
case
NEXTHDR_DEST
:
if
(
**
nexthdr
==
NEXTHDR_ROUTING
)
found_rhdr
=
1
;
if
(
**
nexthdr
==
NEXTHDR_DEST
&&
found_rhdr
)
return
offset
;
offset
+=
ipv6_optlen
(
exthdr
);
*
nexthdr
=
&
exthdr
->
nexthdr
;
exthdr
=
(
struct
ipv6_opt_hdr
*
)(
skb
->
nh
.
raw
+
offset
);
break
;
default
:
return
offset
;
}
}
return
offset
;
}
static
int
ip6_fragment
(
struct
sk_buff
*
skb
,
int
(
*
output
)(
struct
sk_buff
*
))
{
struct
net_device
*
dev
;
struct
rt6_info
*
rt
=
(
struct
rt6_info
*
)
skb
->
dst
;
struct
sk_buff
*
frag
;
struct
ipv6hdr
*
tmp_hdr
;
struct
frag_hdr
*
fh
;
unsigned
int
mtu
,
hlen
,
left
,
len
;
u32
frag_id
=
0
;
int
ptr
,
offset
=
0
,
err
=
0
;
u8
*
prevhdr
,
nexthdr
=
0
;
dev
=
rt
->
u
.
dst
.
dev
;
hlen
=
ip6_found_nexthdr
(
skb
,
&
prevhdr
);
nexthdr
=
*
prevhdr
;
mtu
=
dst_pmtu
(
&
rt
->
u
.
dst
)
-
hlen
-
sizeof
(
struct
frag_hdr
);
if
(
skb_shinfo
(
skb
)
->
frag_list
)
{
int
first_len
=
0
;
if
(
first_len
-
hlen
>
mtu
||
((
first_len
-
hlen
)
&
7
)
||
skb_cloned
(
skb
))
goto
slow_path
;
for
(
frag
=
skb_shinfo
(
skb
)
->
frag_list
;
frag
;
frag
=
frag
->
next
)
{
/* Correct geometry. */
if
(
frag
->
len
>
mtu
||
((
frag
->
len
&
7
)
&&
frag
->
next
)
||
skb_headroom
(
frag
)
<
hlen
)
goto
slow_path
;
/* Correct socket ownership. */
if
(
frag
->
sk
==
NULL
)
goto
slow_path
;
/* Partially cloned skb? */
if
(
skb_shared
(
frag
))
goto
slow_path
;
}
err
=
0
;
offset
=
0
;
frag
=
skb_shinfo
(
skb
)
->
frag_list
;
skb_shinfo
(
skb
)
->
frag_list
=
0
;
/* BUILD HEADER */
tmp_hdr
=
kmalloc
(
hlen
,
GFP_ATOMIC
);
if
(
!
tmp_hdr
)
{
IP6_INC_STATS
(
Ip6FragFails
);
return
-
ENOMEM
;
}
*
prevhdr
=
NEXTHDR_FRAGMENT
;
memcpy
(
tmp_hdr
,
skb
->
nh
.
raw
,
hlen
);
__skb_pull
(
skb
,
hlen
);
fh
=
(
struct
frag_hdr
*
)
__skb_push
(
skb
,
sizeof
(
struct
frag_hdr
));
skb
->
nh
.
raw
=
__skb_push
(
skb
,
hlen
);
memcpy
(
skb
->
nh
.
raw
,
tmp_hdr
,
hlen
);
ipv6_select_ident
(
skb
,
fh
);
fh
->
nexthdr
=
nexthdr
;
fh
->
reserved
=
0
;
fh
->
frag_off
=
htons
(
0x0001
);
frag_id
=
fh
->
identification
;
first_len
=
skb_pagelen
(
skb
);
skb
->
data_len
=
first_len
-
skb_headlen
(
skb
);
skb
->
len
=
first_len
;
skb
->
nh
.
ipv6h
->
payload_len
=
htons
(
first_len
-
sizeof
(
struct
ipv6hdr
));
for
(;;)
{
/* Prepare header of the next frame,
* before previous one went down. */
if
(
frag
)
{
frag
->
h
.
raw
=
frag
->
data
;
fh
=
(
struct
frag_hdr
*
)
__skb_push
(
frag
,
sizeof
(
struct
frag_hdr
));
frag
->
nh
.
raw
=
__skb_push
(
frag
,
hlen
);
memcpy
(
frag
->
nh
.
raw
,
tmp_hdr
,
hlen
);
offset
+=
skb
->
len
-
hlen
-
sizeof
(
struct
frag_hdr
);
fh
->
nexthdr
=
nexthdr
;
fh
->
reserved
=
0
;
if
(
frag
->
next
!=
NULL
)
offset
|=
0x0001
;
fh
->
frag_off
=
htons
(
offset
);
fh
->
identification
=
frag_id
;
frag
->
nh
.
ipv6h
->
payload_len
=
htons
(
frag
->
len
-
sizeof
(
struct
ipv6hdr
));
ip6_copy_metadata
(
frag
,
skb
);
}
err
=
output
(
skb
);
if
(
err
||
!
frag
)
break
;
skb
=
frag
;
frag
=
skb
->
next
;
skb
->
next
=
NULL
;
}
if
(
tmp_hdr
)
kfree
(
tmp_hdr
);
if
(
err
==
0
)
{
IP6_INC_STATS
(
Ip6FragOKs
);
return
0
;
}
while
(
frag
)
{
skb
=
frag
->
next
;
kfree_skb
(
frag
);
frag
=
skb
;
}
IP6_INC_STATS
(
Ip6FragFails
);
return
err
;
}
slow_path:
left
=
skb
->
len
-
hlen
;
/* Space per frame */
ptr
=
hlen
;
/* Where to start from */
/*
* Fragment the datagram.
*/
*
prevhdr
=
NEXTHDR_FRAGMENT
;
/*
* Keep copying data until we run out.
*/
while
(
left
>
0
)
{
len
=
left
;
/* IF: it doesn't fit, use 'mtu' - the data space left */
if
(
len
>
mtu
)
len
=
mtu
;
/* IF: we are not sending upto and including the packet end
then align the next start on an eight byte boundary */
if
(
len
<
left
)
{
len
&=
~
7
;
}
/*
* Allocate buffer.
*/
if
((
frag
=
alloc_skb
(
len
+
hlen
+
sizeof
(
struct
frag_hdr
)
+
LL_RESERVED_SPACE
(
rt
->
u
.
dst
.
dev
),
GFP_ATOMIC
))
==
NULL
)
{
NETDEBUG
(
printk
(
KERN_INFO
"IPv6: frag: no memory for new fragment!
\n
"
));
err
=
-
ENOMEM
;
goto
fail
;
}
/*
* Set up data on packet
*/
ip6_copy_metadata
(
frag
,
skb
);
skb_reserve
(
frag
,
LL_RESERVED_SPACE
(
rt
->
u
.
dst
.
dev
));
skb_put
(
frag
,
len
+
hlen
+
sizeof
(
struct
frag_hdr
));
frag
->
nh
.
raw
=
frag
->
data
;
fh
=
(
struct
frag_hdr
*
)(
frag
->
data
+
hlen
);
frag
->
h
.
raw
=
frag
->
data
+
hlen
+
sizeof
(
struct
frag_hdr
);
/*
* Charge the memory for the fragment to any owner
* it might possess
*/
if
(
skb
->
sk
)
skb_set_owner_w
(
frag
,
skb
->
sk
);
/*
* Copy the packet header into the new buffer.
*/
memcpy
(
frag
->
nh
.
raw
,
skb
->
data
,
hlen
);
/*
* Build fragment header.
*/
fh
->
nexthdr
=
nexthdr
;
fh
->
reserved
=
0
;
if
(
frag_id
)
{
ipv6_select_ident
(
skb
,
fh
);
frag_id
=
fh
->
identification
;
}
else
fh
->
identification
=
frag_id
;
/*
* Copy a block of the IP datagram.
*/
if
(
skb_copy_bits
(
skb
,
ptr
,
frag
->
h
.
raw
,
len
))
BUG
();
left
-=
len
;
fh
->
frag_off
=
htons
(
left
>
0
?
(
offset
|
0x0001
)
:
offset
);
frag
->
nh
.
ipv6h
->
payload_len
=
htons
(
frag
->
len
-
sizeof
(
struct
ipv6hdr
));
ptr
+=
len
;
offset
+=
len
;
/*
* Put this fragment into the sending queue.
*/
IP6_INC_STATS
(
Ip6FragCreates
);
err
=
output
(
frag
);
if
(
err
)
goto
fail
;
}
kfree_skb
(
skb
);
IP6_INC_STATS
(
Ip6FragOKs
);
return
err
;
fail:
kfree_skb
(
skb
);
IP6_INC_STATS
(
Ip6FragFails
);
return
err
;
}
int
ip6_dst_lookup
(
struct
sock
*
sk
,
struct
dst_entry
**
dst
,
struct
flowi
*
fl
,
struct
in6_addr
**
saddr
)
{
struct
ipv6_pinfo
*
np
=
inet6_sk
(
sk
);
int
err
=
0
;
*
dst
=
__sk_dst_check
(
sk
,
np
->
dst_cookie
);
if
(
*
dst
)
{
struct
rt6_info
*
rt
=
(
struct
rt6_info
*
)
*
dst
;
/* Yes, checking route validity in not connected
case is not very simple. Take into account,
that we do not support routing by source, TOS,
and MSG_DONTROUTE --ANK (980726)
1. If route was host route, check that
cached destination is current.
If it is network route, we still may
check its validity using saved pointer
to the last used address: daddr_cache.
We do not want to save whole address now,
(because main consumer of this service
is tcp, which has not this problem),
so that the last trick works only on connected
sockets.
2. oif also should be the same.
*/
if
(((
rt
->
rt6i_dst
.
plen
!=
128
||
ipv6_addr_cmp
(
fl
->
fl6_dst
,
&
rt
->
rt6i_dst
.
addr
))
&&
(
np
->
daddr_cache
==
NULL
||
ipv6_addr_cmp
(
fl
->
fl6_dst
,
np
->
daddr_cache
)))
||
(
fl
->
oif
&&
fl
->
oif
!=
(
*
dst
)
->
dev
->
ifindex
))
{
*
dst
=
NULL
;
}
else
dst_hold
(
*
dst
);
}
if
(
*
dst
==
NULL
)
*
dst
=
ip6_route_output
(
sk
,
fl
);
if
((
*
dst
)
->
error
)
{
IP6_INC_STATS
(
Ip6OutNoRoutes
);
dst_release
(
*
dst
);
return
-
ENETUNREACH
;
}
if
(
fl
->
fl6_src
==
NULL
)
{
*
saddr
=
kmalloc
(
sizeof
(
struct
in6_addr
),
GFP_ATOMIC
);
err
=
ipv6_get_saddr
(
*
dst
,
fl
->
fl6_dst
,
*
saddr
);
if
(
err
)
{
#if IP6_DEBUG >= 2
printk
(
KERN_DEBUG
"ip6_build_xmit: "
"no availiable source address
\n
"
);
#endif
return
err
;
}
fl
->
fl6_src
=
*
saddr
;
}
if
(
*
dst
)
{
if
((
err
=
xfrm_lookup
(
dst
,
fl
,
sk
,
0
))
<
0
)
{
dst_release
(
*
dst
);
return
-
ENETUNREACH
;
}
}
return
0
;
}
int
ip6_append_data
(
struct
sock
*
sk
,
int
getfrag
(
void
*
from
,
char
*
to
,
int
offset
,
int
len
,
int
odd
,
struct
sk_buff
*
skb
),
void
*
from
,
int
length
,
int
transhdrlen
,
int
hlimit
,
struct
ipv6_txoptions
*
opt
,
struct
flowi
*
fl
,
struct
rt6_info
*
rt
,
unsigned
int
flags
)
{
struct
inet_opt
*
inet
=
inet_sk
(
sk
);
struct
ipv6_pinfo
*
np
=
inet6_sk
(
sk
);
struct
sk_buff
*
skb
;
unsigned
int
maxfraglen
,
fragheaderlen
;
int
exthdrlen
;
int
hh_len
;
int
mtu
;
int
copy
=
0
;
int
err
;
int
offset
=
0
;
int
csummode
=
CHECKSUM_NONE
;
if
(
flags
&
MSG_PROBE
)
return
0
;
if
(
skb_queue_empty
(
&
sk
->
write_queue
))
{
/*
* setup for corking
*/
if
(
opt
)
{
if
(
np
->
cork
.
opt
==
NULL
)
np
->
cork
.
opt
=
kmalloc
(
opt
->
tot_len
,
sk
->
allocation
);
memcpy
(
np
->
cork
.
opt
,
opt
,
opt
->
tot_len
);
inet
->
cork
.
flags
|=
IPCORK_OPT
;
/* need source address above miyazawa*/
exthdrlen
+=
opt
->
opt_flen
?
opt
->
opt_flen
:
0
;
}
dst_hold
(
&
rt
->
u
.
dst
);
np
->
cork
.
rt
=
rt
;
np
->
cork
.
fl
=
fl
;
inet
->
cork
.
fragsize
=
mtu
=
dst_pmtu
(
&
rt
->
u
.
dst
);
inet
->
cork
.
length
=
0
;
inet
->
sndmsg_page
=
NULL
;
inet
->
sndmsg_off
=
0
;
if
((
exthdrlen
=
rt
->
u
.
dst
.
header_len
)
!=
0
)
{
length
+=
exthdrlen
;
transhdrlen
+=
exthdrlen
;
}
}
else
{
rt
=
np
->
cork
.
rt
;
if
(
inet
->
cork
.
flags
&
IPCORK_OPT
)
opt
=
np
->
cork
.
opt
;
transhdrlen
=
0
;
exthdrlen
=
0
;
mtu
=
inet
->
cork
.
fragsize
;
}
hh_len
=
(
rt
->
u
.
dst
.
dev
->
hard_header_len
&~
15
)
+
16
;
fragheaderlen
=
sizeof
(
struct
ipv6hdr
)
+
(
opt
?
opt
->
opt_nflen
:
0
);
maxfraglen
=
((
mtu
-
fragheaderlen
)
&
~
7
)
+
fragheaderlen
-
sizeof
(
struct
frag_hdr
);
if
(
mtu
<
65576
)
{
if
(
inet
->
cork
.
length
+
length
>
0xFFFF
-
fragheaderlen
)
{
ipv6_local_error
(
sk
,
EMSGSIZE
,
fl
,
mtu
-
exthdrlen
);
return
-
EMSGSIZE
;
}
}
inet
->
cork
.
length
+=
length
;
if
((
skb
=
skb_peek_tail
(
&
sk
->
write_queue
))
==
NULL
)
goto
alloc_new_skb
;
while
(
length
>
0
)
{
if
((
copy
=
maxfraglen
-
skb
->
len
)
<=
0
)
{
char
*
data
;
unsigned
int
datalen
;
unsigned
int
fraglen
;
unsigned
int
alloclen
;
BUG_TRAP
(
copy
==
0
);
alloc_new_skb:
datalen
=
maxfraglen
-
fragheaderlen
;
if
(
datalen
>
length
)
datalen
=
length
;
fraglen
=
datalen
+
fragheaderlen
;
if
((
flags
&
MSG_MORE
)
&&
!
(
rt
->
u
.
dst
.
dev
->
features
&
NETIF_F_SG
))
alloclen
=
maxfraglen
;
else
alloclen
=
fraglen
;
alloclen
+=
sizeof
(
struct
frag_hdr
);
if
(
transhdrlen
)
{
skb
=
sock_alloc_send_skb
(
sk
,
alloclen
+
hh_len
+
15
,
(
flags
&
MSG_DONTWAIT
),
&
err
);
}
else
{
skb
=
NULL
;
if
(
atomic_read
(
&
sk
->
wmem_alloc
)
<=
2
*
sk
->
sndbuf
)
skb
=
sock_wmalloc
(
sk
,
alloclen
+
hh_len
+
15
,
1
,
sk
->
allocation
);
if
(
unlikely
(
skb
==
NULL
))
err
=
-
ENOBUFS
;
}
if
(
skb
==
NULL
)
goto
error
;
/*
* Fill in the control structures
*/
skb
->
ip_summed
=
csummode
;
skb
->
csum
=
0
;
/* reserve 8 byte for fragmentation */
skb_reserve
(
skb
,
hh_len
+
sizeof
(
struct
frag_hdr
));
/*
* Find where to start putting bytes
*/
data
=
skb_put
(
skb
,
fraglen
);
skb
->
nh
.
raw
=
data
+
exthdrlen
;
data
+=
fragheaderlen
;
skb
->
h
.
raw
=
data
+
exthdrlen
;
copy
=
datalen
-
transhdrlen
;
if
(
copy
>
0
&&
getfrag
(
from
,
data
+
transhdrlen
,
offset
,
copy
,
0
,
skb
)
<
0
)
{
err
=
-
EFAULT
;
kfree_skb
(
skb
);
goto
error
;
}
offset
+=
copy
;
length
-=
datalen
;
transhdrlen
=
0
;
exthdrlen
=
0
;
csummode
=
CHECKSUM_NONE
;
/*
* Put the packet on the pending queue
*/
__skb_queue_tail
(
&
sk
->
write_queue
,
skb
);
continue
;
}
if
(
copy
>
length
)
copy
=
length
;
if
(
!
(
rt
->
u
.
dst
.
dev
->
features
&
NETIF_F_SG
))
{
unsigned
int
off
;
off
=
skb
->
len
;
if
(
getfrag
(
from
,
skb_put
(
skb
,
copy
),
offset
,
copy
,
off
,
skb
)
<
0
)
{
__skb_trim
(
skb
,
off
);
err
=
-
EFAULT
;
goto
error
;
}
}
else
{
int
i
=
skb_shinfo
(
skb
)
->
nr_frags
;
skb_frag_t
*
frag
=
&
skb_shinfo
(
skb
)
->
frags
[
i
-
1
];
struct
page
*
page
=
inet
->
sndmsg_page
;
int
off
=
inet
->
sndmsg_off
;
unsigned
int
left
;
if
(
page
&&
(
left
=
PAGE_SIZE
-
off
)
>
0
)
{
if
(
copy
>=
left
)
copy
=
left
;
if
(
page
!=
frag
->
page
)
{
if
(
i
==
MAX_SKB_FRAGS
)
{
err
=
-
EMSGSIZE
;
goto
error
;
}
get_page
(
page
);
skb_fill_page_desc
(
skb
,
i
,
page
,
inet
->
sndmsg_off
,
0
);
frag
=
&
skb_shinfo
(
skb
)
->
frags
[
i
];
}
}
else
if
(
i
<
MAX_SKB_FRAGS
)
{
if
(
copy
>
PAGE_SIZE
)
copy
=
PAGE_SIZE
;
page
=
alloc_pages
(
sk
->
allocation
,
0
);
if
(
page
==
NULL
)
{
err
=
-
ENOMEM
;
goto
error
;
}
inet
->
sndmsg_page
=
page
;
inet
->
sndmsg_off
=
0
;
skb_fill_page_desc
(
skb
,
i
,
page
,
0
,
0
);
frag
=
&
skb_shinfo
(
skb
)
->
frags
[
i
];
skb
->
truesize
+=
PAGE_SIZE
;
atomic_add
(
PAGE_SIZE
,
&
sk
->
wmem_alloc
);
}
else
{
err
=
-
EMSGSIZE
;
goto
error
;
}
if
(
getfrag
(
from
,
page_address
(
frag
->
page
)
+
frag
->
page_offset
+
frag
->
size
,
offset
,
copy
,
skb
->
len
,
skb
)
<
0
)
{
err
=
-
EFAULT
;
goto
error
;
}
inet
->
sndmsg_off
+=
copy
;
frag
->
size
+=
copy
;
skb
->
len
+=
copy
;
skb
->
data_len
+=
copy
;
}
offset
+=
copy
;
length
-=
copy
;
}
return
0
;
error:
inet
->
cork
.
length
-=
length
;
IP6_INC_STATS
(
Ip6OutDiscards
);
return
err
;
}
int
ip6_push_pending_frames
(
struct
sock
*
sk
)
{
struct
sk_buff
*
skb
,
*
tmp_skb
;
struct
sk_buff
**
tail_skb
;
struct
in6_addr
*
final_dst
=
NULL
;
struct
inet_opt
*
inet
=
inet_sk
(
sk
);
struct
ipv6_pinfo
*
np
=
inet6_sk
(
sk
);
struct
ipv6hdr
*
hdr
;
struct
ipv6_txoptions
*
opt
=
np
->
cork
.
opt
;
struct
rt6_info
*
rt
=
np
->
cork
.
rt
;
struct
flowi
*
fl
=
np
->
cork
.
fl
;
unsigned
char
proto
=
fl
->
proto
;
int
err
=
0
;
if
((
skb
=
__skb_dequeue
(
&
sk
->
write_queue
))
==
NULL
)
goto
out
;
tail_skb
=
&
(
skb_shinfo
(
skb
)
->
frag_list
);
/* move skb->data to ip header from ext header */
if
(
skb
->
data
<
skb
->
nh
.
raw
)
__skb_pull
(
skb
,
skb
->
nh
.
raw
-
skb
->
data
);
while
((
tmp_skb
=
__skb_dequeue
(
&
sk
->
write_queue
))
!=
NULL
)
{
__skb_pull
(
tmp_skb
,
skb
->
h
.
raw
-
skb
->
nh
.
raw
);
*
tail_skb
=
tmp_skb
;
tail_skb
=
&
(
tmp_skb
->
next
);
skb
->
len
+=
tmp_skb
->
len
;
skb
->
data_len
+=
tmp_skb
->
len
;
#if 0 /* Logically correct, but useless work, ip_fragment() will have to undo */
skb->truesize += tmp_skb->truesize;
__sock_put(tmp_skb->sk);
tmp_skb->destructor = NULL;
tmp_skb->sk = NULL;
#endif
}
final_dst
=
fl
->
fl6_dst
;
__skb_pull
(
skb
,
skb
->
h
.
raw
-
skb
->
nh
.
raw
);
if
(
opt
&&
opt
->
opt_flen
)
ipv6_push_frag_opts
(
skb
,
opt
,
&
proto
);
if
(
opt
&&
opt
->
opt_nflen
)
ipv6_push_nfrag_opts
(
skb
,
opt
,
&
proto
,
&
final_dst
);
skb
->
nh
.
ipv6h
=
hdr
=
(
struct
ipv6hdr
*
)
skb_push
(
skb
,
sizeof
(
struct
ipv6hdr
));
*
(
u32
*
)
hdr
=
fl
->
fl6_flowlabel
|
htonl
(
0x60000000
);
if
(
skb
->
len
<
65536
)
hdr
->
payload_len
=
htons
(
skb
->
len
-
sizeof
(
struct
ipv6hdr
));
else
hdr
->
payload_len
=
0
;
hdr
->
hop_limit
=
np
->
hop_limit
;
hdr
->
nexthdr
=
proto
;
memcpy
(
&
hdr
->
saddr
,
fl
->
fl6_src
,
sizeof
(
struct
in6_addr
));
memcpy
(
&
hdr
->
daddr
,
final_dst
,
sizeof
(
struct
in6_addr
));
skb
->
dst
=
dst_clone
(
&
rt
->
u
.
dst
);
err
=
NF_HOOK
(
PF_INET6
,
NF_IP6_LOCAL_OUT
,
skb
,
NULL
,
skb
->
dst
->
dev
,
dst_output
);
if
(
err
)
{
if
(
err
>
0
)
err
=
inet
->
recverr
?
net_xmit_errno
(
err
)
:
0
;
if
(
err
)
goto
error
;
}
out:
inet
->
cork
.
flags
&=
~
IPCORK_OPT
;
if
(
np
->
cork
.
opt
)
{
kfree
(
np
->
cork
.
opt
);
np
->
cork
.
opt
=
NULL
;
}
if
(
np
->
cork
.
rt
)
{
np
->
cork
.
rt
=
NULL
;
}
if
(
np
->
cork
.
fl
)
{
np
->
cork
.
fl
=
NULL
;
}
return
err
;
error:
goto
out
;
}
void
ip6_flush_pending_frames
(
struct
sock
*
sk
)
{
struct
inet_opt
*
inet
=
inet_sk
(
sk
);
struct
ipv6_pinfo
*
np
=
inet6_sk
(
sk
);
struct
sk_buff
*
skb
;
while
((
skb
=
__skb_dequeue_tail
(
&
sk
->
write_queue
))
!=
NULL
)
kfree_skb
(
skb
);
inet
->
cork
.
flags
&=
~
IPCORK_OPT
;
if
(
np
->
cork
.
opt
)
{
kfree
(
np
->
cork
.
opt
);
np
->
cork
.
opt
=
NULL
;
}
if
(
np
->
cork
.
rt
)
{
np
->
cork
.
rt
=
NULL
;
}
if
(
np
->
cork
.
fl
)
{
np
->
cork
.
fl
=
NULL
;
}
}
net/ipv6/raw.c
View file @
6bac9098
...
@@ -12,6 +12,7 @@
...
@@ -12,6 +12,7 @@
* Fixes:
* Fixes:
* Hideaki YOSHIFUJI : sin6_scope_id support
* Hideaki YOSHIFUJI : sin6_scope_id support
* YOSHIFUJI,H.@USAGI : raw checksum (RFC2292(bis) compliance)
* YOSHIFUJI,H.@USAGI : raw checksum (RFC2292(bis) compliance)
* Kazunori MIYAZAWA @USAGI: change process style to use ip6_append_data
*
*
* This program is free software; you can redistribute it and/or
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* modify it under the terms of the GNU General Public License
...
@@ -29,6 +30,8 @@
...
@@ -29,6 +30,8 @@
#include <linux/netdevice.h>
#include <linux/netdevice.h>
#include <linux/if_arp.h>
#include <linux/if_arp.h>
#include <linux/icmpv6.h>
#include <linux/icmpv6.h>
#include <linux/netfilter.h>
#include <linux/netfilter_ipv6.h>
#include <asm/uaccess.h>
#include <asm/uaccess.h>
#include <asm/ioctls.h>
#include <asm/ioctls.h>
...
@@ -438,87 +441,115 @@ static int rawv6_recvmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg
...
@@ -438,87 +441,115 @@ static int rawv6_recvmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg
goto
out_free
;
goto
out_free
;
}
}
/*
static
int
rawv6_push_pending_frames
(
struct
sock
*
sk
,
struct
flowi
*
fl
,
struct
raw6_opt
*
opt
,
int
len
)
* Sending...
{
*/
struct
sk_buff
*
skb
;
int
err
=
0
;
u16
*
csum
;
struct
rawv6_fakehdr
{
if
((
skb
=
skb_peek
(
&
sk
->
write_queue
))
==
NULL
)
struct
iovec
*
iov
;
goto
out
;
struct
sock
*
sk
;
__u32
len
;
__u32
cksum
;
__u32
proto
;
struct
in6_addr
*
daddr
;
};
static
int
rawv6_getfrag
(
const
void
*
data
,
struct
in6_addr
*
saddr
,
if
(
opt
->
offset
+
1
<
len
)
char
*
buff
,
unsigned
int
offset
,
unsigned
int
len
)
csum
=
(
u16
*
)(
skb
->
h
.
raw
+
opt
->
offset
);
{
else
{
struct
iovec
*
iov
=
(
struct
iovec
*
)
data
;
err
=
-
EINVAL
;
goto
out
;
}
if
(
skb_queue_len
(
&
sk
->
write_queue
)
==
1
)
{
/*
* Only one fragment on the socket.
*/
/* should be check HW csum miyazawa */
*
csum
=
csum_ipv6_magic
(
fl
->
fl6_src
,
fl
->
fl6_dst
,
len
,
fl
->
proto
,
skb
->
csum
);
}
else
{
u32
tmp_csum
=
0
;
skb_queue_walk
(
&
sk
->
write_queue
,
skb
)
{
tmp_csum
=
csum_add
(
tmp_csum
,
skb
->
csum
);
}
return
memcpy_fromiovecend
(
buff
,
iov
,
offset
,
len
);
tmp_csum
=
csum_ipv6_magic
(
fl
->
fl6_src
,
fl
->
fl6_dst
,
len
,
fl
->
proto
,
tmp_csum
);
*
csum
=
tmp_csum
;
}
if
(
*
csum
==
0
)
*
csum
=
-
1
;
ip6_push_pending_frames
(
sk
);
out:
return
err
;
}
}
static
int
rawv6_
frag_cksum
(
const
void
*
data
,
struct
in6_addr
*
addr
,
static
int
rawv6_
send_hdrinc
(
struct
sock
*
sk
,
void
*
from
,
int
length
,
char
*
buff
,
unsigned
int
offse
t
,
struct
flowi
*
fl
,
struct
rt6_info
*
r
t
,
unsigned
int
len
)
unsigned
int
flags
)
{
{
struct
rawv6_fakehdr
*
hdr
=
(
struct
rawv6_fakehdr
*
)
data
;
struct
inet_opt
*
inet
=
inet_sk
(
sk
);
struct
ipv6hdr
*
iph
;
struct
sk_buff
*
skb
;
unsigned
int
hh_len
;
int
err
;
if
(
csum_partial_copy_fromiovecend
(
buff
,
hdr
->
iov
,
offset
,
if
(
length
>
rt
->
u
.
dst
.
dev
->
mtu
)
{
len
,
&
hdr
->
cksum
))
ipv6_local_error
(
sk
,
EMSGSIZE
,
fl
,
rt
->
u
.
dst
.
dev
->
mtu
);
return
-
EFAULT
;
return
-
EMSGSIZE
;
}
if
(
flags
&
MSG_PROBE
)
goto
out
;
if
(
offset
==
0
)
{
hh_len
=
LL_RESERVED_SPACE
(
rt
->
u
.
dst
.
dev
);
struct
sock
*
sk
;
struct
raw6_opt
*
opt
;
struct
in6_addr
*
daddr
;
sk
=
hdr
->
sk
;
skb
=
sock_alloc_send_skb
(
sk
,
length
+
hh_len
+
15
,
opt
=
raw6_sk
(
sk
);
flags
&
MSG_DONTWAIT
,
&
err
);
if
(
skb
==
NULL
)
goto
error
;
skb_reserve
(
skb
,
hh_len
);
if
(
hdr
->
daddr
)
skb
->
priority
=
sk
->
priority
;
daddr
=
hdr
->
daddr
;
skb
->
dst
=
dst_clone
(
&
rt
->
u
.
dst
);
else
daddr
=
addr
+
1
;
hdr
->
cksum
=
csum_ipv6_magic
(
addr
,
daddr
,
hdr
->
len
,
skb
->
nh
.
ipv6h
=
iph
=
(
struct
ipv6hdr
*
)
skb_put
(
skb
,
length
);
hdr
->
proto
,
hdr
->
cksum
);
if
(
opt
->
offset
+
1
<
len
)
{
skb
->
ip_summed
=
CHECKSUM_NONE
;
__u16
*
csum
;
csum
=
(
__u16
*
)
(
buff
+
opt
->
offset
);
skb
->
h
.
raw
=
skb
->
nh
.
raw
;
if
(
*
csum
)
{
err
=
memcpy_fromiovecend
((
void
*
)
iph
,
from
,
0
,
length
);
/* in case cksum was not initialized */
if
(
err
)
__u32
sum
=
hdr
->
cksum
;
goto
error_fault
;
sum
+=
*
csum
;
*
csum
=
hdr
->
cksum
=
(
sum
+
(
sum
>>
16
));
}
else
{
*
csum
=
hdr
->
cksum
;
}
}
else
{
if
(
net_ratelimit
())
printk
(
KERN_DEBUG
"icmp: cksum offset too big
\n
"
);
return
-
EINVAL
;
}
}
return
0
;
}
err
=
NF_HOOK
(
PF_INET6
,
NF_IP6_LOCAL_OUT
,
skb
,
NULL
,
rt
->
u
.
dst
.
dev
,
dst_output
);
if
(
err
>
0
)
err
=
inet
->
recverr
?
net_xmit_errno
(
err
)
:
0
;
if
(
err
)
goto
error
;
out:
return
0
;
error_fault:
err
=
-
EFAULT
;
kfree_skb
(
skb
);
error:
IP6_INC_STATS
(
Ip6OutDiscards
);
return
err
;
}
static
int
rawv6_sendmsg
(
struct
kiocb
*
iocb
,
struct
sock
*
sk
,
struct
msghdr
*
msg
,
int
len
)
static
int
rawv6_sendmsg
(
struct
kiocb
*
iocb
,
struct
sock
*
sk
,
struct
msghdr
*
msg
,
int
len
)
{
{
struct
ipv6_txoptions
opt_space
;
struct
ipv6_txoptions
opt_space
;
struct
sockaddr_in6
*
sin6
=
(
struct
sockaddr_in6
*
)
msg
->
msg_name
;
struct
sockaddr_in6
*
sin6
=
(
struct
sockaddr_in6
*
)
msg
->
msg_name
;
struct
in6_addr
*
daddr
,
*
saddr
=
NULL
;
struct
inet_opt
*
inet
=
inet_sk
(
sk
);
struct
inet_opt
*
inet
=
inet_sk
(
sk
);
struct
ipv6_pinfo
*
np
=
inet6_sk
(
sk
);
struct
ipv6_pinfo
*
np
=
inet6_sk
(
sk
);
struct
raw6_opt
*
raw_opt
=
raw6_sk
(
sk
);
struct
ipv6_txoptions
*
opt
=
NULL
;
struct
ipv6_txoptions
*
opt
=
NULL
;
struct
ip6_flowlabel
*
flowlabel
=
NULL
;
struct
ip6_flowlabel
*
flowlabel
=
NULL
;
struct
dst_entry
*
dst
=
NULL
;
struct
flowi
fl
;
struct
flowi
fl
;
int
addr_len
=
msg
->
msg_namelen
;
int
addr_len
=
msg
->
msg_namelen
;
struct
in6_addr
*
daddr
;
struct
raw6_opt
*
raw_opt
;
int
hlimit
=
-
1
;
int
hlimit
=
-
1
;
u16
proto
;
u16
proto
;
int
err
;
int
err
;
...
@@ -552,6 +583,8 @@ static int rawv6_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg
...
@@ -552,6 +583,8 @@ static int rawv6_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg
if
(
!
proto
)
if
(
!
proto
)
proto
=
inet
->
num
;
proto
=
inet
->
num
;
else
if
(
proto
!=
inet
->
num
)
return
(
-
EINVAL
);
if
(
proto
>
255
)
if
(
proto
>
255
)
return
(
-
EINVAL
);
return
(
-
EINVAL
);
...
@@ -590,6 +623,7 @@ static int rawv6_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg
...
@@ -590,6 +623,7 @@ static int rawv6_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg
* unspecfied destination address
* unspecfied destination address
* treated as error... is this correct ?
* treated as error... is this correct ?
*/
*/
fl6_sock_release
(
flowlabel
);
return
(
-
EINVAL
);
return
(
-
EINVAL
);
}
}
...
@@ -619,8 +653,6 @@ static int rawv6_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg
...
@@ -619,8 +653,6 @@ static int rawv6_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg
if
(
flowlabel
)
if
(
flowlabel
)
opt
=
fl6_merge_options
(
&
opt_space
,
flowlabel
,
opt
);
opt
=
fl6_merge_options
(
&
opt_space
,
flowlabel
,
opt
);
raw_opt
=
raw6_sk
(
sk
);
fl
.
proto
=
proto
;
fl
.
proto
=
proto
;
fl
.
fl6_dst
=
daddr
;
fl
.
fl6_dst
=
daddr
;
if
(
fl
.
fl6_src
==
NULL
&&
!
ipv6_addr_any
(
&
np
->
saddr
))
if
(
fl
.
fl6_src
==
NULL
&&
!
ipv6_addr_any
(
&
np
->
saddr
))
...
@@ -628,30 +660,64 @@ static int rawv6_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg
...
@@ -628,30 +660,64 @@ static int rawv6_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg
fl
.
fl_icmp_type
=
0
;
fl
.
fl_icmp_type
=
0
;
fl
.
fl_icmp_code
=
0
;
fl
.
fl_icmp_code
=
0
;
if
(
raw_opt
->
checksum
)
{
/* merge ip6_build_xmit from ip6_output */
struct
rawv6_fakehdr
hdr
;
if
(
opt
&&
opt
->
srcrt
)
{
struct
rt0_hdr
*
rt0
=
(
struct
rt0_hdr
*
)
opt
->
srcrt
;
fl
.
fl6_dst
=
rt0
->
addr
;
}
if
(
!
fl
.
oif
&&
ipv6_addr_is_multicast
(
fl
.
nl_u
.
ip6_u
.
daddr
))
fl
.
oif
=
np
->
mcast_oif
;
hdr
.
iov
=
msg
->
msg_iov
;
err
=
ip6_dst_lookup
(
sk
,
&
dst
,
&
fl
,
&
saddr
);
hdr
.
sk
=
sk
;
if
(
err
)
goto
out
;
hdr
.
len
=
len
;
hdr
.
cksum
=
0
;
hdr
.
proto
=
proto
;
if
(
opt
&&
opt
->
srcrt
)
if
(
hlimit
<
0
)
{
hdr
.
daddr
=
daddr
;
if
(
ipv6_addr_is_multicast
(
fl
.
fl6_dst
))
hlimit
=
np
->
mcast_hops
;
else
else
hdr
.
daddr
=
NULL
;
hlimit
=
np
->
hop_limit
;
if
(
hlimit
<
0
)
hlimit
=
((
struct
rt6_info
*
)
dst
)
->
rt6i_hoplimit
;
}
if
(
msg
->
msg_flags
&
MSG_CONFIRM
)
goto
do_confirm
;
err
=
ip6_build_xmit
(
sk
,
rawv6_frag_cksum
,
&
hdr
,
&
fl
,
len
,
back_from_confirm:
opt
,
hlimit
,
msg
->
msg_flags
);
if
(
inet
->
hdrincl
)
{
err
=
rawv6_send_hdrinc
(
sk
,
msg
->
msg_iov
,
len
,
&
fl
,
(
struct
rt6_info
*
)
dst
,
msg
->
msg_flags
);
}
else
{
}
else
{
err
=
ip6_build_xmit
(
sk
,
rawv6_getfrag
,
msg
->
msg_iov
,
&
fl
,
len
,
lock_sock
(
sk
);
opt
,
hlimit
,
msg
->
msg_flags
);
err
=
ip6_append_data
(
sk
,
ip_generic_getfrag
,
msg
->
msg_iov
,
len
,
0
,
hlimit
,
opt
,
&
fl
,
(
struct
rt6_info
*
)
dst
,
msg
->
msg_flags
);
if
(
err
)
ip6_flush_pending_frames
(
sk
);
else
if
(
!
(
msg
->
msg_flags
&
MSG_MORE
))
{
if
(
raw_opt
->
checksum
)
{
err
=
rawv6_push_pending_frames
(
sk
,
&
fl
,
raw_opt
,
len
);
}
else
{
err
=
ip6_push_pending_frames
(
sk
);
}
}
}
}
done:
ip6_dst_store
(
sk
,
dst
,
fl
.
nl_u
.
ip6_u
.
daddr
==
&
np
->
daddr
?
&
np
->
daddr
:
NULL
);
if
(
err
>
0
)
err
=
np
->
recverr
?
net_xmit_errno
(
err
)
:
0
;
release_sock
(
sk
);
out:
fl6_sock_release
(
flowlabel
);
fl6_sock_release
(
flowlabel
);
if
(
saddr
)
kfree
(
saddr
);
return
err
<
0
?
err
:
len
;
return
err
<
0
?
err
:
len
;
do_confirm:
dst_confirm
(
dst
);
if
(
!
(
msg
->
msg_flags
&
MSG_PROBE
)
||
len
)
goto
back_from_confirm
;
err
=
0
;
goto
done
;
}
}
static
int
rawv6_seticmpfilter
(
struct
sock
*
sk
,
int
level
,
int
optname
,
static
int
rawv6_seticmpfilter
(
struct
sock
*
sk
,
int
level
,
int
optname
,
...
...
net/ipv6/udp.c
View file @
6bac9098
...
@@ -14,6 +14,7 @@
...
@@ -14,6 +14,7 @@
* YOSHIFUJI Hideaki @USAGI and: Support IPV6_V6ONLY socket option, which
* YOSHIFUJI Hideaki @USAGI and: Support IPV6_V6ONLY socket option, which
* Alexey Kuznetsov allow both IPv4 and IPv6 sockets to bind
* Alexey Kuznetsov allow both IPv4 and IPv6 sockets to bind
* a single port at the same time.
* a single port at the same time.
* Kazunori MIYAZAWA @USAGI: change process style to use ip6_append_data
*
*
* This program is free software; you can redistribute it and/or
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* modify it under the terms of the GNU General Public License
...
@@ -738,96 +739,117 @@ static int udpv6_rcv(struct sk_buff **pskb, unsigned int *nhoffp)
...
@@ -738,96 +739,117 @@ static int udpv6_rcv(struct sk_buff **pskb, unsigned int *nhoffp)
kfree_skb
(
skb
);
kfree_skb
(
skb
);
return
(
0
);
return
(
0
);
}
}
/*
/*
*
Sending
*
Throw away all pending data and cancel the corking. Socket is locked.
*/
*/
static
void
udp_v6_flush_pending_frames
(
struct
sock
*
sk
)
struct
udpv6fakehdr
{
{
struct
udphdr
uh
;
struct
udp_opt
*
up
=
udp_sk
(
sk
);
struct
iovec
*
iov
;
__u32
wcheck
;
if
(
up
->
pending
)
{
__u32
pl_len
;
up
->
pending
=
0
;
struct
in6_addr
*
daddr
;
ip6_flush_pending_frames
(
sk
);
};
}
}
/*
/*
*
with checksum
*
Sending
*/
*/
static
int
udpv6_getfrag
(
const
void
*
data
,
struct
in6_addr
*
addr
,
static
int
udp_v6_push_pending_frames
(
struct
sock
*
sk
,
struct
udp_opt
*
up
)
char
*
buff
,
unsigned
int
offset
,
unsigned
int
len
)
{
{
struct
udpv6fakehdr
*
udh
=
(
struct
udpv6fakehdr
*
)
data
;
struct
sk_buff
*
skb
;
char
*
dst
;
struct
udphdr
*
uh
;
int
final
=
0
;
struct
ipv6_pinfo
*
np
=
inet6_sk
(
sk
);
int
clen
=
len
;
struct
flowi
*
fl
=
np
->
cork
.
fl
;
int
err
=
0
;
dst
=
buff
;
if
(
offset
)
{
offset
-=
sizeof
(
struct
udphdr
);
}
else
{
dst
+=
sizeof
(
struct
udphdr
);
final
=
1
;
clen
-=
sizeof
(
struct
udphdr
);
}
if
(
csum_partial_copy_fromiovecend
(
dst
,
udh
->
iov
,
offset
,
/* Grab the skbuff where UDP header space exists. */
clen
,
&
udh
->
wcheck
)
)
if
((
skb
=
skb_peek
(
&
sk
->
write_queue
))
==
NULL
)
return
-
EFAULT
;
goto
out
;
if
(
final
)
{
/*
struct
in6_addr
*
daddr
;
* Create a UDP header
*/
uh
=
skb
->
h
.
uh
;
uh
->
source
=
fl
->
fl_ip_sport
;
uh
->
dest
=
fl
->
fl_ip_dport
;
uh
->
len
=
htons
(
up
->
len
);
uh
->
check
=
0
;
udh
->
wcheck
=
csum_partial
((
char
*
)
udh
,
sizeof
(
struct
udphdr
),
if
(
sk
->
no_check
==
UDP_CSUM_NOXMIT
)
{
udh
->
wcheck
);
skb
->
ip_summed
=
CHECKSUM_NONE
;
goto
send
;
}
if
(
udh
->
daddr
)
{
if
(
skb_queue_len
(
&
sk
->
write_queue
)
==
1
)
{
daddr
=
udh
->
daddr
;
skb
->
csum
=
csum_partial
((
char
*
)
uh
,
sizeof
(
struct
udphdr
),
skb
->
csum
);
uh
->
check
=
csum_ipv6_magic
(
fl
->
fl6_src
,
fl
->
fl6_dst
,
up
->
len
,
fl
->
proto
,
skb
->
csum
);
}
else
{
}
else
{
/*
u32
tmp_csum
=
0
;
* use packet destination address
* this should improve cache locality
skb_queue_walk
(
&
sk
->
write_queue
,
skb
)
{
*/
tmp_csum
=
csum_add
(
tmp_csum
,
skb
->
csum
);
daddr
=
addr
+
1
;
}
}
udh
->
uh
.
check
=
csum_ipv6_magic
(
addr
,
daddr
,
tmp_csum
=
csum_partial
((
char
*
)
uh
,
udh
->
pl_len
,
IPPROTO_UDP
,
sizeof
(
struct
udphdr
),
tmp_csum
);
udh
->
wcheck
);
tmp_csum
=
csum_ipv6_magic
(
fl
->
fl6_src
,
if
(
udh
->
uh
.
check
==
0
)
fl
->
fl6_dst
,
udh
->
uh
.
check
=
-
1
;
up
->
len
,
fl
->
proto
,
tmp_csum
);
uh
->
check
=
tmp_csum
;
memcpy
(
buff
,
udh
,
sizeof
(
struct
udphdr
));
}
}
return
0
;
if
(
uh
->
check
==
0
)
uh
->
check
=
-
1
;
send:
err
=
ip6_push_pending_frames
(
sk
);
out:
up
->
len
=
0
;
up
->
pending
=
0
;
return
err
;
}
}
static
int
udpv6_sendmsg
(
struct
kiocb
*
iocb
,
struct
sock
*
sk
,
struct
msghdr
*
msg
,
int
u
len
)
static
int
udpv6_sendmsg
(
struct
kiocb
*
iocb
,
struct
sock
*
sk
,
struct
msghdr
*
msg
,
int
len
)
{
{
struct
ipv6_txoptions
opt_space
;
struct
ipv6_txoptions
opt_space
;
struct
udp
v6fakehdr
udh
;
struct
udp
_opt
*
up
=
udp_sk
(
sk
)
;
struct
inet_opt
*
inet
=
inet_sk
(
sk
);
struct
inet_opt
*
inet
=
inet_sk
(
sk
);
struct
ipv6_pinfo
*
np
=
inet6_sk
(
sk
);
struct
ipv6_pinfo
*
np
=
inet6_sk
(
sk
);
struct
sockaddr_in6
*
sin6
=
(
struct
sockaddr_in6
*
)
msg
->
msg_name
;
struct
sockaddr_in6
*
sin6
=
(
struct
sockaddr_in6
*
)
msg
->
msg_name
;
struct
in6_addr
*
daddr
,
*
saddr
=
NULL
;
struct
ipv6_txoptions
*
opt
=
NULL
;
struct
ipv6_txoptions
*
opt
=
NULL
;
struct
ip6_flowlabel
*
flowlabel
=
NULL
;
struct
ip6_flowlabel
*
flowlabel
=
NULL
;
struct
flowi
fl
;
struct
flowi
fl
;
struct
dst_entry
*
dst
;
int
addr_len
=
msg
->
msg_namelen
;
int
addr_len
=
msg
->
msg_namelen
;
struct
in6_addr
*
daddr
;
int
ulen
=
len
;
int
len
=
ulen
+
sizeof
(
struct
udphdr
);
int
addr_type
;
int
addr_type
;
int
hlimit
=
-
1
;
int
hlimit
=
-
1
;
int
corkreq
=
up
->
corkflag
||
msg
->
msg_flags
&
MSG_MORE
;
int
err
;
int
err
;
/* Rough check on arithmetic overflow,
/* Rough check on arithmetic overflow,
better check is made in ip6_build_xmit
better check is made in ip6_build_xmit
*/
*/
if
(
ulen
<
0
||
u
len
>
INT_MAX
-
sizeof
(
struct
udphdr
))
if
(
len
<
0
||
len
>
INT_MAX
-
sizeof
(
struct
udphdr
))
return
-
EMSGSIZE
;
return
-
EMSGSIZE
;
if
(
up
->
pending
)
{
/*
* There are pending frames.
* The socket lock must be held while it's corked.
*/
lock_sock
(
sk
);
if
(
likely
(
up
->
pending
))
goto
do_append_data
;
release_sock
(
sk
);
}
ulen
+=
sizeof
(
struct
udphdr
);
fl
.
fl6_flowlabel
=
0
;
fl
.
fl6_flowlabel
=
0
;
fl
.
oif
=
0
;
fl
.
oif
=
0
;
...
@@ -835,7 +857,7 @@ static int udpv6_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg
...
@@ -835,7 +857,7 @@ static int udpv6_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg
if
(
sin6
->
sin6_family
==
AF_INET
)
{
if
(
sin6
->
sin6_family
==
AF_INET
)
{
if
(
__ipv6_only_sock
(
sk
))
if
(
__ipv6_only_sock
(
sk
))
return
-
ENETUNREACH
;
return
-
ENETUNREACH
;
return
udp_sendmsg
(
iocb
,
sk
,
msg
,
u
len
);
return
udp_sendmsg
(
iocb
,
sk
,
msg
,
len
);
}
}
if
(
addr_len
<
SIN6_LEN_RFC2133
)
if
(
addr_len
<
SIN6_LEN_RFC2133
)
...
@@ -847,7 +869,7 @@ static int udpv6_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg
...
@@ -847,7 +869,7 @@ static int udpv6_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg
if
(
sin6
->
sin6_port
==
0
)
if
(
sin6
->
sin6_port
==
0
)
return
-
EINVAL
;
return
-
EINVAL
;
u
dh
.
uh
.
des
t
=
sin6
->
sin6_port
;
u
p
->
dpor
t
=
sin6
->
sin6_port
;
daddr
=
&
sin6
->
sin6_addr
;
daddr
=
&
sin6
->
sin6_addr
;
if
(
np
->
sndflow
)
{
if
(
np
->
sndflow
)
{
...
@@ -873,7 +895,7 @@ static int udpv6_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg
...
@@ -873,7 +895,7 @@ static int udpv6_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg
if
(
sk
->
state
!=
TCP_ESTABLISHED
)
if
(
sk
->
state
!=
TCP_ESTABLISHED
)
return
-
ENOTCONN
;
return
-
ENOTCONN
;
u
dh
.
uh
.
des
t
=
inet
->
dport
;
u
p
->
dpor
t
=
inet
->
dport
;
daddr
=
&
np
->
daddr
;
daddr
=
&
np
->
daddr
;
fl
.
fl6_flowlabel
=
np
->
flow_label
;
fl
.
fl6_flowlabel
=
np
->
flow_label
;
}
}
...
@@ -888,15 +910,14 @@ static int udpv6_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg
...
@@ -888,15 +910,14 @@ static int udpv6_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg
sin
.
sin_family
=
AF_INET
;
sin
.
sin_family
=
AF_INET
;
sin
.
sin_addr
.
s_addr
=
daddr
->
s6_addr32
[
3
];
sin
.
sin_addr
.
s_addr
=
daddr
->
s6_addr32
[
3
];
sin
.
sin_port
=
u
dh
.
uh
.
des
t
;
sin
.
sin_port
=
u
p
->
dpor
t
;
msg
->
msg_name
=
(
struct
sockaddr
*
)(
&
sin
);
msg
->
msg_name
=
(
struct
sockaddr
*
)(
&
sin
);
msg
->
msg_namelen
=
sizeof
(
sin
);
msg
->
msg_namelen
=
sizeof
(
sin
);
fl6_sock_release
(
flowlabel
);
fl6_sock_release
(
flowlabel
);
return
udp_sendmsg
(
iocb
,
sk
,
msg
,
u
len
);
return
udp_sendmsg
(
iocb
,
sk
,
msg
,
len
);
}
}
udh
.
daddr
=
NULL
;
if
(
!
fl
.
oif
)
if
(
!
fl
.
oif
)
fl
.
oif
=
sk
->
bound_dev_if
;
fl
.
oif
=
sk
->
bound_dev_if
;
fl
.
fl6_src
=
NULL
;
fl
.
fl6_src
=
NULL
;
...
@@ -922,33 +943,172 @@ static int udpv6_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg
...
@@ -922,33 +943,172 @@ static int udpv6_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg
opt
=
np
->
opt
;
opt
=
np
->
opt
;
if
(
flowlabel
)
if
(
flowlabel
)
opt
=
fl6_merge_options
(
&
opt_space
,
flowlabel
,
opt
);
opt
=
fl6_merge_options
(
&
opt_space
,
flowlabel
,
opt
);
if
(
opt
&&
opt
->
srcrt
)
udh
.
daddr
=
daddr
;
udh
.
uh
.
source
=
inet
->
sport
;
udh
.
uh
.
len
=
len
<
0x10000
?
htons
(
len
)
:
0
;
udh
.
uh
.
check
=
0
;
udh
.
iov
=
msg
->
msg_iov
;
udh
.
wcheck
=
0
;
udh
.
pl_len
=
len
;
fl
.
proto
=
IPPROTO_UDP
;
fl
.
proto
=
IPPROTO_UDP
;
fl
.
fl6_dst
=
daddr
;
fl
.
fl6_dst
=
daddr
;
if
(
fl
.
fl6_src
==
NULL
&&
!
ipv6_addr_any
(
&
np
->
saddr
))
if
(
fl
.
fl6_src
==
NULL
&&
!
ipv6_addr_any
(
&
np
->
saddr
))
fl
.
fl6_src
=
&
np
->
saddr
;
fl
.
fl6_src
=
&
np
->
saddr
;
fl
.
fl_ip_dport
=
udh
.
uh
.
dest
;
fl
.
fl_ip_dport
=
up
->
dport
;
fl
.
fl_ip_sport
=
udh
.
uh
.
source
;
fl
.
fl_ip_sport
=
inet
->
sport
;
/* merge ip6_build_xmit from ip6_output */
if
(
opt
&&
opt
->
srcrt
)
{
struct
rt0_hdr
*
rt0
=
(
struct
rt0_hdr
*
)
opt
->
srcrt
;
fl
.
fl6_dst
=
rt0
->
addr
;
}
if
(
!
fl
.
oif
&&
ipv6_addr_is_multicast
(
fl
.
nl_u
.
ip6_u
.
daddr
))
fl
.
oif
=
np
->
mcast_oif
;
err
=
ip6_dst_lookup
(
sk
,
&
dst
,
&
fl
,
&
saddr
);
if
(
err
)
goto
out
;
if
(
hlimit
<
0
)
{
if
(
ipv6_addr_is_multicast
(
fl
.
fl6_dst
))
hlimit
=
np
->
mcast_hops
;
else
hlimit
=
np
->
hop_limit
;
if
(
hlimit
<
0
)
hlimit
=
((
struct
rt6_info
*
)
dst
)
->
rt6i_hoplimit
;
}
if
(
msg
->
msg_flags
&
MSG_CONFIRM
)
goto
do_confirm
;
back_from_confirm:
lock_sock
(
sk
);
if
(
unlikely
(
up
->
pending
))
{
/* The socket is already corked while preparing it. */
/* ... which is an evident application bug. --ANK */
release_sock
(
sk
);
NETDEBUG
(
if
(
net_ratelimit
())
printk
(
KERN_DEBUG
"udp cork app bug 2
\n
"
));
err
=
-
EINVAL
;
goto
out
;
}
err
=
ip6_build_xmit
(
sk
,
udpv6_getfrag
,
&
udh
,
&
fl
,
len
,
opt
,
hlimit
,
up
->
pending
=
1
;
msg
->
msg_flags
);
do_append_data:
up
->
len
+=
ulen
;
err
=
ip6_append_data
(
sk
,
ip_generic_getfrag
,
msg
->
msg_iov
,
ulen
,
sizeof
(
struct
udphdr
),
hlimit
,
opt
,
&
fl
,
(
struct
rt6_info
*
)
dst
,
corkreq
?
msg
->
msg_flags
|
MSG_MORE
:
msg
->
msg_flags
);
if
(
err
)
udp_v6_flush_pending_frames
(
sk
);
else
if
(
!
corkreq
)
err
=
udp_v6_push_pending_frames
(
sk
,
up
);
ip6_dst_store
(
sk
,
dst
,
fl
.
nl_u
.
ip6_u
.
daddr
==
&
np
->
daddr
?
&
np
->
daddr
:
NULL
);
if
(
err
>
0
)
err
=
np
->
recverr
?
net_xmit_errno
(
err
)
:
0
;
release_sock
(
sk
);
out:
fl6_sock_release
(
flowlabel
);
fl6_sock_release
(
flowlabel
);
if
(
saddr
)
kfree
(
saddr
);
if
(
!
err
)
{
UDP_INC_STATS_USER
(
UdpOutDatagrams
);
return
len
;
}
return
err
;
do_confirm:
dst_confirm
(
dst
);
if
(
!
(
msg
->
msg_flags
&
MSG_PROBE
)
||
len
)
goto
back_from_confirm
;
err
=
0
;
goto
out
;
}
static
int
udpv6_destroy_sock
(
struct
sock
*
sk
)
{
lock_sock
(
sk
);
udp_v6_flush_pending_frames
(
sk
);
release_sock
(
sk
);
inet6_destroy_sock
(
sk
);
return
0
;
}
/*
* Socket option code for UDP
*/
static
int
udpv6_setsockopt
(
struct
sock
*
sk
,
int
level
,
int
optname
,
char
*
optval
,
int
optlen
)
{
struct
udp_opt
*
up
=
udp_sk
(
sk
);
int
val
;
int
err
=
0
;
if
(
level
!=
SOL_UDP
)
return
ipv6_setsockopt
(
sk
,
level
,
optname
,
optval
,
optlen
);
if
(
optlen
<
sizeof
(
int
))
return
-
EINVAL
;
if
(
get_user
(
val
,
(
int
*
)
optval
))
return
-
EFAULT
;
switch
(
optname
)
{
case
UDP_CORK
:
if
(
val
!=
0
)
{
up
->
corkflag
=
1
;
}
else
{
up
->
corkflag
=
0
;
lock_sock
(
sk
);
udp_v6_push_pending_frames
(
sk
,
up
);
release_sock
(
sk
);
}
break
;
case
UDP_ENCAP
:
up
->
encap_type
=
val
;
break
;
default:
err
=
-
ENOPROTOOPT
;
break
;
};
if
(
err
<
0
)
return
err
;
return
err
;
}
static
int
udpv6_getsockopt
(
struct
sock
*
sk
,
int
level
,
int
optname
,
char
*
optval
,
int
*
optlen
)
{
struct
udp_opt
*
up
=
udp_sk
(
sk
);
int
val
,
len
;
if
(
level
!=
SOL_UDP
)
return
ipv6_getsockopt
(
sk
,
level
,
optname
,
optval
,
optlen
);
UDP6_INC_STATS_USER
(
UdpOutDatagrams
);
if
(
get_user
(
len
,
optlen
))
return
ulen
;
return
-
EFAULT
;
len
=
min_t
(
unsigned
int
,
len
,
sizeof
(
int
));
if
(
len
<
0
)
return
-
EINVAL
;
switch
(
optname
)
{
case
UDP_CORK
:
val
=
up
->
corkflag
;
break
;
case
UDP_ENCAP
:
val
=
up
->
encap_type
;
break
;
default:
return
-
ENOPROTOOPT
;
};
if
(
put_user
(
len
,
optlen
))
return
-
EFAULT
;
if
(
copy_to_user
(
optval
,
&
val
,
len
))
return
-
EFAULT
;
return
0
;
}
}
static
struct
inet6_protocol
udpv6_protocol
=
{
static
struct
inet6_protocol
udpv6_protocol
=
{
...
@@ -1038,9 +1198,9 @@ struct proto udpv6_prot = {
...
@@ -1038,9 +1198,9 @@ struct proto udpv6_prot = {
.
connect
=
udpv6_connect
,
.
connect
=
udpv6_connect
,
.
disconnect
=
udp_disconnect
,
.
disconnect
=
udp_disconnect
,
.
ioctl
=
udp_ioctl
,
.
ioctl
=
udp_ioctl
,
.
destroy
=
inet
6_destroy_sock
,
.
destroy
=
udpv
6_destroy_sock
,
.
setsockopt
=
i
pv6_setsockopt
,
.
setsockopt
=
ud
pv6_setsockopt
,
.
getsockopt
=
i
pv6_getsockopt
,
.
getsockopt
=
ud
pv6_getsockopt
,
.
sendmsg
=
udpv6_sendmsg
,
.
sendmsg
=
udpv6_sendmsg
,
.
recvmsg
=
udpv6_recvmsg
,
.
recvmsg
=
udpv6_recvmsg
,
.
backlog_rcv
=
udpv6_queue_rcv_skb
,
.
backlog_rcv
=
udpv6_queue_rcv_skb
,
...
...
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