/* * Copyright (C) 2013 Red Hat, Inc. * Copyright (C) 2012 Nicira, Inc. * Copyright (C) 2017 IBM Corporation * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library. If not, see * . */ #include #include "virnetdevopenvswitch.h" #include "vircommand.h" #include "viralloc.h" #include "virerror.h" #include "virmacaddr.h" #include "virstring.h" #include "virlog.h" #include "virjson.h" #define VIR_FROM_THIS VIR_FROM_NONE VIR_LOG_INIT("util.netdevopenvswitch"); /* * Set openvswitch default timeout */ static unsigned int virNetDevOpenvswitchTimeout = VIR_NETDEV_OVS_DEFAULT_TIMEOUT; /** * virNetDevOpenvswitchSetTimeout: * @timeout: the timeout in seconds * * Set the openvswitch timeout */ void virNetDevOpenvswitchSetTimeout(unsigned int timeout) { virNetDevOpenvswitchTimeout = timeout; } static virCommandPtr virNetDevOpenvswitchCreateCmd(void) { virCommandPtr cmd = virCommandNew(OVS_VSCTL); virCommandAddArgFormat(cmd, "--timeout=%u", virNetDevOpenvswitchTimeout); return cmd; } /** * virNetDevOpenvswitchConstructVlans: * @cmd: command to construct * @virtVlan: VLAN configuration to be applied * * Construct the VLAN configuration parameters to be passed to * ovs-vsctl command. */ static void virNetDevOpenvswitchConstructVlans(virCommandPtr cmd, const virNetDevVlan *virtVlan) { g_auto(virBuffer) buf = VIR_BUFFER_INITIALIZER; if (!virtVlan || !virtVlan->nTags) return; switch (virtVlan->nativeMode) { case VIR_NATIVE_VLAN_MODE_TAGGED: virCommandAddArg(cmd, "vlan_mode=native-tagged"); virCommandAddArgFormat(cmd, "tag=%d", virtVlan->nativeTag); break; case VIR_NATIVE_VLAN_MODE_UNTAGGED: virCommandAddArg(cmd, "vlan_mode=native-untagged"); virCommandAddArgFormat(cmd, "tag=%d", virtVlan->nativeTag); break; case VIR_NATIVE_VLAN_MODE_DEFAULT: default: break; } if (virtVlan->trunk) { size_t i; virBufferAddLit(&buf, "trunk="); /* * Trunk ports have at least one VLAN. Do the first one * outside the "for" loop so we can put a "," at the * start of the for loop if there are more than one VLANs * on this trunk port. */ virBufferAsprintf(&buf, "%d", virtVlan->tag[0]); for (i = 1; i < virtVlan->nTags; i++) { virBufferAddLit(&buf, ","); virBufferAsprintf(&buf, "%d", virtVlan->tag[i]); } virCommandAddArg(cmd, virBufferCurrentContent(&buf)); } else if (virtVlan->nTags) { virCommandAddArgFormat(cmd, "tag=%d", virtVlan->tag[0]); } } /** * virNetDevOpenvswitchAddPort: * @brname: the bridge name * @ifname: the network interface name * @macaddr: the mac address of the virtual interface * @vmuuid: the Domain UUID that has this interface * @ovsport: the ovs specific fields * * Add an interface to the OVS bridge * * Returns 0 in case of success or -1 in case of failure. */ int virNetDevOpenvswitchAddPort(const char *brname, const char *ifname, const virMacAddr *macaddr, const unsigned char *vmuuid, const virNetDevVPortProfile *ovsport, const virNetDevVlan *virtVlan) { char macaddrstr[VIR_MAC_STRING_BUFLEN]; char ifuuidstr[VIR_UUID_STRING_BUFLEN]; char vmuuidstr[VIR_UUID_STRING_BUFLEN]; g_autoptr(virCommand) cmd = NULL; g_autofree char *attachedmac_ex_id = NULL; g_autofree char *ifaceid_ex_id = NULL; g_autofree char *profile_ex_id = NULL; g_autofree char *vmid_ex_id = NULL; virMacAddrFormat(macaddr, macaddrstr); virUUIDFormat(ovsport->interfaceID, ifuuidstr); virUUIDFormat(vmuuid, vmuuidstr); attachedmac_ex_id = g_strdup_printf("external-ids:attached-mac=\"%s\"", macaddrstr); ifaceid_ex_id = g_strdup_printf("external-ids:iface-id=\"%s\"", ifuuidstr); vmid_ex_id = g_strdup_printf("external-ids:vm-id=\"%s\"", vmuuidstr); if (ovsport->profileID[0] != '\0') { profile_ex_id = g_strdup_printf("external-ids:port-profile=\"%s\"", ovsport->profileID); } cmd = virNetDevOpenvswitchCreateCmd(); virCommandAddArgList(cmd, "--", "--if-exists", "del-port", ifname, "--", "add-port", brname, ifname, NULL); virNetDevOpenvswitchConstructVlans(cmd, virtVlan); if (ovsport->profileID[0] == '\0') { virCommandAddArgList(cmd, "--", "set", "Interface", ifname, attachedmac_ex_id, "--", "set", "Interface", ifname, ifaceid_ex_id, "--", "set", "Interface", ifname, vmid_ex_id, "--", "set", "Interface", ifname, "external-ids:iface-status=active", NULL); } else { virCommandAddArgList(cmd, "--", "set", "Interface", ifname, attachedmac_ex_id, "--", "set", "Interface", ifname, ifaceid_ex_id, "--", "set", "Interface", ifname, vmid_ex_id, "--", "set", "Interface", ifname, profile_ex_id, "--", "set", "Interface", ifname, "external-ids:iface-status=active", NULL); } if (virCommandRun(cmd, NULL) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, _("Unable to add port %s to OVS bridge %s"), ifname, brname); return -1; } return 0; } /** * virNetDevOpenvswitchRemovePort: * @ifname: the network interface name * * Deletes an interface from a OVS bridge * * Returns 0 in case of success or -1 in case of failure. */ int virNetDevOpenvswitchRemovePort(const char *brname G_GNUC_UNUSED, const char *ifname) { g_autoptr(virCommand) cmd = virNetDevOpenvswitchCreateCmd(); virCommandAddArgList(cmd, "--", "--if-exists", "del-port", ifname, NULL); if (virCommandRun(cmd, NULL) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, _("Unable to delete port %s from OVS"), ifname); return -1; } return 0; } /** * virNetDevOpenvswitchGetMigrateData: * @migrate: a pointer to store the data into, allocated by this function * @ifname: name of the interface for which data is being migrated * * Allocates data to be migrated specific to Open vSwitch * * Returns 0 in case of success or -1 in case of failure */ int virNetDevOpenvswitchGetMigrateData(char **migrate, const char *ifname) { size_t len; g_autoptr(virCommand) cmd = virNetDevOpenvswitchCreateCmd(); virCommandAddArgList(cmd, "--if-exists", "get", "Interface", ifname, "external_ids:PortData", NULL); virCommandSetOutputBuffer(cmd, migrate); /* Run the command */ if (virCommandRun(cmd, NULL) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, _("Unable to run command to get OVS port data for " "interface %s"), ifname); return -1; } /* Wipeout the newline, if it exists */ len = strlen(*migrate); if (len > 0) (*migrate)[len - 1] = '\0'; return 0; } /** * virNetDevOpenvswitchSetMigrateData: * @migrate: the data which was transferred during migration * @ifname: the name of the interface the data is associated with * * Repopulates OVS per-port data on destination host * * Returns 0 in case of success or -1 in case of failure */ int virNetDevOpenvswitchSetMigrateData(char *migrate, const char *ifname) { g_autoptr(virCommand) cmd = NULL; if (!migrate) { VIR_DEBUG("No OVS port data for interface %s", ifname); return 0; } cmd = virNetDevOpenvswitchCreateCmd(); virCommandAddArgList(cmd, "set", "Interface", ifname, NULL); virCommandAddArgFormat(cmd, "external_ids:PortData=%s", migrate); /* Run the command */ if (virCommandRun(cmd, NULL) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, _("Unable to run command to set OVS port data for " "interface %s"), ifname); return -1; } return 0; } /** * virNetDevOpenvswitchInterfaceParseStats: * @json: Input string in JSON format * @stats: parsed stats * * For given input string @json parse interface statistics and store them into * @stats. * * Returns: 0 on success, * -1 otherwise (with error reported). */ int virNetDevOpenvswitchInterfaceParseStats(const char *json, virDomainInterfaceStatsPtr stats) { g_autoptr(virJSONValue) jsonStats = NULL; virJSONValuePtr jsonMap = NULL; size_t i; stats->rx_bytes = stats->rx_packets = stats->rx_errs = stats->rx_drop = -1; stats->tx_bytes = stats->tx_packets = stats->tx_errs = stats->tx_drop = -1; if (!(jsonStats = virJSONValueFromString(json)) || !virJSONValueIsArray(jsonStats) || !(jsonMap = virJSONValueArrayGet(jsonStats, 1))) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Unable to parse ovs-vsctl output")); return -1; } for (i = 0; i < virJSONValueArraySize(jsonMap); i++) { virJSONValuePtr item = virJSONValueArrayGet(jsonMap, i); virJSONValuePtr jsonKey; virJSONValuePtr jsonVal; const char *key; long long val; if (!item || (!(jsonKey = virJSONValueArrayGet(item, 0))) || (!(jsonVal = virJSONValueArrayGet(item, 1))) || (!(key = virJSONValueGetString(jsonKey))) || (virJSONValueGetNumberLong(jsonVal, &val) < 0)) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Malformed ovs-vsctl output")); return -1; } /* The TX/RX fields appear to be swapped here * because this is the host view. */ if (STREQ(key, "rx_bytes")) { stats->tx_bytes = val; } else if (STREQ(key, "rx_packets")) { stats->tx_packets = val; } else if (STREQ(key, "rx_errors")) { stats->tx_errs = val; } else if (STREQ(key, "rx_dropped")) { stats->tx_drop = val; } else if (STREQ(key, "tx_bytes")) { stats->rx_bytes = val; } else if (STREQ(key, "tx_packets")) { stats->rx_packets = val; } else if (STREQ(key, "tx_errors")) { stats->rx_errs = val; } else if (STREQ(key, "tx_dropped")) { stats->rx_drop = val; } else { VIR_DEBUG("Unused ovs-vsctl stat key=%s val=%lld", key, val); } } return 0; } /** * virNetDevOpenvswitchInterfaceStats: * @ifname: the name of the interface * @stats: the retrieved domain interface stat * * Retrieves the OVS interfaces stats * * Returns 0 in case of success or -1 in case of failure */ int virNetDevOpenvswitchInterfaceStats(const char *ifname, virDomainInterfaceStatsPtr stats) { g_autoptr(virCommand) cmd = virNetDevOpenvswitchCreateCmd(); g_autofree char *output = NULL; virCommandAddArgList(cmd, "--if-exists", "--format=list", "--data=json", "--no-headings", "--columns=statistics", "list", "Interface", ifname, NULL); virCommandSetOutputBuffer(cmd, &output); /* The above command returns either: * 1) empty string if @ifname doesn't exist, or * 2) a JSON array, for instance: * ["map",[["collisions",0],["rx_bytes",0],["rx_crc_err",0],["rx_dropped",0], * ["rx_errors",0],["rx_frame_err",0],["rx_over_err",0],["rx_packets",0], * ["tx_bytes",12406],["tx_dropped",0],["tx_errors",0],["tx_packets",173]]] */ if (virCommandRun(cmd, NULL) < 0 || STREQ_NULLABLE(output, "")) { /* no ovs-vsctl or interface 'ifname' doesn't exists in ovs */ virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Interface not found")); return -1; } if (virNetDevOpenvswitchInterfaceParseStats(output, stats) < 0) return -1; if (stats->rx_bytes == -1 && stats->rx_packets == -1 && stats->rx_errs == -1 && stats->rx_drop == -1 && stats->tx_bytes == -1 && stats->tx_packets == -1 && stats->tx_errs == -1 && stats->tx_drop == -1) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Interface doesn't have any statistics")); return -1; } return 0; } /** * virNetDeOpenvswitchGetMaster: * @ifname: name of interface we're interested in * @master: used to return a string containing the name of @ifname's "master" * (this is the bridge or bond device that this device is attached to) * * Returns 0 on success, -1 on failure (if @ifname has no master * @master will be NULL, but return value will still be 0 (success)). * * NB: This function is needed because the IFLA_MASTER attribute of an * interface in a netlink dump (see virNetDevGetMaster()) will always * return "ovs-system" for any interface that is attached to an OVS * switch. When that happens, virNetDevOpenvswitchInterfaceGetMaster() * must be called to get the "real" master of the interface. */ int virNetDevOpenvswitchInterfaceGetMaster(const char *ifname, char **master) { g_autoptr(virCommand) cmd = virNetDevOpenvswitchCreateCmd(); int exitstatus; *master = NULL; virCommandAddArgList(cmd, "iface-to-br", ifname, NULL); virCommandSetOutputBuffer(cmd, master); if (virCommandRun(cmd, &exitstatus) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, _("Unable to run command to get OVS master for " "interface %s"), ifname); return -1; } /* non-0 exit code just means that the interface has no master in OVS */ if (exitstatus != 0) VIR_FREE(*master); if (*master) { /* truncate at the first newline */ char *nl = strchr(*master, '\n'); if (nl) *nl = '\0'; } VIR_DEBUG("OVS master for %s is %s", ifname, *master ? *master : "(none)"); return 0; } /** * virNetDevOpenvswitchGetVhostuserIfname: * @path: the path of the unix socket * @server: true if OVS creates the @path * @ifname: the retrieved name of the interface * * Retrieves the OVS ifname from vhostuser UNIX socket path. * There are two types of vhostuser ports which differ in client/server * role: * * dpdkvhostuser - OVS creates the socket and QEMU connects to it * (@server = true) * dpdkvhostuserclient - QEMU creates the socket and OVS connects to it * (@server = false) * * Since the way of retrieving ifname is different in these two cases, * caller must set @server according to the interface definition. * * Returns: 1 if interface is an openvswitch interface, * 0 if it is not, but no other error occurred, * -1 otherwise. */ int virNetDevOpenvswitchGetVhostuserIfname(const char *path, bool server, char **ifname) { g_autoptr(virCommand) cmd = virNetDevOpenvswitchCreateCmd(); int status; if (server) { virCommandAddArgList(cmd, "--no-headings", "--columns=name", "find", "Interface", NULL); virCommandAddArgPair(cmd, "options:vhost-server-path", "path"); } else { const char *tmpIfname = NULL; /* Openvswitch vhostuser path is hardcoded to * //openvswitch/ * for example: /var/run/openvswitch/dpdkvhostuser0 * * so we pick the filename and check it's an openvswitch interface */ if (!path || !(tmpIfname = strrchr(path, '/'))) { return 0; } tmpIfname++; virCommandAddArgList(cmd, "get", "Interface", tmpIfname, "name", NULL); } virCommandSetOutputBuffer(cmd, ifname); if (virCommandRun(cmd, &status) < 0) return -1; if (status != 0) { /* it's not a openvswitch vhostuser interface. */ return 0; } return 1; } /** * virNetDevOpenvswitchUpdateVlan: * @ifname: the network interface name * @virtVlan: VLAN configuration to be applied * * Update VLAN configuration of an OVS port. * * Returns 0 in case of success or -1 in case of failure. */ int virNetDevOpenvswitchUpdateVlan(const char *ifname, const virNetDevVlan *virtVlan) { g_autoptr(virCommand) cmd = virNetDevOpenvswitchCreateCmd(); virCommandAddArgList(cmd, "--", "--if-exists", "clear", "Port", ifname, "tag", "--", "--if-exists", "clear", "Port", ifname, "trunk", "--", "--if-exists", "clear", "Port", ifname, "vlan_mode", "--", "--if-exists", "set", "Port", ifname, NULL); virNetDevOpenvswitchConstructVlans(cmd, virtVlan); if (virCommandRun(cmd, NULL) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, _("Unable to set vlan configuration on port %s"), ifname); return -1; } return 0; }