aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorEric Dumazet <eric.dumazet@gmail.com>2009-11-18 17:24:34 -0500
committerRoland Dreier <rolandd@cisco.com>2009-11-18 17:24:34 -0500
commit0f9ea5d2ab5cef732d5abbe62b9e9af3007bae81 (patch)
treef5d3f3583c991b6f162cd91f9bffdc0937b12720
parenta7ca1f00ed2921b804d7ebda0f6fca8c9078fa42 (diff)
RDMA/addr: Use appropriate locking with for_each_netdev()
for_each_netdev() should be used with RTNL or dev_base_lock held, or else we risk a crash. Signed-off-by: Eric Dumazet <eric.dumazet@gmail.com> Signed-off-by: Sean Hefty <sean.hefty@intel.com> Signed-off-by: Roland Dreier <rolandd@cisco.com>
-rw-r--r--drivers/infiniband/core/addr.c8
1 files changed, 7 insertions, 1 deletions
diff --git a/drivers/infiniband/core/addr.c b/drivers/infiniband/core/addr.c
index bd07803e9183..373f1118d57b 100644
--- a/drivers/infiniband/core/addr.c
+++ b/drivers/infiniband/core/addr.c
@@ -131,6 +131,7 @@ int rdma_translate_ip(struct sockaddr *addr, struct rdma_dev_addr *dev_addr)
131 131
132#if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE) 132#if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
133 case AF_INET6: 133 case AF_INET6:
134 read_lock(&dev_base_lock);
134 for_each_netdev(&init_net, dev) { 135 for_each_netdev(&init_net, dev) {
135 if (ipv6_chk_addr(&init_net, 136 if (ipv6_chk_addr(&init_net,
136 &((struct sockaddr_in6 *) addr)->sin6_addr, 137 &((struct sockaddr_in6 *) addr)->sin6_addr,
@@ -139,6 +140,7 @@ int rdma_translate_ip(struct sockaddr *addr, struct rdma_dev_addr *dev_addr)
139 break; 140 break;
140 } 141 }
141 } 142 }
143 read_unlock(&dev_base_lock);
142 break; 144 break;
143#endif 145#endif
144 } 146 }
@@ -391,14 +393,17 @@ static int addr_resolve_local(struct sockaddr *src_in,
391 { 393 {
392 struct in6_addr *a; 394 struct in6_addr *a;
393 395
396 read_lock(&dev_base_lock);
394 for_each_netdev(&init_net, dev) 397 for_each_netdev(&init_net, dev)
395 if (ipv6_chk_addr(&init_net, 398 if (ipv6_chk_addr(&init_net,
396 &((struct sockaddr_in6 *) dst_in)->sin6_addr, 399 &((struct sockaddr_in6 *) dst_in)->sin6_addr,
397 dev, 1)) 400 dev, 1))
398 break; 401 break;
399 402
400 if (!dev) 403 if (!dev) {
404 read_unlock(&dev_base_lock);
401 return -EADDRNOTAVAIL; 405 return -EADDRNOTAVAIL;
406 }
402 407
403 a = &((struct sockaddr_in6 *) src_in)->sin6_addr; 408 a = &((struct sockaddr_in6 *) src_in)->sin6_addr;
404 409
@@ -416,6 +421,7 @@ static int addr_resolve_local(struct sockaddr *src_in,
416 if (!ret) 421 if (!ret)
417 memcpy(addr->dst_dev_addr, dev->dev_addr, MAX_ADDR_LEN); 422 memcpy(addr->dst_dev_addr, dev->dev_addr, MAX_ADDR_LEN);
418 } 423 }
424 read_unlock(&dev_base_lock);
419 break; 425 break;
420 } 426 }
421#endif 427#endif