diff options
Diffstat (limited to 'drivers/pci/hotplug/pciehp_hpc.c')
| -rw-r--r-- | drivers/pci/hotplug/pciehp_hpc.c | 121 | 
1 files changed, 79 insertions, 42 deletions
| diff --git a/drivers/pci/hotplug/pciehp_hpc.c b/drivers/pci/hotplug/pciehp_hpc.c index 08e84d61874e..b57fc6d6e28a 100644 --- a/drivers/pci/hotplug/pciehp_hpc.c +++ b/drivers/pci/hotplug/pciehp_hpc.c @@ -355,6 +355,18 @@ static int pciehp_link_enable(struct controller *ctrl)  	return __pciehp_link_set(ctrl, true);  } +int pciehp_get_raw_indicator_status(struct hotplug_slot *hotplug_slot, +				    u8 *status) +{ +	struct slot *slot = hotplug_slot->private; +	struct pci_dev *pdev = ctrl_dev(slot->ctrl); +	u16 slot_ctrl; + +	pcie_capability_read_word(pdev, PCI_EXP_SLTCTL, &slot_ctrl); +	*status = (slot_ctrl & (PCI_EXP_SLTCTL_AIC | PCI_EXP_SLTCTL_PIC)) >> 6; +	return 0; +} +  void pciehp_get_attention_status(struct slot *slot, u8 *status)  {  	struct controller *ctrl = slot->ctrl; @@ -431,6 +443,17 @@ int pciehp_query_power_fault(struct slot *slot)  	return !!(slot_status & PCI_EXP_SLTSTA_PFD);  } +int pciehp_set_raw_indicator_status(struct hotplug_slot *hotplug_slot, +				    u8 status) +{ +	struct slot *slot = hotplug_slot->private; +	struct controller *ctrl = slot->ctrl; + +	pcie_write_cmd_nowait(ctrl, status << 6, +			      PCI_EXP_SLTCTL_AIC | PCI_EXP_SLTCTL_PIC); +	return 0; +} +  void pciehp_set_attention_status(struct slot *slot, u8 value)  {  	struct controller *ctrl = slot->ctrl; @@ -535,14 +558,14 @@ void pciehp_power_off_slot(struct slot *slot)  		 PCI_EXP_SLTCTL_PWR_OFF);  } -static irqreturn_t pcie_isr(int irq, void *dev_id) +static irqreturn_t pciehp_isr(int irq, void *dev_id)  {  	struct controller *ctrl = (struct controller *)dev_id;  	struct pci_dev *pdev = ctrl_dev(ctrl);  	struct pci_bus *subordinate = pdev->subordinate;  	struct pci_dev *dev;  	struct slot *slot = ctrl->slot; -	u16 detected, intr_loc; +	u16 status, events;  	u8 present;  	bool link; @@ -550,36 +573,31 @@ static irqreturn_t pcie_isr(int irq, void *dev_id)  	if (pdev->current_state == PCI_D3cold)  		return IRQ_NONE; +	pcie_capability_read_word(pdev, PCI_EXP_SLTSTA, &status); +	if (status == (u16) ~0) { +		ctrl_info(ctrl, "%s: no response from device\n", __func__); +		return IRQ_NONE; +	} +  	/* -	 * In order to guarantee that all interrupt events are -	 * serviced, we need to re-inspect Slot Status register after -	 * clearing what is presumed to be the last pending interrupt. +	 * Slot Status contains plain status bits as well as event +	 * notification bits; right now we only want the event bits.  	 */ -	intr_loc = 0; -	do { -		pcie_capability_read_word(pdev, PCI_EXP_SLTSTA, &detected); -		if (detected == (u16) ~0) { -			ctrl_info(ctrl, "%s: no response from device\n", -				  __func__); -			return IRQ_HANDLED; -		} +	events = status & (PCI_EXP_SLTSTA_ABP | PCI_EXP_SLTSTA_PFD | +			   PCI_EXP_SLTSTA_PDC | PCI_EXP_SLTSTA_CC | +			   PCI_EXP_SLTSTA_DLLSC); +	if (!events) +		return IRQ_NONE; -		detected &= (PCI_EXP_SLTSTA_ABP | PCI_EXP_SLTSTA_PFD | -			     PCI_EXP_SLTSTA_PDC | -			     PCI_EXP_SLTSTA_CC | PCI_EXP_SLTSTA_DLLSC); -		detected &= ~intr_loc; -		intr_loc |= detected; -		if (!intr_loc) -			return IRQ_NONE; -		if (detected) -			pcie_capability_write_word(pdev, PCI_EXP_SLTSTA, -						   intr_loc); -	} while (detected); +	/* Capture link status before clearing interrupts */ +	if (events & PCI_EXP_SLTSTA_DLLSC) +		link = pciehp_check_link_active(ctrl); -	ctrl_dbg(ctrl, "pending interrupts %#06x from Slot Status\n", intr_loc); +	pcie_capability_write_word(pdev, PCI_EXP_SLTSTA, events); +	ctrl_dbg(ctrl, "pending interrupts %#06x from Slot Status\n", events);  	/* Check Command Complete Interrupt Pending */ -	if (intr_loc & PCI_EXP_SLTSTA_CC) { +	if (events & PCI_EXP_SLTSTA_CC) {  		ctrl->cmd_busy = 0;  		smp_mb();  		wake_up(&ctrl->queue); @@ -589,42 +607,38 @@ static irqreturn_t pcie_isr(int irq, void *dev_id)  		list_for_each_entry(dev, &subordinate->devices, bus_list) {  			if (dev->ignore_hotplug) {  				ctrl_dbg(ctrl, "ignoring hotplug event %#06x (%s requested no hotplug)\n", -					 intr_loc, pci_name(dev)); +					 events, pci_name(dev));  				return IRQ_HANDLED;  			}  		}  	} -	if (!(intr_loc & ~PCI_EXP_SLTSTA_CC)) -		return IRQ_HANDLED; -  	/* Check Attention Button Pressed */ -	if (intr_loc & PCI_EXP_SLTSTA_ABP) { -		ctrl_info(ctrl, "Button pressed on Slot(%s)\n", +	if (events & PCI_EXP_SLTSTA_ABP) { +		ctrl_info(ctrl, "Slot(%s): Attention button pressed\n",  			  slot_name(slot));  		pciehp_queue_interrupt_event(slot, INT_BUTTON_PRESS);  	}  	/* Check Presence Detect Changed */ -	if (intr_loc & PCI_EXP_SLTSTA_PDC) { -		pciehp_get_adapter_status(slot, &present); -		ctrl_info(ctrl, "Card %spresent on Slot(%s)\n", -			  present ? "" : "not ", slot_name(slot)); +	if (events & PCI_EXP_SLTSTA_PDC) { +		present = !!(status & PCI_EXP_SLTSTA_PDS); +		ctrl_info(ctrl, "Slot(%s): Card %spresent\n", slot_name(slot), +			  present ? "" : "not ");  		pciehp_queue_interrupt_event(slot, present ? INT_PRESENCE_ON :  					     INT_PRESENCE_OFF);  	}  	/* Check Power Fault Detected */ -	if ((intr_loc & PCI_EXP_SLTSTA_PFD) && !ctrl->power_fault_detected) { +	if ((events & PCI_EXP_SLTSTA_PFD) && !ctrl->power_fault_detected) {  		ctrl->power_fault_detected = 1; -		ctrl_err(ctrl, "Power fault on slot %s\n", slot_name(slot)); +		ctrl_err(ctrl, "Slot(%s): Power fault\n", slot_name(slot));  		pciehp_queue_interrupt_event(slot, INT_POWER_FAULT);  	} -	if (intr_loc & PCI_EXP_SLTSTA_DLLSC) { -		link = pciehp_check_link_active(ctrl); -		ctrl_info(ctrl, "slot(%s): Link %s event\n", -			  slot_name(slot), link ? "Up" : "Down"); +	if (events & PCI_EXP_SLTSTA_DLLSC) { +		ctrl_info(ctrl, "Slot(%s): Link %s\n", slot_name(slot), +			  link ? "Up" : "Down");  		pciehp_queue_interrupt_event(slot, link ? INT_LINK_UP :  					     INT_LINK_DOWN);  	} @@ -632,6 +646,25 @@ static irqreturn_t pcie_isr(int irq, void *dev_id)  	return IRQ_HANDLED;  } +static irqreturn_t pcie_isr(int irq, void *dev_id) +{ +	irqreturn_t rc, handled = IRQ_NONE; + +	/* +	 * To guarantee that all interrupt events are serviced, we need to +	 * re-inspect Slot Status register after clearing what is presumed +	 * to be the last pending interrupt. +	 */ +	do { +		rc = pciehp_isr(irq, dev_id); +		if (rc == IRQ_HANDLED) +			handled = IRQ_HANDLED; +	} while (rc == IRQ_HANDLED); + +	/* Return IRQ_HANDLED if we handled one or more events */ +	return handled; +} +  void pcie_enable_notification(struct controller *ctrl)  {  	u16 cmd, mask; @@ -804,6 +837,10 @@ struct controller *pcie_init(struct pcie_device *dev)  	}  	ctrl->pcie = dev;  	pcie_capability_read_dword(pdev, PCI_EXP_SLTCAP, &slot_cap); + +	if (pdev->hotplug_user_indicators) +		slot_cap &= ~(PCI_EXP_SLTCAP_AIP | PCI_EXP_SLTCAP_PIP); +  	ctrl->slot_cap = slot_cap;  	mutex_init(&ctrl->ctrl_lock);  	init_waitqueue_head(&ctrl->queue); |