On Mon, May 24, 2010 at 05:22:15PM -0700, Scott Feldman wrote: > From: Scott Feldman <scofeldm@xxxxxxxxx> > > This patch builds on the work recently posted by Stefan Berger. It builds > on top of Stefan's three posted patches: > > [PATCH v9] vepa: parsing for 802.1Qb{g|h} XML > [RFC][PATCH 1/3] vepa+vsi: Introduce dependency on libnl > [PATCH v3] Add host UUID (to libvirt capabilities) > > Stefan's RFC patches 2/3 and 3/3 are incorporated into my patch. > > Changes from v3 to v4: > - move from Stafan's 802.1Qb{g|h} XML v8 to v9 > - move hostuuid and vf index calcs to inside doPortProfileOp8021Qbh I don't have the latest if_link.h on my system, so I had to make the changes in the attached patch to get v4 to build and pass make syntax-check; nothing major. The biggest thing is that getPhysfn is only called when IFLA_VF_PORT_MAX is defined, so I moved the entire function into that #ifdef and removed the internal #ifdef; otherwise the compiler complains about the unused static function. Dave > Changes from v2 to v3: > - remove debug fprintfs > - use virGetHostUUID (thanks Stefan!) > - fix compile issue when latest if_link.h isn't available > - change poll timeout to 10s, at 1/8 intervals > - if polling times out, log msg and return -ETIMEDOUT > > Changes from v1 to v2: > - Add Stefan's code for getPortProfileStatus > - Poll for up to 2 secs for port-profile status, at 1/8 sec intervals: > - if status indicates error, abort openMacvtapTap > - if status indicates success, exit polling > - if status is "in-progress" after 2 secs of polling, exit > polling loop silently, without error > > My patch finishes out the 802.1Qbh parts, which Stefan had mostly complete. > I've tested using the recent kernel updates for VF_PORT netlink msgs and > enic for Cisco's 10G Ethernet NIC. I tested many VMs, each with several > direct interfaces, each configured with a port-profile per the XML. VM-to-VM, > and VM-to-external work as expected. VM-to-VM on same host (using same NIC) > works same as VM-to-VM where VMs are on diff hosts. I'm able to change > settings on the port-profile while the VM is running to change the virtual > port behaviour. For example, adjusting a QoS setting like rate limit. All > VMs with interfaces using that port-profile immediatly see the effect of the > change to the port-profile. > > I don't have a SR-IOV device to test so source dev is a non-SR-IOV device, > but most of the code paths include support for specifing the source dev and > VF index. We'll need to complete this by discovering the PF given the VF > linkdev. Once we have the PF, we'll also have the VF index. All this info- > mation is available from sysfs. > > Signed-off-by: Scott Feldman <scofeldm@xxxxxxxxx> > --- > configure.ac | 16 + > src/qemu/qemu_conf.c | 2 > src/qemu/qemu_driver.c | 4 > src/util/macvtap.c | 778 +++++++++++++++++++++++++++++++++++++++++++++++- > src/util/macvtap.h | 1 > 5 files changed, 785 insertions(+), 16 deletions(-) > > > diff --git a/configure.ac b/configure.ac > index 36ba703..885b0ae 100644 > --- a/configure.ac > +++ b/configure.ac > @@ -2005,13 +2005,26 @@ if test "$with_macvtap" != "no" ; then > fi > AM_CONDITIONAL([WITH_MACVTAP], [test "$with_macvtap" = "yes"]) > > +AC_TRY_COMPILE([ #include <sys/socket.h> > + #include <linux/rtnetlink.h> ], > + [ int x = IFLA_PORT_MAX; ], > + [ with_virtualport=yes ], > + [ with_virtualport=no ]) > +if test "$with_virtualport" = "yes"; then > + val=1 > +else > + val=0 > +fi > +AC_DEFINE_UNQUOTED([WITH_VIRTUALPORT], $val, [whether vsi vepa support is enabled]) > +AM_CONDITIONAL([WITH_VIRTUALPORT], [test "$with_virtualport" = "yes"]) > + > > dnl netlink library > > LIBNL_CFLAGS="" > LIBNL_LIBS="" > > -if test "$with_macvtap" = "yes"; then > +if test "$with_macvtap" = "yes" || "$with_virtualport" = "yes"; then > PKG_CHECK_MODULES([LIBNL], [libnl-1 >= $LIBNL_REQUIRED], [ > ], [ > AC_MSG_ERROR([libnl >= $LIBNL_REQUIRED is required for macvtap support]) > @@ -2084,6 +2097,7 @@ AC_MSG_NOTICE([ Network: $with_network]) > AC_MSG_NOTICE([Libvirtd: $with_libvirtd]) > AC_MSG_NOTICE([ netcf: $with_netcf]) > AC_MSG_NOTICE([ macvtap: $with_macvtap]) > +AC_MSG_NOTICE([virtport: $with_virtualport]) > AC_MSG_NOTICE([]) > AC_MSG_NOTICE([Storage Drivers]) > AC_MSG_NOTICE([]) > diff --git a/src/qemu/qemu_conf.c b/src/qemu/qemu_conf.c > index 111fa6e..95d4c1a 100644 > --- a/src/qemu/qemu_conf.c > +++ b/src/qemu/qemu_conf.c > @@ -1505,7 +1505,7 @@ qemudPhysIfaceConnect(virConnectPtr conn, > if (err) { > close(rc); > rc = -1; > - delMacvtap(net->ifname, > + delMacvtap(net->ifname, net->data.direct.linkdev, > &net->data.direct.virtPortProfile); > } > } > diff --git a/src/qemu/qemu_driver.c b/src/qemu/qemu_driver.c > index f02bf3b..f1a0d0e 100644 > --- a/src/qemu/qemu_driver.c > +++ b/src/qemu/qemu_driver.c > @@ -3679,7 +3679,7 @@ static void qemudShutdownVMDaemon(struct qemud_driver *driver, > for (i = 0; i < def->nnets; i++) { > virDomainNetDefPtr net = def->nets[i]; > if (net->type == VIR_DOMAIN_NET_TYPE_DIRECT) > - delMacvtap(net->ifname, > + delMacvtap(net->ifname, net->data.direct.linkdev, > &net->data.direct.virtPortProfile); > } > #endif > @@ -8369,7 +8369,7 @@ qemudDomainDetachNetDevice(struct qemud_driver *driver, > > #if WITH_MACVTAP > if (detach->type == VIR_DOMAIN_NET_TYPE_DIRECT) > - delMacvtap(detach->ifname, > + delMacvtap(detach->ifname, detach->data.direct.linkdev, > &detach->data.direct.virtPortProfile); > #endif > > diff --git a/src/util/macvtap.c b/src/util/macvtap.c > index 5cbd02b..d5a08d9 100644 > --- a/src/util/macvtap.c > +++ b/src/util/macvtap.c > @@ -27,7 +27,7 @@ > > #include <config.h> > > -#if WITH_MACVTAP > +#if WITH_MACVTAP || WITH_VIRTUALPORT > > # include <stdio.h> > # include <errno.h> > @@ -41,6 +41,8 @@ > # include <linux/rtnetlink.h> > # include <linux/if_tun.h> > > +# include <netlink/msg.h> > + > # include "util.h" > # include "memory.h" > # include "logging.h" > @@ -48,6 +50,7 @@ > # include "interface.h" > # include "conf/domain_conf.h" > # include "virterror_internal.h" > +# include "uuid.h" > > # define VIR_FROM_THIS VIR_FROM_NET > > @@ -58,15 +61,23 @@ > # define MACVTAP_NAME_PREFIX "macvtap" > # define MACVTAP_NAME_PATTERN "macvtap%d" > > +# define MICROSEC_PER_SEC (1000 * 1000) > + > > static int associatePortProfileId(const char *macvtap_ifname, > + const char *linkdev, > const virVirtualPortProfileDefPtr virtPort, > - int vf, > const unsigned char *vmuuid); > > static int disassociatePortProfileId(const char *macvtap_ifname, > + const char *linkdev, > const virVirtualPortProfileDefPtr virtPort); > > +enum virVirtualPortOp { > + ASSOCIATE = 0x1, > + DISASSOCIATE = 0x2, > +}; > + > > static int nlOpen(void) > { > @@ -159,6 +170,156 @@ err_exit: > } > > > +# ifdef IFLA_VF_PORT_MAX > + > +/** > + * nlCommWaitSuccess: > + * > + * @nlmsg: pointer to netlink message > + * @nl_grousp: the netlink multicast groups to send to > + * @respbuf: pointer to pointer where response buffer will be allocated > + * @respbuflen: pointer to integer holding the size of the response buffer > + * on return of the function. > + * @to_usecs: timeout in microseconds to wait for a success message > + * to be returned > + * > + * Send the given message to the netlink multicast group and receive > + * responses. Skip responses indicating an error and keep on receiving > + * responses until a success response is returned. > + * Returns 0 on success, -1 on error. In case of error, no response > + * buffer will be returned. > + */ > +static int > +nlCommWaitSuccess(struct nlmsghdr *nlmsg, int nl_groups, > + char **respbuf, int *respbuflen, long to_usecs) > +{ > + int rc = 0; > + struct sockaddr_nl nladdr = { > + .nl_family = AF_NETLINK, > + .nl_pid = getpid(), > + .nl_groups = nl_groups, > + }; > + int rcvChunkSize = 1024; // expecting less than that > + int rcvoffset = 0; > + ssize_t nbytes; > + int n; > + struct timeval tv = { > + .tv_sec = to_usecs / MICROSEC_PER_SEC, > + .tv_usec = to_usecs % MICROSEC_PER_SEC, > + }; > + fd_set rfds; > + bool gotvalid = false; > + int fd = nlOpen(); > + static uint32_t seq = 0x1234; > + uint32_t myseq = seq++; > + uint32_t mypid = getpid(); > + > + if (fd < 0) > + return -1; > + > + nlmsg->nlmsg_pid = mypid; > + nlmsg->nlmsg_seq = myseq; > + nlmsg->nlmsg_flags |= NLM_F_ACK; > + > + nbytes = sendto(fd, (void *)nlmsg, nlmsg->nlmsg_len, 0, > + (struct sockaddr *)&nladdr, sizeof(nladdr)); > + if (nbytes < 0) { > + virReportSystemError(errno, > + "%s", _("cannot send to netlink socket")); > + rc = -1; > + goto err_exit; > + } > + > + while (!gotvalid) { > + rcvoffset = 0; > + while (1) { > + socklen_t addrlen = sizeof(nladdr); > + > + if (VIR_REALLOC_N(*respbuf, rcvoffset+rcvChunkSize) < 0) { > + virReportOOMError(); > + rc = -1; > + goto err_exit; > + } > + > + FD_ZERO(&rfds); > + FD_SET(fd, &rfds); > + > + n = select(fd + 1, &rfds, NULL, NULL, &tv); > + if (n == 0) { > + rc = -1; > + goto err_exit; > + } > + > + nbytes = recvfrom(fd, &((*respbuf)[rcvoffset]), rcvChunkSize, 0, > + (struct sockaddr *)&nladdr, &addrlen); > + if (nbytes < 0) { > + if (errno == EAGAIN || errno == EINTR) > + continue; > + virReportSystemError(errno, "%s", > + _("error receiving from netlink socket")); > + rc = -1; > + goto err_exit; > + } > + rcvoffset += nbytes; > + break; > + } > + *respbuflen = rcvoffset; > + > + /* check message for error */ > + if (*respbuflen > NLMSG_LENGTH(0) && *respbuf != NULL) { > + struct nlmsghdr *resp = (struct nlmsghdr *)*respbuf; > + struct nlmsgerr *err; > + > + if (resp->nlmsg_pid != mypid || > + resp->nlmsg_seq != myseq) > + continue; > + > + /* skip reflected message */ > + if (resp->nlmsg_type & 0x10) > + continue; > + > + switch (resp->nlmsg_type) { > + case NLMSG_ERROR: > + err = (struct nlmsgerr *)NLMSG_DATA(resp); > + if (resp->nlmsg_len >= NLMSG_LENGTH(sizeof(*err))) { > + if (-err->error != EOPNOTSUPP) { > + /* assuming error msg from daemon */ > + gotvalid = true; > + break; > + } > + } > + /* whatever this is, skip it */ > + VIR_FREE(*respbuf); > + *respbuf = NULL; > + *respbuflen = 0; > + break; > + > + case NLMSG_DONE: > + gotvalid = true; > + break; > + > + default: > + VIR_FREE(*respbuf); > + *respbuf = NULL; > + *respbuflen = 0; > + break; > + } > + } > + } > + > +err_exit: > + if (rc == -1) { > + VIR_FREE(*respbuf); > + *respbuf = NULL; > + *respbuflen = 0; > + } > + > + nlClose(fd); > + return rc; > +} > + > +#endif > + > static struct rtattr * > rtattrCreate(char *buffer, int bufsize, int type, > const void *data, int datalen) > @@ -204,6 +365,8 @@ nlAppend(struct nlmsghdr *nlm, int totlen, const void *data, int datalen) > } > > > +# if WITH_MACVTAP > + > static int > link_add(const char *type, > const unsigned char *macaddress, int macaddrsize, > @@ -655,8 +818,8 @@ create_name: > } > > if (associatePortProfileId(cr_ifname, > + linkdev, > virtPortProfile, > - -1, > vmuuid) != 0) { > rc = -1; > goto link_del_exit; > @@ -689,6 +852,7 @@ create_name: > > disassociate_exit: > disassociatePortProfileId(cr_ifname, > + linkdev, > virtPortProfile); > > link_del_exit: > @@ -701,6 +865,7 @@ link_del_exit: > /** > * delMacvtap: > * @ifname : The name of the macvtap interface > + * @linkdev: The interface name of the NIC to connect to the external bridge > * @virtPortProfile: pointer to object holding the virtual port profile data > * > * Delete an interface given its name. Disassociate > @@ -709,24 +874,603 @@ link_del_exit: > */ > void > delMacvtap(const char *ifname, > + const char *linkdev, > virVirtualPortProfileDefPtr virtPortProfile) > { > if (ifname) { > disassociatePortProfileId(ifname, > + linkdev, > virtPortProfile); > link_del(ifname); > } > } > > -#endif > +# endif > + > + > +# ifdef IFLA_PORT_MAX > + > +static struct nla_policy ifla_policy[IFLA_MAX + 1] = > +{ > + [IFLA_VF_PORTS] = { .type = NLA_NESTED }, > +}; > + > +static struct nla_policy ifla_vf_ports_policy[IFLA_VF_PORT_MAX + 1] = > +{ > + [IFLA_VF_PORT] = { .type = NLA_NESTED }, > +}; > + > +static struct nla_policy ifla_port_policy[IFLA_PORT_MAX + 1] = > +{ > + [IFLA_PORT_RESPONSE] = { .type = NLA_U16 }, > +}; > + > + > +static int > +link_dump(int ifindex, struct nlattr **tb, char **recvbuf) > +{ > + int rc = 0; > + char nlmsgbuf[256] = { 0, }; > + struct nlmsghdr *nlm = (struct nlmsghdr *)nlmsgbuf, *resp; > + struct nlmsgerr *err; > + struct ifinfomsg i = { > + .ifi_family = AF_UNSPEC, > + .ifi_index = ifindex > + }; > + int recvbuflen; > + > + *recvbuf = NULL; > + > + nlInit(nlm, NLM_F_REQUEST, RTM_GETLINK); > + > + if (!nlAppend(nlm, sizeof(nlmsgbuf), &i, sizeof(i))) > + goto buffer_too_small; > + > + if (nlComm(nlm, recvbuf, &recvbuflen) < 0) > + return -1; > + > + if (recvbuflen < NLMSG_LENGTH(0) || *recvbuf == NULL) > + goto malformed_resp; > > + resp = (struct nlmsghdr *)*recvbuf; > + > + switch (resp->nlmsg_type) { > + case NLMSG_ERROR: > + err = (struct nlmsgerr *)NLMSG_DATA(resp); > + if (resp->nlmsg_len < NLMSG_LENGTH(sizeof(*err))) > + goto malformed_resp; > + > + switch (-err->error) { > + case 0: > + break; > + > + default: > + virReportSystemError(-err->error, > + _("error dumping %d interface"), > + ifindex); > + rc = -1; > + } > + break; > + > + case GENL_ID_CTRL: > + case NLMSG_DONE: > + if (nlmsg_parse(resp, sizeof(struct ifinfomsg), > + tb, IFLA_MAX, ifla_policy)) { > + goto malformed_resp; > + } > + break; > + > + default: > + goto malformed_resp; > + } > + > + if (rc != 0) > + VIR_FREE(*recvbuf); > + > + return rc; > + > +malformed_resp: > + macvtapError(VIR_ERR_INTERNAL_ERROR, "%s", > + _("malformed netlink response message")); > + VIR_FREE(*recvbuf); > + return -1; > + > +buffer_too_small: > + macvtapError(VIR_ERR_INTERNAL_ERROR, "%s", > + _("internal buffer is too small")); > + return -1; > +} > + > + > +static int > +getPortProfileStatus(struct nlattr **tb, int32_t vf, uint16_t *status) > +{ > + int rc = 1; > + const char *msg = NULL; > + struct nlattr *tb2[IFLA_VF_PORT_MAX + 1], > + *tb3[IFLA_PORT_MAX+1]; > + > + if (vf == PORT_SELF_VF) { > + if (tb[IFLA_PORT_SELF]) { > + if (nla_parse_nested(tb3, IFLA_PORT_MAX, tb[IFLA_PORT_SELF], > + ifla_port_policy)) { > + msg = _("error parsing nested IFLA_VF_PORT part"); > + goto err_exit; > + } > + } > + } else { > + if (tb[IFLA_VF_PORTS]) { > + if (nla_parse_nested(tb2, IFLA_VF_PORT_MAX, tb[IFLA_VF_PORTS], > + ifla_vf_ports_policy)) { > + msg = _("error parsing nested IFLA_VF_PORTS part"); > + goto err_exit; > + } > + if (tb2[IFLA_VF_PORT]) { > + if (nla_parse_nested(tb3, IFLA_PORT_MAX, tb2[IFLA_VF_PORT], > + ifla_port_policy)) { > + msg = _("error parsing nested IFLA_VF_PORT part"); > + goto err_exit; > + } > + } > + } > + } > + > + if (tb3[IFLA_PORT_RESPONSE]) { > + *status = *(uint16_t *)RTA_DATA(tb3[IFLA_PORT_RESPONSE]); > + rc = 0; > + } else { > + msg = _("no IFLA_PORT_RESPONSE found in netlink message"); > + goto err_exit; > + } > + > +err_exit: > + if (msg) > + macvtapError(VIR_ERR_INTERNAL_ERROR, "%s", msg); > + > + return rc; > +} > + > + > +static int > +doPortProfileOpSetLink(bool multicast, > + int ifindex, > + const char *profileId, > + struct ifla_port_vsi *portVsi, > + const unsigned char *instanceId, > + const unsigned char *hostUUID, > + int32_t vf, > + uint8_t op) > +{ > + int rc = 0; > + char nlmsgbuf[256]; > + struct nlmsghdr *nlm = (struct nlmsghdr *)nlmsgbuf, *resp; > + struct nlmsgerr *err; > + char rtattbuf[64]; > + struct rtattr *rta, *vfports, *vfport; > + struct ifinfomsg ifinfo = { > + .ifi_family = AF_UNSPEC, > + .ifi_index = ifindex, > + }; > + char *recvbuf = NULL; > + int recvbuflen = 0; > + > + memset(&nlmsgbuf, 0, sizeof(nlmsgbuf)); > + > + nlInit(nlm, NLM_F_REQUEST, RTM_SETLINK); > + > + if (!nlAppend(nlm, sizeof(nlmsgbuf), &ifinfo, sizeof(ifinfo))) > + goto buffer_too_small; > + > + if (vf == PORT_SELF_VF) { > + rta = rtattrCreate(rtattbuf, sizeof(rtattbuf), IFLA_PORT_SELF, NULL, 0); > + } else { > + rta = rtattrCreate(rtattbuf, sizeof(rtattbuf), IFLA_VF_PORTS, NULL, 0); > + if (!rta) > + goto buffer_too_small; > + > + if (!(vfports = nlAppend(nlm, sizeof(nlmsgbuf), > + rtattbuf, rta->rta_len))) > + goto buffer_too_small; > + > + /* beging nesting vfports */ > + rta = rtattrCreate(rtattbuf, sizeof(rtattbuf), IFLA_VF_PORT, NULL, 0); > + } > + > + if (!rta) > + goto buffer_too_small; > + > + if (!(vfport = nlAppend(nlm, sizeof(nlmsgbuf), rtattbuf, rta->rta_len))) > + goto buffer_too_small; > + > + if (profileId) { > + rta = rtattrCreate(rtattbuf, sizeof(rtattbuf), IFLA_PORT_PROFILE, > + profileId, strlen(profileId) + 1); > + if (!rta) > + goto buffer_too_small; > + > + if (!nlAppend(nlm, sizeof(nlmsgbuf), rtattbuf, rta->rta_len)) > + goto buffer_too_small; > + } > + > + if (portVsi) { > + rta = rtattrCreate(rtattbuf, sizeof(rtattbuf), IFLA_PORT_VSI_TYPE, > + portVsi, sizeof(*portVsi)); > + if (!rta) > + goto buffer_too_small; > + > + if (!nlAppend(nlm, sizeof(nlmsgbuf), rtattbuf, rta->rta_len)) > + goto buffer_too_small; > + } > + > + if (instanceId) { > + rta = rtattrCreate(rtattbuf, sizeof(rtattbuf), IFLA_PORT_INSTANCE_UUID, > + instanceId, VIR_UUID_BUFLEN); > + if (!rta) > + goto buffer_too_small; > + > + if (!nlAppend(nlm, sizeof(nlmsgbuf), rtattbuf, rta->rta_len)) > + goto buffer_too_small; > + } > + > + if (hostUUID) { > + rta = rtattrCreate(rtattbuf, sizeof(rtattbuf), IFLA_PORT_HOST_UUID, > + hostUUID, VIR_UUID_BUFLEN); > + if (!rta) > + goto buffer_too_small; > + > + if (!nlAppend(nlm, sizeof(nlmsgbuf), rtattbuf, rta->rta_len)) > + goto buffer_too_small; > + } > + > + if (vf != PORT_SELF_VF) { > + rta = rtattrCreate(rtattbuf, sizeof(rtattbuf), IFLA_PORT_VF, > + &vf, sizeof(vf)); > + if (!rta) > + goto buffer_too_small; > + > + if (!nlAppend(nlm, sizeof(nlmsgbuf), rtattbuf, rta->rta_len)) > + goto buffer_too_small; > + } > + > + rta = rtattrCreate(rtattbuf, sizeof(rtattbuf), IFLA_PORT_REQUEST, > + &op, sizeof(op)); > + if (!rta) > + goto buffer_too_small; > + > + if (!nlAppend(nlm, sizeof(nlmsgbuf), rtattbuf, rta->rta_len)) > + goto buffer_too_small; > + > + /* end nesting of vport */ > + vfport->rta_len = (char *)nlm + nlm->nlmsg_len - (char *)vfport; > + > + if (vf != PORT_SELF_VF) { > + /* end nesting of vfports */ > + vfports->rta_len = (char *)nlm + nlm->nlmsg_len - (char *)vfports; > + } > + > + if (!multicast) { > + if (nlComm(nlm, &recvbuf, &recvbuflen) < 0) > + return -1; > + } else { > + if (nlCommWaitSuccess(nlm, RTMGRP_LINK, &recvbuf, &recvbuflen, > + 5 * MICROSEC_PER_SEC) < 0) > + return -1; > + } > + > + if (recvbuflen < NLMSG_LENGTH(0) || recvbuf == NULL) > + goto malformed_resp; > + > + resp = (struct nlmsghdr *)recvbuf; > + > + switch (resp->nlmsg_type) { > + case NLMSG_ERROR: > + err = (struct nlmsgerr *)NLMSG_DATA(resp); > + if (resp->nlmsg_len < NLMSG_LENGTH(sizeof(*err))) > + goto malformed_resp; > + > + switch (-err->error) { > + case 0: > + break; > + > + default: > + virReportSystemError(-err->error, > + _("error during virtual port configuration of ifindex %d"), > + ifindex); > + rc = -1; > + } > + break; > + > + case NLMSG_DONE: > + break; > + > + default: > + goto malformed_resp; > + } > + > + VIR_FREE(recvbuf); > + > + return rc; > + > +malformed_resp: > + macvtapError(VIR_ERR_INTERNAL_ERROR, "%s", > + _("malformed netlink response message")); > + VIR_FREE(recvbuf); > + return -1; > + > +buffer_too_small: > + macvtapError(VIR_ERR_INTERNAL_ERROR, "%s", > + _("internal buffer is too small")); > + return -1; > +} > + > + > +static int > +doPortProfileOpCommon(bool multicast, > + int ifindex, > + const char *profileId, > + struct ifla_port_vsi *portVsi, > + const unsigned char *instanceId, > + const unsigned char *hostUUID, > + int32_t vf, > + uint8_t op) > +{ > + int rc; > + char *recvbuf = NULL; > + struct nlattr *tb[IFLA_MAX + 1]; > + int repeats = 80; > + uint16_t status = 0; > + > + rc = doPortProfileOpSetLink(multicast, > + ifindex, > + profileId, > + portVsi, > + instanceId, > + hostUUID, > + vf, > + op); > + > + if (rc != 0) { > + macvtapError(VIR_ERR_INTERNAL_ERROR, "%s", > + _("sending of PortProfileRequest failed.\n")); > + return rc; > + } > + > + if (!multicast) { > + while (--repeats) { > + rc = link_dump(ifindex, tb, &recvbuf); > + if (rc) > + goto err_exit; > + rc = getPortProfileStatus(tb, vf, &status); > + if (rc == 0) { > + if (status == PORT_PROFILE_RESPONSE_SUCCESS || > + status == PORT_VDP_RESPONSE_SUCCESS) { > + break; > + } else if (status == PORT_PROFILE_RESPONSE_INPROGRESS) { > + // keep trying... > + } else { > + virReportSystemError(EINVAL, > + _("error %d during port-profile setlink on ifindex %d"), > + status, ifindex); > + rc = 1; > + break; > + } > + } > + usleep(125000); > + > + VIR_FREE(recvbuf); > + } > + } > + > + if (status == PORT_PROFILE_RESPONSE_INPROGRESS) { > + macvtapError(VIR_ERR_INTERNAL_ERROR, "%s", > + _("port-profile setlink timed out")); > + rc = -ETIMEDOUT; > + } > + > +err_exit: > + VIR_FREE(recvbuf); > + > + return rc; > +} > + > +# endif /* IFLA_PORT_MAX */ > + > +static int > +doPortProfileOp8021Qbg(const char *ifname, > + const virVirtualPortProfileDefPtr virtPort, > + enum virVirtualPortOp virtPortOp) > +{ > + int rc; > + > +# ifndef IFLA_VF_PORT_MAX > + > + (void)ifname; > + (void)virtPort; > + (void)vf; > + (void)virtPortOp; > + macvtapError(VIR_ERR_INTERNAL_ERROR, "%s", > + _("Kernel VF Port support was missing at compile time.")); > + rc = 1; > + > +# else /* IFLA_VF_PORT_MAX */ > + > + int op = PORT_REQUEST_ASSOCIATE; > + struct ifla_port_vsi portVsi = { > + .vsi_mgr_id = virtPort->u.virtPort8021Qbg.managerID, > + .vsi_type_version = virtPort->u.virtPort8021Qbg.typeIDVersion, > + }; > + bool multicast = true; > + int ifindex; > + > + if (ifaceGetIndex(true, ifname, &ifindex) != 0) { > + rc = 1; > + goto err_exit; > + } > + > + portVsi.vsi_type_id[2] = virtPort->u.virtPort8021Qbg.typeID >> 16; > + portVsi.vsi_type_id[1] = virtPort->u.virtPort8021Qbg.typeID >> 8; > + portVsi.vsi_type_id[0] = virtPort->u.virtPort8021Qbg.typeID; > + > + switch (virtPortOp) { > + case ASSOCIATE: > + op = PORT_REQUEST_ASSOCIATE; > + break; > + case DISASSOCIATE: > + op = PORT_REQUEST_DISASSOCIATE; > + break; > + default: > + macvtapError(VIR_ERR_INTERNAL_ERROR, > + _("operation type %d not supported"), op); > + rc = 1; > + goto err_exit; > + } > + > + rc = doPortProfileOpCommon(multicast, ifindex, > + NULL, > + &portVsi, > + virtPort->u.virtPort8021Qbg.instanceID, > + NULL, > + PORT_SELF_VF, > + op); > + > +err_exit: > + > +# endif /* IFLA_VF_PORT_MAX */ > + > + return rc; > +} > + > + > +static int > +getPhysfn(const char *linkdev, > + int32_t *vf, > + char **physfndev) > +{ > + int rc = 0; > + > +# ifndef IFLA_VF_PORT_MAX > + > + (void)linkdev; > + (void)vf; > + (void)physfndev; > + macvtapError(VIR_ERR_INTERNAL_ERROR, "%s", > + _("Kernel VF Port support was missing at compile time.")); > + rc = 1; > + > +# else /* IFLA_VF_PORT_MAX */ > + > + bool virtfn = false; > + > + if (virtfn) { > + > + // XXX: if linkdev is SR-IOV VF, then set vf = VF index > + // XXX: and set linkdev = PF device > + // XXX: need to use get_physical_function_linux() or > + // XXX: something like that to get PF > + // XXX: device and figure out VF index > + > + rc = 1; > + > + } else { > + > + /* Not SR-IOV VF: physfndev is linkdev and VF index > + * refers to linkdev self > + */ > + > + *vf = PORT_SELF_VF; > + *physfndev = (char *)linkdev; > + } > + > +# endif /* IFLA_VF_PORT_MAX */ > + > + return rc; > +} > + > + > +static int > +doPortProfileOp8021Qbh(const char *ifname, > + const virVirtualPortProfileDefPtr virtPort, > + const unsigned char *vm_uuid, > + enum virVirtualPortOp virtPortOp) > +{ > + int rc; > + > +# ifndef IFLA_VF_PORT_MAX > + > + (void)ifname; > + (void)virtPort; > + (void)vm_uuid; > + (void)virtPortOp; > + macvtapError(VIR_ERR_INTERNAL_ERROR, "%s", > + _("Kernel VF Port support was missing at compile time.")); > + rc = 1; > + > +# else /* IFLA_VF_PORT_MAX */ > + > + char *physfndev; > + unsigned char hostuuid[VIR_UUID_BUFLEN]; > + int32_t vf; > + int op = PORT_REQUEST_ASSOCIATE; > + bool multicast = false; > + int ifindex; > + > + rc = virGetHostUUID(hostuuid); > + if (rc) > + goto err_exit; > + > + rc = getPhysfn(ifname, &vf, &physfndev); > + if (rc) > + goto err_exit; > + > + if (ifaceGetIndex(true, physfndev, &ifindex) != 0) { > + rc = 1; > + goto err_exit; > + } > + > + switch (virtPortOp) { > + case ASSOCIATE: > + op = PORT_REQUEST_ASSOCIATE; > + break; > + case DISASSOCIATE: > + op = PORT_REQUEST_DISASSOCIATE; > + break; > + default: > + macvtapError(VIR_ERR_INTERNAL_ERROR, > + _("operation type %d not supported"), op); > + rc = 1; > + goto err_exit; > + } > + > + rc = doPortProfileOpCommon(multicast, ifindex, > + virtPort->u.virtPort8021Qbh.profileID, > + NULL, > + vm_uuid, > + hostuuid, > + vf, > + op); > + > + switch (virtPortOp) { > + case ASSOCIATE: > + ifaceUp(ifname); > + break; > + case DISASSOCIATE: > + ifaceDown(ifname); > + break; > + } > + > +err_exit: > + > +# endif /* IFLA_VF_PORT_MAX */ > + > + return rc; > +} > > /** > * associatePortProfile > * > * @macvtap_ifname: The name of the macvtap device > + * @linkdev: The link device in case of macvtap > * @virtPort: pointer to the object holding port profile parameters > - * @vf: virtual function number, -1 if to be ignored > * @vmuuid : the UUID of the virtual machine > * > * Associate a port on a swtich with a profile. This function > @@ -740,15 +1484,14 @@ delMacvtap(const char *ifname, > */ > static int > associatePortProfileId(const char *macvtap_ifname, > + const char *linkdev, > const virVirtualPortProfileDefPtr virtPort, > - int vf, > const unsigned char *vmuuid) > { > int rc = 0; > + > VIR_DEBUG("Associating port profile '%p' on link device '%s'", > virtPort, macvtap_ifname); > - (void)vf; > - (void)vmuuid; > > switch (virtPort->virtPortType) { > case VIR_VIRTUALPORT_NONE: > @@ -756,11 +1499,14 @@ associatePortProfileId(const char *macvtap_ifname, > break; > > case VIR_VIRTUALPORT_8021QBG: > - > + rc = doPortProfileOp8021Qbg(macvtap_ifname, virtPort, > + ASSOCIATE); > break; > > case VIR_VIRTUALPORT_8021QBH: > - > + rc = doPortProfileOp8021Qbh(linkdev, virtPort, > + vmuuid, > + ASSOCIATE); > break; > } > > @@ -772,6 +1518,7 @@ associatePortProfileId(const char *macvtap_ifname, > * disassociatePortProfile > * > * @macvtap_ifname: The name of the macvtap device > + * @linkdev: The link device in case of macvtap > * @virtPort: point to object holding port profile parameters > * > * Returns 0 in case of success, != 0 otherwise with error > @@ -779,9 +1526,11 @@ associatePortProfileId(const char *macvtap_ifname, > */ > static int > disassociatePortProfileId(const char *macvtap_ifname, > + const char *linkdev, > const virVirtualPortProfileDefPtr virtPort) > { > int rc = 0; > + > VIR_DEBUG("Disassociating port profile id '%p' on link device '%s' ", > virtPort, macvtap_ifname); > > @@ -791,13 +1540,18 @@ disassociatePortProfileId(const char *macvtap_ifname, > break; > > case VIR_VIRTUALPORT_8021QBG: > - > + rc = doPortProfileOp8021Qbg(macvtap_ifname, virtPort, > + DISASSOCIATE); > break; > > case VIR_VIRTUALPORT_8021QBH: > - > + rc = doPortProfileOp8021Qbh(linkdev, virtPort, > + NULL, > + DISASSOCIATE); > break; > } > > return rc; > } > + > +#endif > diff --git a/src/util/macvtap.h b/src/util/macvtap.h > index ae11c5c..35db31c 100644 > --- a/src/util/macvtap.h > +++ b/src/util/macvtap.h > @@ -72,6 +72,7 @@ int openMacvtapTap(const char *ifname, > char **res_ifname); > > void delMacvtap(const char *ifname, > + const char *linkdev, > virVirtualPortProfileDefPtr virtPortProfile); > > # endif /* WITH_MACVTAP */ > > -- > libvir-list mailing list > libvir-list@xxxxxxxxxx > https://www.redhat.com/mailman/listinfo/libvir-list
>From 7b0c513288ad3b0740ac05d9bf47d4320e385639 Mon Sep 17 00:00:00 2001 From: David Allan <dallan@xxxxxxxxxx> Date: Mon, 24 May 2010 22:43:54 -0400 Subject: [PATCH 1/1] Minor fixes to compile ifndef IFLA_VF_PORT_MAX * Make getPhysfn entirely within #ifdef IFLA_VF_PORT_MAX as it's only called when IFLA_VF_PORT_MAX is defined * Fix a few nits found by make syntax check (newline in error message and preprocessor indentation) --- src/util/macvtap.c | 22 ++++------------------ 1 files changed, 4 insertions(+), 18 deletions(-) diff --git a/src/util/macvtap.c b/src/util/macvtap.c index d5a08d9..8e4bea3 100644 --- a/src/util/macvtap.c +++ b/src/util/macvtap.c @@ -318,7 +318,7 @@ err_exit: return rc; } -#endif +# endif static struct rtattr * rtattrCreate(char *buffer, int bufsize, int type, @@ -1231,7 +1231,7 @@ doPortProfileOpCommon(bool multicast, if (rc != 0) { macvtapError(VIR_ERR_INTERNAL_ERROR, "%s", - _("sending of PortProfileRequest failed.\n")); + _("sending of PortProfileRequest failed.")); return rc; } @@ -1286,7 +1286,6 @@ doPortProfileOp8021Qbg(const char *ifname, (void)ifname; (void)virtPort; - (void)vf; (void)virtPortOp; macvtapError(VIR_ERR_INTERNAL_ERROR, "%s", _("Kernel VF Port support was missing at compile time.")); @@ -1341,24 +1340,13 @@ err_exit: } +# ifdef IFLA_VF_PORT_MAX static int getPhysfn(const char *linkdev, int32_t *vf, char **physfndev) { int rc = 0; - -# ifndef IFLA_VF_PORT_MAX - - (void)linkdev; - (void)vf; - (void)physfndev; - macvtapError(VIR_ERR_INTERNAL_ERROR, "%s", - _("Kernel VF Port support was missing at compile time.")); - rc = 1; - -# else /* IFLA_VF_PORT_MAX */ - bool virtfn = false; if (virtfn) { @@ -1381,11 +1369,9 @@ getPhysfn(const char *linkdev, *physfndev = (char *)linkdev; } -# endif /* IFLA_VF_PORT_MAX */ - return rc; } - +# endif /* IFLA_VF_PORT_MAX */ static int doPortProfileOp8021Qbh(const char *ifname, -- 1.7.0.1
-- libvir-list mailing list libvir-list@xxxxxxxxxx https://www.redhat.com/mailman/listinfo/libvir-list