diff options
author | Nicolas Dichtel <nicolas.dichtel@6wind.com> | 2013-03-22 02:28:43 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2013-03-24 17:16:29 -0400 |
commit | 63998ac24f8370caf99e433483532bab8368eb7e (patch) | |
tree | 6128d2e8e783079c404f2f7c5c64616c7db1b507 /net/ipv6/addrconf.c | |
parent | 0465277f6b3fd0535428ae935644ac30ce903de0 (diff) |
ipv6: provide addr and netconf dump consistency info
This patch adds a dev_addr_genid for IPv6. The goal is to use it, combined with
dev_base_seq to check if a change occurs during a netlink dump.
If a change is detected, the flag NLM_F_DUMP_INTR is set in the first message
after the dump was interrupted.
Note that only dump of unicast addresses is checked (multicast and anycast are
not checked).
Reported-by: Junwei Zhang <junwei.zhang@6wind.com>
Reported-by: Hongjun Li <hongjun.li@6wind.com>
Signed-off-by: Nicolas Dichtel <nicolas.dichtel@6wind.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/ipv6/addrconf.c')
-rw-r--r-- | net/ipv6/addrconf.c | 8 |
1 files changed, 8 insertions, 0 deletions
diff --git a/net/ipv6/addrconf.c b/net/ipv6/addrconf.c index 15794fdaf028..1b9f4f25212e 100644 --- a/net/ipv6/addrconf.c +++ b/net/ipv6/addrconf.c | |||
@@ -621,6 +621,8 @@ static int inet6_netconf_dump_devconf(struct sk_buff *skb, | |||
621 | idx = 0; | 621 | idx = 0; |
622 | head = &net->dev_index_head[h]; | 622 | head = &net->dev_index_head[h]; |
623 | rcu_read_lock(); | 623 | rcu_read_lock(); |
624 | cb->seq = atomic_read(&net->ipv6.dev_addr_genid) ^ | ||
625 | net->dev_base_seq; | ||
624 | hlist_for_each_entry_rcu(dev, head, index_hlist) { | 626 | hlist_for_each_entry_rcu(dev, head, index_hlist) { |
625 | if (idx < s_idx) | 627 | if (idx < s_idx) |
626 | goto cont; | 628 | goto cont; |
@@ -638,6 +640,7 @@ static int inet6_netconf_dump_devconf(struct sk_buff *skb, | |||
638 | rcu_read_unlock(); | 640 | rcu_read_unlock(); |
639 | goto done; | 641 | goto done; |
640 | } | 642 | } |
643 | nl_dump_check_consistent(cb, nlmsg_hdr(skb)); | ||
641 | cont: | 644 | cont: |
642 | idx++; | 645 | idx++; |
643 | } | 646 | } |
@@ -3874,6 +3877,7 @@ static int in6_dump_addrs(struct inet6_dev *idev, struct sk_buff *skb, | |||
3874 | NLM_F_MULTI); | 3877 | NLM_F_MULTI); |
3875 | if (err <= 0) | 3878 | if (err <= 0) |
3876 | break; | 3879 | break; |
3880 | nl_dump_check_consistent(cb, nlmsg_hdr(skb)); | ||
3877 | } | 3881 | } |
3878 | break; | 3882 | break; |
3879 | } | 3883 | } |
@@ -3931,6 +3935,7 @@ static int inet6_dump_addr(struct sk_buff *skb, struct netlink_callback *cb, | |||
3931 | s_ip_idx = ip_idx = cb->args[2]; | 3935 | s_ip_idx = ip_idx = cb->args[2]; |
3932 | 3936 | ||
3933 | rcu_read_lock(); | 3937 | rcu_read_lock(); |
3938 | cb->seq = atomic_read(&net->ipv6.dev_addr_genid) ^ net->dev_base_seq; | ||
3934 | for (h = s_h; h < NETDEV_HASHENTRIES; h++, s_idx = 0) { | 3939 | for (h = s_h; h < NETDEV_HASHENTRIES; h++, s_idx = 0) { |
3935 | idx = 0; | 3940 | idx = 0; |
3936 | head = &net->dev_index_head[h]; | 3941 | head = &net->dev_index_head[h]; |
@@ -4407,6 +4412,8 @@ errout: | |||
4407 | 4412 | ||
4408 | static void __ipv6_ifa_notify(int event, struct inet6_ifaddr *ifp) | 4413 | static void __ipv6_ifa_notify(int event, struct inet6_ifaddr *ifp) |
4409 | { | 4414 | { |
4415 | struct net *net = dev_net(ifp->idev->dev); | ||
4416 | |||
4410 | inet6_ifa_notify(event ? : RTM_NEWADDR, ifp); | 4417 | inet6_ifa_notify(event ? : RTM_NEWADDR, ifp); |
4411 | 4418 | ||
4412 | switch (event) { | 4419 | switch (event) { |
@@ -4432,6 +4439,7 @@ static void __ipv6_ifa_notify(int event, struct inet6_ifaddr *ifp) | |||
4432 | dst_free(&ifp->rt->dst); | 4439 | dst_free(&ifp->rt->dst); |
4433 | break; | 4440 | break; |
4434 | } | 4441 | } |
4442 | atomic_inc(&net->ipv6.dev_addr_genid); | ||
4435 | } | 4443 | } |
4436 | 4444 | ||
4437 | static void ipv6_ifa_notify(int event, struct inet6_ifaddr *ifp) | 4445 | static void ipv6_ifa_notify(int event, struct inet6_ifaddr *ifp) |