diff options
-rw-r--r-- | drivers/pci/iov.c | 36 | ||||
-rw-r--r-- | drivers/pci/pci.h | 5 | ||||
-rw-r--r-- | drivers/pci/probe.c | 3 |
3 files changed, 44 insertions, 0 deletions
diff --git a/drivers/pci/iov.c b/drivers/pci/iov.c index b121e47402fa..5ddfc09a8d3f 100644 --- a/drivers/pci/iov.c +++ b/drivers/pci/iov.c | |||
@@ -14,6 +14,18 @@ | |||
14 | #include "pci.h" | 14 | #include "pci.h" |
15 | 15 | ||
16 | 16 | ||
17 | static inline u8 virtfn_bus(struct pci_dev *dev, int id) | ||
18 | { | ||
19 | return dev->bus->number + ((dev->devfn + dev->sriov->offset + | ||
20 | dev->sriov->stride * id) >> 8); | ||
21 | } | ||
22 | |||
23 | static inline u8 virtfn_devfn(struct pci_dev *dev, int id) | ||
24 | { | ||
25 | return (dev->devfn + dev->sriov->offset + | ||
26 | dev->sriov->stride * id) & 0xff; | ||
27 | } | ||
28 | |||
17 | static int sriov_init(struct pci_dev *dev, int pos) | 29 | static int sriov_init(struct pci_dev *dev, int pos) |
18 | { | 30 | { |
19 | int i; | 31 | int i; |
@@ -209,3 +221,27 @@ void pci_restore_iov_state(struct pci_dev *dev) | |||
209 | if (dev->is_physfn) | 221 | if (dev->is_physfn) |
210 | sriov_restore_state(dev); | 222 | sriov_restore_state(dev); |
211 | } | 223 | } |
224 | |||
225 | /** | ||
226 | * pci_iov_bus_range - find bus range used by Virtual Function | ||
227 | * @bus: the PCI bus | ||
228 | * | ||
229 | * Returns max number of buses (exclude current one) used by Virtual | ||
230 | * Functions. | ||
231 | */ | ||
232 | int pci_iov_bus_range(struct pci_bus *bus) | ||
233 | { | ||
234 | int max = 0; | ||
235 | u8 busnr; | ||
236 | struct pci_dev *dev; | ||
237 | |||
238 | list_for_each_entry(dev, &bus->devices, bus_list) { | ||
239 | if (!dev->is_physfn) | ||
240 | continue; | ||
241 | busnr = virtfn_bus(dev, dev->sriov->total - 1); | ||
242 | if (busnr > max) | ||
243 | max = busnr; | ||
244 | } | ||
245 | |||
246 | return max ? max - bus->number : 0; | ||
247 | } | ||
diff --git a/drivers/pci/pci.h b/drivers/pci/pci.h index fd5ea4d445e8..5c29cb2b8e63 100644 --- a/drivers/pci/pci.h +++ b/drivers/pci/pci.h | |||
@@ -223,6 +223,7 @@ extern void pci_iov_release(struct pci_dev *dev); | |||
223 | extern int pci_iov_resource_bar(struct pci_dev *dev, int resno, | 223 | extern int pci_iov_resource_bar(struct pci_dev *dev, int resno, |
224 | enum pci_bar_type *type); | 224 | enum pci_bar_type *type); |
225 | extern void pci_restore_iov_state(struct pci_dev *dev); | 225 | extern void pci_restore_iov_state(struct pci_dev *dev); |
226 | extern int pci_iov_bus_range(struct pci_bus *bus); | ||
226 | #else | 227 | #else |
227 | static inline int pci_iov_init(struct pci_dev *dev) | 228 | static inline int pci_iov_init(struct pci_dev *dev) |
228 | { | 229 | { |
@@ -240,6 +241,10 @@ static inline int pci_iov_resource_bar(struct pci_dev *dev, int resno, | |||
240 | static inline void pci_restore_iov_state(struct pci_dev *dev) | 241 | static inline void pci_restore_iov_state(struct pci_dev *dev) |
241 | { | 242 | { |
242 | } | 243 | } |
244 | static inline int pci_iov_bus_range(struct pci_bus *bus) | ||
245 | { | ||
246 | return 0; | ||
247 | } | ||
243 | #endif /* CONFIG_PCI_IOV */ | 248 | #endif /* CONFIG_PCI_IOV */ |
244 | 249 | ||
245 | #endif /* DRIVERS_PCI_H */ | 250 | #endif /* DRIVERS_PCI_H */ |
diff --git a/drivers/pci/probe.c b/drivers/pci/probe.c index 0471f6ea1466..0ecdaea3915f 100644 --- a/drivers/pci/probe.c +++ b/drivers/pci/probe.c | |||
@@ -1085,6 +1085,9 @@ unsigned int __devinit pci_scan_child_bus(struct pci_bus *bus) | |||
1085 | for (devfn = 0; devfn < 0x100; devfn += 8) | 1085 | for (devfn = 0; devfn < 0x100; devfn += 8) |
1086 | pci_scan_slot(bus, devfn); | 1086 | pci_scan_slot(bus, devfn); |
1087 | 1087 | ||
1088 | /* Reserve buses for SR-IOV capability. */ | ||
1089 | max += pci_iov_bus_range(bus); | ||
1090 | |||
1088 | /* | 1091 | /* |
1089 | * After performing arch-dependent fixup of the bus, look behind | 1092 | * After performing arch-dependent fixup of the bus, look behind |
1090 | * all PCI-to-PCI bridges on this bus. | 1093 | * all PCI-to-PCI bridges on this bus. |