diff options
author | YOSHIFUJI Hideaki / 吉藤英明 <yoshfuji@linux-ipv6.org> | 2013-01-03 22:58:04 -0500 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2013-01-04 18:16:44 -0500 |
commit | b7dc8c3959fd43bfa0dbcf65375628c86665cb94 (patch) | |
tree | e1d73a23c35cd8de6bac74902d6d7b15be732535 /net/ipv6/ndisc.c | |
parent | ff43da86c69d76a726ffe7d1666148960dc1d108 (diff) |
ndisc: Remove unused space at tail of skb for ndisc messages. (TAKE 3)
Currently, the size of skb allocated for NDISC is MAX_HEADER +
LL_RESERVED_SPACE(dev) + packet length + dev->needed_tailroom,
but only LL_RESERVED_SPACE(dev) bytes is "reserved" for headers.
As a result, the skb looks like this (after construction of the
message):
head data tail end
+--------------------------------------------------------------+
+ | | | |
+--------------------------------------------------------------+
|<-hlen---->|<---ipv6 packet------>|<--tlen-->|<--MAX_HEADER-->|
=LL_ = dev
RESERVED_ ->needed_
SPACE(dev) tailroom
As the name implies, "MAX_HEADER" is used for headers, and should
be "reserved" in prior to packet construction. Or, if some space
is really required at the tail of ther skb, it should be
explicitly documented.
We have several option after construction of NDISC message:
Option 1:
head data tail end
+---------------------------------------------+
+ | | |
+---------------------------------------------+
|<-hlen---->|<---ipv6 packet------>|<--tlen-->|
=LL_ = dev
RESERVED_ ->needed_
SPACE(dev) tailroom
Option 2:
head data tail end
+--------------------------------------------------+
+ | | |
+--------------------------------------------------+
|<--MAX_HEADER-->|<---ipv6 packet------>|<--tlen-->|
= dev
->needed_
tailroom
Option 3:
head data tail end
+--------------------------------------------------------------+
+ | | | |
+--------------------------------------------------------------+
|<--MAX_HEADER-->|<-hlen---->|<---ipv6 packet------>|<--tlen-->|
=LL_ = dev
RESERVED_ ->needed_
SPACE(dev) tailroom
Our tunnel drivers try expanding headroom and the space for tunnel
encapsulation was not a mandatory space -- so we are not seeing
bugs here --, but just for optimization for performance critial
situations.
Since NDISC messages are not performance critical unlike TCP,
and as we know outgoing device, LL_RESERVED_SPACE(dev) should be
just enough for the device in most (if not all) cases:
LL_RESERVED_SPACE(dev) <= LL_MAX_HEADER <= MAX_HEADER
Note that LL_RESERVED_SPACE(dev) is also enough for NDISC over
SIT (e.g., ISATAP).
So, I think Option 1 is just fine here.
Signed-off-by: YOSHIFUJI Hideaki <yoshfuji@linux-ipv6.org>
Acked-by: Eric Dumazet <edumazet@google.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/ipv6/ndisc.c')
-rw-r--r-- | net/ipv6/ndisc.c | 4 |
1 files changed, 2 insertions, 2 deletions
diff --git a/net/ipv6/ndisc.c b/net/ipv6/ndisc.c index 6574175795df..4c4ccf79d2c5 100644 --- a/net/ipv6/ndisc.c +++ b/net/ipv6/ndisc.c | |||
@@ -395,7 +395,7 @@ static struct sk_buff *ndisc_build_skb(struct net_device *dev, | |||
395 | len += ndisc_opt_addr_space(dev); | 395 | len += ndisc_opt_addr_space(dev); |
396 | 396 | ||
397 | skb = sock_alloc_send_skb(sk, | 397 | skb = sock_alloc_send_skb(sk, |
398 | (MAX_HEADER + sizeof(struct ipv6hdr) + | 398 | (sizeof(struct ipv6hdr) + |
399 | len + hlen + tlen), | 399 | len + hlen + tlen), |
400 | 1, &err); | 400 | 1, &err); |
401 | if (!skb) { | 401 | if (!skb) { |
@@ -1439,7 +1439,7 @@ void ndisc_send_redirect(struct sk_buff *skb, const struct in6_addr *target) | |||
1439 | hlen = LL_RESERVED_SPACE(dev); | 1439 | hlen = LL_RESERVED_SPACE(dev); |
1440 | tlen = dev->needed_tailroom; | 1440 | tlen = dev->needed_tailroom; |
1441 | buff = sock_alloc_send_skb(sk, | 1441 | buff = sock_alloc_send_skb(sk, |
1442 | (MAX_HEADER + sizeof(struct ipv6hdr) + | 1442 | (sizeof(struct ipv6hdr) + |
1443 | len + hlen + tlen), | 1443 | len + hlen + tlen), |
1444 | 1, &err); | 1444 | 1, &err); |
1445 | if (buff == NULL) { | 1445 | if (buff == NULL) { |