diff options
| author | Ingo Molnar <mingo@kernel.org> | 2012-04-25 14:24:16 +0400 | 
|---|---|---|
| committer | Ingo Molnar <mingo@kernel.org> | 2012-04-25 14:24:16 +0400 | 
| commit | cd32b1616bc79b2f2ce1b1c6164beecfecc2259c (patch) | |
| tree | 2e91c49b5c2bd927b9b74b7414dbb6839af601e1 /drivers/pci/pci.c | |
| parent | 89b8835ec865dddd6673a8dd7003581bf2377176 (diff) | |
| parent | a720b2dd2470a52345df11dca8d6c1466599f812 (diff) | |
| download | linux-cd32b1616bc79b2f2ce1b1c6164beecfecc2259c.tar.xz | |
Merge tag 'l3-fix-for-3.5' of git://git.kernel.org/pub/scm/linux/kernel/git/bp/bp into x86/urgent
A small L3 cache index disable fix from Srivatsa Bhat which unifies the
way the code checks for already disabled indices.
( Pulling it into v3.4 despite the v3.5 tag - the fix is small and we better
  keep the same code across kernel versions for such user facing interfaces. )
Signed-off-by: Ingo Molnar <mingo@kernel.org>
Diffstat (limited to 'drivers/pci/pci.c')
| -rw-r--r-- | drivers/pci/pci.c | 57 | 
1 files changed, 39 insertions, 18 deletions
| diff --git a/drivers/pci/pci.c b/drivers/pci/pci.c index 815674415267..d20f1334792b 100644 --- a/drivers/pci/pci.c +++ b/drivers/pci/pci.c @@ -967,16 +967,47 @@ pci_save_state(struct pci_dev *dev)  	return 0;  } +static void pci_restore_config_dword(struct pci_dev *pdev, int offset, +				     u32 saved_val, int retry) +{ +	u32 val; + +	pci_read_config_dword(pdev, offset, &val); +	if (val == saved_val) +		return; + +	for (;;) { +		dev_dbg(&pdev->dev, "restoring config space at offset " +			"%#x (was %#x, writing %#x)\n", offset, val, saved_val); +		pci_write_config_dword(pdev, offset, saved_val); +		if (retry-- <= 0) +			return; + +		pci_read_config_dword(pdev, offset, &val); +		if (val == saved_val) +			return; + +		mdelay(1); +	} +} + +static void pci_restore_config_space(struct pci_dev *pdev, int start, int end, +				     int retry) +{ +	int index; + +	for (index = end; index >= start; index--) +		pci_restore_config_dword(pdev, 4 * index, +					 pdev->saved_config_space[index], +					 retry); +} +  /**    * pci_restore_state - Restore the saved state of a PCI device   * @dev: - PCI device that we're dealing with   */  void pci_restore_state(struct pci_dev *dev)  { -	int i; -	u32 val; -	int tries; -  	if (!dev->state_saved)  		return; @@ -984,24 +1015,14 @@ void pci_restore_state(struct pci_dev *dev)  	pci_restore_pcie_state(dev);  	pci_restore_ats_state(dev); +	pci_restore_config_space(dev, 10, 15, 0);  	/*  	 * The Base Address register should be programmed before the command  	 * register(s)  	 */ -	for (i = 15; i >= 0; i--) { -		pci_read_config_dword(dev, i * 4, &val); -		tries = 10;		 -		while (tries && val != dev->saved_config_space[i]) { -			dev_dbg(&dev->dev, "restoring config " -				"space at offset %#x (was %#x, writing %#x)\n", -				i, val, (int)dev->saved_config_space[i]); -			pci_write_config_dword(dev,i * 4, -				dev->saved_config_space[i]); -			pci_read_config_dword(dev, i * 4, &val); -			mdelay(10); -			tries--; -		} -	} +	pci_restore_config_space(dev, 4, 9, 10); +	pci_restore_config_space(dev, 0, 3, 0); +  	pci_restore_pcix_state(dev);  	pci_restore_msi_state(dev);  	pci_restore_iov_state(dev); | 
