diff options
Diffstat (limited to 'drivers/pci')
-rw-r--r-- | drivers/pci/iov.c | 29 | ||||
-rw-r--r-- | drivers/pci/pci.c | 1 | ||||
-rw-r--r-- | drivers/pci/pci.h | 4 |
3 files changed, 34 insertions, 0 deletions
diff --git a/drivers/pci/iov.c b/drivers/pci/iov.c index 66cc414ed15f..b121e47402fa 100644 --- a/drivers/pci/iov.c +++ b/drivers/pci/iov.c | |||
@@ -129,6 +129,25 @@ static void sriov_release(struct pci_dev *dev) | |||
129 | dev->sriov = NULL; | 129 | dev->sriov = NULL; |
130 | } | 130 | } |
131 | 131 | ||
132 | static void sriov_restore_state(struct pci_dev *dev) | ||
133 | { | ||
134 | int i; | ||
135 | u16 ctrl; | ||
136 | struct pci_sriov *iov = dev->sriov; | ||
137 | |||
138 | pci_read_config_word(dev, iov->pos + PCI_SRIOV_CTRL, &ctrl); | ||
139 | if (ctrl & PCI_SRIOV_CTRL_VFE) | ||
140 | return; | ||
141 | |||
142 | for (i = PCI_IOV_RESOURCES; i <= PCI_IOV_RESOURCE_END; i++) | ||
143 | pci_update_resource(dev, i); | ||
144 | |||
145 | pci_write_config_dword(dev, iov->pos + PCI_SRIOV_SYS_PGSIZE, iov->pgsz); | ||
146 | pci_write_config_word(dev, iov->pos + PCI_SRIOV_CTRL, iov->ctrl); | ||
147 | if (iov->ctrl & PCI_SRIOV_CTRL_VFE) | ||
148 | msleep(100); | ||
149 | } | ||
150 | |||
132 | /** | 151 | /** |
133 | * pci_iov_init - initialize the IOV capability | 152 | * pci_iov_init - initialize the IOV capability |
134 | * @dev: the PCI device | 153 | * @dev: the PCI device |
@@ -180,3 +199,13 @@ int pci_iov_resource_bar(struct pci_dev *dev, int resno, | |||
180 | return dev->sriov->pos + PCI_SRIOV_BAR + | 199 | return dev->sriov->pos + PCI_SRIOV_BAR + |
181 | 4 * (resno - PCI_IOV_RESOURCES); | 200 | 4 * (resno - PCI_IOV_RESOURCES); |
182 | } | 201 | } |
202 | |||
203 | /** | ||
204 | * pci_restore_iov_state - restore the state of the IOV capability | ||
205 | * @dev: the PCI device | ||
206 | */ | ||
207 | void pci_restore_iov_state(struct pci_dev *dev) | ||
208 | { | ||
209 | if (dev->is_physfn) | ||
210 | sriov_restore_state(dev); | ||
211 | } | ||
diff --git a/drivers/pci/pci.c b/drivers/pci/pci.c index 2b3201ec2b05..676bbcbc272b 100644 --- a/drivers/pci/pci.c +++ b/drivers/pci/pci.c | |||
@@ -775,6 +775,7 @@ pci_restore_state(struct pci_dev *dev) | |||
775 | } | 775 | } |
776 | pci_restore_pcix_state(dev); | 776 | pci_restore_pcix_state(dev); |
777 | pci_restore_msi_state(dev); | 777 | pci_restore_msi_state(dev); |
778 | pci_restore_iov_state(dev); | ||
778 | 779 | ||
779 | return 0; | 780 | return 0; |
780 | } | 781 | } |
diff --git a/drivers/pci/pci.h b/drivers/pci/pci.h index 7d5327c986f5..fd5ea4d445e8 100644 --- a/drivers/pci/pci.h +++ b/drivers/pci/pci.h | |||
@@ -222,6 +222,7 @@ extern int pci_iov_init(struct pci_dev *dev); | |||
222 | extern void pci_iov_release(struct pci_dev *dev); | 222 | 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 | #else | 226 | #else |
226 | static inline int pci_iov_init(struct pci_dev *dev) | 227 | static inline int pci_iov_init(struct pci_dev *dev) |
227 | { | 228 | { |
@@ -236,6 +237,9 @@ static inline int pci_iov_resource_bar(struct pci_dev *dev, int resno, | |||
236 | { | 237 | { |
237 | return 0; | 238 | return 0; |
238 | } | 239 | } |
240 | static inline void pci_restore_iov_state(struct pci_dev *dev) | ||
241 | { | ||
242 | } | ||
239 | #endif /* CONFIG_PCI_IOV */ | 243 | #endif /* CONFIG_PCI_IOV */ |
240 | 244 | ||
241 | #endif /* DRIVERS_PCI_H */ | 245 | #endif /* DRIVERS_PCI_H */ |