aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--include/net/addrconf.h10
-rw-r--r--include/net/if_inet6.h1
-rw-r--r--net/core/pktgen.c4
-rw-r--r--net/ipv6/addrconf.c46
-rw-r--r--net/ipv6/anycast.c4
-rw-r--r--net/ipv6/ipv6_syms.c1
-rw-r--r--net/sctp/ipv6.c6
7 files changed, 36 insertions, 36 deletions
diff --git a/include/net/addrconf.h b/include/net/addrconf.h
index 5fc8627435eb..aa2ed8f0a9dd 100644
--- a/include/net/addrconf.h
+++ b/include/net/addrconf.h
@@ -133,20 +133,18 @@ extern int unregister_inet6addr_notifier(struct notifier_block *nb);
133static inline struct inet6_dev * 133static inline struct inet6_dev *
134__in6_dev_get(struct net_device *dev) 134__in6_dev_get(struct net_device *dev)
135{ 135{
136 return (struct inet6_dev *)dev->ip6_ptr; 136 return rcu_dereference(dev->ip6_ptr);
137} 137}
138 138
139extern rwlock_t addrconf_lock;
140
141static inline struct inet6_dev * 139static inline struct inet6_dev *
142in6_dev_get(struct net_device *dev) 140in6_dev_get(struct net_device *dev)
143{ 141{
144 struct inet6_dev *idev = NULL; 142 struct inet6_dev *idev = NULL;
145 read_lock(&addrconf_lock); 143 rcu_read_lock();
146 idev = dev->ip6_ptr; 144 idev = __in6_dev_get(dev);
147 if (idev) 145 if (idev)
148 atomic_inc(&idev->refcnt); 146 atomic_inc(&idev->refcnt);
149 read_unlock(&addrconf_lock); 147 rcu_read_unlock();
150 return idev; 148 return idev;
151} 149}
152 150
diff --git a/include/net/if_inet6.h b/include/net/if_inet6.h
index e459e1a0ae4a..34489c13c119 100644
--- a/include/net/if_inet6.h
+++ b/include/net/if_inet6.h
@@ -189,6 +189,7 @@ struct inet6_dev
189 struct ipv6_devconf cnf; 189 struct ipv6_devconf cnf;
190 struct ipv6_devstat stats; 190 struct ipv6_devstat stats;
191 unsigned long tstamp; /* ipv6InterfaceTable update timestamp */ 191 unsigned long tstamp; /* ipv6InterfaceTable update timestamp */
192 struct rcu_head rcu;
192}; 193};
193 194
194extern struct ipv6_devconf ipv6_devconf; 195extern struct ipv6_devconf ipv6_devconf;
diff --git a/net/core/pktgen.c b/net/core/pktgen.c
index 6a7320b39ed0..72145d4a2600 100644
--- a/net/core/pktgen.c
+++ b/net/core/pktgen.c
@@ -1786,7 +1786,7 @@ static void pktgen_setup_inject(struct pktgen_dev *pkt_dev)
1786 * use ipv6_get_lladdr if/when it's get exported 1786 * use ipv6_get_lladdr if/when it's get exported
1787 */ 1787 */
1788 1788
1789 read_lock(&addrconf_lock); 1789 rcu_read_lock();
1790 if ((idev = __in6_dev_get(pkt_dev->odev)) != NULL) { 1790 if ((idev = __in6_dev_get(pkt_dev->odev)) != NULL) {
1791 struct inet6_ifaddr *ifp; 1791 struct inet6_ifaddr *ifp;
1792 1792
@@ -1805,7 +1805,7 @@ static void pktgen_setup_inject(struct pktgen_dev *pkt_dev)
1805 } 1805 }
1806 read_unlock_bh(&idev->lock); 1806 read_unlock_bh(&idev->lock);
1807 } 1807 }
1808 read_unlock(&addrconf_lock); 1808 rcu_read_unlock();
1809 if (err) 1809 if (err)
1810 printk("pktgen: ERROR: IPv6 link address not availble.\n"); 1810 printk("pktgen: ERROR: IPv6 link address not availble.\n");
1811 } 1811 }
diff --git a/net/ipv6/addrconf.c b/net/ipv6/addrconf.c
index 825a291d5aa5..c09ebb7bb98a 100644
--- a/net/ipv6/addrconf.c
+++ b/net/ipv6/addrconf.c
@@ -119,9 +119,6 @@ static int ipv6_count_addresses(struct inet6_dev *idev);
119static struct inet6_ifaddr *inet6_addr_lst[IN6_ADDR_HSIZE]; 119static struct inet6_ifaddr *inet6_addr_lst[IN6_ADDR_HSIZE];
120static DEFINE_RWLOCK(addrconf_hash_lock); 120static DEFINE_RWLOCK(addrconf_hash_lock);
121 121
122/* Protects inet6 devices */
123DEFINE_RWLOCK(addrconf_lock);
124
125static void addrconf_verify(unsigned long); 122static void addrconf_verify(unsigned long);
126 123
127static DEFINE_TIMER(addr_chk_timer, addrconf_verify, 0, 0); 124static DEFINE_TIMER(addr_chk_timer, addrconf_verify, 0, 0);
@@ -318,6 +315,12 @@ static void addrconf_mod_timer(struct inet6_ifaddr *ifp,
318 315
319/* Nobody refers to this device, we may destroy it. */ 316/* Nobody refers to this device, we may destroy it. */
320 317
318static void in6_dev_finish_destroy_rcu(struct rcu_head *head)
319{
320 struct inet6_dev *idev = container_of(head, struct inet6_dev, rcu);
321 kfree(idev);
322}
323
321void in6_dev_finish_destroy(struct inet6_dev *idev) 324void in6_dev_finish_destroy(struct inet6_dev *idev)
322{ 325{
323 struct net_device *dev = idev->dev; 326 struct net_device *dev = idev->dev;
@@ -332,7 +335,7 @@ void in6_dev_finish_destroy(struct inet6_dev *idev)
332 return; 335 return;
333 } 336 }
334 snmp6_free_dev(idev); 337 snmp6_free_dev(idev);
335 kfree(idev); 338 call_rcu(&idev->rcu, in6_dev_finish_destroy_rcu);
336} 339}
337 340
338static struct inet6_dev * ipv6_add_dev(struct net_device *dev) 341static struct inet6_dev * ipv6_add_dev(struct net_device *dev)
@@ -408,9 +411,8 @@ static struct inet6_dev * ipv6_add_dev(struct net_device *dev)
408 if (netif_carrier_ok(dev)) 411 if (netif_carrier_ok(dev))
409 ndev->if_flags |= IF_READY; 412 ndev->if_flags |= IF_READY;
410 413
411 write_lock_bh(&addrconf_lock); 414 /* protected by rtnl_lock */
412 dev->ip6_ptr = ndev; 415 rcu_assign_pointer(dev->ip6_ptr, ndev);
413 write_unlock_bh(&addrconf_lock);
414 416
415 ipv6_mc_init_dev(ndev); 417 ipv6_mc_init_dev(ndev);
416 ndev->tstamp = jiffies; 418 ndev->tstamp = jiffies;
@@ -474,7 +476,7 @@ static void addrconf_forward_change(void)
474 476
475 read_lock(&dev_base_lock); 477 read_lock(&dev_base_lock);
476 for (dev=dev_base; dev; dev=dev->next) { 478 for (dev=dev_base; dev; dev=dev->next) {
477 read_lock(&addrconf_lock); 479 rcu_read_lock();
478 idev = __in6_dev_get(dev); 480 idev = __in6_dev_get(dev);
479 if (idev) { 481 if (idev) {
480 int changed = (!idev->cnf.forwarding) ^ (!ipv6_devconf.forwarding); 482 int changed = (!idev->cnf.forwarding) ^ (!ipv6_devconf.forwarding);
@@ -482,7 +484,7 @@ static void addrconf_forward_change(void)
482 if (changed) 484 if (changed)
483 dev_forward_change(idev); 485 dev_forward_change(idev);
484 } 486 }
485 read_unlock(&addrconf_lock); 487 rcu_read_unlock();
486 } 488 }
487 read_unlock(&dev_base_lock); 489 read_unlock(&dev_base_lock);
488} 490}
@@ -543,7 +545,7 @@ ipv6_add_addr(struct inet6_dev *idev, const struct in6_addr *addr, int pfxlen,
543 int hash; 545 int hash;
544 int err = 0; 546 int err = 0;
545 547
546 read_lock_bh(&addrconf_lock); 548 rcu_read_lock_bh();
547 if (idev->dead) { 549 if (idev->dead) {
548 err = -ENODEV; /*XXX*/ 550 err = -ENODEV; /*XXX*/
549 goto out2; 551 goto out2;
@@ -612,7 +614,7 @@ ipv6_add_addr(struct inet6_dev *idev, const struct in6_addr *addr, int pfxlen,
612 in6_ifa_hold(ifa); 614 in6_ifa_hold(ifa);
613 write_unlock(&idev->lock); 615 write_unlock(&idev->lock);
614out2: 616out2:
615 read_unlock_bh(&addrconf_lock); 617 rcu_read_unlock_bh();
616 618
617 if (likely(err == 0)) 619 if (likely(err == 0))
618 atomic_notifier_call_chain(&inet6addr_chain, NETDEV_UP, ifa); 620 atomic_notifier_call_chain(&inet6addr_chain, NETDEV_UP, ifa);
@@ -915,7 +917,7 @@ int ipv6_dev_get_saddr(struct net_device *daddr_dev,
915 memset(&hiscore, 0, sizeof(hiscore)); 917 memset(&hiscore, 0, sizeof(hiscore));
916 918
917 read_lock(&dev_base_lock); 919 read_lock(&dev_base_lock);
918 read_lock(&addrconf_lock); 920 rcu_read_lock();
919 921
920 for (dev = dev_base; dev; dev=dev->next) { 922 for (dev = dev_base; dev; dev=dev->next) {
921 struct inet6_dev *idev; 923 struct inet6_dev *idev;
@@ -1127,7 +1129,7 @@ record_it:
1127 } 1129 }
1128 read_unlock_bh(&idev->lock); 1130 read_unlock_bh(&idev->lock);
1129 } 1131 }
1130 read_unlock(&addrconf_lock); 1132 rcu_read_unlock();
1131 read_unlock(&dev_base_lock); 1133 read_unlock(&dev_base_lock);
1132 1134
1133 if (!ifa_result) 1135 if (!ifa_result)
@@ -1151,7 +1153,7 @@ int ipv6_get_lladdr(struct net_device *dev, struct in6_addr *addr)
1151 struct inet6_dev *idev; 1153 struct inet6_dev *idev;
1152 int err = -EADDRNOTAVAIL; 1154 int err = -EADDRNOTAVAIL;
1153 1155
1154 read_lock(&addrconf_lock); 1156 rcu_read_lock();
1155 if ((idev = __in6_dev_get(dev)) != NULL) { 1157 if ((idev = __in6_dev_get(dev)) != NULL) {
1156 struct inet6_ifaddr *ifp; 1158 struct inet6_ifaddr *ifp;
1157 1159
@@ -1165,7 +1167,7 @@ int ipv6_get_lladdr(struct net_device *dev, struct in6_addr *addr)
1165 } 1167 }
1166 read_unlock_bh(&idev->lock); 1168 read_unlock_bh(&idev->lock);
1167 } 1169 }
1168 read_unlock(&addrconf_lock); 1170 rcu_read_unlock();
1169 return err; 1171 return err;
1170} 1172}
1171 1173
@@ -1466,7 +1468,7 @@ static void ipv6_regen_rndid(unsigned long data)
1466 struct inet6_dev *idev = (struct inet6_dev *) data; 1468 struct inet6_dev *idev = (struct inet6_dev *) data;
1467 unsigned long expires; 1469 unsigned long expires;
1468 1470
1469 read_lock_bh(&addrconf_lock); 1471 rcu_read_lock_bh();
1470 write_lock_bh(&idev->lock); 1472 write_lock_bh(&idev->lock);
1471 1473
1472 if (idev->dead) 1474 if (idev->dead)
@@ -1490,7 +1492,7 @@ static void ipv6_regen_rndid(unsigned long data)
1490 1492
1491out: 1493out:
1492 write_unlock_bh(&idev->lock); 1494 write_unlock_bh(&idev->lock);
1493 read_unlock_bh(&addrconf_lock); 1495 rcu_read_unlock_bh();
1494 in6_dev_put(idev); 1496 in6_dev_put(idev);
1495} 1497}
1496 1498
@@ -2342,10 +2344,10 @@ static int addrconf_ifdown(struct net_device *dev, int how)
2342 Do not dev_put! 2344 Do not dev_put!
2343 */ 2345 */
2344 if (how == 1) { 2346 if (how == 1) {
2345 write_lock_bh(&addrconf_lock);
2346 dev->ip6_ptr = NULL;
2347 idev->dead = 1; 2347 idev->dead = 1;
2348 write_unlock_bh(&addrconf_lock); 2348
2349 /* protected by rtnl_lock */
2350 rcu_assign_pointer(dev->ip6_ptr, NULL);
2349 2351
2350 /* Step 1.5: remove snmp6 entry */ 2352 /* Step 1.5: remove snmp6 entry */
2351 snmp6_unregister_dev(idev); 2353 snmp6_unregister_dev(idev);
@@ -3573,10 +3575,10 @@ static void __ipv6_ifa_notify(int event, struct inet6_ifaddr *ifp)
3573 3575
3574static void ipv6_ifa_notify(int event, struct inet6_ifaddr *ifp) 3576static void ipv6_ifa_notify(int event, struct inet6_ifaddr *ifp)
3575{ 3577{
3576 read_lock_bh(&addrconf_lock); 3578 rcu_read_lock_bh();
3577 if (likely(ifp->idev->dead == 0)) 3579 if (likely(ifp->idev->dead == 0))
3578 __ipv6_ifa_notify(event, ifp); 3580 __ipv6_ifa_notify(event, ifp);
3579 read_unlock_bh(&addrconf_lock); 3581 rcu_read_unlock_bh();
3580} 3582}
3581 3583
3582#ifdef CONFIG_SYSCTL 3584#ifdef CONFIG_SYSCTL
diff --git a/net/ipv6/anycast.c b/net/ipv6/anycast.c
index b80fc502ca03..a9604764e015 100644
--- a/net/ipv6/anycast.c
+++ b/net/ipv6/anycast.c
@@ -56,7 +56,7 @@ ip6_onlink(struct in6_addr *addr, struct net_device *dev)
56 int onlink; 56 int onlink;
57 57
58 onlink = 0; 58 onlink = 0;
59 read_lock(&addrconf_lock); 59 rcu_read_lock();
60 idev = __in6_dev_get(dev); 60 idev = __in6_dev_get(dev);
61 if (idev) { 61 if (idev) {
62 read_lock_bh(&idev->lock); 62 read_lock_bh(&idev->lock);
@@ -68,7 +68,7 @@ ip6_onlink(struct in6_addr *addr, struct net_device *dev)
68 } 68 }
69 read_unlock_bh(&idev->lock); 69 read_unlock_bh(&idev->lock);
70 } 70 }
71 read_unlock(&addrconf_lock); 71 rcu_read_unlock();
72 return onlink; 72 return onlink;
73} 73}
74 74
diff --git a/net/ipv6/ipv6_syms.c b/net/ipv6/ipv6_syms.c
index 7b7b90d9c3d0..0e8e0676a033 100644
--- a/net/ipv6/ipv6_syms.c
+++ b/net/ipv6/ipv6_syms.c
@@ -14,7 +14,6 @@ EXPORT_SYMBOL(ndisc_mc_map);
14EXPORT_SYMBOL(register_inet6addr_notifier); 14EXPORT_SYMBOL(register_inet6addr_notifier);
15EXPORT_SYMBOL(unregister_inet6addr_notifier); 15EXPORT_SYMBOL(unregister_inet6addr_notifier);
16EXPORT_SYMBOL(ip6_route_output); 16EXPORT_SYMBOL(ip6_route_output);
17EXPORT_SYMBOL(addrconf_lock);
18EXPORT_SYMBOL(ipv6_setsockopt); 17EXPORT_SYMBOL(ipv6_setsockopt);
19EXPORT_SYMBOL(ipv6_getsockopt); 18EXPORT_SYMBOL(ipv6_getsockopt);
20EXPORT_SYMBOL(inet6_register_protosw); 19EXPORT_SYMBOL(inet6_register_protosw);
diff --git a/net/sctp/ipv6.c b/net/sctp/ipv6.c
index fd87e3ceb56e..249e5033c1a8 100644
--- a/net/sctp/ipv6.c
+++ b/net/sctp/ipv6.c
@@ -321,9 +321,9 @@ static void sctp_v6_copy_addrlist(struct list_head *addrlist,
321 struct inet6_ifaddr *ifp; 321 struct inet6_ifaddr *ifp;
322 struct sctp_sockaddr_entry *addr; 322 struct sctp_sockaddr_entry *addr;
323 323
324 read_lock(&addrconf_lock); 324 rcu_read_lock();
325 if ((in6_dev = __in6_dev_get(dev)) == NULL) { 325 if ((in6_dev = __in6_dev_get(dev)) == NULL) {
326 read_unlock(&addrconf_lock); 326 rcu_read_unlock();
327 return; 327 return;
328 } 328 }
329 329
@@ -342,7 +342,7 @@ static void sctp_v6_copy_addrlist(struct list_head *addrlist,
342 } 342 }
343 343
344 read_unlock(&in6_dev->lock); 344 read_unlock(&in6_dev->lock);
345 read_unlock(&addrconf_lock); 345 rcu_read_unlock();
346} 346}
347 347
348/* Initialize a sockaddr_storage from in incoming skb. */ 348/* Initialize a sockaddr_storage from in incoming skb. */