Commit 157c1062 authored by Lukas Wunner's avatar Lukas Wunner Committed by Bjorn Helgaas

PCI: pciehp: Avoid returning prematurely from sysfs requests

A sysfs request to enable or disable a PCIe hotplug slot should not
return before it has been carried out.  That is sought to be achieved by
waiting until the controller's "pending_events" have been cleared.

However the IRQ thread pciehp_ist() clears the "pending_events" before
it acts on them.  If pciehp_sysfs_enable_slot() / _disable_slot() happen
to check the "pending_events" after they have been cleared but while
pciehp_ist() is still running, the functions may return prematurely
with an incorrect return value.

Fix by introducing an "ist_running" flag which must be false before a sysfs
request is allowed to return.

Fixes: 32a8cef2 ("PCI: pciehp: Enable/disable exclusively from IRQ thread")
Link: https://lore.kernel.org/linux-pci/1562226638-54134-1-git-send-email-wangxiongfeng2@huawei.com
Link: https://lore.kernel.org/r/4174210466e27eb7e2243dd1d801d5f75baaffd8.1565345211.git.lukas@wunner.deReported-and-tested-by: default avatarXiongfeng Wang <wangxiongfeng2@huawei.com>
Signed-off-by: default avatarLukas Wunner <lukas@wunner.de>
Signed-off-by: default avatarBjorn Helgaas <bhelgaas@google.com>
Cc: stable@vger.kernel.org # v4.19+
parent 54ecb8f7
...@@ -72,6 +72,7 @@ extern int pciehp_poll_time; ...@@ -72,6 +72,7 @@ extern int pciehp_poll_time;
* @reset_lock: prevents access to the Data Link Layer Link Active bit in the * @reset_lock: prevents access to the Data Link Layer Link Active bit in the
* Link Status register and to the Presence Detect State bit in the Slot * Link Status register and to the Presence Detect State bit in the Slot
* Status register during a slot reset which may cause them to flap * Status register during a slot reset which may cause them to flap
* @ist_running: flag to keep user request waiting while IRQ thread is running
* @request_result: result of last user request submitted to the IRQ thread * @request_result: result of last user request submitted to the IRQ thread
* @requester: wait queue to wake up on completion of user request, * @requester: wait queue to wake up on completion of user request,
* used for synchronous slot enable/disable request via sysfs * used for synchronous slot enable/disable request via sysfs
...@@ -101,6 +102,7 @@ struct controller { ...@@ -101,6 +102,7 @@ struct controller {
struct hotplug_slot hotplug_slot; /* hotplug core interface */ struct hotplug_slot hotplug_slot; /* hotplug core interface */
struct rw_semaphore reset_lock; struct rw_semaphore reset_lock;
unsigned int ist_running;
int request_result; int request_result;
wait_queue_head_t requester; wait_queue_head_t requester;
}; };
......
...@@ -375,7 +375,8 @@ int pciehp_sysfs_enable_slot(struct hotplug_slot *hotplug_slot) ...@@ -375,7 +375,8 @@ int pciehp_sysfs_enable_slot(struct hotplug_slot *hotplug_slot)
ctrl->request_result = -ENODEV; ctrl->request_result = -ENODEV;
pciehp_request(ctrl, PCI_EXP_SLTSTA_PDC); pciehp_request(ctrl, PCI_EXP_SLTSTA_PDC);
wait_event(ctrl->requester, wait_event(ctrl->requester,
!atomic_read(&ctrl->pending_events)); !atomic_read(&ctrl->pending_events) &&
!ctrl->ist_running);
return ctrl->request_result; return ctrl->request_result;
case POWERON_STATE: case POWERON_STATE:
ctrl_info(ctrl, "Slot(%s): Already in powering on state\n", ctrl_info(ctrl, "Slot(%s): Already in powering on state\n",
...@@ -408,7 +409,8 @@ int pciehp_sysfs_disable_slot(struct hotplug_slot *hotplug_slot) ...@@ -408,7 +409,8 @@ int pciehp_sysfs_disable_slot(struct hotplug_slot *hotplug_slot)
mutex_unlock(&ctrl->state_lock); mutex_unlock(&ctrl->state_lock);
pciehp_request(ctrl, DISABLE_SLOT); pciehp_request(ctrl, DISABLE_SLOT);
wait_event(ctrl->requester, wait_event(ctrl->requester,
!atomic_read(&ctrl->pending_events)); !atomic_read(&ctrl->pending_events) &&
!ctrl->ist_running);
return ctrl->request_result; return ctrl->request_result;
case POWEROFF_STATE: case POWEROFF_STATE:
ctrl_info(ctrl, "Slot(%s): Already in powering off state\n", ctrl_info(ctrl, "Slot(%s): Already in powering off state\n",
......
...@@ -583,6 +583,7 @@ static irqreturn_t pciehp_ist(int irq, void *dev_id) ...@@ -583,6 +583,7 @@ static irqreturn_t pciehp_ist(int irq, void *dev_id)
irqreturn_t ret; irqreturn_t ret;
u32 events; u32 events;
ctrl->ist_running = true;
pci_config_pm_runtime_get(pdev); pci_config_pm_runtime_get(pdev);
/* rerun pciehp_isr() if the port was inaccessible on interrupt */ /* rerun pciehp_isr() if the port was inaccessible on interrupt */
...@@ -629,6 +630,7 @@ static irqreturn_t pciehp_ist(int irq, void *dev_id) ...@@ -629,6 +630,7 @@ static irqreturn_t pciehp_ist(int irq, void *dev_id)
up_read(&ctrl->reset_lock); up_read(&ctrl->reset_lock);
pci_config_pm_runtime_put(pdev); pci_config_pm_runtime_put(pdev);
ctrl->ist_running = false;
wake_up(&ctrl->requester); wake_up(&ctrl->requester);
return IRQ_HANDLED; return IRQ_HANDLED;
} }
......
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