Skip to content

Commit 15816fb

Browse files
sassmannpopcornmix
authored andcommitted
igb: fix driver reload with VF assigned to guest
commit 781798a upstream. commit fa44f2f broke reloading of igb, when VFs are assigned to a guest, in several ways. 1. on module load adapter->vf_data does not get properly allocated, resulting in a null pointer exception when accessing adapter->vf_data in igb_reset() on module reload. modprobe -r igb ; modprobe igb max_vfs=7 [ 215.215837] igb 0000:01:00.1: removed PHC on eth1 [ 216.932072] igb 0000:01:00.1: IOV Disabled [ 216.937038] igb 0000:01:00.0: removed PHC on eth0 [ 217.127032] igb 0000:01:00.0: Cannot deallocate SR-IOV virtual functions while they are assigned - VFs will not be deallocated [ 217.146178] igb: Intel(R) Gigabit Ethernet Network Driver - version 5.0.5-k [ 217.154050] igb: Copyright (c) 2007-2013 Intel Corporation. [ 217.160688] igb 0000:01:00.0: Enabling SR-IOV VFs using the module parameter is deprecated - please use the pci sysfs interface. [ 217.173703] igb 0000:01:00.0: irq 103 for MSI/MSI-X [ 217.179227] igb 0000:01:00.0: irq 104 for MSI/MSI-X [ 217.184735] igb 0000:01:00.0: irq 105 for MSI/MSI-X [ 217.220082] BUG: unable to handle kernel NULL pointer dereference at 0000000000000048 [ 217.228846] IP: [<ffffffffa007c5e5>] igb_reset+0xc5/0x4b0 [igb] [ 217.235472] PGD 3607ec067 PUD 36170b067 PMD 0 [ 217.240461] Oops: 0002 [#1] SMP [ 217.244085] Modules linked in: igb(+) igbvf mptsas mptscsih mptbase scsi_transport_sas [last unloaded: igb] [ 217.255040] CPU: 4 PID: 4833 Comm: modprobe Not tainted 3.11.0+ #46 [...] [ 217.390007] [<ffffffffa007fab2>] igb_probe+0x892/0xfd0 [igb] [ 217.396422] [<ffffffff81470b3e>] local_pci_probe+0x1e/0x40 [ 217.402641] [<ffffffff81472029>] pci_device_probe+0xf9/0x110 [...] 2. A follow up issue, pci_enable_sriov() should only be called if no VFs were still allocated on module unload. Otherwise pci_enable_sriov() gets called multiple times in a row rendering the NIC unusable until reset. 3. simply calling igb_enable_sriov() in igb_probe_vfs() is not enough as the interrupts need to be re-setup. Switching that to igb_pci_enable_sriov(). Signed-off-by: Stefan Assmann <[email protected]> Tested-by: Aaron Brown <[email protected]> Tested-by: Sibai Li <[email protected]> Signed-off-by: Jeff Kirsher <[email protected]> Signed-off-by: Jiri Slaby <[email protected]>
1 parent d7ca7e6 commit 15816fb

File tree

1 file changed

+16
-21
lines changed

1 file changed

+16
-21
lines changed

drivers/net/ethernet/intel/igb/igb_main.c

Lines changed: 16 additions & 21 deletions
Original file line numberDiff line numberDiff line change
@@ -182,6 +182,7 @@ static void igb_check_vf_rate_limit(struct igb_adapter *);
182182

183183
#ifdef CONFIG_PCI_IOV
184184
static int igb_vf_configure(struct igb_adapter *adapter, int vf);
185+
static int igb_pci_enable_sriov(struct pci_dev *dev, int num_vfs);
185186
#endif
186187

187188
#ifdef CONFIG_PM
@@ -2423,7 +2424,7 @@ static int igb_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
24232424
}
24242425

24252426
#ifdef CONFIG_PCI_IOV
2426-
static int igb_disable_sriov(struct pci_dev *pdev)
2427+
static int igb_disable_sriov(struct pci_dev *pdev)
24272428
{
24282429
struct net_device *netdev = pci_get_drvdata(pdev);
24292430
struct igb_adapter *adapter = netdev_priv(netdev);
@@ -2464,27 +2465,19 @@ static int igb_enable_sriov(struct pci_dev *pdev, int num_vfs)
24642465
int err = 0;
24652466
int i;
24662467

2467-
if (!adapter->msix_entries) {
2468+
if (!adapter->msix_entries || num_vfs > 7) {
24682469
err = -EPERM;
24692470
goto out;
24702471
}
2471-
24722472
if (!num_vfs)
24732473
goto out;
2474-
else if (old_vfs && old_vfs == num_vfs)
2475-
goto out;
2476-
else if (old_vfs && old_vfs != num_vfs)
2477-
err = igb_disable_sriov(pdev);
2478-
2479-
if (err)
2480-
goto out;
24812474

2482-
if (num_vfs > 7) {
2483-
err = -EPERM;
2484-
goto out;
2485-
}
2486-
2487-
adapter->vfs_allocated_count = num_vfs;
2475+
if (old_vfs) {
2476+
dev_info(&pdev->dev, "%d pre-allocated VFs found - override max_vfs setting of %d\n",
2477+
old_vfs, max_vfs);
2478+
adapter->vfs_allocated_count = old_vfs;
2479+
} else
2480+
adapter->vfs_allocated_count = num_vfs;
24882481

24892482
adapter->vf_data = kcalloc(adapter->vfs_allocated_count,
24902483
sizeof(struct vf_data_storage), GFP_KERNEL);
@@ -2498,10 +2491,12 @@ static int igb_enable_sriov(struct pci_dev *pdev, int num_vfs)
24982491
goto out;
24992492
}
25002493

2501-
err = pci_enable_sriov(pdev, adapter->vfs_allocated_count);
2502-
if (err)
2503-
goto err_out;
2504-
2494+
/* only call pci_enable_sriov() if no VFs are allocated already */
2495+
if (!old_vfs) {
2496+
err = pci_enable_sriov(pdev, adapter->vfs_allocated_count);
2497+
if (err)
2498+
goto err_out;
2499+
}
25052500
dev_info(&pdev->dev, "%d VFs allocated\n",
25062501
adapter->vfs_allocated_count);
25072502
for (i = 0; i < adapter->vfs_allocated_count; i++)
@@ -2617,7 +2612,7 @@ static void igb_probe_vfs(struct igb_adapter *adapter)
26172612
return;
26182613

26192614
pci_sriov_set_totalvfs(pdev, 7);
2620-
igb_enable_sriov(pdev, max_vfs);
2615+
igb_pci_enable_sriov(pdev, max_vfs);
26212616

26222617
#endif /* CONFIG_PCI_IOV */
26232618
}

0 commit comments

Comments
 (0)