Skip to content

Commit 6f8174b

Browse files
authored
Add support for automatic bridge selection in HW offload mode (#301)
If deviceID argument is passed it can be used for automatic bridge selection: VF deviceID > PF > Bond(Optional) > Bridge Signed-off-by: Yury Kulazhenkov <[email protected]>
1 parent 3663c3d commit 6f8174b

File tree

5 files changed

+173
-36
lines changed

5 files changed

+173
-36
lines changed

docs/cni-plugin.md

+7-1
Original file line numberDiff line numberDiff line change
@@ -50,7 +50,8 @@ Another example with a port which has an interface of type system:
5050

5151
* `name` (string, required): the name of the network.
5252
* `type` (string, required): "ovs".
53-
* `bridge` (string, required): name of the bridge to use.
53+
* `bridge` (string, optional): name of the bridge to use, can be omitted if `ovnPort` is set in CNI_ARGS, or if `deviceID` is set
54+
* `deviceID` (string, optional): PCI address of a Virtual Function in valid sysfs format to use in HW offloading mode. This value is usually set by Multus.
5455
* `vlan` (integer, optional): VLAN ID of attached port. Trunk port if not
5556
specified.
5657
* `mtu` (integer, optional): MTU.
@@ -61,6 +62,11 @@ Another example with a port which has an interface of type system:
6162
* `configuration_path` (optional): configuration file containing ovsdb
6263
socket file path, etc.
6364

65+
66+
_*Note:* if `deviceID` is provided, then it is possible to omit `bridge` argument. Bridge will be automatically selected by the CNI plugin by following
67+
the chain: Virtual Function PCI address (provided in `deviceID` argument) > Physical Function > Bond interface
68+
(optional, if Physical Function is part of a bond interface) > ovs bridge_
69+
6470
### Flatfile Configuation
6571

6672
There is one option for flat file configuration:

docs/ovs-offload.md

+4
Original file line numberDiff line numberDiff line change
@@ -125,6 +125,10 @@ spec:
125125
}'
126126
```
127127
128+
_*Note:* it is possible to omit `bridge` argument. Bridge will be automatically selected by the CNI plugin by following
129+
the chain: Virtual Function PCI address (injected by Multus to `deviceID` parameter) > Physical Function > Bond interface
130+
(optional, if Physical Function is part of a bond interface) > ovs bridge_
131+
128132
Now deploy a pod with the following config to attach VF into container and its representor net device
129133
attached with ovs bridge `br-snic0`.
130134

pkg/ovsdb/ovsdb.go

+27
Original file line numberDiff line numberDiff line change
@@ -91,6 +91,10 @@ func connectToOvsDb(ovsSocket string) (client.Client, error) {
9191
func NewOvsDriver(ovsSocket string) (*OvsDriver, error) {
9292
ovsDriver := new(OvsDriver)
9393

94+
if ovsSocket == "" {
95+
ovsSocket = "unix:/var/run/openvswitch/db.sock"
96+
}
97+
9498
ovsDB, err := connectToOvsDb(ovsSocket)
9599
if err != nil {
96100
return nil, fmt.Errorf("failed to connect to ovsdb error: %v", err)
@@ -619,6 +623,29 @@ func (ovsd *OvsDriver) IsBridgePresent(bridgeName string) (bool, error) {
619623
return true, nil
620624
}
621625

626+
// FindBridgeByInterface returns name of the bridge that contains provided interface
627+
func (ovsd *OvsDriver) FindBridgeByInterface(ifaceName string) (string, error) {
628+
iface, err := ovsd.findByCondition("Interface",
629+
ovsdb.NewCondition("name", ovsdb.ConditionEqual, ifaceName),
630+
[]string{"name", "_uuid"})
631+
if err != nil {
632+
return "", fmt.Errorf("failed to find interface %s: %v", ifaceName, err)
633+
}
634+
port, err := ovsd.findByCondition("Port",
635+
ovsdb.NewCondition("interfaces", ovsdb.ConditionIncludes, iface["_uuid"]),
636+
[]string{"name", "_uuid"})
637+
if err != nil {
638+
return "", fmt.Errorf("failed to find port %s: %v", ifaceName, err)
639+
}
640+
bridge, err := ovsd.findByCondition("Bridge",
641+
ovsdb.NewCondition("ports", ovsdb.ConditionIncludes, port["_uuid"]),
642+
[]string{"name"})
643+
if err != nil {
644+
return "", fmt.Errorf("failed to find bridge for %s: %v", ifaceName, err)
645+
}
646+
return fmt.Sprintf("%v", bridge["name"]), nil
647+
}
648+
622649
// GetOvsPortForContIface Return ovs port name for an container interface
623650
func (ovsd *OvsDriver) GetOvsPortForContIface(contIface, contNetnsPath string) (string, bool, error) {
624651
searchMap := map[string]string{

pkg/plugin/plugin.go

+69-35
Original file line numberDiff line numberDiff line change
@@ -142,11 +142,27 @@ func assignMacToLink(link netlink.Link, mac net.HardwareAddr, name string) error
142142
return nil
143143
}
144144

145-
func getBridgeName(bridgeName, ovnPort string) (string, error) {
145+
func getBridgeName(driver *ovsdb.OvsDriver, bridgeName, ovnPort, deviceID string) (string, error) {
146146
if bridgeName != "" {
147147
return bridgeName, nil
148148
} else if bridgeName == "" && ovnPort != "" {
149149
return "br-int", nil
150+
} else if deviceID != "" {
151+
possibleUplinkNames, err := sriov.GetBridgeUplinkNameByDeviceID(deviceID)
152+
if err != nil {
153+
return "", fmt.Errorf("failed to get bridge name - failed to resolve uplink name: %v", err)
154+
}
155+
var errList []error
156+
for _, uplinkName := range possibleUplinkNames {
157+
bridgeName, err = driver.FindBridgeByInterface(uplinkName)
158+
if err != nil {
159+
errList = append(errList,
160+
fmt.Errorf("failed to get bridge name - failed to find bridge name by uplink name %s: %v", uplinkName, err))
161+
continue
162+
}
163+
return bridgeName, nil
164+
}
165+
return "", fmt.Errorf("failed to find bridge by uplink names %v: %v", possibleUplinkNames, errList)
150166
}
151167

152168
return "", fmt.Errorf("failed to get bridge name")
@@ -256,19 +272,27 @@ func CmdAdd(args *skel.CmdArgs) error {
256272
} else if netconf.VlanTag != nil {
257273
vlanTagNum = *netconf.VlanTag
258274
}
259-
260-
bridgeName, err := getBridgeName(netconf.BrName, ovnPort)
275+
ovsDriver, err := ovsdb.NewOvsDriver(netconf.SocketFile)
261276
if err != nil {
262277
return err
263278
}
279+
bridgeName, err := getBridgeName(ovsDriver, netconf.BrName, ovnPort, netconf.DeviceID)
280+
if err != nil {
281+
return err
282+
}
283+
// save discovered bridge name to the netconf struct to make
284+
// sure it is save in the cache.
285+
// we need to cache discovered bridge name to make sure that we will
286+
// use the right bridge name in CmdDel
287+
netconf.BrName = bridgeName
264288

265-
ovsDriver, err := ovsdb.NewOvsBridgeDriver(bridgeName, netconf.SocketFile)
289+
ovsBridgeDriver, err := ovsdb.NewOvsBridgeDriver(bridgeName, netconf.SocketFile)
266290
if err != nil {
267291
return err
268292
}
269293

270294
// removes all ports whose interfaces have an error
271-
if err := cleanPorts(ovsDriver); err != nil {
295+
if err := cleanPorts(ovsBridgeDriver); err != nil {
272296
return err
273297
}
274298

@@ -305,19 +329,19 @@ func CmdAdd(args *skel.CmdArgs) error {
305329
}
306330
}
307331

308-
if err = attachIfaceToBridge(ovsDriver, hostIface.Name, contIface.Name, netconf.OfportRequest, vlanTagNum, trunks, portType, netconf.InterfaceType, args.Netns, ovnPort); err != nil {
332+
if err = attachIfaceToBridge(ovsBridgeDriver, hostIface.Name, contIface.Name, netconf.OfportRequest, vlanTagNum, trunks, portType, netconf.InterfaceType, args.Netns, ovnPort); err != nil {
309333
return err
310334
}
311335
defer func() {
312336
if err != nil {
313337
// Unlike veth pair, OVS port will not be automatically removed
314338
// if the following IPAM configuration fails and netns gets removed.
315-
portName, portFound, err := getOvsPortForContIface(ovsDriver, args.IfName, args.Netns)
339+
portName, portFound, err := getOvsPortForContIface(ovsBridgeDriver, args.IfName, args.Netns)
316340
if err != nil {
317341
log.Printf("Failed best-effort cleanup: %v", err)
318342
}
319343
if portFound {
320-
if err := removeOvsPort(ovsDriver, portName); err != nil {
344+
if err := removeOvsPort(ovsBridgeDriver, portName); err != nil {
321345
log.Printf("Failed best-effort cleanup: %v", err)
322346
}
323347
}
@@ -364,7 +388,7 @@ func CmdAdd(args *skel.CmdArgs) error {
364388

365389
// wait until OF port link state becomes up. This is needed to make
366390
// gratuitous arp for args.IfName to be sent over ovs bridge
367-
err = waitLinkUp(ovsDriver, hostIface.Name, netconf.LinkStateCheckRetries, netconf.LinkStateCheckInterval)
391+
err = waitLinkUp(ovsBridgeDriver, hostIface.Name, netconf.LinkStateCheckRetries, netconf.LinkStateCheckInterval)
368392
if err != nil {
369393
return err
370394
}
@@ -502,13 +526,16 @@ func CmdDel(args *skel.CmdArgs) error {
502526
if envArgs != nil {
503527
ovnPort = string(envArgs.OvnPort)
504528
}
505-
506-
bridgeName, err := getBridgeName(cache.Netconf.BrName, ovnPort)
529+
ovsDriver, err := ovsdb.NewOvsDriver(cache.Netconf.SocketFile)
530+
if err != nil {
531+
return err
532+
}
533+
bridgeName, err := getBridgeName(ovsDriver, cache.Netconf.BrName, ovnPort, cache.Netconf.DeviceID)
507534
if err != nil {
508535
return err
509536
}
510537

511-
ovsDriver, err := ovsdb.NewOvsBridgeDriver(bridgeName, cache.Netconf.SocketFile)
538+
ovsBridgeDriver, err := ovsdb.NewOvsBridgeDriver(bridgeName, cache.Netconf.SocketFile)
512539
if err != nil {
513540
return err
514541
}
@@ -530,7 +557,7 @@ func CmdDel(args *skel.CmdArgs) error {
530557
if rep, err = sriov.GetNetRepresentor(cache.Netconf.DeviceID); err != nil {
531558
return err
532559
}
533-
if err = removeOvsPort(ovsDriver, rep); err != nil {
560+
if err = removeOvsPort(ovsBridgeDriver, rep); err != nil {
534561
// Don't throw err as delete can be called multiple times because of error in ResetVF and ovs
535562
// port is already deleted in a previous invocation.
536563
log.Printf("Error: %v\n", err)
@@ -540,7 +567,7 @@ func CmdDel(args *skel.CmdArgs) error {
540567
}
541568
} else {
542569
// In accordance with the spec we clean up as many resources as possible.
543-
if err := cleanPorts(ovsDriver); err != nil {
570+
if err := cleanPorts(ovsBridgeDriver); err != nil {
544571
return err
545572
}
546573
}
@@ -550,15 +577,15 @@ func CmdDel(args *skel.CmdArgs) error {
550577
// Unlike veth pair, OVS port will not be automatically removed when
551578
// container namespace is gone. Find port matching DEL arguments and remove
552579
// it explicitly.
553-
portName, portFound, err := getOvsPortForContIface(ovsDriver, args.IfName, args.Netns)
580+
portName, portFound, err := getOvsPortForContIface(ovsBridgeDriver, args.IfName, args.Netns)
554581
if err != nil {
555582
return fmt.Errorf("Failed to obtain OVS port for given connection: %v", err)
556583
}
557584

558585
// Do not return an error if the port was not found, it may have been
559586
// already removed by someone.
560587
if portFound {
561-
if err := removeOvsPort(ovsDriver, portName); err != nil {
588+
if err := removeOvsPort(ovsBridgeDriver, portName); err != nil {
562589
return err
563590
}
564591
}
@@ -589,7 +616,7 @@ func CmdDel(args *skel.CmdArgs) error {
589616
}
590617

591618
// removes all ports whose interfaces have an error
592-
if err := cleanPorts(ovsDriver); err != nil {
619+
if err := cleanPorts(ovsBridgeDriver); err != nil {
593620
return err
594621
}
595622

@@ -614,12 +641,33 @@ func CmdCheck(args *skel.CmdArgs) error {
614641
}
615642
}
616643

644+
envArgs, err := getEnvArgs(args.Args)
645+
if err != nil {
646+
return err
647+
}
648+
var ovnPort string
649+
if envArgs != nil {
650+
ovnPort = string(envArgs.OvnPort)
651+
}
652+
ovsDriver, err := ovsdb.NewOvsDriver(netconf.SocketFile)
653+
if err != nil {
654+
return err
655+
}
656+
// cached config may contain bridge name which were automatically
657+
// discovered in CmdAdd, we need to re-discover the bridge name before we validating the cache
658+
bridgeName, err := getBridgeName(ovsDriver, netconf.BrName, ovnPort, netconf.DeviceID)
659+
if err != nil {
660+
return err
661+
}
662+
netconf.BrName = bridgeName
663+
617664
// check cache
618665
cRef := config.GetCRef(args.ContainerID, args.IfName)
619666
cache, err := config.LoadConfFromCache(cRef)
620667
if err != nil {
621668
return err
622669
}
670+
623671
if err := validateCache(cache, netconf); err != nil {
624672
return err
625673
}
@@ -754,42 +802,28 @@ func validateInterface(intf current.Interface, isHost bool, hwOffload bool) erro
754802
}
755803

756804
func validateOvs(args *skel.CmdArgs, netconf *types.NetConf, hostIfname string) error {
757-
envArgs, err := getEnvArgs(args.Args)
758-
if err != nil {
759-
return err
760-
}
761-
var ovnPort string
762-
if envArgs != nil {
763-
ovnPort = string(envArgs.OvnPort)
764-
}
765-
766-
bridgeName, err := getBridgeName(netconf.BrName, ovnPort)
805+
ovsBridgeDriver, err := ovsdb.NewOvsBridgeDriver(netconf.BrName, netconf.SocketFile)
767806
if err != nil {
768807
return err
769808
}
770809

771-
ovsDriver, err := ovsdb.NewOvsBridgeDriver(bridgeName, netconf.SocketFile)
772-
if err != nil {
773-
return err
774-
}
775-
776-
found, err := ovsDriver.IsBridgePresent(netconf.BrName)
810+
found, err := ovsBridgeDriver.IsBridgePresent(netconf.BrName)
777811
if err != nil {
778812
return err
779813
}
780814
if !found {
781815
return fmt.Errorf("Error: bridge %s is not found in OVS", netconf.BrName)
782816
}
783817

784-
ifaces, err := ovsDriver.FindInterfacesWithError()
818+
ifaces, err := ovsBridgeDriver.FindInterfacesWithError()
785819
if err != nil {
786820
return err
787821
}
788822
if len(ifaces) > 0 {
789823
return fmt.Errorf("Error: There are some interfaces in error state: %v", ifaces)
790824
}
791825

792-
vlanMode, tag, trunk, err := ovsDriver.GetOFPortVlanState(hostIfname)
826+
vlanMode, tag, trunk, err := ovsBridgeDriver.GetOFPortVlanState(hostIfname)
793827
if err != nil {
794828
return fmt.Errorf("Error: Failed to retrieve port %s state: %v", hostIfname, err)
795829
}

pkg/sriov/sriov.go

+66
Original file line numberDiff line numberDiff line change
@@ -67,6 +67,72 @@ func IsOvsHardwareOffloadEnabled(deviceID string) bool {
6767
return deviceID != ""
6868
}
6969

70+
// GetBridgeUplinkNameByDeviceID tries to automatically resolve uplink interface name
71+
// for provided VF deviceID by following the sequence:
72+
// VF pci address > PF pci address > Bond (optional, if PF is part of a bond)
73+
// return list of candidate names
74+
func GetBridgeUplinkNameByDeviceID(deviceID string) ([]string, error) {
75+
pfName, err := sriovnet.GetUplinkRepresentor(deviceID)
76+
if err != nil {
77+
return nil, err
78+
}
79+
pfLink, err := netlink.LinkByName(pfName)
80+
if err != nil {
81+
return nil, fmt.Errorf("failed to get link info for uplink %s: %v", pfName, err)
82+
}
83+
bond, err := getBondInterface(pfLink)
84+
if err != nil {
85+
return nil, fmt.Errorf("failed to get parent link for uplink %s: %v", pfName, err)
86+
}
87+
if bond == nil {
88+
// PF has no parent bond, return only PF name
89+
return []string{pfLink.Attrs().Name}, nil
90+
}
91+
// for some OVS datapathes, to use bond configuration it is required to attach primary PF (usually first one) to the ovs instead of the bond interface.
92+
// Example:
93+
// - Bond interface bond0 (contains PF0 + PF1)
94+
// - OVS bridge br0 (only PF0 is attached)
95+
// - VF representors from PF0 and PF1 can be attached to OVS bridge br0, traffic will be offloaded and sent through bond0
96+
//
97+
// to support autobridge selection for VFs from the PF1 (which is part of the bond, but not directly attached to the ovs),
98+
// we need to add other interfaces that are part of the bond as candidates, for PF1 candidates list will be: [bond0, PF0, PF1]
99+
bondMembers, err := getBondMembers(bond)
100+
if err != nil {
101+
return nil, fmt.Errorf("failed to retrieve list of bond members for bond %s, uplink %s: %v", bond.Attrs().Name, pfName, err)
102+
}
103+
return bondMembers, nil
104+
}
105+
106+
// getBondInterface returns a parent bond interface for the link if it exists
107+
func getBondInterface(link netlink.Link) (netlink.Link, error) {
108+
if link.Attrs().MasterIndex == 0 {
109+
return nil, nil
110+
}
111+
bondLink, err := netlink.LinkByIndex(link.Attrs().MasterIndex)
112+
if err != nil {
113+
return nil, err
114+
}
115+
if bondLink.Type() != "bond" {
116+
return nil, nil
117+
}
118+
return bondLink, nil
119+
}
120+
121+
// getBondMembers returns list with name of the bond and all bond members
122+
func getBondMembers(bond netlink.Link) ([]string, error) {
123+
allLinks, err := netlink.LinkList()
124+
if err != nil {
125+
return nil, err
126+
}
127+
result := []string{bond.Attrs().Name}
128+
for _, link := range allLinks {
129+
if link.Attrs().MasterIndex == bond.Attrs().Index {
130+
result = append(result, link.Attrs().Name)
131+
}
132+
}
133+
return result, nil
134+
}
135+
70136
// GetNetRepresentor retrieves network representor device for smartvf
71137
func GetNetRepresentor(deviceID string) (string, error) {
72138
// get Uplink netdevice. The uplink is basically the PF name of the deviceID (smart VF).

0 commit comments

Comments
 (0)