From a241e658401dd69fab39b2818c70ef0bafa24e90 Mon Sep 17 00:00:00 2001 From: Leonardo Milleri Date: Mon, 21 Nov 2022 12:37:31 +0000 Subject: [PATCH] Dependency on govdpa v0.1.4 Signed-off-by: Leonardo Milleri --- go.mod | 4 +- go.sum | 12 +- .../govdpa/pkg/kvdpa/device.go | 338 ++++++++++++++++++ .../govdpa/pkg/kvdpa/kvdpa.go | 242 ------------- .../govdpa/pkg/kvdpa/mgmtdev.go | 111 ++++++ .../govdpa/pkg/kvdpa/netlink.go | 182 ++++++++++ .../govdpa/pkg/kvdpa/vhost.go | 62 ++++ .../govdpa/pkg/kvdpa/virtio.go | 68 ++++ .../pkg/types/types.go | 3 - .../pkg/utils/vdpa_provider.go | 18 +- vendor/modules.txt | 6 +- 11 files changed, 791 insertions(+), 255 deletions(-) create mode 100644 vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/device.go delete mode 100644 vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/kvdpa.go create mode 100644 vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/mgmtdev.go create mode 100644 vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/netlink.go create mode 100644 vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/vhost.go create mode 100644 vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/virtio.go diff --git a/go.mod b/go.mod index a109de23cd..80adab8792 100644 --- a/go.mod +++ b/go.mod @@ -13,7 +13,7 @@ require ( github.com/hashicorp/go-retryablehttp v0.7.0 github.com/jaypipes/ghw v0.9.0 github.com/k8snetworkplumbingwg/network-attachment-definition-client v1.1.1-0.20201119153432-9d213757d22d - github.com/k8snetworkplumbingwg/sriov-network-device-plugin v0.0.0-20220614121156-6fff085aed91 + github.com/k8snetworkplumbingwg/sriov-network-device-plugin v0.0.0-20221127172732-a5a7395122e3 github.com/onsi/ginkgo/v2 v2.5.0 github.com/onsi/gomega v1.24.0 github.com/openshift/api v0.0.0-20221220162201-efeef9d83325 @@ -95,7 +95,7 @@ require ( github.com/jaypipes/pcidb v1.0.0 // indirect github.com/josharian/intern v1.0.0 // indirect github.com/json-iterator/go v1.1.12 // indirect - github.com/k8snetworkplumbingwg/govdpa v0.1.3 // indirect + github.com/k8snetworkplumbingwg/govdpa v0.1.4 // indirect github.com/liggitt/tabwriter v0.0.0-20181228230101-89fcab3d43de // indirect github.com/mailru/easyjson v0.7.7 // indirect github.com/mattn/go-isatty v0.0.17 // indirect diff --git a/go.sum b/go.sum index 8e69125ab8..f3646f1453 100644 --- a/go.sum +++ b/go.sum @@ -364,12 +364,16 @@ github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHm github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU= github.com/jstemmer/go-junit-report v0.9.1/go.mod h1:Brl9GWCQeLvo8nXZwPNNblvFj/XSXhF0NWZEnDohbsk= github.com/jtolds/gls v4.2.1+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU= -github.com/k8snetworkplumbingwg/govdpa v0.1.3 h1:FZRhTMB1e3yWwSEy+l4eS73WioyMaL+vmFZ8JNwn+Uk= -github.com/k8snetworkplumbingwg/govdpa v0.1.3/go.mod h1:Jx2rlMquENdCd8M5Oc51xHCt10bQIXTloDU8F4nS4T4= +github.com/jtolds/gls v4.20.0+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU= +github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w= +github.com/julienschmidt/httprouter v1.3.0/go.mod h1:JR6WtHb+2LUe8TCKY3cZOxFyyO8IZAc4RVcycCCAKdM= +github.com/k8snetworkplumbingwg/govdpa v0.1.4 h1:e6mM7JFZkLVJeMQw3px96EigHAhnb4VUlqhNub/2Psk= +github.com/k8snetworkplumbingwg/govdpa v0.1.4/go.mod h1:UQR1xu7A+nnRK1dkLEi12OnNL0OiBPpIKOYDuaQQkck= github.com/k8snetworkplumbingwg/network-attachment-definition-client v1.1.1-0.20201119153432-9d213757d22d h1:9QbZltQGRFe7temwcTDjj8rIbow48Gv6mIKOxuks+OI= github.com/k8snetworkplumbingwg/network-attachment-definition-client v1.1.1-0.20201119153432-9d213757d22d/go.mod h1:+1DpV8uIwteAhxNO0lgRox8gHkTG6w3OeDfAlg+qqjA= -github.com/k8snetworkplumbingwg/sriov-network-device-plugin v0.0.0-20220614121156-6fff085aed91 h1:/030qHe5IK/TS/NVzGXTZQ0jh8B/P+M8zgOfNht+qoM= -github.com/k8snetworkplumbingwg/sriov-network-device-plugin v0.0.0-20220614121156-6fff085aed91/go.mod h1:jdfSQkcB9H8oWUFAd/6t5gkEj930WkLLtO3hFRv0fpg= +github.com/k8snetworkplumbingwg/sriov-network-device-plugin v0.0.0-20221127172732-a5a7395122e3 h1:hIHzF4vNTCFb9UMngrcJAMvdNslCekaSvNbfQt21CUw= +github.com/k8snetworkplumbingwg/sriov-network-device-plugin v0.0.0-20221127172732-a5a7395122e3/go.mod h1:b0YSmUuNOy6CkEmV27XfmZ3a7njs2pjxHoFAvfLbUII= +github.com/kisielk/errcheck v1.1.0/go.mod h1:EZBBE59ingxPouuu3KfxchcWSUPOHkagtvWXihfKN4Q= github.com/kisielk/errcheck v1.2.0/go.mod h1:/BMXB+zMLi60iA8Vv6Ksmxu/1UDYcXs4uQLJ+jE2L00= github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8= github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= diff --git a/vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/device.go b/vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/device.go new file mode 100644 index 0000000000..120b495704 --- /dev/null +++ b/vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/device.go @@ -0,0 +1,338 @@ +package kvdpa + +import ( + "errors" + "os" + "path/filepath" + "strings" + "syscall" + + "github.com/vishvananda/netlink/nl" + "golang.org/x/sys/unix" +) + +// Exported constants +const ( + VhostVdpaDriver = "vhost_vdpa" + VirtioVdpaDriver = "virtio_vdpa" +) + +// Private constants +const ( + vdpaBusDevDir = "/sys/bus/vdpa/devices" + vdpaVhostDevDir = "/dev" + rootDevDir = "/sys/devices" +) + +// VdpaDevice contains information about a Vdpa Device +type VdpaDevice interface { + Driver() string + Name() string + MgmtDev() MgmtDev + VirtioNet() VirtioNet + VhostVdpa() VhostVdpa + ParentDevicePath() (string, error) +} + +// vdpaDev implements VdpaDevice interface +type vdpaDev struct { + name string + driver string + mgmtDev *mgmtDev + virtioNet VirtioNet + vhostVdpa VhostVdpa +} + +// Driver resturns de device's driver name +func (vd *vdpaDev) Driver() string { + return vd.driver +} + +// Driver resturns de device's name +func (vd *vdpaDev) Name() string { + return vd.name +} + +// MgmtDev returns the device's management device +func (vd *vdpaDev) MgmtDev() MgmtDev { + return vd.mgmtDev +} + +// VhostVdpa returns the VhostVdpa device information associated +// or nil if the device is not bound to the vhost_vdpa driver +func (vd *vdpaDev) VhostVdpa() VhostVdpa { + return vd.vhostVdpa +} + +// Virtionet returns the VirtioNet device information associated +// or nil if the device is not bound to the virtio_vdpa driver +func (vd *vdpaDev) VirtioNet() VirtioNet { + return vd.virtioNet +} + +// getBusInfo populates the vdpa bus information +// the vdpa device must have at least the name prepopulated +func (vd *vdpaDev) getBusInfo() error { + driverLink, err := os.Readlink(filepath.Join(vdpaBusDevDir, vd.name, "driver")) + if err != nil { + // No error if driver is not present. The device is simply not bound to any. + return nil + } + + vd.driver = filepath.Base(driverLink) + + switch vd.driver { + case VhostVdpaDriver: + vd.vhostVdpa, err = vd.getVhostVdpaDev() + if err != nil { + return err + } + case VirtioVdpaDriver: + vd.virtioNet, err = vd.getVirtioVdpaDev() + if err != nil { + return err + } + } + + return nil +} + +// parseAttributes populates the vdpa device information from netlink attributes +func (vd *vdpaDev) parseAttributes(attrs []syscall.NetlinkRouteAttr) error { + mgmtDev := &mgmtDev{} + for _, a := range attrs { + switch a.Attr.Type { + case VdpaAttrDevName: + vd.name = string(a.Value[:len(a.Value)-1]) + case VdpaAttrMgmtDevBusName: + mgmtDev.busName = string(a.Value[:len(a.Value)-1]) + case VdpaAttrMgmtDevDevName: + mgmtDev.devName = string(a.Value[:len(a.Value)-1]) + } + } + vd.mgmtDev = mgmtDev + return nil +} + +/* Finds the vhost vdpa device of a vdpa device and returns it's path */ +func (vd *vdpaDev) getVhostVdpaDev() (VhostVdpa, error) { + // vhost vdpa devices live in the vdpa device's path + path := filepath.Join(vdpaBusDevDir, vd.name) + return GetVhostVdpaDevInPath(path) +} + +/* ParentDevice returns the path of the parent device (e.g: PCI) of the device */ +func (vd *vdpaDev) ParentDevicePath() (string, error) { + vdpaDevicePath := filepath.Join(vdpaBusDevDir, vd.name) + + /* For pci devices we have: + /sys/bud/vdpa/devices/vdpaX -> + ../../../devices/pci0000:00/.../0000:05:00:1/vdpaX + + Resolving the symlinks should give us the parent PCI device. + */ + devicePath, err := filepath.EvalSymlinks(vdpaDevicePath) + if err != nil { + return "", err + } + + /* If the parent device is the root device /sys/devices, there is + no parent (e.g: vdpasim). + */ + parent := filepath.Dir(devicePath) + if parent == rootDevDir { + return devicePath, nil + } + + return parent, nil +} + +/* + Finds the virtio vdpa device of a vdpa device and returns its path + +Currently, PCI-based devices have the following sysfs structure: +/sys/bus/vdpa/devices/ + + vdpa1 -> ../../../devices/pci0000:00/0000:00:03.2/0000:05:00.2/vdpa1 + +In order to find the virtio device we look for virtio* devices inside the parent device: + + sys/devices/pci0000:00/0000:00:03.2/0000:05:00.2/virtio{N} + +We also check the virtio device exists in the virtio bus: +/sys/bus/virtio/devices + + virtio{N} -> ../../../devices/pci0000:00/0000:00:03.2/0000:05:00.2/virtio{N} +*/ +func (vd *vdpaDev) getVirtioVdpaDev() (VirtioNet, error) { + parentPath, err := vd.ParentDevicePath() + if err != nil { + return nil, err + } + return GetVirtioNetInPath(parentPath) +} + +/*GetVdpaDevice returns the vdpa device information by a vdpa device name */ +func GetVdpaDevice(name string) (VdpaDevice, error) { + nameAttr, err := GetNetlinkOps().NewAttribute(VdpaAttrDevName, name) + if err != nil { + return nil, err + } + + msgs, err := GetNetlinkOps(). + RunVdpaNetlinkCmd(VdpaCmdDevGet, 0, []*nl.RtAttr{nameAttr}) + if err != nil { + return nil, err + } + + vdpaDevs, err := parseDevLinkVdpaDevList(msgs) + if err != nil { + return nil, err + } + return vdpaDevs[0], nil +} + +/* +GetVdpaDevicesByMgmtDev returns the VdpaDevice objects whose MgmtDev +has the given bus and device names. +*/ +func GetVdpaDevicesByMgmtDev(busName, devName string) ([]VdpaDevice, error) { + result := []VdpaDevice{} + devices, err := ListVdpaDevices() + if err != nil { + return nil, err + } + for _, device := range devices { + if device.MgmtDev() != nil && + device.MgmtDev().BusName() == busName && + device.MgmtDev().DevName() == devName { + result = append(result, device) + } + } + if len(result) == 0 { + return nil, syscall.ENODEV + } + return result, nil +} + +/*ListVdpaDevices returns a list of all available vdpa devices */ +func ListVdpaDevices() ([]VdpaDevice, error) { + msgs, err := GetNetlinkOps().RunVdpaNetlinkCmd(VdpaCmdDevGet, syscall.NLM_F_DUMP, nil) + if err != nil { + return nil, err + } + + vdpaDevs, err := parseDevLinkVdpaDevList(msgs) + if err != nil { + return nil, err + } + return vdpaDevs, nil +} + +func extractBusNameAndMgmtDeviceName(fullMgmtDeviceName string) (busName string, mgmtDeviceName string, err error) { + numSlashes := strings.Count(fullMgmtDeviceName, "/") + if numSlashes > 1 { + return "", "", errors.New("expected mgmtDeviceName to be either in the format / or ") + } else if numSlashes == 0 { + return "", fullMgmtDeviceName, nil + } else { + values := strings.Split(fullMgmtDeviceName, "/") + return values[0], values[1], nil + } +} + +/* +GetVdpaDevicesByPciAddress returns the VdpaDevice objects for the given pciAddress + + The pciAddress must have one of the following formats: + - MgmtBusName/MgmtDevName + - MgmtDevName +*/ +func GetVdpaDevicesByPciAddress(pciAddress string) ([]VdpaDevice, error) { + busName, mgmtDeviceName, err := extractBusNameAndMgmtDeviceName(pciAddress) + if err != nil { + return nil, unix.EINVAL + } + + return GetVdpaDevicesByMgmtDev(busName, mgmtDeviceName) +} + +/*AddVdpaDevice adds a new vdpa device to the given management device */ +func AddVdpaDevice(mgmtDeviceName string, vdpaDeviceName string) error { + if mgmtDeviceName == "" || vdpaDeviceName == "" { + return unix.EINVAL + } + + busName, mgmtDeviceName, err := extractBusNameAndMgmtDeviceName(mgmtDeviceName) + if err != nil { + return unix.EINVAL + } + + var attributes []*nl.RtAttr + var busNameAttr *nl.RtAttr + if busName != "" { + busNameAttr, err = GetNetlinkOps().NewAttribute(VdpaAttrMgmtDevBusName, busName) + if err != nil { + return err + } + attributes = append(attributes, busNameAttr) + } + + mgmtAttr, err := GetNetlinkOps().NewAttribute(VdpaAttrMgmtDevDevName, mgmtDeviceName) + if err != nil { + return err + } + attributes = append(attributes, mgmtAttr) + + nameAttr, err := GetNetlinkOps().NewAttribute(VdpaAttrDevName, vdpaDeviceName) + if err != nil { + return err + } + attributes = append(attributes, nameAttr) + + _, err = GetNetlinkOps().RunVdpaNetlinkCmd(VdpaCmdDevNew, unix.NLM_F_ACK|unix.NLM_F_REQUEST, attributes) + if err != nil { + return err + } + + return nil +} + +/*DeleteVdpaDevice deletes a vdpa device */ +func DeleteVdpaDevice(vdpaDeviceName string) error { + if vdpaDeviceName == "" { + return unix.EINVAL + } + + nameAttr, err := GetNetlinkOps().NewAttribute(VdpaAttrDevName, vdpaDeviceName) + if err != nil { + return err + } + + _, err = GetNetlinkOps().RunVdpaNetlinkCmd(VdpaCmdDevDel, unix.NLM_F_ACK|unix.NLM_F_REQUEST, []*nl.RtAttr{nameAttr}) + if err != nil { + return err + } + + return nil +} + +func parseDevLinkVdpaDevList(msgs [][]byte) ([]VdpaDevice, error) { + devices := make([]VdpaDevice, 0, len(msgs)) + + for _, m := range msgs { + attrs, err := nl.ParseRouteAttr(m[nl.SizeofGenlmsg:]) + if err != nil { + return nil, err + } + dev := &vdpaDev{} + if err = dev.parseAttributes(attrs); err != nil { + return nil, err + } + if err = dev.getBusInfo(); err != nil { + return nil, err + } + devices = append(devices, dev) + } + return devices, nil +} diff --git a/vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/kvdpa.go b/vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/kvdpa.go deleted file mode 100644 index 24471adc26..0000000000 --- a/vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/kvdpa.go +++ /dev/null @@ -1,242 +0,0 @@ -package kvdpa - -import ( - "fmt" - "io/ioutil" - "os" - "path/filepath" - "strings" -) - -/*Exported constants */ -const ( - VhostVdpaDriver = "vhost_vdpa" - VirtioVdpaDriver = "virtio_vdpa" -) - -/*Private constants */ -const ( - vdpaBusDevDir = "/sys/bus/vdpa/devices" - pciBusDevDir = "/sys/bus/pci/devices" - vdpaVhostDevDir = "/dev" - virtioDevDir = "/sys/bus/virtio/devices" - rootDevDir = "/sys/devices" -) - -/*VdpaDevice contains information about a Vdpa Device*/ -type VdpaDevice interface { - GetDriver() string - GetParent() string - GetPath() string - GetNetDev() string -} - -/*vdpaDevimplements VdpaDevice interface */ -type vdpaDev struct { - name string - driver string - path string // Path of the vhost or virtio device - netdev string // VirtioNet netdev (only for virtio-vdpa devices) -} - -func (vd *vdpaDev) GetDriver() string { - return vd.driver -} - -func (vd *vdpaDev) GetParent() string { - return vd.name -} - -func (vd *vdpaDev) GetPath() string { - return vd.path -} - -func (vd *vdpaDev) GetNetDev() string { - return vd.netdev -} - -/*GetVdpaDeviceList returns a list of all available vdpa devices */ -func GetVdpaDeviceList() ([]VdpaDevice, error) { - vdpaDevList := make([]VdpaDevice, 0) - fd, err := os.Open(vdpaBusDevDir) - if err != nil { - return nil, err - } - defer fd.Close() - - fileInfos, err := fd.Readdir(-1) - if err != nil { - return nil, err - } - var errors []string - for _, file := range fileInfos { - if vdpaDev, err := GetVdpaDeviceByName(file.Name()); err != nil { - errors = append(errors, err.Error()) - } else { - vdpaDevList = append(vdpaDevList, vdpaDev) - } - } - - if len(errors) > 0 { - return vdpaDevList, fmt.Errorf(strings.Join(errors, ";")) - } - return vdpaDevList, nil -} - -/*GetVdpaDeviceByName returns the vdpa device information by a vdpa device name */ -func GetVdpaDeviceByName(name string) (VdpaDevice, error) { - var err error - var path string - var netdev string - - driverLink, err := os.Readlink(filepath.Join(vdpaBusDevDir, name, "driver")) - if err != nil { - return nil, err - } - - driver := filepath.Base(driverLink) - switch driver { - case VhostVdpaDriver: - path, err = getVhostVdpaDev(name) - if err != nil { - return nil, err - } - case VirtioVdpaDriver: - path, err = getVirtioVdpaDev(name) - if err != nil { - return nil, err - } - virtioNetDir := filepath.Join(path, "net") - netDeviceFiles, err := ioutil.ReadDir(virtioNetDir) - if err != nil || len(netDeviceFiles) != 1 { - return nil, fmt.Errorf("failed to get network device name from vdpa device in %v %v", name, err) - } - netdev = strings.TrimSpace(netDeviceFiles[0].Name()) - default: - return nil, fmt.Errorf("Unknown vdpa bus driver %s", driver) - } - - return &vdpaDev{ - name: name, - driver: driver, - path: path, - netdev: netdev, - }, nil -} - -/* Finds the vhost vdpa device of a vdpa device and returns it's path */ -func getVhostVdpaDev(name string) (string, error) { - file := filepath.Join(vdpaBusDevDir, name) - fd, err := os.Open(file) - if err != nil { - return "", err - } - defer fd.Close() - - fileInfos, err := fd.Readdir(-1) - for _, file := range fileInfos { - if strings.Contains(file.Name(), "vhost-vdpa") && - file.IsDir() { - devicePath := filepath.Join(vdpaVhostDevDir, file.Name()) - info, err := os.Stat(devicePath) - if err != nil { - return "", err - } - if info.Mode()&os.ModeDevice == 0 { - return "", fmt.Errorf("vhost device %s is not a valid device", devicePath) - } - return devicePath, nil - } - } - return "", fmt.Errorf("vhost device not found for vdpa device %s", name) -} - -/*GetVdpaDeviceByPci returns the vdpa device information corresponding to a PCI device*/ -/* Based on the following directory hiearchy: -/sys/bus/pci/devices/{PCIDev}/ - /vdpa{N}/ - -/sys/bus/vdpa/devices/vdpa{N} -> ../../../devices/pci.../{PCIDev}/vdpa{N} -*/ -func GetVdpaDeviceByPci(pciAddr string) (VdpaDevice, error) { - path, err := filepath.EvalSymlinks(filepath.Join(pciBusDevDir, pciAddr)) - if err != nil { - return nil, err - } - fd, err := os.Open(path) - if err != nil { - return nil, err - } - defer fd.Close() - - fileInfos, err := fd.Readdir(-1) - for _, file := range fileInfos { - if strings.Contains(file.Name(), "vdpa") { - parent, err := getParentDevice(filepath.Join(vdpaBusDevDir, file.Name())) - if err != nil { - return nil, err - } - if parent != path { - return nil, fmt.Errorf("vdpa device %s parent (%s) does not match containing dir (%s)", - file.Name(), parent, path) - } - return GetVdpaDeviceByName(file.Name()) - } - } - return nil, fmt.Errorf("PCI address %s does not contain a vdpa device", pciAddr) -} - -/* Finds the virtio vdpa device of a vdpa device and returns it's path -Currently, PCI-based devices have the following sysfs structure: -/sys/bus/vdpa/devices/ - vdpa1 -> ../../../devices/pci0000:00/0000:00:03.2/0000:05:00.2/vdpa1 - -In order to find the virtio device we look for virtio* devices inside the parent device: - sys/devices/pci0000:00/0000:00:03.2/0000:05:00.2/virtio{N} - -We also check the virtio device exists in the virtio bus: -/sys/bus/virtio/devices - virtio{N} -> ../../../devices/pci0000:00/0000:00:03.2/0000:05:00.2/virtio{N} -*/ -func getVirtioVdpaDev(name string) (string, error) { - vdpaDevicePath := filepath.Join(vdpaBusDevDir, name) - parentPath, err := getParentDevice(vdpaDevicePath) - if err != nil { - return "", err - } - - fd, err := os.Open(parentPath) - if err != nil { - return "", err - } - defer fd.Close() - - fileInfos, err := fd.Readdir(-1) - for _, file := range fileInfos { - if strings.Contains(file.Name(), "virtio") && - file.IsDir() { - virtioDevPath := filepath.Join(virtioDevDir, file.Name()) - if _, err := os.Stat(virtioDevPath); os.IsNotExist(err) { - return "", fmt.Errorf("virtio device %s does not exist", virtioDevPath) - } - return virtioDevPath, nil - } - } - - return "", fmt.Errorf("virtio device not found for vdpa device %s", name) -} - -/* getParentDevice returns the parent's path of a vdpa device path */ -func getParentDevice(path string) (string, error) { - devicePath, err := filepath.EvalSymlinks(path) - if err != nil { - return "", err - } - - parent := filepath.Dir(devicePath) - // if the "parent" is sys/devices, we have reached the "root" device - if parent == rootDevDir { - return devicePath, nil - } - return parent, nil -} diff --git a/vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/mgmtdev.go b/vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/mgmtdev.go new file mode 100644 index 0000000000..dd9e9b7513 --- /dev/null +++ b/vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/mgmtdev.go @@ -0,0 +1,111 @@ +package kvdpa + +import ( + "strings" + "syscall" + + "github.com/vishvananda/netlink/nl" +) + +// MgmtDev represents a Vdpa Management Device +type MgmtDev interface { + BusName() string // Optional + DevName() string // + Name() string // The MgmtDevName is BusName/DevName +} + +type mgmtDev struct { + busName string + devName string +} + +// BusName returns the MgmtDev's bus name +func (m *mgmtDev) BusName() string { + return m.busName +} + +// BusName returns the MgmtDev's device name +func (m *mgmtDev) DevName() string { + return m.devName +} + +// BusName returns the MgmtDev's name: [BusName/]DeviceName +func (m *mgmtDev) Name() string { + if m.busName != "" { + return strings.Join([]string{m.busName, m.devName}, "/") + } + return m.devName +} + +// parseAttributes parses the netlink attributes and populates the fields accordingly +func (m *mgmtDev) parseAttributes(attrs []syscall.NetlinkRouteAttr) error { + for _, a := range attrs { + switch a.Attr.Type { + case VdpaAttrMgmtDevBusName: + m.busName = string(a.Value[:len(a.Value)-1]) + case VdpaAttrMgmtDevDevName: + m.devName = string(a.Value[:len(a.Value)-1]) + } + } + return nil +} + +// ListVdpaMgmtDevices returns the list of all available MgmtDevs +func ListVdpaMgmtDevices() ([]MgmtDev, error) { + msgs, err := GetNetlinkOps().RunVdpaNetlinkCmd(VdpaCmdMgmtDevGet, syscall.NLM_F_DUMP, nil) + if err != nil { + return nil, err + } + + mgtmDevs, err := parseDevLinkVdpaMgmtDevList(msgs) + if err != nil { + return nil, err + } + return mgtmDevs, nil +} + +// GetVdpaMgmtDevices returns a MgmtDev based on a busName and deviceName +func GetVdpaMgmtDevices(busName, devName string) (MgmtDev, error) { + data := []*nl.RtAttr{} + if busName != "" { + bus, err := GetNetlinkOps().NewAttribute(VdpaAttrMgmtDevBusName, busName) + if err != nil { + return nil, err + } + data = append(data, bus) + } + + dev, err := GetNetlinkOps().NewAttribute(VdpaAttrMgmtDevDevName, devName) + if err != nil { + return nil, err + } + data = append(data, dev) + + msgs, err := GetNetlinkOps().RunVdpaNetlinkCmd(VdpaCmdMgmtDevGet, 0, data) + if err != nil { + return nil, err + } + + mgtmDevs, err := parseDevLinkVdpaMgmtDevList(msgs) + if err != nil { + return nil, err + } + return mgtmDevs[0], nil +} + +func parseDevLinkVdpaMgmtDevList(msgs [][]byte) ([]MgmtDev, error) { + devices := make([]MgmtDev, 0, len(msgs)) + + for _, m := range msgs { + attrs, err := nl.ParseRouteAttr(m[nl.SizeofGenlmsg:]) + if err != nil { + return nil, err + } + dev := &mgmtDev{} + if err = dev.parseAttributes(attrs); err != nil { + return nil, err + } + devices = append(devices, dev) + } + return devices, nil +} diff --git a/vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/netlink.go b/vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/netlink.go new file mode 100644 index 0000000000..b5af17f6a6 --- /dev/null +++ b/vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/netlink.go @@ -0,0 +1,182 @@ +package kvdpa + +import ( + "fmt" + "syscall" + + "github.com/vishvananda/netlink" + "github.com/vishvananda/netlink/nl" +) + +/* Vdpa Netlink Name */ +const ( + VdpaGenlName = "vdpa" +) + +/* VDPA Netlink Commands */ +const ( + VdpaCmdUnspec uint8 = iota + VdpaCmdMgmtDevNew + VdpaCmdMgmtDevGet /* can dump */ + VdpaCmdDevNew + VdpaCmdDevDel + VdpaCmdDevGet /* can dump */ + VdpaCmdDevConfigGet /* can dump */ +) + +/* VDPA Netlink Attributes */ +const ( + VdpaAttrUnspec = iota + + /* bus name (optional) + dev name together make the parent device handle */ + VdpaAttrMgmtDevBusName /* string */ + VdpaAttrMgmtDevDevName /* string */ + VdpaAttrMgmtDevSupportedClasses /* u64 */ + + VdpaAttrDevName /* string */ + VdpaAttrDevID /* u32 */ + VdpaAttrDevVendorID /* u32 */ + VdpaAttrDevMaxVqs /* u32 */ + VdpaAttrDevMaxVqSize /* u16 */ + VdpaAttrDevMinVqSize /* u16 */ + + VdpaAttrDevNetCfgMacAddr /* binary */ + VdpaAttrDevNetStatus /* u8 */ + VdpaAttrDevNetCfgMaxVqp /* u16 */ + VdpaAttrGetNetCfgMTU /* u16 */ + + /* new attributes must be added above here */ + VdpaAttrMax +) + +var ( + commonNetlinkFlags = syscall.NLM_F_REQUEST | syscall.NLM_F_ACK +) + +// NetlinkOps defines the Netlink Operations +type NetlinkOps interface { + RunVdpaNetlinkCmd(command uint8, flags int, data []*nl.RtAttr) ([][]byte, error) + NewAttribute(attrType int, data interface{}) (*nl.RtAttr, error) +} + +type defaultNetlinkOps struct { +} + +var netlinkOps NetlinkOps = &defaultNetlinkOps{} + +// SetNetlinkOps method would be used by unit tests +func SetNetlinkOps(mockInst NetlinkOps) { + netlinkOps = mockInst +} + +// GetNetlinkOps will be invoked by functions in other packages that would need access to the sriovnet library methods. +func GetNetlinkOps() NetlinkOps { + return netlinkOps +} + +// RunVdpaNerlinkCmd runs a vdpa netlink command and returns the response +func (defaultNetlinkOps) RunVdpaNetlinkCmd(command uint8, flags int, data []*nl.RtAttr) ([][]byte, error) { + f, err := netlink.GenlFamilyGet(VdpaGenlName) + if err != nil { + return nil, err + } + + msg := &nl.Genlmsg{ + Command: command, + Version: nl.GENL_CTRL_VERSION, + } + req := nl.NewNetlinkRequest(int(f.ID), commonNetlinkFlags|flags) + + req.AddData(msg) + for _, d := range data { + req.AddData(d) + } + + msgs, err := req.Execute(syscall.NETLINK_GENERIC, 0) + if err != nil { + return nil, err + } + return msgs, nil +} + +// NewAttribute returns a new netlink attribute based on the provided data +func (defaultNetlinkOps) NewAttribute(attrType int, data interface{}) (*nl.RtAttr, error) { + switch attrType { + case VdpaAttrMgmtDevBusName, VdpaAttrMgmtDevDevName, VdpaAttrDevName: + strData, ok := data.(string) + if !ok { + return nil, fmt.Errorf("attribute type %d requires string data", attrType) + } + bytes := make([]byte, len(strData)+1) + copy(bytes, strData) + return nl.NewRtAttr(attrType, bytes), nil + /* TODO + case: + VdpaAttrMgmtDevBusName string + VdpaAttrMgmtDevDevName string + VdpaAttrMgmtDevSupportedClasses u64 + + VdpaAttrDevName string + VdpaAttrDevID u32 + VdpaAttrDevVendorID u32 + VdpaAttrDevMaxVqs u32 + VdpaAttrDevMaxVqSize u16 + VdpaAttrDevMinVqSize u16 + + VdpaAttrDevNetCfgMacAddr binary + VdpaAttrDevNetStatus u8 + VdpaAttrDevNetCfgMaxVqp u16 + VdpaAttrGetNetCfgMTU u16 + */ + default: + return nil, fmt.Errorf("invalid attribute type %d", attrType) + } + +} + +func newMockSingleMessage(command uint8, attrs []*nl.RtAttr) []byte { + b := make([]byte, 0) + dataBytes := make([][]byte, len(attrs)+1) + + msg := &nl.Genlmsg{ + Command: command, + Version: nl.GENL_CTRL_VERSION, + } + dataBytes[0] = msg.Serialize() + + for i, attr := range attrs { + dataBytes[i+1] = attr.Serialize() + } + next := 0 + for _, data := range dataBytes { + for _, dataByte := range data { + b = append(b, dataByte) + next = next + 1 + } + } + return b + /* + nlm := &nl.NetlinkRequest{ + NlMsghdr: unix.NlMsghdr{ + Len: uint32(unix.SizeofNlMsghdr), + Type: 0xa, + Flags: 0, + Seq: 1, + }, + } + for _, a := range attrs { + nlm.AddData(a) + } + return nlm.Serialize() + */ +} + +// Used for unit tests +func newMockNetLinkResponse(command uint8, data [][]*nl.RtAttr) [][]byte { + msgs := make([][]byte, len(data)) + for i, msgData := range data { + msgDataBytes := newMockSingleMessage(command, msgData) + msgs[i] = msgDataBytes + } + return msgs +} diff --git a/vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/vhost.go b/vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/vhost.go new file mode 100644 index 0000000000..5067dcf9e1 --- /dev/null +++ b/vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/vhost.go @@ -0,0 +1,62 @@ +package kvdpa + +import ( + "fmt" + "os" + "path/filepath" + "strings" +) + +// VhostVdpa is the vhost-vdpa device information +type VhostVdpa interface { + Name() string + Path() string +} + +// vhostVdpa implements VhostVdpa interface +type vhostVdpa struct { + name string + path string +} + +// Name returns the vhost device's name +func (v *vhostVdpa) Name() string { + return v.name +} + +// Name returns the vhost device's path +func (v *vhostVdpa) Path() string { + return v.path +} + +// GetVhostVdpaDevInPath returns the VhostVdpa found in the provided parent device's path +func GetVhostVdpaDevInPath(parentPath string) (VhostVdpa, error) { + fd, err := os.Open(parentPath) + if err != nil { + return nil, err + } + defer fd.Close() + + fileInfos, err := fd.Readdir(-1) + if err != nil { + return nil, err + } + for _, file := range fileInfos { + if strings.Contains(file.Name(), "vhost-vdpa") && + file.IsDir() { + devicePath := filepath.Join(vdpaVhostDevDir, file.Name()) + info, err := os.Stat(devicePath) + if err != nil { + return nil, err + } + if info.Mode()&os.ModeDevice == 0 { + return nil, fmt.Errorf("vhost device %s is not a valid device", devicePath) + } + return &vhostVdpa{ + name: file.Name(), + path: devicePath, + }, nil + } + } + return nil, fmt.Errorf("no VhostVdpa device foiund in path %s", parentPath) +} diff --git a/vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/virtio.go b/vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/virtio.go new file mode 100644 index 0000000000..1d99d5518d --- /dev/null +++ b/vendor/github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa/virtio.go @@ -0,0 +1,68 @@ +package kvdpa + +import ( + "fmt" + "os" + "path/filepath" + "strings" +) + +const ( + virtioDevDir = "/sys/bus/virtio/devices" +) + +// VirtioNet is the virtio-net device information +type VirtioNet interface { + Name() string + NetDev() string +} + +// virtioNet implements VirtioNet interface +type virtioNet struct { + name string + netDev string +} + +// Name returns the virtio device's name (as appears in the virtio bus) +func (v *virtioNet) Name() string { + return v.name +} + +// NetDev returns the virtio-net netdev name +func (v *virtioNet) NetDev() string { + return v.netDev +} + +// GetVirtioNetInPath returns the VirtioNet found in the provided parent device's path +func GetVirtioNetInPath(parentPath string) (VirtioNet, error) { + fd, err := os.Open(parentPath) + if err != nil { + return nil, err + } + defer fd.Close() + + fileInfos, err := fd.Readdir(-1) + if err != nil { + return nil, err + } + for _, file := range fileInfos { + if strings.Contains(file.Name(), "virtio") && + file.IsDir() { + virtioDevPath := filepath.Join(virtioDevDir, file.Name()) + if _, err := os.Stat(virtioDevPath); os.IsNotExist(err) { + return nil, fmt.Errorf("virtio device %s does not exist", virtioDevPath) + } + var netdev string + // Read the "net" directory in the virtio device path + netDeviceFiles, err := os.ReadDir(filepath.Join(virtioDevPath, "net")) + if err == nil && len(netDeviceFiles) == 1 { + netdev = strings.TrimSpace(netDeviceFiles[0].Name()) + } + return &virtioNet{ + name: file.Name(), + netDev: netdev, + }, nil + } + } + return nil, fmt.Errorf("no VirtioNet device found in path %s", parentPath) +} diff --git a/vendor/github.com/k8snetworkplumbingwg/sriov-network-device-plugin/pkg/types/types.go b/vendor/github.com/k8snetworkplumbingwg/sriov-network-device-plugin/pkg/types/types.go index aa15e5f4dd..69d1eecd71 100644 --- a/vendor/github.com/k8snetworkplumbingwg/sriov-network-device-plugin/pkg/types/types.go +++ b/vendor/github.com/k8snetworkplumbingwg/sriov-network-device-plugin/pkg/types/types.go @@ -184,14 +184,11 @@ type PciDevice interface { GetDeviceCode() string GetPciAddr() string GetPfPciAddr() string - IsSriovPF() bool - GetSubClass() string GetDeviceSpecs() []*pluginapi.DeviceSpec GetEnvVal() string GetMounts() []*pluginapi.Mount GetAPIDevice() *pluginapi.Device GetVFID() int - GetNumaInfo() string } // PciNetDevice extends PciDevice interface diff --git a/vendor/github.com/k8snetworkplumbingwg/sriov-network-device-plugin/pkg/utils/vdpa_provider.go b/vendor/github.com/k8snetworkplumbingwg/sriov-network-device-plugin/pkg/utils/vdpa_provider.go index 49366df46c..3c48c3961d 100644 --- a/vendor/github.com/k8snetworkplumbingwg/sriov-network-device-plugin/pkg/utils/vdpa_provider.go +++ b/vendor/github.com/k8snetworkplumbingwg/sriov-network-device-plugin/pkg/utils/vdpa_provider.go @@ -1,6 +1,9 @@ package utils import ( + "fmt" + + "github.com/golang/glog" vdpa "github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa" ) @@ -25,5 +28,18 @@ func GetVdpaProvider() VdpaProvider { } func (defaultVdpaProvider) GetVdpaDeviceByPci(pciAddr string) (vdpa.VdpaDevice, error) { - return vdpa.GetVdpaDeviceByPci(pciAddr) + // the govdpa library requires the pci address to include the "pci/" prefix + fullPciAddr := "pci/" + pciAddr + vdpaDevices, err := vdpa.GetVdpaDevicesByPciAddress(fullPciAddr) + if err != nil { + return nil, err + } + numVdpaDevices := len(vdpaDevices) + if numVdpaDevices == 0 { + return nil, fmt.Errorf("no vdpa device associated to pciAddress %s", pciAddr) + } + if numVdpaDevices > 1 { + glog.Infof("More than one vDPA device found for pciAddress %s, returning the first one", pciAddr) + } + return vdpaDevices[0], nil } diff --git a/vendor/modules.txt b/vendor/modules.txt index c2f219a51a..cf39f991ba 100644 --- a/vendor/modules.txt +++ b/vendor/modules.txt @@ -284,14 +284,14 @@ github.com/josharian/intern # github.com/json-iterator/go v1.1.12 ## explicit; go 1.12 github.com/json-iterator/go -# github.com/k8snetworkplumbingwg/govdpa v0.1.3 -## explicit; go 1.14 +# github.com/k8snetworkplumbingwg/govdpa v0.1.4 +## explicit; go 1.17 github.com/k8snetworkplumbingwg/govdpa/pkg/kvdpa # github.com/k8snetworkplumbingwg/network-attachment-definition-client v1.1.1-0.20201119153432-9d213757d22d ## explicit; go 1.12 github.com/k8snetworkplumbingwg/network-attachment-definition-client/pkg/apis/k8s.cni.cncf.io github.com/k8snetworkplumbingwg/network-attachment-definition-client/pkg/apis/k8s.cni.cncf.io/v1 -# github.com/k8snetworkplumbingwg/sriov-network-device-plugin v0.0.0-20220614121156-6fff085aed91 +# github.com/k8snetworkplumbingwg/sriov-network-device-plugin v0.0.0-20221127172732-a5a7395122e3 ## explicit; go 1.17 github.com/k8snetworkplumbingwg/sriov-network-device-plugin/pkg/types github.com/k8snetworkplumbingwg/sriov-network-device-plugin/pkg/utils