diff --git a/pkg/utils/utils.go b/pkg/utils/utils.go index ca8ad4f10b..b0e41e1759 100644 --- a/pkg/utils/utils.go +++ b/pkg/utils/utils.go @@ -272,7 +272,7 @@ func configSriovDevice(iface *sriovnetworkv1.Interface, ifaceStatus *sriovnetwor for _, addr := range vfAddrs { var group sriovnetworkv1.VfGroup i := 0 - var driver string + var dpdkDriver string var isRdma bool vfID, err := dputils.GetVFID(addr) for i, group = range iface.VfGroups { @@ -282,22 +282,17 @@ func configSriovDevice(iface *sriovnetworkv1.Interface, ifaceStatus *sriovnetwor if sriovnetworkv1.IndexInRange(vfID, group.VfRange) { isRdma = group.IsRdma if sriovnetworkv1.StringInArray(group.DeviceType, DpdkDrivers) { - driver = group.DeviceType + dpdkDriver = group.DeviceType } break } } - if strings.EqualFold(iface.LinkType, "IB") { - if err = setVfGuid(addr, pfLink); err != nil { - return err - } - } else if err = setVfAdminMac(addr, pfLink); err != nil { - return err - } + if err = unbindDriverIfNeeded(addr, isRdma); err != nil { return err } - if driver == "" { + + if dpdkDriver == "" { if err := BindDefaultDriver(addr); err != nil { glog.Warningf("configSriovDevice(): fail to bind default driver for device %s", addr) return err @@ -309,9 +304,37 @@ func configSriovDevice(iface *sriovnetworkv1.Interface, ifaceStatus *sriovnetwor return err } } + + // only set GUID and MAC for VF with default driver + if strings.EqualFold(iface.LinkType, "IB") { + if err = setVfGuid(addr, pfLink); err != nil { + return err + } + } else { + vfLink, err := vfIsReady(addr) + if err != nil { + err = RebindVfToDefaultDriver(addr) + if err != nil { + glog.Errorf("setVfAdminMac(): failed to rebind VF %s %q", addr, err) + return err + } + + // Try to check the VF status again + vfLink, err = vfIsReady(addr) + if err != nil { + glog.Errorf("setVfAdminMac(): VF link is not ready for device %s %q", addr, err) + return err + } + + } + + if err = setVfAdminMac(addr, pfLink, vfLink); err != nil { + return err + } + } } else { - if err := BindDpdkDriver(addr, driver); err != nil { - glog.Warningf("configSriovDevice(): fail to bind driver %s for device %s", driver, addr) + if err := BindDpdkDriver(addr, dpdkDriver); err != nil { + glog.Warningf("configSriovDevice(): fail to bind driver %s for device %s", dpdkDriver, addr) return err } } @@ -555,7 +578,7 @@ func vfIsReady(pciAddr string) (netlink.Link, error) { return vfLink, nil } -func setVfAdminMac(vfAddr string, pfLink netlink.Link) error { +func setVfAdminMac(vfAddr string, pfLink, vfLink netlink.Link) error { glog.Infof("setVfAdminMac(): VF %s", vfAddr) vfID, err := dputils.GetVFID(vfAddr) @@ -563,11 +586,7 @@ func setVfAdminMac(vfAddr string, pfLink netlink.Link) error { glog.Errorf("setVfAdminMac(): unable to get VF id %+v %q", vfAddr, err) return err } - vfLink, err := vfIsReady(vfAddr) - if err != nil { - glog.Errorf("setVfAdminMac(): VF link is not ready for device %+v %q", vfAddr, err) - return err - } + if err := netlink.LinkSetVfHardwareAddr(pfLink, vfID, vfLink.Attrs().HardwareAddr); err != nil { return err } @@ -722,3 +741,20 @@ func hasMellanoxInterfacesInSpec(newState *sriovnetworkv1.SriovNetworkNodeState) } return false } + +// Workaround function to handle a case where the vf default driver is stuck and not able to create the vf kernel interface. +// This function unbind the VF from the default driver and try to bind it again +// bugzilla: https://bugzilla.redhat.com/show_bug.cgi?id=2045087 +func RebindVfToDefaultDriver(vfAddr string) error { + glog.Infof("RebindVfToDefaultDriver(): VF %s", vfAddr) + if err := Unbind(vfAddr); err != nil { + return err + } + if err := BindDefaultDriver(vfAddr); err != nil { + glog.Errorf("RebindVfToDefaultDriver(): fail to bind default driver for device %s", vfAddr) + return err + } + + glog.Warningf("RebindVfToDefaultDriver(): workaround implemented for VF %s", vfAddr) + return nil +}