aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/net/cnic.c
diff options
context:
space:
mode:
authorMichael Chan <mchan@broadcom.com>2010-12-23 02:42:57 -0500
committerDavid S. Miller <davem@davemloft.net>2010-12-23 14:44:28 -0500
commit8adc9240f98a816f7e9b3d93b9446a790110e062 (patch)
treeab3fbcfb3d29a9a3a1bbafd861a38f649d690e70 /drivers/net/cnic.c
parent9b09336072796378dac46df63bcd43291b24fd12 (diff)
cnic: Prevent "scheduling while atomic" when calling ->cnic_init()
cnic_dev_list is protected by rtnl_lock and cnic_dev_lock spin_lock during modifications. When looping on cnic_dev_list and calling ->cnic_init(), we should just hold rtnl_lock since ->cnic_init() may sleep. Signed-off-by: Michael Chan <mchan@broadcom.com> Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'drivers/net/cnic.c')
-rw-r--r--drivers/net/cnic.c3
1 files changed, 1 insertions, 2 deletions
diff --git a/drivers/net/cnic.c b/drivers/net/cnic.c
index 9c2e7860b18e..3a7d3ce6db7b 100644
--- a/drivers/net/cnic.c
+++ b/drivers/net/cnic.c
@@ -59,6 +59,7 @@ MODULE_DESCRIPTION("Broadcom NetXtreme II CNIC Driver");
59MODULE_LICENSE("GPL"); 59MODULE_LICENSE("GPL");
60MODULE_VERSION(CNIC_MODULE_VERSION); 60MODULE_VERSION(CNIC_MODULE_VERSION);
61 61
62/* cnic_dev_list modifications are protected by both rtnl and cnic_dev_lock */
62static LIST_HEAD(cnic_dev_list); 63static LIST_HEAD(cnic_dev_list);
63static LIST_HEAD(cnic_udev_list); 64static LIST_HEAD(cnic_udev_list);
64static DEFINE_RWLOCK(cnic_dev_lock); 65static DEFINE_RWLOCK(cnic_dev_lock);
@@ -445,14 +446,12 @@ int cnic_register_driver(int ulp_type, struct cnic_ulp_ops *ulp_ops)
445 446
446 /* Prevent race conditions with netdev_event */ 447 /* Prevent race conditions with netdev_event */
447 rtnl_lock(); 448 rtnl_lock();
448 read_lock(&cnic_dev_lock);
449 list_for_each_entry(dev, &cnic_dev_list, list) { 449 list_for_each_entry(dev, &cnic_dev_list, list) {
450 struct cnic_local *cp = dev->cnic_priv; 450 struct cnic_local *cp = dev->cnic_priv;
451 451
452 if (!test_and_set_bit(ULP_F_INIT, &cp->ulp_flags[ulp_type])) 452 if (!test_and_set_bit(ULP_F_INIT, &cp->ulp_flags[ulp_type]))
453 ulp_ops->cnic_init(dev); 453 ulp_ops->cnic_init(dev);
454 } 454 }
455 read_unlock(&cnic_dev_lock);
456 rtnl_unlock(); 455 rtnl_unlock();
457 456
458 return 0; 457 return 0;