Commit 527a150f authored by YOSHIFUJI Hideaki / 吉藤英明's avatar YOSHIFUJI Hideaki / 吉藤英明 Committed by David S. Miller

ndisc: Defer building IPv6 header.

Build ICMPv6 message first and make buffer management easier;
we can use skb->len when filling checksum in ICMPv6 header,
and then build IP header with length field.
Signed-off-by: default avatarYOSHIFUJI Hideaki <yoshfuji@linux-ipv6.org>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent af9a9976
...@@ -387,7 +387,7 @@ static struct sk_buff *ndisc_alloc_skb(struct net_device *dev, ...@@ -387,7 +387,7 @@ static struct sk_buff *ndisc_alloc_skb(struct net_device *dev,
skb->protocol = htons(ETH_P_IPV6); skb->protocol = htons(ETH_P_IPV6);
skb->dev = dev; skb->dev = dev;
skb_reserve(skb, hlen); skb_reserve(skb, hlen + sizeof(struct ipv6hdr));
return skb; return skb;
} }
...@@ -399,8 +399,8 @@ static void ip6_nd_hdr(struct sk_buff *skb, ...@@ -399,8 +399,8 @@ static void ip6_nd_hdr(struct sk_buff *skb,
{ {
struct ipv6hdr *hdr; struct ipv6hdr *hdr;
skb_push(skb, sizeof(*hdr));
skb_reset_network_header(skb); skb_reset_network_header(skb);
skb_put(skb, sizeof(struct ipv6hdr));
hdr = ipv6_hdr(skb); hdr = ipv6_hdr(skb);
ip6_flow_hdr(hdr, 0, 0); ip6_flow_hdr(hdr, 0, 0);
...@@ -438,8 +438,6 @@ static struct sk_buff *ndisc_build_skb(struct net_device *dev, ...@@ -438,8 +438,6 @@ static struct sk_buff *ndisc_build_skb(struct net_device *dev,
if (!skb) if (!skb)
return NULL; return NULL;
ip6_nd_hdr(skb, saddr, daddr, inet6_sk(sk)->hop_limit, len);
skb->transport_header = skb->tail; skb->transport_header = skb->tail;
skb_put(skb, len); skb_put(skb, len);
...@@ -455,10 +453,12 @@ static struct sk_buff *ndisc_build_skb(struct net_device *dev, ...@@ -455,10 +453,12 @@ static struct sk_buff *ndisc_build_skb(struct net_device *dev,
if (llinfo) if (llinfo)
ndisc_fill_addr_option(opt, llinfo, dev->dev_addr, dev); ndisc_fill_addr_option(opt, llinfo, dev->dev_addr, dev);
hdr->icmp6_cksum = csum_ipv6_magic(saddr, daddr, len, hdr->icmp6_cksum = csum_ipv6_magic(saddr, daddr, skb->len,
IPPROTO_ICMPV6, IPPROTO_ICMPV6,
csum_partial(hdr, csum_partial(hdr,
len, 0)); skb->len, 0));
ip6_nd_hdr(skb, saddr, daddr, inet6_sk(sk)->hop_limit, skb->len);
return skb; return skb;
} }
...@@ -1479,9 +1479,6 @@ void ndisc_send_redirect(struct sk_buff *skb, const struct in6_addr *target) ...@@ -1479,9 +1479,6 @@ void ndisc_send_redirect(struct sk_buff *skb, const struct in6_addr *target)
if (!buff) if (!buff)
goto release; goto release;
ip6_nd_hdr(buff, &saddr_buf, &ipv6_hdr(skb)->saddr,
inet6_sk(sk)->hop_limit, len);
skb_set_transport_header(buff, skb_tail_pointer(buff) - buff->data); skb_set_transport_header(buff, skb_tail_pointer(buff) - buff->data);
skb_put(buff, len); skb_put(buff, len);
msg = (struct rd_msg *)icmp6_hdr(buff); msg = (struct rd_msg *)icmp6_hdr(buff);
...@@ -1513,8 +1510,11 @@ void ndisc_send_redirect(struct sk_buff *skb, const struct in6_addr *target) ...@@ -1513,8 +1510,11 @@ void ndisc_send_redirect(struct sk_buff *skb, const struct in6_addr *target)
opt = ndisc_fill_redirect_hdr_option(opt, skb, rd_len); opt = ndisc_fill_redirect_hdr_option(opt, skb, rd_len);
msg->icmph.icmp6_cksum = csum_ipv6_magic(&saddr_buf, &ipv6_hdr(skb)->saddr, msg->icmph.icmp6_cksum = csum_ipv6_magic(&saddr_buf, &ipv6_hdr(skb)->saddr,
len, IPPROTO_ICMPV6, buff->len, IPPROTO_ICMPV6,
csum_partial(msg, len, 0)); csum_partial(msg, buff->len, 0));
ip6_nd_hdr(buff, &saddr_buf, &ipv6_hdr(skb)->saddr,
inet6_sk(sk)->hop_limit, buff->len);
skb_dst_set(buff, dst); skb_dst_set(buff, dst);
rcu_read_lock(); rcu_read_lock();
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment