diff options
author | Brian King <brking@us.ibm.com> | 2005-09-27 04:21:55 -0400 |
---|---|---|
committer | Greg Kroah-Hartman <gregkh@suse.de> | 2005-10-28 18:36:58 -0400 |
commit | e04b0ea2e0f9c1bb0d874db4493fc7f7a623116b (patch) | |
tree | 75937e50de883f69e906a4c8bc9f119d86c14411 /drivers/pci/syscall.c | |
parent | 2cea752f683af1be58ee8f25717c0a8118e0ac5b (diff) |
[PATCH] PCI: Block config access during BIST
Some PCI adapters (eg. ipr scsi adapters) have an exposure today in that they
issue BIST to the adapter to reset the card. If, during the time it takes to
complete BIST, userspace attempts to access PCI config space, the host bus
bridge will master abort the access since the ipr adapter does not respond on
the PCI bus for a brief period of time when running BIST. On PPC64 hardware,
this master abort results in the host PCI bridge isolating that PCI device
from the rest of the system, making the device unusable until Linux is
rebooted. This patch is an attempt to close that exposure by introducing some
blocking code in the PCI code. When blocked, writes will be humored and reads
will return the cached value. Ben Herrenschmidt has also mentioned that he
plans to use this in PPC power management.
Signed-off-by: Brian King <brking@us.ibm.com>
Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
drivers/pci/access.c | 89 ++++++++++++++++++++++++++++++++++++++++++++++++
drivers/pci/pci-sysfs.c | 20 +++++-----
drivers/pci/pci.h | 7 +++
drivers/pci/proc.c | 28 +++++++--------
drivers/pci/syscall.c | 14 +++----
include/linux/pci.h | 7 +++
6 files changed, 134 insertions(+), 31 deletions(-)
Diffstat (limited to 'drivers/pci/syscall.c')
-rw-r--r-- | drivers/pci/syscall.c | 14 |
1 files changed, 7 insertions, 7 deletions
diff --git a/drivers/pci/syscall.c b/drivers/pci/syscall.c index c071790cc983..87fafc08cb9d 100644 --- a/drivers/pci/syscall.c +++ b/drivers/pci/syscall.c | |||
@@ -13,7 +13,7 @@ | |||
13 | #include <linux/smp_lock.h> | 13 | #include <linux/smp_lock.h> |
14 | #include <linux/syscalls.h> | 14 | #include <linux/syscalls.h> |
15 | #include <asm/uaccess.h> | 15 | #include <asm/uaccess.h> |
16 | 16 | #include "pci.h" | |
17 | 17 | ||
18 | asmlinkage long | 18 | asmlinkage long |
19 | sys_pciconfig_read(unsigned long bus, unsigned long dfn, | 19 | sys_pciconfig_read(unsigned long bus, unsigned long dfn, |
@@ -38,13 +38,13 @@ sys_pciconfig_read(unsigned long bus, unsigned long dfn, | |||
38 | lock_kernel(); | 38 | lock_kernel(); |
39 | switch (len) { | 39 | switch (len) { |
40 | case 1: | 40 | case 1: |
41 | cfg_ret = pci_read_config_byte(dev, off, &byte); | 41 | cfg_ret = pci_user_read_config_byte(dev, off, &byte); |
42 | break; | 42 | break; |
43 | case 2: | 43 | case 2: |
44 | cfg_ret = pci_read_config_word(dev, off, &word); | 44 | cfg_ret = pci_user_read_config_word(dev, off, &word); |
45 | break; | 45 | break; |
46 | case 4: | 46 | case 4: |
47 | cfg_ret = pci_read_config_dword(dev, off, &dword); | 47 | cfg_ret = pci_user_read_config_dword(dev, off, &dword); |
48 | break; | 48 | break; |
49 | default: | 49 | default: |
50 | err = -EINVAL; | 50 | err = -EINVAL; |
@@ -112,7 +112,7 @@ sys_pciconfig_write(unsigned long bus, unsigned long dfn, | |||
112 | err = get_user(byte, (u8 __user *)buf); | 112 | err = get_user(byte, (u8 __user *)buf); |
113 | if (err) | 113 | if (err) |
114 | break; | 114 | break; |
115 | err = pci_write_config_byte(dev, off, byte); | 115 | err = pci_user_write_config_byte(dev, off, byte); |
116 | if (err != PCIBIOS_SUCCESSFUL) | 116 | if (err != PCIBIOS_SUCCESSFUL) |
117 | err = -EIO; | 117 | err = -EIO; |
118 | break; | 118 | break; |
@@ -121,7 +121,7 @@ sys_pciconfig_write(unsigned long bus, unsigned long dfn, | |||
121 | err = get_user(word, (u16 __user *)buf); | 121 | err = get_user(word, (u16 __user *)buf); |
122 | if (err) | 122 | if (err) |
123 | break; | 123 | break; |
124 | err = pci_write_config_word(dev, off, word); | 124 | err = pci_user_write_config_word(dev, off, word); |
125 | if (err != PCIBIOS_SUCCESSFUL) | 125 | if (err != PCIBIOS_SUCCESSFUL) |
126 | err = -EIO; | 126 | err = -EIO; |
127 | break; | 127 | break; |
@@ -130,7 +130,7 @@ sys_pciconfig_write(unsigned long bus, unsigned long dfn, | |||
130 | err = get_user(dword, (u32 __user *)buf); | 130 | err = get_user(dword, (u32 __user *)buf); |
131 | if (err) | 131 | if (err) |
132 | break; | 132 | break; |
133 | err = pci_write_config_dword(dev, off, dword); | 133 | err = pci_user_write_config_dword(dev, off, dword); |
134 | if (err != PCIBIOS_SUCCESSFUL) | 134 | if (err != PCIBIOS_SUCCESSFUL) |
135 | err = -EIO; | 135 | err = -EIO; |
136 | break; | 136 | break; |