Commit c9636020 authored by Long Li's avatar Long Li Committed by Thadeu Lima de Souza Cascardo

UBUNTU: SAUCE: pci-hyperv: lock pci bus on device eject

BugLink: http://bugs.launchpad.net/bugs/1665097

A PCI_EJECT message can arrive at the same time we are calling
pci_scan_child_bus in the workqueue for the previous PCI_BUS_RELATIONS
message or in create_root_hv_pci_bus(), in this case we could potentailly
modify the bus from multiple places. Properly lock the bus access.

Thanks Dexuan Cui <decui@microsoft.com> for pointing out the race
condition in create_root_hv_pci_bus().
Signed-off-by: default avatarLong Li <longli@microsoft.com>
Reported-by: default avatarXiaofeng Wang <xiaofwan@redhat.com>
Signed-off-by: default avatarJoseph Salisbury <joseph.salisbury@canonical.com>
Acked-by: default avatarTim Gardner <tim.gardner@canonical.com>
Acked-by: default avatarBrad Figg <brad.figg@canonical.com>
Signed-off-by: default avatarThadeu Lima de Souza Cascardo <cascardo@canonical.com>
parent 909a8ec8
...@@ -1201,9 +1201,11 @@ static int create_root_hv_pci_bus(struct hv_pcibus_device *hbus) ...@@ -1201,9 +1201,11 @@ static int create_root_hv_pci_bus(struct hv_pcibus_device *hbus)
hbus->pci_bus->msi = &hbus->msi_chip; hbus->pci_bus->msi = &hbus->msi_chip;
hbus->pci_bus->msi->dev = &hbus->hdev->device; hbus->pci_bus->msi->dev = &hbus->hdev->device;
pci_lock_rescan_remove();
pci_scan_child_bus(hbus->pci_bus); pci_scan_child_bus(hbus->pci_bus);
pci_bus_assign_resources(hbus->pci_bus); pci_bus_assign_resources(hbus->pci_bus);
pci_bus_add_devices(hbus->pci_bus); pci_bus_add_devices(hbus->pci_bus);
pci_unlock_rescan_remove();
hbus->state = hv_pcibus_installed; hbus->state = hv_pcibus_installed;
return 0; return 0;
} }
...@@ -1592,8 +1594,10 @@ static void hv_eject_device_work(struct work_struct *work) ...@@ -1592,8 +1594,10 @@ static void hv_eject_device_work(struct work_struct *work)
pdev = pci_get_domain_bus_and_slot(hpdev->hbus->sysdata.domain, 0, pdev = pci_get_domain_bus_and_slot(hpdev->hbus->sysdata.domain, 0,
wslot); wslot);
if (pdev) { if (pdev) {
pci_lock_rescan_remove();
pci_stop_and_remove_bus_device(pdev); pci_stop_and_remove_bus_device(pdev);
pci_dev_put(pdev); pci_dev_put(pdev);
pci_unlock_rescan_remove();
} }
memset(&ctxt, 0, sizeof(ctxt)); memset(&ctxt, 0, sizeof(ctxt));
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment