2011-11-02 17:11:02 +00:00
|
|
|
/*
|
2017-03-05 17:32:15 -05:00
|
|
|
* Copyright (C) 2010-2017 Red Hat, Inc.
|
2012-02-22 14:17:14 +01:00
|
|
|
* Copyright (C) 2010-2012 IBM Corporation
|
2011-11-02 17:11:02 +00:00
|
|
|
*
|
|
|
|
* 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
|
2012-09-20 16:30:55 -06:00
|
|
|
* License along with this library. If not, see
|
2012-07-21 18:06:23 +08:00
|
|
|
* <http://www.gnu.org/licenses/>.
|
2011-11-02 17:11:02 +00:00
|
|
|
*
|
|
|
|
* Authors:
|
|
|
|
* Stefan Berger <stefanb@us.ibm.com>
|
|
|
|
*
|
|
|
|
* Notes:
|
|
|
|
* netlink: http://lovezutto.googlepages.com/netlink.pdf
|
|
|
|
* iproute2 package
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include "virnetdevmacvlan.h"
|
2012-01-27 17:23:05 +00:00
|
|
|
#include "virmacaddr.h"
|
2012-12-13 18:21:53 +00:00
|
|
|
#include "virerror.h"
|
2013-02-28 15:59:01 +01:00
|
|
|
#include "virthread.h"
|
2013-05-24 09:19:51 +02:00
|
|
|
#include "virstring.h"
|
2011-11-02 17:11:02 +00:00
|
|
|
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_NET
|
|
|
|
|
|
|
|
VIR_ENUM_IMPL(virNetDevMacVLanMode, VIR_NETDEV_MACVLAN_MODE_LAST,
|
|
|
|
"vepa",
|
|
|
|
"private",
|
|
|
|
"bridge",
|
|
|
|
"passthrough")
|
|
|
|
|
|
|
|
#if WITH_MACVTAP
|
|
|
|
# include <stdint.h>
|
|
|
|
# include <stdio.h>
|
|
|
|
# include <errno.h>
|
|
|
|
# include <fcntl.h>
|
|
|
|
# include <sys/socket.h>
|
|
|
|
# include <sys/ioctl.h>
|
|
|
|
|
2013-06-13 10:26:22 +04:00
|
|
|
# include <net/if.h>
|
2011-11-02 17:11:02 +00:00
|
|
|
# include <linux/if_tun.h>
|
|
|
|
|
|
|
|
/* Older kernels lacked this enum value. */
|
|
|
|
# if !HAVE_DECL_MACVLAN_MODE_PASSTHRU
|
|
|
|
# define MACVLAN_MODE_PASSTHRU 8
|
|
|
|
# endif
|
|
|
|
|
2012-12-12 18:06:53 +00:00
|
|
|
# include "viralloc.h"
|
2012-12-12 17:59:27 +00:00
|
|
|
# include "virlog.h"
|
2012-12-13 18:01:25 +00:00
|
|
|
# include "viruuid.h"
|
2011-11-02 17:11:02 +00:00
|
|
|
# include "virfile.h"
|
2012-02-03 15:13:19 +01:00
|
|
|
# include "virnetlink.h"
|
2011-11-02 17:11:02 +00:00
|
|
|
# include "virnetdev.h"
|
2012-02-22 14:17:14 +01:00
|
|
|
# include "virpidfile.h"
|
2016-01-19 14:20:54 -05:00
|
|
|
# include "virbitmap.h"
|
2012-02-22 14:17:14 +01:00
|
|
|
|
2014-02-28 12:16:17 +00:00
|
|
|
VIR_LOG_INIT("util.netdevmacvlan");
|
2011-11-02 17:11:02 +00:00
|
|
|
|
2017-04-25 14:09:45 -04:00
|
|
|
# define VIR_NET_GENERATED_MACVTAP_PATTERN VIR_NET_GENERATED_MACVTAP_PREFIX "%d"
|
|
|
|
# define VIR_NET_GENERATED_MACVLAN_PATTERN VIR_NET_GENERATED_MACVLAN_PREFIX "%d"
|
|
|
|
# define VIR_NET_GENERATED_PREFIX \
|
|
|
|
((flags & VIR_NETDEV_MACVLAN_CREATE_WITH_TAP) ? \
|
|
|
|
VIR_NET_GENERATED_MACVTAP_PREFIX : VIR_NET_GENERATED_MACVLAN_PREFIX)
|
2011-11-10 10:29:09 +00:00
|
|
|
|
2016-01-19 14:20:54 -05:00
|
|
|
# define MACVLAN_MAX_ID 8191
|
|
|
|
|
2014-03-25 14:54:44 +00:00
|
|
|
virMutex virNetDevMacVLanCreateMutex = VIR_MUTEX_INITIALIZER;
|
2016-01-19 14:20:54 -05:00
|
|
|
virBitmapPtr macvtapIDs = NULL;
|
|
|
|
virBitmapPtr macvlanIDs = NULL;
|
|
|
|
|
|
|
|
static int
|
|
|
|
virNetDevMacVLanOnceInit(void)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (!macvtapIDs &&
|
|
|
|
!(macvtapIDs = virBitmapNew(MACVLAN_MAX_ID + 1)))
|
|
|
|
return -1;
|
|
|
|
if (!macvlanIDs &&
|
|
|
|
!(macvlanIDs = virBitmapNew(MACVLAN_MAX_ID + 1)))
|
|
|
|
return -1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
VIR_ONCE_GLOBAL_INIT(virNetDevMacVLan);
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* virNetDevMacVLanReserveID:
|
|
|
|
*
|
|
|
|
* @id: id 0 - MACVLAN_MAX_ID+1 to reserve (or -1 for "first free")
|
|
|
|
* @flags: set VIR_NETDEV_MACVLAN_CREATE_WITH_TAP for macvtapN else macvlanN
|
|
|
|
* @quietFail: don't log an error if this name is already in-use
|
|
|
|
* @nextFree: reserve the next free ID *after* @id rather than @id itself
|
|
|
|
*
|
|
|
|
* Reserve the indicated ID in the appropriate bitmap, or find the
|
|
|
|
* first free ID if @id is -1.
|
|
|
|
*
|
|
|
|
* Returns newly reserved ID# on success, or -1 to indicate failure.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
virNetDevMacVLanReserveID(int id, unsigned int flags,
|
|
|
|
bool quietFail, bool nextFree)
|
|
|
|
{
|
|
|
|
virBitmapPtr bitmap;
|
|
|
|
|
|
|
|
if (virNetDevMacVLanInitialize() < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
bitmap = (flags & VIR_NETDEV_MACVLAN_CREATE_WITH_TAP) ?
|
|
|
|
macvtapIDs : macvlanIDs;
|
|
|
|
|
|
|
|
if (id > MACVLAN_MAX_ID) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("can't use name %s%d - out of range 0-%d"),
|
2017-04-25 14:09:45 -04:00
|
|
|
VIR_NET_GENERATED_PREFIX, id, MACVLAN_MAX_ID);
|
2016-01-19 14:20:54 -05:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((id < 0 || nextFree) &&
|
2016-03-28 10:14:04 -04:00
|
|
|
(id = virBitmapNextClearBit(bitmap, id)) < 0) {
|
2016-01-19 14:20:54 -05:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("no unused %s names available"),
|
2017-04-25 14:09:45 -04:00
|
|
|
VIR_NET_GENERATED_PREFIX);
|
2016-01-19 14:20:54 -05:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virBitmapIsBitSet(bitmap, id)) {
|
|
|
|
if (quietFail) {
|
|
|
|
VIR_INFO("couldn't reserve name %s%d - already in use",
|
2017-04-25 14:09:45 -04:00
|
|
|
VIR_NET_GENERATED_PREFIX, id);
|
2016-01-19 14:20:54 -05:00
|
|
|
} else {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("couldn't reserve name %s%d - already in use"),
|
2017-04-25 14:09:45 -04:00
|
|
|
VIR_NET_GENERATED_PREFIX, id);
|
2016-01-19 14:20:54 -05:00
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virBitmapSetBit(bitmap, id) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("couldn't mark %s%d as used"),
|
2017-04-25 14:09:45 -04:00
|
|
|
VIR_NET_GENERATED_PREFIX, id);
|
2016-01-19 14:20:54 -05:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2017-04-25 14:09:45 -04:00
|
|
|
VIR_INFO("reserving device %s%d", VIR_NET_GENERATED_PREFIX, id);
|
2016-01-19 14:20:54 -05:00
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* virNetDevMacVLanReleaseID:
|
|
|
|
* @id: id 0 - MACVLAN_MAX_ID+1 to release
|
|
|
|
*
|
|
|
|
* Returns 0 for success or -1 for failure.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
virNetDevMacVLanReleaseID(int id, unsigned int flags)
|
|
|
|
{
|
|
|
|
virBitmapPtr bitmap;
|
|
|
|
|
|
|
|
if (virNetDevMacVLanInitialize() < 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
bitmap = (flags & VIR_NETDEV_MACVLAN_CREATE_WITH_TAP) ?
|
|
|
|
macvtapIDs : macvlanIDs;
|
|
|
|
|
|
|
|
if (id > MACVLAN_MAX_ID) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("can't free name %s%d - out of range 0-%d"),
|
2017-04-25 14:09:45 -04:00
|
|
|
VIR_NET_GENERATED_PREFIX, id, MACVLAN_MAX_ID);
|
2016-01-19 14:20:54 -05:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (id < 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
VIR_INFO("releasing %sdevice %s%d",
|
|
|
|
virBitmapIsBitSet(bitmap, id) ? "" : "unreserved",
|
2017-04-25 14:09:45 -04:00
|
|
|
VIR_NET_GENERATED_PREFIX, id);
|
2016-01-19 14:20:54 -05:00
|
|
|
|
|
|
|
if (virBitmapClearBit(bitmap, id) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("couldn't mark %s%d as unused"),
|
2017-04-25 14:09:45 -04:00
|
|
|
VIR_NET_GENERATED_PREFIX, id);
|
2016-01-19 14:20:54 -05:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* virNetDevMacVLanReserveName:
|
|
|
|
*
|
|
|
|
* @name: already-known name of device
|
|
|
|
* @quietFail: don't log an error if this name is already in-use
|
|
|
|
*
|
|
|
|
* Extract the device type and id from a macvtap/macvlan device name
|
|
|
|
* and mark the appropriate position as in-use in the appropriate
|
|
|
|
* bitmap.
|
|
|
|
*
|
|
|
|
* Returns reserved ID# on success, -1 on failure, -2 if the name
|
|
|
|
* doesn't fit the auto-pattern (so not reserveable).
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
virNetDevMacVLanReserveName(const char *name, bool quietFail)
|
|
|
|
{
|
|
|
|
unsigned int id;
|
|
|
|
unsigned int flags = 0;
|
|
|
|
const char *idstr = NULL;
|
|
|
|
|
|
|
|
if (virNetDevMacVLanInitialize() < 0)
|
|
|
|
return -1;
|
|
|
|
|
2017-04-25 14:09:45 -04:00
|
|
|
if (STRPREFIX(name, VIR_NET_GENERATED_MACVTAP_PREFIX)) {
|
|
|
|
idstr = name + strlen(VIR_NET_GENERATED_MACVTAP_PREFIX);
|
2016-01-19 14:20:54 -05:00
|
|
|
flags |= VIR_NETDEV_MACVLAN_CREATE_WITH_TAP;
|
2017-04-25 14:09:45 -04:00
|
|
|
} else if (STRPREFIX(name, VIR_NET_GENERATED_MACVLAN_PREFIX)) {
|
|
|
|
idstr = name + strlen(VIR_NET_GENERATED_MACVLAN_PREFIX);
|
2016-01-19 14:20:54 -05:00
|
|
|
} else {
|
|
|
|
return -2;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virStrToLong_ui(idstr, NULL, 10, &id) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("couldn't get id value from macvtap device name %s"),
|
|
|
|
name);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return virNetDevMacVLanReserveID(id, flags, quietFail, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* virNetDevMacVLanReleaseName:
|
|
|
|
*
|
|
|
|
* @name: already-known name of device
|
|
|
|
*
|
|
|
|
* Extract the device type and id from a macvtap/macvlan device name
|
|
|
|
* and mark the appropriate position as in-use in the appropriate
|
|
|
|
* bitmap.
|
|
|
|
*
|
|
|
|
* returns 0 on success, -1 on failure
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
virNetDevMacVLanReleaseName(const char *name)
|
|
|
|
{
|
|
|
|
unsigned int id;
|
|
|
|
unsigned int flags = 0;
|
|
|
|
const char *idstr = NULL;
|
|
|
|
|
|
|
|
if (virNetDevMacVLanInitialize() < 0)
|
|
|
|
return -1;
|
|
|
|
|
2017-04-25 14:09:45 -04:00
|
|
|
if (STRPREFIX(name, VIR_NET_GENERATED_MACVTAP_PREFIX)) {
|
|
|
|
idstr = name + strlen(VIR_NET_GENERATED_MACVTAP_PREFIX);
|
2016-01-19 14:20:54 -05:00
|
|
|
flags |= VIR_NETDEV_MACVLAN_CREATE_WITH_TAP;
|
2017-04-25 14:09:45 -04:00
|
|
|
} else if (STRPREFIX(name, VIR_NET_GENERATED_MACVLAN_PREFIX)) {
|
|
|
|
idstr = name + strlen(VIR_NET_GENERATED_MACVLAN_PREFIX);
|
2016-01-19 14:20:54 -05:00
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virStrToLong_ui(idstr, NULL, 10, &id) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("couldn't get id value from macvtap device name %s"),
|
|
|
|
name);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return virNetDevMacVLanReleaseID(id, flags);
|
|
|
|
}
|
|
|
|
|
2013-02-28 15:59:01 +01:00
|
|
|
|
2011-11-02 17:34:41 +00:00
|
|
|
/**
|
|
|
|
* virNetDevMacVLanCreate:
|
|
|
|
*
|
|
|
|
* @ifname: The name the interface is supposed to have; optional parameter
|
2011-11-10 10:29:09 +00:00
|
|
|
* @type: The type of device, i.e., "macvtap", "macvlan"
|
2011-11-02 17:34:41 +00:00
|
|
|
* @macaddress: The MAC address of the device
|
|
|
|
* @srcdev: The name of the 'link' device
|
|
|
|
* @macvlan_mode: The macvlan mode to use
|
|
|
|
* @retry: Pointer to integer that will be '1' upon return if an interface
|
|
|
|
* with the same name already exists and it is worth to try
|
|
|
|
* again with a different name
|
|
|
|
*
|
|
|
|
* Create a macvtap device with the given properties.
|
|
|
|
*
|
|
|
|
* Returns 0 on success, -1 on fatal error.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
virNetDevMacVLanCreate(const char *ifname,
|
|
|
|
const char *type,
|
2013-10-05 13:41:44 -06:00
|
|
|
const virMacAddr *macaddress,
|
2011-11-02 17:34:41 +00:00
|
|
|
const char *srcdev,
|
|
|
|
uint32_t macvlan_mode,
|
|
|
|
int *retry)
|
|
|
|
{
|
util: standardize return from functions calling virNetlinkCommand
There are several functions that call virNetlinkCommand, and they all
follow a common pattern, with three exit labels: err_exit (or
cleanup), malformed_resp, and buffer_too_small. All three of these
labels do their own cleanup and have their own return. However, the
malformed_resp label usually frees the same items as the
cleanup/err_exit label, and the buffer_too_small label just doesn't
free recvbuf (because it's known to always be NULL at the time we goto
buffer_too_small.
In order to simplify and standardize the code, I've made the following
changes to all of these functions:
1) err_exit is replaced with the more libvirt-ish "cleanup", which
makes sense because in all cases this code is also executed in the
case of success, so labelling it err_exit may be confusing.
2) rc is initialized to -1, and set to 0 just before the cleanup
label. Any code that currently sets rc = -1 is made to instead goto
cleanup.
3) malformed_resp and buffer_too_small just log their error and goto
cleanup. This gives us a single return path, and a single place to
free up resources.
4) In one instance, rather then logging an error immediately, a char*
msg was pointed to an error string, then goto cleanup (and cleanup
would log an error if msg != NULL). It takes no more lines of code
to just log the message as we encounter it.
This patch should have 0 functional effects.
2012-03-07 12:44:56 -05:00
|
|
|
int rc = -1;
|
2013-04-03 14:09:19 +01:00
|
|
|
struct nlmsghdr *resp = NULL;
|
2011-11-02 17:34:41 +00:00
|
|
|
struct nlmsgerr *err;
|
|
|
|
struct ifinfomsg ifinfo = { .ifi_family = AF_UNSPEC };
|
|
|
|
int ifindex;
|
|
|
|
unsigned int recvbuflen;
|
|
|
|
struct nl_msg *nl_msg;
|
|
|
|
struct nlattr *linkinfo, *info_data;
|
2015-03-18 14:27:05 -04:00
|
|
|
char macstr[VIR_MAC_STRING_BUFLEN];
|
2011-11-02 17:34:41 +00:00
|
|
|
|
2011-11-03 09:21:35 +00:00
|
|
|
if (virNetDevGetIndex(srcdev, &ifindex) < 0)
|
2011-11-02 17:34:41 +00:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
*retry = 0;
|
|
|
|
|
|
|
|
nl_msg = nlmsg_alloc_simple(RTM_NEWLINK,
|
|
|
|
NLM_F_REQUEST | NLM_F_CREATE | NLM_F_EXCL);
|
|
|
|
if (!nl_msg) {
|
|
|
|
virReportOOMError();
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nlmsg_append(nl_msg, &ifinfo, sizeof(ifinfo), NLMSG_ALIGNTO) < 0)
|
|
|
|
goto buffer_too_small;
|
|
|
|
|
|
|
|
if (nla_put_u32(nl_msg, IFLA_LINK, ifindex) < 0)
|
|
|
|
goto buffer_too_small;
|
|
|
|
|
|
|
|
if (nla_put(nl_msg, IFLA_ADDRESS, VIR_MAC_BUFLEN, macaddress) < 0)
|
|
|
|
goto buffer_too_small;
|
|
|
|
|
|
|
|
if (ifname &&
|
|
|
|
nla_put(nl_msg, IFLA_IFNAME, strlen(ifname)+1, ifname) < 0)
|
|
|
|
goto buffer_too_small;
|
|
|
|
|
|
|
|
if (!(linkinfo = nla_nest_start(nl_msg, IFLA_LINKINFO)))
|
|
|
|
goto buffer_too_small;
|
|
|
|
|
|
|
|
if (nla_put(nl_msg, IFLA_INFO_KIND, strlen(type), type) < 0)
|
|
|
|
goto buffer_too_small;
|
|
|
|
|
|
|
|
if (macvlan_mode > 0) {
|
|
|
|
if (!(info_data = nla_nest_start(nl_msg, IFLA_INFO_DATA)))
|
|
|
|
goto buffer_too_small;
|
|
|
|
|
|
|
|
if (nla_put(nl_msg, IFLA_MACVLAN_MODE, sizeof(macvlan_mode),
|
|
|
|
&macvlan_mode) < 0)
|
|
|
|
goto buffer_too_small;
|
|
|
|
|
|
|
|
nla_nest_end(nl_msg, info_data);
|
|
|
|
}
|
|
|
|
|
|
|
|
nla_nest_end(nl_msg, linkinfo);
|
|
|
|
|
2013-04-03 14:09:19 +01:00
|
|
|
if (virNetlinkCommand(nl_msg, &resp, &recvbuflen, 0, 0,
|
2012-08-22 12:10:23 +08:00
|
|
|
NETLINK_ROUTE, 0) < 0) {
|
2011-11-02 17:34:41 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2013-04-03 14:09:19 +01:00
|
|
|
if (recvbuflen < NLMSG_LENGTH(0) || resp == NULL)
|
2011-11-02 17:34:41 +00:00
|
|
|
goto malformed_resp;
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
case -EEXIST:
|
|
|
|
*retry = 1;
|
util: standardize return from functions calling virNetlinkCommand
There are several functions that call virNetlinkCommand, and they all
follow a common pattern, with three exit labels: err_exit (or
cleanup), malformed_resp, and buffer_too_small. All three of these
labels do their own cleanup and have their own return. However, the
malformed_resp label usually frees the same items as the
cleanup/err_exit label, and the buffer_too_small label just doesn't
free recvbuf (because it's known to always be NULL at the time we goto
buffer_too_small.
In order to simplify and standardize the code, I've made the following
changes to all of these functions:
1) err_exit is replaced with the more libvirt-ish "cleanup", which
makes sense because in all cases this code is also executed in the
case of success, so labelling it err_exit may be confusing.
2) rc is initialized to -1, and set to 0 just before the cleanup
label. Any code that currently sets rc = -1 is made to instead goto
cleanup.
3) malformed_resp and buffer_too_small just log their error and goto
cleanup. This gives us a single return path, and a single place to
free up resources.
4) In one instance, rather then logging an error immediately, a char*
msg was pointed to an error string, then goto cleanup (and cleanup
would log an error if msg != NULL). It takes no more lines of code
to just log the message as we encounter it.
This patch should have 0 functional effects.
2012-03-07 12:44:56 -05:00
|
|
|
goto cleanup;
|
2011-11-02 17:34:41 +00:00
|
|
|
|
|
|
|
default:
|
|
|
|
virReportSystemError(-err->error,
|
2015-03-18 14:27:05 -04:00
|
|
|
_("error creating %s interface %s@%s (%s)"),
|
|
|
|
type, ifname, srcdev,
|
|
|
|
virMacAddrFormat(macaddress, macstr));
|
util: standardize return from functions calling virNetlinkCommand
There are several functions that call virNetlinkCommand, and they all
follow a common pattern, with three exit labels: err_exit (or
cleanup), malformed_resp, and buffer_too_small. All three of these
labels do their own cleanup and have their own return. However, the
malformed_resp label usually frees the same items as the
cleanup/err_exit label, and the buffer_too_small label just doesn't
free recvbuf (because it's known to always be NULL at the time we goto
buffer_too_small.
In order to simplify and standardize the code, I've made the following
changes to all of these functions:
1) err_exit is replaced with the more libvirt-ish "cleanup", which
makes sense because in all cases this code is also executed in the
case of success, so labelling it err_exit may be confusing.
2) rc is initialized to -1, and set to 0 just before the cleanup
label. Any code that currently sets rc = -1 is made to instead goto
cleanup.
3) malformed_resp and buffer_too_small just log their error and goto
cleanup. This gives us a single return path, and a single place to
free up resources.
4) In one instance, rather then logging an error immediately, a char*
msg was pointed to an error string, then goto cleanup (and cleanup
would log an error if msg != NULL). It takes no more lines of code
to just log the message as we encounter it.
This patch should have 0 functional effects.
2012-03-07 12:44:56 -05:00
|
|
|
goto cleanup;
|
2011-11-02 17:34:41 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NLMSG_DONE:
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
goto malformed_resp;
|
|
|
|
}
|
|
|
|
|
util: standardize return from functions calling virNetlinkCommand
There are several functions that call virNetlinkCommand, and they all
follow a common pattern, with three exit labels: err_exit (or
cleanup), malformed_resp, and buffer_too_small. All three of these
labels do their own cleanup and have their own return. However, the
malformed_resp label usually frees the same items as the
cleanup/err_exit label, and the buffer_too_small label just doesn't
free recvbuf (because it's known to always be NULL at the time we goto
buffer_too_small.
In order to simplify and standardize the code, I've made the following
changes to all of these functions:
1) err_exit is replaced with the more libvirt-ish "cleanup", which
makes sense because in all cases this code is also executed in the
case of success, so labelling it err_exit may be confusing.
2) rc is initialized to -1, and set to 0 just before the cleanup
label. Any code that currently sets rc = -1 is made to instead goto
cleanup.
3) malformed_resp and buffer_too_small just log their error and goto
cleanup. This gives us a single return path, and a single place to
free up resources.
4) In one instance, rather then logging an error immediately, a char*
msg was pointed to an error string, then goto cleanup (and cleanup
would log an error if msg != NULL). It takes no more lines of code
to just log the message as we encounter it.
This patch should have 0 functional effects.
2012-03-07 12:44:56 -05:00
|
|
|
rc = 0;
|
2014-03-25 07:53:22 +01:00
|
|
|
cleanup:
|
2011-11-02 17:34:41 +00:00
|
|
|
nlmsg_free(nl_msg);
|
2013-04-03 14:09:19 +01:00
|
|
|
VIR_FREE(resp);
|
2011-11-02 17:34:41 +00:00
|
|
|
return rc;
|
|
|
|
|
2014-03-25 07:53:22 +01:00
|
|
|
malformed_resp:
|
2012-07-18 11:26:24 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2011-11-02 17:34:41 +00:00
|
|
|
_("malformed netlink response message"));
|
util: standardize return from functions calling virNetlinkCommand
There are several functions that call virNetlinkCommand, and they all
follow a common pattern, with three exit labels: err_exit (or
cleanup), malformed_resp, and buffer_too_small. All three of these
labels do their own cleanup and have their own return. However, the
malformed_resp label usually frees the same items as the
cleanup/err_exit label, and the buffer_too_small label just doesn't
free recvbuf (because it's known to always be NULL at the time we goto
buffer_too_small.
In order to simplify and standardize the code, I've made the following
changes to all of these functions:
1) err_exit is replaced with the more libvirt-ish "cleanup", which
makes sense because in all cases this code is also executed in the
case of success, so labelling it err_exit may be confusing.
2) rc is initialized to -1, and set to 0 just before the cleanup
label. Any code that currently sets rc = -1 is made to instead goto
cleanup.
3) malformed_resp and buffer_too_small just log their error and goto
cleanup. This gives us a single return path, and a single place to
free up resources.
4) In one instance, rather then logging an error immediately, a char*
msg was pointed to an error string, then goto cleanup (and cleanup
would log an error if msg != NULL). It takes no more lines of code
to just log the message as we encounter it.
This patch should have 0 functional effects.
2012-03-07 12:44:56 -05:00
|
|
|
goto cleanup;
|
2011-11-02 17:34:41 +00:00
|
|
|
|
2014-03-25 07:53:22 +01:00
|
|
|
buffer_too_small:
|
2012-07-18 11:26:24 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2011-11-02 17:34:41 +00:00
|
|
|
_("allocated netlink buffer is too small"));
|
util: standardize return from functions calling virNetlinkCommand
There are several functions that call virNetlinkCommand, and they all
follow a common pattern, with three exit labels: err_exit (or
cleanup), malformed_resp, and buffer_too_small. All three of these
labels do their own cleanup and have their own return. However, the
malformed_resp label usually frees the same items as the
cleanup/err_exit label, and the buffer_too_small label just doesn't
free recvbuf (because it's known to always be NULL at the time we goto
buffer_too_small.
In order to simplify and standardize the code, I've made the following
changes to all of these functions:
1) err_exit is replaced with the more libvirt-ish "cleanup", which
makes sense because in all cases this code is also executed in the
case of success, so labelling it err_exit may be confusing.
2) rc is initialized to -1, and set to 0 just before the cleanup
label. Any code that currently sets rc = -1 is made to instead goto
cleanup.
3) malformed_resp and buffer_too_small just log their error and goto
cleanup. This gives us a single return path, and a single place to
free up resources.
4) In one instance, rather then logging an error immediately, a char*
msg was pointed to an error string, then goto cleanup (and cleanup
would log an error if msg != NULL). It takes no more lines of code
to just log the message as we encounter it.
This patch should have 0 functional effects.
2012-03-07 12:44:56 -05:00
|
|
|
goto cleanup;
|
2011-11-02 17:34:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* virNetDevMacVLanDelete:
|
|
|
|
*
|
|
|
|
* @ifname: Name of the interface
|
|
|
|
*
|
|
|
|
* Tear down an interface with the given name.
|
|
|
|
*
|
|
|
|
* Returns 0 on success, -1 on fatal error.
|
|
|
|
*/
|
|
|
|
int virNetDevMacVLanDelete(const char *ifname)
|
|
|
|
{
|
util: fallback to ioctl(SIOCBRDELBR) if netlink RTM_DELLINK fails
commit 09778e09 switched from using ioctl(SIOCBRDELBR) for bridge
device deletion to using a netlink RTM_DELLINK message, which is the
more modern way to delete a bridge (and also doesn't require the
bridge to be ~IFF_UP to succeed). However, although older kernels
(e.g. 2.6.32, in RHEL6/CentOS6) support deleting *some* link types
with RTM_NEWLINK, they don't support deleting bridges, and there is no
compile-time way to figure this out.
This patch moves the body of the SIOCBRDELBR version of
virNetDevBridgeDelete() into a static function, calls the new function
from the original, and also calls the new function from the
RTM_DELLINK version if the RTM_DELLINK message generates an EOPNOTSUPP
error. Since RTM_DELLINK is done from the subordinate function
virNetlinkDelLink, which is also called for other purposes (deleting a
macvtap interface), a function pointer called "fallback" has been
added to the arglist of virNetlinkDelLink() - if that arg != NULL, the
provided function will be called when (and only when) RTM_DELLINK
fails with EOPNOTSUPP.
Resolves: https://bugzilla.redhat.com/show_bug.cgi?id=1252780 (part 2)
2015-08-25 23:19:03 -04:00
|
|
|
return virNetlinkDelLink(ifname, NULL);
|
2011-11-02 17:34:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-11-02 17:11:02 +00:00
|
|
|
/**
|
|
|
|
* virNetDevMacVLanTapOpen:
|
|
|
|
* @ifname: Name of the macvtap interface
|
2015-12-04 09:39:02 +01:00
|
|
|
* @tapfd: array of file descriptor return value for the new macvtap device
|
|
|
|
* @tapfdSize: number of file descriptors in @tapfd
|
2011-11-02 17:11:02 +00:00
|
|
|
* @retries : Number of retries in case udev for example may need to be
|
|
|
|
* waited for to create the tap chardev
|
2015-12-04 09:39:02 +01:00
|
|
|
*
|
|
|
|
* Open the macvtap's tap device, possibly multiple times if @tapfdSize > 1.
|
|
|
|
*
|
|
|
|
* Returns 0 on success, -1 otherwise.
|
2011-11-02 17:11:02 +00:00
|
|
|
*/
|
2015-12-04 09:39:02 +01:00
|
|
|
static int
|
|
|
|
virNetDevMacVLanTapOpen(const char *ifname,
|
|
|
|
int *tapfd,
|
|
|
|
size_t tapfdSize,
|
|
|
|
int retries)
|
2011-11-02 17:11:02 +00:00
|
|
|
{
|
2015-04-15 11:45:47 +02:00
|
|
|
int ret = -1;
|
2011-11-02 17:11:02 +00:00
|
|
|
int ifindex;
|
2015-12-04 10:55:49 +01:00
|
|
|
char *tapname = NULL;
|
2015-12-04 09:39:02 +01:00
|
|
|
size_t i = 0;
|
2011-11-02 17:11:02 +00:00
|
|
|
|
2015-12-04 10:55:49 +01:00
|
|
|
if (virNetDevGetIndex(ifname, &ifindex) < 0)
|
2011-11-02 17:11:02 +00:00
|
|
|
return -1;
|
|
|
|
|
2015-12-04 10:55:49 +01:00
|
|
|
if (virAsprintf(&tapname, "/dev/tap%d", ifindex) < 0)
|
2015-04-15 11:45:47 +02:00
|
|
|
goto cleanup;
|
2011-11-02 17:11:02 +00:00
|
|
|
|
2015-12-04 09:39:02 +01:00
|
|
|
for (i = 0; i < tapfdSize; i++) {
|
|
|
|
int fd = -1;
|
|
|
|
|
|
|
|
while (fd < 0) {
|
|
|
|
if ((fd = open(tapname, O_RDWR)) >= 0) {
|
|
|
|
tapfd[i] = fd;
|
|
|
|
} else if (retries-- > 0) {
|
|
|
|
/* may need to wait for udev to be done */
|
|
|
|
usleep(20000);
|
|
|
|
} else {
|
|
|
|
/* However, if haven't succeeded, quit. */
|
|
|
|
virReportSystemError(errno,
|
|
|
|
_("cannot open macvtap tap device %s"),
|
|
|
|
tapname);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2011-11-02 17:11:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-04 09:39:02 +01:00
|
|
|
ret = 0;
|
|
|
|
|
2015-04-15 11:45:47 +02:00
|
|
|
cleanup:
|
2015-12-04 09:39:02 +01:00
|
|
|
if (ret < 0) {
|
|
|
|
while (i--)
|
|
|
|
VIR_FORCE_CLOSE(tapfd[i]);
|
|
|
|
}
|
2015-12-04 10:55:49 +01:00
|
|
|
VIR_FREE(tapname);
|
2015-04-15 11:45:47 +02:00
|
|
|
return ret;
|
2011-11-02 17:11:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* virNetDevMacVLanTapSetup:
|
2015-12-04 11:19:32 +01:00
|
|
|
* @tapfd: array of file descriptors of the macvtap tap
|
|
|
|
* @tapfdSize: number of file descriptors in @tapfd
|
|
|
|
* @vnet_hdr: whether to enable or disable IFF_VNET_HDR
|
2015-12-08 13:17:26 +01:00
|
|
|
*
|
2015-12-13 07:54:46 +01:00
|
|
|
* Turn on the IFF_VNET_HDR flag if requested and available, but make sure
|
|
|
|
* it's off otherwise. Similarly, turn on IFF_MULTI_QUEUE if @tapfdSize is
|
|
|
|
* greater than one, but if it can't be set, consider it a fatal error
|
|
|
|
* (rather than ignoring as with @vnet_hdr).
|
2011-11-02 17:11:02 +00:00
|
|
|
*
|
|
|
|
* A fatal error is defined as the VNET_HDR flag being set but it cannot
|
|
|
|
* be turned off for some reason. This is reported with -1. Other fatal
|
|
|
|
* error is not being able to read the interface flags. In that case the
|
|
|
|
* macvtap device should not be used.
|
2015-12-04 11:19:32 +01:00
|
|
|
*
|
2015-12-08 13:17:26 +01:00
|
|
|
* Returns 0 on success, -1 in case of fatal error.
|
2011-11-02 17:11:02 +00:00
|
|
|
*/
|
|
|
|
static int
|
2015-12-13 07:54:46 +01:00
|
|
|
virNetDevMacVLanTapSetup(int *tapfd, size_t tapfdSize, bool vnet_hdr)
|
2011-11-02 17:11:02 +00:00
|
|
|
{
|
|
|
|
unsigned int features;
|
|
|
|
struct ifreq ifreq;
|
|
|
|
short new_flags = 0;
|
2015-12-04 11:19:32 +01:00
|
|
|
size_t i;
|
2011-11-02 17:11:02 +00:00
|
|
|
|
2015-12-04 11:19:32 +01:00
|
|
|
for (i = 0; i < tapfdSize; i++) {
|
|
|
|
memset(&ifreq, 0, sizeof(ifreq));
|
2011-11-02 17:11:02 +00:00
|
|
|
|
2015-12-04 11:19:32 +01:00
|
|
|
if (ioctl(tapfd[i], TUNGETIFF, &ifreq) < 0) {
|
2011-11-02 17:11:02 +00:00
|
|
|
virReportSystemError(errno, "%s",
|
2015-12-04 11:19:32 +01:00
|
|
|
_("cannot get interface flags on macvtap tap"));
|
2011-11-02 17:11:02 +00:00
|
|
|
return -1;
|
|
|
|
}
|
2015-12-04 11:19:32 +01:00
|
|
|
|
|
|
|
new_flags = ifreq.ifr_flags;
|
|
|
|
|
2015-12-08 13:17:26 +01:00
|
|
|
if (vnet_hdr) {
|
2015-12-04 11:19:32 +01:00
|
|
|
if (ioctl(tapfd[i], TUNGETFEATURES, &features) < 0) {
|
|
|
|
virReportSystemError(errno, "%s",
|
|
|
|
_("cannot get feature flags on macvtap tap"));
|
|
|
|
return -1;
|
|
|
|
}
|
2015-12-08 13:17:26 +01:00
|
|
|
if (features & IFF_VNET_HDR)
|
|
|
|
new_flags |= IFF_VNET_HDR;
|
|
|
|
} else {
|
|
|
|
new_flags &= ~IFF_VNET_HDR;
|
2011-11-02 17:11:02 +00:00
|
|
|
}
|
|
|
|
|
2015-12-12 08:05:17 +01:00
|
|
|
# ifdef IFF_MULTI_QUEUE
|
2015-12-13 07:54:46 +01:00
|
|
|
if (tapfdSize > 1)
|
2015-12-08 13:17:26 +01:00
|
|
|
new_flags |= IFF_MULTI_QUEUE;
|
|
|
|
else
|
|
|
|
new_flags &= ~IFF_MULTI_QUEUE;
|
2015-12-12 08:05:17 +01:00
|
|
|
# else
|
2015-12-13 07:54:46 +01:00
|
|
|
if (tapfdSize > 1) {
|
2015-12-12 08:05:17 +01:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("Multiqueue devices are not supported on this system"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
# endif
|
2015-12-08 13:17:26 +01:00
|
|
|
|
2015-12-04 11:19:32 +01:00
|
|
|
if (new_flags != ifreq.ifr_flags) {
|
|
|
|
ifreq.ifr_flags = new_flags;
|
|
|
|
if (ioctl(tapfd[i], TUNSETIFF, &ifreq) < 0) {
|
2015-12-08 13:17:26 +01:00
|
|
|
virReportSystemError(errno, "%s",
|
|
|
|
_("unable to set vnet or multiqueue flags on macvtap"));
|
|
|
|
return -1;
|
2015-12-04 11:19:32 +01:00
|
|
|
}
|
2011-11-02 17:11:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static const uint32_t modeMap[VIR_NETDEV_MACVLAN_MODE_LAST] = {
|
|
|
|
[VIR_NETDEV_MACVLAN_MODE_VEPA] = MACVLAN_MODE_VEPA,
|
|
|
|
[VIR_NETDEV_MACVLAN_MODE_PRIVATE] = MACVLAN_MODE_PRIVATE,
|
|
|
|
[VIR_NETDEV_MACVLAN_MODE_BRIDGE] = MACVLAN_MODE_BRIDGE,
|
|
|
|
[VIR_NETDEV_MACVLAN_MODE_PASSTHRU] = MACVLAN_MODE_PASSTHRU,
|
|
|
|
};
|
|
|
|
|
2012-02-22 14:17:14 +01:00
|
|
|
/* Struct to hold the state and configuration of a 802.1qbg port */
|
|
|
|
struct virNetlinkCallbackData {
|
|
|
|
char *cr_ifname;
|
|
|
|
virNetDevVPortProfilePtr virtPortProfile;
|
2012-07-17 08:07:59 -04:00
|
|
|
virMacAddr macaddress;
|
2012-02-22 14:17:14 +01:00
|
|
|
char *linkdev;
|
2012-03-05 17:12:39 -08:00
|
|
|
int vf;
|
2012-04-25 07:55:07 -04:00
|
|
|
unsigned char vmuuid[VIR_UUID_BUFLEN];
|
2014-04-26 21:15:22 -03:00
|
|
|
virNetDevVPortProfileOp vmOp;
|
2012-02-22 14:17:14 +01:00
|
|
|
unsigned int linkState;
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct virNetlinkCallbackData *virNetlinkCallbackDataPtr;
|
|
|
|
|
|
|
|
# define INSTANCE_STRLEN 36
|
|
|
|
|
|
|
|
static int instance2str(const unsigned char *p, char *dst, size_t size)
|
|
|
|
{
|
|
|
|
if (dst && size > INSTANCE_STRLEN) {
|
|
|
|
snprintf(dst, size, "%02x%02x%02x%02x-%02x%02x-%02x%02x-"
|
|
|
|
"%02x%02x-%02x%02x%02x%02x%02x%02x",
|
|
|
|
p[0], p[1], p[2], p[3],
|
|
|
|
p[4], p[5], p[6], p[7],
|
|
|
|
p[8], p[9], p[10], p[11], p[12], p[13], p[14], p[15]);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
# define LLDPAD_PID_FILE "/var/run/lldpad.pid"
|
|
|
|
# define VIRIP_PID_FILE "/var/run/virip.pid"
|
|
|
|
|
|
|
|
/**
|
|
|
|
* virNetDevMacVLanVPortProfileCallback:
|
|
|
|
*
|
2013-04-03 14:09:19 +01:00
|
|
|
* @hdr: The buffer containing the received netlink header + payload
|
2012-02-22 14:17:14 +01:00
|
|
|
* @length: The length of the received netlink message.
|
|
|
|
* @peer: The netling sockaddr containing the peer information
|
|
|
|
* @handled: Contains information if the message has been replied to yet
|
|
|
|
* @opaque: Contains vital information regarding the associated vm an interface
|
|
|
|
*
|
|
|
|
* This function is called when a netlink message is received. The function
|
|
|
|
* reads the message and responds if it is pertinent to the running VMs
|
|
|
|
* network interface.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void
|
2013-04-03 14:09:19 +01:00
|
|
|
virNetDevMacVLanVPortProfileCallback(struct nlmsghdr *hdr,
|
|
|
|
unsigned int length,
|
2012-02-22 14:17:14 +01:00
|
|
|
struct sockaddr_nl *peer,
|
|
|
|
bool *handled,
|
|
|
|
void *opaque)
|
|
|
|
{
|
|
|
|
struct nla_policy ifla_vf_policy[IFLA_VF_MAX + 1] = {
|
|
|
|
[IFLA_VF_MAC] = {.minlen = sizeof(struct ifla_vf_mac),
|
|
|
|
.maxlen = sizeof(struct ifla_vf_mac)},
|
|
|
|
[IFLA_VF_VLAN] = {.minlen = sizeof(struct ifla_vf_vlan),
|
|
|
|
.maxlen = sizeof(struct ifla_vf_vlan)},
|
|
|
|
};
|
|
|
|
|
|
|
|
struct nla_policy ifla_port_policy[IFLA_PORT_MAX + 1] = {
|
|
|
|
[IFLA_PORT_RESPONSE] = {.type = NLA_U16},
|
|
|
|
};
|
|
|
|
|
|
|
|
struct nlattr *tb[IFLA_MAX + 1], *tb3[IFLA_PORT_MAX + 1],
|
|
|
|
*tb_vfinfo[IFLA_VF_MAX + 1], *tb_vfinfo_list;
|
|
|
|
|
|
|
|
struct ifinfomsg ifinfo;
|
|
|
|
void *data;
|
|
|
|
int rem;
|
|
|
|
char *ifname;
|
|
|
|
bool indicate = false;
|
|
|
|
virNetlinkCallbackDataPtr calld = opaque;
|
|
|
|
pid_t lldpad_pid = 0;
|
|
|
|
pid_t virip_pid = 0;
|
2013-03-26 12:21:33 +01:00
|
|
|
char macaddr[VIR_MAC_STRING_BUFLEN];
|
2012-02-22 14:17:14 +01:00
|
|
|
|
|
|
|
data = nlmsg_data(hdr);
|
|
|
|
|
|
|
|
/* Quickly decide if we want this or not */
|
|
|
|
|
|
|
|
if (virPidFileReadPath(LLDPAD_PID_FILE, &lldpad_pid) < 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
ignore_value(virPidFileReadPath(VIRIP_PID_FILE, &virip_pid));
|
|
|
|
|
|
|
|
if (hdr->nlmsg_pid != lldpad_pid && hdr->nlmsg_pid != virip_pid)
|
|
|
|
return; /* we only care for lldpad and virip messages */
|
|
|
|
if (hdr->nlmsg_type != RTM_SETLINK)
|
|
|
|
return; /* we only care for RTM_SETLINK */
|
|
|
|
if (*handled)
|
|
|
|
return; /* if it has been handled - dont handle again */
|
|
|
|
|
|
|
|
/* DEBUG start */
|
|
|
|
VIR_INFO("netlink message nl_sockaddr: %p len: %d", peer, length);
|
|
|
|
VIR_DEBUG("nlmsg_type = 0x%02x", hdr->nlmsg_type);
|
|
|
|
VIR_DEBUG("nlmsg_len = 0x%04x", hdr->nlmsg_len);
|
|
|
|
VIR_DEBUG("nlmsg_pid = %d", hdr->nlmsg_pid);
|
|
|
|
VIR_DEBUG("nlmsg_seq = 0x%08x", hdr->nlmsg_seq);
|
|
|
|
VIR_DEBUG("nlmsg_flags = 0x%04x", hdr->nlmsg_flags);
|
|
|
|
|
|
|
|
VIR_DEBUG("lldpad pid = %d", lldpad_pid);
|
|
|
|
|
|
|
|
switch (hdr->nlmsg_type) {
|
|
|
|
case RTM_NEWLINK:
|
|
|
|
case RTM_DELLINK:
|
|
|
|
case RTM_SETLINK:
|
|
|
|
case RTM_GETLINK:
|
2015-10-27 19:14:01 +01:00
|
|
|
VIR_DEBUG(" IFINFOMSG");
|
|
|
|
VIR_DEBUG(" ifi_family = 0x%02x",
|
2012-02-22 14:17:14 +01:00
|
|
|
((struct ifinfomsg *)data)->ifi_family);
|
2015-10-27 19:14:01 +01:00
|
|
|
VIR_DEBUG(" ifi_type = 0x%x",
|
2012-02-22 14:17:14 +01:00
|
|
|
((struct ifinfomsg *)data)->ifi_type);
|
2015-10-27 19:14:01 +01:00
|
|
|
VIR_DEBUG(" ifi_index = %i",
|
2012-02-22 14:17:14 +01:00
|
|
|
((struct ifinfomsg *)data)->ifi_index);
|
2015-10-27 19:14:01 +01:00
|
|
|
VIR_DEBUG(" ifi_flags = 0x%04x",
|
2012-02-22 14:17:14 +01:00
|
|
|
((struct ifinfomsg *)data)->ifi_flags);
|
2015-10-27 19:14:01 +01:00
|
|
|
VIR_DEBUG(" ifi_change = 0x%04x",
|
2012-02-22 14:17:14 +01:00
|
|
|
((struct ifinfomsg *)data)->ifi_change);
|
|
|
|
}
|
|
|
|
/* DEBUG end */
|
|
|
|
|
|
|
|
/* Parse netlink message assume a setlink with vfports */
|
2012-03-29 10:52:04 +01:00
|
|
|
memcpy(&ifinfo, NLMSG_DATA(hdr), sizeof(ifinfo));
|
2012-02-22 14:17:14 +01:00
|
|
|
VIR_DEBUG("family:%#x type:%#x index:%d flags:%#x change:%#x",
|
|
|
|
ifinfo.ifi_family, ifinfo.ifi_type, ifinfo.ifi_index,
|
|
|
|
ifinfo.ifi_flags, ifinfo.ifi_change);
|
2012-03-29 10:52:04 +01:00
|
|
|
if (nlmsg_parse(hdr, sizeof(ifinfo),
|
2012-02-22 14:17:14 +01:00
|
|
|
(struct nlattr **)&tb, IFLA_MAX, NULL)) {
|
|
|
|
VIR_DEBUG("error parsing request...");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tb[IFLA_VFINFO_LIST]) {
|
|
|
|
VIR_DEBUG("FOUND IFLA_VFINFO_LIST!");
|
|
|
|
|
|
|
|
nla_for_each_nested(tb_vfinfo_list, tb[IFLA_VFINFO_LIST], rem) {
|
|
|
|
if (nla_type(tb_vfinfo_list) != IFLA_VF_INFO) {
|
|
|
|
VIR_DEBUG("nested parsing of"
|
|
|
|
"IFLA_VFINFO_LIST failed.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (nla_parse_nested(tb_vfinfo, IFLA_VF_MAX,
|
|
|
|
tb_vfinfo_list, ifla_vf_policy)) {
|
|
|
|
VIR_DEBUG("nested parsing of "
|
|
|
|
"IFLA_VF_INFO failed.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tb_vfinfo[IFLA_VF_MAC]) {
|
|
|
|
struct ifla_vf_mac *mac = RTA_DATA(tb_vfinfo[IFLA_VF_MAC]);
|
|
|
|
unsigned char *m = mac->mac;
|
|
|
|
|
|
|
|
VIR_DEBUG("IFLA_VF_MAC = %2x:%2x:%2x:%2x:%2x:%2x",
|
|
|
|
m[0], m[1], m[2], m[3], m[4], m[5]);
|
|
|
|
|
2014-09-03 16:24:43 -06:00
|
|
|
if (virMacAddrCmpRaw(&calld->macaddress, mac->mac)) {
|
2012-02-22 14:17:14 +01:00
|
|
|
/* Repeat the same check for a broadcast mac */
|
Convert 'int i' to 'size_t i' in src/util/ files
Convert the type of loop iterators named 'i', 'j', k',
'ii', 'jj', 'kk', to be 'size_t' instead of 'int' or
'unsigned int', also santizing 'ii', 'jj', 'kk' to use
the normal 'i', 'j', 'k' naming
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2013-07-08 15:09:33 +01:00
|
|
|
size_t i;
|
2012-02-22 14:17:14 +01:00
|
|
|
|
2012-07-17 08:07:59 -04:00
|
|
|
for (i = 0; i < VIR_MAC_BUFLEN; i++) {
|
|
|
|
if (calld->macaddress.addr[i] != 0xff) {
|
2012-02-22 14:17:14 +01:00
|
|
|
VIR_DEBUG("MAC address match failed (wasn't broadcast)");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tb_vfinfo[IFLA_VF_VLAN]) {
|
|
|
|
struct ifla_vf_vlan *vlan = RTA_DATA(tb_vfinfo[IFLA_VF_VLAN]);
|
|
|
|
|
|
|
|
VIR_DEBUG("IFLA_VF_VLAN = %d", vlan->vlan);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tb[IFLA_IFNAME]) {
|
|
|
|
ifname = (char *)RTA_DATA(tb[IFLA_IFNAME]);
|
2015-10-27 19:14:01 +01:00
|
|
|
VIR_DEBUG("IFLA_IFNAME = %s", ifname);
|
2012-02-22 14:17:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (tb[IFLA_OPERSTATE]) {
|
|
|
|
rem = *(unsigned short *)RTA_DATA(tb[IFLA_OPERSTATE]);
|
2015-10-27 19:14:01 +01:00
|
|
|
VIR_DEBUG("IFLA_OPERSTATE = %d", rem);
|
2012-02-22 14:17:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (tb[IFLA_VF_PORTS]) {
|
|
|
|
struct nlattr *tb_vf_ports;
|
|
|
|
|
2015-10-27 19:14:01 +01:00
|
|
|
VIR_DEBUG("found IFLA_VF_PORTS");
|
2012-02-22 14:17:14 +01:00
|
|
|
nla_for_each_nested(tb_vf_ports, tb[IFLA_VF_PORTS], rem) {
|
|
|
|
|
2015-10-27 19:14:01 +01:00
|
|
|
VIR_DEBUG("iterating");
|
2012-02-22 14:17:14 +01:00
|
|
|
if (nla_type(tb_vf_ports) != IFLA_VF_PORT) {
|
2015-10-27 19:14:01 +01:00
|
|
|
VIR_DEBUG("not a IFLA_VF_PORT. skipping");
|
2012-02-22 14:17:14 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (nla_parse_nested(tb3, IFLA_PORT_MAX, tb_vf_ports,
|
|
|
|
ifla_port_policy)) {
|
|
|
|
VIR_DEBUG("nested parsing on level 2"
|
|
|
|
" failed.");
|
|
|
|
}
|
|
|
|
if (tb3[IFLA_PORT_VF]) {
|
|
|
|
VIR_DEBUG("IFLA_PORT_VF = %d",
|
|
|
|
*(uint32_t *) (RTA_DATA(tb3[IFLA_PORT_VF])));
|
|
|
|
}
|
|
|
|
if (tb3[IFLA_PORT_PROFILE]) {
|
|
|
|
VIR_DEBUG("IFLA_PORT_PROFILE = %s",
|
|
|
|
(char *) RTA_DATA(tb3[IFLA_PORT_PROFILE]));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tb3[IFLA_PORT_VSI_TYPE]) {
|
|
|
|
struct ifla_port_vsi *pvsi;
|
|
|
|
int tid = 0;
|
|
|
|
|
|
|
|
pvsi = (struct ifla_port_vsi *)
|
|
|
|
RTA_DATA(tb3[IFLA_PORT_VSI_TYPE]);
|
|
|
|
tid = ((pvsi->vsi_type_id[2] << 16) |
|
|
|
|
(pvsi->vsi_type_id[1] << 8) |
|
|
|
|
pvsi->vsi_type_id[0]);
|
|
|
|
|
|
|
|
VIR_DEBUG("mgr_id: %d", pvsi->vsi_mgr_id);
|
|
|
|
VIR_DEBUG("type_id: %d", tid);
|
|
|
|
VIR_DEBUG("type_version: %d",
|
|
|
|
pvsi->vsi_type_version);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tb3[IFLA_PORT_INSTANCE_UUID]) {
|
|
|
|
char instance[INSTANCE_STRLEN + 2];
|
|
|
|
unsigned char *uuid;
|
|
|
|
|
|
|
|
uuid = (unsigned char *)
|
|
|
|
RTA_DATA(tb3[IFLA_PORT_INSTANCE_UUID]);
|
|
|
|
instance2str(uuid, instance, sizeof(instance));
|
2015-10-27 19:14:01 +01:00
|
|
|
VIR_DEBUG("IFLA_PORT_INSTANCE_UUID = %s",
|
2012-02-22 14:17:14 +01:00
|
|
|
instance);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tb3[IFLA_PORT_REQUEST]) {
|
|
|
|
uint8_t req = *(uint8_t *) RTA_DATA(tb3[IFLA_PORT_REQUEST]);
|
|
|
|
VIR_DEBUG("IFLA_PORT_REQUEST = %d", req);
|
|
|
|
|
|
|
|
if (req == PORT_REQUEST_DISASSOCIATE) {
|
|
|
|
VIR_DEBUG("Set dissaccociated.");
|
|
|
|
indicate = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tb3[IFLA_PORT_RESPONSE]) {
|
2015-10-27 19:14:01 +01:00
|
|
|
VIR_DEBUG("IFLA_PORT_RESPONSE = %d", *(uint16_t *)
|
2012-02-22 14:17:14 +01:00
|
|
|
RTA_DATA(tb3[IFLA_PORT_RESPONSE]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-13 15:28:18 +01:00
|
|
|
if (!indicate)
|
2012-02-22 14:17:14 +01:00
|
|
|
return;
|
|
|
|
|
|
|
|
VIR_INFO("Re-send 802.1qbg associate request:");
|
|
|
|
VIR_INFO(" if: %s", calld->cr_ifname);
|
|
|
|
VIR_INFO(" lf: %s", calld->linkdev);
|
2013-03-26 12:21:33 +01:00
|
|
|
VIR_INFO(" mac: %s", virMacAddrFormat(&calld->macaddress, macaddr));
|
2012-02-22 14:17:14 +01:00
|
|
|
ignore_value(virNetDevVPortProfileAssociate(calld->cr_ifname,
|
|
|
|
calld->virtPortProfile,
|
2012-07-17 08:07:59 -04:00
|
|
|
&calld->macaddress,
|
2012-02-22 14:17:14 +01:00
|
|
|
calld->linkdev,
|
2012-03-05 17:12:39 -08:00
|
|
|
calld->vf,
|
2012-02-22 14:17:14 +01:00
|
|
|
calld->vmuuid,
|
|
|
|
calld->vmOp, true));
|
|
|
|
*handled = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* virNetlinkCallbackDataFree
|
|
|
|
*
|
|
|
|
* @calld: pointer to a virNetlinkCallbackData object to free
|
|
|
|
*
|
|
|
|
* This function frees all the data associated with a virNetlinkCallbackData object
|
|
|
|
* as well as the object itself. If called with NULL, it does nothing.
|
|
|
|
*
|
|
|
|
* Returns nothing.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
virNetlinkCallbackDataFree(virNetlinkCallbackDataPtr calld)
|
|
|
|
{
|
|
|
|
if (calld) {
|
|
|
|
VIR_FREE(calld->cr_ifname);
|
|
|
|
VIR_FREE(calld->virtPortProfile);
|
|
|
|
VIR_FREE(calld->linkdev);
|
|
|
|
}
|
|
|
|
VIR_FREE(calld);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* virNetDevMacVLanVPortProfileDestroyCallback:
|
|
|
|
*
|
|
|
|
* @watch: watch whose handle to remove
|
|
|
|
* @macaddr: macaddr whose handle to remove
|
|
|
|
* @opaque: Contains vital information regarding the associated vm
|
|
|
|
*
|
|
|
|
* This function is called when a netlink message handler is terminated.
|
|
|
|
* The function frees locally allocated data referenced in the opaque
|
|
|
|
* data, and the opaque object itself.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
virNetDevMacVLanVPortProfileDestroyCallback(int watch ATTRIBUTE_UNUSED,
|
2013-10-05 13:41:44 -06:00
|
|
|
const virMacAddr *macaddr ATTRIBUTE_UNUSED,
|
2012-02-22 14:17:14 +01:00
|
|
|
void *opaque)
|
|
|
|
{
|
|
|
|
virNetlinkCallbackDataFree((virNetlinkCallbackDataPtr)opaque);
|
|
|
|
}
|
|
|
|
|
2012-03-29 13:15:00 +02:00
|
|
|
int
|
2012-03-27 14:38:33 +02:00
|
|
|
virNetDevMacVLanVPortProfileRegisterCallback(const char *ifname,
|
2013-10-05 13:41:44 -06:00
|
|
|
const virMacAddr *macaddress,
|
2012-03-27 14:38:33 +02:00
|
|
|
const char *linkdev,
|
|
|
|
const unsigned char *vmuuid,
|
|
|
|
virNetDevVPortProfilePtr virtPortProfile,
|
2014-04-26 21:15:22 -03:00
|
|
|
virNetDevVPortProfileOp vmOp)
|
2012-03-27 14:38:33 +02:00
|
|
|
{
|
|
|
|
virNetlinkCallbackDataPtr calld = NULL;
|
|
|
|
|
2012-08-22 12:10:23 +08:00
|
|
|
if (virtPortProfile && virNetlinkEventServiceIsRunning(NETLINK_ROUTE)) {
|
2012-03-27 14:38:33 +02:00
|
|
|
if (VIR_ALLOC(calld) < 0)
|
2013-07-04 12:17:18 +02:00
|
|
|
goto error;
|
2013-05-24 09:19:51 +02:00
|
|
|
if (VIR_STRDUP(calld->cr_ifname, ifname) < 0)
|
|
|
|
goto error;
|
2012-03-27 14:38:33 +02:00
|
|
|
if (VIR_ALLOC(calld->virtPortProfile) < 0)
|
2013-07-04 12:17:18 +02:00
|
|
|
goto error;
|
2012-03-27 14:38:33 +02:00
|
|
|
memcpy(calld->virtPortProfile, virtPortProfile, sizeof(*virtPortProfile));
|
2012-07-17 08:07:59 -04:00
|
|
|
virMacAddrSet(&calld->macaddress, macaddress);
|
2013-05-24 09:19:51 +02:00
|
|
|
if (VIR_STRDUP(calld->linkdev, linkdev) < 0)
|
|
|
|
goto error;
|
2012-04-25 07:55:07 -04:00
|
|
|
memcpy(calld->vmuuid, vmuuid, sizeof(calld->vmuuid));
|
2012-03-27 14:38:33 +02:00
|
|
|
|
|
|
|
calld->vmOp = vmOp;
|
|
|
|
|
|
|
|
if (virNetlinkEventAddClient(virNetDevMacVLanVPortProfileCallback,
|
|
|
|
virNetDevMacVLanVPortProfileDestroyCallback,
|
2012-08-22 12:10:23 +08:00
|
|
|
calld, macaddress, NETLINK_ROUTE) < 0)
|
2012-03-27 14:38:33 +02:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
2014-03-25 07:53:22 +01:00
|
|
|
error:
|
2012-03-27 14:38:33 +02:00
|
|
|
virNetlinkCallbackDataFree(calld);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2012-02-22 14:17:14 +01:00
|
|
|
|
2011-11-02 17:11:02 +00:00
|
|
|
/**
|
2011-11-02 17:19:48 +00:00
|
|
|
* virNetDevMacVLanCreateWithVPortProfile:
|
2011-11-02 17:11:02 +00:00
|
|
|
* Create an instance of a macvtap device and open its tap character
|
|
|
|
* device.
|
2016-01-19 14:20:54 -05:00
|
|
|
|
|
|
|
* @ifnameRequested: Interface name that the caller wants the macvtap
|
|
|
|
* device to have, or NULL to pick the first available name
|
|
|
|
* appropriate for the type (macvlan%d or macvtap%d). If the
|
|
|
|
* suggested name fits one of those patterns, but is already in
|
|
|
|
* use, we will fallback to finding the first available. If the
|
|
|
|
* suggested name *doesn't* fit a pattern and the name is in use,
|
|
|
|
* we will fail.
|
2011-11-02 17:11:02 +00:00
|
|
|
* @macaddress: The MAC address for the macvtap device
|
|
|
|
* @linkdev: The interface name of the NIC to connect to the external bridge
|
2016-01-19 14:20:54 -05:00
|
|
|
* @mode: macvtap mode (VIR_NETDEV_MACVLAN_MODE_(BRIDGE|VEPA|PRIVATE|PASSTHRU)
|
2011-11-02 17:11:02 +00:00
|
|
|
* @vmuuid: The UUID of the VM the macvtap belongs to
|
|
|
|
* @virtPortProfile: pointer to object holding the virtual port profile data
|
2016-01-19 14:20:54 -05:00
|
|
|
* @ifnameResult: Pointer to a string pointer where the actual name of the
|
2011-11-02 17:11:02 +00:00
|
|
|
* interface will be stored into if everything succeeded. It is up
|
|
|
|
* to the caller to free the string.
|
2015-12-04 11:31:17 +01:00
|
|
|
* @tapfd: array of file descriptor return value for the new tap device
|
|
|
|
* @tapfdSize: number of file descriptors in @tapfd
|
2014-08-27 10:34:13 -04:00
|
|
|
* @flags: OR of virNetDevMacVLanCreateFlags.
|
2011-11-02 17:11:02 +00:00
|
|
|
*
|
2015-12-04 11:31:17 +01:00
|
|
|
* Creates a macvlan device. Optionally, if flags &
|
|
|
|
* VIR_NETDEV_MACVLAN_CREATE_WITH_TAP is set, @tapfd is populated with FDs of
|
|
|
|
* tap devices up to @tapfdSize.
|
|
|
|
*
|
|
|
|
* Return 0 on success, -1 on error.
|
2011-11-02 17:11:02 +00:00
|
|
|
*/
|
2016-01-19 14:20:54 -05:00
|
|
|
int
|
|
|
|
virNetDevMacVLanCreateWithVPortProfile(const char *ifnameRequested,
|
|
|
|
const virMacAddr *macaddress,
|
|
|
|
const char *linkdev,
|
|
|
|
virNetDevMacVLanMode mode,
|
2016-05-04 13:18:16 -04:00
|
|
|
virNetDevVlanPtr vlan,
|
2016-01-19 14:20:54 -05:00
|
|
|
const unsigned char *vmuuid,
|
|
|
|
virNetDevVPortProfilePtr virtPortProfile,
|
|
|
|
char **ifnameResult,
|
|
|
|
virNetDevVPortProfileOp vmOp,
|
|
|
|
char *stateDir,
|
|
|
|
int *tapfd,
|
|
|
|
size_t tapfdSize,
|
|
|
|
unsigned int flags)
|
2011-11-02 17:11:02 +00:00
|
|
|
{
|
2017-04-25 14:09:45 -04:00
|
|
|
const char *type = VIR_NET_GENERATED_PREFIX;
|
2014-08-27 10:34:13 -04:00
|
|
|
const char *pattern = (flags & VIR_NETDEV_MACVLAN_CREATE_WITH_TAP) ?
|
2017-04-25 14:09:45 -04:00
|
|
|
VIR_NET_GENERATED_MACVTAP_PATTERN : VIR_NET_GENERATED_MACVLAN_PATTERN;
|
2016-08-09 19:23:15 +02:00
|
|
|
int reservedID = -1;
|
2011-11-02 17:11:02 +00:00
|
|
|
char ifname[IFNAMSIZ];
|
|
|
|
int retries, do_retry = 0;
|
|
|
|
uint32_t macvtapMode;
|
2016-01-19 14:20:54 -05:00
|
|
|
const char *ifnameCreated = NULL;
|
2012-03-05 17:12:39 -08:00
|
|
|
int vf = -1;
|
2015-12-03 11:33:55 +01:00
|
|
|
bool vnet_hdr = flags & VIR_NETDEV_MACVLAN_VNET_HDR;
|
2011-11-02 17:11:02 +00:00
|
|
|
|
|
|
|
macvtapMode = modeMap[mode];
|
|
|
|
|
2016-01-19 14:20:54 -05:00
|
|
|
*ifnameResult = NULL;
|
2011-11-02 17:11:02 +00:00
|
|
|
|
|
|
|
/** Note: When using PASSTHROUGH mode with MACVTAP devices the link
|
|
|
|
* device's MAC address must be set to the VMs MAC address. In
|
|
|
|
* order to not confuse the first switch or bridge in line this MAC
|
|
|
|
* address must be reset when the VM is shut down.
|
|
|
|
* This is especially important when using SRIOV capable cards that
|
|
|
|
* emulate their switch in firmware.
|
|
|
|
*/
|
2015-08-26 00:18:26 -04:00
|
|
|
|
2011-11-02 17:11:02 +00:00
|
|
|
if (mode == VIR_NETDEV_MACVLAN_MODE_PASSTHRU) {
|
2017-03-05 17:43:39 -05:00
|
|
|
bool setVlan = true;
|
|
|
|
|
2015-08-26 00:18:26 -04:00
|
|
|
if (virtPortProfile &&
|
|
|
|
virtPortProfile->virtPortType == VIR_NETDEV_VPORT_PROFILE_8021QBH) {
|
2017-03-05 17:43:39 -05:00
|
|
|
/* The Cisco enic driver (the only SRIOV-capable card that
|
|
|
|
* uses 802.1Qbh) doesn't support IFLA_VFINFO_LIST, which
|
|
|
|
* is required to get/set the vlan tag of a VF.
|
2015-08-26 00:18:26 -04:00
|
|
|
*/
|
2017-03-05 17:43:39 -05:00
|
|
|
setVlan = false;
|
2015-08-26 00:18:26 -04:00
|
|
|
}
|
2017-03-05 17:43:39 -05:00
|
|
|
|
|
|
|
if (virNetDevSaveNetConfig(linkdev, -1, stateDir, setVlan) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (virNetDevSetNetConfig(linkdev, -1, NULL, vlan, macaddress, setVlan) < 0)
|
|
|
|
return -1;
|
2011-11-02 17:11:02 +00:00
|
|
|
}
|
|
|
|
|
2016-01-19 14:20:54 -05:00
|
|
|
if (ifnameRequested) {
|
2016-08-09 19:25:44 +02:00
|
|
|
int rc;
|
2016-01-19 14:20:54 -05:00
|
|
|
bool isAutoName
|
2017-04-25 14:09:45 -04:00
|
|
|
= (STRPREFIX(ifnameRequested, VIR_NET_GENERATED_MACVTAP_PREFIX) ||
|
|
|
|
STRPREFIX(ifnameRequested, VIR_NET_GENERATED_MACVLAN_PREFIX));
|
2011-11-03 09:21:35 +00:00
|
|
|
|
2016-01-19 14:20:54 -05:00
|
|
|
VIR_INFO("Requested macvtap device name: %s", ifnameRequested);
|
|
|
|
virMutexLock(&virNetDevMacVLanCreateMutex);
|
|
|
|
|
2016-08-09 19:25:44 +02:00
|
|
|
if ((rc = virNetDevExists(ifnameRequested)) < 0) {
|
2016-01-19 14:20:54 -05:00
|
|
|
virMutexUnlock(&virNetDevMacVLanCreateMutex);
|
|
|
|
return -1;
|
|
|
|
}
|
2016-08-09 19:25:44 +02:00
|
|
|
if (rc) {
|
2016-01-19 14:20:54 -05:00
|
|
|
if (isAutoName)
|
2011-11-02 17:11:02 +00:00
|
|
|
goto create_name;
|
2011-11-03 09:21:35 +00:00
|
|
|
virReportSystemError(EEXIST,
|
2016-01-19 14:20:54 -05:00
|
|
|
_("Unable to create %s device %s"),
|
|
|
|
type, ifnameRequested);
|
|
|
|
virMutexUnlock(&virNetDevMacVLanCreateMutex);
|
2011-11-02 17:11:02 +00:00
|
|
|
return -1;
|
|
|
|
}
|
2016-01-19 14:20:54 -05:00
|
|
|
if (isAutoName &&
|
|
|
|
(reservedID = virNetDevMacVLanReserveName(ifnameRequested, true)) < 0) {
|
|
|
|
reservedID = -1;
|
|
|
|
goto create_name;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virNetDevMacVLanCreate(ifnameRequested, type, macaddress,
|
|
|
|
linkdev, macvtapMode, &do_retry) < 0) {
|
|
|
|
if (isAutoName) {
|
|
|
|
virNetDevMacVLanReleaseName(ifnameRequested);
|
|
|
|
reservedID = -1;
|
|
|
|
goto create_name;
|
|
|
|
}
|
|
|
|
virMutexUnlock(&virNetDevMacVLanCreateMutex);
|
2011-11-02 17:11:02 +00:00
|
|
|
return -1;
|
2016-01-19 14:20:54 -05:00
|
|
|
}
|
|
|
|
/* virNetDevMacVLanCreate() was successful - use this name */
|
|
|
|
ifnameCreated = ifnameRequested;
|
2014-03-25 07:53:22 +01:00
|
|
|
create_name:
|
2016-01-19 14:20:54 -05:00
|
|
|
virMutexUnlock(&virNetDevMacVLanCreateMutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
retries = MACVLAN_MAX_ID;
|
|
|
|
while (!ifnameCreated && retries) {
|
2013-02-28 15:59:01 +01:00
|
|
|
virMutexLock(&virNetDevMacVLanCreateMutex);
|
2016-01-19 14:20:54 -05:00
|
|
|
reservedID = virNetDevMacVLanReserveID(reservedID, flags, false, true);
|
|
|
|
if (reservedID < 0) {
|
|
|
|
virMutexUnlock(&virNetDevMacVLanCreateMutex);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
snprintf(ifname, sizeof(ifname), pattern, reservedID);
|
2016-08-09 19:23:15 +02:00
|
|
|
if (virNetDevMacVLanCreate(ifname, type, macaddress, linkdev,
|
|
|
|
macvtapMode, &do_retry) < 0) {
|
2016-01-19 14:20:54 -05:00
|
|
|
virNetDevMacVLanReleaseID(reservedID, flags);
|
|
|
|
virMutexUnlock(&virNetDevMacVLanCreateMutex);
|
|
|
|
if (!do_retry)
|
2011-11-03 09:21:35 +00:00
|
|
|
return -1;
|
2016-01-19 14:20:54 -05:00
|
|
|
VIR_INFO("Device %s wasn't reserved but already existed, skipping",
|
|
|
|
ifname);
|
|
|
|
retries--;
|
|
|
|
continue;
|
2011-11-02 17:11:02 +00:00
|
|
|
}
|
2016-01-19 14:20:54 -05:00
|
|
|
ifnameCreated = ifname;
|
2013-02-28 15:59:01 +01:00
|
|
|
virMutexUnlock(&virNetDevMacVLanCreateMutex);
|
2011-11-02 17:11:02 +00:00
|
|
|
}
|
|
|
|
|
2016-01-19 14:20:54 -05:00
|
|
|
if (!ifnameCreated) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Too many unreserved %s devices in use"),
|
|
|
|
type);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virNetDevVPortProfileAssociate(ifnameCreated,
|
2011-11-02 17:11:02 +00:00
|
|
|
virtPortProfile,
|
|
|
|
macaddress,
|
|
|
|
linkdev,
|
2012-03-05 17:12:39 -08:00
|
|
|
vf,
|
2016-08-09 19:23:15 +02:00
|
|
|
vmuuid, vmOp, false) < 0)
|
2011-11-02 17:11:02 +00:00
|
|
|
goto link_del_exit;
|
|
|
|
|
2014-09-16 16:50:53 -04:00
|
|
|
if (flags & VIR_NETDEV_MACVLAN_CREATE_IFUP) {
|
2016-08-09 19:23:15 +02:00
|
|
|
if (virNetDevSetOnline(ifnameCreated, true) < 0)
|
2014-09-16 16:50:53 -04:00
|
|
|
goto disassociate_exit;
|
2011-11-02 17:11:02 +00:00
|
|
|
}
|
|
|
|
|
2014-08-27 10:34:13 -04:00
|
|
|
if (flags & VIR_NETDEV_MACVLAN_CREATE_WITH_TAP) {
|
2016-08-09 19:23:15 +02:00
|
|
|
if (virNetDevMacVLanTapOpen(ifnameCreated, tapfd, tapfdSize, 10) < 0)
|
2011-11-10 10:29:09 +00:00
|
|
|
goto disassociate_exit;
|
|
|
|
|
2016-08-09 19:23:15 +02:00
|
|
|
if (virNetDevMacVLanTapSetup(tapfd, tapfdSize, vnet_hdr) < 0)
|
2011-11-02 17:11:02 +00:00
|
|
|
goto disassociate_exit;
|
2016-08-09 19:23:15 +02:00
|
|
|
|
|
|
|
if (VIR_STRDUP(*ifnameResult, ifnameCreated) < 0)
|
2011-11-10 10:29:09 +00:00
|
|
|
goto disassociate_exit;
|
|
|
|
} else {
|
2016-01-19 14:20:54 -05:00
|
|
|
if (VIR_STRDUP(*ifnameResult, ifnameCreated) < 0)
|
2011-11-10 10:29:09 +00:00
|
|
|
goto disassociate_exit;
|
|
|
|
}
|
2011-11-02 17:11:02 +00:00
|
|
|
|
2012-04-13 14:41:16 +02:00
|
|
|
if (vmOp == VIR_NETDEV_VPORT_PROFILE_OP_CREATE ||
|
|
|
|
vmOp == VIR_NETDEV_VPORT_PROFILE_OP_RESTORE) {
|
|
|
|
/* Only directly register upon a create or restore (restarting
|
|
|
|
* a saved image) - migration and libvirtd restart are handled
|
|
|
|
* elsewhere.
|
|
|
|
*/
|
2016-01-19 14:20:54 -05:00
|
|
|
if (virNetDevMacVLanVPortProfileRegisterCallback(ifnameCreated, macaddress,
|
2012-04-13 14:41:16 +02:00
|
|
|
linkdev, vmuuid,
|
|
|
|
virtPortProfile,
|
2012-10-17 10:23:12 +01:00
|
|
|
vmOp) < 0)
|
2016-08-09 19:23:15 +02:00
|
|
|
goto disassociate_exit;
|
2012-04-13 14:41:16 +02:00
|
|
|
}
|
2011-11-02 17:11:02 +00:00
|
|
|
|
2016-08-09 19:23:15 +02:00
|
|
|
return 0;
|
2011-11-02 17:11:02 +00:00
|
|
|
|
2014-03-25 07:53:22 +01:00
|
|
|
disassociate_exit:
|
2016-01-19 14:20:54 -05:00
|
|
|
ignore_value(virNetDevVPortProfileDisassociate(ifnameCreated,
|
2011-11-02 17:11:02 +00:00
|
|
|
virtPortProfile,
|
|
|
|
macaddress,
|
|
|
|
linkdev,
|
2012-03-05 17:12:39 -08:00
|
|
|
vf,
|
2011-11-02 17:11:02 +00:00
|
|
|
vmOp));
|
2015-12-04 11:31:17 +01:00
|
|
|
while (tapfdSize--)
|
|
|
|
VIR_FORCE_CLOSE(tapfd[tapfdSize]);
|
2011-11-02 17:11:02 +00:00
|
|
|
|
2014-03-25 07:53:22 +01:00
|
|
|
link_del_exit:
|
2016-01-19 14:20:54 -05:00
|
|
|
ignore_value(virNetDevMacVLanDelete(ifnameCreated));
|
|
|
|
virNetDevMacVLanReleaseName(ifnameCreated);
|
2011-11-02 17:11:02 +00:00
|
|
|
|
2016-08-09 19:23:15 +02:00
|
|
|
return -1;
|
2011-11-02 17:11:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
2011-11-02 17:19:48 +00:00
|
|
|
* virNetDevMacVLanDeleteWithVPortProfile:
|
2011-11-02 17:11:02 +00:00
|
|
|
* @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
|
|
|
|
* it with the switch if port profile parameters
|
|
|
|
* were provided.
|
|
|
|
*/
|
2011-11-02 17:19:48 +00:00
|
|
|
int virNetDevMacVLanDeleteWithVPortProfile(const char *ifname,
|
2013-10-05 13:41:44 -06:00
|
|
|
const virMacAddr *macaddr,
|
2011-11-02 17:19:48 +00:00
|
|
|
const char *linkdev,
|
|
|
|
int mode,
|
|
|
|
virNetDevVPortProfilePtr virtPortProfile,
|
|
|
|
char *stateDir)
|
2011-11-02 17:11:02 +00:00
|
|
|
{
|
|
|
|
int ret = 0;
|
2012-03-05 17:12:39 -08:00
|
|
|
|
2011-11-02 17:11:02 +00:00
|
|
|
if (ifname) {
|
|
|
|
if (virNetDevVPortProfileDisassociate(ifname,
|
|
|
|
virtPortProfile,
|
|
|
|
macaddr,
|
|
|
|
linkdev,
|
2017-03-05 17:32:15 -05:00
|
|
|
-1,
|
2011-11-02 17:11:02 +00:00
|
|
|
VIR_NETDEV_VPORT_PROFILE_OP_DESTROY) < 0)
|
|
|
|
ret = -1;
|
2011-11-02 17:26:11 +00:00
|
|
|
if (virNetDevMacVLanDelete(ifname) < 0)
|
2011-11-02 17:11:02 +00:00
|
|
|
ret = -1;
|
2016-01-19 14:20:54 -05:00
|
|
|
virNetDevMacVLanReleaseName(ifname);
|
2011-11-02 17:11:02 +00:00
|
|
|
}
|
2012-02-22 14:17:14 +01:00
|
|
|
|
2016-01-21 14:19:56 -05:00
|
|
|
if (mode == VIR_NETDEV_MACVLAN_MODE_PASSTHRU) {
|
2017-03-05 17:43:39 -05:00
|
|
|
virMacAddrPtr MAC = NULL;
|
|
|
|
virMacAddrPtr adminMAC = NULL;
|
|
|
|
virNetDevVlanPtr vlan = NULL;
|
|
|
|
|
util: restructure virNetDevReadNetConfig() to eliminate false error logs
virHostdevRestoreNetConfig() calls virNetDevReadNetConfig() to try and
read the "original config" of a netdev, and if that fails, it tries
again with a different directory/netdev name. This achieves the
desired effect (we end up finding the config wherever it may be), but
for each failure, virNetDevReadNetConfig() places a nice error message
in the system logs. Experience has shown that false-positive error
logs like this lead to erroneous bug reports, and can often mislead
those searching for *real* bugs.
This patch changes virNetDevReadNetConfig() to explicitly check if the
file exists before calling virFileReadAll(); if it doesn't exist,
virNetDevReadNetConfig() returns a success, but leaves all the
variables holding the results as NULL. (This makes sense if you define
the purpose of the function as "read a netdev's config from its config
file *if that file exists*).
To take advantage of that change, the caller,
virHostdevRestoreNetConfig() is modified to fail immediately if
virNetDevReadNetConfig() returns an error, and otherwise to try the
different directory/netdev name if adminMAC & vlan & MAC are all NULL
after the preceding attempt.
2017-08-09 20:49:26 -04:00
|
|
|
if ((virNetDevReadNetConfig(linkdev, -1, stateDir,
|
|
|
|
&adminMAC, &vlan, &MAC) == 0) &&
|
|
|
|
(adminMAC || vlan || MAC)) {
|
2017-03-05 17:43:39 -05:00
|
|
|
|
|
|
|
ignore_value(virNetDevSetNetConfig(linkdev, -1,
|
|
|
|
adminMAC, vlan, MAC, !!vlan));
|
|
|
|
VIR_FREE(MAC);
|
|
|
|
VIR_FREE(adminMAC);
|
|
|
|
virNetDevVlanFree(vlan);
|
|
|
|
}
|
2016-01-21 14:19:56 -05:00
|
|
|
}
|
|
|
|
|
2012-08-22 12:10:23 +08:00
|
|
|
virNetlinkEventRemoveClient(0, macaddr, NETLINK_ROUTE);
|
2012-02-22 14:17:14 +01:00
|
|
|
|
2011-11-02 17:11:02 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-03-27 14:38:33 +02:00
|
|
|
/**
|
|
|
|
* virNetDevMacVLanRestartWithVPortProfile:
|
|
|
|
* Register a port profile callback handler for a VM that
|
|
|
|
* is already running
|
|
|
|
* .
|
|
|
|
* @cr_ifname: Interface name that the macvtap has.
|
|
|
|
* @macaddress: The MAC address for the macvtap device
|
|
|
|
* @linkdev: The interface name of the NIC to connect to the external bridge
|
|
|
|
* @vmuuid: The UUID of the VM the macvtap belongs to
|
|
|
|
* @virtPortProfile: pointer to object holding the virtual port profile data
|
|
|
|
* @vmOp: Operation to use during setup of the association
|
|
|
|
*
|
|
|
|
* Returns 0; returns -1 on error.
|
|
|
|
*/
|
|
|
|
int virNetDevMacVLanRestartWithVPortProfile(const char *cr_ifname,
|
2017-02-10 20:05:23 +05:30
|
|
|
const virMacAddr *macaddress,
|
|
|
|
const char *linkdev,
|
|
|
|
const unsigned char *vmuuid,
|
|
|
|
virNetDevVPortProfilePtr virtPortProfile,
|
|
|
|
virNetDevVPortProfileOp vmOp)
|
2012-03-27 14:38:33 +02:00
|
|
|
{
|
|
|
|
int rc = 0;
|
|
|
|
|
|
|
|
rc = virNetDevMacVLanVPortProfileRegisterCallback(cr_ifname, macaddress,
|
|
|
|
linkdev, vmuuid,
|
|
|
|
virtPortProfile, vmOp);
|
|
|
|
if (rc < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
ignore_value(virNetDevVPortProfileAssociate(cr_ifname,
|
|
|
|
virtPortProfile,
|
|
|
|
macaddress,
|
|
|
|
linkdev,
|
|
|
|
-1,
|
|
|
|
vmuuid,
|
|
|
|
vmOp, true));
|
|
|
|
|
2014-03-25 07:53:22 +01:00
|
|
|
error:
|
2012-03-27 14:38:33 +02:00
|
|
|
return rc;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2011-11-02 17:11:02 +00:00
|
|
|
#else /* ! WITH_MACVTAP */
|
2011-11-02 17:34:41 +00:00
|
|
|
int virNetDevMacVLanCreate(const char *ifname ATTRIBUTE_UNUSED,
|
|
|
|
const char *type ATTRIBUTE_UNUSED,
|
2013-10-05 13:41:44 -06:00
|
|
|
const virMacAddr *macaddress ATTRIBUTE_UNUSED,
|
2011-11-02 17:34:41 +00:00
|
|
|
const char *srcdev ATTRIBUTE_UNUSED,
|
|
|
|
uint32_t macvlan_mode ATTRIBUTE_UNUSED,
|
|
|
|
int *retry ATTRIBUTE_UNUSED)
|
|
|
|
{
|
|
|
|
virReportSystemError(ENOSYS, "%s",
|
|
|
|
_("Cannot create macvlan devices on this platform"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int virNetDevMacVLanDelete(const char *ifname ATTRIBUTE_UNUSED)
|
|
|
|
{
|
|
|
|
virReportSystemError(ENOSYS, "%s",
|
|
|
|
_("Cannot create macvlan devices on this platform"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2011-11-02 17:19:48 +00:00
|
|
|
int virNetDevMacVLanCreateWithVPortProfile(const char *ifname ATTRIBUTE_UNUSED,
|
2013-10-05 13:41:44 -06:00
|
|
|
const virMacAddr *macaddress ATTRIBUTE_UNUSED,
|
2011-11-02 17:19:48 +00:00
|
|
|
const char *linkdev ATTRIBUTE_UNUSED,
|
2014-05-06 09:14:05 +02:00
|
|
|
virNetDevMacVLanMode mode ATTRIBUTE_UNUSED,
|
2016-05-04 13:18:16 -04:00
|
|
|
virNetDevVlanPtr vlan ATTRIBUTE_UNUSED,
|
2011-11-02 17:19:48 +00:00
|
|
|
const unsigned char *vmuuid ATTRIBUTE_UNUSED,
|
|
|
|
virNetDevVPortProfilePtr virtPortProfile ATTRIBUTE_UNUSED,
|
|
|
|
char **res_ifname ATTRIBUTE_UNUSED,
|
2014-05-06 09:14:05 +02:00
|
|
|
virNetDevVPortProfileOp vmop ATTRIBUTE_UNUSED,
|
2011-11-02 17:19:48 +00:00
|
|
|
char *stateDir ATTRIBUTE_UNUSED,
|
2015-12-04 11:31:17 +01:00
|
|
|
int *tapfd ATTRIBUTE_UNUSED,
|
|
|
|
size_t tapfdSize ATTRIBUTE_UNUSED,
|
2015-01-29 14:39:12 +01:00
|
|
|
unsigned int unused_flags ATTRIBUTE_UNUSED)
|
2011-11-02 17:11:02 +00:00
|
|
|
{
|
|
|
|
virReportSystemError(ENOSYS, "%s",
|
|
|
|
_("Cannot create macvlan devices on this platform"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2011-11-02 17:19:48 +00:00
|
|
|
int virNetDevMacVLanDeleteWithVPortProfile(const char *ifname ATTRIBUTE_UNUSED,
|
2013-10-05 13:41:44 -06:00
|
|
|
const virMacAddr *macaddress ATTRIBUTE_UNUSED,
|
2011-11-02 17:19:48 +00:00
|
|
|
const char *linkdev ATTRIBUTE_UNUSED,
|
|
|
|
int mode ATTRIBUTE_UNUSED,
|
|
|
|
virNetDevVPortProfilePtr virtPortProfile ATTRIBUTE_UNUSED,
|
|
|
|
char *stateDir ATTRIBUTE_UNUSED)
|
2011-11-02 17:11:02 +00:00
|
|
|
{
|
|
|
|
virReportSystemError(ENOSYS, "%s",
|
|
|
|
_("Cannot create macvlan devices on this platform"));
|
|
|
|
return -1;
|
|
|
|
}
|
2012-03-27 14:38:33 +02:00
|
|
|
|
|
|
|
int virNetDevMacVLanRestartWithVPortProfile(const char *cr_ifname ATTRIBUTE_UNUSED,
|
2017-02-10 20:05:23 +05:30
|
|
|
const virMacAddr *macaddress ATTRIBUTE_UNUSED,
|
|
|
|
const char *linkdev ATTRIBUTE_UNUSED,
|
|
|
|
const unsigned char *vmuuid ATTRIBUTE_UNUSED,
|
|
|
|
virNetDevVPortProfilePtr virtPortProfile ATTRIBUTE_UNUSED,
|
|
|
|
virNetDevVPortProfileOp vmOp ATTRIBUTE_UNUSED)
|
2012-03-27 14:38:33 +02:00
|
|
|
{
|
|
|
|
virReportSystemError(ENOSYS, "%s",
|
|
|
|
_("Cannot create macvlan devices on this platform"));
|
|
|
|
return -1;
|
|
|
|
}
|
2012-03-29 13:15:00 +02:00
|
|
|
|
|
|
|
int virNetDevMacVLanVPortProfileRegisterCallback(const char *ifname ATTRIBUTE_UNUSED,
|
2017-02-10 20:05:23 +05:30
|
|
|
const virMacAddr *macaddress ATTRIBUTE_UNUSED,
|
|
|
|
const char *linkdev ATTRIBUTE_UNUSED,
|
|
|
|
const unsigned char *vmuuid ATTRIBUTE_UNUSED,
|
|
|
|
virNetDevVPortProfilePtr virtPortProfile ATTRIBUTE_UNUSED,
|
|
|
|
virNetDevVPortProfileOp vmOp ATTRIBUTE_UNUSED)
|
2012-03-29 13:15:00 +02:00
|
|
|
{
|
|
|
|
virReportSystemError(ENOSYS, "%s",
|
|
|
|
_("Cannot create macvlan devices on this platform"));
|
|
|
|
return -1;
|
|
|
|
}
|
2016-01-27 09:49:54 +01:00
|
|
|
|
|
|
|
int virNetDevMacVLanReleaseName(const char *name ATTRIBUTE_UNUSED)
|
|
|
|
{
|
|
|
|
virReportSystemError(ENOSYS, "%s",
|
|
|
|
_("Cannot create macvlan devices on this platform"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int virNetDevMacVLanReserveName(const char *name ATTRIBUTE_UNUSED,
|
|
|
|
bool quietFail ATTRIBUTE_UNUSED)
|
|
|
|
{
|
|
|
|
virReportSystemError(ENOSYS, "%s",
|
|
|
|
_("Cannot create macvlan devices on this platform"));
|
|
|
|
return -1;
|
|
|
|
}
|
2011-11-02 17:11:02 +00:00
|
|
|
#endif /* ! WITH_MACVTAP */
|