forked from Minki/linux
ice: restore VF MSI-X state during PCI reset
During a PCI FLR the MSI-X Enable flag in the VF PCI MSI-X capability register will be cleared. This can lead to issues when a VF is assigned to a VM because in these cases the VF driver receives no indication of the PF PCI error/reset and additionally it is incapable of restoring the cleared flag in the hypervisor configuration space without fully reinitializing the driver interrupt functionality. Since the VF driver is unable to easily resolve this condition on its own, restore the VF MSI-X flag during the PF PCI reset handling. Signed-off-by: Nick Nunley <nicholas.d.nunley@intel.com> Tested-by: Andrew Bowers <andrewx.bowers@intel.com> Signed-off-by: Tony Nguyen <anthony.l.nguyen@intel.com>
This commit is contained in:
parent
0ce6c34a8f
commit
a54a0b24f4
@ -4623,6 +4623,8 @@ static void ice_pci_err_resume(struct pci_dev *pdev)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ice_restore_all_vfs_msi_state(pdev);
|
||||||
|
|
||||||
ice_do_reset(pf, ICE_RESET_PFR);
|
ice_do_reset(pf, ICE_RESET_PFR);
|
||||||
ice_service_task_restart(pf);
|
ice_service_task_restart(pf);
|
||||||
mod_timer(&pf->serv_tmr, round_jiffies(jiffies + pf->serv_tmr_period));
|
mod_timer(&pf->serv_tmr, round_jiffies(jiffies + pf->serv_tmr_period));
|
||||||
|
@ -4071,3 +4071,33 @@ void ice_print_vfs_mdd_events(struct ice_pf *pf)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ice_restore_all_vfs_msi_state - restore VF MSI state after PF FLR
|
||||||
|
* @pdev: pointer to a pci_dev structure
|
||||||
|
*
|
||||||
|
* Called when recovering from a PF FLR to restore interrupt capability to
|
||||||
|
* the VFs.
|
||||||
|
*/
|
||||||
|
void ice_restore_all_vfs_msi_state(struct pci_dev *pdev)
|
||||||
|
{
|
||||||
|
struct pci_dev *vfdev;
|
||||||
|
u16 vf_id;
|
||||||
|
int pos;
|
||||||
|
|
||||||
|
if (!pci_num_vf(pdev))
|
||||||
|
return;
|
||||||
|
|
||||||
|
pos = pci_find_ext_capability(pdev, PCI_EXT_CAP_ID_SRIOV);
|
||||||
|
if (pos) {
|
||||||
|
pci_read_config_word(pdev, pos + PCI_SRIOV_VF_DID,
|
||||||
|
&vf_id);
|
||||||
|
vfdev = pci_get_device(pdev->vendor, vf_id, NULL);
|
||||||
|
while (vfdev) {
|
||||||
|
if (vfdev->is_virtfn && vfdev->physfn == pdev)
|
||||||
|
pci_restore_msi_state(vfdev);
|
||||||
|
vfdev = pci_get_device(pdev->vendor, vf_id,
|
||||||
|
vfdev);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -114,6 +114,7 @@ void ice_vc_notify_link_state(struct ice_pf *pf);
|
|||||||
void ice_vc_notify_reset(struct ice_pf *pf);
|
void ice_vc_notify_reset(struct ice_pf *pf);
|
||||||
bool ice_reset_all_vfs(struct ice_pf *pf, bool is_vflr);
|
bool ice_reset_all_vfs(struct ice_pf *pf, bool is_vflr);
|
||||||
bool ice_reset_vf(struct ice_vf *vf, bool is_vflr);
|
bool ice_reset_vf(struct ice_vf *vf, bool is_vflr);
|
||||||
|
void ice_restore_all_vfs_msi_state(struct pci_dev *pdev);
|
||||||
|
|
||||||
int
|
int
|
||||||
ice_set_vf_port_vlan(struct net_device *netdev, int vf_id, u16 vlan_id, u8 qos,
|
ice_set_vf_port_vlan(struct net_device *netdev, int vf_id, u16 vlan_id, u8 qos,
|
||||||
@ -146,6 +147,7 @@ void ice_print_vf_rx_mdd_event(struct ice_vf *vf);
|
|||||||
#define ice_vf_lan_overflow_event(pf, event) do {} while (0)
|
#define ice_vf_lan_overflow_event(pf, event) do {} while (0)
|
||||||
#define ice_print_vfs_mdd_events(pf) do {} while (0)
|
#define ice_print_vfs_mdd_events(pf) do {} while (0)
|
||||||
#define ice_print_vf_rx_mdd_event(vf) do {} while (0)
|
#define ice_print_vf_rx_mdd_event(vf) do {} while (0)
|
||||||
|
#define ice_restore_all_vfs_msi_state(pdev) do {} while (0)
|
||||||
|
|
||||||
static inline bool
|
static inline bool
|
||||||
ice_reset_all_vfs(struct ice_pf __always_unused *pf,
|
ice_reset_all_vfs(struct ice_pf __always_unused *pf,
|
||||||
|
Loading…
Reference in New Issue
Block a user